os/persistentdata/persistentstorage/sql/SRC/Client/SqlDbSession.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
//
sl@0
    13
// Description:
sl@0
    14
//
sl@0
    15
sl@0
    16
#include <e32math.h>
sl@0
    17
#include <f32file.h>
sl@0
    18
#include <f32file64.h>
sl@0
    19
#include <s32strm.h>
sl@0
    20
#include <s32mem.h>
sl@0
    21
#include "IPCBuf.h"				//HIpcBuf
sl@0
    22
#include "SqlDbSession.h"		//RSqlDbSession
sl@0
    23
#include "SqlSrvStartup.h"		//StartSqlServer()
sl@0
    24
#include "SqlResourceTest.h"	//TSqlResourceTestData
sl@0
    25
#include "SqlSecurityImpl.h"	//CSqlSecurityPolicy
sl@0
    26
#include "OstTraceDefinitions.h"
sl@0
    27
#ifdef OST_TRACE_COMPILER_IN_USE
sl@0
    28
#include "SqlDbSessionTraces.h"
sl@0
    29
#endif
sl@0
    30
#include "SqlTraceDef.h"
sl@0
    31
sl@0
    32
///////////////////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
    33
//////////////////////             TSqlFhCmdFunctor             ///////////////////////////////////////////////
sl@0
    34
///////////////////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
    35
sl@0
    36
/**
sl@0
    37
TSqlFhCmdFunctor derived classes are used by RSqlDbSession for sending to SQL server localy created/opened 
sl@0
    38
file handles.
sl@0
    39
sl@0
    40
@see RSqlDbSession
sl@0
    41
@see RSqlDbSession::CreateAndSendFileHandle()
sl@0
    42
sl@0
    43
@internalComponent
sl@0
    44
*/
sl@0
    45
NONSHARABLE_CLASS(TSqlFhCmdFunctor)
sl@0
    46
	{
sl@0
    47
public: 
sl@0
    48
	TSqlFhCmdFunctor(RSqlDbSession& aDbSession, const TDesC& aDbFileName, const TDesC8* aConfig=NULL) :
sl@0
    49
		iDbSession(aDbSession),
sl@0
    50
		iDbFileName(aDbFileName),
sl@0
    51
		iConfig(aConfig) // ownership not transferred
sl@0
    52
		{
sl@0
    53
		}
sl@0
    54
	virtual TInt operator()(RFile64& aFile, TBool aCreated, TBool aReadOnly) = 0;
sl@0
    55
	
sl@0
    56
public:	
sl@0
    57
	RSqlDbSession&	iDbSession;
sl@0
    58
	const TDesC& 	iDbFileName;
sl@0
    59
	const TDesC8*	iConfig;
sl@0
    60
	};
sl@0
    61
sl@0
    62
///////////////////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
    63
//////////////////////              TSqlFhOpenCmdFunctor         //////////////////////////////////////////////
sl@0
    64
///////////////////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
    65
sl@0
    66
/**
sl@0
    67
Implements the sending of ESqlSrvDbOpenFromHandle command to the SQL server.
sl@0
    68
sl@0
    69
@internalComponent
sl@0
    70
*/
sl@0
    71
NONSHARABLE_CLASS(TSqlFhOpenCmdFunctor) : public TSqlFhCmdFunctor
sl@0
    72
	{
sl@0
    73
public:	
sl@0
    74
	TSqlFhOpenCmdFunctor(RSqlDbSession& aDbSession, const TDesC& aDbFileName,const TDesC8* aConfig=NULL) :
sl@0
    75
		TSqlFhCmdFunctor(aDbSession, aDbFileName, aConfig)
sl@0
    76
		{
sl@0
    77
		}
sl@0
    78
	virtual TInt operator()(RFile64& aDbFile, TBool aCreated, TBool aReadOnly);
sl@0
    79
	
sl@0
    80
	};
sl@0
    81
sl@0
    82
/**
sl@0
    83
Sends a command to the SQL server to use the supplied file session and file handles for adopting,
sl@0
    84
when opening/creating a database.
sl@0
    85
sl@0
    86
Usage of the IPC call arguments:
sl@0
    87
 - Arg 0: [out]  The 32 bits of the argument are used as follow:
sl@0
    88
 @code
sl@0
    89
 MSB                                                                                 LSB
sl@0
    90
 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
    91
 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
    92
 @endcode
sl@0
    93
 Where:
sl@0
    94
 @code
sl@0
    95
  - "Ro" - read-only flag, true if the file is read-only;
sl@0
    96
  - "Cr" - create/open flag, true if the file was created, false if the file was opened;
sl@0
    97
  - "C"  - config string length;
sl@0
    98
  - "F"  - database file name length;
sl@0
    99
 @endcode
sl@0
   100
 - Arg 1: [out]  database file name + configuration string (if there is a configuration string)
sl@0
   101
 - Arg 2: [out]  file session handle
sl@0
   102
 - Arg 3: [out]  database file handle
sl@0
   103
*/		
sl@0
   104
TInt TSqlFhOpenCmdFunctor::operator()(RFile64& aDbFile, TBool aCreated, TBool aReadOnly)
sl@0
   105
	{
sl@0
   106
	const TInt KConfigStrLen = iConfig ? iConfig->Length() : 0;
sl@0
   107
	if(KConfigStrLen > KSqlSrvMaxConfigStrLen)
sl@0
   108
		{
sl@0
   109
		return KErrArgument;
sl@0
   110
		}
sl@0
   111
	TIpcArgs ipcArgs;
sl@0
   112
	HBufC* arg1Buf = HBufC::New(iDbFileName.Length() + KConfigStrLen);
sl@0
   113
	if(!arg1Buf)
sl@0
   114
		{
sl@0
   115
		return KErrNoMemory;	
sl@0
   116
		}
sl@0
   117
	TPtr arg1 = arg1Buf->Des();
sl@0
   118
	arg1.Copy(iDbFileName);
sl@0
   119
	if(iConfig)
sl@0
   120
		{
sl@0
   121
		TBuf<KSqlSrvMaxConfigStrLen> cfgBuf;
sl@0
   122
		cfgBuf.Copy(*iConfig);
sl@0
   123
		arg1.Append(cfgBuf);
sl@0
   124
		}
sl@0
   125
	TUint32 arg0 = iDbFileName.Length() | (KConfigStrLen << 16);
sl@0
   126
	if(aReadOnly)
sl@0
   127
		{
sl@0
   128
		arg0 |= 0x80000000;	
sl@0
   129
		}
sl@0
   130
	if(aCreated)
sl@0
   131
		{
sl@0
   132
		arg0 |= 0x40000000;	
sl@0
   133
		}
sl@0
   134
	ipcArgs.Set(0, arg0);
sl@0
   135
	ipcArgs.Set(1, &arg1);
sl@0
   136
	TInt err = aDbFile.TransferToServer(ipcArgs, 2, 3);
sl@0
   137
	if(err == KErrNone)
sl@0
   138
		{
sl@0
   139
		err = iDbSession.SendReceive(ESqlSrvDbOpenFromHandle, ipcArgs);
sl@0
   140
		}
sl@0
   141
	delete arg1Buf;	
sl@0
   142
	return err;
sl@0
   143
	}
sl@0
   144
sl@0
   145
///////////////////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
   146
//////////////////////              TSqlFhAttachCmdFunctor         ////////////////////////////////////////////
sl@0
   147
///////////////////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
   148
sl@0
   149
/**
sl@0
   150
Implements the sending of ESqlSrvDbAttachFromHandle command to the SQL server.
sl@0
   151
sl@0
   152
@internalComponent
sl@0
   153
*/
sl@0
   154
NONSHARABLE_CLASS(TSqlFhAttachCmdFunctor) : public TSqlFhCmdFunctor
sl@0
   155
	{
sl@0
   156
public:	
sl@0
   157
	TSqlFhAttachCmdFunctor(RSqlDbSession& aDbSession, const TDesC& aDbFileName,
sl@0
   158
		const TDesC& aDbName) :
sl@0
   159
		TSqlFhCmdFunctor(aDbSession, aDbFileName),
sl@0
   160
		iDbName(aDbName)
sl@0
   161
		{
sl@0
   162
		}
sl@0
   163
	virtual TInt operator()(RFile64& aDbFile, TBool aCreated, TBool aReadOnly);
sl@0
   164
	
sl@0
   165
private:
sl@0
   166
	void SerializeToStreamL(RWriteStream& aStream);
sl@0
   167
		
sl@0
   168
private:
sl@0
   169
	const TDesC&	iDbName;
sl@0
   170
	
sl@0
   171
	};
sl@0
   172
	
sl@0
   173
/**
sl@0
   174
Sends a command to the SQL server to use the supplied file session and file handles for adopting,
sl@0
   175
when attaching a database.
sl@0
   176
sl@0
   177
Usage of the IPC call arguments:
sl@0
   178
 - Arg 0: [out]  The 32 bits of the argument are used as follow:
sl@0
   179
 @code
sl@0
   180
 MSB                                                                                 LSB
sl@0
   181
 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
   182
 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
   183
 @endcode
sl@0
   184
 Where:
sl@0
   185
 @code
sl@0
   186
  - "Ro" - read-only flag, true if the file is read-only;
sl@0
   187
  - "F"  - database file name length;
sl@0
   188
 @endcode
sl@0
   189
 - Arg 1: [out]  db names buffer
sl@0
   190
 - Arg 2: [out]  file session handle
sl@0
   191
 - Arg 3: [out]  database file handle
sl@0
   192
*/		
sl@0
   193
TInt TSqlFhAttachCmdFunctor::operator()(RFile64& aDbFile, TBool /*aCreated*/, TBool aReadOnly)
sl@0
   194
	{
sl@0
   195
	const TInt KMaxBufLen = iDbFileName.Length() * sizeof(TText) + iDbName.Length() * sizeof(TText) + 32;//"32" should be enough for all additional data like name length, alignment, etc.
sl@0
   196
	HBufC8* buf = HBufC8::New(KMaxBufLen);
sl@0
   197
	if(!buf)
sl@0
   198
		{
sl@0
   199
		return KErrNoMemory;	
sl@0
   200
		}
sl@0
   201
	TPtr8 bufPtr = buf->Des();
sl@0
   202
	RDesWriteStream out(bufPtr);
sl@0
   203
	TRAPD(err, SerializeToStreamL(out));
sl@0
   204
	__ASSERT_DEBUG(err == KErrNone, __SQLPANIC(ESqlPanicInternalError));//"Write to descriptor" streaming operatons can't fail
sl@0
   205
   TUint32 arg0 = (TUint32)bufPtr.Length() | (aReadOnly ? 0x80000000 : 0);
sl@0
   206
    TIpcArgs ipcArgs(arg0, &bufPtr);
sl@0
   207
    err = aDbFile.TransferToServer(ipcArgs, 2, 3);
sl@0
   208
    if(err == KErrNone)
sl@0
   209
        {
sl@0
   210
        err = iDbSession.SendReceive(ESqlSrvDbAttachFromHandle, ipcArgs);
sl@0
   211
        }
sl@0
   212
	delete buf;
sl@0
   213
	return err;
sl@0
   214
	}
sl@0
   215
	
sl@0
   216
/**
sl@0
   217
Serializes TSqlFhAttachCmdFunctor object content to a stream (aStream parameter).
sl@0
   218
*/
sl@0
   219
void TSqlFhAttachCmdFunctor::SerializeToStreamL(RWriteStream& aStream)
sl@0
   220
	{
sl@0
   221
	aStream << iDbFileName;
sl@0
   222
	aStream << iDbName;
sl@0
   223
	aStream.CommitL();
sl@0
   224
	}
sl@0
   225
sl@0
   226
///////////////////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
   227
sl@0
   228
//Database type.
sl@0
   229
enum TDbType 
sl@0
   230
	{
sl@0
   231
	EDbTypeUnknown,			//The database resides outside the client's private data cage 
sl@0
   232
	EDbTypeClientPrivate	//The database resides in the client's private data cage
sl@0
   233
	};
sl@0
   234
sl@0
   235
/**
sl@0
   236
Returns the database type.
sl@0
   237
sl@0
   238
@param aDbFileName Database file name, including the path. If it is a request for creating/opening
sl@0
   239
			secure database, then the name format is <drive>:<[SID]database_file_name>.
sl@0
   240
sl@0
   241
@return EDbTypeClientPrivate If the database resides in the client's private data cage;
sl@0
   242
		EDbTypeUnknown This may be a database in the SQL server private data cage or somewhere else;
sl@0
   243
		If the error code is less than 0, then the function cannot determine the database type
sl@0
   244
		(the function was unable to connect the local file session instance).
sl@0
   245
sl@0
   246
@internalComponent
sl@0
   247
*/
sl@0
   248
static TInt GetDbType(const TDesC& aDbFileName)
sl@0
   249
	{
sl@0
   250
	RFs fs;
sl@0
   251
	TInt err = fs.Connect();
sl@0
   252
	if(err == KErrNone)
sl@0
   253
		{
sl@0
   254
		TFileName clientPrivatePath;
sl@0
   255
		err = fs.PrivatePath(clientPrivatePath);
sl@0
   256
		fs.Close();
sl@0
   257
		if(err == KErrNone)
sl@0
   258
			{
sl@0
   259
			TInt pos = aDbFileName.FindF(clientPrivatePath);
sl@0
   260
			return (TUint)pos <= (TUint)KMaxDriveName ? EDbTypeClientPrivate : EDbTypeUnknown;
sl@0
   261
			}
sl@0
   262
		}
sl@0
   263
	return err;
sl@0
   264
	}
sl@0
   265
sl@0
   266
///////////////////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
   267
//////////////////////                RSqlDbSession           /////////////////////////////////////////////////
sl@0
   268
///////////////////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
   269
sl@0
   270
/**
sl@0
   271
Establishes a connection with the SQL server.
sl@0
   272
sl@0
   273
The method initializes RSqlDbSession object establishing a connection with the SQL server.
sl@0
   274
If the connection has established successfully, the method sends a message to the server to
sl@0
   275
create or open (depending on aFunction parameter) a database file with aDbFileName name.
sl@0
   276
sl@0
   277
@param aFunction Specifies which operation has to be performed:
sl@0
   278
  ESqlSrvDbCreate       - Create a shared non-secure or private secure database;
sl@0
   279
  ESqlSrvDbCreateSecure - Create a shared secure database;
sl@0
   280
  ESqlSrvDbOpen         - Open a shared non-secure, shared secure or private secure database;
sl@0
   281
sl@0
   282
@param aDbFileName Database file name, including the path. If it is a request for creating/opening
sl@0
   283
			secure database, then the name format is <drive>:<[SID]database_file_name>.
sl@0
   284
@param aSecurityPolicyData Security policy data. Non-empty descriptor only for secure databases
sl@0
   285
@param aConfig the configuration string "PARAM=VALUE;...."
sl@0
   286
sl@0
   287
@return KErrNone, the operation completed successfully;
sl@0
   288
		KErrNoMemory, an out of memory condition has occured;
sl@0
   289
		KErrBadName, bad database file name: zero length, directory name;
sl@0
   290
		KErrAlreadyExists, database file already exists;
sl@0
   291
		KErrNotReady, the drive does not exist or is not ready;
sl@0
   292
		KErrInUse, the file has been opened already;
sl@0
   293
		KErrNotFound, file not found;
sl@0
   294
		KErrGeneral, no or invalid security policies (if the database to be opened is a secure database with bad security data);
sl@0
   295
		KErrNotSupported, incompatible sql security version (if the database to be opened is a secure database with bad security data);
sl@0
   296
		KErrPermissionDenied, the caller does not satisfy the relevant database security policies.
sl@0
   297
                      Note that database specific errors categorised as ESqlDbError, and
sl@0
   298
                      other system-wide error codes may also be returned.
sl@0
   299
        KErrArgument invalid config string or config string length exceeds KSqlSrvMaxConfigStrLen.
sl@0
   300
sl@0
   301
Usage of the IPC call arguments:
sl@0
   302
 - Arg 0: [out]  database file name length
sl@0
   303
 - Arg 1: [out]  database file name
sl@0
   304
 - Arg 2: [out]  security policies buffer length if aFunction is ESqlSrvDbCreateSecure
sl@0
   305
 - Arg 3: [out]  security policies buffer if aFunction is ESqlSrvDbCreateSecure
sl@0
   306
*/
sl@0
   307
TInt RSqlDbSession::Connect(TSqlSrvFunction aFunction, const TDesC& aDbFileName, const TDesC8& aSecurityPolicyData, const TDesC8* aConfig)
sl@0
   308
	{
sl@0
   309
    SQL_TRACE_SESSION(OstTraceExt2(TRACE_INTERNALS, RSQLDBSESSION_CONNECT_ENTRY, "Entry;0x%X;RSqlDbSession::Connect;aDbFileName=%S", (TUint)this, __SQLPRNSTR(aDbFileName)));
sl@0
   310
#ifdef SYSLIBS_TEST
sl@0
   311
    const TInt KDefaultMsgBufLen = 4;
sl@0
   312
#else	
sl@0
   313
	const TInt KDefaultMsgBufLen = 128;
sl@0
   314
#endif	
sl@0
   315
	iLastErrorMessage = HBufC::New(KDefaultMsgBufLen);
sl@0
   316
	if(!iLastErrorMessage)
sl@0
   317
		{
sl@0
   318
		return KErrNoMemory;	
sl@0
   319
		}
sl@0
   320
	TInt err = DoCreateSession();
sl@0
   321
	if(err == KErrNone)
sl@0
   322
		{
sl@0
   323
		err = InitResourceTestData();
sl@0
   324
		if(err == KErrNone)
sl@0
   325
			{
sl@0
   326
			err = ::GetDbType(aDbFileName);
sl@0
   327
			if(err >= 0)	
sl@0
   328
				{
sl@0
   329
				if(err == EDbTypeClientPrivate)
sl@0
   330
					{//The database is in the client's private data cage. Set err = KErrPermissionDenied.
sl@0
   331
					err = KErrPermissionDenied;
sl@0
   332
					}
sl@0
   333
				else
sl@0
   334
					{//Unknown database type. Try to connect.
sl@0
   335
					err = DoConnect(aFunction, aDbFileName, aSecurityPolicyData, aConfig);
sl@0
   336
					}
sl@0
   337
				if(err == KErrPermissionDenied && aFunction != ESqlSrvDbCreateSecure)
sl@0
   338
					{
sl@0
   339
					//What do we have here now? - the operation is create/open non-secure, the error is KErrPermissionDenied.
sl@0
   340
					//So, the dll will try now to create/open the database in the application's private data cage.
sl@0
   341
					//What if the used database file name was "C:MyDb.db"!? - the dll will try to create/open 
sl@0
   342
					//"C:MyDb.db" in the application's data cage and there
sl@0
   343
					//is a possibility that this operation my return a result, like KErrNotFound, which will hide the original
sl@0
   344
					//error code (KErrPermissionDenied).
sl@0
   345
					//The dll shall not try to create/open file in the application's data cage, if the format of the
sl@0
   346
					//database file name is secure.
sl@0
   347
					TParsePtrC parse(aDbFileName);//this call may panic if aDbFileName cannot be parsed, but it will panic the client
sl@0
   348
					if(parse.PathPresent())
sl@0
   349
						{
sl@0
   350
						//The caller or the SQL server has no enough rights to create or open the database with aDbFileName name.
sl@0
   351
						//One of the reasons may be that the database file is in the caller's private data cage where the SQL
sl@0
   352
						//server cannot create/open the database file.
sl@0
   353
						//The SQL dll will try to create/open the database file  on the client side and pass the file handle to the
sl@0
   354
						//SQL server.
sl@0
   355
						TSqlFhOpenCmdFunctor fhOpenCmdSender(*this, aDbFileName, aConfig);
sl@0
   356
						err = CreateAndSendFileHandle(fhOpenCmdSender,
sl@0
   357
							aFunction == ESqlSrvDbCreate ? RSqlDbSession::EDbfCreate : RSqlDbSession::EDbfOpen);
sl@0
   358
						}
sl@0
   359
					}	
sl@0
   360
				}
sl@0
   361
			}
sl@0
   362
		}
sl@0
   363
	if(err != KErrNone)
sl@0
   364
		{
sl@0
   365
		Close();	
sl@0
   366
		}
sl@0
   367
    SQL_TRACE_SESSION(OstTraceExt3(TRACE_INTERNALS, RSQLDBSESSION_CONNECT_EXIT, "Exit;0x%X;RSqlDbSession::Connect;err=%d;handle=0x%X", (TUint)this, err, (TUint)Handle()));
sl@0
   368
	return err;
sl@0
   369
	}
sl@0
   370
sl@0
   371
/**
sl@0
   372
Sends a command to the server to attach an existing database to the current connection.
sl@0
   373
sl@0
   374
The database can be:
sl@0
   375
@code
sl@0
   376
 - shared secure database;
sl@0
   377
 - shared non-secure database;
sl@0
   378
 - private secure database;
sl@0
   379
@endcode
sl@0
   380
sl@0
   381
@param aDbFileName Database file name.
sl@0
   382
@param aDbName Logical database name. 
sl@0
   383
sl@0
   384
@return KErrNone, the operation completed successfully;
sl@0
   385
		KErrNoMemory, an out of memory condition has occured;
sl@0
   386
		KErrBadName, bad database file name: zero length, directory name;
sl@0
   387
		KErrNotReady, the drive does not exist or is not ready;
sl@0
   388
		KErrInUse, the file has been opened already;
sl@0
   389
		KErrNotFound, file not found;
sl@0
   390
		KErrGeneral, no or invalid security policies (if the database to be opened is a secure database with bad security data);
sl@0
   391
		KErrNotSupported, incompatible sql security version (if the database to be opened is a secure database with bad security data);
sl@0
   392
		KErrPermissionDenied, the caller does not satisfy the relevant database security policies.
sl@0
   393
                      Note that database specific errors categorised as ESqlDbError, and
sl@0
   394
                      other system-wide error codes may also be returned.
sl@0
   395
sl@0
   396
Usage of the IPC call arguments: 
sl@0
   397
Arg 0: [out]	Database file name length.
sl@0
   398
Arg 1: [out]	Database file name.
sl@0
   399
Arg 2: [out]	Logical database name length.
sl@0
   400
Arg 3: [out]	Logical database name.
sl@0
   401
*/
sl@0
   402
TInt RSqlDbSession::Attach(const TDesC& aDbFileName, const TDesC& aDbName)
sl@0
   403
	{
sl@0
   404
	TInt err = ::GetDbType(aDbFileName);
sl@0
   405
	if(err >= 0)	
sl@0
   406
		{
sl@0
   407
		if(err == EDbTypeClientPrivate)
sl@0
   408
			{//The database is in the client's private data cage. Set err = KErrPermissionDenied.
sl@0
   409
			err = KErrPermissionDenied;
sl@0
   410
			}
sl@0
   411
		else
sl@0
   412
			{//Unknown database type. Try to attach.
sl@0
   413
			err = SendReceive(ESqlSrvDbAttach, TIpcArgs(aDbFileName.Length(), &aDbFileName, aDbName.Length(), &aDbName));
sl@0
   414
			}
sl@0
   415
		if(err == KErrPermissionDenied)
sl@0
   416
			{
sl@0
   417
			//What do we have here now? - the operation is "attach", the error is KErrPermissionDenied.
sl@0
   418
			//Which means, it is (possibly) a secure database in the server private data cage and the caller 
sl@0
   419
			//failed to pass the security checks.
sl@0
   420
			//So, the dll will try now to open the database assuming it is in the application's private data cage
sl@0
   421
			//and pass the file and session handles to the server which will try to attach the database using the handles.
sl@0
   422
			//What if the used database file name was "C:MyDb.db"!? (which means that the client's intention was
sl@0
   423
			//to attach a public shared secure database)
sl@0
   424
			//The dll will try to open "C:MyDb.db" in the application's data cage and there
sl@0
   425
			//is a possibility that this operation may return a result, like KErrNotFound, which will
sl@0
   426
			//hide the original error code (KErrPermissionDenied).
sl@0
   427
			//The dll shall not try to attach a database from the application's data cage, if the format of the
sl@0
   428
			//database file name is secure.
sl@0
   429
			TParsePtrC parse(aDbFileName);//this call may panic if aDbFileName cannot be parsed, but it will panic the client
sl@0
   430
			if(parse.PathPresent())
sl@0
   431
				{
sl@0
   432
				TSqlFhAttachCmdFunctor fhAttachCmdSender(*this, aDbFileName, aDbName);
sl@0
   433
				err = CreateAndSendFileHandle(fhAttachCmdSender, RSqlDbSession::EDbfOpen);
sl@0
   434
				}
sl@0
   435
			}
sl@0
   436
		}
sl@0
   437
	return err;
sl@0
   438
	}
sl@0
   439
	
sl@0
   440
/**
sl@0
   441
Copies a database.
sl@0
   442
sl@0
   443
The method establishes a temporary connection with the SQL server and issues a request for copying
sl@0
   444
aSrcDbFileName database file to aDestDbFileName file. After the server completes the request 
sl@0
   445
CopyDatabase() closes the connection.
sl@0
   446
sl@0
   447
@param aSrcDbFileName Source database file name.
sl@0
   448
@param aDestDbFileName Destination database file name.
sl@0
   449
sl@0
   450
@return KErrNone, the operation completed successfully;
sl@0
   451
		KErrNoMemory, an out of memory condition has occured;
sl@0
   452
		KErrBadName, bad database file name: zero length, directory name;
sl@0
   453
		KErrAlreadyExists, target database file already exists;
sl@0
   454
		KErrNotReady, the drive does not exist or is not ready;
sl@0
   455
		KErrInUse, the file has been opened already;
sl@0
   456
		KErrNotFound, file not found;
sl@0
   457
        KErrPermissionDenied, the SID of the calling application does not match the SID of source or destination database.
sl@0
   458
                      Note that other system-wide error codes may also be returned.
sl@0
   459
sl@0
   460
Usage of the IPC call arguments: 
sl@0
   461
Arg 0: [out]  source database file name length
sl@0
   462
Arg 1: [out]  source database file name
sl@0
   463
Arg 2: [out]  destination database file name length
sl@0
   464
Arg 3: [out]  destination database file name
sl@0
   465
*/
sl@0
   466
TInt RSqlDbSession::CopyDatabase(const TDesC& aSrcDbFileName, const TDesC& aDestDbFileName)
sl@0
   467
	{
sl@0
   468
	RSqlDbSession sess;
sl@0
   469
	TInt err = sess.DoCreateSession();
sl@0
   470
	if(err == KErrNone)
sl@0
   471
		{
sl@0
   472
		err = sess.InitResourceTestData();
sl@0
   473
		if(err == KErrNone)
sl@0
   474
			{
sl@0
   475
			err = sess.SendReceive(ESqlSrvDbCopy, TIpcArgs(aSrcDbFileName.Length(), &aSrcDbFileName, aDestDbFileName.Length(), &aDestDbFileName));
sl@0
   476
			}
sl@0
   477
		sess.Close();
sl@0
   478
		}
sl@0
   479
	return err;
sl@0
   480
	}
sl@0
   481
sl@0
   482
/**
sl@0
   483
Deletes a database.
sl@0
   484
sl@0
   485
The database can be:
sl@0
   486
@code
sl@0
   487
 - shared secure database;
sl@0
   488
 - shared non-secure database;
sl@0
   489
 - private secure database;
sl@0
   490
@endcode
sl@0
   491
sl@0
   492
The method establishes a temporary connection with the SQL server and issues a "delete" request regarding
sl@0
   493
aDbFileName database file. After the server completes the request DeleteDatabase() closes the connection.
sl@0
   494
sl@0
   495
@param aDbFileName Database file name. It must include the path if it is not a secure database.
sl@0
   496
sl@0
   497
@return KErrNone, the operation completed successfully;
sl@0
   498
		KErrNoMemory, an out of memory condition has occured;
sl@0
   499
		KErrBadName, bad database file name: zero length, directory name;
sl@0
   500
		KErrNotReady, the drive does not exist or is not ready;
sl@0
   501
		KErrInUse, the file has been opened already;
sl@0
   502
		KErrNotFound, file not found;
sl@0
   503
		KErrAccessDenied, access to the database file is denied (it might be a read-only file);
sl@0
   504
        KErrPermissionDenied, the SID of the calling application does not match the SID of the database.
sl@0
   505
                      Note that other system-wide error codes may also be returned.
sl@0
   506
                      
sl@0
   507
Usage of the IPC call arguments: 
sl@0
   508
Arg 0: [out]  database file name length
sl@0
   509
Arg 1: [out]  database file name
sl@0
   510
*/
sl@0
   511
TInt RSqlDbSession::DeleteDatabase(const TDesC& aDbFileName)
sl@0
   512
	{
sl@0
   513
	RSqlDbSession sess;
sl@0
   514
	TInt err = sess.DoCreateSession();
sl@0
   515
	if(err == KErrNone)
sl@0
   516
		{
sl@0
   517
		err = sess.InitResourceTestData();
sl@0
   518
		if(err == KErrNone)
sl@0
   519
			{
sl@0
   520
			err = ::GetDbType(aDbFileName);
sl@0
   521
			if(err >= 0)	
sl@0
   522
				{
sl@0
   523
				if(err == EDbTypeClientPrivate)
sl@0
   524
					{//The database is in the client's private data cage. Set err = KErrPermissionDenied.
sl@0
   525
					err = KErrPermissionDenied;
sl@0
   526
					}
sl@0
   527
				else
sl@0
   528
					{//Unknown database type. Try to delete.
sl@0
   529
					err = sess.SendReceive(ESqlSrvDbDelete, TIpcArgs(aDbFileName.Length(), &aDbFileName));
sl@0
   530
					}
sl@0
   531
				if(err == KErrPermissionDenied)
sl@0
   532
					{
sl@0
   533
					//What do we have here now? - the operation is "delete db", the error is KErrPermissionDenied.
sl@0
   534
					//So, the dll will try now to delete a database with the same name from the application's private data cage.
sl@0
   535
					//What if the used database file name was "C:MyDb.db" and there is a such file in server's private
sl@0
   536
					//data cage? - the dll will try to delete a file with name "C:MyDb.db" from the application's data cage 
sl@0
   537
					//and there is a possibility that this operation my pass or may return strange result, like KErrNotFound.
sl@0
   538
					//Bith cases are not what the user would expect.
sl@0
   539
					//The dll shall not try to delete a file from the application's data cage, if the format of the
sl@0
   540
					//database file name is secure.
sl@0
   541
					TParsePtrC parse(aDbFileName);//this call may panic if aDbFileName cannot be parsed, but it will panic the client
sl@0
   542
					if(parse.PathPresent())
sl@0
   543
						{
sl@0
   544
						//The caller or the SQL server has no enough rights to delete the database with aDbFileName name.
sl@0
   545
						//One of the reasons may be that the database file is in the caller's private data cage where the SQL
sl@0
   546
						//server cannot delete the database file.
sl@0
   547
						//The SQL dll will try to delete the database file on the client side.
sl@0
   548
						RFs fs;
sl@0
   549
						err = fs.Connect();
sl@0
   550
						if(err == KErrNone)
sl@0
   551
							{
sl@0
   552
							err = fs.Delete(aDbFileName);
sl@0
   553
							fs.Close();
sl@0
   554
							}
sl@0
   555
						}
sl@0
   556
					}
sl@0
   557
				}
sl@0
   558
			}
sl@0
   559
		sess.Close();
sl@0
   560
		}
sl@0
   561
	return err;
sl@0
   562
	}
sl@0
   563
sl@0
   564
/**
sl@0
   565
Retrieves a reference to the textual description of the error returned by the
sl@0
   566
most recent call to any of the functions:
sl@0
   567
- RSqlDatabase::Exec()
sl@0
   568
- RSqlStatement::Exec()
sl@0
   569
- RSqlStatement::Next()
sl@0
   570
- RSqlStatement::Reset()
sl@0
   571
sl@0
   572
Note that the function can only return a reference to text for
sl@0
   573
database-specific type errors, i.e. those errors that are categorised as of
sl@0
   574
type ESqlDbError.
sl@0
   575
sl@0
   576
If an error occurs during the retrieval of the last error message, the function silently ignores the error
sl@0
   577
and returns a NULL descriptor.
sl@0
   578
sl@0
   579
@return A non-modifiable pointer descriptor representing the most recent error
sl@0
   580
        message. Note that message may be NULL, i.e. the descriptor may have
sl@0
   581
        zero length.
sl@0
   582
sl@0
   583
Usage of the IPC call arguments:
sl@0
   584
Arg 0: [out]	max length of the message buffer
sl@0
   585
Arg 1: [in/out]	buffer for the last error message
sl@0
   586
*/
sl@0
   587
TPtrC RSqlDbSession::LastErrorMessage()
sl@0
   588
	{
sl@0
   589
	TPtr msg(iLastErrorMessage->Des());
sl@0
   590
	msg.Zero();
sl@0
   591
	TInt size = 0;
sl@0
   592
	while((size = SendReceive(ESqlSrvLastErrorMsg, TIpcArgs(msg.MaxLength(), &msg))) > KSqlClientBufOverflowCode)
sl@0
   593
		{
sl@0
   594
		HBufC* newMsgBuf = iLastErrorMessage->ReAlloc(size - KSqlClientBufOverflowCode);
sl@0
   595
		if(!newMsgBuf)
sl@0
   596
			{
sl@0
   597
			break;	
sl@0
   598
			}
sl@0
   599
		iLastErrorMessage = newMsgBuf;	
sl@0
   600
		msg.Set(iLastErrorMessage->Des());
sl@0
   601
		}
sl@0
   602
	return msg;
sl@0
   603
	}
sl@0
   604
sl@0
   605
/**
sl@0
   606
Returns the ROWID of the most recent successful INSERT into the database 
sl@0
   607
from this database connection.
sl@0
   608
sl@0
   609
@return >0, the ROWID of the most recent successful INSERT into the database
sl@0
   610
			from this database connection;
sl@0
   611
		0, 	if no successful INSERTs have ever occurred from this database connection
sl@0
   612
		<0, if one of the system-wide error codes is returned
sl@0
   613
sl@0
   614
Usage of the IPC call arguments:
sl@0
   615
Arg 0: [in/out]	the receiving buffer for the last inserted ROWID
sl@0
   616
*/	
sl@0
   617
TInt64 RSqlDbSession::LastInsertedRowId()
sl@0
   618
	{
sl@0
   619
	TInt64 res;
sl@0
   620
	TPtr8 ptr(reinterpret_cast <TUint8*> (&res), sizeof(res));
sl@0
   621
	TInt err = SendReceive(ESqlSrvDbLastInsertedRowId, TIpcArgs(&ptr));
sl@0
   622
	return err == KErrNone ? res : err;
sl@0
   623
	}
sl@0
   624
sl@0
   625
/**
sl@0
   626
Retrieves the database security policies.
sl@0
   627
sl@0
   628
@return A pointer to the created and internalized CSqlSecurityPolicy instance.
sl@0
   629
sl@0
   630
@return KErrNone, the operation has completed successfully;
sl@0
   631
		KErrNoMemory, an out of memory condition has occurred;
sl@0
   632
                      Note that the function may leave with some database specific errors categorised as 
sl@0
   633
                      ESqlDbError or other system-wide error codes.
sl@0
   634
sl@0
   635
Usage of the IPC call arguments:
sl@0
   636
Arg 0: [out]	security policy buffer size
sl@0
   637
Arg 1: [in/out]	buffer for the database security policies
sl@0
   638
*/
sl@0
   639
TInt RSqlDbSession::GetSecurityPolicy(RSqlBufFlat& aSecurityPolicyBuf)
sl@0
   640
	{
sl@0
   641
	TPtr8 ptr(aSecurityPolicyBuf.BufPtr());
sl@0
   642
	TInt rc = KErrNone;
sl@0
   643
	while((rc = SendReceive(ESqlSrvDbGetSecurityPolicy, TIpcArgs(ptr.MaxLength(), &ptr))) > KSqlClientBufOverflowCode)
sl@0
   644
		{
sl@0
   645
		rc = aSecurityPolicyBuf.ReAlloc(rc - KSqlClientBufOverflowCode);
sl@0
   646
		if(rc != KErrNone)
sl@0
   647
			{
sl@0
   648
			break;
sl@0
   649
			}
sl@0
   650
		ptr.Set(aSecurityPolicyBuf.BufPtr());
sl@0
   651
		}
sl@0
   652
	return rc;
sl@0
   653
	}
sl@0
   654
	
sl@0
   655
/**
sl@0
   656
Closes the database and releases the connection with the database server.
sl@0
   657
*/
sl@0
   658
void RSqlDbSession::Close()
sl@0
   659
	{
sl@0
   660
    SQL_TRACE_SESSION(OstTraceExt2(TRACE_INTERNALS, RSQLDBSESSION_CLOSE, "0x%X;RSqlDbSession::Close;handle=0x%X", (TUint)this, (TUint)Handle()));
sl@0
   661
	if(Handle())
sl@0
   662
		{
sl@0
   663
		(void)SendReceive(ESqlSrvDbClose);
sl@0
   664
		}
sl@0
   665
	TSqlResourceTestData::Release();
sl@0
   666
	delete iLastErrorMessage;
sl@0
   667
	iLastErrorMessage = NULL;
sl@0
   668
	RSessionBase::Close();
sl@0
   669
	}
sl@0
   670
sl@0
   671
/**
sl@0
   672
The method establishes a connection with the SQL server.
sl@0
   673
sl@0
   674
@return KErrNone, The connection was established successfully;
sl@0
   675
		KErrAlreadyExists, the connection already exists.
sl@0
   676
                 The function may also return some other system-wide error codes.
sl@0
   677
*/
sl@0
   678
TInt RSqlDbSession::DoCreateSession()
sl@0
   679
	{
sl@0
   680
	const TInt KTimesToRetryConnection = 2;
sl@0
   681
	TInt retry = KTimesToRetryConnection;
sl@0
   682
	for(;;)
sl@0
   683
		{
sl@0
   684
		TInt err = CreateSession(KSqlSrvName, ::SqlSrvVersion());
sl@0
   685
		if(err != KErrNotFound && err != KErrServerTerminated)
sl@0
   686
			{
sl@0
   687
			return err;
sl@0
   688
			}
sl@0
   689
		if(--retry == 0)
sl@0
   690
			{
sl@0
   691
			return err;
sl@0
   692
			}
sl@0
   693
		err = ::StartSqlServer();
sl@0
   694
		if(err != KErrNone && err != KErrAlreadyExists)
sl@0
   695
			{
sl@0
   696
			return err;
sl@0
   697
			}
sl@0
   698
		}
sl@0
   699
	}
sl@0
   700
sl@0
   701
/**
sl@0
   702
The method gets called immediatelly after the establishing client-server connection but before the 
sl@0
   703
create/open database message. 
sl@0
   704
If a request was made by the client (using TSqlResourceTester methods) to test the server under out of memory
sl@0
   705
conditions, InitResourceTestData() will send this request to the server, putting the just created connection
sl@0
   706
in a "out of memory" test mode.
sl@0
   707
*/
sl@0
   708
TInt RSqlDbSession::InitResourceTestData()
sl@0
   709
	{
sl@0
   710
	TInt err = KErrNone;
sl@0
   711
#ifdef _DEBUG	
sl@0
   712
	TSqlResourceTestData* data = TSqlResourceTestData::Instance();
sl@0
   713
	if(data)
sl@0
   714
		{
sl@0
   715
		data->Init(*this);
sl@0
   716
		}
sl@0
   717
	else
sl@0
   718
		{
sl@0
   719
		err = KErrNoMemory;	
sl@0
   720
		}
sl@0
   721
#endif	
sl@0
   722
	return err;	
sl@0
   723
	}
sl@0
   724
sl@0
   725
/**
sl@0
   726
Creates/opens database file locally and sends the file handle to the SQL server.
sl@0
   727
sl@0
   728
This function is used only when the SQL server fails to create/open/attach the requested database file with
sl@0
   729
KErrPermissionDenied error. One of the reasons for that error may be that the database file is in the 
sl@0
   730
calling application's private data cage. In this case an attempt is made to create/open the database file locally
sl@0
   731
and if the operation completes successfully, then the file handle is passed to the SQL server.
sl@0
   732
sl@0
   733
@param aFhCmdSender A reference to a functor instance which is used for sending the file handles to the SQL server. 
sl@0
   734
@param aDbfAction It is set by the caller to RSqlDbSession::EDbfCreate if the database file has to be created or
sl@0
   735
				  to RSqlDbSession::EDbfOpen if the database file already exists and has to be opened
sl@0
   736
sl@0
   737
@return KErrNone, the operation completed successfully;
sl@0
   738
		KErrNoMemory, an out of memory condition has occured;
sl@0
   739
		KErrBadName, bad database file name: zero length, directory name, missing drive in the file name;
sl@0
   740
		KErrAlreadyExists, database file already exists;
sl@0
   741
		KErrNotReady, the drive does not exist or is not ready;
sl@0
   742
		KErrInUse, the file has been opened already;
sl@0
   743
		KErrNotFound, file not found;
sl@0
   744
		KErrPermissionDenied, the caller does not satisfy the relevant database security policies.
sl@0
   745
                      Note that database specific errors categorised as ESqlDbError, and
sl@0
   746
                      other system-wide error codes may also be returned.
sl@0
   747
*/
sl@0
   748
TInt RSqlDbSession::CreateAndSendFileHandle(TSqlFhCmdFunctor& aFhCmdSender, RSqlDbSession::TDbfAction aDbfAction)
sl@0
   749
	{
sl@0
   750
	//Create a file session
sl@0
   751
	RFs fs;
sl@0
   752
	TInt err = fs.Connect();
sl@0
   753
	if(err != KErrNone)
sl@0
   754
		{
sl@0
   755
		return err;	
sl@0
   756
		}
sl@0
   757
	//Share the file session
sl@0
   758
	err = fs.ShareProtected();
sl@0
   759
	if(err != KErrNone)
sl@0
   760
		{
sl@0
   761
		fs.Close();	
sl@0
   762
		return err;	
sl@0
   763
		}
sl@0
   764
	//Create/open the database file locally
sl@0
   765
	TBool readOnly = EFalse;
sl@0
   766
	TBool fileCreated = EFalse;
sl@0
   767
	RFile64 dbFile;
sl@0
   768
	if(aDbfAction == RSqlDbSession::EDbfCreate)
sl@0
   769
		{//Create the database file in R/W mode
sl@0
   770
		err = dbFile.Create(fs, aFhCmdSender.iDbFileName, EFileShareAny | EFileWrite);
sl@0
   771
		if(err == KErrNone)
sl@0
   772
			{
sl@0
   773
			fileCreated = ETrue;	
sl@0
   774
			}
sl@0
   775
		}
sl@0
   776
	else //aDbfAction == RSqlDbSession::EDbfOpen
sl@0
   777
		{//Open the database file in shared R/W mode
sl@0
   778
		err = dbFile.Open(fs, aFhCmdSender.iDbFileName, EFileShareAny | EFileWrite);
sl@0
   779
		if(err != KErrNone)
sl@0
   780
			{//If the the database open operation in R/W mode has failed - try to open the database in shared read-only mode.
sl@0
   781
			readOnly = ETrue;
sl@0
   782
			err = dbFile.Open(fs, aFhCmdSender.iDbFileName, EFileShareReadersOnly);
sl@0
   783
			}
sl@0
   784
		}
sl@0
   785
	if(err == KErrNone)
sl@0
   786
		{//Successful create/open database file operation. Send the database file and session handles to the server.
sl@0
   787
		err = aFhCmdSender(dbFile, fileCreated, readOnly);
sl@0
   788
		dbFile.Close();
sl@0
   789
		}
sl@0
   790
	if(err != KErrNone && fileCreated)
sl@0
   791
		{
sl@0
   792
		dbFile.Close();
sl@0
   793
		TInt err2 = fs.Delete(aFhCmdSender.iDbFileName);
sl@0
   794
		if(err2 != KErrNone)
sl@0
   795
			{//Sometimes it is not possible the file to be deleted, the reported error is - KErrInUse. 			
sl@0
   796
			fs.Close();
sl@0
   797
			User::After(100000);
sl@0
   798
			err2 = fs.Connect();
sl@0
   799
			if(err2 == KErrNone)
sl@0
   800
				{
sl@0
   801
				(void)fs.Delete(aFhCmdSender.iDbFileName);
sl@0
   802
				}
sl@0
   803
			}
sl@0
   804
		}
sl@0
   805
	fs.Close();	
sl@0
   806
	return err;
sl@0
   807
	}
sl@0
   808
sl@0
   809
/**
sl@0
   810
Prepares the "security & configuration" string and sends the command to the SQL server.
sl@0
   811
sl@0
   812
@param aFunction Specifies which operation has to be performed:
sl@0
   813
  ESqlSrvDbCreate       - Create a shared non-secure or private secure database;
sl@0
   814
  ESqlSrvDbCreateSecure - Create a shared secure database;
sl@0
   815
  ESqlSrvDbOpen         - Open a shared non-secure, shared secure or private secure database;
sl@0
   816
sl@0
   817
@param aDbFileName Database file name, including the path. If it is a request for creating/opening
sl@0
   818
			secure database, then the name format is <drive>:<[SID]database_file_name>.
sl@0
   819
@param aSecurityPolicyData Security policy data. Non-empty descriptor only for secure databases
sl@0
   820
@param aConfig the configuration string "PARAM=VALUE;...."
sl@0
   821
sl@0
   822
@return KErrNone, the operation completed successfully;
sl@0
   823
		KErrNoMemory, an out of memory condition has occured;
sl@0
   824
        KErrArgument invalid config string or config string length exceeds KSqlSrvMaxConfigStrLen.
sl@0
   825
                     Other system-wide error codes may also be returned.
sl@0
   826
		
sl@0
   827
*/
sl@0
   828
TInt RSqlDbSession::DoConnect(TSqlSrvFunction aFunction, const TDesC& aDbFileName, const TDesC8& aSecurityPolicyData, const TDesC8* aConfig)
sl@0
   829
	{
sl@0
   830
	const TInt KConfigStrLen = aConfig ? aConfig->Length() : 0;
sl@0
   831
	if(KConfigStrLen > KSqlSrvMaxConfigStrLen)
sl@0
   832
		{
sl@0
   833
		return KErrArgument;
sl@0
   834
		}
sl@0
   835
	const TInt KSecPolicyLen = aFunction == ESqlSrvDbCreateSecure ? aSecurityPolicyData.Length() : 0;
sl@0
   836
	TIpcArgs ipcArgs(aDbFileName.Length(), &aDbFileName);
sl@0
   837
	// pack the length xxxxyyyy where xxxx is the policy length
sl@0
   838
	// and yyyy is the config length.
sl@0
   839
	ipcArgs.Set(2, (TUint)(KSecPolicyLen << 16) | (TUint)KConfigStrLen);
sl@0
   840
	const TInt KTotalLen = KConfigStrLen + KSecPolicyLen;
sl@0
   841
	ipcArgs.Set(3, 0);
sl@0
   842
	HBufC8* arg3Buf = NULL;
sl@0
   843
	if(KTotalLen > 0)
sl@0
   844
		{
sl@0
   845
		arg3Buf = HBufC8::New(KTotalLen);
sl@0
   846
		if(!arg3Buf)
sl@0
   847
			{
sl@0
   848
			return KErrNoMemory;
sl@0
   849
			}
sl@0
   850
		TPtr8 arg3Ptr = arg3Buf->Des();
sl@0
   851
		if(KSecPolicyLen > 0)
sl@0
   852
			{
sl@0
   853
			arg3Ptr.Copy(aSecurityPolicyData);
sl@0
   854
			}
sl@0
   855
		if(KConfigStrLen > 0)
sl@0
   856
			{
sl@0
   857
			//coverity[DEADCODE]
sl@0
   858
			//The ASSERT might be useful in catching future defect in this function
sl@0
   859
			__ASSERT_DEBUG(aConfig != NULL, __SQLPANIC(ESqlPanicInternalError));
sl@0
   860
			arg3Ptr.Append(*aConfig);
sl@0
   861
			}
sl@0
   862
		ipcArgs.Set(3, &arg3Ptr);
sl@0
   863
		}
sl@0
   864
	TInt err = SendReceive(aFunction, ipcArgs);
sl@0
   865
	delete arg3Buf;
sl@0
   866
	return err;
sl@0
   867
	}