Update contrib.
1 // Copyright (c) 2005-2010 Nokia Corporation and/or its subsidiary(-ies).
2 // All rights reserved.
3 // This component and the accompanying materials are made available
4 // under the terms of "Eclipse Public License v1.0"
5 // which accompanies this distribution, and is available
6 // at the URL "http://www.eclipse.org/legal/epl-v10.html".
8 // Initial Contributors:
9 // Nokia Corporation - initial contribution.
12 // NTT DOCOMO, INC - Fix for Bug 1915 "SQL server panics when using long column type strings"
17 #include "SqlSrvMain.h" //CSqlServer
18 #include "SqlSrvSession.h" //CSqlSrvSession
19 #include "SqlSrvStatement.h" //CSqlSrvStatement, HSqlSrvStmtParamBuf
20 #include "SqlSecurityImpl.h" //CSqlSecurityPolicy
21 #include "SqlSrvUtil.h" //Global server functions
22 #include "SqlUtil.h" // config length
23 #include "SqlSrvDriveSpace.h" //CSqlDriveSpace, RSqlDriveSpaceCol
24 #include "SqlSrvBlob.h"
25 #include "SqlResourceProfiler.h"
26 #include "SqlCompact.h"
27 #include "OstTraceDefinitions.h"
28 #ifdef OST_TRACE_COMPILER_IN_USE
29 #include "SqlSrvSessionTraces.h"
31 #include "SqlTraceDef.h"
33 ///////////////////////////////////////////////////////////////////////////////////////////////////////////////
34 ///////////////////////////////////////////////////////////////////////////////////////////////////////////////
36 #pragma BullseyeCoverage off
40 const TInt KDelayedDbHeapFailureMask = 0x1000;
42 //Puts the database connection in a test mode
43 //Returns true if the heap failure simulation has to be delayed untill the database is opened
44 //Initialises iDbResourceTestMode and iFailedAllocNumber (if delayed simulation) data members
45 inline TBool CSqlSrvSession::ActivateDbTestMode(TInt aHeapFailureMode, TInt aFailedAllocNumber)
47 iDbResourceTestMode = aHeapFailureMode;
48 if(aHeapFailureMode & KDelayedDbHeapFailureMask)
50 iFailedAllocNumber = aFailedAllocNumber;
56 //If the database connection is in a test mode then the macro will reset the heap allocation failure type.
57 //and stop the test mode.
58 inline void CSqlSrvSession::StopDbTestMode()
60 if(iDbResourceTestMode)
62 iDbResourceTestMode = 0;
63 User::__DbgSetAllocFail(RHeap::EUser, RHeap::ENone, 0);
67 //If the database connection is in a test mode then the function will mark the allocated by the
69 inline void CSqlSrvSession::DbResourceMark()
71 if(iDbResourceTestMode)
73 ResourceCountMarkStart();
77 //If the database connection is in a test mode then the macro will check the allocated by the server resources,
78 //comparing their count with the resource count at the moment DbResourceMark() has been called.
79 //The client will be panicked if the resource count now is different.
80 inline void CSqlSrvSession::DbResourceEnd(const RMessage2& aMessage)
82 if(iDbResourceTestMode)
84 ResourceCountMarkEnd(aMessage);
88 //Executes the heap simulation failure.
89 inline void CSqlSrvSession::DbSetAllocFail(TInt aHeapFailureMode, TInt aFailedAllocNumber)
91 TInt mode = aHeapFailureMode & (KDelayedDbHeapFailureMask - 1);
92 if(mode >= RAllocator::EBurstRandom && mode <= RAllocator::EBurstFailNext)
94 const TUint KBurst = 50;
95 User::__DbgSetBurstAllocFail(RHeap::EUser, static_cast <RHeap::TAllocFail> (mode), aFailedAllocNumber, KBurst);
99 User::__DbgSetAllocFail(RHeap::EUser, static_cast <RHeap::TAllocFail> (mode), aFailedAllocNumber);
103 //Executes the delayed heap simulation failure, if the connection is in test mode
104 inline void CSqlSrvSession::DbSetDelayedAllocFail()
106 if(iDbResourceTestMode & KDelayedDbHeapFailureMask)
108 TInt mode = iDbResourceTestMode & (KDelayedDbHeapFailureMask - 1);
109 if(mode >= RAllocator::EBurstRandom && mode <= RAllocator::EBurstFailNext)
111 const TUint KBurst = 50;
112 User::__DbgSetBurstAllocFail(RHeap::EUser, static_cast <RHeap::TAllocFail> (mode), iFailedAllocNumber, KBurst);
116 User::__DbgSetAllocFail(RHeap::EUser, static_cast <RHeap::TAllocFail> (mode), iFailedAllocNumber);
123 inline TBool CSqlSrvSession::ActivateDbTestMode(TInt, TInt)
128 inline void CSqlSrvSession::StopDbTestMode()
132 inline void CSqlSrvSession::DbResourceMark()
136 inline void CSqlSrvSession::DbResourceEnd(const RMessage2&)
140 inline void CSqlSrvSession::DbSetAllocFail(TInt, TInt)
144 inline void CSqlSrvSession::DbSetDelayedAllocFail()
150 #pragma BullseyeCoverage on
152 ///////////////////////////////////////////////////////////////////////////////////////////////////////////////
153 ///////////////////////////////////////////////////////////////////////////////////////////////////////////////
156 Searches aContainer for an object identified by aHandle.
157 If such object exists, a reference to it is returned.
158 If there is no object, the client gets a panic.
160 @panic SqlDb 4 Client panic. Invalid aHandle parameter value (zero, negative or out of range).
164 template <class T> T& SqlSessObjFind(RDbObjContainer<T>& aContainer, TInt aHandle, const RMessage2& aMessage)
166 T* obj = aContainer.Find(aHandle);
167 __SQLPANIC_CLIENT2(obj != NULL, aMessage, ESqlPanicBadArgument);
171 //This function return true, if there is free disk space on drive where the main database file is.
172 static TBool HasFreeDiskSpace(RFs& aFs, TDriveNumber aDrive)
175 TInt err = aFs.Volume(volInfo, aDrive);
178 const TInt64 KDiskSpaceThreshold = 1024 * 4;
179 return volInfo.iFree > KDiskSpaceThreshold;
184 //If aError is KSqlErrFull and there is no free disk space, then KSqlErrFull is converted to KErrDiskFull
186 static TInt ConvertSqlFull2DiskFullErr(TInt aError, RFs& aFs, TDriveNumber aDrive)
188 if(aError == KSqlErrFull && !HasFreeDiskSpace(aFs, aDrive))
190 aError = KErrDiskFull;
191 SQL_TRACE_SESSION(OstTrace1(TRACE_INTERNALS, CONVERTSQLFULL2DISKFULLERRROR, "0;ConvertSqlFull2DiskFullErr;aError=KSqlErrFull;!HasFreeDiskSpace();aDrive=%d", (TInt)aDrive));
196 ///////////////////////////////////////////////////////////////////////////////////////////////////////////////
197 ///////////////////////////////////////////////////////////////////////////////////////////////////////////////
200 Creates a new instance of CSqlSrvSession class.
202 This function shall never be called directly.
203 It is CSqlServer responsibility to create a new server side session object as a responce to the criation of a
204 client side session instance.
206 @return A pointer to the created CSqlSrvSession instance.
208 @leave KErrNoMemory, an out of memory condition has occurred;
211 @see CSqlServer::NewSessionL()
213 CSqlSrvSession* CSqlSrvSession::NewL()
215 SQL_TRACE_SESSION(OstTrace0(TRACE_INTERNALS, CSQLSRVSESSION_NEWL_ENTRY, "Entry;0;CSqlSrvSession::NewL"));
216 CSqlSrvSession* self = new (ELeave) CSqlSrvSession;
217 CleanupStack::PushL(self);
219 CleanupStack::Pop(self);
220 SQL_TRACE_SESSION(OstTrace1(TRACE_INTERNALS, CSQLSRVSESSION_NEWL_EXIT, "Exit;0x%X;CSqlSrvSession::NewL", (TUint)self));
225 Frees the allocated by CSqlSrvSession instance resources and memory.
227 CSqlSrvSession::~CSqlSrvSession()
229 SQL_TRACE_SESSION(OstTraceExt2(TRACE_INTERNALS, CSQLSRVSESSION_CSQLSRVSESSION2_ENTRY, "Entry;0x%X;CSqlSrvSession::~CSqlSrvSession;iDatabase=0x%X", (TUint)this, (TUint)iDatabase));
231 DbFreeReservedSpace();
235 SQL_TRACE_SESSION(OstTrace1(TRACE_INTERNALS, CSQLSRVSESSION_CSQLSRVSESSION2_EXIT, "Exit;0x%X;CSqlSrvSession::~CSqlSrvSession", (TUint)this));
239 Receives and dispatches all client side requests.
241 CSession2::ServiceL() implementation.
243 @param aMessage Client message containing the request (function code and data)
245 @leave The function may leave with some database specific
246 errors categorised as ESqlDbError or system-wide error codes.
248 @see CSession2::ServiceL()
250 void CSqlSrvSession::ServiceL(const RMessage2& aMessage)
252 TSqlSrvFunction funcCode = ESqlSrvTestBase;
254 Extract(aMessage, funcCode, handle);
255 TInt retCode = KErrNone;
256 if(funcCode >= ESqlSrvDbBase)
258 SQLPROFILER_REPORT_IPC(ESqlIpcRq, 0);
260 __SQLTRACE_SESSIONEXPR(++iIpcCallCounter);
261 SQL_TRACE_SESSION(OstTraceExt4(TRACE_INTERNALS, CSQLSRVSESSION_SERVICEL_ENTRY, "Entry;0x%X;CSqlSrvSession::ServiceL;aMessage.Handle()=0x%X;funcCode=0x%X;iIpcCallCounter=%u", (TUint)this, (TUint)aMessage.Handle(), (TUint)funcCode, iIpcCallCounter));
262 SQLPROFILER_IPC_START(iIpcCallCounter, iDatabase ? (TUint)iDatabase->RawDbHandle() : 0);
265 ////////////////////// resource check operations ///////////////////////////
266 case ESqlSrvResourceMark:
267 ResourceCountMarkStart();
269 case ESqlSrvResourceCheck:
270 ResourceCountMarkEnd(aMessage);
272 case ESqlSrvResourceCount:
273 retCode = CountResources(); //Returns the recourse count
275 case ESqlSrvSetDbHeapFailure:
276 if(ActivateDbTestMode(aMessage.Int0(), aMessage.Int1()))
280 case ESqlSrvSetHeapFailure:
281 DbSetAllocFail(aMessage.Int0(), aMessage.Int1());
283 ////////////////////// profiling operations //////////////////////////////////
284 case ESqlSrvProfilerStart:
285 TSqlSrvResourceProfiler::StartL(aMessage);
287 case ESqlSrvProfilerStop:
288 TSqlSrvResourceProfiler::StopL(aMessage);
290 case ESqlSrvProfilerReset:
291 TSqlSrvResourceProfiler::ResetL(aMessage);
293 case ESqlSrvProfilerQuery:
294 ProfilerQueryL(aMessage);
296 ////////////////////// database operations //////////////////////////////////
297 case ESqlSrvDbCreate:
298 case ESqlSrvDbCreateSecure:
301 DbCreateObjectL(aMessage, funcCode);
302 DbSetDelayedAllocFail();
304 case ESqlSrvDbOpenFromHandle:
306 DbCreateObjectFromHandleL(aMessage);
307 DbSetDelayedAllocFail();
309 case ESqlSrvDbAttach:
312 case ESqlSrvDbAttachFromHandle:
313 DbAttachFromHandleL(aMessage);
315 case ESqlSrvDbDetach:
320 DbResourceEnd(aMessage);
325 DbCopyFileL(aMessage);
327 case ESqlSrvDbDelete:
329 DbDeleteFileL(aMessage);
331 case ESqlSrvLastErrorMsg:
332 retCode = DbLastErrorMessageL(aMessage);//may return that the client buffer is not big enough for the message
334 case ESqlSrvDbLastInsertedRowId:
335 DbLastInsertedRowIdL(aMessage);
338 retCode = DbExecSql8L(aMessage); //returns the count of affected records
340 case ESqlSrvDbExec16:
341 retCode = DbExecSql16L(aMessage); //returns the count of affected records
343 case ESqlSrvDbSetIsolationLevel:
344 DbSetIsolationLevelL(aMessage);
346 case ESqlSrvDbGetSecurityPolicy:
347 retCode = DbGetSecurityPolicyL(aMessage);//may return that the client buffer is not big enough for the security policy
349 case ESqlSrvDbScalarFullSelect8:
350 retCode = DbScalarFullSelectL(aMessage, EFalse);//may return that the client buffer is not big enough for the column value
352 case ESqlSrvDbScalarFullSelect16:
353 retCode = DbScalarFullSelectL(aMessage, ETrue);//may return that the client buffer is not big enough for the column value
355 case ESqlSrvDbInTransaction:
356 retCode = DbInTransaction(aMessage); //Returns whether the database in in transaction or not - true/false
359 retCode = DbSizeL(aMessage); //Returns the database size
364 case ESqlSrvDbCompact:
365 retCode = DbCompactL(aMessage); //Returns the amount of the removed free database space
367 ////////////////////// reserved drive space management ////////////////////////
368 case ESqlSrvDbReserveDriveSpace:
369 __SQLPANIC_CLIENT(iDatabase != NULL, aMessage, ESqlPanicInvalidObj);
370 DbReserveDriveSpaceL();
372 case ESqlSrvDbFreeReservedSpace:
373 __SQLPANIC_CLIENT(iDatabase != NULL, aMessage, ESqlPanicInvalidObj);
374 DbFreeReservedSpace();
376 case ESqlSrvDbGetReserveAccess:
377 __SQLPANIC_CLIENT(iDatabase != NULL, aMessage, ESqlPanicInvalidObj);
378 DbGetReserveAccessL();
380 case ESqlSrvDbReleaseReserveAccess:
381 __SQLPANIC_CLIENT(iDatabase != NULL, aMessage, ESqlPanicInvalidObj);
382 DbReleaseReserveAccess();
384 ////////////////////// BLOB source ///////////////////////////////////////////
385 case ESqlSrvDbBlobSource:
386 retCode = DbBlobSourceL(aMessage); //Returns the BLOB handle
388 ////////////////////// statement operations //////////////////////////////////
389 case ESqlSrvStmtPrepare8:
390 retCode = StmtPrepareL(aMessage, EFalse);//returns the statement handle
392 case ESqlSrvStmtPrepare16:
393 retCode = StmtPrepareL(aMessage, ETrue);//returns the statement handle
395 case ESqlSrvStmtClose:
396 iStatements.Remove(handle);
398 case ESqlSrvStmtReset:
399 retCode = ::SqlSessObjFind(iStatements, handle, aMessage).Reset();//May return that the statement has expired
401 case ESqlSrvStmtExec:
402 case ESqlSrvStmtAsyncExec:
403 case ESqlSrvStmtBindExec:
404 case ESqlSrvStmtAsyncBindExec:
405 retCode = StmtExecL(aMessage, handle, funcCode);//returns the count of affected records
407 case ESqlSrvStmtNext:
408 case ESqlSrvStmtBindNext:
409 retCode = StmtNextL(aMessage, handle, funcCode);//returns a non-negative number if the client side buffer is too small
411 case ESqlSrvStmtColumnNames:
412 case ESqlSrvStmtParamNames:
413 retCode = StmtNamesL(aMessage, handle, funcCode);//returns a non-negative number if the client side buffer is too small
415 case ESqlSrvStmtColumnSource:
416 retCode = StmtColumnSourceL(aMessage, handle);//returns an IPC stream handle
418 case ESqlSrvStmtBinParamSink:
419 case ESqlSrvStmtTxtParamSink16:
420 retCode = StmtParamSinkL(aMessage, handle, funcCode);//returns an IPC stream handle
422 case ESqlSrvStmtBufFlat:
423 StmtGetBufFlatL(aMessage, handle);
425 case ESqlSrvStmtColumnValue:
426 StmtColumnValueL(aMessage, handle);
428 case ESqlSrvStmtDeclColumnTypes:
429 retCode = StmtDeclColumnTypesL(aMessage, handle);
431 ////////////////////// stream operations //////////////////////////////////
432 case ESqlSrvStreamRead:
433 retCode = ::SqlSessObjFind(iIpcStreams, handle, aMessage).ReadL(aMessage);
435 case ESqlSrvStreamWrite:
436 ::SqlSessObjFind(iIpcStreams, handle, aMessage).WriteL(aMessage);
438 case ESqlSrvStreamSize:
439 retCode = ::SqlSessObjFind(iIpcStreams, handle, aMessage).SizeL();
441 case ESqlSrvStreamSynch:
442 ::SqlSessObjFind(iIpcStreams, handle, aMessage).SynchL();
444 case ESqlSrvStreamClose:
445 iIpcStreams.Remove(handle);
447 ////////////////////// //////////////////////////////////
449 retCode = KErrNotSupported;
452 Server().Compactor().RestartTimer();
453 Server().MinimizeBuffers();
454 if(!aMessage.IsNull())
456 aMessage.Complete(retCode);
458 SQL_TRACE_SESSION(OstTraceExt4(TRACE_INTERNALS, CSQLSRVSESSION_SERVICEL_EXIT, "Exit;0x%X;CSqlSrvSession::ServiceL;funcCode=0x%X;retCode=%d;iIpcCallCounter=%u", (TUint)this, (TUint)funcCode, retCode, iIpcCallCounter));
462 If aError is KErrBadDescriptor, then panic the client, else - default error handling.
463 KErrBadDescriptor error may be thrown from "message write" operations, if the client supplied a bad
464 descriptor to the server.
466 void CSqlSrvSession::ServiceError(const RMessage2& aMessage, TInt aError)
468 SQL_TRACE_SESSION(OstTraceExt4(TRACE_INTERNALS, CSQLSRVSESSION_SERVICEERROR_ENTRY, "Entry;0x%X;CSqlSrvSession::ServiceError;aMessage.Function()=0x%X;aError=%d;iIpcCallCounter=%u", (TUint)this, (TUint)aMessage.Function(), aError, iIpcCallCounter));
469 Server().MinimizeBuffers();
470 aError = ::ConvertSqlFull2DiskFullErr(aError, Server().FileData().Fs(), iDrive);
471 if(aError == KErrBadDescriptor)
473 //The __SQLPANIC_CLIENT() macro cannot be used here because it calls a leaving function. A leaving call
474 //from a leaving call will terminate the server.
475 _LIT(KPanicCategory, "SqlDb");
476 aMessage.Panic(KPanicCategory, ESqlPanicBadDescriptor);
478 SQLPROFILER_IPC_ERROR(iIpcCallCounter, static_cast <TSqlSrvFunction> (KSqlSrvFunctionMask & aMessage.Function()),
479 iDatabase ? (TUint)iDatabase->RawDbHandle() : 0, aError);
480 CSession2::ServiceError(aMessage, aError);
481 SQL_TRACE_SESSION(OstTraceExt3(TRACE_INTERNALS, CSQLSRVSESSION_SERVICEERROR_EXIT, "Exit;0x%X;CSqlSrvSession::ServiceError;aMessage.Function()=0x%X;iIpcCallCounter=%u", (TUint)this, (TUint)aMessage.Function(), iIpcCallCounter));
484 ///////////////////////////////////////////////////////////////////////////////////////////////////////
485 //////////////////////////// Profiler operations ///////////////////////////////////
486 ///////////////////////////////////////////////////////////////////////////////////////////////////////
488 #pragma BullseyeCoverage off
491 Retrieves the counter values for the specified profiling counter.
493 @leave KErrNone, the operation completed successfully,
494 KErrOverflow, the receiving buffer size is too small;
495 One of the other system-wide error codes may also be returned.
497 @see TSqlResourceProfiler
499 Usage of the IPC call arguments:
500 - Arg 0: [in] profiling counter type, one of the TSqlResourceProfiler::TSqlCounter enum item values.
501 - Arg 1: [in] the size of the buffer for the profiling counter values.
502 - Arg 2: [out] the buffer for the profiling counter values.
504 @panic SqlDb 2 Client panic. iDatabase is NULL (the database object is not created yet).
506 void CSqlSrvSession::ProfilerQueryL(const RMessage2& aMessage)
508 __SQLPANIC_CLIENT(iDatabase != NULL, aMessage, ESqlPanicInvalidObj);
509 if(aMessage.Int0() == TSqlResourceProfiler::ESqlCounterConfig)
511 const TInt KConfigBufLen = 128;
512 if(aMessage.Int1() < KConfigBufLen)
514 __SQLLEAVE(KErrOverflow);
516 TBuf8<KConfigBufLen> res;
517 iDatabase->QueryConfigL(res);
518 aMessage.WriteL(2, res);
522 TSqlSrvResourceProfiler::QueryL(aMessage);
526 #pragma BullseyeCoverage on
528 ///////////////////////////////////////////////////////////////////////////////////////////////////////
529 //////////////////////////// Database operations ///////////////////////////////////
530 ///////////////////////////////////////////////////////////////////////////////////////////////////////
533 Processes the request for creating/opening a database.
535 The function initializes iDatabase and iDrive data members.
537 Usage of the IPC call arguments:
538 Arg 0: [in] database file name length in 16-bit characters
539 Arg 1: [in] database file name
540 Arg 2: [in] PPPPCCCC, where PPPP is the security policy length, CCCC is the config string length.
541 Arg 3: [in] security policies buffer | config string
543 @panic SqlDb 1 Client panic. iDatabase is not NULL (it has been created already)
544 @panic SqlDb 4 Client panic. Negative or too big config string length
545 @panic SqlDb 4 Client panic. Negative security policy length, or zero length if the request is to create a secure database
547 void CSqlSrvSession::DbCreateObjectL(const RMessage2& aMessage, TSqlSrvFunction aFunction)
549 __SQLPANIC_CLIENT(!iDatabase, aMessage, ESqlPanicObjExists);
550 const TInt KSecurityPolicyLen = (aMessage.Int2() & 0x7fff0000) >> 16;
551 //If the security policy length is negative then this is a programming error.
552 __SQLPANIC_CLIENT(KSecurityPolicyLen >= 0, aMessage, ESqlPanicBadArgument);
553 const TInt KConfigStringLen = aMessage.Int2() & 0xffff;
554 //If KConfigStringLen is invalid then this is a programming error.
555 //If the client sends a too big config string - this is handled in the client side session.
556 __SQLPANIC_CLIENT((TUint)KConfigStringLen <= KSqlSrvMaxConfigStrLen, aMessage, ESqlPanicBadArgument);
557 RBuf8 securityAndConfigBuf;
558 CleanupClosePushL(securityAndConfigBuf);
559 if((KSecurityPolicyLen + KConfigStringLen) > 0)
561 securityAndConfigBuf.CreateL(KSecurityPolicyLen + KConfigStringLen);
562 aMessage.ReadL(3, securityAndConfigBuf);
563 SQLPROFILER_REPORT_IPC(ESqlIpcRead, (KSecurityPolicyLen + KConfigStringLen));
565 TSqlSrvFileData& fileData = Server().FileData();
566 TPtrC8 configStr(KNullDesC8);
567 if(KConfigStringLen > 0)
569 configStr.Set(securityAndConfigBuf.Mid(KSecurityPolicyLen));//the first part of the buffer is for the security policies
571 fileData.SetL(aMessage, aMessage.Int0(), 1, &configStr);
572 iDrive = fileData.Drive();
575 case ESqlSrvDbCreate:
576 if(fileData.IsSecureFileNameFmt())
578 __SQLLEAVE(KErrArgument);
580 iDatabase = CSqlSrvDatabase::CreateL(fileData);
582 case ESqlSrvDbCreateSecure:
584 __SQLPANIC_CLIENT(KSecurityPolicyLen > 0, aMessage, ESqlPanicBadArgument);
585 if(!fileData.IsSecureFileNameFmt())
587 __SQLLEAVE(KErrArgument);
589 //The caller can create a secure database which secure UID matches his secure UID.
590 if(fileData.SecureUid() != aMessage.SecureId())
592 __SQLLEAVE(KErrPermissionDenied);
594 CSqlSecurityPolicy* policy = CreateSecurityPolicyL(securityAndConfigBuf.Left(KSecurityPolicyLen));
595 iDatabase = CSqlSrvDatabase::CreateSecureL(fileData, policy);
599 iDatabase = CSqlSrvDatabase::OpenL(fileData);
602 __SQLLEAVE(KErrArgument);
605 CleanupStack::PopAndDestroy(&securityAndConfigBuf);
609 Processes the request for opening a database from file handle.
610 The server expects that the database to be opened/created is in the applicatio's private data cage.
612 Usage of the IPC call arguments:
613 - Arg 0: [in] The 32 bits of the argument are used as follow:
616 31 30 29 28 27 26 25 24 23 22 21 20 19 18 17 16 15 14 13 12 11 10 9 8 7 6 5 4 3 2 1 0
617 Ro Cr C C C C C C C C C C C C C C F F F F F F F F F F F F F F F F
621 - "Ro" - read-only flag, true if the file is read-only;
622 - "Cr" - create/open flag, true if the file was created, false if the file was opened;
623 - "C" - config string length in 16-bit characters;
624 - "F" - database file name length in 16-bit characters;
626 - Arg 1: [in] database file name | config string
627 - Arg 2: [in] file session handle
628 - Arg 3: [in] database file handle
630 @panic SqlDb 1 Client panic. iDatabase is not NULL (it has been created already)
632 void CSqlSrvSession::DbCreateObjectFromHandleL(const RMessage2& aMessage)
634 __SQLPANIC_CLIENT(!iDatabase, aMessage, ESqlPanicObjExists);
635 const TBool KReadOnly = (aMessage.Int0() & 0x80000000) != 0;
636 const TBool KCreated = (aMessage.Int0() & 0x40000000) != 0;
637 const TInt KDbFileNameLen = aMessage.Int0() & 0x0000FFFF;
638 const TInt KConfigStringLen = (aMessage.Int0() & 0x3FFF0000) >> 16;
639 __SQLPANIC_CLIENT((TUint)KConfigStringLen <= KSqlSrvMaxConfigStrLen, aMessage, ESqlPanicBadArgument);
640 __SQLPANIC_CLIENT((TUint)KDbFileNameLen <= KMaxFileName, aMessage, ESqlPanicBadArgument);
641 TDes16& buffer = Server().GetBuf16L(KDbFileNameLen + KConfigStringLen);
642 aMessage.ReadL(1, buffer);
643 SQLPROFILER_REPORT_IPC(ESqlIpcRead, ((KDbFileNameLen + KConfigStringLen) * sizeof(TText)));
644 TFileName dbFileName;
645 dbFileName.Copy(buffer.LeftTPtr(KDbFileNameLen));
646 TBuf8<KSqlSrvMaxConfigStrLen> configStr;
647 if(KConfigStringLen > 0)
649 configStr.Copy(buffer.MidTPtr(KDbFileNameLen, KConfigStringLen));
651 TSqlSrvFileData& fileData = Server().FileData();
652 fileData.SetFromHandleL(aMessage, dbFileName, KCreated, KReadOnly, &configStr);
653 iDrive = fileData.Drive();
654 iDatabase = CSqlSrvDatabase::OpenL(fileData);
658 Processes the request for copying a database.
660 Only the database creator can copy the database if the database is a secure database.
662 Usage of the IPC call arguments:
663 Arg 0: [in] source database file name length
664 Arg 1: [in] source database file name
665 Arg 2: [in] destination database file name length
666 Arg 3: [in] destination database file name
668 void CSqlSrvSession::DbCopyFileL(const RMessage2& aMessage)
670 const TInt KDbCnt = 2; //"2" - because we have 2 dbases: src and dest
671 const TInt KSrcDbIdx = 0;
672 const TInt KDestDbIdx = 1;
673 TInt fileNameLen[KDbCnt] = {aMessage.Int0(), aMessage.Int2()};
674 TSqlSrvFileData& fileData = Server().FileData();
675 TUint dbSecureFlag[KDbCnt];
676 TUid dbSID[KDbCnt] = {KNullUid, KNullUid};
677 TFileName dbFileName[KDbCnt];
678 //Initialize dbSecureFlag[], dbSID[] and dbFileName[] array elements
679 for(TInt i=0;i<KDbCnt;++i)
681 fileData.SetL(aMessage, fileNameLen[i], i * KDbCnt + 1); //"i * KDbCnt + 1" is the RMessage2 parameter number: 1 for src db, 3 for dest db
682 dbSecureFlag[i] = fileData.IsSecureFileNameFmt() ? 1 : 0;
685 dbSID[i] = fileData.SecureUid();
687 dbFileName[i].Copy(fileData.FileName());
689 //It is not allowed to copy non-secure to a secure or secure to a non-secure database.
690 if(dbSecureFlag[KSrcDbIdx] ^ dbSecureFlag[KDestDbIdx])
692 __SQLLEAVE(KErrPermissionDenied);
694 //If this is a secure database "copy" operation, then...
695 if(dbSecureFlag[KSrcDbIdx])
697 TUid callerSid = aMessage.SecureId();
698 //A secure database can be copied only by its owner (database SID matches caller SID).
699 if(callerSid != dbSID[KSrcDbIdx] || callerSid != dbSID[KDestDbIdx])
701 __SQLLEAVE(KErrPermissionDenied);
705 CFileMan* fileMan = CFileMan::NewL(fileData.Fs());
706 CleanupStack::PushL(fileMan);
707 __SQLLEAVE_IF_ERROR(fileMan->Copy(dbFileName[KSrcDbIdx], dbFileName[KDestDbIdx]));
708 //"Copy" operation executed without errors. Now it is a time to turn off the read-only
709 //flag of the target file (which may be on if the source file is on a read-only drive)
710 __SQLLEAVE_IF_ERROR(fileData.Fs().SetAtt(dbFileName[KDestDbIdx], 0, KEntryAttReadOnly));
711 CleanupStack::PopAndDestroy(fileMan);
715 Processes the request for deleting a database.
717 Only the database creator can delete the database if the database is a secure database.
719 Usage of the IPC call arguments:
720 Arg 0: [in] database file name length
721 Arg 1: [in] database file name
723 void CSqlSrvSession::DbDeleteFileL(const RMessage2& aMessage)
725 TSqlSrvFileData& fileData = Server().FileData();
726 fileData.SetL(aMessage, aMessage.Int0(), 1);
727 if(fileData.IsSecureFileNameFmt())
729 //A secure database can be deleted only by its owner (database SID matches caller SID).
730 if(fileData.SecureUid() != aMessage.SecureId())
732 __SQLLEAVE(KErrPermissionDenied);
735 __SQLTRACE_SESSIONVAR(TPtrC fname(fileData.FileName()));
736 SQL_TRACE_SESSION(OstTraceExt2(TRACE_INTERNALS, CSQLSRVSESSION_DBDELETEFILEL, "0x%X;CSqlSrvSession::DbDeleteFileL;file='%S'", (TUint)this, __SQLPRNSTR(fname)));
737 __SQLLEAVE_IF_ERROR(fileData.Fs().Delete(fileData.FileName()));
741 Processes the request for retrieving the last error message.
743 If the client side buffer size is not big enough, the function returns the needed
744 buffer size + KSqlClientBufOverflowCode.
745 In this case the client must increase the buffer and try again to get the message.
747 Usage of the IPC call arguments:
748 Arg 0: [in] Message buffer length in 16-bit characters
749 Arg 1: [in/out] Message buffer
751 @panic SqlDb 2 Client panic. iDatabase is NULL (the database object is not created yet).
752 @panic SqlDb 4 Client panic. Negative client message buffer length (Arg 0).
754 TInt CSqlSrvSession::DbLastErrorMessageL(const RMessage2& aMessage)
756 __SQLPANIC_CLIENT(iDatabase != NULL, aMessage, ESqlPanicInvalidObj);
757 TPtrC msg = iDatabase->LastErrorMessage();
758 TInt msgLen = msg.Length();
759 TInt bufSize = aMessage.Int0();
760 __SQLPANIC_CLIENT(bufSize >= 0, aMessage, ESqlPanicBadArgument);
761 if(msgLen <= bufSize)
763 aMessage.WriteL(1, msg);
764 SQLPROFILER_REPORT_IPC(ESqlIpcWrite, (msgLen * sizeof(TText)));
767 return msgLen + KSqlClientBufOverflowCode;
771 Processes the request for retrieving the last inserted ROWID of this database connection.
773 Usage of the IPC call arguments:
774 Arg 0: [in/out] Receiving buffer
776 @panic SqlDb 2 Client panic. The database object is not yet created yet (iDatabase is NULL).
778 void CSqlSrvSession::DbLastInsertedRowIdL(const RMessage2& aMessage)
780 __SQLPANIC_CLIENT(iDatabase != NULL, aMessage, ESqlPanicInvalidObj);
781 TInt64 rowid = iDatabase->LastInsertedRowId();
782 aMessage.WriteL(0, TPtrC8(reinterpret_cast <const TUint8*> (&rowid), sizeof(rowid)));
783 SQLPROFILER_REPORT_IPC(ESqlIpcWrite, sizeof(rowid));
787 Processes the request for retrieving the database security policies.
789 The method leaves with KErrNotSupported if the database is not a secure database.
791 If the client side buffer size is not big enough, the function returns the needed
792 buffer size + KSqlClientBufOverflowCode.
793 In this case the client must increase the buffer and try again to get the database security policy.
795 Usage of the IPC call arguments:
796 Arg 0: [in] security policy buffer length in bytes
797 Arg 1: [in/out] buffer for the database security policies
799 @panic SqlDb 2 Client panic. iDatabase is NULL (the database object is not created yet).
801 TInt CSqlSrvSession::DbGetSecurityPolicyL(const RMessage2& aMessage)
803 __SQLPANIC_CLIENT(iDatabase != NULL, aMessage, ESqlPanicInvalidObj);
804 const CSqlSecurityPolicy* securityPolicy = iDatabase->SecurityPolicy();
807 __SQLLEAVE(KErrNotSupported);
809 const RSqlBufFlat& bufFlat = securityPolicy->BufFlat();
810 TInt size = bufFlat.Size();
811 if(size <= aMessage.Int0())
813 aMessage.WriteL(1, bufFlat.BufDes());
814 SQLPROFILER_REPORT_IPC(ESqlIpcWrite, size);
817 return size + KSqlClientBufOverflowCode;
821 If an error occurs during the execution the function leaves with the error code.
822 Possible non-leaving return values:
823 - KErrNone - the function has completed successfully;
824 - Positive return value - the length of the column, which means - the destination buffer is too small.
825 This return value is possible only with text or binary columns.
827 @panic SqlDb 2 Client panic. iDatabase is NULL (the database object is not created yet).
829 Usage of the IPC call arguments:
830 Arg 0: [in] (8/16-bit character length of SQL statement) | (expected column value type << 24).
831 Arg 1: [in] SQL statement.
832 Arg 2: [in] Byte max length of the receiving buffer
833 Arg 3: [in/out] The receiving buffer
835 TInt CSqlSrvSession::DbScalarFullSelectL(const RMessage2& aMessage, TBool aIsText16)
837 __SQLPANIC_CLIENT(iDatabase != NULL, aMessage, ESqlPanicInvalidObj);
838 TUint sqlLen = static_cast <TUint> (aMessage.Int0()) & 0x00FFFFFF;
839 TSqlColumnType colType = static_cast <TSqlColumnType> ((static_cast <TUint> (aMessage.Int0()) & 0xFF000000) >> 24);
840 TInt columnCount = -1;
841 TInt paramCount = -1;
842 CSqlSrvStatement* stmt = aIsText16 ?
843 CSqlSrvStatement::NewLC(iDatabase->RawDbHandle(), ReadString16ZL(aMessage, 1, sqlLen), columnCount, paramCount) :
844 CSqlSrvStatement::NewLC(iDatabase->RawDbHandle(), ReadString8ZL(aMessage, 1, sqlLen), columnCount, paramCount);
845 if(columnCount != 1 || paramCount != 0)
847 __SQLLEAVE(KErrArgument);
849 TInt err = stmt->Next();
852 err = GetColumnValueL(aMessage, *stmt, colType);
856 __SQLLEAVE(err == KSqlAtEnd ? KErrNotFound : err);
858 CleanupStack::PopAndDestroy(stmt);
863 @return True, if the database is in transaction, false otherwise.
865 @panic SqlDb 2 Client panic. iDatabase is NULL (the database object is not created yet).
867 TBool CSqlSrvSession::DbInTransaction(const RMessage2& aMessage)
869 __SQLPANIC_CLIENT(iDatabase != NULL, aMessage, ESqlPanicInvalidObj);
870 return iDatabase->InTransaction();
874 Main database size in bytes.
876 @return Main database size in bytes.
878 @leave KErrNoMemory, an out of memory condition has occurred,
879 Note that the function may also leave with some other system wide errors or
880 database specific errors categorised as ESqlDbError,
881 KErrTooBig, The database is very big and the size cannot fit in a 32-bit signed integer.
883 @panic SqlDb 2 Client panic. iDatabase is NULL (the database object is not created yet).
885 TInt CSqlSrvSession::DbSizeL(const RMessage2& aMessage)
887 __SQLPANIC_CLIENT(iDatabase != NULL, aMessage, ESqlPanicInvalidObj);
888 TInt64 size = iDatabase->SizeL();
891 __SQLLEAVE(KErrTooBig);
897 Retrieves the database size and free space.
899 Usage of the IPC call arguments:
900 Arg 0: [in/out] Points to a RSqlDatabase::TSize object, where the database size and free space values
902 Arg 1: [in] The database name length in 16-bit characters
903 Arg 2: [in] The attached database name or KNullDesC for the main database
905 @leave KErrNoMemory, an out of memory condition has occurred,
906 Note that the function may also leave with some other system wide errors or
907 database specific errors categorised as ESqlDbError.
908 KErrBadName, Invalid database name
910 @panic SqlDb 2 Client panic. iDatabase is NULL (the database object is not created yet).
912 void CSqlSrvSession::DbSize2L(const RMessage2& aMessage)
914 __SQLPANIC_CLIENT(iDatabase != NULL, aMessage, ESqlPanicInvalidObj);
915 const TInt KDbNameLen = aMessage.Int1();
916 if((TUint)KDbNameLen > KMaxFileName)
918 __SQLLEAVE(KErrBadName);
920 TPtrC dbName(KNullDesC);
923 dbName.Set(ReadString16L(aMessage, 2, KDbNameLen));
925 TPckgBuf<RSqlDatabase::TSize> data;
926 data().iSize = iDatabase->SizeL(dbName);
927 data().iFree = iDatabase->FreeSpaceL(dbName);
928 aMessage.WriteL(0, data);
929 SQLPROFILER_REPORT_IPC(ESqlIpcWrite, sizeof(RSqlDatabase::TSize));
933 Runs database compaction.
935 Usage of the IPC call arguments:
936 Arg 0: [in] How much space in bytes should be compacted, all free pages should be removed if the
937 parameter value is RSqlDatabase::EMaxCompaction.
938 Note that the requested space to be compacted will be rounded up to the nearest page count,
939 e.g. request for removing 1 byte will remove one free page from the database file.
940 Arg 1: [in] The database name length in characters
941 Arg 2: [in] The attached database name or KNullDesC for the main database
943 @return The size of the removed free space
945 @panic SqlDb 2 Client panic. iDatabase is NULL (the database object is not created yet).
947 TInt CSqlSrvSession::DbCompactL(const RMessage2& aMessage)
949 __SQLPANIC_CLIENT(iDatabase != NULL, aMessage, ESqlPanicInvalidObj);
950 const TInt KSize = aMessage.Int0();
953 if(KSize != RSqlDatabase::EMaxCompaction)
955 __SQLLEAVE(KErrArgument);
962 const TInt KDbNameLen = aMessage.Int1();
963 if((TUint)KDbNameLen > KMaxFileName)
965 __SQLLEAVE(KErrBadName);
967 TPtrC dbName(KNullDesC);
970 dbName.Set(ReadString16L(aMessage, 2, KDbNameLen));
972 return iDatabase->CompactL(KSize, dbName);
976 Usage of the IPC call arguments:
977 Arg 0: [in] requested size of the space to be reserved - not used
979 The function leaves with KErrAlreadyExists if a drive space has been reserved already by this session.
981 void CSqlSrvSession::DbReserveDriveSpaceL()
983 if(iDriveSpaceReserved)
985 __SQLLEAVE(KErrAlreadyExists);
987 RSqlDriveSpaceCol& driveSpaceCol = Server().DriveSpaceCol();
988 if(!driveSpaceCol.Find(iDrive))
990 (void)driveSpaceCol.AddL(iDrive);
993 iDriveSpaceReserved = ETrue;
994 //Only iDriveSpaceReserved is set, nothing more needs to be done, because RSqlDriveSpaceCol::AddL() will
995 //reserve a drive space on the specified drive.
996 //Although it looks like that the implementation can ommit "iDriveSpaceReserved" flag, this flag plays important
997 //role, because it is used to ensure that every "reserve drive space" request is matched by a "free drive space"
1002 If the client has been given an access to the reserved drive space, that access will be released.
1004 void CSqlSrvSession::DbFreeReservedSpace()
1006 DbReleaseReserveAccess();
1007 iDriveSpaceReserved = EFalse;
1011 The function leaves with KErrInUse if an access to the reserved drive space has been given to the client.
1012 The function leaves with KErrNotFound if no drive space has been reserved for the drive, where the database file is.
1014 void CSqlSrvSession::DbGetReserveAccessL()
1016 if(iDriveSpaceInUse)
1018 __SQLLEAVE(KErrInUse);
1020 if(!iDriveSpaceReserved)
1022 __SQLLEAVE(KErrNotFound);
1024 CSqlDriveSpace* driveSpace = Server().DriveSpaceCol().Find(iDrive);
1027 __SQLLEAVE(KErrNotFound);
1029 driveSpace->GetAccessL();
1030 iDriveSpaceInUse = ETrue;
1034 Releases the drive space reserve if it has been in use by this session (resp. client).
1036 void CSqlSrvSession::DbReleaseReserveAccess()
1038 if(iDriveSpaceInUse)
1040 CSqlDriveSpace* driveSpace = Server().DriveSpaceCol().Find(iDrive);
1043 driveSpace->ReleaseAccess();
1045 iDriveSpaceInUse = EFalse;
1050 Processes the request for attaching a secure or non-secure database.
1052 Usage of the IPC call arguments:
1053 Arg 0: [in] Database file name length (counted in 16-bit characters).
1054 Arg 1: [in] Database file name.
1055 Arg 2: [in] Logical database name length (counted in 16-bit characters).
1056 Arg 3: [in] Logical database name.
1058 @panic SqlDb 2 Client panic. iDatabase is NULL (the database object is not created yet).
1060 void CSqlSrvSession::DbAttachL(const RMessage2& aMessage)
1062 __SQLPANIC_CLIENT(iDatabase != NULL, aMessage, ESqlPanicInvalidObj);
1063 TSqlSrvFileData& fileData = Server().FileData();
1064 fileData.SetL(aMessage, aMessage.Int0(), 1);
1065 TInt logicalDbNameLen = aMessage.Int2();
1066 if(logicalDbNameLen < 1 || logicalDbNameLen > KMaxFileName)
1068 __SQLLEAVE(KErrBadName);
1070 iDatabase->AttachDbL(fileData, ReadString16L(aMessage, 3, logicalDbNameLen));
1074 Processes the request for attaching a database using file session and file handles sent by the client.
1076 Usage of the IPC call arguments:
1077 - Arg 0: [in] The 32 bits of the argument are used as follow:
1080 31 30 29 28 27 26 25 24 23 22 21 20 19 18 17 16 15 14 13 12 11 10 9 8 7 6 5 4 3 2 1 0
1081 Ro F F F F F F F F F F F F F F F F F F F F F F F F F F F F F F F
1085 - "Ro" - read-only flag, true if the file is read-only;
1086 - "F" - database file name length in 16-bit characters;
1088 Arg 1: [in] db names buffer
1089 Arg 2: [in] file session handle
1090 Arg 3: [in] database file handle
1092 @panic SqlDb 2 Client panic. iDatabase is NULL (the database object is not created yet).
1093 @panic SqlDb 4 Client panic. Invalid IPC data, an indication of a problme in client side sql library.
1095 void CSqlSrvSession::DbAttachFromHandleL(const RMessage2& aMessage)
1097 __SQLPANIC_CLIENT(iDatabase != NULL, aMessage, ESqlPanicInvalidObj);
1098 //Read-only flag, buffer length, buffer allocation
1099 TBool readOnly = (aMessage.Int0() & 0x80000000) != 0;
1100 const TInt KBufLen = aMessage.Int0() & 0x7FFFFFFF;
1101 __SQLPANIC_CLIENT(KBufLen > 0, aMessage, ESqlPanicBadArgument);
1102 HBufC8* buf = HBufC8::NewLC(KBufLen);
1103 TPtr8 bufPtr = buf->Des();
1104 aMessage.ReadL(1, bufPtr);
1105 SQLPROFILER_REPORT_IPC(ESqlIpcRead, KBufLen);
1106 if(KBufLen != bufPtr.Length())
1108 __SQLLEAVE(KErrArgument);
1110 RDesReadStream in(bufPtr);
1111 TDes& dbFileName = Server().FileNameBuf();
1112 TDes16& dbName = Server().GetBuf16L(KMaxFileName);
1115 CleanupStack::PopAndDestroy(buf);
1116 TSqlSrvFileData& fileData = Server().FileData();
1117 fileData.SetFromHandleL(aMessage, dbFileName, EFalse, readOnly);
1118 iDatabase->AttachDbL(fileData, dbName);
1122 Processes the request for detaching a secure or non-secure database.
1124 Usage of the IPC call arguments:
1125 Arg 0: [in] Logical database name length (counted in 16-bit characters).
1126 Arg 1: [in] Logical database name.
1128 @panic SqlDb 2 Client panic. iDatabase is NULL (the database object is not created yet).
1130 void CSqlSrvSession::DbDetachL(const RMessage2& aMessage)
1132 __SQLPANIC_CLIENT(iDatabase != NULL, aMessage, ESqlPanicInvalidObj);
1133 TInt logicalDbNameLen = aMessage.Int0();
1134 if(logicalDbNameLen < 1 || logicalDbNameLen > KMaxFileName)
1136 __SQLLEAVE(KErrBadName);
1138 iDatabase->DetachDbL(ReadString16L(aMessage, 1, logicalDbNameLen));
1142 Reads a 16-bit string from the specified stream and returns it in zero-terminated
1143 8-bit format in aNameOut.
1144 If the string is of zero length then the substitute string provided will be used instead.
1146 @param aStrm The read stream
1147 @param aNameOut The output parameter that will contain the string read
1148 @param aEmptyNameSubstitute The substitute string to use if the string to be read from
1149 the stream is zero length
1151 @leave KErrNoMemory, An out of memory condition has occurred;
1152 KErrArgument, The UTF-16 to UTF-8 string conversion failed;
1153 KErrBadName, The string has an invalid length;
1155 void CSqlSrvSession::ExtractNameL(RDesReadStream& aStrm, TDes8& aNameOut, const TDesC& aEmptyNameSubstitute)
1157 TBool replace = EFalse;
1163 if(aEmptyNameSubstitute.Length() > 0)
1165 len = aEmptyNameSubstitute.Length();
1170 __SQLLEAVE(KErrBadName);
1173 __ASSERT_DEBUG(len > 0, __SQLPANIC2(ESqlPanicInternalError));//The "if" above should have hanled the case with "len == 0"
1174 if((TUint)len > KMaxFileName)
1176 __SQLLEAVE(KErrBadName);
1179 HBufC* buf = HBufC::NewLC(len + 1);
1180 TPtr ptr = buf->Des();
1183 ptr.Copy(aEmptyNameSubstitute);
1191 if(!::UTF16ZToUTF8Z(ptr, aNameOut))
1193 __SQLLEAVE(KErrArgument);
1196 CleanupStack::PopAndDestroy(buf);
1200 Processes the request for creating an IPC stream for accessing the content of a blob column.
1202 @param aMessage The client request wrapped in an IPC message
1204 @return The blob stream handle
1206 @leave KErrNoMemory, An out of memory condition has occurred;
1207 KErrArgument, The ROWID is invalid or UTF-16 to UTF-8 string conversion failed;
1208 KErrBadDescriptor The transferred data is bigger than the specified length;
1209 KErrBadName, The table name, column name or database name has an invalid length;
1210 KErrPermissionDenied, The client does not have the required security capabilites for this operation;
1211 Note that the function may also leave with some other system wide errors or
1212 database specific errors categorised as ESqlDbError.
1214 @panic SqlDb 2 Client panic. The database object is not yet created (iDatabase is NULL).
1215 @panic SqlDb 3 Client panic. Failed to create a blob stream handle.
1216 @panic SqlDb 4 Client panic. IPC buffer length is 0.
1218 Usage of the IPC call arguments:
1219 Arg 0: [in] The length of the IPC data buffer
1220 Arg 1: [in] IPC data buffer containing blob parameters: table name, column name, rowid, mode, database name.
1221 Arg 2: [out] IPC buffer containing the blob stream handle
1223 TInt CSqlSrvSession::DbBlobSourceL(const RMessage2& aMessage)
1225 __SQLPANIC_CLIENT(iDatabase != NULL, aMessage, ESqlPanicInvalidObj);
1227 TInt ipcPrmLen = aMessage.Int0();
1228 __SQLPANIC_CLIENT(ipcPrmLen > 0, aMessage, ESqlPanicBadArgument);
1230 iIpcStreams.AllocL();
1232 TDes8& ipcPrmDes = ReadString8ZL(aMessage, 1, ipcPrmLen);
1233 RDesReadStream strm(ipcPrmDes);
1235 TBuf8<KMaxFileName + 1> tblName;
1236 ExtractNameL(strm, tblName);
1238 TBuf8<KMaxFileName + 1> colName;
1239 ExtractNameL(strm, colName);
1245 rowId = iDatabase->LastInsertedRowId();
1249 __SQLLEAVE(KErrArgument);
1254 TBool isReadOnly = tmp != 0;
1256 TBuf8<KMaxFileName + 1> dbName;
1257 ExtractNameL(strm, dbName, KMainDb16);
1261 // If the database is secure then check that the client has the required capabilities for the operation
1262 TInt dbOpType = isReadOnly ? SQLITE_READ : SQLITE_UPDATE;
1263 if(CSqlSrvDatabase::AuthorizeCallback(iDatabase, dbOpType, (char*)tblName.Ptr(), (char*)colName.Ptr(), (char*)dbName.Ptr(), '\0') != SQLITE_OK)
1265 __SQLLEAVE(KErrPermissionDenied);
1268 // Create the stream buffer
1269 HBlobBuf* blobBuf = HBlobBuf::NewL(iDatabase->RawDbHandle(), dbName, tblName, colName, rowId, isReadOnly ? HBlobBuf::EReadOnly : HBlobBuf::EReadWrite);
1272 // Return the blob size to the client
1273 TPckgBuf<TIpcStreamBuf> ipcBuf;
1274 TInt size = blobBuf->SizeL();
1275 ipcBuf().iExt = size;
1277 // If this is a read stream then return the first client buffer-full of data
1279 if(isReadOnly && (size > 0))
1281 len = Min(size, KIpcBufSize);
1282 blobBuf->ReadL(ipcBuf().iData, len);
1285 // Create the stream object
1286 HIpcStream* ipcStream = new (ELeave) HIpcStream(blobBuf, len);
1287 TInt strmHandle = iIpcStreams.Add(ipcStream);
1288 __SQLPANIC_CLIENT(strmHandle > 0, aMessage, ESqlPanicBadHandle);
1289 CleanupStack::Pop(blobBuf);
1291 // Send the size and data to the client
1292 aMessage.WriteL(2, ipcBuf);
1293 SQLPROFILER_REPORT_IPC(ESqlIpcWrite, size);
1298 ///////////////////////////////////////////////////////////////////////////////////////////////////////
1299 //////////////////////////// Statement operations //////////////////////////////////
1300 ///////////////////////////////////////////////////////////////////////////////////////////////////////
1303 Processes the request for preparing a 8/16-bit SQL statement.
1305 @panic SqlDb 2 Client panic. iDatabase is NULL (the database object is not created yet).
1306 @panic SqlDb 3 Client panic. Internal error - invalid statement handle.
1308 Usage of the IPC call arguments:
1309 Arg 0: [out] Column count and parameter count
1310 Arg 1: [in] 8/16-bit character length of SQL statement
1311 Arg 2: [in] SQL statement
1313 TInt CSqlSrvSession::StmtPrepareL(const RMessage2& aMessage, TBool aIsText16)
1315 __SQLPANIC_CLIENT(iDatabase != NULL, aMessage, ESqlPanicInvalidObj);
1316 iStatements.AllocL();
1317 TInt columnCount = -1;
1318 TInt paramCount = -1;
1319 TUint len = static_cast <TUint> (aMessage.Int1());
1320 CSqlSrvStatement* stmt = aIsText16 ?
1321 CSqlSrvStatement::NewLC(iDatabase->RawDbHandle(), ReadString16ZL(aMessage, 2, len), columnCount, paramCount) :
1322 CSqlSrvStatement::NewLC(iDatabase->RawDbHandle(), ReadString8ZL(aMessage, 2, len), columnCount, paramCount);
1323 TPckgBuf<TSqlIpcData> data;
1324 data().iPrm1 = static_cast <TUint32> (columnCount);
1325 data().iPrm2 = static_cast <TUint32> (paramCount);
1326 aMessage.WriteL(0, data);
1327 SQLPROFILER_REPORT_IPC(ESqlIpcWrite, sizeof(TSqlIpcData));
1328 TInt stmtHandle = iStatements.Add(stmt);
1329 __SQLPANIC_CLIENT(stmtHandle > 0, aMessage, ESqlPanicBadHandle);
1330 CleanupStack::Pop(stmt);
1335 Processes the request for executing the SQL statement.
1337 @param aFunction ESqlSrvStmtExec, ESqlSrvStmtAsyncExec, ESqlSrvStmtBindExec, ESqlSrvStmtBindExecRowId, ESqlSrvStmtAsyncBindExec
1339 Usage of the IPC call arguments:
1340 Arg 0: [in] parameter buffer length in bytes (if aFunction == ESqlSrvStmtBindExec || aFunction == ESqlSrvStmtAsyncBindExec)
1341 Arg 1: [in] parameter buffer (if aFunction == ESqlSrvStmtBindExec || aFunction == ESqlSrvStmtAsyncBindExec)
1343 TInt CSqlSrvSession::StmtExecL(const RMessage2& aMessage, TInt aStmtHandle, TSqlSrvFunction aFunction)
1345 CSqlSrvStatement& stmt = ::SqlSessObjFind(iStatements, aStmtHandle, aMessage);
1346 if(aFunction == ESqlSrvStmtBindExec || aFunction == ESqlSrvStmtAsyncBindExec)
1348 DoStmtBindL(aMessage, stmt);
1350 __SQLLEAVE_IF_ERROR(stmt.Exec());
1351 return iDatabase->LastChangesCount();
1355 Processes the request for moving the SQL statement on the next record.
1357 If the call does not fail, the only valid acceptable return codes should be KSqlAtRow and KSqlAtEnd.
1359 @panic SqlDb 7 In _DEBUG mode. The call completed with no error but the return code is not KSqlAtRow or KSqlAtEnd.
1361 Usage of the IPC call arguments:
1362 Arg 0: [in] parameter buffer length in bytes (if aFunction == ESqlSrvStmtBindNext)
1363 Arg 1: [in] parameter buffer (if aFunction == ESqlSrvStmtBindNext)
1364 Arg 2: [in] client side column buffer length in bytes
1365 Arg 3: [out] column buffer
1367 TInt CSqlSrvSession::StmtNextL(const RMessage2& aMessage, TInt aStmtHandle, TSqlSrvFunction aFunction)
1369 CSqlSrvStatement& stmt = ::SqlSessObjFind(iStatements, aStmtHandle, aMessage);
1370 if(aFunction == ESqlSrvStmtBindNext)
1372 DoStmtBindL(aMessage, stmt);
1374 TInt err = stmt.Next();
1375 if(err == KSqlAtRow)
1377 const RSqlBufFlat& bufFlat = stmt.ColumnValuesL();
1378 TInt size = bufFlat.Size();
1379 if(size > aMessage.Int2())
1381 return size + KSqlClientBufOverflowCode;
1383 aMessage.WriteL(3, bufFlat.BufDes());
1384 SQLPROFILER_REPORT_IPC(ESqlIpcWrite, size);
1386 __SQLLEAVE_IF_ERROR(err);
1387 __ASSERT_DEBUG(err == KSqlAtRow || err == KSqlAtEnd, __SQLPANIC(ESqlPanicInternalError));
1392 Processes the request for retrieving the statement column or parameter names.
1394 If the client side buffer size is not big enough, the function returns the size + KSqlClientBufOverflowCode.
1395 In this case the client must increase the buffer and try again to get the buffer only.
1397 Usage of the IPC call arguments:
1398 Arg 0: [in] size of the client side buffer for the names (in bytes)
1399 Arg 1: [out] ipc buffer, column or parameter names
1401 TInt CSqlSrvSession::StmtNamesL(const RMessage2& aMessage, TInt aStmtHandle, TSqlSrvFunction aFunction)
1403 CSqlSrvStatement& stmt = ::SqlSessObjFind(iStatements, aStmtHandle, aMessage);
1404 const RSqlBufFlat& namesBuf = aFunction == ESqlSrvStmtParamNames ? stmt.ParamNamesL() : stmt.ColumnNamesL();
1405 TInt size = namesBuf.Size();
1406 if(size <= aMessage.Int0())
1408 aMessage.WriteL(1, namesBuf.BufDes());
1409 SQLPROFILER_REPORT_IPC(ESqlIpcWrite, size);
1412 return size + KSqlClientBufOverflowCode;
1416 Processes the request for accessing a large column value as a stream of bytes/characters.
1418 Usage of the IPC call arguments:
1419 Arg 0: [in] column index (0 based)
1420 Arg 2: [out] ipc buffer, column source
1422 TInt CSqlSrvSession::StmtColumnSourceL(const RMessage2& aMessage, TInt aStmtHandle)
1424 CSqlSrvStatement& stmt = ::SqlSessObjFind(iStatements, aStmtHandle, aMessage);
1425 TInt columnIndex = aMessage.Int0();
1426 TPtrC8 columnSource;
1427 TInt err = stmt.ColumnSource(columnIndex, columnSource);
1428 __SQLLEAVE_IF_ERROR(err);
1429 HIpcReadBuf* ipcBuf = HIpcReadBuf::NewL(columnSource);
1430 return NewOutputStreamL(aMessage, ipcBuf);
1434 Processes the request for setting a large parameter value from a stream of bytes or 8/16-bit characters.
1436 @panic SqlDb 3 Client panic. Internal error - invalid stream handle.
1438 Usage of the IPC call arguments:
1439 Arg 0: [in] parameter index (0 based)
1440 Arg 2: [out] ipc buffer, parameter source
1442 TInt CSqlSrvSession::StmtParamSinkL(const RMessage2& aMessage, TInt aStmtHandle, TSqlSrvFunction aFunction)
1444 iIpcStreams.AllocL();
1445 TInt parameterIndex = aMessage.Int0();
1446 CSqlSrvStatement& stmt = ::SqlSessObjFind(iStatements, aStmtHandle, aMessage);
1447 HSqlSrvStmtParamBuf::TDataType dataType = aFunction == ESqlSrvStmtBinParamSink ? HSqlSrvStmtParamBuf::EBinary : HSqlSrvStmtParamBuf::EText16;
1448 HSqlSrvStmtParamBuf* paramBuf = stmt.GetParamBufL(parameterIndex, dataType, HSqlSrvStmtParamBuf::EBufIpcStream);
1449 HIpcStream* ipcStream = new (ELeave) HIpcStream(paramBuf, 0);
1450 TInt strmHandle = iIpcStreams.Add(ipcStream);
1451 __SQLPANIC_CLIENT(strmHandle > 0, aMessage, ESqlPanicBadHandle);
1456 Usage of the IPC call arguments:
1457 Arg 0: [in] the client side buffer length in bytes
1458 Arg 1: [out] refers to a place where the buffer data will be copied to.
1460 void CSqlSrvSession::StmtGetBufFlatL(const RMessage2& aMessage, TInt aStmtHandle)
1462 CSqlSrvStatement& stmt = ::SqlSessObjFind(iStatements, aStmtHandle, aMessage);
1463 const RSqlBufFlat& bufFlat = stmt.BufFlatL(static_cast <TSqlBufFlatType> (aMessage.Int0()));
1464 aMessage.WriteL(1, bufFlat.BufDes());
1465 SQLPROFILER_REPORT_IPC(ESqlIpcWrite, bufFlat.Size());
1469 Usage of the IPC call arguments:
1470 Arg 0: [in] column index
1471 Arg 1: [in] column buffer length in bytes
1472 Arg 2: [in/out] column buffer
1474 void CSqlSrvSession::StmtColumnValueL(const RMessage2& aMessage, TInt aStmtHandle)
1476 CSqlSrvStatement& stmt = ::SqlSessObjFind(iStatements, aStmtHandle, aMessage);
1477 TPtrC8 columnSource;
1478 TInt columnIndex = aMessage.Int0();
1479 TInt err = stmt.ColumnSource(columnIndex, columnSource);
1480 __SQLLEAVE_IF_ERROR(err);
1481 TInt len = aMessage.Int1();
1482 if(columnSource.Length() > len)
1484 TPtr8 ptr(const_cast <TUint8*> (columnSource.Ptr()), columnSource.Length(), columnSource.Length());
1486 aMessage.WriteL(2, ptr);
1487 SQLPROFILER_REPORT_IPC(ESqlIpcWrite, len);
1488 __SQLLEAVE(KErrOverflow);
1492 aMessage.WriteL(2, columnSource);
1493 SQLPROFILER_REPORT_IPC(ESqlIpcWrite, len);
1498 Usage of the IPC call arguments:
1499 Arg 0: [in] parameter buffer length in bytes
1500 Arg 1: [in] parameter buffer
1502 void CSqlSrvSession::DoStmtBindL(const RMessage2& aMessage, CSqlSrvStatement& aStmt)
1504 TInt prmLen = aMessage.Int0();
1505 RSqlBufFlat& prmBuf = Server().GetFlatBufL(prmLen);
1506 aMessage.ReadL(1, prmBuf.BufPtr());
1507 SQLPROFILER_REPORT_IPC(ESqlIpcRead, prmLen);
1508 aStmt.BindL(prmBuf);
1512 Processes the request for retrieving the statement declared column type names.
1514 If the client side buffer size is not big enough, the function returns the size + KSqlClientBufOverflowCode.
1515 In this case the client must increase the buffer and try again to get the buffer only
1517 Usage of the IPC call arguments:
1518 Arg 0: [in] input buffer max length in 16-bit characters
1519 Arg 1: [out] ipc buffer, declared column type names
1521 TInt CSqlSrvSession::StmtDeclColumnTypesL(const RMessage2& aMessage, TInt aStmtHandle)
1523 CSqlSrvStatement& stmt = ::SqlSessObjFind(iStatements, aStmtHandle, aMessage);
1524 const RSqlBufFlat& declColumnTypesBuf = stmt.GetDeclColumnTypesL();
1525 TInt size = declColumnTypesBuf.Size();
1526 if(size <= aMessage.Int0())
1528 aMessage.WriteL(1, declColumnTypesBuf.BufDes());
1531 return size + KSqlClientBufOverflowCode;
1535 //////////////////////////////////////////////////////////////////////////////////////////////////////////
1536 //////////////////////// Helper methods /////////////////////////////////
1537 //////////////////////////////////////////////////////////////////////////////////////////////////////////
1540 Creates a new output IPC stream object using the aStreamBuf parameter as a stream buffer (stream data source).
1542 This method immediately pushes aStreamBuf onto the cleanup stack before creating a new output IPC
1543 stream and so callers of this method should ensure that aStreamBuf is not already on the cleanup stack.
1545 Returns the handle of the created stream.
1547 @panic SqlDb 3 Client panic. Internal error - invalid stream handle.
1549 Usage of the IPC call arguments:
1550 Arg 2: [in/out] IPC buffer
1552 TInt CSqlSrvSession::NewOutputStreamL(const RMessage2& aMessage, MStreamBuf* aStreamBuf)
1554 aStreamBuf->PushL();
1555 iIpcStreams.AllocL();
1556 TInt size = aStreamBuf->SizeL();
1557 __ASSERT_DEBUG(size >= 0, __SQLPANIC(ESqlPanicInternalError));
1558 TPckgBuf<TIpcStreamBuf> ipcBuf;
1559 // read the first buffer-full
1560 TInt len = Min(size, KIpcBufSize);
1561 aStreamBuf->ReadL(ipcBuf().iData, len);
1563 if(size > KIpcBufSize)
1564 { // create the stream object
1565 HIpcStream* ipcStream = new (ELeave) HIpcStream(aStreamBuf, KIpcBufSize);
1566 handle = iIpcStreams.Add(ipcStream);
1567 __SQLPANIC_CLIENT(handle > 0, aMessage, ESqlPanicBadHandle);
1568 CleanupStack::Pop(aStreamBuf);
1570 else // no more data to send
1572 CleanupStack::PopAndDestroy(aStreamBuf);
1574 ipcBuf().iExt = size;
1575 aMessage.WriteL(2, ipcBuf);
1576 SQLPROFILER_REPORT_IPC(ESqlIpcWrite, size);
1581 Reads a 8-bit string with "aByteLen" bytes length, which is in "aArgNum" argument of aMessage.
1582 The string will be zero terminated after the "read" operation.
1583 Returns TDes8 reference pointing to the zero-terminated string.
1585 @panic SqlDb 3 Client panic. The string length is not equal to aByteLen. If happens then it is an indication of a
1586 problem inside client side sql library.
1587 @panic SqlDb 4 Client panic. Negative aByteLen value.
1589 TDes8& CSqlSrvSession::ReadString8ZL(const RMessage2& aMessage, TInt aArgNum, TInt aByteLen)
1591 __SQLPANIC_CLIENT(aByteLen >= 0, aMessage, ESqlPanicBadArgument);
1592 TDes8& buf = Server().GetBuf8L(aByteLen + 1);
1593 aMessage.ReadL(aArgNum, buf);
1594 SQLPROFILER_REPORT_IPC(ESqlIpcRead, aByteLen);
1595 __SQLPANIC_CLIENT(buf.Length() == aByteLen, aMessage, ESqlPanicBadHandle);
1596 buf.Append(TChar(0));
1601 Reads a 16-bit string with "aCharLen" character length, which is in "aArgNum" argument of aMessage.
1602 The string will be zero terminated after the "read" operation.
1603 Returns TDes16 reference pointing to the zero-terminated string.
1605 @panic SqlDb 3 Client panic. The string length is not equal to aCharLen. If happens then it is an indication of a
1606 problem inside client side sql library.
1607 @panic SqlDb 4 Client panic. Negative aCharLen value.
1609 TDes16& CSqlSrvSession::ReadString16ZL(const RMessage2& aMessage, TInt aArgNum, TInt aCharLen)
1611 __SQLPANIC_CLIENT(aCharLen >= 0, aMessage, ESqlPanicBadArgument);
1612 TDes16& buf = Server().GetBuf16L(aCharLen + 1);
1613 aMessage.ReadL(aArgNum, buf);
1614 SQLPROFILER_REPORT_IPC(ESqlIpcRead, (aCharLen * sizeof(TText)));
1615 __SQLPANIC_CLIENT(buf.Length() == aCharLen, aMessage, ESqlPanicBadHandle);
1616 buf.Append(TChar(0));
1621 Reads a 16-bit string with "aCharLen" character length, which is in "aArgNum" argument of aMessage.
1622 Returns TDes16 reference pointing to the string.
1624 @panic SqlDb 3 Client panic. The string length is not equal to aCharLen. If happens then it is an indication of a
1625 problem inside client side sql library.
1626 @panic SqlDb 4 Client panic. Negative aCharLen value.
1628 TDes16& CSqlSrvSession::ReadString16L(const RMessage2& aMessage, TInt aArgNum, TInt aCharLen)
1630 __SQLPANIC_CLIENT(aCharLen >= 0, aMessage, ESqlPanicBadArgument);
1631 TDes16& buf = Server().GetBuf16L(aCharLen);
1632 aMessage.ReadL(aArgNum, buf);
1633 SQLPROFILER_REPORT_IPC(ESqlIpcRead, (aCharLen * sizeof(TText)));
1634 __SQLPANIC_CLIENT(buf.Length() == aCharLen, aMessage, ESqlPanicBadHandle);
1639 The method constructs a CSqlSecurityPolicy object from the passed as an argument descriptor.
1641 @param aSecurityPolicyData A descriptor with the security policy data.
1643 @return A pointer to the created CSqlSecurityPolicy instance.
1645 @leave KErrNoMemory, out of memory condition has occured.
1647 CSqlSecurityPolicy* CSqlSrvSession::CreateSecurityPolicyL(const TDesC8& aSecurityPolicyData)
1649 TSecurityPolicy defaultPolicy(TSecurityPolicy::EAlwaysFail);
1650 CSqlSecurityPolicy* dbPolicy = CSqlSecurityPolicy::NewLC(defaultPolicy);
1651 RSqlBufFlat& bufFlat = dbPolicy->BufFlat();
1652 __SQLLEAVE_IF_ERROR(bufFlat.ReAlloc(aSecurityPolicyData.Length()));
1653 bufFlat.BufPtr().Copy(aSecurityPolicyData);
1654 CleanupStack::Pop(dbPolicy);
1659 Reports how many objects are allocated by the client.
1660 If the database connection is not in a test mode, the allocated memory cells count will be ignored.
1662 TInt CSqlSrvSession::CountResources()
1664 return iStatements.Count() + iIpcStreams.Count() + (iDbResourceTestMode ? User::CountAllocCells() : 0);
1668 Extracts from aMessage:
1670 - stream or statement handle;
1671 The function will panic the client if aMessage contains bad function code or bad handle encoded in it.
1673 void CSqlSrvSession::Extract(const RMessage2& aMessage, TSqlSrvFunction& aFunction, TInt& aHandle)
1675 TInt msgCode = aMessage.Function();
1676 aFunction = static_cast <TSqlSrvFunction> (KSqlSrvFunctionMask & msgCode);
1677 //All operations with code > ESqlSrvDbDelete require valid iDatabase object.
1678 if(aFunction > ESqlSrvDbDelete)
1680 __SQLPANIC_CLIENT(iDatabase != NULL, aMessage, ESqlPanicInvalidObj);
1682 if(aFunction >= ESqlSrvStmtClose)
1684 //Extracting handle and handle type from the message code
1685 TSqlSrvHandleType handleType = static_cast <TSqlSrvHandleType> (msgCode & KSqlSrvHandleTypeMask);
1686 aHandle = (msgCode & KSqlSrvHandleMask) >> KSqlSrvHandleShiftBits;
1687 __SQLPANIC_CLIENT(aHandle > 0, aMessage, ESqlPanicBadArgument);
1688 if(aFunction < ESqlSrvStreamBase)
1690 __SQLPANIC_CLIENT(handleType == ESqlSrvStatementHandle, aMessage, ESqlPanicBadArgument);
1694 __SQLPANIC_CLIENT(handleType == ESqlSrvStreamHandle, aMessage, ESqlPanicBadArgument);
1700 The function reads aStmt column 0 value and copies it into the client buffer, accessed via aMessage argument.
1702 If an error occurs during the execution the function leaves with the error code.
1703 Possible non-leaving return values:
1704 - KErrNone - the function has completed successfully;
1705 - Positive return value - the length of the column, which means - the destination buffer is too small.
1706 This return value is possible only with text or binary columns.
1708 Usage of the IPC call arguments:
1709 Arg 0: [in] (8/16-bit character length of SQL statement) | (expected column value type << 24).
1710 Arg 1: [in] SQL statement.
1711 Arg 2: [in] Byte max length of the receiving buffer
1712 Arg 3: [in/out] The receiving buffer
1714 TInt CSqlSrvSession::GetColumnValueL(const RMessage2& aMessage, CSqlSrvStatement& aStmt, TSqlColumnType aColType)
1721 TInt val = aStmt.ColumnInt(0);
1722 aMessage.WriteL(3, TPtrC8(reinterpret_cast <const TUint8*> (&val), sizeof(val)));
1723 SQLPROFILER_REPORT_IPC(ESqlIpcWrite, sizeof(val));
1728 TInt64 val = aStmt.ColumnInt64(0);
1729 aMessage.WriteL(3, TPtrC8(reinterpret_cast <const TUint8*> (&val), sizeof(val)));
1730 SQLPROFILER_REPORT_IPC(ESqlIpcWrite, sizeof(val));
1735 TReal val = aStmt.ColumnReal(0);
1736 aMessage.WriteL(3, TPtrC8(reinterpret_cast <const TUint8*> (&val), sizeof(val)));
1737 SQLPROFILER_REPORT_IPC(ESqlIpcWrite, sizeof(val));
1745 if(aColType == ESqlText)
1747 TPtrC textVal = aStmt.ColumnTextL(0);
1748 val.Set(reinterpret_cast <const TUint8*> (textVal.Ptr()), textVal.Length() * sizeof(TUint16));
1752 val.Set(aStmt.ColumnBinary(0));
1754 TInt len = val.Length();
1755 if(len > aMessage.Int2())
1757 rc = aColType == ESqlText ? (TUint)len / sizeof(TUint16) : len;
1758 len = aMessage.Int2();
1760 aMessage.WriteL(3, val.Left(len));
1761 SQLPROFILER_REPORT_IPC(ESqlIpcWrite, len);