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