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".
8 // Initial Contributors:
9 // Nokia Corporation - initial contribution.
18 #include <f32file64.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"
30 #include "SqlTraceDef.h"
32 ///////////////////////////////////////////////////////////////////////////////////////////////////////////////
33 ////////////////////// TSqlFhCmdFunctor ///////////////////////////////////////////////
34 ///////////////////////////////////////////////////////////////////////////////////////////////////////////////
37 TSqlFhCmdFunctor derived classes are used by RSqlDbSession for sending to SQL server localy created/opened
41 @see RSqlDbSession::CreateAndSendFileHandle()
45 NONSHARABLE_CLASS(TSqlFhCmdFunctor)
48 TSqlFhCmdFunctor(RSqlDbSession& aDbSession, const TDesC& aDbFileName, const TDesC8* aConfig=NULL) :
49 iDbSession(aDbSession),
50 iDbFileName(aDbFileName),
51 iConfig(aConfig) // ownership not transferred
54 virtual TInt operator()(RFile64& aFile, TBool aCreated, TBool aReadOnly) = 0;
57 RSqlDbSession& iDbSession;
58 const TDesC& iDbFileName;
59 const TDesC8* iConfig;
62 ///////////////////////////////////////////////////////////////////////////////////////////////////////////////
63 ////////////////////// TSqlFhOpenCmdFunctor //////////////////////////////////////////////
64 ///////////////////////////////////////////////////////////////////////////////////////////////////////////////
67 Implements the sending of ESqlSrvDbOpenFromHandle command to the SQL server.
71 NONSHARABLE_CLASS(TSqlFhOpenCmdFunctor) : public TSqlFhCmdFunctor
74 TSqlFhOpenCmdFunctor(RSqlDbSession& aDbSession, const TDesC& aDbFileName,const TDesC8* aConfig=NULL) :
75 TSqlFhCmdFunctor(aDbSession, aDbFileName, aConfig)
78 virtual TInt operator()(RFile64& aDbFile, TBool aCreated, TBool aReadOnly);
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.
86 Usage of the IPC call arguments:
87 - Arg 0: [out] The 32 bits of the argument are used as follow:
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
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;
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
104 TInt TSqlFhOpenCmdFunctor::operator()(RFile64& aDbFile, TBool aCreated, TBool aReadOnly)
106 const TInt KConfigStrLen = iConfig ? iConfig->Length() : 0;
107 if(KConfigStrLen > KSqlSrvMaxConfigStrLen)
112 HBufC* arg1Buf = HBufC::New(iDbFileName.Length() + KConfigStrLen);
117 TPtr arg1 = arg1Buf->Des();
118 arg1.Copy(iDbFileName);
121 TBuf<KSqlSrvMaxConfigStrLen> cfgBuf;
122 cfgBuf.Copy(*iConfig);
125 TUint32 arg0 = iDbFileName.Length() | (KConfigStrLen << 16);
134 ipcArgs.Set(0, arg0);
135 ipcArgs.Set(1, &arg1);
136 TInt err = aDbFile.TransferToServer(ipcArgs, 2, 3);
139 err = iDbSession.SendReceive(ESqlSrvDbOpenFromHandle, ipcArgs);
145 ///////////////////////////////////////////////////////////////////////////////////////////////////////////////
146 ////////////////////// TSqlFhAttachCmdFunctor ////////////////////////////////////////////
147 ///////////////////////////////////////////////////////////////////////////////////////////////////////////////
150 Implements the sending of ESqlSrvDbAttachFromHandle command to the SQL server.
154 NONSHARABLE_CLASS(TSqlFhAttachCmdFunctor) : public TSqlFhCmdFunctor
157 TSqlFhAttachCmdFunctor(RSqlDbSession& aDbSession, const TDesC& aDbFileName,
158 const TDesC& aDbName) :
159 TSqlFhCmdFunctor(aDbSession, aDbFileName),
163 virtual TInt operator()(RFile64& aDbFile, TBool aCreated, TBool aReadOnly);
166 void SerializeToStreamL(RWriteStream& aStream);
169 const TDesC& iDbName;
174 Sends a command to the SQL server to use the supplied file session and file handles for adopting,
175 when attaching a database.
177 Usage of the IPC call arguments:
178 - Arg 0: [out] The 32 bits of the argument are used as follow:
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
186 - "Ro" - read-only flag, true if the file is read-only;
187 - "F" - database file name length;
189 - Arg 1: [out] db names buffer
190 - Arg 2: [out] file session handle
191 - Arg 3: [out] database file handle
193 TInt TSqlFhAttachCmdFunctor::operator()(RFile64& aDbFile, TBool /*aCreated*/, TBool aReadOnly)
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);
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);
210 err = iDbSession.SendReceive(ESqlSrvDbAttachFromHandle, ipcArgs);
217 Serializes TSqlFhAttachCmdFunctor object content to a stream (aStream parameter).
219 void TSqlFhAttachCmdFunctor::SerializeToStreamL(RWriteStream& aStream)
221 aStream << iDbFileName;
226 ///////////////////////////////////////////////////////////////////////////////////////////////////////////////
231 EDbTypeUnknown, //The database resides outside the client's private data cage
232 EDbTypeClientPrivate //The database resides in the client's private data cage
236 Returns the database type.
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>.
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).
248 static TInt GetDbType(const TDesC& aDbFileName)
251 TInt err = fs.Connect();
254 TFileName clientPrivatePath;
255 err = fs.PrivatePath(clientPrivatePath);
259 TInt pos = aDbFileName.FindF(clientPrivatePath);
260 return (TUint)pos <= (TUint)KMaxDriveName ? EDbTypeClientPrivate : EDbTypeUnknown;
266 ///////////////////////////////////////////////////////////////////////////////////////////////////////////////
267 ////////////////////// RSqlDbSession /////////////////////////////////////////////////
268 ///////////////////////////////////////////////////////////////////////////////////////////////////////////////
271 Establishes a connection with the SQL server.
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.
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;
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;...."
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.
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
307 TInt RSqlDbSession::Connect(TSqlSrvFunction aFunction, const TDesC& aDbFileName, const TDesC8& aSecurityPolicyData, const TDesC8* aConfig)
309 SQL_TRACE_SESSION(OstTraceExt2(TRACE_INTERNALS, RSQLDBSESSION_CONNECT_ENTRY, "Entry;0x%X;RSqlDbSession::Connect;aDbFileName=%S", (TUint)this, __SQLPRNSTR(aDbFileName)));
311 const TInt KDefaultMsgBufLen = 4;
313 const TInt KDefaultMsgBufLen = 128;
315 iLastErrorMessage = HBufC::New(KDefaultMsgBufLen);
316 if(!iLastErrorMessage)
320 TInt err = DoCreateSession();
323 err = InitResourceTestData();
326 err = ::GetDbType(aDbFileName);
329 if(err == EDbTypeClientPrivate)
330 {//The database is in the client's private data cage. Set err = KErrPermissionDenied.
331 err = KErrPermissionDenied;
334 {//Unknown database type. Try to connect.
335 err = DoConnect(aFunction, aDbFileName, aSecurityPolicyData, aConfig);
337 if(err == KErrPermissionDenied && aFunction != ESqlSrvDbCreateSecure)
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())
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
355 TSqlFhOpenCmdFunctor fhOpenCmdSender(*this, aDbFileName, aConfig);
356 err = CreateAndSendFileHandle(fhOpenCmdSender,
357 aFunction == ESqlSrvDbCreate ? RSqlDbSession::EDbfCreate : RSqlDbSession::EDbfOpen);
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()));
372 Sends a command to the server to attach an existing database to the current connection.
376 - shared secure database;
377 - shared non-secure database;
378 - private secure database;
381 @param aDbFileName Database file name.
382 @param aDbName Logical database name.
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.
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.
402 TInt RSqlDbSession::Attach(const TDesC& aDbFileName, const TDesC& aDbName)
404 TInt err = ::GetDbType(aDbFileName);
407 if(err == EDbTypeClientPrivate)
408 {//The database is in the client's private data cage. Set err = KErrPermissionDenied.
409 err = KErrPermissionDenied;
412 {//Unknown database type. Try to attach.
413 err = SendReceive(ESqlSrvDbAttach, TIpcArgs(aDbFileName.Length(), &aDbFileName, aDbName.Length(), &aDbName));
415 if(err == KErrPermissionDenied)
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())
432 TSqlFhAttachCmdFunctor fhAttachCmdSender(*this, aDbFileName, aDbName);
433 err = CreateAndSendFileHandle(fhAttachCmdSender, RSqlDbSession::EDbfOpen);
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.
447 @param aSrcDbFileName Source database file name.
448 @param aDestDbFileName Destination database file name.
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.
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
466 TInt RSqlDbSession::CopyDatabase(const TDesC& aSrcDbFileName, const TDesC& aDestDbFileName)
469 TInt err = sess.DoCreateSession();
472 err = sess.InitResourceTestData();
475 err = sess.SendReceive(ESqlSrvDbCopy, TIpcArgs(aSrcDbFileName.Length(), &aSrcDbFileName, aDestDbFileName.Length(), &aDestDbFileName));
487 - shared secure database;
488 - shared non-secure database;
489 - private secure database;
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.
495 @param aDbFileName Database file name. It must include the path if it is not a secure database.
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.
507 Usage of the IPC call arguments:
508 Arg 0: [out] database file name length
509 Arg 1: [out] database file name
511 TInt RSqlDbSession::DeleteDatabase(const TDesC& aDbFileName)
514 TInt err = sess.DoCreateSession();
517 err = sess.InitResourceTestData();
520 err = ::GetDbType(aDbFileName);
523 if(err == EDbTypeClientPrivate)
524 {//The database is in the client's private data cage. Set err = KErrPermissionDenied.
525 err = KErrPermissionDenied;
528 {//Unknown database type. Try to delete.
529 err = sess.SendReceive(ESqlSrvDbDelete, TIpcArgs(aDbFileName.Length(), &aDbFileName));
531 if(err == KErrPermissionDenied)
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())
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.
552 err = fs.Delete(aDbFileName);
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()
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
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.
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
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
587 TPtrC RSqlDbSession::LastErrorMessage()
589 TPtr msg(iLastErrorMessage->Des());
592 while((size = SendReceive(ESqlSrvLastErrorMsg, TIpcArgs(msg.MaxLength(), &msg))) > KSqlClientBufOverflowCode)
594 HBufC* newMsgBuf = iLastErrorMessage->ReAlloc(size - KSqlClientBufOverflowCode);
599 iLastErrorMessage = newMsgBuf;
600 msg.Set(iLastErrorMessage->Des());
606 Returns the ROWID of the most recent successful INSERT into the database
607 from this database connection.
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
614 Usage of the IPC call arguments:
615 Arg 0: [in/out] the receiving buffer for the last inserted ROWID
617 TInt64 RSqlDbSession::LastInsertedRowId()
620 TPtr8 ptr(reinterpret_cast <TUint8*> (&res), sizeof(res));
621 TInt err = SendReceive(ESqlSrvDbLastInsertedRowId, TIpcArgs(&ptr));
622 return err == KErrNone ? res : err;
626 Retrieves the database security policies.
628 @return A pointer to the created and internalized CSqlSecurityPolicy instance.
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.
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
639 TInt RSqlDbSession::GetSecurityPolicy(RSqlBufFlat& aSecurityPolicyBuf)
641 TPtr8 ptr(aSecurityPolicyBuf.BufPtr());
643 while((rc = SendReceive(ESqlSrvDbGetSecurityPolicy, TIpcArgs(ptr.MaxLength(), &ptr))) > KSqlClientBufOverflowCode)
645 rc = aSecurityPolicyBuf.ReAlloc(rc - KSqlClientBufOverflowCode);
650 ptr.Set(aSecurityPolicyBuf.BufPtr());
656 Closes the database and releases the connection with the database server.
658 void RSqlDbSession::Close()
660 SQL_TRACE_SESSION(OstTraceExt2(TRACE_INTERNALS, RSQLDBSESSION_CLOSE, "0x%X;RSqlDbSession::Close;handle=0x%X", (TUint)this, (TUint)Handle()));
663 (void)SendReceive(ESqlSrvDbClose);
665 TSqlResourceTestData::Release();
666 delete iLastErrorMessage;
667 iLastErrorMessage = NULL;
668 RSessionBase::Close();
672 The method establishes a connection with the SQL server.
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.
678 TInt RSqlDbSession::DoCreateSession()
680 const TInt KTimesToRetryConnection = 2;
681 TInt retry = KTimesToRetryConnection;
684 TInt err = CreateSession(KSqlSrvName, ::SqlSrvVersion());
685 if(err != KErrNotFound && err != KErrServerTerminated)
693 err = ::StartSqlServer();
694 if(err != KErrNone && err != KErrAlreadyExists)
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.
708 TInt RSqlDbSession::InitResourceTestData()
712 TSqlResourceTestData* data = TSqlResourceTestData::Instance();
726 Creates/opens database file locally and sends the file handle to the SQL server.
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.
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
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.
748 TInt RSqlDbSession::CreateAndSendFileHandle(TSqlFhCmdFunctor& aFhCmdSender, RSqlDbSession::TDbfAction aDbfAction)
750 //Create a file session
752 TInt err = fs.Connect();
757 //Share the file session
758 err = fs.ShareProtected();
764 //Create/open the database file locally
765 TBool readOnly = EFalse;
766 TBool fileCreated = EFalse;
768 if(aDbfAction == RSqlDbSession::EDbfCreate)
769 {//Create the database file in R/W mode
770 err = dbFile.Create(fs, aFhCmdSender.iDbFileName, EFileShareAny | EFileWrite);
776 else //aDbfAction == RSqlDbSession::EDbfOpen
777 {//Open the database file in shared R/W mode
778 err = dbFile.Open(fs, aFhCmdSender.iDbFileName, EFileShareAny | EFileWrite);
780 {//If the the database open operation in R/W mode has failed - try to open the database in shared read-only mode.
782 err = dbFile.Open(fs, aFhCmdSender.iDbFileName, EFileShareReadersOnly);
786 {//Successful create/open database file operation. Send the database file and session handles to the server.
787 err = aFhCmdSender(dbFile, fileCreated, readOnly);
790 if(err != KErrNone && fileCreated)
793 TInt err2 = fs.Delete(aFhCmdSender.iDbFileName);
795 {//Sometimes it is not possible the file to be deleted, the reported error is - KErrInUse.
801 (void)fs.Delete(aFhCmdSender.iDbFileName);
810 Prepares the "security & configuration" string and sends the command to the SQL server.
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;
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;...."
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.
828 TInt RSqlDbSession::DoConnect(TSqlSrvFunction aFunction, const TDesC& aDbFileName, const TDesC8& aSecurityPolicyData, const TDesC8* aConfig)
830 const TInt KConfigStrLen = aConfig ? aConfig->Length() : 0;
831 if(KConfigStrLen > KSqlSrvMaxConfigStrLen)
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;
842 HBufC8* arg3Buf = NULL;
845 arg3Buf = HBufC8::New(KTotalLen);
850 TPtr8 arg3Ptr = arg3Buf->Des();
851 if(KSecPolicyLen > 0)
853 arg3Ptr.Copy(aSecurityPolicyData);
855 if(KConfigStrLen > 0)
858 //The ASSERT might be useful in catching future defect in this function
859 __ASSERT_DEBUG(aConfig != NULL, __SQLPANIC(ESqlPanicInternalError));
860 arg3Ptr.Append(*aConfig);
862 ipcArgs.Set(3, &arg3Ptr);
864 TInt err = SendReceive(aFunction, ipcArgs);