os/persistentdata/persistentstorage/sql/SRC/Server/SqlSrvSession.cpp
author sl@SLION-WIN7.fritz.box
Fri, 15 Jun 2012 03:10:57 +0200
changeset 0 bde4ae8d615e
permissions -rw-r--r--
First public contribution.
     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".
     7 //
     8 // Initial Contributors:
     9 // Nokia Corporation - initial contribution.
    10 //
    11 // Contributors:
    12 // NTT DOCOMO, INC - Fix for Bug 1915 "SQL server panics when using long column type strings"
    13 //
    14 // Description:
    15 //
    16 
    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"
    30 #endif
    31 #include "SqlTraceDef.h"
    32 
    33 ///////////////////////////////////////////////////////////////////////////////////////////////////////////////
    34 ///////////////////////////////////////////////////////////////////////////////////////////////////////////////
    35 
    36 #pragma BullseyeCoverage off
    37 
    38 #ifdef _DEBUG
    39 
    40 const TInt KDelayedDbHeapFailureMask = 0x1000;
    41 
    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)
    46 	{
    47 	iDbResourceTestMode = aHeapFailureMode;
    48 	if(aHeapFailureMode & KDelayedDbHeapFailureMask)
    49 		{
    50 		iFailedAllocNumber = aFailedAllocNumber;
    51 		return ETrue;
    52 		}
    53 	return EFalse;
    54 	}
    55 
    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()
    59 	{
    60 	if(iDbResourceTestMode)
    61 		{
    62 		iDbResourceTestMode = 0;
    63 		User::__DbgSetAllocFail(RHeap::EUser, RHeap::ENone, 0);
    64 		}
    65 	}
    66 
    67 //If the database connection is in a test mode then the function will mark the allocated by the 
    68 //server resources.
    69 inline void CSqlSrvSession::DbResourceMark()
    70 	{
    71 	if(iDbResourceTestMode)
    72 		{
    73 		ResourceCountMarkStart();
    74 		}
    75 	}
    76 
    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)
    81 	{
    82 	if(iDbResourceTestMode)
    83 		{
    84 		ResourceCountMarkEnd(aMessage);
    85 		}
    86 	}
    87 
    88 //Executes the heap simulation failure.
    89 inline void CSqlSrvSession::DbSetAllocFail(TInt aHeapFailureMode, TInt aFailedAllocNumber)
    90 	{
    91 	TInt mode = aHeapFailureMode & (KDelayedDbHeapFailureMask - 1);
    92 	if(mode >= RAllocator::EBurstRandom && mode <= RAllocator::EBurstFailNext)
    93 	    {
    94 	    const TUint KBurst = 50; 
    95 	    User::__DbgSetBurstAllocFail(RHeap::EUser, static_cast <RHeap::TAllocFail> (mode), aFailedAllocNumber, KBurst);
    96 	    }
    97 	else
    98 	    {
    99         User::__DbgSetAllocFail(RHeap::EUser, static_cast <RHeap::TAllocFail> (mode), aFailedAllocNumber);
   100 	    }
   101 	}
   102 	
   103 //Executes the delayed heap simulation failure, if the connection is in test mode
   104 inline void CSqlSrvSession::DbSetDelayedAllocFail()
   105 	{
   106 	if(iDbResourceTestMode & KDelayedDbHeapFailureMask)
   107 		{
   108 	    TInt mode = iDbResourceTestMode & (KDelayedDbHeapFailureMask - 1);
   109 	    if(mode >= RAllocator::EBurstRandom && mode <= RAllocator::EBurstFailNext)
   110 	        {
   111 	        const TUint KBurst = 50; 
   112 	        User::__DbgSetBurstAllocFail(RHeap::EUser, static_cast <RHeap::TAllocFail> (mode), iFailedAllocNumber, KBurst);
   113 	        }
   114 	    else
   115 	        {
   116 	        User::__DbgSetAllocFail(RHeap::EUser, static_cast <RHeap::TAllocFail> (mode), iFailedAllocNumber);
   117 	        }
   118 		}
   119 	}
   120 	
   121 #else //_DEBUG
   122 
   123 inline TBool CSqlSrvSession::ActivateDbTestMode(TInt, TInt)
   124 	{
   125 	return EFalse;
   126 	}
   127 
   128 inline void CSqlSrvSession::StopDbTestMode()
   129 	{
   130 	}
   131 
   132 inline void CSqlSrvSession::DbResourceMark()
   133 	{
   134 	}
   135 
   136 inline void CSqlSrvSession::DbResourceEnd(const RMessage2&)
   137 	{
   138 	}
   139 
   140 inline void CSqlSrvSession::DbSetAllocFail(TInt, TInt)
   141 	{
   142 	}
   143 	
   144 inline void CSqlSrvSession::DbSetDelayedAllocFail()
   145 	{
   146 	}
   147 
   148 #endif//_DEBUG
   149 
   150 #pragma BullseyeCoverage on
   151 
   152 ///////////////////////////////////////////////////////////////////////////////////////////////////////////////
   153 ///////////////////////////////////////////////////////////////////////////////////////////////////////////////
   154 
   155 /**
   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.
   159 
   160 @panic SqlDb 4 Client panic. Invalid aHandle parameter value (zero, negative or out of range).
   161 
   162 @internalComponent
   163 */
   164 template <class T> T& SqlSessObjFind(RDbObjContainer<T>& aContainer, TInt aHandle, const RMessage2& aMessage)
   165 	{
   166 	T* obj = aContainer.Find(aHandle);
   167 	__SQLPANIC_CLIENT2(obj != NULL, aMessage, ESqlPanicBadArgument);
   168 	return *obj;
   169 	}
   170 
   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)
   173 	{
   174 	TVolumeInfo volInfo;
   175 	TInt err = aFs.Volume(volInfo, aDrive);
   176 	if(err == KErrNone)
   177 		{
   178 		const TInt64 KDiskSpaceThreshold = 1024 * 4;
   179 		return volInfo.iFree > KDiskSpaceThreshold;
   180 		}
   181 	return ETrue;
   182 	}
   183 	
   184 //If aError is KSqlErrFull and  there is no free disk space, then KSqlErrFull is converted to KErrDiskFull
   185 //and returned.
   186 static TInt ConvertSqlFull2DiskFullErr(TInt aError, RFs& aFs, TDriveNumber aDrive)
   187 	{
   188 	if(aError == KSqlErrFull && !HasFreeDiskSpace(aFs, aDrive))
   189 		{
   190 		aError = KErrDiskFull;
   191 		SQL_TRACE_SESSION(OstTrace1(TRACE_INTERNALS, CONVERTSQLFULL2DISKFULLERRROR, "0;ConvertSqlFull2DiskFullErr;aError=KSqlErrFull;!HasFreeDiskSpace();aDrive=%d", (TInt)aDrive));
   192 		}
   193 	return aError;
   194 	}
   195 
   196 ///////////////////////////////////////////////////////////////////////////////////////////////////////////////
   197 ///////////////////////////////////////////////////////////////////////////////////////////////////////////////
   198 
   199 /**
   200 Creates a new instance of CSqlSrvSession class.
   201 
   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.
   205 
   206 @return A pointer to the created CSqlSrvSession instance.
   207 
   208 @leave KErrNoMemory, an out of memory condition has occurred;
   209 
   210 @see CSqlServer
   211 @see CSqlServer::NewSessionL()
   212 */
   213 CSqlSrvSession* CSqlSrvSession::NewL()
   214 	{
   215 	SQL_TRACE_SESSION(OstTrace0(TRACE_INTERNALS, CSQLSRVSESSION_NEWL_ENTRY, "Entry;0;CSqlSrvSession::NewL"));
   216 	CSqlSrvSession* self = new (ELeave)	CSqlSrvSession;
   217 	CleanupStack::PushL(self);
   218 	self->ConstructL();
   219 	CleanupStack::Pop(self);
   220 	SQL_TRACE_SESSION(OstTrace1(TRACE_INTERNALS, CSQLSRVSESSION_NEWL_EXIT, "Exit;0x%X;CSqlSrvSession::NewL", (TUint)self));
   221 	return self;
   222 	}
   223 
   224 /**
   225 Frees the allocated by CSqlSrvSession instance resources and memory.
   226 */
   227 CSqlSrvSession::~CSqlSrvSession()
   228 	{
   229 	SQL_TRACE_SESSION(OstTraceExt2(TRACE_INTERNALS, CSQLSRVSESSION_CSQLSRVSESSION2_ENTRY, "Entry;0x%X;CSqlSrvSession::~CSqlSrvSession;iDatabase=0x%X", (TUint)this, (TUint)iDatabase));
   230 	StopDbTestMode();
   231 	DbFreeReservedSpace();
   232 	iIpcStreams.Close();
   233 	iStatements.Close();
   234 	delete iDatabase;
   235 	SQL_TRACE_SESSION(OstTrace1(TRACE_INTERNALS, CSQLSRVSESSION_CSQLSRVSESSION2_EXIT, "Exit;0x%X;CSqlSrvSession::~CSqlSrvSession", (TUint)this));
   236 	}
   237 
   238 /**
   239 Receives and dispatches all client side requests.
   240 
   241 CSession2::ServiceL() implementation.
   242 
   243 @param aMessage Client message containing the request (function code and data)
   244 
   245 @leave The function may leave with some database specific 
   246 	   errors categorised as ESqlDbError or system-wide error codes.
   247 
   248 @see CSession2::ServiceL()
   249 */
   250 void CSqlSrvSession::ServiceL(const RMessage2& aMessage)
   251 	{
   252 	TSqlSrvFunction funcCode = ESqlSrvTestBase;
   253 	TInt handle = 0;
   254 	Extract(aMessage, funcCode, handle);
   255 	TInt retCode = KErrNone;
   256 	if(funcCode >= ESqlSrvDbBase)
   257 		{
   258 		SQLPROFILER_REPORT_IPC(ESqlIpcRq, 0);
   259 		}
   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);
   263 	switch(funcCode)
   264 		{
   265 		//////////////////////  resource check operations  ///////////////////////////
   266 		case ESqlSrvResourceMark:
   267 			ResourceCountMarkStart();
   268 			break;
   269 		case ESqlSrvResourceCheck:
   270 			ResourceCountMarkEnd(aMessage);
   271 			break;
   272 		case ESqlSrvResourceCount:
   273 			retCode = CountResources();					//Returns the recourse count
   274 			break;
   275 		case ESqlSrvSetDbHeapFailure:
   276 			if(ActivateDbTestMode(aMessage.Int0(), aMessage.Int1()))
   277 				{
   278 				break;	
   279 				}
   280 		case ESqlSrvSetHeapFailure:
   281 			DbSetAllocFail(aMessage.Int0(), aMessage.Int1());
   282 			break;
   283 		//////////////////////   profiling operations  //////////////////////////////////
   284 		case ESqlSrvProfilerStart:
   285 			TSqlSrvResourceProfiler::StartL(aMessage);
   286 			break;
   287 		case ESqlSrvProfilerStop:
   288 			TSqlSrvResourceProfiler::StopL(aMessage);
   289 			break;
   290 		case ESqlSrvProfilerReset:
   291 			TSqlSrvResourceProfiler::ResetL(aMessage);
   292 			break;
   293 		case ESqlSrvProfilerQuery:
   294 			ProfilerQueryL(aMessage);
   295 			break;
   296 		//////////////////////   database operations //////////////////////////////////
   297 		case ESqlSrvDbCreate:
   298 		case ESqlSrvDbCreateSecure:
   299 		case ESqlSrvDbOpen:
   300 			DbResourceMark();
   301 			DbCreateObjectL(aMessage, funcCode);
   302 			DbSetDelayedAllocFail();
   303 			break;
   304 		case ESqlSrvDbOpenFromHandle:
   305 			DbResourceMark();
   306 			DbCreateObjectFromHandleL(aMessage);
   307 			DbSetDelayedAllocFail();
   308 			break;
   309 		case ESqlSrvDbAttach:
   310 			DbAttachL(aMessage);
   311 			break;
   312 		case ESqlSrvDbAttachFromHandle:
   313 			DbAttachFromHandleL(aMessage);
   314 			break;
   315 		case ESqlSrvDbDetach:
   316 			DbDetachL(aMessage);
   317 			break;
   318 		case ESqlSrvDbClose:
   319 			DbDestroyObject();
   320 			DbResourceEnd(aMessage);
   321 			StopDbTestMode();
   322 			break;
   323 		case ESqlSrvDbCopy:
   324 			DbResourceMark();
   325 			DbCopyFileL(aMessage);
   326 			break;
   327 		case ESqlSrvDbDelete:
   328 			DbResourceMark();
   329 			DbDeleteFileL(aMessage);
   330 			break;
   331 		case ESqlSrvLastErrorMsg:
   332 			retCode = DbLastErrorMessageL(aMessage);//may return that the client buffer is not big enough for the message
   333 			break;
   334 		case ESqlSrvDbLastInsertedRowId:
   335 			DbLastInsertedRowIdL(aMessage);
   336 			break;
   337 		case ESqlSrvDbExec8:
   338 			retCode = DbExecSql8L(aMessage);		//returns the count of affected records
   339 			break;
   340 		case ESqlSrvDbExec16:
   341 			retCode = DbExecSql16L(aMessage);	//returns the count of affected records
   342 			break;
   343 		case ESqlSrvDbSetIsolationLevel:
   344 			DbSetIsolationLevelL(aMessage);
   345 			break;
   346 		case ESqlSrvDbGetSecurityPolicy:
   347 			retCode = DbGetSecurityPolicyL(aMessage);//may return that the client buffer is not big enough for the security policy
   348 			break;
   349 		case ESqlSrvDbScalarFullSelect8:
   350 			retCode = DbScalarFullSelectL(aMessage, EFalse);//may return that the client buffer is not big enough for the column value
   351 			break;
   352 		case ESqlSrvDbScalarFullSelect16:
   353 			retCode = DbScalarFullSelectL(aMessage, ETrue);//may return that the client buffer is not big enough for the column value
   354 			break;
   355 		case ESqlSrvDbInTransaction:
   356 			retCode = DbInTransaction(aMessage);	//Returns whether the database in in transaction or not - true/false
   357 			break;
   358 		case ESqlSrvDbSize:
   359 			retCode = DbSizeL(aMessage);			//Returns the database size
   360 			break;
   361 		case ESqlSrvDbSize2:
   362 			DbSize2L(aMessage);
   363 			break;
   364 		case ESqlSrvDbCompact:
   365 			retCode = DbCompactL(aMessage);			//Returns the amount of the removed free database space
   366 			break;
   367 		//////////////////////   reserved drive space management ////////////////////////
   368 		case ESqlSrvDbReserveDriveSpace:
   369 			__SQLPANIC_CLIENT(iDatabase != NULL, aMessage, ESqlPanicInvalidObj);
   370 			DbReserveDriveSpaceL();
   371 			break;
   372 		case ESqlSrvDbFreeReservedSpace:
   373 			__SQLPANIC_CLIENT(iDatabase != NULL, aMessage, ESqlPanicInvalidObj);
   374 			DbFreeReservedSpace();
   375 			break;
   376 		case ESqlSrvDbGetReserveAccess:
   377 			__SQLPANIC_CLIENT(iDatabase != NULL, aMessage, ESqlPanicInvalidObj);
   378 			DbGetReserveAccessL();
   379 			break;
   380 		case ESqlSrvDbReleaseReserveAccess:
   381 			__SQLPANIC_CLIENT(iDatabase != NULL, aMessage, ESqlPanicInvalidObj);
   382 			DbReleaseReserveAccess();
   383 			break;
   384 		//////////////////////   BLOB source ///////////////////////////////////////////
   385 		case ESqlSrvDbBlobSource:
   386 			retCode = DbBlobSourceL(aMessage);		//Returns the BLOB handle
   387 			break;
   388 		//////////////////////   statement operations //////////////////////////////////
   389 		case ESqlSrvStmtPrepare8:
   390 			retCode = StmtPrepareL(aMessage, EFalse);//returns the statement handle
   391 			break;
   392 		case ESqlSrvStmtPrepare16:
   393 			retCode = StmtPrepareL(aMessage, ETrue);//returns the statement handle
   394 			break;
   395 		case ESqlSrvStmtClose:
   396 			iStatements.Remove(handle);
   397 			break;
   398 		case ESqlSrvStmtReset:
   399 			retCode = ::SqlSessObjFind(iStatements, handle, aMessage).Reset();//May return that the statement has expired
   400 			break;
   401 		case ESqlSrvStmtExec:
   402 		case ESqlSrvStmtAsyncExec:
   403 		case ESqlSrvStmtBindExec:
   404 		case ESqlSrvStmtAsyncBindExec:
   405 			retCode = StmtExecL(aMessage, handle, funcCode);//returns the count of affected records
   406 			break;
   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
   410 			break;
   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
   414 			break;
   415 		case ESqlSrvStmtColumnSource:
   416 			retCode = StmtColumnSourceL(aMessage, handle);//returns an IPC stream handle
   417 			break;
   418 		case ESqlSrvStmtBinParamSink:
   419 		case ESqlSrvStmtTxtParamSink16:
   420 			retCode = StmtParamSinkL(aMessage, handle, funcCode);//returns an IPC stream handle
   421 			break;
   422 		case ESqlSrvStmtBufFlat:
   423 			StmtGetBufFlatL(aMessage, handle);
   424 			break;
   425 		case ESqlSrvStmtColumnValue:
   426 			StmtColumnValueL(aMessage, handle);
   427 			break;
   428 		case ESqlSrvStmtDeclColumnTypes:
   429 			retCode = StmtDeclColumnTypesL(aMessage, handle);
   430 			break;
   431 		//////////////////////   stream operations //////////////////////////////////
   432 		case ESqlSrvStreamRead:
   433 			retCode = ::SqlSessObjFind(iIpcStreams, handle, aMessage).ReadL(aMessage);
   434 			break;
   435 		case ESqlSrvStreamWrite:
   436 			::SqlSessObjFind(iIpcStreams, handle, aMessage).WriteL(aMessage);
   437 			break;
   438 		case ESqlSrvStreamSize:
   439 			retCode = ::SqlSessObjFind(iIpcStreams, handle, aMessage).SizeL();
   440 			break;
   441 		case ESqlSrvStreamSynch:
   442 			::SqlSessObjFind(iIpcStreams, handle, aMessage).SynchL();
   443 			break;
   444 		case ESqlSrvStreamClose:
   445 			iIpcStreams.Remove(handle);
   446 			break;
   447 		//////////////////////                     //////////////////////////////////
   448 		default:	
   449 			retCode = KErrNotSupported;
   450 			break;
   451 		}
   452 	Server().Compactor().RestartTimer();
   453 	Server().MinimizeBuffers();		
   454 	if(!aMessage.IsNull())
   455 		{
   456 		aMessage.Complete(retCode);
   457 		}
   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));
   459 	}
   460 
   461 /**
   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.
   465 */
   466 void CSqlSrvSession::ServiceError(const RMessage2& aMessage, TInt aError)
   467  	{
   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)
   472  		{
   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);
   477  		}
   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));
   482    	}
   483 
   484 ///////////////////////////////////////////////////////////////////////////////////////////////////////
   485 ////////////////////////////          Profiler  operations           ///////////////////////////////////
   486 ///////////////////////////////////////////////////////////////////////////////////////////////////////
   487 
   488 #pragma BullseyeCoverage off
   489 
   490 /**
   491 Retrieves the counter values for the specified profiling counter.
   492 
   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.
   496 
   497 @see TSqlResourceProfiler
   498 
   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.
   503 
   504 @panic SqlDb 2 Client panic. iDatabase is NULL (the database object is not created yet).
   505 */
   506 void CSqlSrvSession::ProfilerQueryL(const RMessage2& aMessage)
   507 	{
   508 	__SQLPANIC_CLIENT(iDatabase != NULL, aMessage, ESqlPanicInvalidObj);
   509 	if(aMessage.Int0() == TSqlResourceProfiler::ESqlCounterConfig)
   510 		{
   511 		const TInt KConfigBufLen = 128;
   512 		if(aMessage.Int1() < KConfigBufLen)
   513 			{
   514 			__SQLLEAVE(KErrOverflow);	
   515 			}
   516 		TBuf8<KConfigBufLen> res;
   517 		iDatabase->QueryConfigL(res);
   518 		aMessage.WriteL(2, res);
   519 		}
   520 	else
   521 		{
   522 		TSqlSrvResourceProfiler::QueryL(aMessage);
   523 		}				
   524 	}
   525 
   526 #pragma BullseyeCoverage on
   527 
   528 ///////////////////////////////////////////////////////////////////////////////////////////////////////
   529 ////////////////////////////          Database operations           ///////////////////////////////////
   530 ///////////////////////////////////////////////////////////////////////////////////////////////////////
   531 
   532 /**
   533 Processes the request for creating/opening a database.
   534 
   535 The function initializes iDatabase and iDrive data members.
   536 
   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
   542 
   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 
   546 */
   547 void CSqlSrvSession::DbCreateObjectL(const RMessage2& aMessage, TSqlSrvFunction aFunction)
   548 	{
   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)
   560 	    {
   561 	    securityAndConfigBuf.CreateL(KSecurityPolicyLen + KConfigStringLen);
   562         aMessage.ReadL(3, securityAndConfigBuf); 
   563         SQLPROFILER_REPORT_IPC(ESqlIpcRead, (KSecurityPolicyLen + KConfigStringLen));
   564 	    }
   565 	TSqlSrvFileData& fileData = Server().FileData();
   566 	TPtrC8 configStr(KNullDesC8);
   567 	if(KConfigStringLen > 0)
   568 	    {
   569 	    configStr.Set(securityAndConfigBuf.Mid(KSecurityPolicyLen));//the first part of the buffer is for the security policies
   570 	    }
   571 	fileData.SetL(aMessage, aMessage.Int0(), 1, &configStr);
   572 	iDrive = fileData.Drive();
   573 	switch(aFunction)
   574 		{
   575 		case ESqlSrvDbCreate:
   576 			if(fileData.IsSecureFileNameFmt())
   577 				{
   578 				__SQLLEAVE(KErrArgument);	
   579 				}
   580 			iDatabase = CSqlSrvDatabase::CreateL(fileData);
   581 			break;
   582 		case ESqlSrvDbCreateSecure:
   583 			{
   584 		    __SQLPANIC_CLIENT(KSecurityPolicyLen > 0, aMessage, ESqlPanicBadArgument);
   585 			if(!fileData.IsSecureFileNameFmt())
   586 				{
   587 				__SQLLEAVE(KErrArgument);	
   588 				}
   589 			//The caller can create a secure database which secure UID matches his secure UID.
   590 			if(fileData.SecureUid() != aMessage.SecureId())
   591 				{
   592 				__SQLLEAVE(KErrPermissionDenied);	
   593 				}
   594 			CSqlSecurityPolicy* policy = CreateSecurityPolicyL(securityAndConfigBuf.Left(KSecurityPolicyLen));
   595 			iDatabase = CSqlSrvDatabase::CreateSecureL(fileData, policy);
   596 			}
   597 			break;
   598 		case ESqlSrvDbOpen:
   599 			iDatabase = CSqlSrvDatabase::OpenL(fileData);
   600 			break;
   601 		default:
   602 			__SQLLEAVE(KErrArgument);	
   603 			break;
   604 		}
   605 	CleanupStack::PopAndDestroy(&securityAndConfigBuf);
   606 	}
   607 
   608 /**
   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.
   611 
   612 Usage of the IPC call arguments:
   613  - Arg 0: [in]  The 32 bits of the argument are used as follow:
   614  @code
   615  MSB                                                                                 LSB
   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 
   618  @endcode
   619  Where:
   620  @code
   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;
   625  @endcode
   626  - Arg 1: [in]  database file name | config string
   627  - Arg 2: [in]  file session handle
   628  - Arg 3: [in]  database file handle
   629 
   630 @panic SqlDb 1 Client panic. iDatabase is not NULL (it has been created already)
   631 */
   632 void CSqlSrvSession::DbCreateObjectFromHandleL(const RMessage2& aMessage)
   633 	{
   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)
   648 		{
   649 		configStr.Copy(buffer.MidTPtr(KDbFileNameLen, KConfigStringLen));
   650 		}
   651 	TSqlSrvFileData& fileData = Server().FileData();
   652 	fileData.SetFromHandleL(aMessage, dbFileName, KCreated, KReadOnly, &configStr);
   653 	iDrive = fileData.Drive();
   654 	iDatabase = CSqlSrvDatabase::OpenL(fileData);
   655 	}
   656 
   657 /**
   658 Processes the request for copying a database.
   659 
   660 Only the database creator can copy the database if the database is a secure database.
   661 
   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
   667 */
   668 void CSqlSrvSession::DbCopyFileL(const RMessage2& aMessage)
   669 	{
   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)									
   680 		{
   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;
   683 		if(dbSecureFlag[i])
   684 			{
   685 			dbSID[i] = fileData.SecureUid();
   686 			}
   687 		dbFileName[i].Copy(fileData.FileName());
   688 		}
   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])
   691 		{
   692 		__SQLLEAVE(KErrPermissionDenied);	
   693 		}
   694 	//If this is a secure database "copy" operation, then...
   695 	if(dbSecureFlag[KSrcDbIdx])
   696 		{
   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])
   700 			{
   701 			__SQLLEAVE(KErrPermissionDenied);	
   702 			}
   703 		}
   704 	//Copy the database
   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);
   712 	}
   713 
   714 /**
   715 Processes the request for deleting a database.
   716 
   717 Only the database creator can delete the database if the database is a secure database.
   718 
   719 Usage of the IPC call arguments:
   720 Arg 0: [in]  database file name length
   721 Arg 1: [in]  database file name
   722 */
   723 void CSqlSrvSession::DbDeleteFileL(const RMessage2& aMessage)
   724 	{	
   725 	TSqlSrvFileData& fileData = Server().FileData();
   726 	fileData.SetL(aMessage, aMessage.Int0(), 1);
   727 	if(fileData.IsSecureFileNameFmt())
   728 		{
   729 		//A secure database can be deleted only by its owner (database SID matches caller SID).
   730 		if(fileData.SecureUid() != aMessage.SecureId())
   731 			{
   732 			__SQLLEAVE(KErrPermissionDenied);	
   733 			}
   734 		}
   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()));
   738 	}
   739 
   740 /**
   741 Processes the request for retrieving the last error message.
   742 
   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.
   746 
   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
   750 
   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).
   753 */
   754 TInt CSqlSrvSession::DbLastErrorMessageL(const RMessage2& aMessage)
   755 	{
   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)
   762 		{
   763 		aMessage.WriteL(1, msg);
   764 		SQLPROFILER_REPORT_IPC(ESqlIpcWrite, (msgLen * sizeof(TText)));
   765 		return 0;
   766 		}
   767 	return msgLen + KSqlClientBufOverflowCode;
   768 	}
   769 	
   770 /**
   771 Processes the request for retrieving the last inserted ROWID of this database connection.
   772 
   773 Usage of the IPC call arguments:
   774 Arg 0: [in/out]	Receiving buffer
   775 
   776 @panic SqlDb 2 Client panic. The database object is not yet created yet (iDatabase is NULL).
   777 */
   778 void CSqlSrvSession::DbLastInsertedRowIdL(const RMessage2& aMessage)
   779 	{
   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));
   784 	}
   785 
   786 /**
   787 Processes the request for retrieving the database security policies.
   788 
   789 The method leaves with KErrNotSupported if the database is not a secure database.
   790 
   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.
   794 
   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
   798 
   799 @panic SqlDb 2 Client panic. iDatabase is NULL (the database object is not created yet).
   800 */
   801 TInt CSqlSrvSession::DbGetSecurityPolicyL(const RMessage2& aMessage)
   802 	{
   803 	__SQLPANIC_CLIENT(iDatabase != NULL, aMessage, ESqlPanicInvalidObj);
   804 	const CSqlSecurityPolicy* securityPolicy = iDatabase->SecurityPolicy();
   805 	if(!securityPolicy)
   806 		{
   807 		__SQLLEAVE(KErrNotSupported);
   808 		}
   809 	const RSqlBufFlat& bufFlat = securityPolicy->BufFlat();
   810 	TInt size = bufFlat.Size();
   811 	if(size <= aMessage.Int0())
   812 		{
   813 		aMessage.WriteL(1, bufFlat.BufDes());
   814 		SQLPROFILER_REPORT_IPC(ESqlIpcWrite, size);
   815 		return 0;
   816 		}
   817 	return size + KSqlClientBufOverflowCode;
   818 	}
   819 
   820 /**
   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.
   826 
   827 @panic SqlDb 2 Client panic. iDatabase is NULL (the database object is not created yet).
   828 
   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
   834 */
   835 TInt CSqlSrvSession::DbScalarFullSelectL(const RMessage2& aMessage, TBool aIsText16)
   836 	{
   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)
   846 		{
   847 		__SQLLEAVE(KErrArgument);	
   848 		}
   849 	TInt err = stmt->Next();
   850 	if(err == KSqlAtRow)
   851 		{
   852 		err = GetColumnValueL(aMessage, *stmt, colType);
   853 		}
   854 	else
   855 		{
   856 		__SQLLEAVE(err == KSqlAtEnd ? KErrNotFound : err);
   857 		}
   858 	CleanupStack::PopAndDestroy(stmt);
   859 	return err;
   860 	}
   861 
   862 /**
   863 @return True, if the database is in transaction, false otherwise.
   864 
   865 @panic SqlDb 2 Client panic. iDatabase is NULL (the database object is not created yet).
   866 */
   867 TBool CSqlSrvSession::DbInTransaction(const RMessage2& aMessage)
   868 	{
   869 	__SQLPANIC_CLIENT(iDatabase != NULL, aMessage, ESqlPanicInvalidObj);
   870 	return iDatabase->InTransaction();
   871 	}
   872 
   873 /**
   874 Main database size in bytes.
   875 
   876 @return Main database size in bytes.
   877 
   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.
   882 
   883 @panic SqlDb 2 Client panic. iDatabase is NULL (the database object is not created yet).
   884 */
   885 TInt CSqlSrvSession::DbSizeL(const RMessage2& aMessage)
   886 	{
   887 	__SQLPANIC_CLIENT(iDatabase != NULL, aMessage, ESqlPanicInvalidObj);
   888 	TInt64 size = iDatabase->SizeL();
   889 	if(size > KMaxTInt)
   890 		{
   891 		__SQLLEAVE(KErrTooBig);
   892 		}
   893 	return size;
   894 	}
   895 
   896 /**
   897 Retrieves the database size and free space.
   898 
   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
   901 			    will be copied.
   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
   904 
   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
   909 
   910 @panic SqlDb 2 Client panic. iDatabase is NULL (the database object is not created yet).
   911 */
   912 void CSqlSrvSession::DbSize2L(const RMessage2& aMessage)
   913 	{
   914 	__SQLPANIC_CLIENT(iDatabase != NULL, aMessage, ESqlPanicInvalidObj);
   915 	const TInt KDbNameLen = aMessage.Int1();
   916 	if((TUint)KDbNameLen > KMaxFileName)
   917 		{
   918 		__SQLLEAVE(KErrBadName);
   919 		}
   920 	TPtrC dbName(KNullDesC);
   921 	if(KDbNameLen > 0)
   922 		{
   923 		dbName.Set(ReadString16L(aMessage, 2, KDbNameLen));
   924 		}
   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));
   930 	}
   931 
   932 /**
   933 Runs database compaction.
   934 
   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
   942 
   943 @return The size of the removed free space
   944 
   945 @panic SqlDb 2 Client panic. iDatabase is NULL (the database object is not created yet).
   946 */
   947 TInt CSqlSrvSession::DbCompactL(const RMessage2& aMessage)
   948 	{
   949 	__SQLPANIC_CLIENT(iDatabase != NULL, aMessage, ESqlPanicInvalidObj);
   950 	const TInt KSize = aMessage.Int0();
   951 	if(KSize < 0)
   952 		{
   953 		if(KSize != RSqlDatabase::EMaxCompaction)
   954 			{
   955 			__SQLLEAVE(KErrArgument);
   956 			}
   957 		}
   958 	if(KSize == 0)
   959 		{
   960 		return 0;	
   961 		}
   962 	const TInt KDbNameLen = aMessage.Int1();
   963 	if((TUint)KDbNameLen > KMaxFileName)
   964 		{
   965 		__SQLLEAVE(KErrBadName);
   966 		}
   967 	TPtrC dbName(KNullDesC);
   968 	if(KDbNameLen > 0)
   969 		{
   970 		dbName.Set(ReadString16L(aMessage, 2, KDbNameLen));
   971 		}
   972 	return iDatabase->CompactL(KSize, dbName);
   973 	}
   974 	
   975 /**
   976 Usage of the IPC call arguments:
   977 Arg 0: [in]		requested size of the space to be reserved - not used
   978 
   979 The function leaves with KErrAlreadyExists if a drive space has been reserved already by this session.
   980 */
   981 void CSqlSrvSession::DbReserveDriveSpaceL()
   982 	{
   983 	if(iDriveSpaceReserved)
   984 		{
   985 		__SQLLEAVE(KErrAlreadyExists);
   986 		}
   987 	RSqlDriveSpaceCol& driveSpaceCol = Server().DriveSpaceCol();
   988     if(!driveSpaceCol.Find(iDrive))
   989     	{
   990     	(void)driveSpaceCol.AddL(iDrive);
   991     	}
   992 	
   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"
   998 	//call.
   999 	}
  1000 	
  1001 /**
  1002 If the client has been given an access to the reserved drive space, that access will be released.
  1003 */
  1004 void CSqlSrvSession::DbFreeReservedSpace()
  1005 	{
  1006 	DbReleaseReserveAccess();
  1007 	iDriveSpaceReserved = EFalse;
  1008 	}
  1009 	
  1010 /**
  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.
  1013 */
  1014 void CSqlSrvSession::DbGetReserveAccessL()
  1015 	{
  1016 	if(iDriveSpaceInUse)
  1017 		{
  1018 		__SQLLEAVE(KErrInUse);	
  1019 		}
  1020 	if(!iDriveSpaceReserved)
  1021 		{
  1022 		__SQLLEAVE(KErrNotFound);
  1023 		}
  1024 	CSqlDriveSpace* driveSpace = Server().DriveSpaceCol().Find(iDrive);
  1025    	if(!driveSpace)
  1026    		{
  1027 		__SQLLEAVE(KErrNotFound);
  1028    		}
  1029    	driveSpace->GetAccessL();
  1030    	iDriveSpaceInUse = ETrue;
  1031 	}
  1032 	
  1033 /**
  1034 Releases the drive space reserve if it has been in use by this session (resp. client).
  1035 */
  1036 void CSqlSrvSession::DbReleaseReserveAccess()
  1037 	{
  1038 	if(iDriveSpaceInUse)
  1039 		{
  1040 		CSqlDriveSpace* driveSpace = Server().DriveSpaceCol().Find(iDrive);
  1041 		if(driveSpace)
  1042 			{
  1043    			driveSpace->ReleaseAccess();
  1044 			}
  1045    		iDriveSpaceInUse = EFalse;
  1046 		}
  1047 	}
  1048 
  1049 /**
  1050 Processes the request for attaching a secure or non-secure database.
  1051 
  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.
  1057 
  1058 @panic SqlDb 2 Client panic. iDatabase is NULL (the database object is not created yet).
  1059 */
  1060 void CSqlSrvSession::DbAttachL(const RMessage2& aMessage)
  1061 	{
  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)
  1067 		{
  1068 		__SQLLEAVE(KErrBadName);
  1069 		}
  1070 	iDatabase->AttachDbL(fileData, ReadString16L(aMessage, 3, logicalDbNameLen));
  1071 	}
  1072 
  1073 /**
  1074 Processes the request for attaching a database using file session and file handles sent by the client.
  1075 
  1076 Usage of the IPC call arguments:
  1077  - Arg 0: [in]  The 32 bits of the argument are used as follow:
  1078  @code
  1079  MSB                                                                                 LSB
  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
  1082  @endcode
  1083  Where:
  1084  @code
  1085   - "Ro" - read-only flag, true if the file is read-only;
  1086   - "F"  - database file name length in 16-bit characters;
  1087  @endcode
  1088 Arg 1: [in]  db names buffer
  1089 Arg 2: [in]  file session handle
  1090 Arg 3: [in]  database file handle
  1091 
  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.
  1094 */
  1095 void CSqlSrvSession::DbAttachFromHandleL(const RMessage2& aMessage)
  1096 	{
  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())
  1107 		{
  1108 		__SQLLEAVE(KErrArgument);
  1109 		}
  1110 	RDesReadStream in(bufPtr);
  1111 	TDes& dbFileName = Server().FileNameBuf();
  1112 	TDes16& dbName = Server().GetBuf16L(KMaxFileName);
  1113 	in >> dbFileName;
  1114 	in >> dbName;
  1115 	CleanupStack::PopAndDestroy(buf);
  1116 	TSqlSrvFileData& fileData = Server().FileData();
  1117 	fileData.SetFromHandleL(aMessage, dbFileName, EFalse, readOnly);
  1118 	iDatabase->AttachDbL(fileData, dbName);
  1119 	}
  1120 	
  1121 /**
  1122 Processes the request for detaching a secure or non-secure database.
  1123 
  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.
  1127 
  1128 @panic SqlDb 2 Client panic. iDatabase is NULL (the database object is not created yet).
  1129 */
  1130 void CSqlSrvSession::DbDetachL(const RMessage2& aMessage)
  1131 	{
  1132 	__SQLPANIC_CLIENT(iDatabase != NULL, aMessage, ESqlPanicInvalidObj);
  1133 	TInt logicalDbNameLen = aMessage.Int0();
  1134 	if(logicalDbNameLen < 1 || logicalDbNameLen > KMaxFileName)
  1135 		{
  1136 		__SQLLEAVE(KErrBadName);
  1137 		}
  1138 	iDatabase->DetachDbL(ReadString16L(aMessage, 1, logicalDbNameLen));
  1139 	}
  1140 
  1141 /**
  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.
  1145 
  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
  1150 
  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;
  1154 */
  1155 void CSqlSrvSession::ExtractNameL(RDesReadStream& aStrm, TDes8& aNameOut, const TDesC& aEmptyNameSubstitute)
  1156 	{
  1157 	 TBool replace = EFalse;
  1158 	 TInt32 len;
  1159 	 aStrm >> len;
  1160 
  1161 	 if(len == 0)
  1162 	 	{
  1163 	   	if(aEmptyNameSubstitute.Length() > 0)
  1164 			{
  1165 		 	len = aEmptyNameSubstitute.Length();
  1166 	 	 	replace = ETrue;
  1167 			}
  1168 	 	else
  1169 			{
  1170 			__SQLLEAVE(KErrBadName);
  1171 			}
  1172 	  	}
  1173 	 __ASSERT_DEBUG(len > 0, __SQLPANIC2(ESqlPanicInternalError));//The "if" above should have hanled the case with "len == 0"
  1174 	 if((TUint)len > KMaxFileName)
  1175 	  {
  1176 	  __SQLLEAVE(KErrBadName);
  1177 	  }
  1178 
  1179 	 HBufC* buf = HBufC::NewLC(len + 1);
  1180 	 TPtr ptr = buf->Des();
  1181 	 if(replace)
  1182 	 	{
  1183 	  	ptr.Copy(aEmptyNameSubstitute);
  1184 	  	}
  1185 	 else
  1186 	 	{
  1187 	  	aStrm >> ptr;
  1188 	 	}
  1189 	 ptr.Append(0);
  1190 	 
  1191 	 if(!::UTF16ZToUTF8Z(ptr, aNameOut))
  1192 	  {
  1193 	  __SQLLEAVE(KErrArgument);
  1194 	  }
  1195 	  
  1196 	 CleanupStack::PopAndDestroy(buf);
  1197 	 }
  1198 
  1199 /**
  1200 Processes the request for creating an IPC stream for accessing the content of a blob column.
  1201 
  1202 @param aMessage The client request wrapped in an IPC message
  1203 
  1204 @return The blob stream handle
  1205 
  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.
  1213 
  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.
  1217 
  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
  1222 */
  1223 TInt CSqlSrvSession::DbBlobSourceL(const RMessage2& aMessage)
  1224 	{	
  1225 	__SQLPANIC_CLIENT(iDatabase != NULL, aMessage, ESqlPanicInvalidObj);
  1226 	
  1227 	TInt ipcPrmLen = aMessage.Int0();
  1228 	__SQLPANIC_CLIENT(ipcPrmLen > 0, aMessage, ESqlPanicBadArgument);
  1229 	
  1230 	iIpcStreams.AllocL();
  1231 	
  1232 	TDes8& ipcPrmDes = ReadString8ZL(aMessage, 1, ipcPrmLen);
  1233 	RDesReadStream strm(ipcPrmDes);
  1234 	
  1235 	TBuf8<KMaxFileName + 1> tblName;
  1236 	ExtractNameL(strm, tblName);
  1237 	
  1238 	TBuf8<KMaxFileName + 1> colName;
  1239 	ExtractNameL(strm, colName);
  1240 	
  1241 	TInt64 rowId;
  1242 	strm >> rowId;
  1243 	if(rowId == -1)
  1244 		{
  1245 		rowId = iDatabase->LastInsertedRowId();
  1246 		}
  1247 	if(rowId <= 0)
  1248 		{
  1249 		__SQLLEAVE(KErrArgument);
  1250 		}
  1251 
  1252 	TInt32 tmp;	
  1253 	strm >> tmp;
  1254 	TBool isReadOnly = tmp != 0;
  1255 	
  1256 	TBuf8<KMaxFileName + 1> dbName;
  1257 	ExtractNameL(strm, dbName, KMainDb16);
  1258 						
  1259 	strm.Close();
  1260 			 		
  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)	
  1264 		{
  1265 		__SQLLEAVE(KErrPermissionDenied);	
  1266 		}
  1267 
  1268 	// Create the stream buffer
  1269 	HBlobBuf* blobBuf = HBlobBuf::NewL(iDatabase->RawDbHandle(), dbName, tblName, colName, rowId, isReadOnly ? HBlobBuf::EReadOnly : HBlobBuf::EReadWrite);
  1270 	blobBuf->PushL();
  1271 		
  1272 	// Return the blob size to the client
  1273 	TPckgBuf<TIpcStreamBuf> ipcBuf;
  1274 	TInt size = blobBuf->SizeL();
  1275 	ipcBuf().iExt = size;
  1276 	
  1277 	// If this is a read stream then return the first client buffer-full of data
  1278 	TInt len = 0;
  1279 	if(isReadOnly && (size > 0))
  1280 		{
  1281 		len = Min(size, KIpcBufSize);
  1282 		blobBuf->ReadL(ipcBuf().iData, len);	
  1283 		}
  1284 	
  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);
  1290 
  1291 	// Send the size and data to the client
  1292 	aMessage.WriteL(2, ipcBuf);
  1293 	SQLPROFILER_REPORT_IPC(ESqlIpcWrite, size);
  1294 
  1295 	return strmHandle;
  1296 	}
  1297 	
  1298 ///////////////////////////////////////////////////////////////////////////////////////////////////////
  1299 ////////////////////////////          Statement operations           //////////////////////////////////
  1300 ///////////////////////////////////////////////////////////////////////////////////////////////////////
  1301 
  1302 /**
  1303 Processes the request for preparing a 8/16-bit SQL statement.
  1304 
  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.
  1307 
  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
  1312 */
  1313 TInt CSqlSrvSession::StmtPrepareL(const RMessage2& aMessage, TBool aIsText16)
  1314 	{
  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);
  1331 	return stmtHandle;
  1332 	}
  1333 
  1334 /**
  1335 Processes the request for executing the SQL statement.
  1336 
  1337 @param aFunction ESqlSrvStmtExec, ESqlSrvStmtAsyncExec, ESqlSrvStmtBindExec, ESqlSrvStmtBindExecRowId, ESqlSrvStmtAsyncBindExec
  1338 
  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)
  1342 */
  1343 TInt CSqlSrvSession::StmtExecL(const RMessage2& aMessage, TInt aStmtHandle, TSqlSrvFunction aFunction)
  1344 	{	
  1345 	CSqlSrvStatement& stmt = ::SqlSessObjFind(iStatements, aStmtHandle, aMessage);
  1346 	if(aFunction == ESqlSrvStmtBindExec || aFunction == ESqlSrvStmtAsyncBindExec)
  1347 		{
  1348 		DoStmtBindL(aMessage, stmt);
  1349 		}
  1350 	__SQLLEAVE_IF_ERROR(stmt.Exec());
  1351 	return iDatabase->LastChangesCount();
  1352 	}
  1353 
  1354 /**
  1355 Processes the request for moving the SQL statement on the next record.
  1356 
  1357 If the call does not fail, the only valid acceptable return codes should be KSqlAtRow and KSqlAtEnd.
  1358 
  1359 @panic SqlDb 7 In _DEBUG mode. The call completed with no error but the return code is not KSqlAtRow or KSqlAtEnd.
  1360 
  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
  1366 */
  1367 TInt CSqlSrvSession::StmtNextL(const RMessage2& aMessage, TInt aStmtHandle, TSqlSrvFunction aFunction)
  1368 	{
  1369 	CSqlSrvStatement& stmt = ::SqlSessObjFind(iStatements, aStmtHandle, aMessage);
  1370 	if(aFunction == ESqlSrvStmtBindNext)
  1371 		{
  1372 		DoStmtBindL(aMessage, stmt);
  1373 		}
  1374 	TInt err = stmt.Next();
  1375 	if(err == KSqlAtRow)
  1376 		{
  1377 		const RSqlBufFlat& bufFlat = stmt.ColumnValuesL();
  1378 		TInt size = bufFlat.Size();
  1379 		if(size > aMessage.Int2())
  1380 			{
  1381 			return size + KSqlClientBufOverflowCode;
  1382 			}
  1383 		aMessage.WriteL(3, bufFlat.BufDes());
  1384 		SQLPROFILER_REPORT_IPC(ESqlIpcWrite, size);
  1385 		}
  1386 	__SQLLEAVE_IF_ERROR(err);
  1387 	__ASSERT_DEBUG(err == KSqlAtRow || err == KSqlAtEnd, __SQLPANIC(ESqlPanicInternalError));
  1388 	return err;
  1389 	}
  1390 
  1391 /**
  1392 Processes the request for retrieving the statement column or parameter names.
  1393 
  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.
  1396 
  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
  1400 */
  1401 TInt CSqlSrvSession::StmtNamesL(const RMessage2& aMessage, TInt aStmtHandle, TSqlSrvFunction aFunction)
  1402 	{
  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())
  1407 		{
  1408 		aMessage.WriteL(1, namesBuf.BufDes());
  1409 		SQLPROFILER_REPORT_IPC(ESqlIpcWrite, size);
  1410 		return 0;
  1411 		}
  1412 	return size + KSqlClientBufOverflowCode;
  1413 	}
  1414 
  1415 /**
  1416 Processes the request for accessing a large column value as a stream of bytes/characters.
  1417 
  1418 Usage of the IPC call arguments:
  1419 Arg 0: [in]		column index (0 based)
  1420 Arg 2: [out]	ipc buffer, column source
  1421 */
  1422 TInt CSqlSrvSession::StmtColumnSourceL(const RMessage2& aMessage, TInt aStmtHandle)
  1423 	{
  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);
  1431 	}
  1432 
  1433 /**
  1434 Processes the request for setting a large parameter value from a stream of bytes or 8/16-bit characters.
  1435 
  1436 @panic SqlDb 3 Client panic. Internal error - invalid stream handle.
  1437 
  1438 Usage of the IPC call arguments:
  1439 Arg 0: [in]		parameter index (0 based)
  1440 Arg 2: [out]	ipc buffer, parameter source
  1441 */
  1442 TInt CSqlSrvSession::StmtParamSinkL(const RMessage2& aMessage, TInt aStmtHandle, TSqlSrvFunction aFunction)
  1443 	{
  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);
  1452 	return strmHandle;
  1453 	}
  1454 
  1455 /**
  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.
  1459 */
  1460 void CSqlSrvSession::StmtGetBufFlatL(const RMessage2& aMessage, TInt aStmtHandle)
  1461 	{
  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());
  1466 	}
  1467 
  1468 /**
  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
  1473 */
  1474 void CSqlSrvSession::StmtColumnValueL(const RMessage2& aMessage, TInt aStmtHandle)
  1475 	{
  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)
  1483 		{
  1484 		TPtr8 ptr(const_cast <TUint8*> (columnSource.Ptr()), columnSource.Length(), columnSource.Length());
  1485 		ptr.SetLength(len);
  1486 		aMessage.WriteL(2, ptr);
  1487 		SQLPROFILER_REPORT_IPC(ESqlIpcWrite, len);
  1488 		__SQLLEAVE(KErrOverflow);	
  1489 		}
  1490 	else
  1491 		{
  1492 		aMessage.WriteL(2, columnSource);
  1493 		SQLPROFILER_REPORT_IPC(ESqlIpcWrite, len);
  1494 		}
  1495 	}
  1496 
  1497 /**
  1498 Usage of the IPC call arguments:
  1499 Arg 0: [in]    parameter buffer length in bytes
  1500 Arg 1: [in]    parameter buffer
  1501 */
  1502 void CSqlSrvSession::DoStmtBindL(const RMessage2& aMessage, CSqlSrvStatement& aStmt)
  1503 	{
  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);
  1509 	}
  1510 
  1511 /**
  1512 Processes the request for retrieving the statement declared column type names.
  1513 
  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
  1516 
  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
  1520 */
  1521 TInt CSqlSrvSession::StmtDeclColumnTypesL(const RMessage2& aMessage, TInt aStmtHandle)
  1522 	{	
  1523 	CSqlSrvStatement& stmt = ::SqlSessObjFind(iStatements, aStmtHandle, aMessage);
  1524 	const RSqlBufFlat& declColumnTypesBuf = stmt.GetDeclColumnTypesL();
  1525 	TInt size = declColumnTypesBuf.Size();
  1526 	if(size <= aMessage.Int0())
  1527 		{
  1528 		aMessage.WriteL(1, declColumnTypesBuf.BufDes());
  1529 		return 0;
  1530 		}
  1531 	return size + KSqlClientBufOverflowCode;
  1532 	}
  1533 
  1534 
  1535 //////////////////////////////////////////////////////////////////////////////////////////////////////////	
  1536 ////////////////////////              Helper methods                     /////////////////////////////////
  1537 //////////////////////////////////////////////////////////////////////////////////////////////////////////	
  1538 
  1539 /**
  1540 Creates a new output IPC stream object using the aStreamBuf parameter as a stream buffer (stream data source).
  1541 
  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.
  1544 
  1545 Returns the handle of the created stream.
  1546 
  1547 @panic SqlDb 3 Client panic. Internal error - invalid stream handle.
  1548 
  1549 Usage of the IPC call arguments:
  1550 Arg 2: [in/out]   IPC buffer
  1551 */
  1552 TInt CSqlSrvSession::NewOutputStreamL(const RMessage2& aMessage, MStreamBuf* aStreamBuf)
  1553 	{
  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);
  1562 	TInt handle = 0;
  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);
  1569 		}
  1570 	else								// no more data to send
  1571 		{
  1572 		CleanupStack::PopAndDestroy(aStreamBuf);
  1573 		}
  1574     ipcBuf().iExt = size;
  1575     aMessage.WriteL(2, ipcBuf);
  1576     SQLPROFILER_REPORT_IPC(ESqlIpcWrite, size);
  1577 	return handle;
  1578 	}
  1579 
  1580 /**
  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.
  1584 
  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.
  1588 */
  1589 TDes8& CSqlSrvSession::ReadString8ZL(const RMessage2& aMessage, TInt aArgNum, TInt aByteLen)
  1590 	{
  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));
  1597 	return buf;
  1598 	}
  1599 
  1600 /**
  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.
  1604 
  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.
  1608 */
  1609 TDes16& CSqlSrvSession::ReadString16ZL(const RMessage2& aMessage, TInt aArgNum, TInt aCharLen)
  1610 	{
  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));
  1617 	return buf;
  1618 	}
  1619 
  1620 /**
  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.
  1623 
  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.
  1627 */
  1628 TDes16& CSqlSrvSession::ReadString16L(const RMessage2& aMessage, TInt aArgNum, TInt aCharLen)
  1629 	{
  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);
  1635 	return buf;
  1636 	}
  1637 
  1638 /**
  1639 The method constructs a CSqlSecurityPolicy object from the passed as an argument descriptor.
  1640 
  1641 @param aSecurityPolicyData A descriptor with the security policy data.
  1642 
  1643 @return A pointer to the created CSqlSecurityPolicy instance.
  1644 
  1645 @leave KErrNoMemory, out of memory condition has occured.
  1646 */
  1647 CSqlSecurityPolicy* CSqlSrvSession::CreateSecurityPolicyL(const TDesC8& aSecurityPolicyData)
  1648 	{
  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);
  1655 	return dbPolicy;
  1656 	}
  1657 
  1658 /**
  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.
  1661 */
  1662 TInt CSqlSrvSession::CountResources()
  1663 	{
  1664 	return iStatements.Count() + iIpcStreams.Count() + (iDbResourceTestMode ? User::CountAllocCells() : 0);
  1665 	}
  1666 
  1667 /**
  1668 Extracts from aMessage:
  1669 - function code;
  1670 - stream or statement handle;
  1671 The function will panic the client if aMessage contains bad function code or bad handle encoded in it.
  1672 */
  1673 void CSqlSrvSession::Extract(const RMessage2& aMessage, TSqlSrvFunction& aFunction, TInt& aHandle)
  1674 	{
  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)
  1679 		{
  1680 		__SQLPANIC_CLIENT(iDatabase != NULL, aMessage, ESqlPanicInvalidObj);
  1681 		}
  1682 	if(aFunction >= ESqlSrvStmtClose)
  1683 		{
  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)
  1689 			{
  1690 			__SQLPANIC_CLIENT(handleType == ESqlSrvStatementHandle, aMessage, ESqlPanicBadArgument);
  1691 			}
  1692 		else
  1693 			{
  1694 			__SQLPANIC_CLIENT(handleType == ESqlSrvStreamHandle, aMessage, ESqlPanicBadArgument);
  1695 			}
  1696 		}
  1697 	}
  1698 
  1699 /**
  1700 The function reads aStmt column 0 value and copies it into the client buffer, accessed via aMessage argument.
  1701 
  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.
  1707 
  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
  1713 */
  1714 TInt CSqlSrvSession::GetColumnValueL(const RMessage2& aMessage, CSqlSrvStatement& aStmt, TSqlColumnType aColType)
  1715 	{
  1716 	TInt rc = KErrNone;
  1717 	switch(aColType)
  1718 		{
  1719 		case ESqlInt:
  1720 			{
  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));
  1724 			}
  1725 			break;
  1726 		case ESqlInt64:
  1727 			{
  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));
  1731 			}
  1732 			break;
  1733 		case ESqlReal:
  1734 			{
  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));
  1738 			}
  1739 			break;
  1740 		case ESqlText:
  1741 		case ESqlBinary:
  1742 		default:
  1743 			{
  1744 			TPtrC8 val;
  1745 			if(aColType == ESqlText)
  1746 				{
  1747 				TPtrC textVal = aStmt.ColumnTextL(0);
  1748 				val.Set(reinterpret_cast <const TUint8*> (textVal.Ptr()), textVal.Length() * sizeof(TUint16));
  1749 				}
  1750 			else
  1751 				{
  1752 				val.Set(aStmt.ColumnBinary(0));
  1753 				}
  1754 			TInt len = val.Length();
  1755 			if(len > aMessage.Int2())
  1756 				{
  1757 				rc = aColType == ESqlText ? (TUint)len / sizeof(TUint16) : len;
  1758 				len = aMessage.Int2();
  1759 				}
  1760 			aMessage.WriteL(3, val.Left(len));
  1761 			SQLPROFILER_REPORT_IPC(ESqlIpcWrite, len);
  1762 			}
  1763 			break;
  1764 		}
  1765 	return rc;
  1766 	}