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