sl@0: // Copyright (c) 2005-2010 Nokia Corporation and/or its subsidiary(-ies). sl@0: // All rights reserved. sl@0: // This component and the accompanying materials are made available sl@0: // under the terms of "Eclipse Public License v1.0" sl@0: // which accompanies this distribution, and is available sl@0: // at the URL "http://www.eclipse.org/legal/epl-v10.html". sl@0: // sl@0: // Initial Contributors: sl@0: // Nokia Corporation - initial contribution. sl@0: // sl@0: // Contributors: sl@0: // sl@0: // Description: sl@0: // sl@0: sl@0: #include sl@0: #include sl@0: #include sl@0: #include sl@0: #include sl@0: #include "IPCBuf.h" //HIpcBuf sl@0: #include "SqlDbSession.h" //RSqlDbSession sl@0: #include "SqlSrvStartup.h" //StartSqlServer() sl@0: #include "SqlResourceTest.h" //TSqlResourceTestData sl@0: #include "SqlSecurityImpl.h" //CSqlSecurityPolicy sl@0: #include "OstTraceDefinitions.h" sl@0: #ifdef OST_TRACE_COMPILER_IN_USE sl@0: #include "SqlDbSessionTraces.h" sl@0: #endif sl@0: #include "SqlTraceDef.h" sl@0: sl@0: /////////////////////////////////////////////////////////////////////////////////////////////////////////////// sl@0: ////////////////////// TSqlFhCmdFunctor /////////////////////////////////////////////// sl@0: /////////////////////////////////////////////////////////////////////////////////////////////////////////////// sl@0: sl@0: /** sl@0: TSqlFhCmdFunctor derived classes are used by RSqlDbSession for sending to SQL server localy created/opened sl@0: file handles. sl@0: sl@0: @see RSqlDbSession sl@0: @see RSqlDbSession::CreateAndSendFileHandle() sl@0: sl@0: @internalComponent sl@0: */ sl@0: NONSHARABLE_CLASS(TSqlFhCmdFunctor) sl@0: { sl@0: public: sl@0: TSqlFhCmdFunctor(RSqlDbSession& aDbSession, const TDesC& aDbFileName, const TDesC8* aConfig=NULL) : sl@0: iDbSession(aDbSession), sl@0: iDbFileName(aDbFileName), sl@0: iConfig(aConfig) // ownership not transferred sl@0: { sl@0: } sl@0: virtual TInt operator()(RFile64& aFile, TBool aCreated, TBool aReadOnly) = 0; sl@0: sl@0: public: sl@0: RSqlDbSession& iDbSession; sl@0: const TDesC& iDbFileName; sl@0: const TDesC8* iConfig; sl@0: }; sl@0: sl@0: /////////////////////////////////////////////////////////////////////////////////////////////////////////////// sl@0: ////////////////////// TSqlFhOpenCmdFunctor ////////////////////////////////////////////// sl@0: /////////////////////////////////////////////////////////////////////////////////////////////////////////////// sl@0: sl@0: /** sl@0: Implements the sending of ESqlSrvDbOpenFromHandle command to the SQL server. sl@0: sl@0: @internalComponent sl@0: */ sl@0: NONSHARABLE_CLASS(TSqlFhOpenCmdFunctor) : public TSqlFhCmdFunctor sl@0: { sl@0: public: sl@0: TSqlFhOpenCmdFunctor(RSqlDbSession& aDbSession, const TDesC& aDbFileName,const TDesC8* aConfig=NULL) : sl@0: TSqlFhCmdFunctor(aDbSession, aDbFileName, aConfig) sl@0: { sl@0: } sl@0: virtual TInt operator()(RFile64& aDbFile, TBool aCreated, TBool aReadOnly); sl@0: sl@0: }; sl@0: sl@0: /** sl@0: Sends a command to the SQL server to use the supplied file session and file handles for adopting, sl@0: when opening/creating a database. sl@0: sl@0: Usage of the IPC call arguments: sl@0: - Arg 0: [out] The 32 bits of the argument are used as follow: sl@0: @code sl@0: MSB LSB sl@0: 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: 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: @endcode sl@0: Where: sl@0: @code sl@0: - "Ro" - read-only flag, true if the file is read-only; sl@0: - "Cr" - create/open flag, true if the file was created, false if the file was opened; sl@0: - "C" - config string length; sl@0: - "F" - database file name length; sl@0: @endcode sl@0: - Arg 1: [out] database file name + configuration string (if there is a configuration string) sl@0: - Arg 2: [out] file session handle sl@0: - Arg 3: [out] database file handle sl@0: */ sl@0: TInt TSqlFhOpenCmdFunctor::operator()(RFile64& aDbFile, TBool aCreated, TBool aReadOnly) sl@0: { sl@0: const TInt KConfigStrLen = iConfig ? iConfig->Length() : 0; sl@0: if(KConfigStrLen > KSqlSrvMaxConfigStrLen) sl@0: { sl@0: return KErrArgument; sl@0: } sl@0: TIpcArgs ipcArgs; sl@0: HBufC* arg1Buf = HBufC::New(iDbFileName.Length() + KConfigStrLen); sl@0: if(!arg1Buf) sl@0: { sl@0: return KErrNoMemory; sl@0: } sl@0: TPtr arg1 = arg1Buf->Des(); sl@0: arg1.Copy(iDbFileName); sl@0: if(iConfig) sl@0: { sl@0: TBuf cfgBuf; sl@0: cfgBuf.Copy(*iConfig); sl@0: arg1.Append(cfgBuf); sl@0: } sl@0: TUint32 arg0 = iDbFileName.Length() | (KConfigStrLen << 16); sl@0: if(aReadOnly) sl@0: { sl@0: arg0 |= 0x80000000; sl@0: } sl@0: if(aCreated) sl@0: { sl@0: arg0 |= 0x40000000; sl@0: } sl@0: ipcArgs.Set(0, arg0); sl@0: ipcArgs.Set(1, &arg1); sl@0: TInt err = aDbFile.TransferToServer(ipcArgs, 2, 3); sl@0: if(err == KErrNone) sl@0: { sl@0: err = iDbSession.SendReceive(ESqlSrvDbOpenFromHandle, ipcArgs); sl@0: } sl@0: delete arg1Buf; sl@0: return err; sl@0: } sl@0: sl@0: /////////////////////////////////////////////////////////////////////////////////////////////////////////////// sl@0: ////////////////////// TSqlFhAttachCmdFunctor //////////////////////////////////////////// sl@0: /////////////////////////////////////////////////////////////////////////////////////////////////////////////// sl@0: sl@0: /** sl@0: Implements the sending of ESqlSrvDbAttachFromHandle command to the SQL server. sl@0: sl@0: @internalComponent sl@0: */ sl@0: NONSHARABLE_CLASS(TSqlFhAttachCmdFunctor) : public TSqlFhCmdFunctor sl@0: { sl@0: public: sl@0: TSqlFhAttachCmdFunctor(RSqlDbSession& aDbSession, const TDesC& aDbFileName, sl@0: const TDesC& aDbName) : sl@0: TSqlFhCmdFunctor(aDbSession, aDbFileName), sl@0: iDbName(aDbName) sl@0: { sl@0: } sl@0: virtual TInt operator()(RFile64& aDbFile, TBool aCreated, TBool aReadOnly); sl@0: sl@0: private: sl@0: void SerializeToStreamL(RWriteStream& aStream); sl@0: sl@0: private: sl@0: const TDesC& iDbName; sl@0: sl@0: }; sl@0: sl@0: /** sl@0: Sends a command to the SQL server to use the supplied file session and file handles for adopting, sl@0: when attaching a database. sl@0: sl@0: Usage of the IPC call arguments: sl@0: - Arg 0: [out] The 32 bits of the argument are used as follow: sl@0: @code sl@0: MSB LSB sl@0: 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: 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: @endcode sl@0: Where: sl@0: @code sl@0: - "Ro" - read-only flag, true if the file is read-only; sl@0: - "F" - database file name length; sl@0: @endcode sl@0: - Arg 1: [out] db names buffer sl@0: - Arg 2: [out] file session handle sl@0: - Arg 3: [out] database file handle sl@0: */ sl@0: TInt TSqlFhAttachCmdFunctor::operator()(RFile64& aDbFile, TBool /*aCreated*/, TBool aReadOnly) sl@0: { sl@0: 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: HBufC8* buf = HBufC8::New(KMaxBufLen); sl@0: if(!buf) sl@0: { sl@0: return KErrNoMemory; sl@0: } sl@0: TPtr8 bufPtr = buf->Des(); sl@0: RDesWriteStream out(bufPtr); sl@0: TRAPD(err, SerializeToStreamL(out)); sl@0: __ASSERT_DEBUG(err == KErrNone, __SQLPANIC(ESqlPanicInternalError));//"Write to descriptor" streaming operatons can't fail sl@0: TUint32 arg0 = (TUint32)bufPtr.Length() | (aReadOnly ? 0x80000000 : 0); sl@0: TIpcArgs ipcArgs(arg0, &bufPtr); sl@0: err = aDbFile.TransferToServer(ipcArgs, 2, 3); sl@0: if(err == KErrNone) sl@0: { sl@0: err = iDbSession.SendReceive(ESqlSrvDbAttachFromHandle, ipcArgs); sl@0: } sl@0: delete buf; sl@0: return err; sl@0: } sl@0: sl@0: /** sl@0: Serializes TSqlFhAttachCmdFunctor object content to a stream (aStream parameter). sl@0: */ sl@0: void TSqlFhAttachCmdFunctor::SerializeToStreamL(RWriteStream& aStream) sl@0: { sl@0: aStream << iDbFileName; sl@0: aStream << iDbName; sl@0: aStream.CommitL(); sl@0: } sl@0: sl@0: /////////////////////////////////////////////////////////////////////////////////////////////////////////////// sl@0: sl@0: //Database type. sl@0: enum TDbType sl@0: { sl@0: EDbTypeUnknown, //The database resides outside the client's private data cage sl@0: EDbTypeClientPrivate //The database resides in the client's private data cage sl@0: }; sl@0: sl@0: /** sl@0: Returns the database type. sl@0: sl@0: @param aDbFileName Database file name, including the path. If it is a request for creating/opening sl@0: secure database, then the name format is :<[SID]database_file_name>. sl@0: sl@0: @return EDbTypeClientPrivate If the database resides in the client's private data cage; sl@0: EDbTypeUnknown This may be a database in the SQL server private data cage or somewhere else; sl@0: If the error code is less than 0, then the function cannot determine the database type sl@0: (the function was unable to connect the local file session instance). sl@0: sl@0: @internalComponent sl@0: */ sl@0: static TInt GetDbType(const TDesC& aDbFileName) sl@0: { sl@0: RFs fs; sl@0: TInt err = fs.Connect(); sl@0: if(err == KErrNone) sl@0: { sl@0: TFileName clientPrivatePath; sl@0: err = fs.PrivatePath(clientPrivatePath); sl@0: fs.Close(); sl@0: if(err == KErrNone) sl@0: { sl@0: TInt pos = aDbFileName.FindF(clientPrivatePath); sl@0: return (TUint)pos <= (TUint)KMaxDriveName ? EDbTypeClientPrivate : EDbTypeUnknown; sl@0: } sl@0: } sl@0: return err; sl@0: } sl@0: sl@0: /////////////////////////////////////////////////////////////////////////////////////////////////////////////// sl@0: ////////////////////// RSqlDbSession ///////////////////////////////////////////////// sl@0: /////////////////////////////////////////////////////////////////////////////////////////////////////////////// sl@0: sl@0: /** sl@0: Establishes a connection with the SQL server. sl@0: sl@0: The method initializes RSqlDbSession object establishing a connection with the SQL server. sl@0: If the connection has established successfully, the method sends a message to the server to sl@0: create or open (depending on aFunction parameter) a database file with aDbFileName name. sl@0: sl@0: @param aFunction Specifies which operation has to be performed: sl@0: ESqlSrvDbCreate - Create a shared non-secure or private secure database; sl@0: ESqlSrvDbCreateSecure - Create a shared secure database; sl@0: ESqlSrvDbOpen - Open a shared non-secure, shared secure or private secure database; sl@0: sl@0: @param aDbFileName Database file name, including the path. If it is a request for creating/opening sl@0: secure database, then the name format is :<[SID]database_file_name>. sl@0: @param aSecurityPolicyData Security policy data. Non-empty descriptor only for secure databases sl@0: @param aConfig the configuration string "PARAM=VALUE;...." sl@0: sl@0: @return KErrNone, the operation completed successfully; sl@0: KErrNoMemory, an out of memory condition has occured; sl@0: KErrBadName, bad database file name: zero length, directory name; sl@0: KErrAlreadyExists, database file already exists; sl@0: KErrNotReady, the drive does not exist or is not ready; sl@0: KErrInUse, the file has been opened already; sl@0: KErrNotFound, file not found; sl@0: KErrGeneral, no or invalid security policies (if the database to be opened is a secure database with bad security data); sl@0: KErrNotSupported, incompatible sql security version (if the database to be opened is a secure database with bad security data); sl@0: KErrPermissionDenied, the caller does not satisfy the relevant database security policies. sl@0: Note that database specific errors categorised as ESqlDbError, and sl@0: other system-wide error codes may also be returned. sl@0: KErrArgument invalid config string or config string length exceeds KSqlSrvMaxConfigStrLen. sl@0: sl@0: Usage of the IPC call arguments: sl@0: - Arg 0: [out] database file name length sl@0: - Arg 1: [out] database file name sl@0: - Arg 2: [out] security policies buffer length if aFunction is ESqlSrvDbCreateSecure sl@0: - Arg 3: [out] security policies buffer if aFunction is ESqlSrvDbCreateSecure sl@0: */ sl@0: TInt RSqlDbSession::Connect(TSqlSrvFunction aFunction, const TDesC& aDbFileName, const TDesC8& aSecurityPolicyData, const TDesC8* aConfig) sl@0: { sl@0: SQL_TRACE_SESSION(OstTraceExt2(TRACE_INTERNALS, RSQLDBSESSION_CONNECT_ENTRY, "Entry;0x%X;RSqlDbSession::Connect;aDbFileName=%S", (TUint)this, __SQLPRNSTR(aDbFileName))); sl@0: #ifdef SYSLIBS_TEST sl@0: const TInt KDefaultMsgBufLen = 4; sl@0: #else sl@0: const TInt KDefaultMsgBufLen = 128; sl@0: #endif sl@0: iLastErrorMessage = HBufC::New(KDefaultMsgBufLen); sl@0: if(!iLastErrorMessage) sl@0: { sl@0: return KErrNoMemory; sl@0: } sl@0: TInt err = DoCreateSession(); sl@0: if(err == KErrNone) sl@0: { sl@0: err = InitResourceTestData(); sl@0: if(err == KErrNone) sl@0: { sl@0: err = ::GetDbType(aDbFileName); sl@0: if(err >= 0) sl@0: { sl@0: if(err == EDbTypeClientPrivate) sl@0: {//The database is in the client's private data cage. Set err = KErrPermissionDenied. sl@0: err = KErrPermissionDenied; sl@0: } sl@0: else sl@0: {//Unknown database type. Try to connect. sl@0: err = DoConnect(aFunction, aDbFileName, aSecurityPolicyData, aConfig); sl@0: } sl@0: if(err == KErrPermissionDenied && aFunction != ESqlSrvDbCreateSecure) sl@0: { sl@0: //What do we have here now? - the operation is create/open non-secure, the error is KErrPermissionDenied. sl@0: //So, the dll will try now to create/open the database in the application's private data cage. sl@0: //What if the used database file name was "C:MyDb.db"!? - the dll will try to create/open sl@0: //"C:MyDb.db" in the application's data cage and there sl@0: //is a possibility that this operation my return a result, like KErrNotFound, which will hide the original sl@0: //error code (KErrPermissionDenied). sl@0: //The dll shall not try to create/open file in the application's data cage, if the format of the sl@0: //database file name is secure. sl@0: TParsePtrC parse(aDbFileName);//this call may panic if aDbFileName cannot be parsed, but it will panic the client sl@0: if(parse.PathPresent()) sl@0: { sl@0: //The caller or the SQL server has no enough rights to create or open the database with aDbFileName name. sl@0: //One of the reasons may be that the database file is in the caller's private data cage where the SQL sl@0: //server cannot create/open the database file. sl@0: //The SQL dll will try to create/open the database file on the client side and pass the file handle to the sl@0: //SQL server. sl@0: TSqlFhOpenCmdFunctor fhOpenCmdSender(*this, aDbFileName, aConfig); sl@0: err = CreateAndSendFileHandle(fhOpenCmdSender, sl@0: aFunction == ESqlSrvDbCreate ? RSqlDbSession::EDbfCreate : RSqlDbSession::EDbfOpen); sl@0: } sl@0: } sl@0: } sl@0: } sl@0: } sl@0: if(err != KErrNone) sl@0: { sl@0: Close(); sl@0: } sl@0: 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: return err; sl@0: } sl@0: sl@0: /** sl@0: Sends a command to the server to attach an existing database to the current connection. sl@0: sl@0: The database can be: sl@0: @code sl@0: - shared secure database; sl@0: - shared non-secure database; sl@0: - private secure database; sl@0: @endcode sl@0: sl@0: @param aDbFileName Database file name. sl@0: @param aDbName Logical database name. sl@0: sl@0: @return KErrNone, the operation completed successfully; sl@0: KErrNoMemory, an out of memory condition has occured; sl@0: KErrBadName, bad database file name: zero length, directory name; sl@0: KErrNotReady, the drive does not exist or is not ready; sl@0: KErrInUse, the file has been opened already; sl@0: KErrNotFound, file not found; sl@0: KErrGeneral, no or invalid security policies (if the database to be opened is a secure database with bad security data); sl@0: KErrNotSupported, incompatible sql security version (if the database to be opened is a secure database with bad security data); sl@0: KErrPermissionDenied, the caller does not satisfy the relevant database security policies. sl@0: Note that database specific errors categorised as ESqlDbError, and sl@0: other system-wide error codes may also be returned. sl@0: sl@0: Usage of the IPC call arguments: sl@0: Arg 0: [out] Database file name length. sl@0: Arg 1: [out] Database file name. sl@0: Arg 2: [out] Logical database name length. sl@0: Arg 3: [out] Logical database name. sl@0: */ sl@0: TInt RSqlDbSession::Attach(const TDesC& aDbFileName, const TDesC& aDbName) sl@0: { sl@0: TInt err = ::GetDbType(aDbFileName); sl@0: if(err >= 0) sl@0: { sl@0: if(err == EDbTypeClientPrivate) sl@0: {//The database is in the client's private data cage. Set err = KErrPermissionDenied. sl@0: err = KErrPermissionDenied; sl@0: } sl@0: else sl@0: {//Unknown database type. Try to attach. sl@0: err = SendReceive(ESqlSrvDbAttach, TIpcArgs(aDbFileName.Length(), &aDbFileName, aDbName.Length(), &aDbName)); sl@0: } sl@0: if(err == KErrPermissionDenied) sl@0: { sl@0: //What do we have here now? - the operation is "attach", the error is KErrPermissionDenied. sl@0: //Which means, it is (possibly) a secure database in the server private data cage and the caller sl@0: //failed to pass the security checks. sl@0: //So, the dll will try now to open the database assuming it is in the application's private data cage sl@0: //and pass the file and session handles to the server which will try to attach the database using the handles. sl@0: //What if the used database file name was "C:MyDb.db"!? (which means that the client's intention was sl@0: //to attach a public shared secure database) sl@0: //The dll will try to open "C:MyDb.db" in the application's data cage and there sl@0: //is a possibility that this operation may return a result, like KErrNotFound, which will sl@0: //hide the original error code (KErrPermissionDenied). sl@0: //The dll shall not try to attach a database from the application's data cage, if the format of the sl@0: //database file name is secure. sl@0: TParsePtrC parse(aDbFileName);//this call may panic if aDbFileName cannot be parsed, but it will panic the client sl@0: if(parse.PathPresent()) sl@0: { sl@0: TSqlFhAttachCmdFunctor fhAttachCmdSender(*this, aDbFileName, aDbName); sl@0: err = CreateAndSendFileHandle(fhAttachCmdSender, RSqlDbSession::EDbfOpen); sl@0: } sl@0: } sl@0: } sl@0: return err; sl@0: } sl@0: sl@0: /** sl@0: Copies a database. sl@0: sl@0: The method establishes a temporary connection with the SQL server and issues a request for copying sl@0: aSrcDbFileName database file to aDestDbFileName file. After the server completes the request sl@0: CopyDatabase() closes the connection. sl@0: sl@0: @param aSrcDbFileName Source database file name. sl@0: @param aDestDbFileName Destination database file name. sl@0: sl@0: @return KErrNone, the operation completed successfully; sl@0: KErrNoMemory, an out of memory condition has occured; sl@0: KErrBadName, bad database file name: zero length, directory name; sl@0: KErrAlreadyExists, target database file already exists; sl@0: KErrNotReady, the drive does not exist or is not ready; sl@0: KErrInUse, the file has been opened already; sl@0: KErrNotFound, file not found; sl@0: KErrPermissionDenied, the SID of the calling application does not match the SID of source or destination database. sl@0: Note that other system-wide error codes may also be returned. sl@0: sl@0: Usage of the IPC call arguments: sl@0: Arg 0: [out] source database file name length sl@0: Arg 1: [out] source database file name sl@0: Arg 2: [out] destination database file name length sl@0: Arg 3: [out] destination database file name sl@0: */ sl@0: TInt RSqlDbSession::CopyDatabase(const TDesC& aSrcDbFileName, const TDesC& aDestDbFileName) sl@0: { sl@0: RSqlDbSession sess; sl@0: TInt err = sess.DoCreateSession(); sl@0: if(err == KErrNone) sl@0: { sl@0: err = sess.InitResourceTestData(); sl@0: if(err == KErrNone) sl@0: { sl@0: err = sess.SendReceive(ESqlSrvDbCopy, TIpcArgs(aSrcDbFileName.Length(), &aSrcDbFileName, aDestDbFileName.Length(), &aDestDbFileName)); sl@0: } sl@0: sess.Close(); sl@0: } sl@0: return err; sl@0: } sl@0: sl@0: /** sl@0: Deletes a database. sl@0: sl@0: The database can be: sl@0: @code sl@0: - shared secure database; sl@0: - shared non-secure database; sl@0: - private secure database; sl@0: @endcode sl@0: sl@0: The method establishes a temporary connection with the SQL server and issues a "delete" request regarding sl@0: aDbFileName database file. After the server completes the request DeleteDatabase() closes the connection. sl@0: sl@0: @param aDbFileName Database file name. It must include the path if it is not a secure database. sl@0: sl@0: @return KErrNone, the operation completed successfully; sl@0: KErrNoMemory, an out of memory condition has occured; sl@0: KErrBadName, bad database file name: zero length, directory name; sl@0: KErrNotReady, the drive does not exist or is not ready; sl@0: KErrInUse, the file has been opened already; sl@0: KErrNotFound, file not found; sl@0: KErrAccessDenied, access to the database file is denied (it might be a read-only file); sl@0: KErrPermissionDenied, the SID of the calling application does not match the SID of the database. sl@0: Note that other system-wide error codes may also be returned. sl@0: sl@0: Usage of the IPC call arguments: sl@0: Arg 0: [out] database file name length sl@0: Arg 1: [out] database file name sl@0: */ sl@0: TInt RSqlDbSession::DeleteDatabase(const TDesC& aDbFileName) sl@0: { sl@0: RSqlDbSession sess; sl@0: TInt err = sess.DoCreateSession(); sl@0: if(err == KErrNone) sl@0: { sl@0: err = sess.InitResourceTestData(); sl@0: if(err == KErrNone) sl@0: { sl@0: err = ::GetDbType(aDbFileName); sl@0: if(err >= 0) sl@0: { sl@0: if(err == EDbTypeClientPrivate) sl@0: {//The database is in the client's private data cage. Set err = KErrPermissionDenied. sl@0: err = KErrPermissionDenied; sl@0: } sl@0: else sl@0: {//Unknown database type. Try to delete. sl@0: err = sess.SendReceive(ESqlSrvDbDelete, TIpcArgs(aDbFileName.Length(), &aDbFileName)); sl@0: } sl@0: if(err == KErrPermissionDenied) sl@0: { sl@0: //What do we have here now? - the operation is "delete db", the error is KErrPermissionDenied. sl@0: //So, the dll will try now to delete a database with the same name from the application's private data cage. sl@0: //What if the used database file name was "C:MyDb.db" and there is a such file in server's private sl@0: //data cage? - the dll will try to delete a file with name "C:MyDb.db" from the application's data cage sl@0: //and there is a possibility that this operation my pass or may return strange result, like KErrNotFound. sl@0: //Bith cases are not what the user would expect. sl@0: //The dll shall not try to delete a file from the application's data cage, if the format of the sl@0: //database file name is secure. sl@0: TParsePtrC parse(aDbFileName);//this call may panic if aDbFileName cannot be parsed, but it will panic the client sl@0: if(parse.PathPresent()) sl@0: { sl@0: //The caller or the SQL server has no enough rights to delete the database with aDbFileName name. sl@0: //One of the reasons may be that the database file is in the caller's private data cage where the SQL sl@0: //server cannot delete the database file. sl@0: //The SQL dll will try to delete the database file on the client side. sl@0: RFs fs; sl@0: err = fs.Connect(); sl@0: if(err == KErrNone) sl@0: { sl@0: err = fs.Delete(aDbFileName); sl@0: fs.Close(); sl@0: } sl@0: } sl@0: } sl@0: } sl@0: } sl@0: sess.Close(); sl@0: } sl@0: return err; sl@0: } sl@0: sl@0: /** sl@0: Retrieves a reference to the textual description of the error returned by the sl@0: most recent call to any of the functions: sl@0: - RSqlDatabase::Exec() sl@0: - RSqlStatement::Exec() sl@0: - RSqlStatement::Next() sl@0: - RSqlStatement::Reset() sl@0: sl@0: Note that the function can only return a reference to text for sl@0: database-specific type errors, i.e. those errors that are categorised as of sl@0: type ESqlDbError. sl@0: sl@0: If an error occurs during the retrieval of the last error message, the function silently ignores the error sl@0: and returns a NULL descriptor. sl@0: sl@0: @return A non-modifiable pointer descriptor representing the most recent error sl@0: message. Note that message may be NULL, i.e. the descriptor may have sl@0: zero length. sl@0: sl@0: Usage of the IPC call arguments: sl@0: Arg 0: [out] max length of the message buffer sl@0: Arg 1: [in/out] buffer for the last error message sl@0: */ sl@0: TPtrC RSqlDbSession::LastErrorMessage() sl@0: { sl@0: TPtr msg(iLastErrorMessage->Des()); sl@0: msg.Zero(); sl@0: TInt size = 0; sl@0: while((size = SendReceive(ESqlSrvLastErrorMsg, TIpcArgs(msg.MaxLength(), &msg))) > KSqlClientBufOverflowCode) sl@0: { sl@0: HBufC* newMsgBuf = iLastErrorMessage->ReAlloc(size - KSqlClientBufOverflowCode); sl@0: if(!newMsgBuf) sl@0: { sl@0: break; sl@0: } sl@0: iLastErrorMessage = newMsgBuf; sl@0: msg.Set(iLastErrorMessage->Des()); sl@0: } sl@0: return msg; sl@0: } sl@0: sl@0: /** sl@0: Returns the ROWID of the most recent successful INSERT into the database sl@0: from this database connection. sl@0: sl@0: @return >0, the ROWID of the most recent successful INSERT into the database sl@0: from this database connection; sl@0: 0, if no successful INSERTs have ever occurred from this database connection sl@0: <0, if one of the system-wide error codes is returned sl@0: sl@0: Usage of the IPC call arguments: sl@0: Arg 0: [in/out] the receiving buffer for the last inserted ROWID sl@0: */ sl@0: TInt64 RSqlDbSession::LastInsertedRowId() sl@0: { sl@0: TInt64 res; sl@0: TPtr8 ptr(reinterpret_cast (&res), sizeof(res)); sl@0: TInt err = SendReceive(ESqlSrvDbLastInsertedRowId, TIpcArgs(&ptr)); sl@0: return err == KErrNone ? res : err; sl@0: } sl@0: sl@0: /** sl@0: Retrieves the database security policies. sl@0: sl@0: @return A pointer to the created and internalized CSqlSecurityPolicy instance. sl@0: sl@0: @return KErrNone, the operation has completed successfully; sl@0: KErrNoMemory, an out of memory condition has occurred; sl@0: Note that the function may leave with some database specific errors categorised as sl@0: ESqlDbError or other system-wide error codes. sl@0: sl@0: Usage of the IPC call arguments: sl@0: Arg 0: [out] security policy buffer size sl@0: Arg 1: [in/out] buffer for the database security policies sl@0: */ sl@0: TInt RSqlDbSession::GetSecurityPolicy(RSqlBufFlat& aSecurityPolicyBuf) sl@0: { sl@0: TPtr8 ptr(aSecurityPolicyBuf.BufPtr()); sl@0: TInt rc = KErrNone; sl@0: while((rc = SendReceive(ESqlSrvDbGetSecurityPolicy, TIpcArgs(ptr.MaxLength(), &ptr))) > KSqlClientBufOverflowCode) sl@0: { sl@0: rc = aSecurityPolicyBuf.ReAlloc(rc - KSqlClientBufOverflowCode); sl@0: if(rc != KErrNone) sl@0: { sl@0: break; sl@0: } sl@0: ptr.Set(aSecurityPolicyBuf.BufPtr()); sl@0: } sl@0: return rc; sl@0: } sl@0: sl@0: /** sl@0: Closes the database and releases the connection with the database server. sl@0: */ sl@0: void RSqlDbSession::Close() sl@0: { sl@0: SQL_TRACE_SESSION(OstTraceExt2(TRACE_INTERNALS, RSQLDBSESSION_CLOSE, "0x%X;RSqlDbSession::Close;handle=0x%X", (TUint)this, (TUint)Handle())); sl@0: if(Handle()) sl@0: { sl@0: (void)SendReceive(ESqlSrvDbClose); sl@0: } sl@0: TSqlResourceTestData::Release(); sl@0: delete iLastErrorMessage; sl@0: iLastErrorMessage = NULL; sl@0: RSessionBase::Close(); sl@0: } sl@0: sl@0: /** sl@0: The method establishes a connection with the SQL server. sl@0: sl@0: @return KErrNone, The connection was established successfully; sl@0: KErrAlreadyExists, the connection already exists. sl@0: The function may also return some other system-wide error codes. sl@0: */ sl@0: TInt RSqlDbSession::DoCreateSession() sl@0: { sl@0: const TInt KTimesToRetryConnection = 2; sl@0: TInt retry = KTimesToRetryConnection; sl@0: for(;;) sl@0: { sl@0: TInt err = CreateSession(KSqlSrvName, ::SqlSrvVersion()); sl@0: if(err != KErrNotFound && err != KErrServerTerminated) sl@0: { sl@0: return err; sl@0: } sl@0: if(--retry == 0) sl@0: { sl@0: return err; sl@0: } sl@0: err = ::StartSqlServer(); sl@0: if(err != KErrNone && err != KErrAlreadyExists) sl@0: { sl@0: return err; sl@0: } sl@0: } sl@0: } sl@0: sl@0: /** sl@0: The method gets called immediatelly after the establishing client-server connection but before the sl@0: create/open database message. sl@0: If a request was made by the client (using TSqlResourceTester methods) to test the server under out of memory sl@0: conditions, InitResourceTestData() will send this request to the server, putting the just created connection sl@0: in a "out of memory" test mode. sl@0: */ sl@0: TInt RSqlDbSession::InitResourceTestData() sl@0: { sl@0: TInt err = KErrNone; sl@0: #ifdef _DEBUG sl@0: TSqlResourceTestData* data = TSqlResourceTestData::Instance(); sl@0: if(data) sl@0: { sl@0: data->Init(*this); sl@0: } sl@0: else sl@0: { sl@0: err = KErrNoMemory; sl@0: } sl@0: #endif sl@0: return err; sl@0: } sl@0: sl@0: /** sl@0: Creates/opens database file locally and sends the file handle to the SQL server. sl@0: sl@0: This function is used only when the SQL server fails to create/open/attach the requested database file with sl@0: KErrPermissionDenied error. One of the reasons for that error may be that the database file is in the sl@0: calling application's private data cage. In this case an attempt is made to create/open the database file locally sl@0: and if the operation completes successfully, then the file handle is passed to the SQL server. sl@0: sl@0: @param aFhCmdSender A reference to a functor instance which is used for sending the file handles to the SQL server. sl@0: @param aDbfAction It is set by the caller to RSqlDbSession::EDbfCreate if the database file has to be created or sl@0: to RSqlDbSession::EDbfOpen if the database file already exists and has to be opened sl@0: sl@0: @return KErrNone, the operation completed successfully; sl@0: KErrNoMemory, an out of memory condition has occured; sl@0: KErrBadName, bad database file name: zero length, directory name, missing drive in the file name; sl@0: KErrAlreadyExists, database file already exists; sl@0: KErrNotReady, the drive does not exist or is not ready; sl@0: KErrInUse, the file has been opened already; sl@0: KErrNotFound, file not found; sl@0: KErrPermissionDenied, the caller does not satisfy the relevant database security policies. sl@0: Note that database specific errors categorised as ESqlDbError, and sl@0: other system-wide error codes may also be returned. sl@0: */ sl@0: TInt RSqlDbSession::CreateAndSendFileHandle(TSqlFhCmdFunctor& aFhCmdSender, RSqlDbSession::TDbfAction aDbfAction) sl@0: { sl@0: //Create a file session sl@0: RFs fs; sl@0: TInt err = fs.Connect(); sl@0: if(err != KErrNone) sl@0: { sl@0: return err; sl@0: } sl@0: //Share the file session sl@0: err = fs.ShareProtected(); sl@0: if(err != KErrNone) sl@0: { sl@0: fs.Close(); sl@0: return err; sl@0: } sl@0: //Create/open the database file locally sl@0: TBool readOnly = EFalse; sl@0: TBool fileCreated = EFalse; sl@0: RFile64 dbFile; sl@0: if(aDbfAction == RSqlDbSession::EDbfCreate) sl@0: {//Create the database file in R/W mode sl@0: err = dbFile.Create(fs, aFhCmdSender.iDbFileName, EFileShareAny | EFileWrite); sl@0: if(err == KErrNone) sl@0: { sl@0: fileCreated = ETrue; sl@0: } sl@0: } sl@0: else //aDbfAction == RSqlDbSession::EDbfOpen sl@0: {//Open the database file in shared R/W mode sl@0: err = dbFile.Open(fs, aFhCmdSender.iDbFileName, EFileShareAny | EFileWrite); sl@0: if(err != KErrNone) sl@0: {//If the the database open operation in R/W mode has failed - try to open the database in shared read-only mode. sl@0: readOnly = ETrue; sl@0: err = dbFile.Open(fs, aFhCmdSender.iDbFileName, EFileShareReadersOnly); sl@0: } sl@0: } sl@0: if(err == KErrNone) sl@0: {//Successful create/open database file operation. Send the database file and session handles to the server. sl@0: err = aFhCmdSender(dbFile, fileCreated, readOnly); sl@0: dbFile.Close(); sl@0: } sl@0: if(err != KErrNone && fileCreated) sl@0: { sl@0: dbFile.Close(); sl@0: TInt err2 = fs.Delete(aFhCmdSender.iDbFileName); sl@0: if(err2 != KErrNone) sl@0: {//Sometimes it is not possible the file to be deleted, the reported error is - KErrInUse. sl@0: fs.Close(); sl@0: User::After(100000); sl@0: err2 = fs.Connect(); sl@0: if(err2 == KErrNone) sl@0: { sl@0: (void)fs.Delete(aFhCmdSender.iDbFileName); sl@0: } sl@0: } sl@0: } sl@0: fs.Close(); sl@0: return err; sl@0: } sl@0: sl@0: /** sl@0: Prepares the "security & configuration" string and sends the command to the SQL server. sl@0: sl@0: @param aFunction Specifies which operation has to be performed: sl@0: ESqlSrvDbCreate - Create a shared non-secure or private secure database; sl@0: ESqlSrvDbCreateSecure - Create a shared secure database; sl@0: ESqlSrvDbOpen - Open a shared non-secure, shared secure or private secure database; sl@0: sl@0: @param aDbFileName Database file name, including the path. If it is a request for creating/opening sl@0: secure database, then the name format is :<[SID]database_file_name>. sl@0: @param aSecurityPolicyData Security policy data. Non-empty descriptor only for secure databases sl@0: @param aConfig the configuration string "PARAM=VALUE;...." sl@0: sl@0: @return KErrNone, the operation completed successfully; sl@0: KErrNoMemory, an out of memory condition has occured; sl@0: KErrArgument invalid config string or config string length exceeds KSqlSrvMaxConfigStrLen. sl@0: Other system-wide error codes may also be returned. sl@0: sl@0: */ sl@0: TInt RSqlDbSession::DoConnect(TSqlSrvFunction aFunction, const TDesC& aDbFileName, const TDesC8& aSecurityPolicyData, const TDesC8* aConfig) sl@0: { sl@0: const TInt KConfigStrLen = aConfig ? aConfig->Length() : 0; sl@0: if(KConfigStrLen > KSqlSrvMaxConfigStrLen) sl@0: { sl@0: return KErrArgument; sl@0: } sl@0: const TInt KSecPolicyLen = aFunction == ESqlSrvDbCreateSecure ? aSecurityPolicyData.Length() : 0; sl@0: TIpcArgs ipcArgs(aDbFileName.Length(), &aDbFileName); sl@0: // pack the length xxxxyyyy where xxxx is the policy length sl@0: // and yyyy is the config length. sl@0: ipcArgs.Set(2, (TUint)(KSecPolicyLen << 16) | (TUint)KConfigStrLen); sl@0: const TInt KTotalLen = KConfigStrLen + KSecPolicyLen; sl@0: ipcArgs.Set(3, 0); sl@0: HBufC8* arg3Buf = NULL; sl@0: if(KTotalLen > 0) sl@0: { sl@0: arg3Buf = HBufC8::New(KTotalLen); sl@0: if(!arg3Buf) sl@0: { sl@0: return KErrNoMemory; sl@0: } sl@0: TPtr8 arg3Ptr = arg3Buf->Des(); sl@0: if(KSecPolicyLen > 0) sl@0: { sl@0: arg3Ptr.Copy(aSecurityPolicyData); sl@0: } sl@0: if(KConfigStrLen > 0) sl@0: { sl@0: //coverity[DEADCODE] sl@0: //The ASSERT might be useful in catching future defect in this function sl@0: __ASSERT_DEBUG(aConfig != NULL, __SQLPANIC(ESqlPanicInternalError)); sl@0: arg3Ptr.Append(*aConfig); sl@0: } sl@0: ipcArgs.Set(3, &arg3Ptr); sl@0: } sl@0: TInt err = SendReceive(aFunction, ipcArgs); sl@0: delete arg3Buf; sl@0: return err; sl@0: }