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 "SqlSrvFileData.h" //TSqlSrvFileData sl@0: #include "SqlSrvMain.h" //CSqlServer sl@0: #include "SqlSrvAuthorizer.h" //MSqlPolicyInspector sl@0: #include "SqlSrvDatabase.h" sl@0: #include "SqlSrvStatement.h" sl@0: #include "SqlUtil.h" //Panic codes, Sql2OsErrCode() sl@0: #include "SqlSrvUtil.h" //Global server functions sl@0: #include "SqlCompact.h" sl@0: #include "SqlSrvResourceProfiler.h" sl@0: #include "OstTraceDefinitions.h" sl@0: #ifdef OST_TRACE_COMPILER_IN_USE sl@0: #include "SqlSrvDatabaseTraces.h" sl@0: #endif sl@0: #include "SqlTraceDef.h" sl@0: sl@0: // sl@0: // The following macro disables the creation/loading of the settings table. sl@0: // It is for internal testing purposes only! sl@0: // sl@0: // __SQL_DISABLE_SYMBIAN_SETTINGS_TABLE__ sl@0: // sl@0: // This means that the database index will always be rebuilt when loaded by sl@0: // by the server regardless of the current system collation/locale in use. sl@0: // The benefit of enabling this macro is that a client can send PRAGMA sl@0: // commands to the SQL server before any tables have been explicity created in sl@0: // a NON-SECURE database (secure databases still automatically get a security sl@0: // policy table). sl@0: // sl@0: // The macro is applied inside: sl@0: // CSqlSrvDatabase::StoreSettingsL sl@0: // CSqlSrvDatabase::ProcessSettingsL sl@0: // sl@0: // We should inform the user at compile-time if this macro has been enabled: sl@0: #if defined(__SQL_DISABLE_SYMBIAN_SETTINGS_TABLE__) sl@0: #pragma message(">>> WARNING: Use of SYMBIAN_SETTINGS table has been disabled <<<") sl@0: #endif sl@0: sl@0: sl@0: /////////////////////////////////////////////////////////////////////////////////////////////////////////////// sl@0: ///////////////////////////// Local const data /////////////////////////////////////////////////////// sl@0: /////////////////////////////////////////////////////////////////////////////////////////////////////////////// sl@0: sl@0: const char* KErrMsg1 = "Missing ESCAPE expression"; sl@0: const char* KErrMsg2 = "ESCAPE expression must be a single character"; sl@0: sl@0: //////////////////////////////////////////////////////// sl@0: //Attach/detach SQL statements (zero-terminated strings) sl@0: _LIT(KAttachDb, "ATTACH DATABASE :FileName AS :DbName\x0"); sl@0: _LIT(KDetachDb, "DETACH DATABASE :DbName\x0"); sl@0: //////////////////////////////////////////////////////// sl@0: // Pragma SQL statements. The database names in all statements are quoted to avoid the "sql injection" threat. sl@0: // (At the moment there is no way to pass an invalid database name for these statements, because the database has to be attached sl@0: // first and a parameter binding is used there. So, the quoting is just for safety and against changes in the future) sl@0: _LIT(KCacheSizePragma, "PRAGMA \"%S\".cache_size=%d\x0"); sl@0: _LIT(KPageSizePragma, "PRAGMA \"%S\".page_size=%d\x0"); sl@0: _LIT(KAutoVacuumPragma, "PRAGMA \"%S\".auto_vacuum=%d\x0"); sl@0: //_LIT(KPersist, "persist"); sl@0: //_LIT(KPersistentJournalPragma, "PRAGMA \"%S\".journal_mode=%S\x0"); sl@0: _LIT(KJournalSizeLimitPragma, "PRAGMA \"%S\".journal_size_limit=%d\x0"); sl@0: //////////////////////////////////////////////////////// sl@0: //"LIKE" - user defined function name sl@0: _LIT8(KStrLikeFuncName, "LIKE\x0"); sl@0: sl@0: /////////////////////////////////////////////////////////////////////////////////////////////////////////////// sl@0: ///////////////////////////// Local functions /////////////////////////////////////////////////////// sl@0: /////////////////////////////////////////////////////////////////////////////////////////////////////////////// sl@0: sl@0: //Local function, used for comparing TSqlAttachDbPair objects. sl@0: //(TSqlAttachDbPair structure represents type of the objects, members sl@0: // of the map used for keeping the information regarding attached databases) sl@0: // sl@0: //Note that iKey members of aLeft and aRight function parameters are expected to be sl@0: //UTF8 encoded, zero-terminated strings. sl@0: // sl@0: //The function will panic with panic code 7 in _DEBUG mode if iKey member of aLeft or sl@0: //aRight argument is NULL. sl@0: static TInt Compare(const TSqlAttachDbPair& aLeft, const TSqlAttachDbPair& aRight) sl@0: { sl@0: __ASSERT_DEBUG(aLeft.iKey != NULL && aRight.iKey != NULL, __SQLPANIC2(ESqlPanicInternalError)); sl@0: return ::CompareNoCase8(TPtrC8(aLeft.iKey), TPtrC8(aRight.iKey)); sl@0: } sl@0: sl@0: //Local function, used for comparing TSqlCompactDbPair objects. sl@0: //(TSqlCompactDbPair structure represents type of the objects, members sl@0: // of the map used for keeping the information regarding compacted databases) sl@0: // sl@0: //Note that iKey members of aLeft and aRight function parameters are expected to be sl@0: //UTF16 encoded strings. sl@0: // sl@0: //The function will panic with panic code 7 in _DEBUG mode if iKey member of aLeft or sl@0: //aRight argument is NULL. sl@0: static TInt Compare2(const TSqlCompactDbPair& aLeft, const TSqlCompactDbPair& aRight) sl@0: { sl@0: __ASSERT_DEBUG(aLeft.iKey != NULL && aRight.iKey != NULL, __SQLPANIC2(ESqlPanicInternalError)); sl@0: return ::CompareNoCase(*aLeft.iKey, *aRight.iKey); sl@0: } sl@0: sl@0: //Creates/opens database file (database file name in aFileData parameter) and initializes aDbHandle parameter. sl@0: //The database will be created either with UTF-16 or UTF-8 encoding, depending on the sl@0: //TSqlSrvFileData::IsUTF16() property. sl@0: static void CreateDbHandleL(const TSqlSrvFileData& aFileData, sqlite3*& aDbHandle) sl@0: { sl@0: if(aFileData.ConfigParams().iDbEncoding == TSqlSrvConfigParams::EEncUtf8) sl@0: { sl@0: TBuf8 fileNameZ; sl@0: if(!::UTF16ZToUTF8Z(aFileData.FileNameZ(), fileNameZ)) sl@0: { sl@0: __SQLLEAVE2(KErrGeneral); sl@0: } sl@0: __SQLLEAVE_IF_ERROR2(::CreateDbHandle8(fileNameZ, aDbHandle)); sl@0: } sl@0: else sl@0: { sl@0: __SQLLEAVE_IF_ERROR2(::CreateDbHandle16(aFileData.FileNameZ(), aDbHandle)); sl@0: } sl@0: } sl@0: sl@0: //LoadAttachedDbSecurityPolicyLC() creates a new CSqlSecurityPolicy object and initializes it with the sl@0: //security policies read from the attached database file, which name is in aFileData parameter. sl@0: //The created database security policy object is placed in the cleanup stack. sl@0: //The caller is responsible for the destruction of the created and returned security policy object. sl@0: //This function is used to read the security policies of attached databases. sl@0: static CSqlSecurityPolicy* LoadAttachedDbSecurityPolicyLC(const TSqlSrvFileData& aFileData) sl@0: { sl@0: //Create new database security policy object and initialize it with a default security policy sl@0: TSecurityPolicy defaultPolicy(TSecurityPolicy::EAlwaysFail); sl@0: CSqlSecurityPolicy* dbPolicy = CSqlSecurityPolicy::NewLC(defaultPolicy); sl@0: //This is an attached database and has to be opened sl@0: sqlite3* dbHandle = NULL; sl@0: CreateDbHandleL(aFileData, dbHandle); sl@0: CleanupStack::PushL(TCleanupItem(&CloseDbCleanup, dbHandle)); sl@0: //Read the security policies. sl@0: TSqlDbSysSettings dbSysSettings(dbHandle); sl@0: dbSysSettings.LoadSecurityPolicyL(*dbPolicy); sl@0: CleanupStack::PopAndDestroy();//TCleanupItem(&CloseDbCleanup, dbHandle) sl@0: return dbPolicy; sl@0: } sl@0: sl@0: //LoadDbSecurityPolicyLC() creates a new CSqlSecurityPolicy object and initializes it with the sl@0: //security policies read from the database file. sl@0: //The created database security policy object is placed in the cleanup stack. sl@0: //The caller is responsible for destroying the returned database security policy object. sl@0: //The function is used to read the security policy of the main database. sl@0: static CSqlSecurityPolicy* LoadDbSecurityPolicyLC(sqlite3* aDbHandle) sl@0: { sl@0: __ASSERT_DEBUG(aDbHandle != NULL, __SQLPANIC2(ESqlPanicBadArgument)); sl@0: //Create new database security policy object and initialize it with a default security policy sl@0: TSecurityPolicy defaultPolicy(TSecurityPolicy::EAlwaysFail); sl@0: CSqlSecurityPolicy* dbPolicy = CSqlSecurityPolicy::NewLC(defaultPolicy); sl@0: //Read the security policies. sl@0: TSqlDbSysSettings dbSysSettings(aDbHandle); sl@0: dbSysSettings.LoadSecurityPolicyL(*dbPolicy); sl@0: return dbPolicy; sl@0: } sl@0: sl@0: //CreateStrCopyLC() makes a copy of aSrc string and places it in the cleanup stack. sl@0: //aSrc is expected to be UTF8 encoded, zero terminated string. sl@0: //The function panics in _DEBUG mode if aSrc is NULL. sl@0: static TUint8* CreateStrCopyLC(const TUint8* aSrc) sl@0: { sl@0: __ASSERT_DEBUG(aSrc != NULL, __SQLPANIC2(ESqlPanicBadArgument)); sl@0: TInt len = User::StringLength(aSrc) + 1; sl@0: TUint8* copy = new (ELeave) TUint8[len]; sl@0: Mem::Copy(copy, aSrc, len); sl@0: CleanupStack::PushL(copy); sl@0: return copy; sl@0: } sl@0: sl@0: //EnableAuthorizer() function is used to reenable the authorizer callback sl@0: //during the stack cleanup. sl@0: static void EnableAuthorizer(void* aAuthorizerDisabled) sl@0: { sl@0: __ASSERT_DEBUG(aAuthorizerDisabled != NULL, __SQLPANIC2(ESqlPanicBadArgument)); sl@0: TBool* authorizerDisabled = static_cast (aAuthorizerDisabled); sl@0: *authorizerDisabled = EFalse; sl@0: } sl@0: sl@0: //Used by DbFileCleanup() sl@0: NONSHARABLE_STRUCT(TDbFileCleanup) sl@0: { sl@0: TDbFileCleanup(const TSqlSrvFileData& aSqlSrvFileData, sqlite3*& aDbHandle) : sl@0: iSqlSrvFileData(aSqlSrvFileData), sl@0: iDbHandle(aDbHandle) sl@0: { sl@0: //aDbHandle can be NULL (it is a reference to a pointer and the pointer can be initialized later) sl@0: } sl@0: void Cleanup() sl@0: { sl@0: ::CloseDbHandle(iDbHandle);//This operation also deletes the journal file sl@0: iDbHandle = NULL; sl@0: (void)iSqlSrvFileData.Fs().Delete(iSqlSrvFileData.FileName()); sl@0: } sl@0: private: sl@0: const TSqlSrvFileData& iSqlSrvFileData; sl@0: sqlite3*& iDbHandle; sl@0: }; sl@0: sl@0: //DbFileCleanup() is used to close and delete the database file during the stack cleanup, if sl@0: //CSqlSrvDatabase's ConstructL() method(s) leave (when creating a new database file). sl@0: static void DbFileCleanup(void* aDbFileCleanup) sl@0: { sl@0: __ASSERT_DEBUG(aDbFileCleanup != NULL, __SQLPANIC2(ESqlPanicBadArgument)); sl@0: TDbFileCleanup* dbFileCleanup = static_cast (aDbFileCleanup); sl@0: dbFileCleanup->Cleanup(); sl@0: } sl@0: sl@0: //Executes "PRAGMA" SQL statement + INTEGER value. sl@0: //Pragma parameters: sl@0: // aValue - integer pragma value to be set; sl@0: // aPragma - pragma statement, the format is: ""PRAGMA .=%d\x0"" sl@0: // aDbName - "main" or the attached database name sl@0: //This function is used for setting "cache_size", "page_size", "auto_vacuum" pragmas. sl@0: //During the call the authorizer will be disabled. sl@0: static TInt ExecPragma(sqlite3* aDbHandle, TBool& aAuthorizerDisabled, const TDesC& aPragma, TInt aValue, const TDesC& aDbName = KMainDb16) sl@0: { sl@0: __SQLTRACE_INTERNALSEXPR(TPtrC pragmaprnptr(aPragma.Left(aPragma.Length() - 1))); sl@0: SQL_TRACE_INTERNALS(OstTraceExt4(TRACE_INTERNALS, EXECPRAGMA_ENTRY, "Entry;0;ExecPragma;sqlite3*=0x%X;aPragma=%S;aValue=%d;aDbName=%S", (TUint)aDbHandle, __SQLPRNSTR(pragmaprnptr), aValue, __SQLPRNSTR(aDbName))); sl@0: __ASSERT_DEBUG(aDbHandle != NULL, __SQLPANIC2(ESqlPanicBadArgument)); sl@0: TBuf pragmaSql;//(KMaxFileName + 64) characters buffer length is enough for the longest possible PRAGMA statement sl@0: pragmaSql.Format(aPragma, &aDbName, aValue); sl@0: TBool authorizerDisabledState = aAuthorizerDisabled; sl@0: aAuthorizerDisabled = ETrue; sl@0: TInt err = DbExecStmt16(aDbHandle, pragmaSql); sl@0: aAuthorizerDisabled = authorizerDisabledState; sl@0: SQL_TRACE_INTERNALS(OstTraceExt2(TRACE_INTERNALS, EXECPRAGMA_EXIT, "Exit;0;ExecPragma;sqlite3*=0x%X;err=%d", (TUint)aDbHandle, err)); sl@0: return err; sl@0: } sl@0: sl@0: //The journal size limit is set to be at lest 16 pages and no less than 64 Kb. sl@0: static void SetJournalSizeLimitL(sqlite3* aDbHandle, TBool& aAuthorizerDisabled, TInt aPageSize, const TDesC& aDbName = KMainDb16) sl@0: { sl@0: __ASSERT_DEBUG(aDbHandle != NULL, __SQLPANIC2(ESqlPanicBadArgument)); sl@0: if(aPageSize == TSqlSrvConfigParams::KConfigPrmValueNotSet) sl@0: { sl@0: __SQLLEAVE_IF_ERROR2(DbPageSize(aDbHandle, aDbName, aPageSize)); sl@0: } sl@0: const TInt KPageMultiplier = 16; sl@0: const TInt KDefaultJournalSizeLimit = 64 * 1024; sl@0: const TInt KMaxJournalSizeLimit = 512 * 1024; sl@0: const TInt KJournalSizeLimit = Min((aPageSize * KPageMultiplier), KMaxJournalSizeLimit); sl@0: if(KJournalSizeLimit > KDefaultJournalSizeLimit) sl@0: { sl@0: __SQLLEAVE_IF_ERROR2(::ExecPragma(aDbHandle, aAuthorizerDisabled, KJournalSizeLimitPragma, KJournalSizeLimit)); sl@0: } sl@0: } sl@0: sl@0: ////////////////////////////////////////////////////////////////////////////////////////////////////// sl@0: ///////////////////////////// CSqlSrvDatabase class ///////////////////////////////////////////// sl@0: ////////////////////////////////////////////////////////////////////////////////////////////////////// sl@0: sl@0: sl@0: sl@0: ///////////////////////////// Object creation methods ///////////////////////////////////////////// sl@0: sl@0: /** sl@0: Creates new CSqlSrvDatabase instance which creates and manages new secure SQL database. sl@0: sl@0: @param aFileData A reference to a TSqlSrvFileData instance, which keeps the database name, a sl@0: file session reference and some other database file related properties. sl@0: The format of the name must be: sl@0: \:\<[SID]database file name excluding the path\>. sl@0: "[SID]" refers to SID of the application which creates the database. sl@0: @param aSecurityPolicy The database security policies container. sl@0: CSqlSrvDatabase instance stores the pointer in the security policy map, sl@0: if it does not exist there already. The security policies map is owned by the CSqlServer class. sl@0: sl@0: @return A pointer to the created CSqlSrvDatabase instance. sl@0: sl@0: @see CSqlServer sl@0: @see TSqlSrvFileData sl@0: @see MSqlPolicyInspector sl@0: @see RSqlSecurityMap sl@0: @see CSqlSecurityPolicy sl@0: sl@0: @leave KErrNoMemory, an out of memory condition has occurred; sl@0: KErrArgument, if a system table name found in the security policies (aSecurityPolicy); sl@0: KErrAlreadyExists, the file already exists; sl@0: KErrNotReady, the drive does not exist or is not ready; sl@0: KErrInUse, the file is already open; sl@0: KErrPermissionDenied, the client application does not satisfy the relevant database security policies. sl@0: Note that the function may also leave with some other database specific sl@0: errors categorised as ESqlDbError, and other system-wide error codes. sl@0: sl@0: @panic SqlDb 4 In _DEBUG mode if aFileData does not refer to a secure database file name. sl@0: @panic SqlDb 4 In _DEBUG mode if aSecurityPolicy is NULL. sl@0: */ sl@0: CSqlSrvDatabase* CSqlSrvDatabase::CreateSecureL(const TSqlSrvFileData& aFileData, CSqlSecurityPolicy* aSecurityPolicy) sl@0: { sl@0: SQL_TRACE_INTERNALS(OstTrace0(TRACE_INTERNALS, CSQLSRVDATABASE_CREATESECUREL_ENTRY, "Entry;0;CSqlSrvDatabase::CreateSecureL")); sl@0: __ASSERT_DEBUG(aFileData.IsSecureFileNameFmt(), __SQLPANIC2(ESqlPanicBadArgument)); sl@0: __ASSERT_DEBUG(aSecurityPolicy != NULL, __SQLPANIC2(ESqlPanicBadArgument)); sl@0: if(!::SqlServer().SecurityInspector().Check(aSecurityPolicy->DbPolicy(RSqlSecurityPolicy::ESchemaPolicy))) sl@0: { sl@0: //The caller has no "SCHEMA" policy. Then the client is not given a permission to create the database. sl@0: //Delete aSecurityPolicy since no database object is going to be created and the security policy object sl@0: //won't be put in the security policies map. sl@0: delete aSecurityPolicy; sl@0: __SQLLEAVE2(KErrPermissionDenied); sl@0: } sl@0: //What does happen with aSecurityPolicy instance? sl@0: // If the database is created successfully, then a lookup will be made in the security policies map. sl@0: // (the security policies map contains reference counted security policies) sl@0: // If the same security policy already exists in the map, then aSecurityPolicy will be deleted. sl@0: // The reference counter of the found policy will be incremented. sl@0: // If aSecurityPolicy is not in the map, then it will be put in the map and removed sl@0: // from the map when CSqlSrvDatabase object is destroyed. sl@0: // (the "remove" operation decrements the security policy counter sl@0: // and destroys the policy if it reaches 0) sl@0: // sl@0: //The security policy map pair is: sl@0: //{secure_db_name, reference_counted db_security_policy} sl@0: //The security policy is reference counted, because the same database can be shared between one or more sl@0: //connections (clients), in which case only a single, reference counted instance of the database security sl@0: //policy is kept in the map. sl@0: CleanupStack::PushL(aSecurityPolicy); sl@0: CSqlSrvDatabase* self = new (ELeave) CSqlSrvDatabase(); sl@0: CleanupStack::Pop(aSecurityPolicy); sl@0: CleanupStack::PushL(self); sl@0: self->ConstructCreateSecureL(aFileData, aSecurityPolicy); sl@0: CleanupStack::Pop(self); sl@0: SQL_TRACE_INTERNALS(OstTraceExt2(TRACE_INTERNALS, CSQLSRVDATABASE_CREATESECUREL_EXIT, "Exit;0x%X;CSqlSrvDatabase::CreateSecureL;sqlite3*=0x%X", (TUint)self, (TUint)self->iDbHandle)); sl@0: return self; sl@0: } sl@0: sl@0: /** sl@0: Creates new CSqlSrvDatabase instance which creates and manages new SQL database. sl@0: sl@0: @param aFileData A reference to a TSqlSrvFileData instance, which keeps the database name, a sl@0: file session reference and some other database file related properties. sl@0: sl@0: @return A pointer to the created CSqlSrvDatabase instance. sl@0: sl@0: @see CSqlServer sl@0: @see TSqlSrvFileData sl@0: @see MSqlPolicyInspector sl@0: @see RSqlSecurityMap sl@0: @see CSqlSecurityPolicy sl@0: sl@0: @leave KErrNoMemory, an out of memory condition has occurred; sl@0: KErrArgument, the file name is a name of a secure database; sl@0: KErrAlreadyExists, the file already exists; sl@0: KErrNotReady, the drive does not exist or is not ready; sl@0: KErrInUse, the file is already open; sl@0: KErrArgument, the file name refers to a secure database, but aSecurityPolicy is NULL; sl@0: Note that the function may also leave with some other database specific sl@0: errors categorised as ESqlDbError, and other system-wide error codes. sl@0: sl@0: @panic SqlDb 4 In _DEBUG mode if aFileData refers to a secure database file name. sl@0: */ sl@0: CSqlSrvDatabase* CSqlSrvDatabase::CreateL(const TSqlSrvFileData& aFileData) sl@0: { sl@0: SQL_TRACE_INTERNALS(OstTrace0(TRACE_INTERNALS, CSQLSRVDATABASE_CREATEL_ENTRY, "Entry;0;CSqlSrvDatabase::CreateL")); sl@0: __ASSERT_DEBUG(!aFileData.IsSecureFileNameFmt(), __SQLPANIC2(ESqlPanicBadArgument)); sl@0: CSqlSrvDatabase* self = new (ELeave) CSqlSrvDatabase(); sl@0: CleanupStack::PushL(self); sl@0: self->ConstructCreateL(aFileData); sl@0: CleanupStack::Pop(self); sl@0: SQL_TRACE_INTERNALS(OstTraceExt2(TRACE_INTERNALS, CSQLSRVDATABASE_CREATEL_EXIT, "Exit;0x%X;CSqlSrvDatabase::CreateL;sqlite3*=0x%X", (TUint)self, (TUint)self->iDbHandle)); sl@0: return self; sl@0: } sl@0: sl@0: /** sl@0: Creates new CSqlSrvDatabase instance which opens and manages an existing SQL database. sl@0: sl@0: @param aFileData A reference to a TSqlSrvFileData instance, which keeps the database name, a sl@0: file session reference and some other database file related properties. sl@0: If this is a secure database, then the format of the name must be: sl@0: \:\<[SID]database file name excluding the path\>. sl@0: If this is a non-secure database, then the file name has to be the full database file name. sl@0: "[SID]" refers to SID of the application which created the database. sl@0: If this is application's private database, then the format of aFileData is as it is described sl@0: in TSqlSrvFileData::SetFromHandleL() comments. sl@0: sl@0: @return A pointer to the created CSqlSrvDatabase instance. sl@0: sl@0: @leave KErrNoMemory, an out of memory condition has occurred; sl@0: KErrNotReady, the drive does not exist or is not ready; sl@0: KErrNotFound, the database file does not exist; sl@0: KErrInUse, the file is already open; sl@0: KErrPermissionDenied, the client application does not satisfy the relevant database security policies. sl@0: Note that the function may also leave with some other database specific sl@0: errors categorised as ESqlDbError, and other system-wide error codes. sl@0: sl@0: @see CSqlServer sl@0: @see TSqlSrvFileData sl@0: @see MSqlPolicyInspector sl@0: @see RSqlSecurityMap sl@0: @see CSqlSecurityPolicy sl@0: @see TSqlSrvFileData::SetFromHandleL() sl@0: */ sl@0: CSqlSrvDatabase* CSqlSrvDatabase::OpenL(const TSqlSrvFileData& aFileData) sl@0: { sl@0: SQL_TRACE_INTERNALS(OstTrace0(TRACE_INTERNALS, CSQLSRVDATABASE_OPENL_ENTRY, "Entry;0;CSqlSrvDatabase::OpenL")); sl@0: CSqlSrvDatabase* self = new (ELeave) CSqlSrvDatabase(); sl@0: CleanupStack::PushL(self); sl@0: aFileData.IsSecureFileNameFmt() ? self->ConstructOpenSecureL(aFileData) : self->ConstructOpenL(aFileData); sl@0: CleanupStack::Pop(self); sl@0: SQL_TRACE_INTERNALS(OstTraceExt2(TRACE_INTERNALS, CSQLSRVDATABASE_OPENL_EXIT, "Exit;0x%X;CSqlSrvDatabase::OpenL;sqlite3*=0x%X", (TUint)self, (TUint)self->iDbHandle)); sl@0: return self; sl@0: } sl@0: sl@0: /** sl@0: Cleans up the allocated for the database connection memory and other resources. sl@0: */ sl@0: CSqlSrvDatabase::~CSqlSrvDatabase() sl@0: { sl@0: SQL_TRACE_INTERNALS(OstTraceExt2(TRACE_INTERNALS, CSQLSRVDATABASE_CSQLSRVDATABASE2_ENTRY, "Entry;0x%X;CSqlSrvDatabase::~CSqlSrvDatabase;sqlite3*=0x%X", (TUint)this, (TUint)iDbHandle)); sl@0: TSqlCompactDbMapIterator compactDbIt(iCompactDbMap); sl@0: TSqlCompactDbPair compactDbPair; sl@0: while(compactDbIt.Next(compactDbPair)) sl@0: { sl@0: __ASSERT_DEBUG(compactDbPair.iData, __SQLPANIC2(ESqlPanicInvalidObj)); sl@0: ::SqlServer().Compactor().ReleaseEntry(*compactDbPair.iData); sl@0: } sl@0: iCompactDbMap.Close(); sl@0: //If iSecureDbName is not NULL, the current CSqlSrvDatabase object operates on a secure database. sl@0: //The database security policy has to be removed from the security policy map. sl@0: //(The "remove" operation actually decrements the security policy reference counter and destroys the policy sl@0: //when the counter reaches 0. The counter value may be greater than 1 if the database is shared between sl@0: //more than one connection) sl@0: if(iSecureDbName) sl@0: { sl@0: ::SqlServer().SecurityMap().Remove(iSecureDbName); sl@0: //The security policy map owns iSecureDbName and iSecurityPolicy and is responsible for sl@0: //iSecureDbName and iSecurityPolicy destruction. sl@0: } sl@0: //The next step of the "resource release" process is to walk over iAttachDbMap entries, get the data part of sl@0: //the found TSqlAttachDbPair objects, which is secure database name used as a key in iSecurityMap, and remove sl@0: //the related entry from iSecurityMap. If the database is closed in normal circumstances, the iAttachDbMap sl@0: //has no entries. But if the database client forgets to detach the used attached databases or if the Detach() call sl@0: //fails (for example, with KErrNoMemory error), then at this point iAttachDbMap has one or more entries. sl@0: //That means there are still some attached databases to this connection. This is not a problem, SQLite will take sl@0: //care of them. The problem is that there are related entries in iSecurityMap map, owned by CSqlServer object, sl@0: //and they won't be removed from the map till CSqlServer object is alive. This causes problems in OOM tests and in sl@0: //real life of the device. sl@0: //For example, one database client opens "c:[11111111]a.db" and attaches "c:[11111111]b.db": sl@0: // - c:[11111111]a.db database has been opened successfully. iSecurityMap has 1 entry: sl@0: // {"c:[11111111]a.db", }. sl@0: // - c:[11111111]b.db is attached to c:[11111111]a.db with name "db2". There will be 1 entry in iAttachDbMap: sl@0: // {"db2", "c:[11111111]a.db"} sl@0: // and a new entry will be added to iSecurityMap: sl@0: // {"c:[11111111]b.db", }. 2 entries in total in iSecurityMap. sl@0: // - The database client attempts to detach the attached database but the opertaion fails during the execution sl@0: // of the DETACH sql statement. Both maps are intact. sl@0: // - The database client attempts to close the database. The previous implementation of CSqlSrvDatabase::~CSqlSrvDatabase() sl@0: // would only remove "c:[11111111]a.db" entry from iSecurityMap and close the iAttachDbMap map. sl@0: // The result: no opened or attached databases but there will be one entry in iSecurityMap: "c:[11111111]b.db". sl@0: // OOM tests will report a memory leak in this case. On a real device, if "c:[11111111]b.db" gets deleted and sl@0: // recreated again, unexpected memory leak will occur in CSqlSrvDatabase::ConstructCreateSecureL() because sl@0: // the code there "expects" that is the first time when a "c:[11111111]b.db" entry is added to iSecurityMap. sl@0: TSqlAttachDbMapIterator it(iAttachDbMap); sl@0: TSqlAttachDbPair attachDbPair; sl@0: while(it.Next(attachDbPair)) sl@0: { sl@0: __ASSERT_DEBUG(attachDbPair.iData, __SQLPANIC2(ESqlPanicInvalidObj)); sl@0: ::SqlServer().SecurityMap().Remove(attachDbPair.iData); sl@0: } sl@0: iAttachDbMap.Close(); sl@0: ::CloseDbHandle(iDbHandle); sl@0: SQL_TRACE_INTERNALS(OstTrace1(TRACE_INTERNALS, CSQLSRVDATABASE_CSQLSRVDATABASE2_EXIT, "Exit;0x%X;CSqlSrvDatabase::~CSqlSrvDatabase", (TUint)this)); sl@0: } sl@0: sl@0: /** sl@0: Initializes CSqlSrvDatabase data memebers with their default values. sl@0: sl@0: sl@0: @see MSqlPolicyInspector sl@0: @see RSqlSecurityMap sl@0: @see CSqlServer sl@0: */ sl@0: CSqlSrvDatabase::CSqlSrvDatabase() : sl@0: iAttachDbMap(TSqlAttachDbLinearOrder(&Compare), TSqlAttachDbDestructor()), sl@0: iCompactDbMap(TSqlCompactDbLinearOrder(&Compare2), TSqlCompactDbDestructor()) sl@0: { sl@0: } sl@0: sl@0: /** sl@0: Creates a new SQL database file and executes config pragmas. sl@0: This function is part of CSqlSrvDatabase instance initialization. sl@0: If the function leaves and the error is not KErrAlreadyExists, the database file will be deleted. sl@0: sl@0: @param aFileData A reference to a TSqlSrvFileData instance, which keeps the database name, a sl@0: file session reference and some other database file related properties. sl@0: If this is a secure database, then the format of the name must be: sl@0: \:\<[SID]database file name excluding the path\>. sl@0: If this is a non-secure database, then the file name has to be the full database file name. sl@0: "[SID]" refers to SID of the application which creates the database. sl@0: sl@0: @see TSqlSrvFileData sl@0: sl@0: @leave KErrNoMemory, an out of memory condition has occurred; sl@0: KErrAlreadyExists, the file already exists. sl@0: Note that the function may also leave with some other database specific sl@0: errors categorised as ESqlDbError, and other system-wide error codes. sl@0: */ sl@0: void CSqlSrvDatabase::CreateNewDbFileL(const TSqlSrvFileData& aFileData) sl@0: { sl@0: __SQLTRACE_INTERNALSVAR(TPtrC fname = aFileData.FileName()); sl@0: SQL_TRACE_INTERNALS(OstTraceExt2(TRACE_INTERNALS, CSQLSRVDATABASE_CREATENEWDBFILEL, "0x%x;CSqlSrvDatabase::CreateNewDbFileL;fname=%S", (TUint)this, __SQLPRNSTR(fname))); sl@0: if(::FileExists(aFileData.Fs(), aFileData.FileName())) sl@0: { sl@0: __SQLLEAVE(KErrAlreadyExists); sl@0: } sl@0: TDbFileCleanup dbFileCleanup(aFileData, iDbHandle); sl@0: CleanupStack::PushL(TCleanupItem(&DbFileCleanup, &dbFileCleanup)); sl@0: ::CreateDbHandleL(aFileData, iDbHandle); sl@0: SetConfigL(aFileData.ConfigParams(), ETrue); sl@0: CleanupStack::Pop();//DbFileCleanup sl@0: } sl@0: sl@0: /** sl@0: Opens an existing SQL database file and executes config pragmas. sl@0: sl@0: This function is part of CSqlSrvDatabase instance initialization. sl@0: sl@0: @param aFileData A reference to a TSqlSrvFileData instance, which keeps the database name, a sl@0: file session reference and some other database file related properties. sl@0: If this is a secure database, then the format of the name must be: sl@0: \:\<[SID]database file name excluding the path\>. sl@0: If this is a non-secure database, then the file name has to be the full database file name. sl@0: "[SID]" refers to SID of the application which creates the database. sl@0: If this is application's private database, then the format of aFileData is as it is described sl@0: in TSqlSrvFileData::SetFromHandleL() comments. sl@0: sl@0: @leave KErrNoMemory, an out of memory condition has occurred; sl@0: KErrNotFound, SQL database file not found. sl@0: Note that the function may also leave with some other database specific sl@0: errors categorised as ESqlDbError, and other system-wide error codes. sl@0: sl@0: @see TSqlSrvFileData sl@0: @see TSqlSrvFileData::SetFromHandleL() sl@0: */ sl@0: void CSqlSrvDatabase::OpenExistingDbFileL(const TSqlSrvFileData& aFileData) sl@0: { sl@0: __SQLTRACE_INTERNALSVAR(TPtrC fname = aFileData.FileName()); sl@0: SQL_TRACE_INTERNALS(OstTraceExt2(TRACE_INTERNALS, CSQLSRVDATABASE_OPENEXISTINGDBFILEL, "0x%x;CSqlSrvDatabase::OpenExistingDbFileL;fname=%S", (TUint)this, __SQLPRNSTR(fname))); sl@0: if(!aFileData.ContainHandles()) sl@0: {//This check is valid only if the database is outside application's private data cage sl@0: if(!::FileExists(aFileData.Fs(), aFileData.FileName())) sl@0: { sl@0: __SQLLEAVE(KErrNotFound); sl@0: } sl@0: } sl@0: ::CreateDbHandleL(aFileData, iDbHandle); sl@0: //this is an existing database, only the cache size can be changed, the page size is persistent database property. sl@0: //But for private databases, opened on the client side - the page size can be set (for the "create database" operations). sl@0: SetConfigL(aFileData.ConfigParams(), aFileData.ContainHandles()); sl@0: } sl@0: sl@0: /** sl@0: Installs an authorizer callback function. sl@0: sl@0: The callback function is invoked by the SQL parser at SQL statement compile time for each attempt sl@0: to access a column of a table in the database and is used to assert the calling application's rights to sl@0: perform specific SQL operation. sl@0: sl@0: This function is part of CSqlSrvDatabase instance initialization. sl@0: sl@0: @leave The function may leave with some database specific errors categorised as ESqlDbError. sl@0: */ sl@0: void CSqlSrvDatabase::InstallAuthorizerL() sl@0: { sl@0: //Install the authorizer just once. "Install authorizer" may be expensive and dangerous operation because sl@0: //it will expire the already prepared statements. sl@0: if(!iAuthorizerInstalled) sl@0: { sl@0: (void)sqlite3SymbianLastOsError();//clear last OS error sl@0: TInt err = sqlite3_set_authorizer(iDbHandle, &CSqlSrvDatabase::AuthorizeCallback, this); sl@0: __SQLLEAVE_IF_ERROR(::Sql2OsErrCode(err, sqlite3SymbianLastOsError())); sl@0: } sl@0: iAuthorizerInstalled = ETrue; sl@0: } sl@0: sl@0: #ifdef SYMBIAN_USE_SQLITE_VERSION_3_6_4 sl@0: extern "C" int sqlite3RegisterInternalUtf8Like(sqlite3 *db); sl@0: #endif sl@0: /** sl@0: Installs user-defined functions. At the moment there is only one: LIKE. sl@0: sl@0: Replacing the LIKE operator default implementation with user defined LIKE functions. sl@0: The default implementation need a replacement because it does not work correctly with UTF16 encoded strings. sl@0: */ sl@0: void CSqlSrvDatabase::InstallUDFsL() sl@0: { sl@0: //Registering user defined LIKE function with 2 parameters for UTF16 encoding sl@0: TInt err = sqlite3_create_function(iDbHandle, reinterpret_cast (KStrLikeFuncName().Ptr()), sl@0: 2/*arg*/, SQLITE_UTF16, this /*user data*/, sl@0: &CSqlSrvDatabase::LikeSqlFunc, NULL/*xStep*/, NULL/*xFinal*/); sl@0: __SQLLEAVE_IF_ERROR(::Sql2OsErrCode(err, sqlite3SymbianLastOsError())); sl@0: //Registering user defined LIKE function with 3 parameters for UTF16 encoding sl@0: err = sqlite3_create_function(iDbHandle, reinterpret_cast (KStrLikeFuncName().Ptr()), sl@0: 3/*arg*/, SQLITE_UTF16, this/*user data*/, sl@0: &CSqlSrvDatabase::LikeSqlFunc, NULL/*xStep*/, NULL/*xFinal*/); sl@0: __SQLLEAVE_IF_ERROR(::Sql2OsErrCode(err, sqlite3SymbianLastOsError())); sl@0: sl@0: #ifdef SYMBIAN_USE_SQLITE_VERSION_3_6_4 sl@0: //Registering user defined LIKE function with 2 and 3 parameters for UTF8 encoding sl@0: //Once registered, these will be treated as built-in functions sl@0: err = sqlite3RegisterInternalUtf8Like(iDbHandle); sl@0: __SQLLEAVE_IF_ERROR(::Sql2OsErrCode(err, sqlite3SymbianLastOsError())); sl@0: #endif sl@0: } sl@0: sl@0: /** sl@0: Constructs a key for searching in the security policies map. sl@0: sl@0: The key is UTF8 encoded, zero-terminated string. sl@0: sl@0: Every time when CSqlSrvDatabase instance creates, opens or attaches a secure database, it updates sl@0: the contents of the security policies map (RSqlSecurityMap class), which is a single instance owned sl@0: by the CSqlServer class. sl@0: sl@0: @param aDbFileName Full secure database file name, from which the security policies map key sl@0: will be constructed. sl@0: sl@0: @return A const pointer to the constructed security map key. No memory is allocated for the key. sl@0: sl@0: @leave KErrGeneral the UTF16 to UTF8 conversion of aDbFileName parameter failed. sl@0: sl@0: @see RSqlSecurityMap sl@0: @see CSqlServer sl@0: */ sl@0: const TUint8* CSqlSrvDatabase::SecurityMapKeyL(const TDesC& aDbFileName) sl@0: { sl@0: //Form the map item key - the secure database name sl@0: TParsePtrC parse(aDbFileName);//this call may panic if aDbFileName cannot be parsed. But aDbFileName was preprocessed by TSqlSrvFileData::Set sl@0: TFileName secureDbName; sl@0: secureDbName.Copy(parse.Drive()); sl@0: secureDbName.Append(parse.NameAndExt()); sl@0: secureDbName.Append(TChar(0)); sl@0: TPtr8 ptr(iFileNameBuf, sizeof(iFileNameBuf)); sl@0: if(!::UTF16ZToUTF8Z(secureDbName, ptr)) sl@0: { sl@0: __SQLLEAVE(KErrGeneral); sl@0: } sl@0: return iFileNameBuf; sl@0: } sl@0: sl@0: /** sl@0: Attaches a secure or non-secure database to the current database connection. sl@0: sl@0: @param aFileData A reference to a TSqlSrvFileData instance, which keeps the database name, a sl@0: file session reference and some other database file related properties. sl@0: If this is a secure database, then the format of the name must be: sl@0: \:\<[SID]database file name excluding the path\>. sl@0: If this is a non-secure database, then the file name has to be the full database file name. sl@0: "[SID]" refers to SID of the application which creates the database. sl@0: @param aDbName Database name. It must be unique (per database connection). This is the name which can sl@0: be used for accessing tables in the attached database. The syntax is "database-name.table-name". sl@0: sl@0: @leave KErrNotFound, the database file which has to be attached does not exist. sl@0: KErrPermissionDenied, the client application does not satisfy the relevant security policies of sl@0: the attached database. sl@0: Note that the function may also leave with some other database specific sl@0: errors categorised as ESqlDbError, and other system-wide error codes. sl@0: sl@0: @see TSqlSrvFileData sl@0: */ sl@0: void CSqlSrvDatabase::AttachDbL(const TSqlSrvFileData& aFileData, const TDesC& aDbName) sl@0: { sl@0: SQL_TRACE_INTERNALS(OstTraceExt2(TRACE_INTERNALS, CSQLSRVDATABASE_ATTACHDBL_ENTRY, "Entry;0x%X;CSqlSrvDatabase::AttachDbL;aDbName=%S", (TUint)this, __SQLPRNSTR(aDbName))); sl@0: if(!aFileData.ContainHandles()) sl@0: {//This check is valid only if the database is outside application's private data cage sl@0: if(!::FileExists(aFileData.Fs(), aFileData.FileName())) sl@0: { sl@0: __SQLLEAVE(KErrNotFound); sl@0: } sl@0: } sl@0: if(!aFileData.IsSecureFileNameFmt()) sl@0: {//A non-secure database to be attached - execute the "ATTACH DB" SQL statement and initialize the attached database. sl@0: InitAttachedDbL(aFileData, aDbName); sl@0: } sl@0: else sl@0: {//A secure database has to be attached. This is a complex operation and if it fails, proper cleanup sl@0: //has to be performed. "state" variable keeps the state, after which the "attach db" operation failed. sl@0: //There are three things needed to be done when atatching a secure database: sl@0: // 1. Executing the "ATTACH DB" SQL statement and initializing the attached database sl@0: // 2. Updating the security policy map sl@0: // 3. Updating the {db name, logical db name} map sl@0: //The additional map (3) is needed because when the authorizer callback is called by SQLITE, the callback sl@0: //is given the logical database name, if that's an operation on an attached database. Since the security sl@0: //policy map uses the database name as a key, the map (3) is used to find what is the physical database sl@0: //name, which the logical database name points to. sl@0: // sl@0: //There is an additional step (0), which may happen when a secure database is attached to a sl@0: //non-secure database. But this step does not require a cleanup. sl@0: CSqlSrvDatabase::TAttachState state = CSqlSrvDatabase::EAStNone; sl@0: const TUint8* securityMapKey = NULL; sl@0: TRAPD(err, DoAttachSecureDbL(state, aFileData, aDbName, securityMapKey)); sl@0: if(err != KErrNone) sl@0: { sl@0: //Cleanup sl@0: if(state > CSqlSrvDatabase::EAStNone) sl@0: { sl@0: (void)FinalizeAttachedDb(aDbName); sl@0: if(state > CSqlSrvDatabase::EAStDbAttached) sl@0: { sl@0: ::SqlServer().SecurityMap().Remove(securityMapKey); sl@0: } sl@0: } sl@0: __SQLLEAVE(err); sl@0: } sl@0: } sl@0: SQL_TRACE_INTERNALS(OstTrace1(TRACE_INTERNALS, CSQLSRVDATABASE_ATTACHDBL_EXIT, "Exit;0x%X;CSqlSrvDatabase::AttachDbL", (TUint)this)); sl@0: } sl@0: sl@0: /** sl@0: Attaches a secure database to the existing connection. sl@0: sl@0: The function also updates the following maps: sl@0: - Security policies map (CSqlServer::iSecurityMap), where a reference counted copy of database security policies sl@0: is kept for each created/opened/attached database during the life time of the SQL server; sl@0: - Attached secure databases map (CSqlSrvDatabase::iAttachDbMap), where an information is kept what SQL database sl@0: file name corresponds to a specific attached database name. This information is used by the authorizer callback sl@0: function in order to find what database security policies have to be asserted when a SQL operation is issued sl@0: for a particular attached database (the attached database name is identified always by its name, not the file name); sl@0: sl@0: @param aState Output parameter - the attach progress state, used by the calling function to perform correctly sl@0: the cleanup, if the attach operation fails. sl@0: It may have the following values set: sl@0: - CSqlSrvDatabase::EAStNone - no resources need to be freed; sl@0: - CSqlSrvDatabase::EAStDbAttached - the function was able to execute the "ATTACH DATABASE" sl@0: SQL statement before an error occured. The calling function has to execute "DETACH DATABASE" sl@0: SQL statement to detach the database; sl@0: - CSqlSrvDatabase::EAStSecurityMapUpdated - the function was able to update the security policies sl@0: map (CSqlServer::iSecurityMap) before an error occured. The calling function has to remove sl@0: the attached database security policies from the map and detach the database. sl@0: @param aFileData A reference to a TSqlSrvFileData instance, which keeps the database name, a sl@0: file session reference and some other database file related properties. sl@0: The secure database name format must be: sl@0: \:\<[SID]database file name excluding the path\>. sl@0: "[SID]" refers to SID of the application which creates the database. sl@0: @param aDbName Database name. It must be unique (per database connection). This is the name which can sl@0: be used for accessing tables in the attached database. The syntax is "database-name.table-name". sl@0: @param aMapKey Output parameter, UTF8 encoded, zero-terminated string, which can be used for searching sl@0: of the attached database security policies in the security policies map (CSqlServer::iSecurityMap). sl@0: No memory is allocated for the key. sl@0: sl@0: @see RSqlSecurityMap sl@0: @see RSqlAttachDbMap sl@0: @see CSqlServer sl@0: @see TSqlSrvFileData sl@0: @see CSqlSrvDatabase sl@0: sl@0: @leave KErrNoMemory, an out of memory condition has occurred; sl@0: KErrPermissionDenied, the client application does not satisfy the relevant security policies of sl@0: the attached database. sl@0: Note that the function may also leave with some other database specific sl@0: errors categorised as ESqlDbError, and other system-wide error codes. sl@0: */ sl@0: void CSqlSrvDatabase::DoAttachSecureDbL(CSqlSrvDatabase::TAttachState& aState, sl@0: const TSqlSrvFileData& aFileData, sl@0: const TDesC& aDbName, const TUint8*& aMapKey) sl@0: { sl@0: //Step 1: Attach the database and initialize the attached database sl@0: InitAttachedDbL(aFileData, aDbName); sl@0: aState = CSqlSrvDatabase::EAStDbAttached; sl@0: //Step 2: Load the database security policy and update the security map sl@0: const CSqlSecurityPolicy* securityPolicy = NULL; sl@0: UpdateSecurityMapL(ETrue, aFileData, aMapKey, securityPolicy); sl@0: aState = CSqlSrvDatabase::EAStSecurityMapUpdated; sl@0: //Check that the caller has at least one of {Schema, Read, Write} policies. sl@0: BasicSecurityPolicyCheckL(*securityPolicy); sl@0: //Step 3: Update the attached databases map sl@0: InsertInAttachDbMapL(aFileData.FileName(), aDbName); sl@0: } sl@0: sl@0: /** sl@0: Detaches a database from the current database connection. sl@0: sl@0: The function also will search the security policies map (CSqlServer::iSecurityMap) and the attached databases sl@0: map (CSqlSrvDatabase::iAttachDbMap) for entries which correspond to the database , sl@0: which is about to be detached, and will remove the entries. sl@0: sl@0: @param aDbName Attached database name. It must be unique (per database connection). sl@0: sl@0: @leave The function may leave with some database specific errors categorised as ESqlDbError, sl@0: and other system-wide error codes. sl@0: sl@0: @see CSqlSrvDatabase::DoAttachSecureDbL() sl@0: @see RSqlSecurityMap sl@0: @see RSqlAttachDbMap sl@0: @see CSqlServer sl@0: @see CSqlSrvDatabase sl@0: */ sl@0: void CSqlSrvDatabase::DetachDbL(const TDesC& aDbName) sl@0: { sl@0: SQL_TRACE_INTERNALS(OstTraceExt2(TRACE_INTERNALS, CSQLSRVDATABASE_DETACHDBL_ENTRY, "Entry;0x%X;CSqlSrvDatabase::DetachDbL;aDbName=%S", (TUint)this, __SQLPRNSTR(aDbName))); sl@0: TInt err = FinalizeAttachedDb(aDbName); sl@0: if(err == KErrNone) sl@0: { sl@0: TRAP_IGNORE(RemoveFromMapsL(aDbName)); sl@0: } sl@0: else sl@0: { sl@0: __SQLLEAVE(err); sl@0: } sl@0: SQL_TRACE_INTERNALS(OstTrace1(TRACE_INTERNALS, CSQLSRVDATABASE_DETACHDBL_EXIT, "Exit;0x%X;CSqlSrvDatabase::DetachDbL", (TUint)this)); sl@0: } sl@0: sl@0: /** sl@0: Calculates and returns the database size. sl@0: sl@0: @param aDbName Attached database name or KNullDesC for the main database sl@0: sl@0: @leave The function may leave with some database specific errors categorised as ESqlDbError, sl@0: and other system-wide error codes. sl@0: sl@0: @return Database size in bytes. sl@0: */ sl@0: TInt64 CSqlSrvDatabase::SizeL(const TDesC& aDbName) sl@0: { sl@0: iAuthorizerDisabled = ETrue; sl@0: CleanupStack::PushL(TCleanupItem(&EnableAuthorizer, &iAuthorizerDisabled)); sl@0: TInt pageCount = 0; sl@0: __SQLLEAVE_IF_ERROR(::DbPageCount(iDbHandle, aDbName, pageCount)); sl@0: __ASSERT_DEBUG(pageCount >= 0, __SQLPANIC(ESqlPanicInternalError)); sl@0: CleanupStack::PopAndDestroy(); sl@0: return (TInt64)pageCount * PageSizeL(aDbName); sl@0: } sl@0: sl@0: /** sl@0: Retrieves the database free space. sl@0: sl@0: @param aDbName Attached database name or KNullDesC for the main database sl@0: sl@0: @return Database free space in bytes. sl@0: sl@0: @leave The function may leave with some database specific errors categorised as ESqlDbError, sl@0: and other system-wide error codes. sl@0: */ sl@0: TInt64 CSqlSrvDatabase::FreeSpaceL(const TDesC& aDbName) sl@0: { sl@0: iAuthorizerDisabled = ETrue; sl@0: CleanupStack::PushL(TCleanupItem(&EnableAuthorizer, &iAuthorizerDisabled)); sl@0: TInt freePageCount = 0; sl@0: __SQLLEAVE_IF_ERROR(::DbFreePageCount(iDbHandle, aDbName, freePageCount)); sl@0: CleanupStack::PopAndDestroy(); sl@0: __ASSERT_DEBUG(freePageCount >= 0, __SQLPANIC(ESqlPanicInternalError)); sl@0: return (TInt64)freePageCount * PageSizeL(aDbName); sl@0: } sl@0: sl@0: /** sl@0: Collects information regarding the current cache size, page, size, encoding, etc. and puts the values sl@0: in the aDest output string. sl@0: sl@0: @param aDest Output parameter, where the result values will be stored. The string format is: ";...;". sl@0: sl@0: @leave The function may leave with some database specific errors categorised as ESqlDbError, sl@0: and other system-wide error codes. sl@0: */ sl@0: void CSqlSrvDatabase::QueryConfigL(TDes8& aDest) sl@0: { sl@0: iAuthorizerDisabled = ETrue; sl@0: CleanupStack::PushL(TCleanupItem(&EnableAuthorizer, &iAuthorizerDisabled)); sl@0: sl@0: TInt cacheSize = 0; sl@0: __SQLLEAVE_IF_ERROR(::DbCacheSize(iDbHandle, KNullDesC, cacheSize)); sl@0: sl@0: TInt pageSize = 0; sl@0: __SQLLEAVE_IF_ERROR(::DbPageSize(iDbHandle, KNullDesC, pageSize)); sl@0: sl@0: TBuf8<16> encoding; sl@0: __SQLLEAVE_IF_ERROR(::DbEncoding(iDbHandle, KNullDesC, encoding)); sl@0: sl@0: TInt defaultSoftHeapLimit = TSqlSrvConfigParams::KDefaultSoftHeapLimitKb; sl@0: sl@0: TInt vacuum = 0; sl@0: __SQLLEAVE_IF_ERROR(::DbVacuumMode(iDbHandle, KNullDesC, vacuum)); sl@0: sl@0: aDest.AppendNum(cacheSize); sl@0: _LIT8(KSemiColon, ";"); sl@0: aDest.Append(KSemiColon); sl@0: aDest.AppendNum(pageSize); sl@0: aDest.Append(KSemiColon); sl@0: aDest.Append(encoding); sl@0: aDest.Append(KSemiColon); sl@0: aDest.AppendNum(defaultSoftHeapLimit); sl@0: aDest.Append(KSemiColon); sl@0: aDest.AppendNum(vacuum); sl@0: aDest.Append(KSemiColon); sl@0: sl@0: CleanupStack::PopAndDestroy(); sl@0: } sl@0: sl@0: /** sl@0: Compacts the database free space. sl@0: sl@0: @param aSize The requested database space to be compacted, in bytes. sl@0: If aSize value is RSqlDatabase::EMaxCompaction, then all free pages will be removed. sl@0: Note that the requested space to be compacted will be rounded up to the nearest page count, sl@0: e.g. request for removing 1 byte will remove one free page from the database file. sl@0: sl@0: @param aDbName Attached database name or KNullDesC for the main database sl@0: sl@0: @return The size of the removed free space sl@0: sl@0: @leave The function may leave with some database specific errors categorised as ESqlDbError, sl@0: and other system-wide error codes. sl@0: */ sl@0: TInt CSqlSrvDatabase::CompactL(TInt aSize, const TDesC& aDbName) sl@0: { sl@0: SQL_TRACE_INTERNALS(OstTraceExt3(TRACE_INTERNALS, CSQLSRVDATABASE_COMPACTL_ENTRY, "Entry;0x%X;CSqlSrvDatabase::CompactL;aSize=%d;aDbName=%S", (TUint)this, aSize, __SQLPRNSTR(aDbName))); sl@0: __ASSERT_DEBUG(aSize > 0 || aSize == RSqlDatabase::EMaxCompaction, __SQLPANIC(ESqlPanicBadArgument)); sl@0: TInt pageSize = PageSizeL(aDbName);//PageSizeL() will disable/enable the authorizer sl@0: TInt pageCount = KMaxTInt; sl@0: if(aSize > 0) sl@0: {//64-bit calculations to avoid the overflow in case if (aSize + pageSize) >= KMaxTInt. sl@0: pageCount = (TInt64)((TInt64)aSize + pageSize - 1) / pageSize; sl@0: } sl@0: if(pageCount > 0) sl@0: { sl@0: iAuthorizerDisabled = ETrue; sl@0: CleanupStack::PushL(TCleanupItem(&EnableAuthorizer, &iAuthorizerDisabled)); sl@0: __SQLLEAVE_IF_ERROR(::DbCompact(iDbHandle, aDbName, pageCount, pageCount)); sl@0: CleanupStack::PopAndDestroy(); sl@0: } sl@0: SQL_TRACE_INTERNALS(OstTraceExt3(TRACE_INTERNALS, CSQLSRVDATABASE_COMPACTL_EXIT, "Exit;0x%X;CSqlSrvDatabase::CompactL;pageCount=%d;pageSize=%d", (TUint)this, pageCount, pageSize)); sl@0: return pageCount * pageSize; sl@0: } sl@0: sl@0: /** sl@0: This structure is used in case if the InitAttachedDbL() execution fails and the sl@0: executed operations ("attach database", "init compact") have to be reverted calling sl@0: FinalizeAttachedDb(). sl@0: sl@0: @see CSqlSrvDatabase::InitAttachedDbL sl@0: @see CSqlSrvDatabase::FinalizeAttachedDb sl@0: @see CSqlSrvDatabase::AttachCleanup sl@0: sl@0: @internalComponent sl@0: */ sl@0: NONSHARABLE_STRUCT(TAttachCleanup) sl@0: { sl@0: inline TAttachCleanup(CSqlSrvDatabase& aSelf, const TDesC& aDbName) : sl@0: iSelf(aSelf), sl@0: iDbName(aDbName) sl@0: { sl@0: } sl@0: CSqlSrvDatabase& iSelf; sl@0: const TDesC& iDbName; sl@0: }; sl@0: sl@0: /** sl@0: Cleanup function. Calls FinalizeAttachedDb() if InitAttachedDbL() has failed. sl@0: sl@0: @param aCleanup A pointer to a TAttachCleanup object that contains the needed for the cleanup information. sl@0: sl@0: @see TAttachCleanup sl@0: sl@0: @internalComponent sl@0: */ sl@0: void CSqlSrvDatabase::AttachCleanup(void* aCleanup) sl@0: { sl@0: TAttachCleanup* cleanup = reinterpret_cast (aCleanup); sl@0: __ASSERT_DEBUG(cleanup != NULL, __SQLPANIC2(ESqlPanicBadArgument)); sl@0: (void)cleanup->iSelf.FinalizeAttachedDb(cleanup->iDbName); sl@0: } sl@0: sl@0: /** sl@0: Forms and executes "ATTACH DATABASE" SQL statement. sl@0: If the database is not read-only: sl@0: - Makes the attached database journal file persistent. sl@0: - Initializes the attached database compaction mode. sl@0: - The attached database will be reindexed if the default collation has been changed. sl@0: sl@0: @param aFileData Attached database file data sl@0: @param aDbName Attached database name. It must be unique (per database connection). sl@0: sl@0: @leave KErrNoMemory, an out of memory condition has occurred; sl@0: Note that the function may also leave with some other database specific sl@0: errors categorised as ESqlDbError, and other system-wide error codes. sl@0: sl@0: @see InitCompactionL sl@0: @see ProcessSettingsL sl@0: */ sl@0: void CSqlSrvDatabase::InitAttachedDbL(const TSqlSrvFileData& aFileData, const TDesC& aDbName) sl@0: { sl@0: TPtrC dbFileName = aFileData.FileName(); sl@0: TBool readOnlyDbFile = aFileData.IsReadOnly(); sl@0: sl@0: InstallAuthorizerL(); sl@0: iAuthorizerDisabled = ETrue; sl@0: CleanupStack::PushL(TCleanupItem(&EnableAuthorizer, &iAuthorizerDisabled)); sl@0: sqlite3_stmt* stmtHandle = StmtPrepare16L(iDbHandle, KAttachDb); sl@0: TInt err = ::Sql2OsErrCode(sqlite3_bind_text16(stmtHandle, 1, dbFileName.Ptr(), dbFileName.Length() * sizeof(TUint16), SQLITE_STATIC), sqlite3SymbianLastOsError()); sl@0: if(err == KErrNone) sl@0: { sl@0: err = ::Sql2OsErrCode(sqlite3_bind_text16(stmtHandle, 2, aDbName.Ptr(), aDbName.Length() * sizeof(TUint16), SQLITE_STATIC), sqlite3SymbianLastOsError()); sl@0: if(err == KErrNone) sl@0: { sl@0: err = StmtExec(stmtHandle); sl@0: } sl@0: } sl@0: (void)::FinalizeStmtHandle(stmtHandle);//sqlite3_finalize() fails only if an invalid statement handle is passed. sl@0: CleanupStack::PopAndDestroy();//TCleanupItem(&EnableAuthorizer, &iAuthorizerDisabled) sl@0: __SQLLEAVE_IF_ERROR(err); sl@0: sl@0: TAttachCleanup attachCleanup(*this, aDbName); sl@0: CleanupStack::PushL(TCleanupItem(&CSqlSrvDatabase::AttachCleanup, &attachCleanup)); sl@0: sl@0: SetConfigL(aFileData.ConfigParams(), EFalse, aDbName); sl@0: sl@0: if(!readOnlyDbFile) sl@0: { sl@0: iAuthorizerDisabled = ETrue; sl@0: CleanupStack::PushL(TCleanupItem(&EnableAuthorizer, &iAuthorizerDisabled)); sl@0: ProcessSettingsL(aFileData, aDbName); sl@0: CleanupStack::PopAndDestroy();//TCleanupItem(&EnableAuthorizer, &iAuthorizerDisabled) sl@0: } sl@0: sl@0: CleanupStack::Pop();//TCleanupItem(&CSqlSrvDatabase::AttachCleanup, &attachCleanup) sl@0: } sl@0: sl@0: /** sl@0: Forms and executes "DETACH DATABASE" SQL statement. sl@0: If the database was scheduled for background compacting, the related database entry will be removed from sl@0: the vaccum db map. sl@0: sl@0: @param aDbName Attached database name. It must be unique (per database connection). sl@0: sl@0: @return KErrNone, Operation completed successfully; sl@0: KErrNoMemory, an out of memory condition has occurred. sl@0: Note that the function may also return some other database specific sl@0: errors categorised as ESqlDbError, and other system-wide error codes. sl@0: */ sl@0: TInt CSqlSrvDatabase::FinalizeAttachedDb(const TDesC& aDbName) sl@0: { sl@0: ReleaseCompactEntry(aDbName); sl@0: iAuthorizerDisabled = ETrue; sl@0: sqlite3_stmt* stmtHandle = NULL; sl@0: TRAPD(err, stmtHandle = StmtPrepare16L(iDbHandle, KDetachDb)); sl@0: if(err == KErrNone) sl@0: { sl@0: err = ::Sql2OsErrCode(sqlite3_bind_text16(stmtHandle, 1, aDbName.Ptr(), aDbName.Length() * sizeof(TUint16), SQLITE_STATIC), sqlite3SymbianLastOsError()); sl@0: if(err == KErrNone) sl@0: { sl@0: err = StmtExec(stmtHandle); sl@0: } sl@0: } sl@0: (void)::FinalizeStmtHandle(stmtHandle);//sqlite3_finalize() fails only if an invalid statement handle is passed. sl@0: iAuthorizerDisabled = EFalse; sl@0: return err; sl@0: } sl@0: sl@0: /** sl@0: Updates the security policies map (CSqlServer::iSecurityMap). sl@0: sl@0: Inserts a new item in the security map, or if such item already exists there - its reference counter will sl@0: be incremented. sl@0: The method guarantees that either the security map will be updated, or the method leaves and the security sl@0: policies map stays unchanged. sl@0: sl@0: @param aAttachedDb True if the currently processed database is an attached database, false if it is the main db. sl@0: @param aFileData A reference to a TSqlSrvFileData instance, which keeps the database name, a sl@0: file session reference and some other database file related properties. sl@0: The secure database name format must be: sl@0: \:\<[SID]database file name excluding the path\>. sl@0: "[SID]" refers to SID of the application which creates the database. sl@0: @param aMapKey Output parameter. UTF8 encoded, zero-terminated string. On function exit cannot be NULL. sl@0: It will be set to point to the security policies map key. No memory is allocated for the key. sl@0: @param aSecurityPolicy Output parameter. On function exit cannot be NULL. It will be set to point to the security policies. sl@0: sl@0: @leave KErrNoMemory, an out of memory condition has occurred; sl@0: Note that the function may also leave with some other database specific sl@0: errors categorised as ESqlDbError, and other system-wide error codes. sl@0: sl@0: @see RSqlSecurityMap sl@0: @see CSqlServer sl@0: @see TSqlSrvFileData sl@0: @see CSqlSecurityPolicy sl@0: */ sl@0: void CSqlSrvDatabase::UpdateSecurityMapL(TBool aAttachedDb, const TSqlSrvFileData& aFileData, sl@0: const TUint8*& aMapKey, const CSqlSecurityPolicy*& aSecurityPolicy) sl@0: { sl@0: //Check if a copy of the database security policies is already in the security map. sl@0: aMapKey = SecurityMapKeyL(aFileData.FileName()); sl@0: TSqlSecurityPair* pair = ::SqlServer().SecurityMap().Entry(aMapKey); sl@0: if(pair) sl@0: { sl@0: //Yes, it is in the map. Increment the reference counter. sl@0: //(It will be decremented when detaching the database). sl@0: pair->iRefCounter.Increment(); sl@0: aMapKey = pair->iKey; sl@0: aSecurityPolicy = pair->iData; sl@0: } sl@0: else sl@0: { sl@0: //No, it is not in the map. Read the security policies from the security policies tables and sl@0: //insert a new item in the map. sl@0: __ASSERT_DEBUG(aMapKey != NULL, __SQLPANIC(ESqlPanicInternalError)); sl@0: aMapKey = ::CreateStrCopyLC(aMapKey); sl@0: CSqlSecurityPolicy* securityPolicy = aAttachedDb ? ::LoadAttachedDbSecurityPolicyLC(aFileData) : sl@0: ::LoadDbSecurityPolicyLC(iDbHandle); sl@0: __ASSERT_DEBUG(!::SqlServer().SecurityMap().Entry(aMapKey), __SQLPANIC2(ESqlPanicObjExists)); sl@0: __SQLLEAVE_IF_ERROR(::SqlServer().SecurityMap().Insert(aMapKey, securityPolicy)); sl@0: CleanupStack::Pop(2);//iSecurityMap owns aMapKey and securityPolicy objects sl@0: aSecurityPolicy = securityPolicy; sl@0: } sl@0: __ASSERT_DEBUG(aMapKey != NULL, __SQLPANIC(ESqlPanicInternalError)); sl@0: __ASSERT_DEBUG(aSecurityPolicy != NULL, __SQLPANIC(ESqlPanicInternalError)); sl@0: } sl@0: sl@0: /** sl@0: Removes attached secure database entries from the maps. sl@0: sl@0: The function will search the security policies map (CSqlServer::iSecurityMap) and the attached databases sl@0: map (CSqlSrvDatabase::iAttachDbMap) for entries which correspond to the database with aDbName name, sl@0: and will remove the entries. sl@0: sl@0: The sequence of the performed by the function operations is: sl@0: 1. Looks for a map item which key is aDbName in iAttachDbMap map. sl@0: 2. If such pair exists, the data part of the pair is used as a key in iSecurityMap map. sl@0: 3. Remove the iSecurityMap map item pointed by the data part of the found pair. sl@0: 4. Remove the iAttachDbMap map item pointed by aDbName. sl@0: sl@0: @param aDbName Attached database name. It must be unique (per database connection). sl@0: sl@0: @leave KErrGeneral It is not possible to convert aDbName parameter to UTF8 encoded string. sl@0: sl@0: @see CSqlSrvDatabase::DoAttachDbL() sl@0: @see CSqlSrvDatabase::DoAttachSecureDbL() sl@0: @see RSqlSecurityMap sl@0: @see RSqlAttachDbMap sl@0: @see CSqlServer sl@0: @see CSqlSrvDatabase sl@0: */ sl@0: void CSqlSrvDatabase::RemoveFromMapsL(const TDesC& aDbName) sl@0: { sl@0: SQL_TRACE_INTERNALS(OstTraceExt2(TRACE_INTERNALS, CSQLSRVDATABASE_REMOVEFROMMAPSL, "0x%X;CSqlSrvDatabase::RemoveFromMapsL;aDbName=%S", (TUint)this, __SQLPRNSTR(aDbName))); sl@0: TPtr8 ptr(iFileNameBuf, sizeof(iFileNameBuf)); sl@0: if(!::UTF16ToUTF8Z(aDbName, ptr)) sl@0: { sl@0: __SQLLEAVE(KErrGeneral); sl@0: } sl@0: TSqlAttachDbPair* attachDbPair = iAttachDbMap.Entry(iFileNameBuf); sl@0: if(attachDbPair) sl@0: {//aDbName refers to attached secure database sl@0: ::SqlServer().SecurityMap().Remove(attachDbPair->iData); sl@0: iAttachDbMap.Remove(iFileNameBuf); sl@0: } sl@0: } sl@0: sl@0: /** sl@0: Inserts a new entry in the attached databases map (CSqlSrvDatabase::iAttachDbMap). sl@0: sl@0: The method guarantees that either a new [logical db name, secure db name] pair will be inserted in sl@0: iAttachDbMap or the method fails and the content of iAttachDbMap remains unchanged. sl@0: sl@0: @param aDbFileName Database file name, including the path. sl@0: @param aDbName Database name. sl@0: sl@0: @leave KErrNoMemory, an out of memory condition has occurred; sl@0: KErrGeneral, it is not possible to convert the function parameters to UTF8 encoded strings. sl@0: sl@0: @see RSqlAttachDbMap sl@0: @see CSqlSrvDatabase sl@0: */ sl@0: void CSqlSrvDatabase::InsertInAttachDbMapL(const TDesC& aDbFileName, const TDesC& aDbName) sl@0: { sl@0: SQL_TRACE_INTERNALS(OstTraceExt3(TRACE_INTERNALS, CSQLSRVDATABASE_INSERTINATTACHDBMAPL, "0x%X;CSqlSrvDatabase::InsertInAttachDbMapL;aDbFileName=%S;aDbName=%S", (TUint)this, __SQLPRNSTR(aDbFileName), __SQLPRNSTR(aDbName))); sl@0: //Convert aDbName to UTF8, zero-terminated name sl@0: TPtr8 ptr(iFileNameBuf, sizeof(iFileNameBuf)); sl@0: if(!::UTF16ToUTF8Z(aDbName, ptr)) sl@0: { sl@0: __SQLLEAVE(KErrGeneral); sl@0: } sl@0: const TUint8* mapKey = ::CreateStrCopyLC(iFileNameBuf); sl@0: const TUint8* mapData = SecurityMapKeyL(aDbFileName); sl@0: mapData = ::CreateStrCopyLC(mapData); sl@0: __ASSERT_DEBUG(!iAttachDbMap.Entry(mapKey), __SQLPANIC(ESqlPanicObjExists)); sl@0: __SQLLEAVE_IF_ERROR(iAttachDbMap.Insert(mapKey, mapData)); sl@0: CleanupStack::Pop(2);//iAttachDbMap owns mapKey amd mapData. sl@0: } sl@0: sl@0: /** sl@0: Processes the database settings that are currently stored in the settings table. sl@0: Makes the journal file persistent. sl@0: Initializes the database compaction mode. sl@0: Based on the current settings the database may be configured to become 'up-to-date'. sl@0: This configuration may include reindexing the database if the collation dll has sl@0: changed and/or executing database configuration file operations. sl@0: sl@0: @param aFileData The file data object, sl@0: @param aDbName Logical database name: "main" for the main database or attached database name, sl@0: sl@0: @leave The function may leave with system-wide error codes or SQL errors of ESqlDbError type sl@0: sl@0: @panic SqlDb 7 In _DEBUG mode if aFileData does not refer to a r/w database file. sl@0: */ sl@0: void CSqlSrvDatabase::ProcessSettingsL(const TSqlSrvFileData& aFileData, const TDesC& aDbName) sl@0: { sl@0: __ASSERT_DEBUG(!aFileData.IsReadOnly(), __SQLPANIC(ESqlPanicInternalError)); sl@0: #if !defined(__SQL_DISABLE_SYMBIAN_SETTINGS_TABLE__) sl@0: //Make the journal file persistent - done by SQLite automatically because the locking mode is EXCLUSIVE sl@0: //__SQLLEAVE_IF_ERROR(::ExecPragma(iDbHandle, iAuthorizerDisabled, KPersistentJournalPragma, KPersist, aDbName)); sl@0: //Load the current settings sl@0: TFileName storedCollationDllName; sl@0: TInt storedDbConfigFileVer = -1; sl@0: TInt currVacuumMode = -1; sl@0: __SQLLEAVE_IF_ERROR(::DbVacuumMode(iDbHandle, aDbName, currVacuumMode)); sl@0: //currVacuumMode == ESqliteVacuumOff ==> This is a database created not by the SQL server sl@0: TSqlCompactionMode compactionMode = currVacuumMode == ESqliteVacuumOff ? ESqlCompactionManual : ESqlCompactionNotSet; sl@0: TSqlDbSysSettings dbSettings(iDbHandle); sl@0: dbSettings.LoadSettingsL(aDbName, storedCollationDllName, storedDbConfigFileVer, compactionMode); sl@0: if(currVacuumMode == ESqliteVacuumOff && compactionMode != ESqlCompactionManual) sl@0: {//The database has been opened and the vacuum mode is "off". Then this is a database, not created by SQL sl@0: //server or it is a corrupted database. The compaction mode read from the system table does not match the sl@0: //database vacuum mode. Conclusion: this is a corrupted database. sl@0: __SQLLEAVE(KErrCorrupt); sl@0: } sl@0: if(aFileData.ContainHandles() && aFileData.IsCreated()) sl@0: { sl@0: compactionMode = aFileData.ConfigParams().iCompactionMode; sl@0: if(compactionMode == ESqlCompactionNotSet) sl@0: { sl@0: compactionMode = KSqlDefaultCompactionMode; sl@0: } sl@0: //This is a just created private database. Store the compaction mode (the compaction mode may have been set using a config string). sl@0: StoreSettingsL(storedCollationDllName, storedDbConfigFileVer, compactionMode); sl@0: } sl@0: //Init the database compaction mode sl@0: InitCompactionL(compactionMode, aFileData.ConfigParams().iFreePageThresholdKb, aFileData.FileName(), sl@0: (TSqliteVacuumMode)currVacuumMode, aDbName); sl@0: //Based on the current settings, apply any necessary configuration updates to the database sl@0: ApplyConfigUpdatesL(storedCollationDllName, storedDbConfigFileVer, aFileData, aDbName); sl@0: #endif // !(__SQL_DISABLE_SYMBIAN_SETTINGS_TABLE__) sl@0: } sl@0: sl@0: /** sl@0: Applies any necessary configuration updates to the database, based on the current settings sl@0: in the settings table and how the database is being used (i.e. 'Opened' or 'Attached'). sl@0: The applied configuration may include: sl@0: - Reindexing the main database and all attached databases, if the collation dll has been changed. sl@0: After the reindexation the new collation dll name will be stored in the settings table of the database. sl@0: - Executing all supported operations on the database that are specified in a database configuration sl@0: file, if such a file exists and has not already been processed. sl@0: The settings table will updated with the current version of the database configuration file if the file sl@0: is processed. sl@0: sl@0: @param aStoredCollationDllName The name of the collation dll that is stored in the settings table, sl@0: @param aStoredDbConfigFileVersion The database configuration file version that is stored in the settings table, sl@0: @param aFileData The file data object, sl@0: @param aDbName Logical database name: "main" for the main database or attached database name, sl@0: sl@0: @leave The function may leave with system-wide error codes or SQL errors of ESqlDbError type sl@0: */ sl@0: void CSqlSrvDatabase::ApplyConfigUpdatesL(const TDesC& aStoredCollationDllName, const TInt& aStoredDbConfigFileVersion, sl@0: const TSqlSrvFileData& aFileData, const TDesC& aDbName) sl@0: { sl@0: TSqlDbSysSettings dbSettings(iDbHandle); sl@0: //Check whether reindexing is necessary sl@0: if(::SqlServer().CollationDllName().CompareF(aStoredCollationDllName) != 0) sl@0: { sl@0: SQL_TRACE_INTERNALS(OstTraceExt3(TRACE_INTERNALS, CSQLSRVDATABASE_APPLYCONFIGUPDATES2L, "0x%X;CSqlSrvDatabase::ApplyConfigUpdatesL;Reindex db;aStoredCollationDllName=%S;aDbName=%S", (TUint)this, __SQLPRNSTR(aStoredCollationDllName), __SQLPRNSTR(aDbName))); sl@0: dbSettings.ReindexDatabaseL(aDbName, ::SqlServer().CollationDllName()); sl@0: } sl@0: sl@0: //Perform any necessary configuration file updates to the database. sl@0: //We do not want failures here to cause the database to fail sl@0: //to be opened and so any leave error is TRAPed and ignored sl@0: //(the error is logged in _DEBUG mode) sl@0: TRAPD(err, dbSettings.ConfigureDatabaseL(aStoredDbConfigFileVersion, aFileData, aDbName)); sl@0: if(KErrNone != err) sl@0: { sl@0: SQL_TRACE_INTERNALS(OstTraceExt2(TRACE_INTERNALS, CSQLSRVDATABASE_APPLYCONFIGUPDATESL, "0x%X;CSqlSrvDatabase::ApplyConfigUpdatesL;ConfigureDatabaseL() failed with error code %d", (TUint)this, err)); sl@0: } sl@0: } sl@0: sl@0: /** sl@0: Sets the "cache size" and "page size" parameter values, if their values are valid. sl@0: This is done formatting and executing PRAGMA statements. sl@0: @param aConfigParams This object contains the cofiguration parameters sl@0: @param aSetPageSize If true, the page size will be set, otherwise not. sl@0: The aSetPageSize is set to true if: sl@0: 1) The operation is "create database" sl@0: 2) The operation is "open private database" sl@0: @param aLogicalDbName Parameter with default value of KNullDesC. If aLogicalDbName length is not 0, then the sl@0: "cache_size" pragma will be executed on the attached database with aLogicalDbName name. sl@0: */ sl@0: void CSqlSrvDatabase::SetConfigL(const TSqlSrvConfigParams& aConfigParams, TBool aSetPageSize, const TDesC& aLogicalDbName) sl@0: { sl@0: __ASSERT_DEBUG(aConfigParams.iPageSize == TSqlSrvConfigParams::KConfigPrmValueNotSet || aConfigParams.iPageSize >= 0, __SQLPANIC(ESqlPanicBadArgument)); sl@0: __ASSERT_DEBUG(aConfigParams.iCacheSize == TSqlSrvConfigParams::KConfigPrmValueNotSet || aConfigParams.iCacheSize >= 0, __SQLPANIC(ESqlPanicBadArgument)); sl@0: if(aSetPageSize && aConfigParams.iPageSize != TSqlSrvConfigParams::KConfigPrmValueNotSet) sl@0: { sl@0: __SQLLEAVE_IF_ERROR(::ExecPragma(iDbHandle, iAuthorizerDisabled, KPageSizePragma, aConfigParams.iPageSize)); sl@0: } sl@0: sl@0: const TDesC& logicalDbName = aLogicalDbName.Length() > 0 ? aLogicalDbName : KMainDb16; sl@0: sl@0: ::SetJournalSizeLimitL(iDbHandle, iAuthorizerDisabled, aConfigParams.iPageSize, logicalDbName); sl@0: sl@0: //Setting the cache size. sl@0: //Step 1: Check if aConfigParams.iCacheSize value is set. If it is set, then use it. sl@0: if(aConfigParams.iCacheSize != TSqlSrvConfigParams::KConfigPrmValueNotSet) sl@0: { sl@0: __SQLLEAVE_IF_ERROR(::ExecPragma(iDbHandle, iAuthorizerDisabled, KCacheSizePragma, aConfigParams.iCacheSize, logicalDbName)); sl@0: } sl@0: else sl@0: { sl@0: //Step 2: aConfigParams.iCacheSize value is not set. Then check if aConfigParams.iSoftHeapLimitKb value is set. sl@0: if(aConfigParams.iSoftHeapLimitKb != TSqlSrvConfigParams::KConfigPrmValueNotSet) sl@0: { sl@0: __ASSERT_DEBUG(aConfigParams.iSoftHeapLimitKb >= TSqlSrvConfigParams::KMinSoftHeapLimitKb && sl@0: aConfigParams.iSoftHeapLimitKb <= TSqlSrvConfigParams::KMaxSoftHeapLimitKb, __SQLPANIC(ESqlPanicInternalError)); sl@0: //Step 3: aConfigParams.iSoftHeapLimitKb value is set. Then use it to calculate the cache size. But we need the page size first. sl@0: // aLogicalDbName is used instead of logicalDbName because PageSizeL() if called with non-zero length name, sl@0: // "thinks" it is the main database name. KMainDb16 will be interpreted as an attached database name. sl@0: TInt pageSize = PageSizeL(aLogicalDbName); sl@0: //Step 4: Calculate the cache size. sl@0: TInt cacheSize = ((TInt64)aConfigParams.iSoftHeapLimitKb * 1024) / pageSize;//"TInt64" cast is used because of a possible overflow sl@0: //Step 5: Set the cache size. sl@0: __SQLLEAVE_IF_ERROR(::ExecPragma(iDbHandle, iAuthorizerDisabled, KCacheSizePragma, cacheSize, logicalDbName)); sl@0: } sl@0: } sl@0: } sl@0: sl@0: /** sl@0: Initializes the database compaction mode. sl@0: If the aCompactionMode parameter is ESqlCompactionBackground, the database with aDbFileName file name will be added sl@0: to the compactor for background compacting. sl@0: sl@0: @param aCompactionMode The database compaction mode. See TSqlCompactionMode enum for the supported database compaction modes. sl@0: @param aFreePageThresholdKb Free page threshold. The background compaction won't start if the free pages size in Kb is less than sl@0: the free page threshold. sl@0: @param aDbFileName Database file name including full path sl@0: @param aCurrentVacuumMode The current SQLite vacuum mode, one of TSqliteVacuumMode enum item values. sl@0: If the current database vacuum mode is ESqliteVacuumOff, which means sl@0: the database has been created not by the SQL server, sl@0: then the "off" vacuum mode is kept unchanged. sl@0: @param aDbName "main" or the attached database name sl@0: sl@0: @leave KErrNoMemory, an out of memory condition has occurred; sl@0: Note that the function may also leave with some other database specific sl@0: errors categorised as ESqlDbError, and other system-wide error codes. sl@0: sl@0: @see TSqlCompactionMode sl@0: @see CSqlCompactor sl@0: @see TSqliteVacuumMode sl@0: sl@0: @panic SqlDb 4 In _DEBUG mode if aCompactionMode parameter value is invalid. sl@0: */ sl@0: void CSqlSrvDatabase::InitCompactionL(TSqlCompactionMode aCompactionMode, TInt aFreePageThresholdKb, sl@0: const TDesC& aDbFileName, TSqliteVacuumMode aCurrentVacuumMode, const TDesC& aDbName) sl@0: { sl@0: __ASSERT_DEBUG(aCompactionMode == ESqlCompactionManual || aCompactionMode == ESqlCompactionBackground || aCompactionMode == ESqlCompactionAuto, __SQLPANIC(ESqlPanicBadArgument)); sl@0: __ASSERT_DEBUG(aCurrentVacuumMode == ESqliteVacuumOff || aCurrentVacuumMode == ESqliteVacuumAuto || sl@0: aCurrentVacuumMode == ESqliteVacuumIncremental, __SQLPANIC(ESqlPanicBadArgument)); sl@0: __ASSERT_DEBUG(aFreePageThresholdKb >= 0, __SQLPANIC(ESqlPanicBadArgument)); sl@0: TSqliteVacuumMode newSqliteVacuumMode = aCompactionMode == ESqlCompactionAuto ? ESqliteVacuumAuto : ESqliteVacuumIncremental; sl@0: if(aCurrentVacuumMode == ESqliteVacuumOff) sl@0: { sl@0: newSqliteVacuumMode = ESqliteVacuumOff; sl@0: } sl@0: if(aCurrentVacuumMode != newSqliteVacuumMode) sl@0: { sl@0: __SQLLEAVE_IF_ERROR(::ExecPragma(iDbHandle, iAuthorizerDisabled, KAutoVacuumPragma, newSqliteVacuumMode, aDbName)); sl@0: } sl@0: if(aCompactionMode == ESqlCompactionBackground) sl@0: { sl@0: NewCompactEntryL(aFreePageThresholdKb, aDbFileName, aDbName); sl@0: } sl@0: } sl@0: sl@0: /** sl@0: Adds the aDbFileName database to the compactor object for background compacting. sl@0: sl@0: @param aFreePageThresholdKb Free page threshold in Kb. The background compaction won't start if the total size of the free pages sl@0: is less than the free page threshold. sl@0: @param aDbFileName Database file name including full path sl@0: @param aDbName "main" or the attached database name sl@0: sl@0: @leave KErrNoMemory, an out of memory condition has occurred; sl@0: Note that the function may also leave with some other database specific sl@0: errors categorised as ESqlDbError, and other system-wide error codes. sl@0: sl@0: @see CSqlCompactor sl@0: @see RSqlCompactDbMap sl@0: */ sl@0: void CSqlSrvDatabase::NewCompactEntryL(TInt aFreePageThresholdKb, const TDesC& aDbFileName, const TDesC& aDbName) sl@0: { sl@0: SQL_TRACE_INTERNALS(OstTraceExt4(TRACE_INTERNALS, CSQLSRVDATABASE_NEWCOMPACTENTRYL_ENTRY, "Entry;0x%X;CSqlSrvDatabase::NewCompactEntryL;aFreePageThresholdKb=%d;aDbFileName=%S;aDbName=%S", (TUint)this, aFreePageThresholdKb, __SQLPRNSTR(aDbFileName), __SQLPRNSTR(aDbName))); sl@0: TSqlCompactSettings settings; sl@0: settings.iFreePageThresholdKb = aFreePageThresholdKb; sl@0: ::SqlServer().Compactor().AddEntryL(aDbFileName, settings); sl@0: TInt err = KErrNoMemory; sl@0: HBufC* key = aDbName.Alloc(); sl@0: HBufC* data = aDbFileName.Alloc(); sl@0: if(key && data) sl@0: { sl@0: __ASSERT_DEBUG(!iCompactDbMap.Entry(key), __SQLPANIC(ESqlPanicObjExists)); sl@0: err = iCompactDbMap.Insert(key, data);//returns the index of the new entry sl@0: } sl@0: if(err < 0) //If either "key" or "data" or both is NULL, then "err" is KErrNoMemory and the next "if" will be executed. sl@0: { sl@0: delete data; sl@0: delete key; sl@0: ::SqlServer().Compactor().ReleaseEntry(aDbFileName); sl@0: } sl@0: SQL_TRACE_INTERNALS(OstTraceExt2(TRACE_INTERNALS, CSQLSRVDATABASE_NEWCOMPACTENTRYL_EXIT, "Exit;0x%X;CSqlSrvDatabase::NewCompactEntryL;err=%d", (TUint)this, err)); sl@0: __SQLLEAVE_IF_ERROR(err); sl@0: } sl@0: sl@0: /** sl@0: Removes database (identified by its logical name) from the compactor. sl@0: sl@0: @param aDbName "main" or the attached database name sl@0: sl@0: @see CSqlCompactor sl@0: @see RSqlCompactDbMap sl@0: */ sl@0: void CSqlSrvDatabase::ReleaseCompactEntry(const TDesC& aDbName) sl@0: { sl@0: TSqlCompactDbMapIterator compactDbIt(iCompactDbMap); sl@0: TSqlCompactDbPair compactDbPair; sl@0: while(compactDbIt.Next(compactDbPair)) sl@0: { sl@0: if(::CompareNoCase(*compactDbPair.iKey, aDbName) == 0) sl@0: { sl@0: ::SqlServer().Compactor().ReleaseEntry(*compactDbPair.iData); sl@0: iCompactDbMap.Remove(compactDbPair.iKey); sl@0: SQL_TRACE_INTERNALS(OstTrace1(TRACE_INTERNALS, CSQLSRVDATABASE_RELEASECOMPACTENTRY, "0x%X;CSqlSrvDatabase::ReleaseCompactEntry", (TUint)this)); sl@0: break; sl@0: } sl@0: } sl@0: } sl@0: sl@0: /** sl@0: Cleanup function. sl@0: Used during the construction phase of the CSqlSrvDatabase instance when a new database is created. sl@0: If the database creation succeeds, the "init compaction" operation succeeds but some other init operation fail, sl@0: the just created database file has to be deleted. But before that the database has to be removed from the compactor, sl@0: because the compactor creates independent database connection that has to be closed before the database deletion. sl@0: sl@0: The database will be removed from the compactor as a result of the call. sl@0: sl@0: @param aCleanup A pointer to the CSqlSrvDatabase object sl@0: */ sl@0: void CSqlSrvDatabase::CompactCleanup(void* aCleanup) sl@0: { sl@0: CSqlSrvDatabase* self = reinterpret_cast (aCleanup); sl@0: __ASSERT_DEBUG(self != NULL, __SQLPANIC2(ESqlPanicBadArgument)); sl@0: self->ReleaseCompactEntry(KMainDb16); sl@0: } sl@0: sl@0: /** sl@0: Retrieves the database page size. sl@0: If the request is for the main database page size and if the size is not retrieved yet, then the page size value will be sl@0: cached in iPageSize data member. sl@0: sl@0: @param aDbName Attached database name or KNullDesC for the main database sl@0: sl@0: @leave KErrNoMemory, an out of memory condition has occurred; sl@0: Note that the function may also leave with some other database specific sl@0: errors categorised as ESqlDbError, and other system-wide error codes. sl@0: sl@0: @return The database page size sl@0: */ sl@0: TInt CSqlSrvDatabase::PageSizeL(const TDesC& aDbName) sl@0: { sl@0: if(iPageSize > 0 && aDbName == KNullDesC) sl@0: { sl@0: return iPageSize; sl@0: } sl@0: iAuthorizerDisabled = ETrue; sl@0: CleanupStack::PushL(TCleanupItem(&EnableAuthorizer, &iAuthorizerDisabled)); sl@0: TInt pageSize = 0; sl@0: __SQLLEAVE_IF_ERROR(::DbPageSize(iDbHandle, aDbName, pageSize)); sl@0: CleanupStack::PopAndDestroy(); sl@0: __ASSERT_DEBUG(pageSize > 0, __SQLPANIC(ESqlPanicInternalError)); sl@0: if(aDbName == KNullDesC) sl@0: { sl@0: iPageSize = pageSize; sl@0: } sl@0: return pageSize; sl@0: } sl@0: sl@0: ////////////////////////////////////////////////////////////////////////////////////////////////////// sl@0: ///////////////////////////// ConstructL() methods /////////////////////////////////////// sl@0: ////////////////////////////////////////////////////////////////////////////////////////////////////// sl@0: sl@0: /** sl@0: Second phase construction method. Creates a new secure database file. sl@0: If the function fails, the database file will be closed and deleted. sl@0: sl@0: @param aFileData A reference to a TSqlSrvFileData instance, which keeps the database name, a sl@0: file session reference and some other database file related properties. sl@0: If this is a secure database, then the format of the name must be: sl@0: \:\<[SID]database file name excluding the path\>. sl@0: If this is a non-secure database, then the file name has to be the full database file name. sl@0: "[SID]" refers to SID of the application which creates the database. sl@0: @param aSecurityPolicy Database security policy sl@0: sl@0: @panic SqlDb 4 In _DEBUG mode if aSecurityPolicy is NULL. sl@0: */ sl@0: void CSqlSrvDatabase::ConstructCreateSecureL(const TSqlSrvFileData& aFileData, CSqlSecurityPolicy* aSecurityPolicy) sl@0: { sl@0: __ASSERT_DEBUG(aSecurityPolicy != NULL, __SQLPANIC(ESqlPanicBadArgument)); sl@0: //Insert a new item in the security policies map. sl@0: CleanupStack::PushL(aSecurityPolicy); sl@0: const TUint8* mapKey = SecurityMapKeyL(aFileData.FileName()); sl@0: mapKey = ::CreateStrCopyLC(mapKey); sl@0: __ASSERT_DEBUG(!::SqlServer().SecurityMap().Entry(mapKey), __SQLPANIC(ESqlPanicObjExists)); sl@0: __SQLLEAVE_IF_ERROR(::SqlServer().SecurityMap().Insert(mapKey, aSecurityPolicy)); sl@0: CleanupStack::Pop(2);//iSecurityMap owns mapKey and aSecurityPolicy. sl@0: iSecureDbName = mapKey; sl@0: iSecurityPolicy = aSecurityPolicy; sl@0: // sl@0: DoCommonConstructCreateL(aFileData, ETrue); sl@0: } sl@0: sl@0: /** sl@0: Second phase construction method. Creates a new non-secure database file. sl@0: If the function fails, the database file will be closed and deleted. sl@0: sl@0: @param aFileData A reference to a TSqlSrvFileData instance, which keeps the database name, a sl@0: file session reference and some other database file related properties. sl@0: If this is a secure database, then the format of the name must be: sl@0: \:\<[SID]database file name excluding the path\>. sl@0: If this is a non-secure database, then the file name has to be the full database file name. sl@0: "[SID]" refers to SID of the application which creates the database. sl@0: */ sl@0: void CSqlSrvDatabase::ConstructCreateL(const TSqlSrvFileData& aFileData) sl@0: { sl@0: DoCommonConstructCreateL(aFileData, EFalse); sl@0: } sl@0: sl@0: //Called by the two "Contruct&Create" methods: ConstructCreateL() and ConstructCreateSecureL(). sl@0: //The aSecureDb parameter tells which method is the caller. sl@0: //The function performs common construction and initialization: sl@0: // - creates the database file sl@0: // - makes the journal file persistent sl@0: // - initializes the database compaction mode sl@0: // - stores the initial settings in the settings table, including the current collation dll name sl@0: // - stores the security settings and installs the authorizer, if aSecureDb is true sl@0: // - installs the user-defined functions sl@0: // - installs collations sl@0: //If the method fails and the error is not KErrAlreadyExists, the database file will be closed and deleted. sl@0: void CSqlSrvDatabase::DoCommonConstructCreateL(const TSqlSrvFileData& aFileData, TBool aSecureDb) sl@0: { sl@0: __ASSERT_DEBUG(!iDbHandle, __SQLPANIC(ESqlPanicInternalError)); sl@0: __ASSERT_DEBUG(aSecureDb ? iSecurityPolicy != NULL : ETrue, __SQLPANIC(ESqlPanicInternalError)); sl@0: CreateNewDbFileL(aFileData); sl@0: TDbFileCleanup dbFileCleanup(aFileData, iDbHandle); sl@0: CleanupStack::PushL(TCleanupItem(&DbFileCleanup, &dbFileCleanup)); sl@0: //Make the journal file persistent - done by SQLite automatically because the locking mode is EXCLUSIVE sl@0: //::ExecPragma(iDbHandle, iAuthorizerDisabled, KPersistentJournalPragma, KPersist); sl@0: //Init database compaction mode sl@0: TSqlCompactionMode compactionMode = aFileData.ConfigParams().iCompactionMode; sl@0: if(compactionMode == ESqlCompactionNotSet) sl@0: { sl@0: compactionMode = KSqlDefaultCompactionMode; sl@0: } sl@0: TInt currVacuumMode = -1; sl@0: __SQLLEAVE_IF_ERROR(::DbVacuumMode(iDbHandle, KNullDesC, currVacuumMode)); sl@0: //currVacuumMode == ESqliteVacuumOff ==> This is a database created not by the SQL server sl@0: InitCompactionL(compactionMode, aFileData.ConfigParams().iFreePageThresholdKb, aFileData.FileName(), (TSqliteVacuumMode)currVacuumMode); sl@0: CleanupStack::PushL(TCleanupItem(&CSqlSrvDatabase::CompactCleanup, this)); sl@0: //Store the initial settings in the settings table (including the current collation dll name) sl@0: StoreSettingsL(::SqlServer().CollationDllName(), KSqlNullDbConfigFileVersion, compactionMode); sl@0: if(aSecureDb) sl@0: { sl@0: //Store the security policies in the security policies tables. sl@0: TSqlDbSysSettings dbSysSettings(iDbHandle); sl@0: dbSysSettings.StoreSecurityPolicyL(*iSecurityPolicy); sl@0: } sl@0: InstallAuthorizerL(); sl@0: InstallUDFsL(); sl@0: InstallCollationsL(); sl@0: CleanupStack::Pop(2);//CompactCleanup, DbFileCleanup sl@0: SQLPROFILER_DB_CREATE((TUint)iDbHandle, aFileData.FileName()); sl@0: } sl@0: sl@0: /** sl@0: Second phase construction method. Opens an existing secure database file. sl@0: sl@0: @param aFileData A reference to a TSqlSrvFileData instance, which keeps the database name, a sl@0: file session reference and some other database file related properties. sl@0: If this is a secure database, then the format of the name must be: sl@0: \:\<[SID]database file name excluding the path\>. sl@0: If this is a non-secure database, then the file name has to be the full database file name. sl@0: "[SID]" refers to SID of the application which creates the database. sl@0: */ sl@0: void CSqlSrvDatabase::ConstructOpenSecureL(const TSqlSrvFileData& aFileData) sl@0: { sl@0: DoCommonConstructOpenL(aFileData, ETrue); sl@0: } sl@0: sl@0: /** sl@0: Second phase construction method. Opens an existing non-secure database file. sl@0: sl@0: @param aFileData A reference to a TSqlSrvFileData instance, which keeps the database name, a sl@0: file session reference and some other database file related properties. sl@0: If this is a secure database, then the format of the name must be: sl@0: \:\<[SID]database file name excluding the path\>. sl@0: If this is a non-secure database, then the file name has to be the full database file name. sl@0: "[SID]" refers to SID of the application which creates the database. sl@0: If this is application's private database, then the format of aFileData is as it is described sl@0: in TSqlSrvFileData::SetFromHandleL() comments. sl@0: @see TSqlSrvFileData::SetFromHandleL() sl@0: */ sl@0: void CSqlSrvDatabase::ConstructOpenL(const TSqlSrvFileData& aFileData) sl@0: { sl@0: DoCommonConstructOpenL(aFileData, EFalse); sl@0: } sl@0: sl@0: //Opens a database and does all necessary initializations sl@0: //Called by the two "Contruct&Open" methods: ConstructOpenL() and ConstructOpenSecureL(). sl@0: //The aSecureDb parameter tells which method is the caller. sl@0: //The function performs common construction and initialization: sl@0: // - opens the database file sl@0: // - installs the user-defined functions sl@0: // - installs collations sl@0: // - installs the authoriser callback sl@0: void CSqlSrvDatabase::DoCommonConstructOpenL(const TSqlSrvFileData& aFileData, TBool aSecureDb) sl@0: { sl@0: OpenExistingDbFileL(aFileData);//iDbHandle is valid after a successful call sl@0: //The user-defined collations must be installed before the possible database reindexing!!! sl@0: InstallCollationsL(); sl@0: if(!aFileData.IsReadOnly()) sl@0: {//No need to disable the authorizer since it is not installed yet. sl@0: //Make sure that the user-defined collation have been installed before the reindexing operation. sl@0: ProcessSettingsL(aFileData, KMainDb16); sl@0: } sl@0: if(aSecureDb) sl@0: { sl@0: const TUint8* mapKey = NULL; sl@0: //Load database security policy, update the security policy map sl@0: UpdateSecurityMapL(EFalse, aFileData, mapKey, iSecurityPolicy); sl@0: iSecureDbName = mapKey;//used in CSqlSrvDatabase destructor. sl@0: mapKey = NULL;//it is not used sl@0: //Check that the caller has at least one of {Schema, Read, Write} policies. sl@0: BasicSecurityPolicyCheckL(*iSecurityPolicy); sl@0: } sl@0: //Install user-defined functions. sl@0: InstallUDFsL(); sl@0: sl@0: //Install the authorizer. sl@0: InstallAuthorizerL(); sl@0: sl@0: SQLPROFILER_DB_OPEN((TUint)iDbHandle, aFileData.FileName()); sl@0: } sl@0: sl@0: /* sl@0: Implementation of the like() SQL function. This function implements sl@0: the user defined LIKE operator. The first argument to the function is the sl@0: pattern and the second argument is the string. So, the SQL statements: sl@0: A LIKE B sl@0: is implemented as like(B, A). sl@0: sl@0: @param aContext Function call context; sl@0: @param aArgc Number of LIKE arguments: 2 for the standard LIKE operator, 3 for the LIKE operator with an ESCAPE clause; sl@0: @param aArgv LIKE arguments; sl@0: sl@0: @internalComponent sl@0: */ sl@0: void CSqlSrvDatabase::LikeSqlFunc(sqlite3_context* aContext, int aArgc, sqlite3_value** aArgv) sl@0: { sl@0: TUint escapeChar = 0; sl@0: if(aArgc == 3) sl@0: { sl@0: //The escape character string must consist of a single UTF16 character. sl@0: //Otherwise, return an error. sl@0: const TUint16* esc = static_cast (sqlite3_value_text16(aArgv[2])); sl@0: if(!esc) sl@0: { sl@0: sqlite3_result_error(aContext, KErrMsg1, -1); sl@0: return; sl@0: } sl@0: if(User::StringLength(esc) != 1) sl@0: { sl@0: sqlite3_result_error(aContext, KErrMsg2, -1); sl@0: return; sl@0: } sl@0: escapeChar = *esc; sl@0: } sl@0: const TUint16* pattern = static_cast (sqlite3_value_text16(aArgv[0])); sl@0: const TUint16* candidate = static_cast (sqlite3_value_text16(aArgv[1])); sl@0: if(pattern && candidate) sl@0: { sl@0: TInt wildChar = '_'; sl@0: TInt wildSeqChar = '%'; sl@0: TPtrC16 patternStr(pattern, (TUint)sqlite3_value_bytes16(aArgv[0]) / sizeof(TUint16)); sl@0: TPtrC16 candidateStr(candidate, (TUint)sqlite3_value_bytes16(aArgv[1]) / sizeof(TUint16)); sl@0: TInt res = candidateStr.MatchC(patternStr, wildChar, wildSeqChar, escapeChar, 0/*collation level*/); sl@0: sqlite3_result_int(aContext, res >= 0); sl@0: //RDebug::Print(_L("--res=%d, pattern=%S, candidate=%S\r\n"), res, &patternStr, &candidateStr); sl@0: } sl@0: } sl@0: