os/persistentdata/persistentstorage/sql/SRC/Server/SqlSrvSession.cpp
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
sl@0
     1
// Copyright (c) 2005-2010 Nokia Corporation and/or its subsidiary(-ies).
sl@0
     2
// All rights reserved.
sl@0
     3
// This component and the accompanying materials are made available
sl@0
     4
// under the terms of "Eclipse Public License v1.0"
sl@0
     5
// which accompanies this distribution, and is available
sl@0
     6
// at the URL "http://www.eclipse.org/legal/epl-v10.html".
sl@0
     7
//
sl@0
     8
// Initial Contributors:
sl@0
     9
// Nokia Corporation - initial contribution.
sl@0
    10
//
sl@0
    11
// Contributors:
sl@0
    12
// NTT DOCOMO, INC - Fix for Bug 1915 "SQL server panics when using long column type strings"
sl@0
    13
//
sl@0
    14
// Description:
sl@0
    15
//
sl@0
    16
sl@0
    17
#include "SqlSrvMain.h"			//CSqlServer
sl@0
    18
#include "SqlSrvSession.h"		//CSqlSrvSession
sl@0
    19
#include "SqlSrvStatement.h"	//CSqlSrvStatement, HSqlSrvStmtParamBuf
sl@0
    20
#include "SqlSecurityImpl.h"	//CSqlSecurityPolicy
sl@0
    21
#include "SqlSrvUtil.h"			//Global server functions
sl@0
    22
#include "SqlUtil.h"			// config length
sl@0
    23
#include "SqlSrvDriveSpace.h"	//CSqlDriveSpace, RSqlDriveSpaceCol
sl@0
    24
#include "SqlSrvBlob.h"
sl@0
    25
#include "SqlResourceProfiler.h"
sl@0
    26
#include "SqlCompact.h"
sl@0
    27
#include "OstTraceDefinitions.h"
sl@0
    28
#ifdef OST_TRACE_COMPILER_IN_USE
sl@0
    29
#include "SqlSrvSessionTraces.h"
sl@0
    30
#endif
sl@0
    31
#include "SqlTraceDef.h"
sl@0
    32
sl@0
    33
///////////////////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
    34
///////////////////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
    35
sl@0
    36
#pragma BullseyeCoverage off
sl@0
    37
sl@0
    38
#ifdef _DEBUG
sl@0
    39
sl@0
    40
const TInt KDelayedDbHeapFailureMask = 0x1000;
sl@0
    41
sl@0
    42
//Puts the database connection in a test mode
sl@0
    43
//Returns true if the heap failure simulation has to be delayed untill the database is opened
sl@0
    44
//Initialises iDbResourceTestMode and iFailedAllocNumber (if delayed simulation) data members
sl@0
    45
inline TBool CSqlSrvSession::ActivateDbTestMode(TInt aHeapFailureMode, TInt aFailedAllocNumber)
sl@0
    46
	{
sl@0
    47
	iDbResourceTestMode = aHeapFailureMode;
sl@0
    48
	if(aHeapFailureMode & KDelayedDbHeapFailureMask)
sl@0
    49
		{
sl@0
    50
		iFailedAllocNumber = aFailedAllocNumber;
sl@0
    51
		return ETrue;
sl@0
    52
		}
sl@0
    53
	return EFalse;
sl@0
    54
	}
sl@0
    55
sl@0
    56
//If the database connection is in a test mode then the macro will reset the heap allocation failure type.
sl@0
    57
//and stop the test mode.
sl@0
    58
inline void CSqlSrvSession::StopDbTestMode()
sl@0
    59
	{
sl@0
    60
	if(iDbResourceTestMode)
sl@0
    61
		{
sl@0
    62
		iDbResourceTestMode = 0;
sl@0
    63
		User::__DbgSetAllocFail(RHeap::EUser, RHeap::ENone, 0);
sl@0
    64
		}
sl@0
    65
	}
sl@0
    66
sl@0
    67
//If the database connection is in a test mode then the function will mark the allocated by the 
sl@0
    68
//server resources.
sl@0
    69
inline void CSqlSrvSession::DbResourceMark()
sl@0
    70
	{
sl@0
    71
	if(iDbResourceTestMode)
sl@0
    72
		{
sl@0
    73
		ResourceCountMarkStart();
sl@0
    74
		}
sl@0
    75
	}
sl@0
    76
sl@0
    77
//If the database connection is in a test mode then the macro will check the allocated by the server resources,
sl@0
    78
//comparing their count with the resource count at the moment DbResourceMark() has been called.
sl@0
    79
//The client will be panicked if the resource count now is different.
sl@0
    80
inline void CSqlSrvSession::DbResourceEnd(const RMessage2& aMessage)
sl@0
    81
	{
sl@0
    82
	if(iDbResourceTestMode)
sl@0
    83
		{
sl@0
    84
		ResourceCountMarkEnd(aMessage);
sl@0
    85
		}
sl@0
    86
	}
sl@0
    87
sl@0
    88
//Executes the heap simulation failure.
sl@0
    89
inline void CSqlSrvSession::DbSetAllocFail(TInt aHeapFailureMode, TInt aFailedAllocNumber)
sl@0
    90
	{
sl@0
    91
	TInt mode = aHeapFailureMode & (KDelayedDbHeapFailureMask - 1);
sl@0
    92
	if(mode >= RAllocator::EBurstRandom && mode <= RAllocator::EBurstFailNext)
sl@0
    93
	    {
sl@0
    94
	    const TUint KBurst = 50; 
sl@0
    95
	    User::__DbgSetBurstAllocFail(RHeap::EUser, static_cast <RHeap::TAllocFail> (mode), aFailedAllocNumber, KBurst);
sl@0
    96
	    }
sl@0
    97
	else
sl@0
    98
	    {
sl@0
    99
        User::__DbgSetAllocFail(RHeap::EUser, static_cast <RHeap::TAllocFail> (mode), aFailedAllocNumber);
sl@0
   100
	    }
sl@0
   101
	}
sl@0
   102
	
sl@0
   103
//Executes the delayed heap simulation failure, if the connection is in test mode
sl@0
   104
inline void CSqlSrvSession::DbSetDelayedAllocFail()
sl@0
   105
	{
sl@0
   106
	if(iDbResourceTestMode & KDelayedDbHeapFailureMask)
sl@0
   107
		{
sl@0
   108
	    TInt mode = iDbResourceTestMode & (KDelayedDbHeapFailureMask - 1);
sl@0
   109
	    if(mode >= RAllocator::EBurstRandom && mode <= RAllocator::EBurstFailNext)
sl@0
   110
	        {
sl@0
   111
	        const TUint KBurst = 50; 
sl@0
   112
	        User::__DbgSetBurstAllocFail(RHeap::EUser, static_cast <RHeap::TAllocFail> (mode), iFailedAllocNumber, KBurst);
sl@0
   113
	        }
sl@0
   114
	    else
sl@0
   115
	        {
sl@0
   116
	        User::__DbgSetAllocFail(RHeap::EUser, static_cast <RHeap::TAllocFail> (mode), iFailedAllocNumber);
sl@0
   117
	        }
sl@0
   118
		}
sl@0
   119
	}
sl@0
   120
	
sl@0
   121
#else //_DEBUG
sl@0
   122
sl@0
   123
inline TBool CSqlSrvSession::ActivateDbTestMode(TInt, TInt)
sl@0
   124
	{
sl@0
   125
	return EFalse;
sl@0
   126
	}
sl@0
   127
sl@0
   128
inline void CSqlSrvSession::StopDbTestMode()
sl@0
   129
	{
sl@0
   130
	}
sl@0
   131
sl@0
   132
inline void CSqlSrvSession::DbResourceMark()
sl@0
   133
	{
sl@0
   134
	}
sl@0
   135
sl@0
   136
inline void CSqlSrvSession::DbResourceEnd(const RMessage2&)
sl@0
   137
	{
sl@0
   138
	}
sl@0
   139
sl@0
   140
inline void CSqlSrvSession::DbSetAllocFail(TInt, TInt)
sl@0
   141
	{
sl@0
   142
	}
sl@0
   143
	
sl@0
   144
inline void CSqlSrvSession::DbSetDelayedAllocFail()
sl@0
   145
	{
sl@0
   146
	}
sl@0
   147
sl@0
   148
#endif//_DEBUG
sl@0
   149
sl@0
   150
#pragma BullseyeCoverage on
sl@0
   151
sl@0
   152
///////////////////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
   153
///////////////////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
   154
sl@0
   155
/**
sl@0
   156
Searches aContainer for an object identified by aHandle.
sl@0
   157
If such object exists, a reference to it is returned.
sl@0
   158
If there is no object, the client gets a panic.
sl@0
   159
sl@0
   160
@panic SqlDb 4 Client panic. Invalid aHandle parameter value (zero, negative or out of range).
sl@0
   161
sl@0
   162
@internalComponent
sl@0
   163
*/
sl@0
   164
template <class T> T& SqlSessObjFind(RDbObjContainer<T>& aContainer, TInt aHandle, const RMessage2& aMessage)
sl@0
   165
	{
sl@0
   166
	T* obj = aContainer.Find(aHandle);
sl@0
   167
	__SQLPANIC_CLIENT2(obj != NULL, aMessage, ESqlPanicBadArgument);
sl@0
   168
	return *obj;
sl@0
   169
	}
sl@0
   170
sl@0
   171
//This function return true, if there is free disk space on drive where the main database file is.
sl@0
   172
static TBool HasFreeDiskSpace(RFs& aFs, TDriveNumber aDrive)
sl@0
   173
	{
sl@0
   174
	TVolumeInfo volInfo;
sl@0
   175
	TInt err = aFs.Volume(volInfo, aDrive);
sl@0
   176
	if(err == KErrNone)
sl@0
   177
		{
sl@0
   178
		const TInt64 KDiskSpaceThreshold = 1024 * 4;
sl@0
   179
		return volInfo.iFree > KDiskSpaceThreshold;
sl@0
   180
		}
sl@0
   181
	return ETrue;
sl@0
   182
	}
sl@0
   183
	
sl@0
   184
//If aError is KSqlErrFull and  there is no free disk space, then KSqlErrFull is converted to KErrDiskFull
sl@0
   185
//and returned.
sl@0
   186
static TInt ConvertSqlFull2DiskFullErr(TInt aError, RFs& aFs, TDriveNumber aDrive)
sl@0
   187
	{
sl@0
   188
	if(aError == KSqlErrFull && !HasFreeDiskSpace(aFs, aDrive))
sl@0
   189
		{
sl@0
   190
		aError = KErrDiskFull;
sl@0
   191
		SQL_TRACE_SESSION(OstTrace1(TRACE_INTERNALS, CONVERTSQLFULL2DISKFULLERRROR, "0;ConvertSqlFull2DiskFullErr;aError=KSqlErrFull;!HasFreeDiskSpace();aDrive=%d", (TInt)aDrive));
sl@0
   192
		}
sl@0
   193
	return aError;
sl@0
   194
	}
sl@0
   195
sl@0
   196
///////////////////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
   197
///////////////////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
   198
sl@0
   199
/**
sl@0
   200
Creates a new instance of CSqlSrvSession class.
sl@0
   201
sl@0
   202
This function shall never be called directly.
sl@0
   203
It is CSqlServer responsibility to create a new server side session object as a responce to the criation of a
sl@0
   204
client side session instance.
sl@0
   205
sl@0
   206
@return A pointer to the created CSqlSrvSession instance.
sl@0
   207
sl@0
   208
@leave KErrNoMemory, an out of memory condition has occurred;
sl@0
   209
sl@0
   210
@see CSqlServer
sl@0
   211
@see CSqlServer::NewSessionL()
sl@0
   212
*/
sl@0
   213
CSqlSrvSession* CSqlSrvSession::NewL()
sl@0
   214
	{
sl@0
   215
	SQL_TRACE_SESSION(OstTrace0(TRACE_INTERNALS, CSQLSRVSESSION_NEWL_ENTRY, "Entry;0;CSqlSrvSession::NewL"));
sl@0
   216
	CSqlSrvSession* self = new (ELeave)	CSqlSrvSession;
sl@0
   217
	CleanupStack::PushL(self);
sl@0
   218
	self->ConstructL();
sl@0
   219
	CleanupStack::Pop(self);
sl@0
   220
	SQL_TRACE_SESSION(OstTrace1(TRACE_INTERNALS, CSQLSRVSESSION_NEWL_EXIT, "Exit;0x%X;CSqlSrvSession::NewL", (TUint)self));
sl@0
   221
	return self;
sl@0
   222
	}
sl@0
   223
sl@0
   224
/**
sl@0
   225
Frees the allocated by CSqlSrvSession instance resources and memory.
sl@0
   226
*/
sl@0
   227
CSqlSrvSession::~CSqlSrvSession()
sl@0
   228
	{
sl@0
   229
	SQL_TRACE_SESSION(OstTraceExt2(TRACE_INTERNALS, CSQLSRVSESSION_CSQLSRVSESSION2_ENTRY, "Entry;0x%X;CSqlSrvSession::~CSqlSrvSession;iDatabase=0x%X", (TUint)this, (TUint)iDatabase));
sl@0
   230
	StopDbTestMode();
sl@0
   231
	DbFreeReservedSpace();
sl@0
   232
	iIpcStreams.Close();
sl@0
   233
	iStatements.Close();
sl@0
   234
	delete iDatabase;
sl@0
   235
	SQL_TRACE_SESSION(OstTrace1(TRACE_INTERNALS, CSQLSRVSESSION_CSQLSRVSESSION2_EXIT, "Exit;0x%X;CSqlSrvSession::~CSqlSrvSession", (TUint)this));
sl@0
   236
	}
sl@0
   237
sl@0
   238
/**
sl@0
   239
Receives and dispatches all client side requests.
sl@0
   240
sl@0
   241
CSession2::ServiceL() implementation.
sl@0
   242
sl@0
   243
@param aMessage Client message containing the request (function code and data)
sl@0
   244
sl@0
   245
@leave The function may leave with some database specific 
sl@0
   246
	   errors categorised as ESqlDbError or system-wide error codes.
sl@0
   247
sl@0
   248
@see CSession2::ServiceL()
sl@0
   249
*/
sl@0
   250
void CSqlSrvSession::ServiceL(const RMessage2& aMessage)
sl@0
   251
	{
sl@0
   252
	TSqlSrvFunction funcCode = ESqlSrvTestBase;
sl@0
   253
	TInt handle = 0;
sl@0
   254
	Extract(aMessage, funcCode, handle);
sl@0
   255
	TInt retCode = KErrNone;
sl@0
   256
	if(funcCode >= ESqlSrvDbBase)
sl@0
   257
		{
sl@0
   258
		SQLPROFILER_REPORT_IPC(ESqlIpcRq, 0);
sl@0
   259
		}
sl@0
   260
	__SQLTRACE_SESSIONEXPR(++iIpcCallCounter);
sl@0
   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));
sl@0
   262
	SQLPROFILER_IPC_START(iIpcCallCounter, iDatabase ? (TUint)iDatabase->RawDbHandle() : 0);
sl@0
   263
	switch(funcCode)
sl@0
   264
		{
sl@0
   265
		//////////////////////  resource check operations  ///////////////////////////
sl@0
   266
		case ESqlSrvResourceMark:
sl@0
   267
			ResourceCountMarkStart();
sl@0
   268
			break;
sl@0
   269
		case ESqlSrvResourceCheck:
sl@0
   270
			ResourceCountMarkEnd(aMessage);
sl@0
   271
			break;
sl@0
   272
		case ESqlSrvResourceCount:
sl@0
   273
			retCode = CountResources();					//Returns the recourse count
sl@0
   274
			break;
sl@0
   275
		case ESqlSrvSetDbHeapFailure:
sl@0
   276
			if(ActivateDbTestMode(aMessage.Int0(), aMessage.Int1()))
sl@0
   277
				{
sl@0
   278
				break;	
sl@0
   279
				}
sl@0
   280
		case ESqlSrvSetHeapFailure:
sl@0
   281
			DbSetAllocFail(aMessage.Int0(), aMessage.Int1());
sl@0
   282
			break;
sl@0
   283
		//////////////////////   profiling operations  //////////////////////////////////
sl@0
   284
		case ESqlSrvProfilerStart:
sl@0
   285
			TSqlSrvResourceProfiler::StartL(aMessage);
sl@0
   286
			break;
sl@0
   287
		case ESqlSrvProfilerStop:
sl@0
   288
			TSqlSrvResourceProfiler::StopL(aMessage);
sl@0
   289
			break;
sl@0
   290
		case ESqlSrvProfilerReset:
sl@0
   291
			TSqlSrvResourceProfiler::ResetL(aMessage);
sl@0
   292
			break;
sl@0
   293
		case ESqlSrvProfilerQuery:
sl@0
   294
			ProfilerQueryL(aMessage);
sl@0
   295
			break;
sl@0
   296
		//////////////////////   database operations //////////////////////////////////
sl@0
   297
		case ESqlSrvDbCreate:
sl@0
   298
		case ESqlSrvDbCreateSecure:
sl@0
   299
		case ESqlSrvDbOpen:
sl@0
   300
			DbResourceMark();
sl@0
   301
			DbCreateObjectL(aMessage, funcCode);
sl@0
   302
			DbSetDelayedAllocFail();
sl@0
   303
			break;
sl@0
   304
		case ESqlSrvDbOpenFromHandle:
sl@0
   305
			DbResourceMark();
sl@0
   306
			DbCreateObjectFromHandleL(aMessage);
sl@0
   307
			DbSetDelayedAllocFail();
sl@0
   308
			break;
sl@0
   309
		case ESqlSrvDbAttach:
sl@0
   310
			DbAttachL(aMessage);
sl@0
   311
			break;
sl@0
   312
		case ESqlSrvDbAttachFromHandle:
sl@0
   313
			DbAttachFromHandleL(aMessage);
sl@0
   314
			break;
sl@0
   315
		case ESqlSrvDbDetach:
sl@0
   316
			DbDetachL(aMessage);
sl@0
   317
			break;
sl@0
   318
		case ESqlSrvDbClose:
sl@0
   319
			DbDestroyObject();
sl@0
   320
			DbResourceEnd(aMessage);
sl@0
   321
			StopDbTestMode();
sl@0
   322
			break;
sl@0
   323
		case ESqlSrvDbCopy:
sl@0
   324
			DbResourceMark();
sl@0
   325
			DbCopyFileL(aMessage);
sl@0
   326
			break;
sl@0
   327
		case ESqlSrvDbDelete:
sl@0
   328
			DbResourceMark();
sl@0
   329
			DbDeleteFileL(aMessage);
sl@0
   330
			break;
sl@0
   331
		case ESqlSrvLastErrorMsg:
sl@0
   332
			retCode = DbLastErrorMessageL(aMessage);//may return that the client buffer is not big enough for the message
sl@0
   333
			break;
sl@0
   334
		case ESqlSrvDbLastInsertedRowId:
sl@0
   335
			DbLastInsertedRowIdL(aMessage);
sl@0
   336
			break;
sl@0
   337
		case ESqlSrvDbExec8:
sl@0
   338
			retCode = DbExecSql8L(aMessage);		//returns the count of affected records
sl@0
   339
			break;
sl@0
   340
		case ESqlSrvDbExec16:
sl@0
   341
			retCode = DbExecSql16L(aMessage);	//returns the count of affected records
sl@0
   342
			break;
sl@0
   343
		case ESqlSrvDbSetIsolationLevel:
sl@0
   344
			DbSetIsolationLevelL(aMessage);
sl@0
   345
			break;
sl@0
   346
		case ESqlSrvDbGetSecurityPolicy:
sl@0
   347
			retCode = DbGetSecurityPolicyL(aMessage);//may return that the client buffer is not big enough for the security policy
sl@0
   348
			break;
sl@0
   349
		case ESqlSrvDbScalarFullSelect8:
sl@0
   350
			retCode = DbScalarFullSelectL(aMessage, EFalse);//may return that the client buffer is not big enough for the column value
sl@0
   351
			break;
sl@0
   352
		case ESqlSrvDbScalarFullSelect16:
sl@0
   353
			retCode = DbScalarFullSelectL(aMessage, ETrue);//may return that the client buffer is not big enough for the column value
sl@0
   354
			break;
sl@0
   355
		case ESqlSrvDbInTransaction:
sl@0
   356
			retCode = DbInTransaction(aMessage);	//Returns whether the database in in transaction or not - true/false
sl@0
   357
			break;
sl@0
   358
		case ESqlSrvDbSize:
sl@0
   359
			retCode = DbSizeL(aMessage);			//Returns the database size
sl@0
   360
			break;
sl@0
   361
		case ESqlSrvDbSize2:
sl@0
   362
			DbSize2L(aMessage);
sl@0
   363
			break;
sl@0
   364
		case ESqlSrvDbCompact:
sl@0
   365
			retCode = DbCompactL(aMessage);			//Returns the amount of the removed free database space
sl@0
   366
			break;
sl@0
   367
		//////////////////////   reserved drive space management ////////////////////////
sl@0
   368
		case ESqlSrvDbReserveDriveSpace:
sl@0
   369
			__SQLPANIC_CLIENT(iDatabase != NULL, aMessage, ESqlPanicInvalidObj);
sl@0
   370
			DbReserveDriveSpaceL();
sl@0
   371
			break;
sl@0
   372
		case ESqlSrvDbFreeReservedSpace:
sl@0
   373
			__SQLPANIC_CLIENT(iDatabase != NULL, aMessage, ESqlPanicInvalidObj);
sl@0
   374
			DbFreeReservedSpace();
sl@0
   375
			break;
sl@0
   376
		case ESqlSrvDbGetReserveAccess:
sl@0
   377
			__SQLPANIC_CLIENT(iDatabase != NULL, aMessage, ESqlPanicInvalidObj);
sl@0
   378
			DbGetReserveAccessL();
sl@0
   379
			break;
sl@0
   380
		case ESqlSrvDbReleaseReserveAccess:
sl@0
   381
			__SQLPANIC_CLIENT(iDatabase != NULL, aMessage, ESqlPanicInvalidObj);
sl@0
   382
			DbReleaseReserveAccess();
sl@0
   383
			break;
sl@0
   384
		//////////////////////   BLOB source ///////////////////////////////////////////
sl@0
   385
		case ESqlSrvDbBlobSource:
sl@0
   386
			retCode = DbBlobSourceL(aMessage);		//Returns the BLOB handle
sl@0
   387
			break;
sl@0
   388
		//////////////////////   statement operations //////////////////////////////////
sl@0
   389
		case ESqlSrvStmtPrepare8:
sl@0
   390
			retCode = StmtPrepareL(aMessage, EFalse);//returns the statement handle
sl@0
   391
			break;
sl@0
   392
		case ESqlSrvStmtPrepare16:
sl@0
   393
			retCode = StmtPrepareL(aMessage, ETrue);//returns the statement handle
sl@0
   394
			break;
sl@0
   395
		case ESqlSrvStmtClose:
sl@0
   396
			iStatements.Remove(handle);
sl@0
   397
			break;
sl@0
   398
		case ESqlSrvStmtReset:
sl@0
   399
			retCode = ::SqlSessObjFind(iStatements, handle, aMessage).Reset();//May return that the statement has expired
sl@0
   400
			break;
sl@0
   401
		case ESqlSrvStmtExec:
sl@0
   402
		case ESqlSrvStmtAsyncExec:
sl@0
   403
		case ESqlSrvStmtBindExec:
sl@0
   404
		case ESqlSrvStmtAsyncBindExec:
sl@0
   405
			retCode = StmtExecL(aMessage, handle, funcCode);//returns the count of affected records
sl@0
   406
			break;
sl@0
   407
		case ESqlSrvStmtNext:
sl@0
   408
		case ESqlSrvStmtBindNext:
sl@0
   409
			retCode = StmtNextL(aMessage, handle, funcCode);//returns a non-negative number if the client side buffer is too small
sl@0
   410
			break;
sl@0
   411
		case ESqlSrvStmtColumnNames:
sl@0
   412
		case ESqlSrvStmtParamNames:
sl@0
   413
			retCode = StmtNamesL(aMessage, handle, funcCode);//returns a non-negative number if the client side buffer is too small
sl@0
   414
			break;
sl@0
   415
		case ESqlSrvStmtColumnSource:
sl@0
   416
			retCode = StmtColumnSourceL(aMessage, handle);//returns an IPC stream handle
sl@0
   417
			break;
sl@0
   418
		case ESqlSrvStmtBinParamSink:
sl@0
   419
		case ESqlSrvStmtTxtParamSink16:
sl@0
   420
			retCode = StmtParamSinkL(aMessage, handle, funcCode);//returns an IPC stream handle
sl@0
   421
			break;
sl@0
   422
		case ESqlSrvStmtBufFlat:
sl@0
   423
			StmtGetBufFlatL(aMessage, handle);
sl@0
   424
			break;
sl@0
   425
		case ESqlSrvStmtColumnValue:
sl@0
   426
			StmtColumnValueL(aMessage, handle);
sl@0
   427
			break;
sl@0
   428
		case ESqlSrvStmtDeclColumnTypes:
sl@0
   429
			retCode = StmtDeclColumnTypesL(aMessage, handle);
sl@0
   430
			break;
sl@0
   431
		//////////////////////   stream operations //////////////////////////////////
sl@0
   432
		case ESqlSrvStreamRead:
sl@0
   433
			retCode = ::SqlSessObjFind(iIpcStreams, handle, aMessage).ReadL(aMessage);
sl@0
   434
			break;
sl@0
   435
		case ESqlSrvStreamWrite:
sl@0
   436
			::SqlSessObjFind(iIpcStreams, handle, aMessage).WriteL(aMessage);
sl@0
   437
			break;
sl@0
   438
		case ESqlSrvStreamSize:
sl@0
   439
			retCode = ::SqlSessObjFind(iIpcStreams, handle, aMessage).SizeL();
sl@0
   440
			break;
sl@0
   441
		case ESqlSrvStreamSynch:
sl@0
   442
			::SqlSessObjFind(iIpcStreams, handle, aMessage).SynchL();
sl@0
   443
			break;
sl@0
   444
		case ESqlSrvStreamClose:
sl@0
   445
			iIpcStreams.Remove(handle);
sl@0
   446
			break;
sl@0
   447
		//////////////////////                     //////////////////////////////////
sl@0
   448
		default:	
sl@0
   449
			retCode = KErrNotSupported;
sl@0
   450
			break;
sl@0
   451
		}
sl@0
   452
	Server().Compactor().RestartTimer();
sl@0
   453
	Server().MinimizeBuffers();		
sl@0
   454
	if(!aMessage.IsNull())
sl@0
   455
		{
sl@0
   456
		aMessage.Complete(retCode);
sl@0
   457
		}
sl@0
   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));
sl@0
   459
	}
sl@0
   460
sl@0
   461
/**
sl@0
   462
If aError is KErrBadDescriptor, then panic the client, else - default error handling.
sl@0
   463
KErrBadDescriptor error may be thrown from "message write" operations, if the client supplied a bad
sl@0
   464
descriptor to the server.
sl@0
   465
*/
sl@0
   466
void CSqlSrvSession::ServiceError(const RMessage2& aMessage, TInt aError)
sl@0
   467
 	{
sl@0
   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));
sl@0
   469
	Server().MinimizeBuffers();		
sl@0
   470
	aError = ::ConvertSqlFull2DiskFullErr(aError, Server().FileData().Fs(), iDrive);
sl@0
   471
 	if(aError == KErrBadDescriptor)
sl@0
   472
 		{
sl@0
   473
		//The __SQLPANIC_CLIENT() macro cannot be used here because it calls a leaving function. A leaving call
sl@0
   474
		//from a leaving call will terminate the server.
sl@0
   475
		_LIT(KPanicCategory, "SqlDb");
sl@0
   476
		aMessage.Panic(KPanicCategory, ESqlPanicBadDescriptor);
sl@0
   477
 		}
sl@0
   478
 	SQLPROFILER_IPC_ERROR(iIpcCallCounter, static_cast <TSqlSrvFunction> (KSqlSrvFunctionMask & aMessage.Function()), 
sl@0
   479
 	       iDatabase ? (TUint)iDatabase->RawDbHandle() : 0, aError);
sl@0
   480
 	CSession2::ServiceError(aMessage, aError);
sl@0
   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));
sl@0
   482
   	}
sl@0
   483
sl@0
   484
///////////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
   485
////////////////////////////          Profiler  operations           ///////////////////////////////////
sl@0
   486
///////////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
   487
sl@0
   488
#pragma BullseyeCoverage off
sl@0
   489
sl@0
   490
/**
sl@0
   491
Retrieves the counter values for the specified profiling counter.
sl@0
   492
sl@0
   493
@leave  KErrNone, the operation completed successfully,
sl@0
   494
        KErrOverflow, the receiving buffer size is too small;
sl@0
   495
                  One of the other system-wide error codes may also be returned.
sl@0
   496
sl@0
   497
@see TSqlResourceProfiler
sl@0
   498
sl@0
   499
Usage of the IPC call arguments:
sl@0
   500
 - Arg 0: [in]  profiling counter type, one of the TSqlResourceProfiler::TSqlCounter enum item values.
sl@0
   501
 - Arg 1: [in]  the size of the buffer for the profiling counter values.
sl@0
   502
 - Arg 2: [out] the buffer for the profiling counter values.
sl@0
   503
sl@0
   504
@panic SqlDb 2 Client panic. iDatabase is NULL (the database object is not created yet).
sl@0
   505
*/
sl@0
   506
void CSqlSrvSession::ProfilerQueryL(const RMessage2& aMessage)
sl@0
   507
	{
sl@0
   508
	__SQLPANIC_CLIENT(iDatabase != NULL, aMessage, ESqlPanicInvalidObj);
sl@0
   509
	if(aMessage.Int0() == TSqlResourceProfiler::ESqlCounterConfig)
sl@0
   510
		{
sl@0
   511
		const TInt KConfigBufLen = 128;
sl@0
   512
		if(aMessage.Int1() < KConfigBufLen)
sl@0
   513
			{
sl@0
   514
			__SQLLEAVE(KErrOverflow);	
sl@0
   515
			}
sl@0
   516
		TBuf8<KConfigBufLen> res;
sl@0
   517
		iDatabase->QueryConfigL(res);
sl@0
   518
		aMessage.WriteL(2, res);
sl@0
   519
		}
sl@0
   520
	else
sl@0
   521
		{
sl@0
   522
		TSqlSrvResourceProfiler::QueryL(aMessage);
sl@0
   523
		}				
sl@0
   524
	}
sl@0
   525
sl@0
   526
#pragma BullseyeCoverage on
sl@0
   527
sl@0
   528
///////////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
   529
////////////////////////////          Database operations           ///////////////////////////////////
sl@0
   530
///////////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
   531
sl@0
   532
/**
sl@0
   533
Processes the request for creating/opening a database.
sl@0
   534
sl@0
   535
The function initializes iDatabase and iDrive data members.
sl@0
   536
sl@0
   537
Usage of the IPC call arguments:
sl@0
   538
Arg 0: [in]  database file name length in 16-bit characters
sl@0
   539
Arg 1: [in]  database file name
sl@0
   540
Arg 2: [in]  PPPPCCCC, where PPPP is the security policy length, CCCC is the config string length.
sl@0
   541
Arg 3: [in]  security policies buffer | config string
sl@0
   542
sl@0
   543
@panic SqlDb 1 Client panic. iDatabase is not NULL (it has been created already)
sl@0
   544
@panic SqlDb 4 Client panic. Negative or too big config string length
sl@0
   545
@panic SqlDb 4 Client panic. Negative security policy length, or zero length if the request is to create a secure database 
sl@0
   546
*/
sl@0
   547
void CSqlSrvSession::DbCreateObjectL(const RMessage2& aMessage, TSqlSrvFunction aFunction)
sl@0
   548
	{
sl@0
   549
	__SQLPANIC_CLIENT(!iDatabase, aMessage, ESqlPanicObjExists);
sl@0
   550
	const TInt KSecurityPolicyLen = (aMessage.Int2() & 0x7fff0000) >> 16;
sl@0
   551
    //If the security policy length is negative then this is a programming error.
sl@0
   552
    __SQLPANIC_CLIENT(KSecurityPolicyLen >= 0, aMessage, ESqlPanicBadArgument);
sl@0
   553
	const TInt KConfigStringLen = aMessage.Int2() & 0xffff;
sl@0
   554
	//If KConfigStringLen is invalid then this is a programming error. 
sl@0
   555
	//If the client sends a too big config string - this is handled in the client side session.
sl@0
   556
    __SQLPANIC_CLIENT((TUint)KConfigStringLen <= KSqlSrvMaxConfigStrLen, aMessage, ESqlPanicBadArgument);
sl@0
   557
	RBuf8 securityAndConfigBuf;
sl@0
   558
	CleanupClosePushL(securityAndConfigBuf);
sl@0
   559
	if((KSecurityPolicyLen + KConfigStringLen) > 0)
sl@0
   560
	    {
sl@0
   561
	    securityAndConfigBuf.CreateL(KSecurityPolicyLen + KConfigStringLen);
sl@0
   562
        aMessage.ReadL(3, securityAndConfigBuf); 
sl@0
   563
        SQLPROFILER_REPORT_IPC(ESqlIpcRead, (KSecurityPolicyLen + KConfigStringLen));
sl@0
   564
	    }
sl@0
   565
	TSqlSrvFileData& fileData = Server().FileData();
sl@0
   566
	TPtrC8 configStr(KNullDesC8);
sl@0
   567
	if(KConfigStringLen > 0)
sl@0
   568
	    {
sl@0
   569
	    configStr.Set(securityAndConfigBuf.Mid(KSecurityPolicyLen));//the first part of the buffer is for the security policies
sl@0
   570
	    }
sl@0
   571
	fileData.SetL(aMessage, aMessage.Int0(), 1, &configStr);
sl@0
   572
	iDrive = fileData.Drive();
sl@0
   573
	switch(aFunction)
sl@0
   574
		{
sl@0
   575
		case ESqlSrvDbCreate:
sl@0
   576
			if(fileData.IsSecureFileNameFmt())
sl@0
   577
				{
sl@0
   578
				__SQLLEAVE(KErrArgument);	
sl@0
   579
				}
sl@0
   580
			iDatabase = CSqlSrvDatabase::CreateL(fileData);
sl@0
   581
			break;
sl@0
   582
		case ESqlSrvDbCreateSecure:
sl@0
   583
			{
sl@0
   584
		    __SQLPANIC_CLIENT(KSecurityPolicyLen > 0, aMessage, ESqlPanicBadArgument);
sl@0
   585
			if(!fileData.IsSecureFileNameFmt())
sl@0
   586
				{
sl@0
   587
				__SQLLEAVE(KErrArgument);	
sl@0
   588
				}
sl@0
   589
			//The caller can create a secure database which secure UID matches his secure UID.
sl@0
   590
			if(fileData.SecureUid() != aMessage.SecureId())
sl@0
   591
				{
sl@0
   592
				__SQLLEAVE(KErrPermissionDenied);	
sl@0
   593
				}
sl@0
   594
			CSqlSecurityPolicy* policy = CreateSecurityPolicyL(securityAndConfigBuf.Left(KSecurityPolicyLen));
sl@0
   595
			iDatabase = CSqlSrvDatabase::CreateSecureL(fileData, policy);
sl@0
   596
			}
sl@0
   597
			break;
sl@0
   598
		case ESqlSrvDbOpen:
sl@0
   599
			iDatabase = CSqlSrvDatabase::OpenL(fileData);
sl@0
   600
			break;
sl@0
   601
		default:
sl@0
   602
			__SQLLEAVE(KErrArgument);	
sl@0
   603
			break;
sl@0
   604
		}
sl@0
   605
	CleanupStack::PopAndDestroy(&securityAndConfigBuf);
sl@0
   606
	}
sl@0
   607
sl@0
   608
/**
sl@0
   609
Processes the request for opening a database from file handle.
sl@0
   610
The server expects that the database to be opened/created is in the applicatio's private data cage.
sl@0
   611
sl@0
   612
Usage of the IPC call arguments:
sl@0
   613
 - Arg 0: [in]  The 32 bits of the argument are used as follow:
sl@0
   614
 @code
sl@0
   615
 MSB                                                                                 LSB
sl@0
   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
sl@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 
sl@0
   618
 @endcode
sl@0
   619
 Where:
sl@0
   620
 @code
sl@0
   621
  - "Ro" - read-only flag, true if the file is read-only;
sl@0
   622
  - "Cr" - create/open flag, true if the file was created, false if the file was opened;
sl@0
   623
  - "C"  - config string length in 16-bit characters;
sl@0
   624
  - "F"  - database file name length in 16-bit characters;
sl@0
   625
 @endcode
sl@0
   626
 - Arg 1: [in]  database file name | config string
sl@0
   627
 - Arg 2: [in]  file session handle
sl@0
   628
 - Arg 3: [in]  database file handle
sl@0
   629
sl@0
   630
@panic SqlDb 1 Client panic. iDatabase is not NULL (it has been created already)
sl@0
   631
*/
sl@0
   632
void CSqlSrvSession::DbCreateObjectFromHandleL(const RMessage2& aMessage)
sl@0
   633
	{
sl@0
   634
	__SQLPANIC_CLIENT(!iDatabase, aMessage, ESqlPanicObjExists);
sl@0
   635
	const TBool KReadOnly = (aMessage.Int0() & 0x80000000) != 0;
sl@0
   636
	const TBool KCreated = (aMessage.Int0() & 0x40000000) != 0;
sl@0
   637
	const TInt KDbFileNameLen = aMessage.Int0() & 0x0000FFFF;
sl@0
   638
	const TInt KConfigStringLen = (aMessage.Int0() & 0x3FFF0000) >> 16;
sl@0
   639
    __SQLPANIC_CLIENT((TUint)KConfigStringLen <= KSqlSrvMaxConfigStrLen, aMessage, ESqlPanicBadArgument);
sl@0
   640
    __SQLPANIC_CLIENT((TUint)KDbFileNameLen <= KMaxFileName, aMessage, ESqlPanicBadArgument);
sl@0
   641
	TDes16& buffer = Server().GetBuf16L(KDbFileNameLen + KConfigStringLen);
sl@0
   642
	aMessage.ReadL(1, buffer);
sl@0
   643
	SQLPROFILER_REPORT_IPC(ESqlIpcRead, ((KDbFileNameLen + KConfigStringLen) * sizeof(TText)));
sl@0
   644
	TFileName dbFileName;
sl@0
   645
	dbFileName.Copy(buffer.LeftTPtr(KDbFileNameLen));
sl@0
   646
	TBuf8<KSqlSrvMaxConfigStrLen> configStr;
sl@0
   647
	if(KConfigStringLen > 0)
sl@0
   648
		{
sl@0
   649
		configStr.Copy(buffer.MidTPtr(KDbFileNameLen, KConfigStringLen));
sl@0
   650
		}
sl@0
   651
	TSqlSrvFileData& fileData = Server().FileData();
sl@0
   652
	fileData.SetFromHandleL(aMessage, dbFileName, KCreated, KReadOnly, &configStr);
sl@0
   653
	iDrive = fileData.Drive();
sl@0
   654
	iDatabase = CSqlSrvDatabase::OpenL(fileData);
sl@0
   655
	}
sl@0
   656
sl@0
   657
/**
sl@0
   658
Processes the request for copying a database.
sl@0
   659
sl@0
   660
Only the database creator can copy the database if the database is a secure database.
sl@0
   661
sl@0
   662
Usage of the IPC call arguments:
sl@0
   663
Arg 0: [in]  source database file name length
sl@0
   664
Arg 1: [in]  source database file name
sl@0
   665
Arg 2: [in]  destination database file name length
sl@0
   666
Arg 3: [in]  destination database file name
sl@0
   667
*/
sl@0
   668
void CSqlSrvSession::DbCopyFileL(const RMessage2& aMessage)
sl@0
   669
	{
sl@0
   670
	const TInt KDbCnt = 2;											//"2" - because we have 2 dbases: src and dest
sl@0
   671
	const TInt KSrcDbIdx = 0;
sl@0
   672
	const TInt KDestDbIdx = 1;
sl@0
   673
	TInt fileNameLen[KDbCnt] = {aMessage.Int0(), aMessage.Int2()};
sl@0
   674
	TSqlSrvFileData& fileData = Server().FileData();
sl@0
   675
	TUint dbSecureFlag[KDbCnt];
sl@0
   676
	TUid  dbSID[KDbCnt] = {KNullUid, KNullUid};
sl@0
   677
	TFileName dbFileName[KDbCnt];
sl@0
   678
	//Initialize dbSecureFlag[], dbSID[] and dbFileName[] array elements
sl@0
   679
	for(TInt i=0;i<KDbCnt;++i)									
sl@0
   680
		{
sl@0
   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
sl@0
   682
		dbSecureFlag[i] = fileData.IsSecureFileNameFmt() ? 1 : 0;
sl@0
   683
		if(dbSecureFlag[i])
sl@0
   684
			{
sl@0
   685
			dbSID[i] = fileData.SecureUid();
sl@0
   686
			}
sl@0
   687
		dbFileName[i].Copy(fileData.FileName());
sl@0
   688
		}
sl@0
   689
	//It is not allowed to copy non-secure to a secure or secure to a non-secure database.
sl@0
   690
	if(dbSecureFlag[KSrcDbIdx] ^ dbSecureFlag[KDestDbIdx])
sl@0
   691
		{
sl@0
   692
		__SQLLEAVE(KErrPermissionDenied);	
sl@0
   693
		}
sl@0
   694
	//If this is a secure database "copy" operation, then...
sl@0
   695
	if(dbSecureFlag[KSrcDbIdx])
sl@0
   696
		{
sl@0
   697
		TUid callerSid = aMessage.SecureId();
sl@0
   698
		//A secure database can be copied only by its owner (database SID matches caller SID).
sl@0
   699
		if(callerSid != dbSID[KSrcDbIdx] || callerSid != dbSID[KDestDbIdx])
sl@0
   700
			{
sl@0
   701
			__SQLLEAVE(KErrPermissionDenied);	
sl@0
   702
			}
sl@0
   703
		}
sl@0
   704
	//Copy the database
sl@0
   705
	CFileMan* fileMan = CFileMan::NewL(fileData.Fs());
sl@0
   706
	CleanupStack::PushL(fileMan);
sl@0
   707
	__SQLLEAVE_IF_ERROR(fileMan->Copy(dbFileName[KSrcDbIdx], dbFileName[KDestDbIdx]));
sl@0
   708
	//"Copy" operation executed without errors. Now it is a time to turn off the read-only
sl@0
   709
	//flag of the target file (which may be on if the source file is on a read-only drive)
sl@0
   710
	__SQLLEAVE_IF_ERROR(fileData.Fs().SetAtt(dbFileName[KDestDbIdx], 0, KEntryAttReadOnly));
sl@0
   711
	CleanupStack::PopAndDestroy(fileMan);
sl@0
   712
	}
sl@0
   713
sl@0
   714
/**
sl@0
   715
Processes the request for deleting a database.
sl@0
   716
sl@0
   717
Only the database creator can delete the database if the database is a secure database.
sl@0
   718
sl@0
   719
Usage of the IPC call arguments:
sl@0
   720
Arg 0: [in]  database file name length
sl@0
   721
Arg 1: [in]  database file name
sl@0
   722
*/
sl@0
   723
void CSqlSrvSession::DbDeleteFileL(const RMessage2& aMessage)
sl@0
   724
	{	
sl@0
   725
	TSqlSrvFileData& fileData = Server().FileData();
sl@0
   726
	fileData.SetL(aMessage, aMessage.Int0(), 1);
sl@0
   727
	if(fileData.IsSecureFileNameFmt())
sl@0
   728
		{
sl@0
   729
		//A secure database can be deleted only by its owner (database SID matches caller SID).
sl@0
   730
		if(fileData.SecureUid() != aMessage.SecureId())
sl@0
   731
			{
sl@0
   732
			__SQLLEAVE(KErrPermissionDenied);	
sl@0
   733
			}
sl@0
   734
		}
sl@0
   735
	__SQLTRACE_SESSIONVAR(TPtrC fname(fileData.FileName()));
sl@0
   736
	SQL_TRACE_SESSION(OstTraceExt2(TRACE_INTERNALS, CSQLSRVSESSION_DBDELETEFILEL, "0x%X;CSqlSrvSession::DbDeleteFileL;file='%S'", (TUint)this, __SQLPRNSTR(fname)));
sl@0
   737
	__SQLLEAVE_IF_ERROR(fileData.Fs().Delete(fileData.FileName()));
sl@0
   738
	}
sl@0
   739
sl@0
   740
/**
sl@0
   741
Processes the request for retrieving the last error message.
sl@0
   742
sl@0
   743
If the client side buffer size is not big enough, the function returns the needed 
sl@0
   744
buffer size + KSqlClientBufOverflowCode.
sl@0
   745
In this case the client must increase the buffer and try again to get the message.
sl@0
   746
sl@0
   747
Usage of the IPC call arguments:
sl@0
   748
Arg 0: [in]		Message buffer length in 16-bit characters
sl@0
   749
Arg 1: [in/out]	Message buffer
sl@0
   750
sl@0
   751
@panic SqlDb 2 Client panic. iDatabase is NULL (the database object is not created yet).
sl@0
   752
@panic SqlDb 4 Client panic. Negative client message buffer length (Arg 0).
sl@0
   753
*/
sl@0
   754
TInt CSqlSrvSession::DbLastErrorMessageL(const RMessage2& aMessage)
sl@0
   755
	{
sl@0
   756
	__SQLPANIC_CLIENT(iDatabase != NULL, aMessage, ESqlPanicInvalidObj);
sl@0
   757
	TPtrC msg = iDatabase->LastErrorMessage();
sl@0
   758
	TInt msgLen = msg.Length();
sl@0
   759
	TInt bufSize = aMessage.Int0();
sl@0
   760
	__SQLPANIC_CLIENT(bufSize >= 0, aMessage, ESqlPanicBadArgument);
sl@0
   761
	if(msgLen <= bufSize)
sl@0
   762
		{
sl@0
   763
		aMessage.WriteL(1, msg);
sl@0
   764
		SQLPROFILER_REPORT_IPC(ESqlIpcWrite, (msgLen * sizeof(TText)));
sl@0
   765
		return 0;
sl@0
   766
		}
sl@0
   767
	return msgLen + KSqlClientBufOverflowCode;
sl@0
   768
	}
sl@0
   769
	
sl@0
   770
/**
sl@0
   771
Processes the request for retrieving the last inserted ROWID of this database connection.
sl@0
   772
sl@0
   773
Usage of the IPC call arguments:
sl@0
   774
Arg 0: [in/out]	Receiving buffer
sl@0
   775
sl@0
   776
@panic SqlDb 2 Client panic. The database object is not yet created yet (iDatabase is NULL).
sl@0
   777
*/
sl@0
   778
void CSqlSrvSession::DbLastInsertedRowIdL(const RMessage2& aMessage)
sl@0
   779
	{
sl@0
   780
	__SQLPANIC_CLIENT(iDatabase != NULL, aMessage, ESqlPanicInvalidObj);
sl@0
   781
	TInt64 rowid = iDatabase->LastInsertedRowId();
sl@0
   782
	aMessage.WriteL(0, TPtrC8(reinterpret_cast <const TUint8*> (&rowid), sizeof(rowid)));
sl@0
   783
	SQLPROFILER_REPORT_IPC(ESqlIpcWrite, sizeof(rowid));
sl@0
   784
	}
sl@0
   785
sl@0
   786
/**
sl@0
   787
Processes the request for retrieving the database security policies.
sl@0
   788
sl@0
   789
The method leaves with KErrNotSupported if the database is not a secure database.
sl@0
   790
sl@0
   791
If the client side buffer size is not big enough, the function returns the needed 
sl@0
   792
buffer size + KSqlClientBufOverflowCode.
sl@0
   793
In this case the client must increase the buffer and try again to get the database security policy.
sl@0
   794
sl@0
   795
Usage of the IPC call arguments:
sl@0
   796
Arg 0: [in]		security policy buffer length in bytes
sl@0
   797
Arg 1: [in/out]	buffer for the database security policies
sl@0
   798
sl@0
   799
@panic SqlDb 2 Client panic. iDatabase is NULL (the database object is not created yet).
sl@0
   800
*/
sl@0
   801
TInt CSqlSrvSession::DbGetSecurityPolicyL(const RMessage2& aMessage)
sl@0
   802
	{
sl@0
   803
	__SQLPANIC_CLIENT(iDatabase != NULL, aMessage, ESqlPanicInvalidObj);
sl@0
   804
	const CSqlSecurityPolicy* securityPolicy = iDatabase->SecurityPolicy();
sl@0
   805
	if(!securityPolicy)
sl@0
   806
		{
sl@0
   807
		__SQLLEAVE(KErrNotSupported);
sl@0
   808
		}
sl@0
   809
	const RSqlBufFlat& bufFlat = securityPolicy->BufFlat();
sl@0
   810
	TInt size = bufFlat.Size();
sl@0
   811
	if(size <= aMessage.Int0())
sl@0
   812
		{
sl@0
   813
		aMessage.WriteL(1, bufFlat.BufDes());
sl@0
   814
		SQLPROFILER_REPORT_IPC(ESqlIpcWrite, size);
sl@0
   815
		return 0;
sl@0
   816
		}
sl@0
   817
	return size + KSqlClientBufOverflowCode;
sl@0
   818
	}
sl@0
   819
sl@0
   820
/**
sl@0
   821
If an error occurs during the execution the function leaves with the error code.
sl@0
   822
Possible non-leaving return values:
sl@0
   823
 - KErrNone              - the function has completed successfully;
sl@0
   824
 - Positive return value - the length of the column, which means - the destination buffer is too small.
sl@0
   825
                           This return value is possible only with text or binary columns.
sl@0
   826
sl@0
   827
@panic SqlDb 2 Client panic. iDatabase is NULL (the database object is not created yet).
sl@0
   828
sl@0
   829
Usage of the IPC call arguments: 
sl@0
   830
Arg 0: [in]		(8/16-bit character length of SQL statement) | (expected column value type << 24).
sl@0
   831
Arg 1: [in]		SQL statement.
sl@0
   832
Arg 2: [in]		Byte max length of the receiving buffer
sl@0
   833
Arg 3: [in/out]	The receiving buffer
sl@0
   834
*/
sl@0
   835
TInt CSqlSrvSession::DbScalarFullSelectL(const RMessage2& aMessage, TBool aIsText16)
sl@0
   836
	{
sl@0
   837
	__SQLPANIC_CLIENT(iDatabase != NULL, aMessage, ESqlPanicInvalidObj);
sl@0
   838
	TUint sqlLen = static_cast <TUint> (aMessage.Int0()) & 0x00FFFFFF;
sl@0
   839
	TSqlColumnType  colType = static_cast <TSqlColumnType> ((static_cast <TUint> (aMessage.Int0()) & 0xFF000000) >> 24);
sl@0
   840
	TInt columnCount = -1;
sl@0
   841
	TInt paramCount = -1;
sl@0
   842
	CSqlSrvStatement*  stmt = aIsText16 ? 
sl@0
   843
								CSqlSrvStatement::NewLC(iDatabase->RawDbHandle(), ReadString16ZL(aMessage, 1, sqlLen), columnCount, paramCount) :
sl@0
   844
								CSqlSrvStatement::NewLC(iDatabase->RawDbHandle(), ReadString8ZL(aMessage, 1, sqlLen), columnCount, paramCount);
sl@0
   845
	if(columnCount != 1 || paramCount != 0)
sl@0
   846
		{
sl@0
   847
		__SQLLEAVE(KErrArgument);	
sl@0
   848
		}
sl@0
   849
	TInt err = stmt->Next();
sl@0
   850
	if(err == KSqlAtRow)
sl@0
   851
		{
sl@0
   852
		err = GetColumnValueL(aMessage, *stmt, colType);
sl@0
   853
		}
sl@0
   854
	else
sl@0
   855
		{
sl@0
   856
		__SQLLEAVE(err == KSqlAtEnd ? KErrNotFound : err);
sl@0
   857
		}
sl@0
   858
	CleanupStack::PopAndDestroy(stmt);
sl@0
   859
	return err;
sl@0
   860
	}
sl@0
   861
sl@0
   862
/**
sl@0
   863
@return True, if the database is in transaction, false otherwise.
sl@0
   864
sl@0
   865
@panic SqlDb 2 Client panic. iDatabase is NULL (the database object is not created yet).
sl@0
   866
*/
sl@0
   867
TBool CSqlSrvSession::DbInTransaction(const RMessage2& aMessage)
sl@0
   868
	{
sl@0
   869
	__SQLPANIC_CLIENT(iDatabase != NULL, aMessage, ESqlPanicInvalidObj);
sl@0
   870
	return iDatabase->InTransaction();
sl@0
   871
	}
sl@0
   872
sl@0
   873
/**
sl@0
   874
Main database size in bytes.
sl@0
   875
sl@0
   876
@return Main database size in bytes.
sl@0
   877
sl@0
   878
@leave KErrNoMemory, an out of memory condition has occurred,
sl@0
   879
                     Note that the function may also leave with some other system wide errors or 
sl@0
   880
                     database specific errors categorised as ESqlDbError,
sl@0
   881
	   KErrTooBig,   The database is very big and the size cannot fit in a 32-bit signed integer.
sl@0
   882
sl@0
   883
@panic SqlDb 2 Client panic. iDatabase is NULL (the database object is not created yet).
sl@0
   884
*/
sl@0
   885
TInt CSqlSrvSession::DbSizeL(const RMessage2& aMessage)
sl@0
   886
	{
sl@0
   887
	__SQLPANIC_CLIENT(iDatabase != NULL, aMessage, ESqlPanicInvalidObj);
sl@0
   888
	TInt64 size = iDatabase->SizeL();
sl@0
   889
	if(size > KMaxTInt)
sl@0
   890
		{
sl@0
   891
		__SQLLEAVE(KErrTooBig);
sl@0
   892
		}
sl@0
   893
	return size;
sl@0
   894
	}
sl@0
   895
sl@0
   896
/**
sl@0
   897
Retrieves the database size and free space.
sl@0
   898
sl@0
   899
Usage of the IPC call arguments:
sl@0
   900
Arg 0: [in/out]	Points to a RSqlDatabase::TSize object, where the database size and free space values
sl@0
   901
			    will be copied.
sl@0
   902
Arg 1: [in]		The database name length in 16-bit characters
sl@0
   903
Arg 2: [in]		The attached database name or KNullDesC for the main database
sl@0
   904
sl@0
   905
@leave KErrNoMemory, an out of memory condition has occurred,
sl@0
   906
                     Note that the function may also leave with some other system wide errors or 
sl@0
   907
                     database specific errors categorised as ESqlDbError.
sl@0
   908
       KErrBadName, Invalid database name
sl@0
   909
sl@0
   910
@panic SqlDb 2 Client panic. iDatabase is NULL (the database object is not created yet).
sl@0
   911
*/
sl@0
   912
void CSqlSrvSession::DbSize2L(const RMessage2& aMessage)
sl@0
   913
	{
sl@0
   914
	__SQLPANIC_CLIENT(iDatabase != NULL, aMessage, ESqlPanicInvalidObj);
sl@0
   915
	const TInt KDbNameLen = aMessage.Int1();
sl@0
   916
	if((TUint)KDbNameLen > KMaxFileName)
sl@0
   917
		{
sl@0
   918
		__SQLLEAVE(KErrBadName);
sl@0
   919
		}
sl@0
   920
	TPtrC dbName(KNullDesC);
sl@0
   921
	if(KDbNameLen > 0)
sl@0
   922
		{
sl@0
   923
		dbName.Set(ReadString16L(aMessage, 2, KDbNameLen));
sl@0
   924
		}
sl@0
   925
	TPckgBuf<RSqlDatabase::TSize> data;
sl@0
   926
	data().iSize = iDatabase->SizeL(dbName);
sl@0
   927
	data().iFree = iDatabase->FreeSpaceL(dbName);
sl@0
   928
	aMessage.WriteL(0, data);
sl@0
   929
	SQLPROFILER_REPORT_IPC(ESqlIpcWrite, sizeof(RSqlDatabase::TSize));
sl@0
   930
	}
sl@0
   931
sl@0
   932
/**
sl@0
   933
Runs database compaction.
sl@0
   934
sl@0
   935
Usage of the IPC call arguments:
sl@0
   936
Arg 0: [in]	    How much space in bytes should be compacted, all free pages should be removed if the
sl@0
   937
				parameter value is RSqlDatabase::EMaxCompaction.
sl@0
   938
				Note that the requested space to be compacted will be rounded up to the nearest page count,
sl@0
   939
				e.g. request for removing 1 byte will remove one free page from the database file. 
sl@0
   940
Arg 1: [in]		The database name length in characters
sl@0
   941
Arg 2: [in]		The attached database name or KNullDesC for the main database
sl@0
   942
sl@0
   943
@return The size of the removed free space
sl@0
   944
sl@0
   945
@panic SqlDb 2 Client panic. iDatabase is NULL (the database object is not created yet).
sl@0
   946
*/
sl@0
   947
TInt CSqlSrvSession::DbCompactL(const RMessage2& aMessage)
sl@0
   948
	{
sl@0
   949
	__SQLPANIC_CLIENT(iDatabase != NULL, aMessage, ESqlPanicInvalidObj);
sl@0
   950
	const TInt KSize = aMessage.Int0();
sl@0
   951
	if(KSize < 0)
sl@0
   952
		{
sl@0
   953
		if(KSize != RSqlDatabase::EMaxCompaction)
sl@0
   954
			{
sl@0
   955
			__SQLLEAVE(KErrArgument);
sl@0
   956
			}
sl@0
   957
		}
sl@0
   958
	if(KSize == 0)
sl@0
   959
		{
sl@0
   960
		return 0;	
sl@0
   961
		}
sl@0
   962
	const TInt KDbNameLen = aMessage.Int1();
sl@0
   963
	if((TUint)KDbNameLen > KMaxFileName)
sl@0
   964
		{
sl@0
   965
		__SQLLEAVE(KErrBadName);
sl@0
   966
		}
sl@0
   967
	TPtrC dbName(KNullDesC);
sl@0
   968
	if(KDbNameLen > 0)
sl@0
   969
		{
sl@0
   970
		dbName.Set(ReadString16L(aMessage, 2, KDbNameLen));
sl@0
   971
		}
sl@0
   972
	return iDatabase->CompactL(KSize, dbName);
sl@0
   973
	}
sl@0
   974
	
sl@0
   975
/**
sl@0
   976
Usage of the IPC call arguments:
sl@0
   977
Arg 0: [in]		requested size of the space to be reserved - not used
sl@0
   978
sl@0
   979
The function leaves with KErrAlreadyExists if a drive space has been reserved already by this session.
sl@0
   980
*/
sl@0
   981
void CSqlSrvSession::DbReserveDriveSpaceL()
sl@0
   982
	{
sl@0
   983
	if(iDriveSpaceReserved)
sl@0
   984
		{
sl@0
   985
		__SQLLEAVE(KErrAlreadyExists);
sl@0
   986
		}
sl@0
   987
	RSqlDriveSpaceCol& driveSpaceCol = Server().DriveSpaceCol();
sl@0
   988
    if(!driveSpaceCol.Find(iDrive))
sl@0
   989
    	{
sl@0
   990
    	(void)driveSpaceCol.AddL(iDrive);
sl@0
   991
    	}
sl@0
   992
	
sl@0
   993
	iDriveSpaceReserved = ETrue;
sl@0
   994
	//Only iDriveSpaceReserved is set, nothing more needs to be done, because RSqlDriveSpaceCol::AddL() will
sl@0
   995
	//reserve a drive space on the specified drive.
sl@0
   996
	//Although it looks like that the implementation can ommit "iDriveSpaceReserved" flag, this flag plays important
sl@0
   997
	//role, because it is used to ensure that every "reserve drive space" request is matched by a "free drive space"
sl@0
   998
	//call.
sl@0
   999
	}
sl@0
  1000
	
sl@0
  1001
/**
sl@0
  1002
If the client has been given an access to the reserved drive space, that access will be released.
sl@0
  1003
*/
sl@0
  1004
void CSqlSrvSession::DbFreeReservedSpace()
sl@0
  1005
	{
sl@0
  1006
	DbReleaseReserveAccess();
sl@0
  1007
	iDriveSpaceReserved = EFalse;
sl@0
  1008
	}
sl@0
  1009
	
sl@0
  1010
/**
sl@0
  1011
The function leaves with KErrInUse if an access to the reserved drive space has been given to the client.
sl@0
  1012
The function leaves with KErrNotFound if no drive space has been reserved for the drive, where the database file is.
sl@0
  1013
*/
sl@0
  1014
void CSqlSrvSession::DbGetReserveAccessL()
sl@0
  1015
	{
sl@0
  1016
	if(iDriveSpaceInUse)
sl@0
  1017
		{
sl@0
  1018
		__SQLLEAVE(KErrInUse);	
sl@0
  1019
		}
sl@0
  1020
	if(!iDriveSpaceReserved)
sl@0
  1021
		{
sl@0
  1022
		__SQLLEAVE(KErrNotFound);
sl@0
  1023
		}
sl@0
  1024
	CSqlDriveSpace* driveSpace = Server().DriveSpaceCol().Find(iDrive);
sl@0
  1025
   	if(!driveSpace)
sl@0
  1026
   		{
sl@0
  1027
		__SQLLEAVE(KErrNotFound);
sl@0
  1028
   		}
sl@0
  1029
   	driveSpace->GetAccessL();
sl@0
  1030
   	iDriveSpaceInUse = ETrue;
sl@0
  1031
	}
sl@0
  1032
	
sl@0
  1033
/**
sl@0
  1034
Releases the drive space reserve if it has been in use by this session (resp. client).
sl@0
  1035
*/
sl@0
  1036
void CSqlSrvSession::DbReleaseReserveAccess()
sl@0
  1037
	{
sl@0
  1038
	if(iDriveSpaceInUse)
sl@0
  1039
		{
sl@0
  1040
		CSqlDriveSpace* driveSpace = Server().DriveSpaceCol().Find(iDrive);
sl@0
  1041
		if(driveSpace)
sl@0
  1042
			{
sl@0
  1043
   			driveSpace->ReleaseAccess();
sl@0
  1044
			}
sl@0
  1045
   		iDriveSpaceInUse = EFalse;
sl@0
  1046
		}
sl@0
  1047
	}
sl@0
  1048
sl@0
  1049
/**
sl@0
  1050
Processes the request for attaching a secure or non-secure database.
sl@0
  1051
sl@0
  1052
Usage of the IPC call arguments: 
sl@0
  1053
Arg 0: [in]	Database file name length (counted in 16-bit characters).
sl@0
  1054
Arg 1: [in]	Database file name.
sl@0
  1055
Arg 2: [in]	Logical database name length (counted in 16-bit characters).
sl@0
  1056
Arg 3: [in]	Logical database name.
sl@0
  1057
sl@0
  1058
@panic SqlDb 2 Client panic. iDatabase is NULL (the database object is not created yet).
sl@0
  1059
*/
sl@0
  1060
void CSqlSrvSession::DbAttachL(const RMessage2& aMessage)
sl@0
  1061
	{
sl@0
  1062
	__SQLPANIC_CLIENT(iDatabase != NULL, aMessage, ESqlPanicInvalidObj);
sl@0
  1063
	TSqlSrvFileData& fileData = Server().FileData();
sl@0
  1064
	fileData.SetL(aMessage, aMessage.Int0(), 1);
sl@0
  1065
	TInt logicalDbNameLen = aMessage.Int2();
sl@0
  1066
	if(logicalDbNameLen < 1 || logicalDbNameLen > KMaxFileName)
sl@0
  1067
		{
sl@0
  1068
		__SQLLEAVE(KErrBadName);
sl@0
  1069
		}
sl@0
  1070
	iDatabase->AttachDbL(fileData, ReadString16L(aMessage, 3, logicalDbNameLen));
sl@0
  1071
	}
sl@0
  1072
sl@0
  1073
/**
sl@0
  1074
Processes the request for attaching a database using file session and file handles sent by the client.
sl@0
  1075
sl@0
  1076
Usage of the IPC call arguments:
sl@0
  1077
 - Arg 0: [in]  The 32 bits of the argument are used as follow:
sl@0
  1078
 @code
sl@0
  1079
 MSB                                                                                 LSB
sl@0
  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
sl@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
sl@0
  1082
 @endcode
sl@0
  1083
 Where:
sl@0
  1084
 @code
sl@0
  1085
  - "Ro" - read-only flag, true if the file is read-only;
sl@0
  1086
  - "F"  - database file name length in 16-bit characters;
sl@0
  1087
 @endcode
sl@0
  1088
Arg 1: [in]  db names buffer
sl@0
  1089
Arg 2: [in]  file session handle
sl@0
  1090
Arg 3: [in]  database file handle
sl@0
  1091
sl@0
  1092
@panic SqlDb 2 Client panic. iDatabase is NULL (the database object is not created yet).
sl@0
  1093
@panic SqlDb 4 Client panic. Invalid IPC data, an indication of a problme in client side sql library.
sl@0
  1094
*/
sl@0
  1095
void CSqlSrvSession::DbAttachFromHandleL(const RMessage2& aMessage)
sl@0
  1096
	{
sl@0
  1097
	__SQLPANIC_CLIENT(iDatabase != NULL, aMessage, ESqlPanicInvalidObj);
sl@0
  1098
	//Read-only flag, buffer length, buffer allocation
sl@0
  1099
	TBool readOnly = (aMessage.Int0() & 0x80000000) != 0;
sl@0
  1100
	const TInt KBufLen = aMessage.Int0() & 0x7FFFFFFF;
sl@0
  1101
    __SQLPANIC_CLIENT(KBufLen > 0, aMessage, ESqlPanicBadArgument);
sl@0
  1102
	HBufC8* buf = HBufC8::NewLC(KBufLen);
sl@0
  1103
	TPtr8 bufPtr = buf->Des();
sl@0
  1104
	aMessage.ReadL(1, bufPtr);
sl@0
  1105
	SQLPROFILER_REPORT_IPC(ESqlIpcRead, KBufLen);
sl@0
  1106
	if(KBufLen != bufPtr.Length())
sl@0
  1107
		{
sl@0
  1108
		__SQLLEAVE(KErrArgument);
sl@0
  1109
		}
sl@0
  1110
	RDesReadStream in(bufPtr);
sl@0
  1111
	TDes& dbFileName = Server().FileNameBuf();
sl@0
  1112
	TDes16& dbName = Server().GetBuf16L(KMaxFileName);
sl@0
  1113
	in >> dbFileName;
sl@0
  1114
	in >> dbName;
sl@0
  1115
	CleanupStack::PopAndDestroy(buf);
sl@0
  1116
	TSqlSrvFileData& fileData = Server().FileData();
sl@0
  1117
	fileData.SetFromHandleL(aMessage, dbFileName, EFalse, readOnly);
sl@0
  1118
	iDatabase->AttachDbL(fileData, dbName);
sl@0
  1119
	}
sl@0
  1120
	
sl@0
  1121
/**
sl@0
  1122
Processes the request for detaching a secure or non-secure database.
sl@0
  1123
sl@0
  1124
Usage of the IPC call arguments: 
sl@0
  1125
Arg 0: [in]	Logical database name length (counted in 16-bit characters).
sl@0
  1126
Arg 1: [in]	Logical database name.
sl@0
  1127
sl@0
  1128
@panic SqlDb 2 Client panic. iDatabase is NULL (the database object is not created yet).
sl@0
  1129
*/
sl@0
  1130
void CSqlSrvSession::DbDetachL(const RMessage2& aMessage)
sl@0
  1131
	{
sl@0
  1132
	__SQLPANIC_CLIENT(iDatabase != NULL, aMessage, ESqlPanicInvalidObj);
sl@0
  1133
	TInt logicalDbNameLen = aMessage.Int0();
sl@0
  1134
	if(logicalDbNameLen < 1 || logicalDbNameLen > KMaxFileName)
sl@0
  1135
		{
sl@0
  1136
		__SQLLEAVE(KErrBadName);
sl@0
  1137
		}
sl@0
  1138
	iDatabase->DetachDbL(ReadString16L(aMessage, 1, logicalDbNameLen));
sl@0
  1139
	}
sl@0
  1140
sl@0
  1141
/**
sl@0
  1142
Reads a 16-bit string from the specified stream and returns it in zero-terminated
sl@0
  1143
8-bit format in aNameOut.
sl@0
  1144
If the string is of zero length then the substitute string provided will be used instead.
sl@0
  1145
sl@0
  1146
@param aStrm 				The read stream
sl@0
  1147
@param aNameOut 			The output parameter that will contain the string read
sl@0
  1148
@param aEmptyNameSubstitute The substitute string to use if the string to be read from
sl@0
  1149
							the stream is zero length
sl@0
  1150
sl@0
  1151
@leave KErrNoMemory, 		 An out of memory condition has occurred;
sl@0
  1152
	   KErrArgument, 		 The UTF-16 to UTF-8 string conversion failed;	
sl@0
  1153
	   KErrBadName,	 		 The string has an invalid length;
sl@0
  1154
*/
sl@0
  1155
void CSqlSrvSession::ExtractNameL(RDesReadStream& aStrm, TDes8& aNameOut, const TDesC& aEmptyNameSubstitute)
sl@0
  1156
	{
sl@0
  1157
	 TBool replace = EFalse;
sl@0
  1158
	 TInt32 len;
sl@0
  1159
	 aStrm >> len;
sl@0
  1160
sl@0
  1161
	 if(len == 0)
sl@0
  1162
	 	{
sl@0
  1163
	   	if(aEmptyNameSubstitute.Length() > 0)
sl@0
  1164
			{
sl@0
  1165
		 	len = aEmptyNameSubstitute.Length();
sl@0
  1166
	 	 	replace = ETrue;
sl@0
  1167
			}
sl@0
  1168
	 	else
sl@0
  1169
			{
sl@0
  1170
			__SQLLEAVE(KErrBadName);
sl@0
  1171
			}
sl@0
  1172
	  	}
sl@0
  1173
	 __ASSERT_DEBUG(len > 0, __SQLPANIC2(ESqlPanicInternalError));//The "if" above should have hanled the case with "len == 0"
sl@0
  1174
	 if((TUint)len > KMaxFileName)
sl@0
  1175
	  {
sl@0
  1176
	  __SQLLEAVE(KErrBadName);
sl@0
  1177
	  }
sl@0
  1178
sl@0
  1179
	 HBufC* buf = HBufC::NewLC(len + 1);
sl@0
  1180
	 TPtr ptr = buf->Des();
sl@0
  1181
	 if(replace)
sl@0
  1182
	 	{
sl@0
  1183
	  	ptr.Copy(aEmptyNameSubstitute);
sl@0
  1184
	  	}
sl@0
  1185
	 else
sl@0
  1186
	 	{
sl@0
  1187
	  	aStrm >> ptr;
sl@0
  1188
	 	}
sl@0
  1189
	 ptr.Append(0);
sl@0
  1190
	 
sl@0
  1191
	 if(!::UTF16ZToUTF8Z(ptr, aNameOut))
sl@0
  1192
	  {
sl@0
  1193
	  __SQLLEAVE(KErrArgument);
sl@0
  1194
	  }
sl@0
  1195
	  
sl@0
  1196
	 CleanupStack::PopAndDestroy(buf);
sl@0
  1197
	 }
sl@0
  1198
sl@0
  1199
/**
sl@0
  1200
Processes the request for creating an IPC stream for accessing the content of a blob column.
sl@0
  1201
sl@0
  1202
@param aMessage The client request wrapped in an IPC message
sl@0
  1203
sl@0
  1204
@return The blob stream handle
sl@0
  1205
sl@0
  1206
@leave KErrNoMemory, 		 An out of memory condition has occurred;
sl@0
  1207
	   KErrArgument, 		 The ROWID is invalid or UTF-16 to UTF-8 string conversion failed;	
sl@0
  1208
	   KErrBadDescriptor 	 The transferred data is bigger than the specified length;
sl@0
  1209
	   KErrBadName,	 		 The table name, column name or database name has an invalid length;
sl@0
  1210
       KErrPermissionDenied, The client does not have the required security capabilites for this operation; 						 
sl@0
  1211
       						 Note that the function may also leave with some other system wide errors or 
sl@0
  1212
                     		 database specific errors categorised as ESqlDbError.
sl@0
  1213
sl@0
  1214
@panic SqlDb 2 Client panic. The database object is not yet created (iDatabase is NULL).
sl@0
  1215
@panic SqlDb 3 Client panic. Failed to create a blob stream handle.
sl@0
  1216
@panic SqlDb 4 Client panic. IPC buffer length is 0.
sl@0
  1217
sl@0
  1218
Usage of the IPC call arguments:
sl@0
  1219
Arg 0: [in]	    The length of the IPC data buffer
sl@0
  1220
Arg 1: [in]	    IPC data buffer containing blob parameters: table name, column name, rowid, mode, database name.
sl@0
  1221
Arg 2: [out]	IPC buffer containing the blob stream handle
sl@0
  1222
*/
sl@0
  1223
TInt CSqlSrvSession::DbBlobSourceL(const RMessage2& aMessage)
sl@0
  1224
	{	
sl@0
  1225
	__SQLPANIC_CLIENT(iDatabase != NULL, aMessage, ESqlPanicInvalidObj);
sl@0
  1226
	
sl@0
  1227
	TInt ipcPrmLen = aMessage.Int0();
sl@0
  1228
	__SQLPANIC_CLIENT(ipcPrmLen > 0, aMessage, ESqlPanicBadArgument);
sl@0
  1229
	
sl@0
  1230
	iIpcStreams.AllocL();
sl@0
  1231
	
sl@0
  1232
	TDes8& ipcPrmDes = ReadString8ZL(aMessage, 1, ipcPrmLen);
sl@0
  1233
	RDesReadStream strm(ipcPrmDes);
sl@0
  1234
	
sl@0
  1235
	TBuf8<KMaxFileName + 1> tblName;
sl@0
  1236
	ExtractNameL(strm, tblName);
sl@0
  1237
	
sl@0
  1238
	TBuf8<KMaxFileName + 1> colName;
sl@0
  1239
	ExtractNameL(strm, colName);
sl@0
  1240
	
sl@0
  1241
	TInt64 rowId;
sl@0
  1242
	strm >> rowId;
sl@0
  1243
	if(rowId == -1)
sl@0
  1244
		{
sl@0
  1245
		rowId = iDatabase->LastInsertedRowId();
sl@0
  1246
		}
sl@0
  1247
	if(rowId <= 0)
sl@0
  1248
		{
sl@0
  1249
		__SQLLEAVE(KErrArgument);
sl@0
  1250
		}
sl@0
  1251
sl@0
  1252
	TInt32 tmp;	
sl@0
  1253
	strm >> tmp;
sl@0
  1254
	TBool isReadOnly = tmp != 0;
sl@0
  1255
	
sl@0
  1256
	TBuf8<KMaxFileName + 1> dbName;
sl@0
  1257
	ExtractNameL(strm, dbName, KMainDb16);
sl@0
  1258
						
sl@0
  1259
	strm.Close();
sl@0
  1260
			 		
sl@0
  1261
	// If the database is secure then check that the client has the required capabilities for the operation
sl@0
  1262
	TInt dbOpType = isReadOnly ? SQLITE_READ : SQLITE_UPDATE;
sl@0
  1263
	if(CSqlSrvDatabase::AuthorizeCallback(iDatabase, dbOpType, (char*)tblName.Ptr(), (char*)colName.Ptr(), (char*)dbName.Ptr(), '\0') != SQLITE_OK)	
sl@0
  1264
		{
sl@0
  1265
		__SQLLEAVE(KErrPermissionDenied);	
sl@0
  1266
		}
sl@0
  1267
sl@0
  1268
	// Create the stream buffer
sl@0
  1269
	HBlobBuf* blobBuf = HBlobBuf::NewL(iDatabase->RawDbHandle(), dbName, tblName, colName, rowId, isReadOnly ? HBlobBuf::EReadOnly : HBlobBuf::EReadWrite);
sl@0
  1270
	blobBuf->PushL();
sl@0
  1271
		
sl@0
  1272
	// Return the blob size to the client
sl@0
  1273
	TPckgBuf<TIpcStreamBuf> ipcBuf;
sl@0
  1274
	TInt size = blobBuf->SizeL();
sl@0
  1275
	ipcBuf().iExt = size;
sl@0
  1276
	
sl@0
  1277
	// If this is a read stream then return the first client buffer-full of data
sl@0
  1278
	TInt len = 0;
sl@0
  1279
	if(isReadOnly && (size > 0))
sl@0
  1280
		{
sl@0
  1281
		len = Min(size, KIpcBufSize);
sl@0
  1282
		blobBuf->ReadL(ipcBuf().iData, len);	
sl@0
  1283
		}
sl@0
  1284
	
sl@0
  1285
	// Create the stream object
sl@0
  1286
	HIpcStream* ipcStream = new (ELeave) HIpcStream(blobBuf, len);
sl@0
  1287
	TInt strmHandle = iIpcStreams.Add(ipcStream);
sl@0
  1288
	__SQLPANIC_CLIENT(strmHandle > 0, aMessage, ESqlPanicBadHandle);
sl@0
  1289
	CleanupStack::Pop(blobBuf);
sl@0
  1290
sl@0
  1291
	// Send the size and data to the client
sl@0
  1292
	aMessage.WriteL(2, ipcBuf);
sl@0
  1293
	SQLPROFILER_REPORT_IPC(ESqlIpcWrite, size);
sl@0
  1294
sl@0
  1295
	return strmHandle;
sl@0
  1296
	}
sl@0
  1297
	
sl@0
  1298
///////////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
  1299
////////////////////////////          Statement operations           //////////////////////////////////
sl@0
  1300
///////////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
  1301
sl@0
  1302
/**
sl@0
  1303
Processes the request for preparing a 8/16-bit SQL statement.
sl@0
  1304
sl@0
  1305
@panic SqlDb 2 Client panic. iDatabase is NULL (the database object is not created yet).
sl@0
  1306
@panic SqlDb 3 Client panic. Internal error - invalid statement handle.
sl@0
  1307
sl@0
  1308
Usage of the IPC call arguments:
sl@0
  1309
Arg 0: [out]	Column count and parameter count
sl@0
  1310
Arg 1: [in]		8/16-bit character length of SQL statement
sl@0
  1311
Arg 2: [in]		SQL statement
sl@0
  1312
*/
sl@0
  1313
TInt CSqlSrvSession::StmtPrepareL(const RMessage2& aMessage, TBool aIsText16)
sl@0
  1314
	{
sl@0
  1315
	__SQLPANIC_CLIENT(iDatabase != NULL, aMessage, ESqlPanicInvalidObj);
sl@0
  1316
	iStatements.AllocL();
sl@0
  1317
	TInt columnCount = -1;
sl@0
  1318
	TInt paramCount = -1;
sl@0
  1319
	TUint len = static_cast <TUint> (aMessage.Int1());
sl@0
  1320
	CSqlSrvStatement*  stmt = aIsText16 ? 
sl@0
  1321
						CSqlSrvStatement::NewLC(iDatabase->RawDbHandle(), ReadString16ZL(aMessage, 2, len), columnCount, paramCount) :
sl@0
  1322
						CSqlSrvStatement::NewLC(iDatabase->RawDbHandle(), ReadString8ZL(aMessage, 2, len), columnCount, paramCount);
sl@0
  1323
	TPckgBuf<TSqlIpcData> data;
sl@0
  1324
	data().iPrm1 = static_cast <TUint32> (columnCount);
sl@0
  1325
	data().iPrm2 = static_cast <TUint32> (paramCount);
sl@0
  1326
	aMessage.WriteL(0, data);
sl@0
  1327
	SQLPROFILER_REPORT_IPC(ESqlIpcWrite, sizeof(TSqlIpcData));
sl@0
  1328
	TInt stmtHandle = iStatements.Add(stmt);
sl@0
  1329
	__SQLPANIC_CLIENT(stmtHandle > 0, aMessage, ESqlPanicBadHandle);
sl@0
  1330
	CleanupStack::Pop(stmt);
sl@0
  1331
	return stmtHandle;
sl@0
  1332
	}
sl@0
  1333
sl@0
  1334
/**
sl@0
  1335
Processes the request for executing the SQL statement.
sl@0
  1336
sl@0
  1337
@param aFunction ESqlSrvStmtExec, ESqlSrvStmtAsyncExec, ESqlSrvStmtBindExec, ESqlSrvStmtBindExecRowId, ESqlSrvStmtAsyncBindExec
sl@0
  1338
sl@0
  1339
Usage of the IPC call arguments:
sl@0
  1340
Arg 0: [in]    parameter buffer length in bytes (if aFunction == ESqlSrvStmtBindExec || aFunction == ESqlSrvStmtAsyncBindExec)
sl@0
  1341
Arg 1: [in]    parameter buffer 		(if aFunction == ESqlSrvStmtBindExec || aFunction == ESqlSrvStmtAsyncBindExec)
sl@0
  1342
*/
sl@0
  1343
TInt CSqlSrvSession::StmtExecL(const RMessage2& aMessage, TInt aStmtHandle, TSqlSrvFunction aFunction)
sl@0
  1344
	{	
sl@0
  1345
	CSqlSrvStatement& stmt = ::SqlSessObjFind(iStatements, aStmtHandle, aMessage);
sl@0
  1346
	if(aFunction == ESqlSrvStmtBindExec || aFunction == ESqlSrvStmtAsyncBindExec)
sl@0
  1347
		{
sl@0
  1348
		DoStmtBindL(aMessage, stmt);
sl@0
  1349
		}
sl@0
  1350
	__SQLLEAVE_IF_ERROR(stmt.Exec());
sl@0
  1351
	return iDatabase->LastChangesCount();
sl@0
  1352
	}
sl@0
  1353
sl@0
  1354
/**
sl@0
  1355
Processes the request for moving the SQL statement on the next record.
sl@0
  1356
sl@0
  1357
If the call does not fail, the only valid acceptable return codes should be KSqlAtRow and KSqlAtEnd.
sl@0
  1358
sl@0
  1359
@panic SqlDb 7 In _DEBUG mode. The call completed with no error but the return code is not KSqlAtRow or KSqlAtEnd.
sl@0
  1360
sl@0
  1361
Usage of the IPC call arguments:
sl@0
  1362
Arg 0: [in]    parameter buffer length in bytes (if aFunction == ESqlSrvStmtBindNext)
sl@0
  1363
Arg 1: [in]    parameter buffer 		(if aFunction == ESqlSrvStmtBindNext)
sl@0
  1364
Arg 2: [in]    client side column buffer length in bytes
sl@0
  1365
Arg 3: [out]   column buffer
sl@0
  1366
*/
sl@0
  1367
TInt CSqlSrvSession::StmtNextL(const RMessage2& aMessage, TInt aStmtHandle, TSqlSrvFunction aFunction)
sl@0
  1368
	{
sl@0
  1369
	CSqlSrvStatement& stmt = ::SqlSessObjFind(iStatements, aStmtHandle, aMessage);
sl@0
  1370
	if(aFunction == ESqlSrvStmtBindNext)
sl@0
  1371
		{
sl@0
  1372
		DoStmtBindL(aMessage, stmt);
sl@0
  1373
		}
sl@0
  1374
	TInt err = stmt.Next();
sl@0
  1375
	if(err == KSqlAtRow)
sl@0
  1376
		{
sl@0
  1377
		const RSqlBufFlat& bufFlat = stmt.ColumnValuesL();
sl@0
  1378
		TInt size = bufFlat.Size();
sl@0
  1379
		if(size > aMessage.Int2())
sl@0
  1380
			{
sl@0
  1381
			return size + KSqlClientBufOverflowCode;
sl@0
  1382
			}
sl@0
  1383
		aMessage.WriteL(3, bufFlat.BufDes());
sl@0
  1384
		SQLPROFILER_REPORT_IPC(ESqlIpcWrite, size);
sl@0
  1385
		}
sl@0
  1386
	__SQLLEAVE_IF_ERROR(err);
sl@0
  1387
	__ASSERT_DEBUG(err == KSqlAtRow || err == KSqlAtEnd, __SQLPANIC(ESqlPanicInternalError));
sl@0
  1388
	return err;
sl@0
  1389
	}
sl@0
  1390
sl@0
  1391
/**
sl@0
  1392
Processes the request for retrieving the statement column or parameter names.
sl@0
  1393
sl@0
  1394
If the client side buffer size is not big enough, the function returns the size + KSqlClientBufOverflowCode.
sl@0
  1395
In this case the client must increase the buffer and try again to get the buffer only.
sl@0
  1396
sl@0
  1397
Usage of the IPC call arguments:
sl@0
  1398
Arg 0: [in]		size of the client side buffer for the names (in bytes)
sl@0
  1399
Arg 1: [out]	ipc buffer, column or parameter names
sl@0
  1400
*/
sl@0
  1401
TInt CSqlSrvSession::StmtNamesL(const RMessage2& aMessage, TInt aStmtHandle, TSqlSrvFunction aFunction)
sl@0
  1402
	{
sl@0
  1403
	CSqlSrvStatement& stmt = ::SqlSessObjFind(iStatements, aStmtHandle, aMessage);
sl@0
  1404
	const RSqlBufFlat& namesBuf = aFunction == ESqlSrvStmtParamNames ? stmt.ParamNamesL() : stmt.ColumnNamesL();
sl@0
  1405
	TInt size = namesBuf.Size();
sl@0
  1406
	if(size <= aMessage.Int0())
sl@0
  1407
		{
sl@0
  1408
		aMessage.WriteL(1, namesBuf.BufDes());
sl@0
  1409
		SQLPROFILER_REPORT_IPC(ESqlIpcWrite, size);
sl@0
  1410
		return 0;
sl@0
  1411
		}
sl@0
  1412
	return size + KSqlClientBufOverflowCode;
sl@0
  1413
	}
sl@0
  1414
sl@0
  1415
/**
sl@0
  1416
Processes the request for accessing a large column value as a stream of bytes/characters.
sl@0
  1417
sl@0
  1418
Usage of the IPC call arguments:
sl@0
  1419
Arg 0: [in]		column index (0 based)
sl@0
  1420
Arg 2: [out]	ipc buffer, column source
sl@0
  1421
*/
sl@0
  1422
TInt CSqlSrvSession::StmtColumnSourceL(const RMessage2& aMessage, TInt aStmtHandle)
sl@0
  1423
	{
sl@0
  1424
	CSqlSrvStatement& stmt = ::SqlSessObjFind(iStatements, aStmtHandle, aMessage);
sl@0
  1425
	TInt columnIndex = aMessage.Int0();
sl@0
  1426
	TPtrC8 columnSource;
sl@0
  1427
	TInt err = stmt.ColumnSource(columnIndex, columnSource);
sl@0
  1428
	__SQLLEAVE_IF_ERROR(err);
sl@0
  1429
	HIpcReadBuf* ipcBuf = HIpcReadBuf::NewL(columnSource);
sl@0
  1430
	return NewOutputStreamL(aMessage, ipcBuf);
sl@0
  1431
	}
sl@0
  1432
sl@0
  1433
/**
sl@0
  1434
Processes the request for setting a large parameter value from a stream of bytes or 8/16-bit characters.
sl@0
  1435
sl@0
  1436
@panic SqlDb 3 Client panic. Internal error - invalid stream handle.
sl@0
  1437
sl@0
  1438
Usage of the IPC call arguments:
sl@0
  1439
Arg 0: [in]		parameter index (0 based)
sl@0
  1440
Arg 2: [out]	ipc buffer, parameter source
sl@0
  1441
*/
sl@0
  1442
TInt CSqlSrvSession::StmtParamSinkL(const RMessage2& aMessage, TInt aStmtHandle, TSqlSrvFunction aFunction)
sl@0
  1443
	{
sl@0
  1444
	iIpcStreams.AllocL();
sl@0
  1445
	TInt parameterIndex = aMessage.Int0();
sl@0
  1446
	CSqlSrvStatement& stmt = ::SqlSessObjFind(iStatements, aStmtHandle, aMessage);
sl@0
  1447
	HSqlSrvStmtParamBuf::TDataType dataType = aFunction == ESqlSrvStmtBinParamSink ? HSqlSrvStmtParamBuf::EBinary : HSqlSrvStmtParamBuf::EText16;
sl@0
  1448
	HSqlSrvStmtParamBuf* paramBuf = stmt.GetParamBufL(parameterIndex, dataType, HSqlSrvStmtParamBuf::EBufIpcStream);
sl@0
  1449
	HIpcStream* ipcStream = new (ELeave) HIpcStream(paramBuf, 0);
sl@0
  1450
	TInt strmHandle = iIpcStreams.Add(ipcStream);
sl@0
  1451
	__SQLPANIC_CLIENT(strmHandle > 0, aMessage, ESqlPanicBadHandle);
sl@0
  1452
	return strmHandle;
sl@0
  1453
	}
sl@0
  1454
sl@0
  1455
/**
sl@0
  1456
Usage of the IPC call arguments:
sl@0
  1457
Arg 0: [in]		the client side buffer length in bytes
sl@0
  1458
Arg 1: [out]	refers to a place where the buffer data will be copied to.
sl@0
  1459
*/
sl@0
  1460
void CSqlSrvSession::StmtGetBufFlatL(const RMessage2& aMessage, TInt aStmtHandle)
sl@0
  1461
	{
sl@0
  1462
	CSqlSrvStatement& stmt = ::SqlSessObjFind(iStatements, aStmtHandle, aMessage);
sl@0
  1463
	const RSqlBufFlat& bufFlat = stmt.BufFlatL(static_cast <TSqlBufFlatType> (aMessage.Int0()));
sl@0
  1464
	aMessage.WriteL(1, bufFlat.BufDes());
sl@0
  1465
	SQLPROFILER_REPORT_IPC(ESqlIpcWrite, bufFlat.Size());
sl@0
  1466
	}
sl@0
  1467
sl@0
  1468
/**
sl@0
  1469
Usage of the IPC call arguments:
sl@0
  1470
Arg 0: [in]		column index
sl@0
  1471
Arg 1: [in]		column buffer length in bytes
sl@0
  1472
Arg 2: [in/out]	column buffer
sl@0
  1473
*/
sl@0
  1474
void CSqlSrvSession::StmtColumnValueL(const RMessage2& aMessage, TInt aStmtHandle)
sl@0
  1475
	{
sl@0
  1476
	CSqlSrvStatement& stmt = ::SqlSessObjFind(iStatements, aStmtHandle, aMessage);
sl@0
  1477
	TPtrC8 columnSource;
sl@0
  1478
	TInt columnIndex = aMessage.Int0();
sl@0
  1479
	TInt err = stmt.ColumnSource(columnIndex, columnSource);
sl@0
  1480
	__SQLLEAVE_IF_ERROR(err);
sl@0
  1481
	TInt len = aMessage.Int1();
sl@0
  1482
	if(columnSource.Length() > len)
sl@0
  1483
		{
sl@0
  1484
		TPtr8 ptr(const_cast <TUint8*> (columnSource.Ptr()), columnSource.Length(), columnSource.Length());
sl@0
  1485
		ptr.SetLength(len);
sl@0
  1486
		aMessage.WriteL(2, ptr);
sl@0
  1487
		SQLPROFILER_REPORT_IPC(ESqlIpcWrite, len);
sl@0
  1488
		__SQLLEAVE(KErrOverflow);	
sl@0
  1489
		}
sl@0
  1490
	else
sl@0
  1491
		{
sl@0
  1492
		aMessage.WriteL(2, columnSource);
sl@0
  1493
		SQLPROFILER_REPORT_IPC(ESqlIpcWrite, len);
sl@0
  1494
		}
sl@0
  1495
	}
sl@0
  1496
sl@0
  1497
/**
sl@0
  1498
Usage of the IPC call arguments:
sl@0
  1499
Arg 0: [in]    parameter buffer length in bytes
sl@0
  1500
Arg 1: [in]    parameter buffer
sl@0
  1501
*/
sl@0
  1502
void CSqlSrvSession::DoStmtBindL(const RMessage2& aMessage, CSqlSrvStatement& aStmt)
sl@0
  1503
	{
sl@0
  1504
	TInt prmLen = aMessage.Int0();
sl@0
  1505
	RSqlBufFlat& prmBuf = Server().GetFlatBufL(prmLen);
sl@0
  1506
	aMessage.ReadL(1, prmBuf.BufPtr());
sl@0
  1507
	SQLPROFILER_REPORT_IPC(ESqlIpcRead, prmLen);
sl@0
  1508
	aStmt.BindL(prmBuf);
sl@0
  1509
	}
sl@0
  1510
sl@0
  1511
/**
sl@0
  1512
Processes the request for retrieving the statement declared column type names.
sl@0
  1513
sl@0
  1514
If the client side buffer size is not big enough, the function returns the size + KSqlClientBufOverflowCode.
sl@0
  1515
In this case the client must increase the buffer and try again to get the buffer only
sl@0
  1516
sl@0
  1517
Usage of the IPC call arguments:
sl@0
  1518
Arg 0: [in]    	input buffer max length in 16-bit characters
sl@0
  1519
Arg 1: [out]	ipc buffer, declared column type names
sl@0
  1520
*/
sl@0
  1521
TInt CSqlSrvSession::StmtDeclColumnTypesL(const RMessage2& aMessage, TInt aStmtHandle)
sl@0
  1522
	{	
sl@0
  1523
	CSqlSrvStatement& stmt = ::SqlSessObjFind(iStatements, aStmtHandle, aMessage);
sl@0
  1524
	const RSqlBufFlat& declColumnTypesBuf = stmt.GetDeclColumnTypesL();
sl@0
  1525
	TInt size = declColumnTypesBuf.Size();
sl@0
  1526
	if(size <= aMessage.Int0())
sl@0
  1527
		{
sl@0
  1528
		aMessage.WriteL(1, declColumnTypesBuf.BufDes());
sl@0
  1529
		return 0;
sl@0
  1530
		}
sl@0
  1531
	return size + KSqlClientBufOverflowCode;
sl@0
  1532
	}
sl@0
  1533
sl@0
  1534
sl@0
  1535
//////////////////////////////////////////////////////////////////////////////////////////////////////////	
sl@0
  1536
////////////////////////              Helper methods                     /////////////////////////////////
sl@0
  1537
//////////////////////////////////////////////////////////////////////////////////////////////////////////	
sl@0
  1538
sl@0
  1539
/**
sl@0
  1540
Creates a new output IPC stream object using the aStreamBuf parameter as a stream buffer (stream data source).
sl@0
  1541
sl@0
  1542
This method immediately pushes aStreamBuf onto the cleanup stack before creating a new output IPC 
sl@0
  1543
stream and so callers of this method should ensure that aStreamBuf is not already on the cleanup stack.
sl@0
  1544
sl@0
  1545
Returns the handle of the created stream.
sl@0
  1546
sl@0
  1547
@panic SqlDb 3 Client panic. Internal error - invalid stream handle.
sl@0
  1548
sl@0
  1549
Usage of the IPC call arguments:
sl@0
  1550
Arg 2: [in/out]   IPC buffer
sl@0
  1551
*/
sl@0
  1552
TInt CSqlSrvSession::NewOutputStreamL(const RMessage2& aMessage, MStreamBuf* aStreamBuf)
sl@0
  1553
	{
sl@0
  1554
	aStreamBuf->PushL();
sl@0
  1555
	iIpcStreams.AllocL();
sl@0
  1556
	TInt size = aStreamBuf->SizeL();
sl@0
  1557
	__ASSERT_DEBUG(size >= 0, __SQLPANIC(ESqlPanicInternalError));
sl@0
  1558
	TPckgBuf<TIpcStreamBuf> ipcBuf;
sl@0
  1559
    // read the first buffer-full
sl@0
  1560
    TInt len = Min(size, KIpcBufSize);
sl@0
  1561
    aStreamBuf->ReadL(ipcBuf().iData, len);
sl@0
  1562
	TInt handle = 0;
sl@0
  1563
	if(size > KIpcBufSize)
sl@0
  1564
		{								// create the stream object
sl@0
  1565
		HIpcStream* ipcStream = new (ELeave) HIpcStream(aStreamBuf, KIpcBufSize);
sl@0
  1566
		handle = iIpcStreams.Add(ipcStream);
sl@0
  1567
		__SQLPANIC_CLIENT(handle > 0, aMessage, ESqlPanicBadHandle);
sl@0
  1568
		CleanupStack::Pop(aStreamBuf);
sl@0
  1569
		}
sl@0
  1570
	else								// no more data to send
sl@0
  1571
		{
sl@0
  1572
		CleanupStack::PopAndDestroy(aStreamBuf);
sl@0
  1573
		}
sl@0
  1574
    ipcBuf().iExt = size;
sl@0
  1575
    aMessage.WriteL(2, ipcBuf);
sl@0
  1576
    SQLPROFILER_REPORT_IPC(ESqlIpcWrite, size);
sl@0
  1577
	return handle;
sl@0
  1578
	}
sl@0
  1579
sl@0
  1580
/**
sl@0
  1581
Reads a 8-bit string with "aByteLen" bytes length, which is in "aArgNum" argument of aMessage.
sl@0
  1582
The string will be zero terminated after the "read" operation.
sl@0
  1583
Returns TDes8 reference pointing to the zero-terminated string.
sl@0
  1584
sl@0
  1585
@panic SqlDb 3 Client panic. The string length is not equal to aByteLen. If happens then it is an indication of a 
sl@0
  1586
                             problem inside client side sql library.
sl@0
  1587
@panic SqlDb 4 Client panic. Negative aByteLen value.
sl@0
  1588
*/
sl@0
  1589
TDes8& CSqlSrvSession::ReadString8ZL(const RMessage2& aMessage, TInt aArgNum, TInt aByteLen)
sl@0
  1590
	{
sl@0
  1591
	__SQLPANIC_CLIENT(aByteLen >= 0, aMessage, ESqlPanicBadArgument);
sl@0
  1592
	TDes8& buf = Server().GetBuf8L(aByteLen + 1);
sl@0
  1593
	aMessage.ReadL(aArgNum, buf);
sl@0
  1594
	SQLPROFILER_REPORT_IPC(ESqlIpcRead, aByteLen);
sl@0
  1595
    __SQLPANIC_CLIENT(buf.Length() == aByteLen, aMessage, ESqlPanicBadHandle);
sl@0
  1596
	buf.Append(TChar(0));
sl@0
  1597
	return buf;
sl@0
  1598
	}
sl@0
  1599
sl@0
  1600
/**
sl@0
  1601
Reads a 16-bit string with "aCharLen" character length, which is in "aArgNum" argument of aMessage.
sl@0
  1602
The string will be zero terminated after the "read" operation.
sl@0
  1603
Returns TDes16 reference pointing to the zero-terminated string.
sl@0
  1604
sl@0
  1605
@panic SqlDb 3 Client panic. The string length is not equal to aCharLen. If happens then it is an indication of a 
sl@0
  1606
                             problem inside client side sql library.
sl@0
  1607
@panic SqlDb 4 Client panic. Negative aCharLen value.
sl@0
  1608
*/
sl@0
  1609
TDes16& CSqlSrvSession::ReadString16ZL(const RMessage2& aMessage, TInt aArgNum, TInt aCharLen)
sl@0
  1610
	{
sl@0
  1611
	__SQLPANIC_CLIENT(aCharLen >= 0, aMessage, ESqlPanicBadArgument);
sl@0
  1612
	TDes16& buf = Server().GetBuf16L(aCharLen + 1);
sl@0
  1613
	aMessage.ReadL(aArgNum, buf);
sl@0
  1614
	SQLPROFILER_REPORT_IPC(ESqlIpcRead, (aCharLen * sizeof(TText)));
sl@0
  1615
    __SQLPANIC_CLIENT(buf.Length() == aCharLen, aMessage, ESqlPanicBadHandle);
sl@0
  1616
	buf.Append(TChar(0));
sl@0
  1617
	return buf;
sl@0
  1618
	}
sl@0
  1619
sl@0
  1620
/**
sl@0
  1621
Reads a 16-bit string with "aCharLen" character length, which is in "aArgNum" argument of aMessage.
sl@0
  1622
Returns TDes16 reference pointing to the string.
sl@0
  1623
sl@0
  1624
@panic SqlDb 3 Client panic. The string length is not equal to aCharLen. If happens then it is an indication of a 
sl@0
  1625
                             problem inside client side sql library.
sl@0
  1626
@panic SqlDb 4 Client panic. Negative aCharLen value.
sl@0
  1627
*/
sl@0
  1628
TDes16& CSqlSrvSession::ReadString16L(const RMessage2& aMessage, TInt aArgNum, TInt aCharLen)
sl@0
  1629
	{
sl@0
  1630
	__SQLPANIC_CLIENT(aCharLen >= 0, aMessage, ESqlPanicBadArgument);
sl@0
  1631
	TDes16& buf = Server().GetBuf16L(aCharLen);
sl@0
  1632
	aMessage.ReadL(aArgNum, buf);
sl@0
  1633
	SQLPROFILER_REPORT_IPC(ESqlIpcRead, (aCharLen * sizeof(TText)));
sl@0
  1634
    __SQLPANIC_CLIENT(buf.Length() == aCharLen, aMessage, ESqlPanicBadHandle);
sl@0
  1635
	return buf;
sl@0
  1636
	}
sl@0
  1637
sl@0
  1638
/**
sl@0
  1639
The method constructs a CSqlSecurityPolicy object from the passed as an argument descriptor.
sl@0
  1640
sl@0
  1641
@param aSecurityPolicyData A descriptor with the security policy data.
sl@0
  1642
sl@0
  1643
@return A pointer to the created CSqlSecurityPolicy instance.
sl@0
  1644
sl@0
  1645
@leave KErrNoMemory, out of memory condition has occured.
sl@0
  1646
*/
sl@0
  1647
CSqlSecurityPolicy* CSqlSrvSession::CreateSecurityPolicyL(const TDesC8& aSecurityPolicyData)
sl@0
  1648
	{
sl@0
  1649
	TSecurityPolicy defaultPolicy(TSecurityPolicy::EAlwaysFail);
sl@0
  1650
	CSqlSecurityPolicy* dbPolicy = CSqlSecurityPolicy::NewLC(defaultPolicy);
sl@0
  1651
	RSqlBufFlat& bufFlat = dbPolicy->BufFlat();
sl@0
  1652
    __SQLLEAVE_IF_ERROR(bufFlat.ReAlloc(aSecurityPolicyData.Length()));
sl@0
  1653
	bufFlat.BufPtr().Copy(aSecurityPolicyData);
sl@0
  1654
	CleanupStack::Pop(dbPolicy);
sl@0
  1655
	return dbPolicy;
sl@0
  1656
	}
sl@0
  1657
sl@0
  1658
/**
sl@0
  1659
Reports how many objects are allocated by the client.
sl@0
  1660
If the database connection is not in a test mode, the allocated memory cells count will be ignored.
sl@0
  1661
*/
sl@0
  1662
TInt CSqlSrvSession::CountResources()
sl@0
  1663
	{
sl@0
  1664
	return iStatements.Count() + iIpcStreams.Count() + (iDbResourceTestMode ? User::CountAllocCells() : 0);
sl@0
  1665
	}
sl@0
  1666
sl@0
  1667
/**
sl@0
  1668
Extracts from aMessage:
sl@0
  1669
- function code;
sl@0
  1670
- stream or statement handle;
sl@0
  1671
The function will panic the client if aMessage contains bad function code or bad handle encoded in it.
sl@0
  1672
*/
sl@0
  1673
void CSqlSrvSession::Extract(const RMessage2& aMessage, TSqlSrvFunction& aFunction, TInt& aHandle)
sl@0
  1674
	{
sl@0
  1675
	TInt msgCode = aMessage.Function();
sl@0
  1676
	aFunction = static_cast <TSqlSrvFunction> (KSqlSrvFunctionMask & msgCode);
sl@0
  1677
	//All operations with code > ESqlSrvDbDelete require valid iDatabase object.
sl@0
  1678
	if(aFunction > ESqlSrvDbDelete)
sl@0
  1679
		{
sl@0
  1680
		__SQLPANIC_CLIENT(iDatabase != NULL, aMessage, ESqlPanicInvalidObj);
sl@0
  1681
		}
sl@0
  1682
	if(aFunction >= ESqlSrvStmtClose)
sl@0
  1683
		{
sl@0
  1684
		//Extracting handle and handle type from the message code
sl@0
  1685
		TSqlSrvHandleType handleType = static_cast <TSqlSrvHandleType> (msgCode & KSqlSrvHandleTypeMask);
sl@0
  1686
		aHandle = (msgCode & KSqlSrvHandleMask) >> KSqlSrvHandleShiftBits;
sl@0
  1687
		__SQLPANIC_CLIENT(aHandle > 0, aMessage, ESqlPanicBadArgument);
sl@0
  1688
		if(aFunction < ESqlSrvStreamBase)
sl@0
  1689
			{
sl@0
  1690
			__SQLPANIC_CLIENT(handleType == ESqlSrvStatementHandle, aMessage, ESqlPanicBadArgument);
sl@0
  1691
			}
sl@0
  1692
		else
sl@0
  1693
			{
sl@0
  1694
			__SQLPANIC_CLIENT(handleType == ESqlSrvStreamHandle, aMessage, ESqlPanicBadArgument);
sl@0
  1695
			}
sl@0
  1696
		}
sl@0
  1697
	}
sl@0
  1698
sl@0
  1699
/**
sl@0
  1700
The function reads aStmt column 0 value and copies it into the client buffer, accessed via aMessage argument.
sl@0
  1701
sl@0
  1702
If an error occurs during the execution the function leaves with the error code.
sl@0
  1703
Possible non-leaving return values:
sl@0
  1704
 - KErrNone              - the function has completed successfully;
sl@0
  1705
 - Positive return value - the length of the column, which means - the destination buffer is too small.
sl@0
  1706
                           This return value is possible only with text or binary columns.
sl@0
  1707
sl@0
  1708
Usage of the IPC call arguments: 
sl@0
  1709
Arg 0: [in]		(8/16-bit character length of SQL statement) | (expected column value type << 24).
sl@0
  1710
Arg 1: [in]		SQL statement.
sl@0
  1711
Arg 2: [in]		Byte max length of the receiving buffer
sl@0
  1712
Arg 3: [in/out]	The receiving buffer
sl@0
  1713
*/
sl@0
  1714
TInt CSqlSrvSession::GetColumnValueL(const RMessage2& aMessage, CSqlSrvStatement& aStmt, TSqlColumnType aColType)
sl@0
  1715
	{
sl@0
  1716
	TInt rc = KErrNone;
sl@0
  1717
	switch(aColType)
sl@0
  1718
		{
sl@0
  1719
		case ESqlInt:
sl@0
  1720
			{
sl@0
  1721
			TInt val = aStmt.ColumnInt(0);
sl@0
  1722
			aMessage.WriteL(3, TPtrC8(reinterpret_cast <const TUint8*> (&val), sizeof(val)));
sl@0
  1723
			SQLPROFILER_REPORT_IPC(ESqlIpcWrite, sizeof(val));
sl@0
  1724
			}
sl@0
  1725
			break;
sl@0
  1726
		case ESqlInt64:
sl@0
  1727
			{
sl@0
  1728
			TInt64 val = aStmt.ColumnInt64(0);
sl@0
  1729
			aMessage.WriteL(3, TPtrC8(reinterpret_cast <const TUint8*> (&val), sizeof(val)));
sl@0
  1730
			SQLPROFILER_REPORT_IPC(ESqlIpcWrite, sizeof(val));
sl@0
  1731
			}
sl@0
  1732
			break;
sl@0
  1733
		case ESqlReal:
sl@0
  1734
			{
sl@0
  1735
			TReal val = aStmt.ColumnReal(0);
sl@0
  1736
			aMessage.WriteL(3, TPtrC8(reinterpret_cast <const TUint8*> (&val), sizeof(val)));
sl@0
  1737
			SQLPROFILER_REPORT_IPC(ESqlIpcWrite, sizeof(val));
sl@0
  1738
			}
sl@0
  1739
			break;
sl@0
  1740
		case ESqlText:
sl@0
  1741
		case ESqlBinary:
sl@0
  1742
		default:
sl@0
  1743
			{
sl@0
  1744
			TPtrC8 val;
sl@0
  1745
			if(aColType == ESqlText)
sl@0
  1746
				{
sl@0
  1747
				TPtrC textVal = aStmt.ColumnTextL(0);
sl@0
  1748
				val.Set(reinterpret_cast <const TUint8*> (textVal.Ptr()), textVal.Length() * sizeof(TUint16));
sl@0
  1749
				}
sl@0
  1750
			else
sl@0
  1751
				{
sl@0
  1752
				val.Set(aStmt.ColumnBinary(0));
sl@0
  1753
				}
sl@0
  1754
			TInt len = val.Length();
sl@0
  1755
			if(len > aMessage.Int2())
sl@0
  1756
				{
sl@0
  1757
				rc = aColType == ESqlText ? (TUint)len / sizeof(TUint16) : len;
sl@0
  1758
				len = aMessage.Int2();
sl@0
  1759
				}
sl@0
  1760
			aMessage.WriteL(3, val.Left(len));
sl@0
  1761
			SQLPROFILER_REPORT_IPC(ESqlIpcWrite, len);
sl@0
  1762
			}
sl@0
  1763
			break;
sl@0
  1764
		}
sl@0
  1765
	return rc;
sl@0
  1766
	}