os/persistentdata/persistentstorage/sql/SRC/Server/SqlSrvDbSysSettings.cpp
author sl@SLION-WIN7.fritz.box
Fri, 15 Jun 2012 03:10:57 +0200
changeset 0 bde4ae8d615e
permissions -rw-r--r--
First public contribution.
     1 // Copyright (c) 2006-2010 Nokia Corporation and/or its subsidiary(-ies).
     2 // All rights reserved.
     3 // This component and the accompanying materials are made available
     4 // under the terms of "Eclipse Public License v1.0"
     5 // which accompanies this distribution, and is available
     6 // at the URL "http://www.eclipse.org/legal/epl-v10.html".
     7 //
     8 // Initial Contributors:
     9 // Nokia Corporation - initial contribution.
    10 //
    11 // Contributors:
    12 //
    13 // Description:
    14 // Store/Load database system settings (database settings, security policies)
    15 // 
    16 //
    17 
    18 #include <f32file.h>
    19 #include <f32file64.h>
    20 #include <utf.h>				//CnvUtfConverter
    21 #include <sqldb.h>				//RSqlSecurityPolicy enum items
    22 #include "SqlSrvDbSysSettings.h"
    23 #include "SqlUtil.h"			//Sql2OsErrCode()
    24 #include "sqlite3.h"			//SQLITE API
    25 #include "SqliteSymbian.h"		//sqlite3SymbianLastOsError()
    26 #include "SqlSecurityImpl.h"	//CSqlSecurityPolicy
    27 #include "SqlSrvStatementUtil.h"//Global SQL execution functions
    28 #include "SqlSrvCollation.h"	//TSqlCollate
    29 #include "SqlSrvStrings.h"		//System table names
    30 #include "SqlSrvUtil.h"			//Global functions
    31 #include "SqlSrvFileData.h"		//TSqlSrvFileData
    32 #include "OstTraceDefinitions.h"
    33 #ifdef OST_TRACE_COMPILER_IN_USE
    34 #include "SqlSrvDbSysSettingsTraces.h"
    35 #endif
    36 #include "SqlTraceDef.h"
    37 
    38 extern TBool IsStatementSupported(const TDesC& aStatementIn, const TDesC& aDbName, TDes& aStatementOut);
    39 
    40 ///////////////////////////////////////////////////////////////////////////////////////////////////////////////
    41 /////////////////////////////        Local const data   ///////////////////////////////////////////////////////
    42 ///////////////////////////////////////////////////////////////////////////////////////////////////////////////
    43 
    44 enum
    45 	{
    46 	ESqlSystemVersion1 = 1,	//Base version
    47 	ESqlSystemVersion2 = 2,	//New field into "Settings" table - "Reserved" (unused, set to 0)
    48 	ESqlSystemVersion3 = 3,	//New field into "Settings" table - "CollationDllName"
    49 				            //Existing field in "Settings" table - "Reserved" - now used to store processed version of db config file
    50 	ESqlSystemVersion4 = 4	//"CompactionMode" field added
    51 	};
    52 
    53 //Current version of the system settings - reflects changes in the system tables.
    54 const TInt KSqlSystemVersion = ESqlSystemVersion4;
    55 
    56 //The database names in all statements are quoted to avoid the "sql injection" threat.
    57 //(At the moment there is no way to pass an invalid database name for these statements, because the database has to be attached
    58 // first and a parameter binding is used there. So, the quoting is just for safety and against changes in the future)
    59 
    60 _LIT(KCreateSettingsSql,"CREATE TABLE \"%S\".symbian_settings(Id INTEGER,Reserved INTEGER,CollationDllName TEXT, CompactionMode INTEGER);\x0");
    61 _LIT(KDropSettingsSql, "DROP TABLE IF EXISTS \"%S\".symbian_settings;\x0");
    62 _LIT(KInsertSettingsSql, "INSERT INTO \"%S\".symbian_settings VALUES(%d, %d,'%S',%d);\x0");
    63 _LIT(KUpdateCollationSettingsSql, "UPDATE \"%S\".symbian_settings SET CollationDllName='%S';\x0");
    64 _LIT(KUpdateFileVersionSettingsSql, "UPDATE \"%S\".symbian_settings SET Reserved=%d;\x0");
    65 _LIT(KGetSettingsSql, "SELECT * FROM \"%S\".symbian_settings;\x0");
    66 _LIT(KGetIndexSql, "SELECT name FROM \"%S\".sqlite_master WHERE type = 'index' AND sql LIKE '%%%S%%'\x0");
    67 _LIT(KSettingsTableCheckSql, "SELECT name from \"%S\".sqlite_master WHERE type = 'table' AND name = 'symbian_settings';\x0");
    68 _LIT(KReindexSql, "REINDEX \"%S\";\x0");
    69 
    70 _LIT8(KCreateSecuritySql,"CREATE TABLE symbian_security(Id INTEGER PRIMARY KEY AUTOINCREMENT,ObjectType INTEGER,ObjectName TEXT,PolicyType INTEGER,PolicyData BLOB);\x0");
    71 _LIT8(KInsertSecuritySql, "INSERT INTO symbian_security(ObjectType,ObjectName,PolicyType,PolicyData) VALUES(:V1,:V2,:V3,:V4);\x0");
    72 _LIT8(KGetSecuritySql, "SELECT * FROM symbian_security;\x0");
    73 
    74 _LIT8(KBeginTransactionSql, "BEGIN;\x0");
    75 _LIT8(KCommitTransactionSql, "COMMIT;\x0");
    76 _LIT8(KRollbackTransactionSql, "ROLLBACK;\x0");
    77 
    78 
    79 
    80 //KInsertSecuritySql statement - parameter indices
    81 const TInt KObjTypePrmIdx = 1;
    82 const TInt KObjNamePrmIdx = 2;
    83 const TInt KObjPolicyTypePrmIdx = 3;
    84 const TInt KObjPolicyDataPrmIdx = 4;
    85 
    86 //Default security policy - object type code and policy type code
    87 const TInt KDefaultObjType = -2;
    88 const TInt KDefaultPolicyType = -1;
    89 
    90 //Database security policy - object type code
    91 const TInt KDbObjType = -1;
    92 
    93 //KGetSettingsSql sql statement - column indices
    94 const TInt KSysVersionColIdx = 0;
    95 const TInt KConfigFileVersionColIdx = 1;
    96 const TInt KCollationDllNameColIdx = 2;
    97 const TInt KCompactionModeColIdx = 3;
    98 
    99 //KGetSecuritySql sql statement - column indices
   100 const TInt KObjTypeColIdx = 1;
   101 const TInt KObjNameColIdx = 2;
   102 const TInt KObjPolicyTypeColIdx = 3;
   103 const TInt KObjPolicyDataColIdx = 4;
   104 
   105 ///////////////////////////////////////////////////////////////////////////////////////////////////////////////
   106 /////////////////////////////        Local functions   ////////////////////////////////////////////////////////
   107 ///////////////////////////////////////////////////////////////////////////////////////////////////////////////
   108 
   109 //Finalzes the statement handle (aHandle argument). Used in stack cleanup operations.
   110 //Panic SqlDb 4 In _DEBUG mode if aHandle argument is NULL.
   111 static void FinalizeStatementHandle(void* aHandle)
   112 	{
   113 	__ASSERT_DEBUG(aHandle != NULL, __SQLPANIC2(ESqlPanicBadArgument));
   114 	sqlite3_stmt* stmtHandle = static_cast <sqlite3_stmt*> (aHandle);
   115 	(void)sqlite3_finalize(stmtHandle);
   116 	}
   117 
   118 //Transaction rollback. Used in stack cleanup operations.
   119 //Panic SqlDb 4 In _DEBUG mode if aDbHandle argument is NULL.
   120 static void RollbackTransaction(void* aDbHandle)
   121 	{
   122 	__ASSERT_DEBUG(aDbHandle != NULL, __SQLPANIC2(ESqlPanicBadArgument));
   123 	(void)::DbExecStmt8(reinterpret_cast <sqlite3*> (aDbHandle), KRollbackTransactionSql);
   124 	}
   125 
   126 ///////////////////////////////////////////////////////////////////////////////////////////////////////////////
   127 /////////////////////////////      TSqlDbSysSettings   ////////////////////////////////////////////////////////
   128 ///////////////////////////////////////////////////////////////////////////////////////////////////////////////
   129 
   130 /**
   131 Initializes TSqlDbSysSettings data members with default values.
   132 
   133 @param aDbHandle Database handle. TSqlDbSysSettings does not own aDbHandle argument.
   134 
   135 @panic SqlDb 4 In _DEBUG mode if aDbHandle argument is NULL.
   136 */
   137 TSqlDbSysSettings::TSqlDbSysSettings(sqlite3* aDbHandle) :
   138 	iDbHandle(aDbHandle)
   139 	{
   140 	__ASSERT_DEBUG(iDbHandle != NULL, __SQLPANIC(ESqlPanicBadArgument));
   141 	}
   142 
   143 /**
   144 Creates the database security policy table and stores the security policy in the table.
   145 
   146 @param aSecurityPolicy   Database security policies container, which data needs to be persisted.
   147 
   148 @leave KErrNoMemory, an out of memory condition has occurred.
   149                   Note that the function may also leave with some other database specific 
   150                   errors categorised as ESqlDbError, and other system-wide error codes.
   151 @panic SqlDb 2 In _DEBUG mode if iDbHandle is NULL (uninitialized TSqlDbSysSettings object).
   152 */
   153 void TSqlDbSysSettings::StoreSecurityPolicyL(const CSqlSecurityPolicy& aSecurityPolicyCon)
   154 	{
   155 	__ASSERT_DEBUG(iDbHandle != NULL, __SQLPANIC(ESqlPanicInvalidObj));
   156 	__SQLLEAVE_IF_ERROR(::DbExecStmt8(iDbHandle, KBeginTransactionSql()));
   157 	CleanupStack::PushL(TCleanupItem(&RollbackTransaction, iDbHandle));
   158 	__SQLLEAVE_IF_ERROR(::DbExecStmt8(iDbHandle, KCreateSecuritySql()));
   159 	StoreSecurityPoliciesL(aSecurityPolicyCon);
   160 	__SQLLEAVE_IF_ERROR(::DbExecStmt8(iDbHandle, KCommitTransactionSql()));
   161 	CleanupStack::Pop();//TCleanupItem(&RollbackTransaction, iDbHandle)
   162 	}
   163 
   164 /**
   165 Stores the database system settings in the settings table. The settings table is recreated.
   166 
   167 @param aDbName Logical database name: "main" for the main database or attached database name,
   168 @param aCollationDllName Collation dll name. It uniquely identifies the current collation method in use.
   169                          If the default collation method changes later then the database will be reindexed 
   170                          and the new collation dll name will replace the existing one in the settings table.
   171 @param aDbConfigFileVersion Current config file version or KSqlNullDbConfigFileVersion
   172 @param aCompactionMode Database compaction mode, one of TSqlCompactionMode enum item values (except ESqlCompactionNotSet)
   173 
   174 @see TSqlCompactionMode
   175 
   176 @leave KErrNoMemory, an out of memory condition has occurred.
   177                   Note that the function may also leave with some other database specific 
   178                   errors categorised as ESqlDbError, and other system-wide error codes.
   179 @panic SqlDb 2 In _DEBUG mode if iDbHandle is NULL (uninitialized TSqlDbSysSettings object).
   180 @panic SqlDb 4 In _DEBUG mode if aCompactionMode parameter value is invalid.
   181 */
   182 void TSqlDbSysSettings::StoreSettingsL(const TDesC& aDbName, const TDesC& aCollationDllName, TInt aDbConfigFileVersion, TSqlCompactionMode aCompactionMode)
   183 	{
   184 	__ASSERT_DEBUG(aCompactionMode == ESqlCompactionManual || aCompactionMode == ESqlCompactionBackground || aCompactionMode == ESqlCompactionAuto, __SQLPANIC(ESqlPanicBadArgument));
   185 	__ASSERT_DEBUG(iDbHandle != NULL, __SQLPANIC(ESqlPanicInvalidObj));
   186 	HBufC* buf = HBufC::NewLC(Max((TInt)sizeof(KDropSettingsSql), 
   187 							  Max((TInt)sizeof(KCreateSettingsSql), (TInt)sizeof(KInsertSettingsSql))) + 
   188 							  aDbName.Length() + aCollationDllName.Length() + 10);
   189 	TPtr sql = buf->Des();
   190 	//Begin transaction
   191 	__SQLLEAVE_IF_ERROR(::DbExecStmt8(iDbHandle, KBeginTransactionSql()));
   192 	CleanupStack::PushL(TCleanupItem(&RollbackTransaction, iDbHandle));	
   193 	//Recreate the "settings" table and store the default collation there (the DLL name)
   194 	sql.Format(KDropSettingsSql(), &aDbName);
   195 	__SQLLEAVE_IF_ERROR(::DbExecStmt16(iDbHandle, sql));
   196 	sql.Format(KCreateSettingsSql, &aDbName);
   197 	__SQLLEAVE_IF_ERROR(::DbExecStmt16(iDbHandle, sql));
   198 	sql.Format(KInsertSettingsSql(), &aDbName, KSqlSystemVersion, aDbConfigFileVersion, &aCollationDllName, aCompactionMode);
   199 	__SQLLEAVE_IF_ERROR(::DbExecStmt16(iDbHandle, sql));
   200 	//Commit transaction
   201 	__SQLLEAVE_IF_ERROR(::DbExecStmt8(iDbHandle, KCommitTransactionSql()));
   202 	CleanupStack::Pop();//TCleanupItem(&RollbackTransaction, iDbHandle)
   203 	CleanupStack::PopAndDestroy(buf);
   204 	}
   205 
   206 /**
   207 Read security policies from security policy table.
   208 
   209 The method does not guarantee that either the security policies will be read from the 
   210 security table and stored in aSecurityPolicy parameter or aSecurityPolicy argument 
   211 stays unchanged in a case of failure.
   212 
   213 @param aSecurityPolicyCon Security policies container which needs to be initializeed with 
   214 						  the database security policies.
   215 
   216 @leave KErrNoMemory, an out of memory condition has occurred;
   217 	   KErrGeneral, missing or invalid data in the system tables;
   218                   Note that the function may also leave with some other database specific 
   219                   errors categorised as ESqlDbError, and other system-wide error codes.
   220 @panic SqlDb 2 In _DEBUG mode if iDbHandle is NULL (uninitialized TSqlDbSysSettings object).
   221 */
   222 void TSqlDbSysSettings::LoadSecurityPolicyL(CSqlSecurityPolicy& aSecurityPolicyCon)
   223 	{
   224 	__ASSERT_DEBUG(iDbHandle != NULL, __SQLPANIC(ESqlPanicInvalidObj));
   225 	//Even if the version of the system settings is bigger than the current one (KSqlSystemVersion constant),
   226 	//I think that all future modifications of the system tables shall not affect the already existing
   227 	//fields. So it is correct to think that all information available in version 1 should be available 
   228 	//(and accessible) in all future versions of the system settings.
   229 	//Note: no attempt shall be made here to modify the system tables structure! There may be more than one 
   230 	//      connection to the database being processed!
   231 	//
   232 	//Prepare statement handle
   233 	sqlite3_stmt* stmtHandle = ::StmtPrepare8L(iDbHandle, KGetSecuritySql());
   234 	CleanupStack::PushL(TCleanupItem(&FinalizeStatementHandle, stmtHandle));
   235 	//Read the security policies
   236 	TBool defaultPolicySet = EFalse;
   237 	TInt dbPolicySetFlag = 0;
   238 	TInt err;
   239 	while((err = ::StmtNext(stmtHandle)) == KSqlAtRow)
   240 		{
   241 		TInt objType = KDefaultObjType - 1;
   242 		TPtrC objName(KNullDesC);
   243 		TInt policyType = RSqlSecurityPolicy::ESchemaPolicy - 1;
   244 		TSecurityPolicy policy = ReadCurrSecurityPolicyL(stmtHandle, objType, objName, policyType);
   245 		switch(objType)
   246 			{
   247 			case KDefaultObjType:
   248 				if(defaultPolicySet)
   249 					{
   250 					__SQLLEAVE(KErrGeneral);//two "default policy" records in the table
   251 					}
   252 				StoreDefaultSecurityPolicy(aSecurityPolicyCon, policy, dbPolicySetFlag);
   253 				defaultPolicySet = ETrue;
   254 				break;
   255 			case KDbObjType:
   256 				StoreDbSecurityPolicyL(aSecurityPolicyCon, policyType, policy, dbPolicySetFlag);
   257 				break;
   258 			case RSqlSecurityPolicy::ETable:
   259 				StoreDbObjSecurityPolicyL(aSecurityPolicyCon, objType, objName, policyType, policy);
   260 				break;
   261 			default:
   262 				__SQLLEAVE(KErrGeneral);
   263 				break;
   264 			}//end of "switch(aObjType)"
   265 		}
   266 	CleanupStack::PopAndDestroy();//cleanupItem (statement handle)
   267 	__SQLLEAVE_IF_ERROR(err);
   268 	if(!defaultPolicySet)
   269 		{
   270 		__SQLLEAVE(KErrGeneral);//no default policy
   271 		}
   272 	}
   273 	
   274 /**
   275 Loads the database system settings from the settings table.
   276 If the settings table does not exist then it is created with the default settings
   277 (the stored collation dll name will be empty and the stored database configuration 
   278 file version will be 0).
   279 
   280 @param aDbName Logical database name: "main" for the main database or attached database name,
   281 @param aCollationDllName Output parameter, will contain the stored collation dll name,
   282 @param aDbConfigFileVersion Output parameter, will contain the stored database config file version.
   283 @param aCompactionMode Output parameter. Database compaction mode (one of TSqlCompactionMode enum item values except ESqlCompactionNotSet).
   284 					   Note that the input value of this parameter might be ESqlCompactionManual if the this is a legacy database,
   285 					   created not by the SQL server.
   286 
   287 @see TSqlCompactionMode
   288 
   289 @leave KErrNoMemory, an out of memory condition has occurred.
   290        Note that the function may also leave with some other database specific 
   291        errors categorised as ESqlDbError, and other system-wide error codes..
   292 @panic SqlDb 2 In _DEBUG mode if iDbHandle is NULL (uninitialized TSqlDbSysSettings object).
   293 @panic SqlDb 7 In _DEBUG mode if the stored compaction mode is invalid.
   294 */	
   295 void TSqlDbSysSettings::LoadSettingsL(const TDesC& aDbName, TDes& aCollationDllName, TInt& aDbConfigFileVersion, TSqlCompactionMode& aCompactionMode)
   296 	{
   297 	__ASSERT_DEBUG(iDbHandle != NULL, __SQLPANIC(ESqlPanicInvalidObj));
   298 
   299 	aCollationDllName.Zero();
   300 	aDbConfigFileVersion = KSqlNullDbConfigFileVersion;	
   301 	if(aCompactionMode == ESqlCompactionNotSet)
   302 		{
   303 		aCompactionMode = KSqlDefaultCompactionMode;
   304 		}
   305 		
   306 	//If the system settings table does not exist then create it now.
   307 	//For a database created by the SQL Server this will only occur 
   308 	//when the database is being created in an application's private data cage -
   309 	//as part of this create call it is now being opened by the server.
   310 	//An externally created database is likely to not contain the settings table 
   311 	//and adding it here makes the database 'SQL Server-compatible'
   312 	if(!SettingsTableExistsL(aDbName))
   313 		{
   314 		StoreSettingsL(aDbName, aCollationDllName, KSqlNullDbConfigFileVersion, aCompactionMode); // store empty collation dll name, then reindexing will occur
   315 		}
   316 	else
   317 		{
   318 		//Get the settings from the existing table
   319 		TInt settingsVersion = 0;
   320 		GetSettingsL(aDbName, aCollationDllName, aDbConfigFileVersion, settingsVersion, aCompactionMode);
   321 		if(settingsVersion < KSqlSystemVersion)
   322 			{
   323 			//Recreate the settings table using the last version number format (this is what the old code did during reindexing)
   324 			StoreSettingsL(aDbName, aCollationDllName, aDbConfigFileVersion, aCompactionMode); // store empty collation dll name, then reindexing will occur
   325 			}
   326 		}
   327 	__ASSERT_DEBUG(aCompactionMode == ESqlCompactionManual || aCompactionMode == ESqlCompactionBackground || aCompactionMode == ESqlCompactionAuto, __SQLPANIC(ESqlPanicInternalError));
   328 	}
   329 	
   330 /**
   331 Retrieves the database system settings from the settings table.
   332 
   333 @param aDbName Logical database name: "main" for the main database or attached database name,
   334 @param aCollationDllName Output parameter, will contain the stored collation dll name,
   335 @param aDbConfigFileVersion Output parameter, will contain the stored database config file version,
   336 @param aSettingsVersion Output parameter, will contain the version of the settings table.
   337 @param aCompactionMode Output parameter. Database compaction mode (one of TSqlCompactionMode enum item values except ESqlCompactionNotSet).
   338 
   339 @see TSqlCompactionMode
   340 
   341 @leave KErrGeneral, either unable to retrieve the data from the settings table or the 
   342 					stored table version or config file version is invalid or the stored compaction mode is invalid.
   343 	   KErrOverflow, aCollationDllName is not large enough to store the name of the 
   344 	   				 collation dll that is stored in the settings table.
   345        KErrNoMemory, an out of memory condition has occurred.
   346 	   Note that the function may also leave with other system-wide error codes or SQL
   347 	   errors of ESqlDbError type
   348 @panic SqlDb 2 In _DEBUG mode if iDbHandle is NULL (uninitialized TSqlDbSysSettings object).
   349 */
   350 void TSqlDbSysSettings::GetSettingsL(const TDesC& aDbName, TDes& aCollationDllName, TInt& aDbConfigFileVersion, 
   351 									 TInt& aSettingsVersion, TSqlCompactionMode& aCompactionMode)
   352 	{
   353 	__ASSERT_DEBUG(iDbHandle != NULL, __SQLPANIC(ESqlPanicInvalidObj));
   354 
   355 	HBufC* buf = HBufC::NewLC(sizeof(KGetSettingsSql) + aDbName.Length());
   356 	TPtr sql = buf->Des();
   357 			
   358 	//Prepare statement handle
   359 	sql.Format(KGetSettingsSql(), &aDbName);
   360 	sqlite3_stmt* stmtHandle = ::StmtPrepare16L(iDbHandle, sql);
   361 	CleanupStack::PushL(TCleanupItem(&FinalizeStatementHandle, stmtHandle));
   362 	//Move to the first record
   363 	TInt err = ::StmtNext(stmtHandle);
   364 	__SQLLEAVE_IF_ERROR(err);
   365 	//Check that it is a valid row. The error is checked on the previous line. 
   366 	//The "if" bellow will check whether there is a valid record or not.
   367 	if(err != KSqlAtRow)
   368 		{
   369 		__SQLLEAVE(KErrGeneral);
   370 		}
   371 	//Get the system settings version 
   372 	aSettingsVersion = sqlite3_column_int(stmtHandle, KSysVersionColIdx);
   373 	if(aSettingsVersion < ESqlSystemVersion1)
   374 		{
   375 		__SQLLEAVE(KErrGeneral);
   376 		}
   377 	if(aSettingsVersion > ESqlSystemVersion2)
   378 		{
   379 		//The "ConfigFileVersion" column exists and is used to store the most recent
   380 		//version of the database's config file (if there is one) that has 
   381 		//been successfully processed
   382 		aDbConfigFileVersion = sqlite3_column_int(stmtHandle, KConfigFileVersionColIdx);
   383 		if(aDbConfigFileVersion < KSqlNullDbConfigFileVersion)
   384 			{
   385 			__SQLLEAVE(KErrGeneral);
   386 			}
   387 			
   388 		//The "CollationDllName" column exists and its value can be read.
   389         //The column type might be different than SQLITE_TEXT - malformed database.
   390         if(sqlite3_column_type(stmtHandle, KCollationDllNameColIdx) != SQLITE_TEXT)
   391             {
   392             __SQLLEAVE(KErrGeneral);   
   393             }
   394 		const void* ptr = sqlite3_column_text16(stmtHandle, KCollationDllNameColIdx);
   395         //Null column value - this might be an indication of an "out of memory" problem, if the column text  
   396         //is in UTF8 format. (sqlite3_column_text16() may allocate memory for UTF8->UTF16 conversion)
   397 		__SQLLEAVE_IF_NULL(const_cast<void*>(ptr));
   398         TPtrC16 src(reinterpret_cast <const TUint16*> (ptr));
   399         if(src.Length() > aCollationDllName.MaxLength())
   400             {
   401             __SQLLEAVE(KErrOverflow);	
   402             }
   403         aCollationDllName.Copy(src);
   404 		}
   405 	if(aSettingsVersion > ESqlSystemVersion3)
   406 		{
   407 		aCompactionMode = static_cast <TSqlCompactionMode> (sqlite3_column_int(stmtHandle, KCompactionModeColIdx));
   408 		if(aCompactionMode != ESqlCompactionManual && aCompactionMode != ESqlCompactionBackground && aCompactionMode != ESqlCompactionAuto)
   409 			{
   410 			__SQLLEAVE(KErrGeneral);
   411 			}
   412 		}
   413 	CleanupStack::PopAndDestroy();//TCleanupItem(&FinalizeStatementHandle, stmtHandle)	
   414 	CleanupStack::PopAndDestroy(buf);
   415 	}
   416 	
   417 /**
   418 Deletes and recreates all database indexes which use user-defined collation methods, from scratch.
   419 Also updates the database settings table with the name of the current collation dll.
   420 This is all performed as a single atomic transaction.
   421 
   422 @param aDbName Logical database name: "main" for the main database or attached database name,
   423 @param aCurrentCollationDllName The name of the current collation dll.
   424 
   425 @leave The function may leave with system-wide error codes or SQL errors of ESqlDbError type
   426 
   427 @panic SqlDb 2 In _DEBUG mode if iDbHandle is NULL (uninitialized TSqlDbSysSettings object).
   428 */
   429 void TSqlDbSysSettings::ReindexDatabaseL(const TDesC& aDbName, const TDesC& aCurrentCollationDllName)
   430 	{	
   431 	__ASSERT_DEBUG(iDbHandle != NULL, __SQLPANIC(ESqlPanicInvalidObj));
   432 
   433 	//Allocate memory for the SQL statements
   434 	HBufC* buf = HBufC::NewLC(Max((TInt)sizeof(KUpdateCollationSettingsSql), (TInt)sizeof(KReindexSql)) + 
   435 							  aDbName.Length() + aCurrentCollationDllName.Length());
   436 	TPtr sql = buf->Des();
   437 	
   438 	//Begin transaction
   439 	__SQLLEAVE_IF_ERROR(::DbExecStmt8(iDbHandle, KBeginTransactionSql()));
   440 	CleanupStack::PushL(TCleanupItem(&RollbackTransaction, iDbHandle));	
   441 	
   442 	const TSqlCollationUtil collationUtil(iDbHandle);
   443 	//Issue a "REINDEX" command for each collation that has at least one index using it
   444 	for(TInt i = collationUtil.CollationCount()-1; i >= 0 ;--i)
   445 		{
   446 		TPtrC collationName = collationUtil.CollationName(i);
   447 		if(IndexExistsL(aDbName, collationName))
   448 			{
   449 			sql.Format(KReindexSql(), &collationName);
   450 			__SQLLEAVE_IF_ERROR(::DbExecStmt16(iDbHandle, sql));
   451 			}
   452 		}
   453 		
   454 	//Update the settings table to store the current collation DLL name
   455 	sql.Format(KUpdateCollationSettingsSql(), &aDbName, &aCurrentCollationDllName);
   456 	__SQLLEAVE_IF_ERROR(::DbExecStmt16(iDbHandle, sql));
   457 
   458 	//Commit transaction
   459 	__SQLLEAVE_IF_ERROR(::DbExecStmt8(iDbHandle, KCommitTransactionSql()));
   460 	CleanupStack::Pop(); //TCleanupItem(&RollbackTransaction, iDbHandle)
   461 	CleanupStack::PopAndDestroy(buf);
   462 	}
   463 
   464 /**
   465 Performs any necessary configuration file updates to the database if a 
   466 configuration file exists for the database that has not yet been processed.
   467 If a configuration file is processed then the database settings table will 
   468 be updated with the version of the configuration file that was processed.
   469 
   470 @param aStoredDbConfigFileVersion The configuration file version that is
   471 								  currently stored in the settings table
   472 @param aFileData The database file data
   473 
   474 @param aDbName Logical database name: "main" for the main database or attached database name
   475 
   476 @leave One of the system-wide error codes or SQL errors of ESqlDbError type
   477 
   478 @panic SqlDb 2 In _DEBUG mode if iDbHandle is NULL (uninitialized TSqlDbSysSettings object)
   479 */
   480 void TSqlDbSysSettings::ConfigureDatabaseL(TInt aStoredDbConfigFileVersion, const TSqlSrvFileData& aFileData, 
   481 										   const TDesC& aDbName)
   482 	{	
   483 	SQL_TRACE_INTERNALS(OstTrace1(TRACE_INTERNALS, TSQLDBSYSSETTINGS_CONFIGUREDATABASEL_ENTRY, "Entry;0x%X;TSqlDbSysSettings::ConfigureDatabaseL", (TUint)this));    
   484 	
   485 	__ASSERT_DEBUG(iDbHandle != NULL, __SQLPANIC(ESqlPanicInvalidObj));
   486 
   487 	if(!aFileData.IsSecureFileNameFmt())
   488 		{
   489 		//As a first implementation, config files will only be supported for 
   490 		//shared, secure databases - not for private, secure databases or public databases
   491 		SQL_TRACE_INTERNALS(OstTrace1(TRACE_INTERNALS, TSQLDBSYSSETTINGS_CONFIGUREDATABASEL_EXIT1, "Exit;0x%X;TSqlDbSysSettings::ConfigureDatabaseL;Not a secure db", (TUint)this));    
   492 		return;	
   493 		}
   494 		
   495 	//Determine whether there is a configuration file 
   496 	//for this database that has not yet been processed
   497 	const CDbConfigFiles* dbConfigFiles = aFileData.DbConfigFiles();
   498 	if(dbConfigFiles)
   499 		{
   500 		//Get the database file name and extension to search for a corresponding config file
   501 		TPtrC dbName = aFileData.FileName();
   502 		TParse parse;
   503 		__SQLLEAVE_IF_ERROR(parse.Set(dbName, NULL, NULL));
   504 		TPtrC dbFileNameAndExt = parse.NameAndExt();
   505 		HBufC* matchingConfigFile = dbConfigFiles->FindConfigFile(dbFileNameAndExt);
   506 		if(matchingConfigFile)
   507 			{
   508 			//There exists a config file for this database - now check whether its
   509 			//version is greater than the version stored in the settings table and 
   510 			//only process the file if it is
   511 			TFileName configFileName;
   512 			configFileName.Copy(matchingConfigFile->Des());
   513 			TInt offset = configFileName.LocateReverse('.') ;
   514 			if(KErrNotFound != offset)
   515 				{
   516 				TInt lengthOfExt = configFileName.Length() - offset - 1;
   517 				TPtrC versionNum = configFileName.Right(lengthOfExt);				
   518 				TLex lex(versionNum);
   519 				TInt fileVersion = 0;
   520 				if(KErrNone == lex.Val(fileVersion))
   521 					{
   522 					if(fileVersion > aStoredDbConfigFileVersion)
   523 						{
   524 						//The latest version of the configuration file has not yet been processed, so do it now
   525 						SQL_TRACE_INTERNALS(OstTraceExt2(TRACE_INTERNALS, TSQLDBSYSSETTINGS_CONFIGUREDATABASEL1, "0x%X;TSqlDbSysSettings::ConfigureDatabaseL;Config file '%S'", (TUint)this, __SQLPRNSTR(configFileName)));	
   526 						ExecuteConfigurationUpdateL(aFileData, configFileName, fileVersion, aDbName);
   527 						SQL_TRACE_INTERNALS(OstTraceExt2(TRACE_INTERNALS, TSQLDBSYSSETTINGS_CONFIGUREDATABASEL2, "0x%X;TSqlDbSysSettings::ConfigureDatabaseL;Config file '%S' was processed, no errors", (TUint)this, __SQLPRNSTR(configFileName)));	
   528 						}
   529 					else
   530 						{
   531 						SQL_TRACE_INTERNALS(OstTraceExt2(TRACE_INTERNALS, TSQLDBSYSSETTINGS_CONFIGUREDATABASEL3, "0x%X;TSqlDbSysSettings::ConfigureDatabaseL;Config file '%S' as this or a later version has already been processed", (TUint)this, __SQLPRNSTR(configFileName)));	
   532 						}
   533 					}
   534 				else
   535 					{
   536 					//Invalid config file name extension - it cannot be converted to an integer
   537 					__SQLLEAVE(KErrCorrupt);
   538 					}
   539 				}
   540 			else
   541 				{
   542 				//Invalid config file name - it doesn't contain the character '.'
   543 				__SQLLEAVE(KErrCorrupt);
   544 				}
   545 			}
   546 		else
   547 			{
   548 			SQL_TRACE_INTERNALS(OstTraceExt2(TRACE_INTERNALS, TSQLDBSYSSETTINGS_CONFIGUREDATABASEL4, "0x%X;TSqlDbSysSettings::ConfigureDatabaseL;No config file found for database '%S'", (TUint)this, __SQLPRNSTR(dbFileNameAndExt)));	
   549 			}
   550 		}		
   551 	SQL_TRACE_INTERNALS(OstTrace1(TRACE_INTERNALS, TSQLDBSYSSETTINGS_CONFIGUREDATABASEL_EXIT2, "Exit;0x%X;TSqlDbSysSettings::ConfigureDatabaseL", (TUint)this));    
   552 	}
   553 	
   554 /**
   555 Executes the supported operations specified in the given database 
   556 configuration file and updates the settings table to store the 
   557 version of this configuration file.
   558 
   559 @param aFileData The database file data
   560 @param aMatchingConfigFile The configuration file that is to be processed
   561 @param aDbConfigFileVersion The configuration file version
   562 @param aDbName Logical database name: "main" for the main database or attached database name
   563 
   564 @leave KErrNoMemory, if an out of memory condition occurs.
   565        One of the other system-wide error codes if the configuration
   566        file fails to be opened or read.
   567        One of the SQL errors of ESqlDbError type if the update to the
   568        database settings table fails
   569 
   570 @panic SqlDb 2 In _DEBUG mode if iDbHandle is NULL (uninitialized TSqlDbSysSettings object).
   571 */
   572 void TSqlDbSysSettings::ExecuteConfigurationUpdateL(const TSqlSrvFileData& aFileData, 
   573 													const TDesC& aMatchingConfigFile,
   574 													TInt aDbConfigFileVersion,
   575 													const TDesC& aDbName)
   576 	{
   577 	__ASSERT_DEBUG(iDbHandle != NULL, __SQLPANIC(ESqlPanicInvalidObj));
   578 														
   579 	//Execute the specified database config file operations that are supported
   580 #ifdef SYSLIBS_TEST
   581 	TDriveUnit drive = EDriveC;
   582 #else
   583 	TDriveUnit drive = EDriveZ;
   584 #endif			
   585 	TFileName configFilePath;
   586 	TDriveName drvName = drive.Name();
   587 	configFilePath.Append(drvName);
   588 	configFilePath.Append(aFileData.PrivatePath());
   589 	configFilePath.Append(aMatchingConfigFile);
   590 	//If this method leaves then either the config file could not be 
   591 	//opened or read or an out of memory condition occured. Either way
   592 	//another attempt will be made to process the config file when the
   593 	//database is next opened
   594 	DoExecuteDbConfigFileOpsL(aFileData.Fs(), configFilePath, aDbName);
   595 												
   596 	//Now update the settings table to store the current version of the database config file.
   597 	//If this fails then another attempt will be made to process the config file and update
   598 	//the settings table when the database is next opened
   599 	TBuf<sizeof(KUpdateFileVersionSettingsSql) + KMaxFileName + 10> buf;
   600 	buf.Format(KUpdateFileVersionSettingsSql(), &aDbName, aDbConfigFileVersion);
   601 	__SQLLEAVE_IF_ERROR(::DbExecStmt16(iDbHandle, buf));
   602 	}
   603 
   604 /**
   605 Opens the specified database configuration file and executes those operations that 
   606 are specified in it which are currently supported (currently only 'CREATE INDEX' 
   607 operations are supported and will be executed).
   608 
   609 @param aFs The server's file server session
   610 @param aConfigFilePath The configuration file path
   611 @param aDbName Logical database name: "main" for the main database or attached database name
   612 
   613 @leave KErrNoMemory, if an out of memory condition occurs.
   614        One of the other system-wide error codes if the configuration
   615        file fails to be opened or read
   616 
   617 @panic SqlDb 2 In _DEBUG mode if iDbHandle is NULL (uninitialized TSqlDbSysSettings object).
   618 */
   619 void TSqlDbSysSettings::DoExecuteDbConfigFileOpsL(RFs& aFs, const TDesC& aConfigFilePath, const TDesC& aDbName)
   620 	{
   621 	__ASSERT_DEBUG(iDbHandle != NULL, __SQLPANIC(ESqlPanicInvalidObj));
   622 	
   623 	//Open the config file and read it into a buffer
   624 	RFile64 file;
   625 	__SQLLEAVE_IF_ERROR(file.Open(aFs, aConfigFilePath, EFileRead)); 
   626 	CleanupClosePushL(file);
   627 	TInt64 size = 0;
   628 	__SQLLEAVE_IF_ERROR(file.Size(size));
   629 	if(size == 0)
   630 		{
   631 		//Config file is empty so just return
   632 		SQL_TRACE_INTERNALS(OstTraceExt1(TRACE_INTERNALS, TSQLDBSYSSETTINGS_DOEXECUTEDBCONFIGFILEOPSL, "0;TSqlDbSysSettings::DoExecuteDbConfigFileOpsL();Config file %S is empty", __SQLPRNSTR(aConfigFilePath)));	
   633 		CleanupStack::PopAndDestroy(); // file
   634 		return;
   635 		}
   636 	HBufC8* buffer = HBufC8::NewLC(size);
   637 	TPtr8 bufPtr = buffer->Des();
   638 	__SQLLEAVE_IF_ERROR(file.Read(bufPtr));
   639 
   640 	//Convert buffer to Unicode for processing (the configuration file is expected to be ASCII or UTF-8)
   641 	HBufC16* buf16 = CnvUtfConverter::ConvertToUnicodeFromUtf8L(bufPtr);
   642 	CleanupStack::PushL(buf16);			
   643 	TPtrC16 ptr16 = buf16->Des();
   644 	
   645 	//Skip the BOM (byte ordering mark) at the start if there is one
   646 	if((ptr16.Locate(TChar(0xFEFF)) == 0) || (ptr16.Locate(TChar(0xFFFE)) == 0))
   647 		{
   648 		ptr16.Set(ptr16.Mid(1));	
   649 		}
   650 	 
   651 	//Parse the file contents and execute the specified
   652 	//config statements that are supported
   653 	ParseFileL(ptr16, aDbName);
   654 
   655 	CleanupStack::PopAndDestroy(3); // buf16, buffer, file
   656 	}
   657 	
   658 /**
   659 Parses the configuration file buffer, reading each SQL statement
   660 and processing it.
   661 
   662 @param aBuffer Buffer containing the configuration file contents
   663 @param aDbName Logical database name: "main" for the main database or attached database name
   664 
   665 @leave KErrNoMemory, if an out of memory condition occurs
   666 */
   667 void TSqlDbSysSettings::ParseFileL(const TDesC& aBuffer, const TDesC& aDbName)
   668 	{
   669 	TLex fileParser(aBuffer);
   670 	//While not end of file
   671 	while (!fileParser.Eos())
   672 		{
   673 		//Get the next of the SQL statements, which are seperated by semicolons
   674 		fileParser.SkipSpaceAndMark(); // skip any preceding whitespace before the next statement
   675 		while ((!fileParser.Eos()) && (fileParser.Peek() != ';'))
   676 			{
   677 			fileParser.Inc(); // continue to next character
   678 			}
   679 		TPtrC stmt = fileParser.MarkedToken(); // extract the marked token
   680 		fileParser.Inc();  // to skip the terminating ';' for next iteration
   681 	
   682 		//Process this statement
   683 		ProcessStatementL(stmt, aDbName);
   684 		}
   685 	}
   686 
   687 /**
   688 Process an SQL statement contained in the configuration file
   689 and executes it if it is a supported statement.
   690 Currently only 'CREATE INDEX' statements are supported.
   691 The statement can contain comments that are understood by SQLite
   692 - these can be in the form of an SQL comment or a 'C' comment, 
   693 as defined at http://www.sqlite.org/lang_comment.html.
   694 
   695 @param aStmt An SQL statement string, stripped of any whitespace 
   696 			 at the beginning of it
   697 @param aDbName Logical database name: "main" for the main database or attached database name
   698 
   699 @leave KErrNoMemory, if an out of memory condition occurs
   700 */
   701 void TSqlDbSysSettings::ProcessStatementL(const TDesC& aStmt, const TDesC& aDbName)
   702 	{	
   703 	SQL_TRACE_INTERNALS(OstTraceExt2(TRACE_INTERNALS, TSQLDBSYSSETTINGS_PROCESSSTATEMENTL_ENTRY, "Entry;0x%X;TSqlDbSysSettings::ProcessStatementL;Processing statement '%S'", (TUint)this, __SQLPRNSTR(aStmt)));	
   704 	
   705 	//If the statement only contained whitespace then just return
   706 	if(aStmt.Length() == 0)
   707 		{
   708 		SQL_TRACE_INTERNALS(OstTrace1(TRACE_INTERNALS, TSQLDBSYSSETTINGS_PROCESSSTATEMENTL_EXIT1, "Exit;0x%X;TSqlDbSysSettings::ProcessStatementL;The statement ignored because contains only whitespace", (TUint)this));	
   709 		return;	
   710 		}
   711 		
   712 	//Check that the statement does not contain an unsupported comment style
   713 	if(KErrNotFound != aStmt.Find(_L("//")))
   714 		{
   715 		//The statement contains '//' which is an unsupported comment style, but rather
   716 		//than leave here and cause the full file to fail, we just ignore this statement
   717 		SQL_TRACE_INTERNALS(OstTrace1(TRACE_INTERNALS, TSQLDBSYSSETTINGS_PROCESSSTATEMENTL_EXIT2, "Exit;0x%X;TSqlDbSysSettings::ProcessStatementL;The statement ignored because contains invalid comment style", (TUint)this));	
   718 		return;
   719 		}
   720 		
   721 	HBufC* stmtBuf = HBufC::NewLC(aStmt.Length() + 1 + aDbName.Length() + 1);
   722 	TPtr stmtPtr = stmtBuf->Des();
   723 	//If the 'aStmt' string is an SQL statement that is supported then execute it
   724 	if(::IsStatementSupported(aStmt, aDbName, stmtPtr))
   725 		{
   726 		TInt err = ::DbExecStmt16(iDbHandle, stmtPtr);
   727 		if(KErrNone == err)
   728 			{
   729 			SQL_TRACE_INTERNALS(OstTrace1(TRACE_INTERNALS, TSQLDBSYSSETTINGS_PROCESSSTATEMENTL1, "0x%X;TSqlDbSysSettings::ProcessStatementL;Successfully executed statement", (TUint)this));		
   730 			}
   731 		else
   732 			{
   733 			SQL_TRACE_INTERNALS(OstTraceExt2(TRACE_INTERNALS, TSQLDBSYSSETTINGS_PROCESSSTATEMENTL2, "0x%X;TSqlDbSysSettings::ProcessStatementL;Failed to execute the statement;err=%d", (TUint)this, err));		
   734 			if(err == KErrNoMemory)
   735 				{
   736 				__SQLLEAVE(err);	
   737 				}
   738 			}
   739 		}
   740 	else
   741 		{
   742 		SQL_TRACE_INTERNALS(OstTrace1(TRACE_INTERNALS, TSQLDBSYSSETTINGS_PROCESSSTATEMENTL3, "0x%X;TSqlDbSysSettings::ProcessStatementL;Non-supported statement, will be ignored", (TUint)this));		
   743 		}
   744 	CleanupStack::PopAndDestroy(); // stmtBuf
   745 	SQL_TRACE_INTERNALS(OstTrace1(TRACE_INTERNALS, TSQLDBSYSSETTINGS_PROCESSSTATEMENTL_EXIT3, "Exit;0x%X;TSqlDbSysSettings::ProcessStatementL", (TUint)this));  
   746 	}
   747 
   748 
   749 /**
   750 Stores the security policies into the security policies table.
   751 
   752 @param aSecurityPolicyCon Security policies container, which data needs to be persisted.
   753 
   754 @leave KErrNoMemory, an out of memory condition has occurred;
   755 	   KErrArgument, system table name found in the list of the tables, which need special protection;
   756                   	 Note that the function may also leave with some other database specific 
   757                   	 errors categorised as ESqlDbError, and other system-wide error codes.
   758 
   759 @panic SqlDb 4 In _DEBUG mode if aSecurityPolicy argument is NULL.
   760 */
   761 void TSqlDbSysSettings::StoreSecurityPoliciesL(const CSqlSecurityPolicy& aSecurityPolicyCon)
   762 	{
   763 	//Prepare the INSERT sql statement
   764 	sqlite3_stmt* stmtHandle = ::StmtPrepare8L(iDbHandle, KInsertSecuritySql());
   765 	CleanupStack::PushL(TCleanupItem(&FinalizeStatementHandle, stmtHandle));
   766 	//Store default security policy into the table
   767 	StoreSecurityPolicyL(stmtHandle, KDefaultObjType, KNullDesC, KDefaultPolicyType, aSecurityPolicyCon.DefaultPolicy());
   768 	//Store database security policies into the table
   769 	StoreSecurityPolicyL(stmtHandle, KDbObjType, KNullDesC, RSqlSecurityPolicy::ESchemaPolicy, aSecurityPolicyCon.DbPolicy(RSqlSecurityPolicy::ESchemaPolicy));
   770 	StoreSecurityPolicyL(stmtHandle, KDbObjType, KNullDesC, RSqlSecurityPolicy::EReadPolicy, aSecurityPolicyCon.DbPolicy(RSqlSecurityPolicy::EReadPolicy));
   771 	StoreSecurityPolicyL(stmtHandle, KDbObjType, KNullDesC, RSqlSecurityPolicy::EWritePolicy, aSecurityPolicyCon.DbPolicy(RSqlSecurityPolicy::EWritePolicy));
   772 	//Store database objects security policies into the table
   773 	TSqlSecurityPolicyIterator it(aSecurityPolicyCon);
   774 	RSqlSecurityPolicy::TObjectType objectType;
   775 	TPtrC objectName;
   776 	RSqlSecurityPolicy::TPolicyType policyType;
   777 	TSecurityPolicy policy;
   778 	while(it.Next(objectType, objectName, policyType, policy))
   779 		{
   780 		if(objectType == RSqlSecurityPolicy::ETable && ::IsSystemTableName(objectName))
   781 	   		{
   782 	   		//Clients are not allowed to put system table names in the RSqlSecurityPolicy container.
   783 			__SQLLEAVE(KErrArgument);		   			
   784 	   		}
   785 		StoreSecurityPolicyL(stmtHandle, objectType, objectName, policyType, policy);
   786 		}
   787 	CleanupStack::PopAndDestroy();//cleanupItem (statement handle)
   788 	}
   789 
   790 /**
   791 Stores a security policy object in the security policies table.
   792 
   793 @param aStmtHandle Statement handle.
   794 @param aObjType Database object type: default security policies, database security policies, table security policies
   795 @param aObjName Database object name. It is expected to be a non-empty string only for tables.
   796 @param aPolicyType Security policy type: schema, write, read.
   797 @param aPolicy Security policy object.
   798 
   799 @leave KErrNoMemory, an out of memory condition has occurred;
   800                   	 Note that the function may also leave with some other database specific 
   801                   	 errors categorised as ESqlDbError, and other system-wide error codes.
   802 
   803 @panic SqlDb 4 In _DEBUG mode if aHandle argument is NULL.
   804 */
   805 void TSqlDbSysSettings::StoreSecurityPolicyL(sqlite3_stmt* aStmtHandle, TInt aObjType, const TDesC& aObjName, 
   806 											 TInt aPolicyType, const TSecurityPolicy& aPolicy)
   807 	{
   808 	__ASSERT_DEBUG(aStmtHandle != NULL, __SQLPANIC(ESqlPanicBadArgument));
   809 	__SQLLEAVE_IF_ERROR(::StmtReset(aStmtHandle));
   810 	__SQLLEAVE_IF_ERROR(BindSecurityPolicyPrm(aStmtHandle, aObjType, aObjName, aPolicyType, aPolicy));
   811 	__SQLLEAVE_IF_ERROR(::StmtExec(aStmtHandle));
   812 	}
   813 
   814 /**
   815 Binds the parameter values for the insert SQL statement for security policies table.
   816 
   817 @param aStmtHandle Statement handle.
   818 @param aObjType Database object type: default security policy, database security policies, table security policies
   819 @param aObjName Database object name. It is expected to be a non-empty string only for tables.
   820 @param aPolicyType Security policy type: schema, write, read.
   821 @param aPolicy Security policy object.
   822 
   823 @see TSqlDbSysSettings::StoreSecurityPolicyL()
   824 
   825 @panic SqlDb 4 In _DEBUG mode if aHandle argument is NULL.
   826 */
   827 TInt TSqlDbSysSettings::BindSecurityPolicyPrm(sqlite3_stmt* aStmtHandle, TInt aObjType, const TDesC& aObjName, 
   828 												  TInt aPolicyType, const TSecurityPolicy& aPolicy)
   829 	{
   830 	(void)sqlite3SymbianLastOsError();//clear last OS error
   831 	TInt err = sqlite3_bind_int(aStmtHandle, KObjTypePrmIdx, aObjType);
   832 	if(err == SQLITE_OK)
   833 		{
   834 		err = sqlite3_bind_text16(aStmtHandle, KObjNamePrmIdx, aObjName.Ptr(), aObjName.Length() * sizeof(TText), SQLITE_STATIC);
   835 		if(err == SQLITE_OK)
   836 			{
   837 			err = sqlite3_bind_int(aStmtHandle, KObjPolicyTypePrmIdx, aPolicyType);
   838 			if(err == SQLITE_OK)
   839 				{
   840 				TPtrC8 policyData(aPolicy.Package());
   841 				err = sqlite3_bind_blob(aStmtHandle, KObjPolicyDataPrmIdx, policyData.Ptr(), policyData.Length(), SQLITE_STATIC);
   842 				}
   843 			}
   844 		}
   845 	return ::Sql2OsErrCode(err, sqlite3SymbianLastOsError());
   846 	}
   847 
   848 /**
   849 Reads a record from security policies table.
   850 
   851 @param aStmtHandle Statement handle. It cannot be NULL.
   852 @param aSecurityPolicy Security policies container.
   853 @param aObjType Output parameter, will be initialized with the database object type: KDefaultObjType,
   854 				KDbObjType, RSqlSecurityPolicy::ETable.
   855 @param aObjName Output parameter, database object name (for example, table name), which is protected by the
   856 				current security policy.
   857 @param aPolicyType Output parameter, will be initialized with the database policy type: RSqlSecurityPolicy::EReadPolicy,
   858 				   RSqlSecurityPolicy::EWritePolicy, RSqlSecurityPolicy::ESchemaPolicy.
   859 @return The created security policy object.
   860 
   861 @leave KErrGeneral, invalid security policy data;
   862 	   KErrNoMemory, Out of memory.
   863 */
   864 TSecurityPolicy TSqlDbSysSettings::ReadCurrSecurityPolicyL(sqlite3_stmt* aStmtHandle, TInt& aObjType, 
   865 														   TPtrC& aObjName, TInt& aPolicyType)
   866 	{
   867 	__ASSERT_DEBUG(aStmtHandle != NULL, __SQLPANIC(ESqlPanicBadArgument));
   868 	aObjType = sqlite3_column_int(aStmtHandle, KObjTypeColIdx);
   869     //The "ObjectName" column type might be different than SQLITE_TEXT - malformed database.
   870     if(sqlite3_column_type(aStmtHandle, KObjNameColIdx) != SQLITE_TEXT)
   871         {
   872         __SQLLEAVE(KErrGeneral);   
   873         }
   874     const void* text = sqlite3_column_text16(aStmtHandle, KObjNameColIdx);
   875     //Null column value - this might be an indication of an "out of memory" problem, if the column text  
   876     //is in UTF8 format. (sqlite3_column_text16() may allocate memory for UTF8->UTF16 conversion)
   877     __SQLLEAVE_IF_NULL(const_cast<void*>(text));
   878 	TInt len = (TUint)sqlite3_column_bytes16(aStmtHandle, KObjNameColIdx) / sizeof(TUint16);
   879 	aObjName.Set(reinterpret_cast <const TUint16*> (text), len);
   880 	aPolicyType = sqlite3_column_int(aStmtHandle, KObjPolicyTypeColIdx);
   881 	len = sqlite3_column_bytes(aStmtHandle, KObjPolicyDataColIdx);
   882 	if(len != sizeof(TSecurityPolicy))
   883 		{
   884 		//Check if the error is "out of memory" (which may happen when retrieving text column data
   885 		//and the column encoding is different, in which case  the column text has to be converted 
   886 		//and a new block of memory has to be allocated for the conversion).
   887 		TInt err2 = ::StmtReset(aStmtHandle);
   888 		__SQLLEAVE(err2 == KErrNoMemory ? KErrNoMemory : KErrGeneral);
   889 		}
   890 	const void* data = sqlite3_column_blob(aStmtHandle, KObjPolicyDataColIdx);
   891 	TSecurityPolicy policy;
   892 	policy.Set(TPtrC8(reinterpret_cast <const TUint8*> (data), len));
   893 	return policy;
   894 	}
   895 
   896 /**
   897 Stores the default security policy into aSecurityPolicyCon container.
   898 Initialises all database security policies not set yet with the default security policy.
   899 
   900 @param aSecurityPolicyCon Security policies container.
   901 @param aPolicy Default security policy object
   902 @param aDbPolicySetFlag Bit flag. Keeps information which database security policies are set and which aren't.
   903 
   904 @see CSqlSecurityPolicy::SetDbPolicy()
   905 */
   906 void TSqlDbSysSettings::StoreDefaultSecurityPolicy(CSqlSecurityPolicy& aSecurityPolicyCon, 
   907 												   const TSecurityPolicy& aPolicy, TInt aDbPolicySetFlag)
   908 	{
   909 	aSecurityPolicyCon.SetDefaultPolicy(aPolicy);
   910 	if(!(aDbPolicySetFlag & (1 << RSqlSecurityPolicy::ESchemaPolicy)))
   911 		{
   912 		aSecurityPolicyCon.SetDbPolicy(RSqlSecurityPolicy::ESchemaPolicy, aPolicy);
   913 		}
   914 	if(!(aDbPolicySetFlag & (1 << RSqlSecurityPolicy::EReadPolicy)))
   915 		{
   916 		aSecurityPolicyCon.SetDbPolicy(RSqlSecurityPolicy::EReadPolicy, aPolicy);
   917 		}
   918 	if(!(aDbPolicySetFlag & (1 << RSqlSecurityPolicy::EWritePolicy)))
   919 		{
   920 		aSecurityPolicyCon.SetDbPolicy(RSqlSecurityPolicy::EWritePolicy, aPolicy);
   921 		}
   922 	}
   923 
   924 /**
   925 Stores a database security policy into aSecurityPolicyCon container.
   926 
   927 @param aSecurityPolicyCon Security policies container.
   928 @param aPolicyType Database policy type: RSqlSecurityPolicy::EReadPolicy,
   929 				   RSqlSecurityPolicy::EWritePolicy, RSqlSecurityPolicy::ESchemaPolicy.
   930 @param aPolicy Database security policy object
   931 @param aDbPolicySetFlag Bit flag. Keeps information which database security policies are set and which aren't.
   932 						The parameter value may change if some of the database policies are set by
   933 						StoreDbSecurityPolicyL().
   934 
   935 @leave KErrGeneral Invalid security policy type or the corresponding database security policy has been set already.
   936 */
   937 void TSqlDbSysSettings::StoreDbSecurityPolicyL(CSqlSecurityPolicy& aSecurityPolicyCon, TInt aPolicyType,
   938 											   const TSecurityPolicy& aPolicy, TInt& aDbPolicySetFlag)
   939 	{
   940 	if(aPolicyType < (TInt)RSqlSecurityPolicy::ESchemaPolicy || aPolicyType > (TInt)RSqlSecurityPolicy::EWritePolicy)
   941 		{
   942 		__SQLLEAVE(KErrGeneral);
   943 		}
   944 	if(aDbPolicySetFlag & (1 << aPolicyType))
   945 		{
   946 		__SQLLEAVE(KErrGeneral);
   947 		}
   948 	aSecurityPolicyCon.SetDbPolicy(static_cast <RSqlSecurityPolicy::TPolicyType> (aPolicyType), aPolicy);
   949 	aDbPolicySetFlag |= (1 << aPolicyType);
   950 	}
   951 
   952 /**
   953 Stores a database object security policy into aSecurityPolicyCon container.
   954 
   955 @param aSecurityPolicyCon Security policies container.
   956 @param aObjType Database object type: RSqlSecurityPolicy::ETable.
   957 @param aObjName Database object name (for example, table name), which is protected by the current security policy.
   958 @param aPolicyType Database policy type: RSqlSecurityPolicy::EReadPolicy,
   959 				   RSqlSecurityPolicy::EWritePolicy, RSqlSecurityPolicy::ESchemaPolicy.
   960 @param aPolicy Database security policy object
   961 @param aDbPolicySetFlag Bit flag. Keeps information which database security policies are set and which aren't.
   962 						The parameter value may change if some of the database policies are set by
   963 						StoreDbSecurityPolicyL().
   964 
   965 @leave KErrGeneral Invalid security policy type or invalid database object name length.
   966 */
   967 void TSqlDbSysSettings::StoreDbObjSecurityPolicyL(CSqlSecurityPolicy& aSecurityPolicyCon, 
   968 							   					  TInt aObjType, const TDesC& aObjName,
   969 							   					  TInt aPolicyType, const TSecurityPolicy& aPolicy)
   970 	{
   971 	if(aPolicyType < (TInt)RSqlSecurityPolicy::EReadPolicy || aPolicyType > (TInt)RSqlSecurityPolicy::EWritePolicy)
   972 		{
   973 		__SQLLEAVE(KErrGeneral);
   974 		}
   975 	if(aObjName.Length() < 1)
   976 		{
   977 		__SQLLEAVE(KErrGeneral);
   978 		}
   979 	__SQLLEAVE_IF_ERROR(aSecurityPolicyCon.SetPolicy(static_cast<RSqlSecurityPolicy::TObjectType> (aObjType), 
   980 							aObjName, static_cast <RSqlSecurityPolicy::TPolicyType> (aPolicyType), aPolicy));
   981 	}
   982 
   983 /**
   984 Returns true if settigns table exists. 
   985 (Non-secure databases prior version 3 don't have settings table)
   986 
   987 @param aDbName Logical database name: "main" for the main database or attached database name.
   988 */
   989 TBool TSqlDbSysSettings::SettingsTableExistsL(const TDesC& aDbName)
   990 	{
   991 	HBufC* buf = HBufC::NewLC(sizeof(KSettingsTableCheckSql) + aDbName.Length());
   992 	TPtr sql = buf->Des();
   993 	sql.Format(KSettingsTableCheckSql(), &aDbName);
   994 	sqlite3_stmt* stmtHandle = ::StmtPrepare16L(iDbHandle, sql);
   995 	CleanupStack::PushL(TCleanupItem(&FinalizeStatementHandle, stmtHandle));
   996 	TInt rc = ::StmtNext(stmtHandle);
   997 	CleanupStack::PopAndDestroy();//cleanupItem (statement handle)
   998 	CleanupStack::PopAndDestroy(buf);
   999 	__SQLLEAVE_IF_ERROR(rc);
  1000 	return rc == KSqlAtRow;
  1001 	}
  1002 
  1003 /**
  1004 Returns true if at least one index using aCollationName exists.
  1005 
  1006 @param aDbName Logical database name: "main" for the main database or attached database name,
  1007 @param aCollationName SQLITE collation name.
  1008 */
  1009 TBool TSqlDbSysSettings::IndexExistsL(const TDesC& aDbName, const TDesC& aCollationName)
  1010 	{
  1011 	//KGetIndexSql will contain '%<CollationName>%' search pattern, while Symbian OS search patterns use '*' symbol by default.
  1012 	//This time the '%' is OK because the serach is performed in the SQLITE master table and before installing the
  1013 	//user-defined Like() function.
  1014 	HBufC* buf = HBufC::NewLC(sizeof(KGetIndexSql) + aDbName.Length() + aCollationName.Length());
  1015 	TPtr sql = buf->Des();
  1016 	sql.Format(KGetIndexSql(), &aDbName, &aCollationName);
  1017 	sqlite3_stmt* stmtHandle = ::StmtPrepare16L(iDbHandle, sql);
  1018 	CleanupStack::PushL(TCleanupItem(&FinalizeStatementHandle, stmtHandle));
  1019 	TInt rc = ::StmtNext(stmtHandle);
  1020 	CleanupStack::PopAndDestroy(2);//buf, cleanupItem (statement handle)
  1021 	__SQLLEAVE_IF_ERROR(rc);
  1022 	return rc == KSqlAtRow;
  1023 	}