os/persistentdata/persistentstorage/sql/SRC/Server/SqlSrvStatementUtil.cpp
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
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
//
sl@0
    15
sl@0
    16
#include <e32debug.h>
sl@0
    17
#include <hal.h>
sl@0
    18
#include <sqldb.h>
sl@0
    19
#include "sqlite3.h"
sl@0
    20
#include "SqlSrvStatementUtil.h"
sl@0
    21
#include "SqlAssert.h"
sl@0
    22
#include "SqlSrvUtil.h"
sl@0
    23
#include "SqlUtil.h"
sl@0
    24
#include "SqliteSymbian.h"		//sqlite3SymbianLastOsError()
sl@0
    25
#include "OstTraceDefinitions.h"
sl@0
    26
#include "SqlSrvResourceProfiler.h"
sl@0
    27
#ifdef OST_TRACE_COMPILER_IN_USE
sl@0
    28
#include "SqlSrvStatementUtilTraces.h"
sl@0
    29
#endif
sl@0
    30
#include "SqlTraceDef.h"
sl@0
    31
sl@0
    32
//The database names in all statements are quoted to avoid the "sql injection" threat.
sl@0
    33
_LIT8(KPageCountPragma, "PRAGMA \"%S\".page_count\x0");
sl@0
    34
_LIT8(KPageSizePragma,  "PRAGMA \"%S\".page_size\x0");
sl@0
    35
_LIT8(KCacheSizePragma,  "PRAGMA \"%S\".cache_size\x0");
sl@0
    36
_LIT8(KEncodingPragma,  "PRAGMA \"%S\".encoding\x0");
sl@0
    37
_LIT8(KFreePageCountPragma, "PRAGMA \"%S\".freelist_count\x0");
sl@0
    38
_LIT8(KVacuumModePragma, "PRAGMA \"%S\".auto_vacuum\x0");
sl@0
    39
_LIT8(KIncrementalVacuumPragma, "PRAGMA \"%S\".incremental_vacuum(%d)\x0");
sl@0
    40
sl@0
    41
///////////////////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
    42
sl@0
    43
//Calls sqlite3_open16() to create or open database file with aFileNameZ.
sl@0
    44
//aFileNameZ is UTF16 encoded, zero-terminated.
sl@0
    45
//The function returns system-wide errors or database specific errors.
sl@0
    46
//The function may panic with code 7 in _DEBUG mode - internal error: the created database handle is NULL.
sl@0
    47
TInt CreateDbHandle16(const TDesC& aFileNameZ, sqlite3*& aDbHandle)
sl@0
    48
	{
sl@0
    49
	(void)sqlite3SymbianLastOsError();//clear last OS error
sl@0
    50
	TInt err = sqlite3_open16(aFileNameZ.Ptr(), &aDbHandle);
sl@0
    51
	__ASSERT_DEBUG(err == SQLITE_OK ? aDbHandle != NULL : ETrue, __SQLPANIC2(ESqlPanicInternalError));
sl@0
    52
	if(err == SQLITE_OK)
sl@0
    53
		{
sl@0
    54
		(void)sqlite3_extended_result_codes(aDbHandle, 0);
sl@0
    55
		}
sl@0
    56
	//Get the return error code now, because the next "if" may destroy it.
sl@0
    57
	TInt rc = ::Sql2OsErrCode(err, sqlite3SymbianLastOsError());
sl@0
    58
	if(err != SQLITE_OK)
sl@0
    59
		{//Yes, it is possible error code != SQLITE_OK and aDbHandle != NULL.
sl@0
    60
		CloseDbHandle(aDbHandle);
sl@0
    61
		aDbHandle = NULL;
sl@0
    62
		}
sl@0
    63
	return rc;
sl@0
    64
	}
sl@0
    65
sl@0
    66
//Calls sqlite3_open() to create or open database file with aFileNameZ.
sl@0
    67
//aFileNameZ is UTF8 encoded, zero-terminated.
sl@0
    68
//The function returns system-wide errors or database specific errors.
sl@0
    69
//The function may panic with code 7 in _DEBUG mode - internal error: the created database handle is NULL.
sl@0
    70
TInt CreateDbHandle8(const TDesC8& aFileNameZ, sqlite3*& aDbHandle)
sl@0
    71
	{
sl@0
    72
	(void)sqlite3SymbianLastOsError();//clear last OS error
sl@0
    73
	TInt err = sqlite3_open((const char *) aFileNameZ.Ptr(), &aDbHandle);
sl@0
    74
	__ASSERT_DEBUG(err == SQLITE_OK ? aDbHandle != NULL : ETrue, __SQLPANIC2(ESqlPanicInternalError));
sl@0
    75
	if(err == SQLITE_OK)
sl@0
    76
		{
sl@0
    77
		(void)sqlite3_extended_result_codes(aDbHandle, 0);
sl@0
    78
		}
sl@0
    79
	//Get the return error code now, because the next "if" may destroy it.
sl@0
    80
	TInt rc = ::Sql2OsErrCode(err, sqlite3SymbianLastOsError());
sl@0
    81
	if(err != SQLITE_OK)
sl@0
    82
		{//Yes, it is possible error code != SQLITE_OK and aDbHandle != NULL.
sl@0
    83
		CloseDbHandle(aDbHandle);
sl@0
    84
		aDbHandle = NULL;
sl@0
    85
		}
sl@0
    86
	return rc;
sl@0
    87
	}
sl@0
    88
	
sl@0
    89
///////////////////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
    90
/////                     16-bit and 8-bit SQL statements execution upon completion                        ////
sl@0
    91
///////////////////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
    92
sl@0
    93
//16-bit SQL statement execution.
sl@0
    94
//
sl@0
    95
//aSql - zero-terminated string
sl@0
    96
//
sl@0
    97
//Prepares the supplied as an argument (aSql) 16-bit SQL statement and executes it.
sl@0
    98
//If aSql argument contains more than one SQL statements, separated with ';', then
sl@0
    99
//the function panics in _DEBUG mode (panic code 7).
sl@0
   100
//
sl@0
   101
//The function panics in debug mode (panic code 2) if aDbHandle is NULL.
sl@0
   102
//
sl@0
   103
//If the function completes successfully, it returns SQLITE_ROW or SQLITE_DONE.
sl@0
   104
//If the function fails then it returns one of the SQLITE error codes.
sl@0
   105
static TInt DoSingleStmtExec16(sqlite3 *aDbHandle, const TDesC16& aSql)
sl@0
   106
	{
sl@0
   107
    __ASSERT_DEBUG(aDbHandle != NULL, __SQLPANIC2(ESqlPanicInvalidObj));
sl@0
   108
    __ASSERT_DEBUG(aSql.Length() > 0 ? (TInt)aSql[aSql.Length() - 1] == 0 : ETrue, __SQLPANIC2(ESqlPanicBadArgument));
sl@0
   109
	sqlite3_stmt* stmtHandle = NULL;
sl@0
   110
	const void* stmtTail = NULL;
sl@0
   111
    //sqlite3_prepare16_v2() expects parameter #3 to be one of the following:
sl@0
   112
    // - byte length of the sql statement (parameter #2), excluding terminating zero;
sl@0
   113
    // - negative value - the sql statement (parameter #2) is zero-terminated;
sl@0
   114
	TInt err = sqlite3_prepare16_v2(aDbHandle, aSql.Ptr(), aSql.Length() * sizeof(TUint16) - sizeof(TUint16), &stmtHandle, &stmtTail);
sl@0
   115
	__ASSERT_DEBUG(err == SQLITE_OK ? !stmtTail || User::StringLength((const TUint16*)stmtTail) == 0 : !stmtHandle, __SQLPANIC2(ESqlPanicInternalError));
sl@0
   116
	if(stmtHandle)	//stmtHandle can be NULL for statements like this: ";".
sl@0
   117
		{
sl@0
   118
		if(err == SQLITE_OK)
sl@0
   119
			{
sl@0
   120
			while((err = sqlite3_step(stmtHandle)) == SQLITE_ROW)
sl@0
   121
				{
sl@0
   122
				}
sl@0
   123
			if(err == SQLITE_ERROR)	//It may be "out of memory" problem
sl@0
   124
				{
sl@0
   125
				err = sqlite3_reset(stmtHandle);
sl@0
   126
				__ASSERT_DEBUG(err != SQLITE_OK, __SQLPANIC2(ESqlPanicInternalError));
sl@0
   127
				}
sl@0
   128
			}
sl@0
   129
		(void)sqlite3_finalize(stmtHandle);//sqlite3_finalize() fails only if an invalid statement handle is passed.
sl@0
   130
		}
sl@0
   131
	return err;
sl@0
   132
	}
sl@0
   133
sl@0
   134
/**
sl@0
   135
This function searches aString argument for ';' occurences.
sl@0
   136
Every time when it finds a ';' character, the function places a zero character right after the ';' and
sl@0
   137
tests the just created, zero-terminated substring if it is a comlpete SQL statement.
sl@0
   138
sl@0
   139
If it is a SQL statement, the function replaces the found ';' character with zero and returns the just created
sl@0
   140
zero-terminated substring.Also the function modifies aString argument to point right after the found
sl@0
   141
SQL string. If it is not SQL statement, the function will continue the searching.
sl@0
   142
sl@0
   143
If there is no ';' inside aString argument, the function returns the same string as a return result and
sl@0
   144
modifies aString argument - sets it to TPtr(NULL, 0, 0).
sl@0
   145
sl@0
   146
The function expects aString argument to be zero-terminated.
sl@0
   147
sl@0
   148
@internalComponent
sl@0
   149
*/
sl@0
   150
TPtrC GetFirstSqlStmt(TPtr& aString)
sl@0
   151
	{
sl@0
   152
	const TChar KDelimitier(';');
sl@0
   153
	TPtr originalStr(aString);
sl@0
   154
	TPtr str(const_cast <TUint16*> (aString.Ptr()), aString.Length(), aString.Length());
sl@0
   155
	TInt afterDelimitierPos = 0;
sl@0
   156
	TInt pos;
sl@0
   157
	while((pos = str.Locate(KDelimitier) + 1) > 0 && pos < str.Length())
sl@0
   158
		{
sl@0
   159
		//There is a possibility that the string, which terminates with the found ';' character, is a SQL statement.
sl@0
   160
		//Zero terminate the string placing a zero right after the ';' character and test it using sqlite3_complete16().
sl@0
   161
		//If it is not a SQL string, restore the original character and continue searching.
sl@0
   162
		afterDelimitierPos += pos;
sl@0
   163
		TChar ch = aString[afterDelimitierPos];
sl@0
   164
		aString[afterDelimitierPos] = 0;
sl@0
   165
		TInt res = sqlite3_complete16(aString.Ptr());
sl@0
   166
		aString[afterDelimitierPos] = ch;
sl@0
   167
		if(res)
sl@0
   168
			{
sl@0
   169
			str.Set(const_cast <TUint16*> (aString.Ptr()), afterDelimitierPos, afterDelimitierPos);	
sl@0
   170
			//Replace the found ';' character with 0.
sl@0
   171
			str[afterDelimitierPos - 1] = 0;
sl@0
   172
			aString.Set(const_cast <TUint16*> (aString.Ptr()) + afterDelimitierPos, aString.Length() - afterDelimitierPos, aString.Length() - afterDelimitierPos);
sl@0
   173
			return str;
sl@0
   174
			}
sl@0
   175
		str.Set(const_cast <TUint16*> (str.Ptr()) + pos, str.Length() - pos, str.Length() - pos);	
sl@0
   176
		}
sl@0
   177
	//aString argument does not contain valid SQL statement or there is no ';' character inside aString.
sl@0
   178
	//Set aString to TPtr(NULL, 0, 0) and return the original string.
sl@0
   179
	aString.Set(NULL, 0, 0);
sl@0
   180
	str.Set(originalStr);
sl@0
   181
	return str;
sl@0
   182
	}
sl@0
   183
sl@0
   184
/**
sl@0
   185
Executes one or more 16-bit SQL statements. SQL statements of any kind can be executed, but the 
sl@0
   186
method won't return any record(s) if the SQL statement type is "SELECT".
sl@0
   187
If the SQL statement(s) contains one or more parameters, the method will execute it giving the parameters
sl@0
   188
default NULL values.
sl@0
   189
sl@0
   190
@param aDbHandle Database handle. Not NULL.
sl@0
   191
@param aSqlStmt String containing one or more 16-bit SQL statements, separated with ';'. Zero-terminated string.
sl@0
   192
				Note: The ExecL() call can modify the content of aSqlStmt argument.
sl@0
   193
sl@0
   194
@return KErrNone, Operation completed successfully;
sl@0
   195
		KErrNoMemory, Out of memory;
sl@0
   196
		KSqlErrGeneral, Syntax error. A text message describing the problem can be obtained calling
sl@0
   197
						RSqlDatabase::LastErrorMessage();
sl@0
   198
						Other system-wide error codes or SQL errors of ESqlDbError type.
sl@0
   199
sl@0
   200
@panic SqlDb 4 In _DEBUG mode if aSqlStmt is not zero-terminated string
sl@0
   201
@panic SqlDb 7 In _DEBUG mode if aDbHandle is NULL
sl@0
   202
sl@0
   203
@internalComponent
sl@0
   204
*/
sl@0
   205
TInt DbExecStmt16(sqlite3* aDbHandle, TDes16& aSqlStmt)
sl@0
   206
	{
sl@0
   207
    __SQLTRACE_INTERNALSEXPR(TPtrC sqlprnptr(aSqlStmt.Left(aSqlStmt.Length() - 1)));
sl@0
   208
    SQL_TRACE_INTERNALS(OstTraceExt2(TRACE_INTERNALS, DBEXECSTMT16_ENTRY, "Entry;0x%X;DbExecStmt16;sql=%S", (TUint)aDbHandle, __SQLPRNSTR(sqlprnptr)));
sl@0
   209
	
sl@0
   210
	__ASSERT_DEBUG(aDbHandle != NULL, __SQLPANIC2(ESqlPanicInternalError));
sl@0
   211
	__ASSERT_DEBUG(aSqlStmt.Length() > 0 ? (TInt)aSqlStmt[aSqlStmt.Length() - 1] == 0: ETrue, __SQLPANIC2(ESqlPanicBadArgument));
sl@0
   212
sl@0
   213
	(void)sqlite3SymbianLastOsError();//clear last OS error
sl@0
   214
	
sl@0
   215
	TInt err = SQLITE_DONE;
sl@0
   216
	//Execute SQL statement(s)
sl@0
   217
	//16-bit SQL string - no sqlite3_exec16() function, so the execution is made with 
sl@0
   218
	//sqlite3_prepare16_v2() and sqlite3_step() functions.
sl@0
   219
	TPtr16 sql(const_cast <TUint16*> (aSqlStmt.Ptr()), aSqlStmt.Length(), aSqlStmt.Length());
sl@0
   220
	TPtrC firstSqlStmt(KNullDesC);
sl@0
   221
	while(err == SQLITE_DONE && sql.Length() > 1) //"> 1" because it is a zero terminated string
sl@0
   222
		{
sl@0
   223
		firstSqlStmt.Set(GetFirstSqlStmt(sql));
sl@0
   224
        SQLPROFILER_SQL16_PRINT((TUint)aDbHandle, firstSqlStmt.Left(firstSqlStmt.Length() - 1), EFalse);
sl@0
   225
		err = ::DoSingleStmtExec16(aDbHandle, firstSqlStmt);
sl@0
   226
		}
sl@0
   227
		
sl@0
   228
	err = ::Sql2OsErrCode(err, sqlite3SymbianLastOsError());
sl@0
   229
	if(err == KSqlAtEnd)
sl@0
   230
		{
sl@0
   231
		err = KErrNone;	
sl@0
   232
		}
sl@0
   233
	SQL_TRACE_INTERNALS(OstTraceExt2(TRACE_INTERNALS, DBEXECSTMT16_EXIT, "Exit;0x%X;DbExecStmt16;err=%d", (TUint)aDbHandle, err));
sl@0
   234
	return err;
sl@0
   235
	}
sl@0
   236
sl@0
   237
/**
sl@0
   238
Executes one or more 8-bit SQL statements. SQL statements of any kind can be executed, but the 
sl@0
   239
method won't return any record(s) if the SQL statement type is "SELECT".
sl@0
   240
If the SQL statement(s) contains one or more parameters, the method will execute it giving the parameters
sl@0
   241
default NULL values.
sl@0
   242
sl@0
   243
@param aDbHandle Database handle. Not NULL.
sl@0
   244
@param aSqlStmt String containing one or more 8-bit SQL statements, separated with ';'. Zero-terminated string.
sl@0
   245
sl@0
   246
@return KErrNone, Operation completed successfully;
sl@0
   247
		KErrNoMemory, Out of memory;
sl@0
   248
		KSqlErrGeneral, Syntax error. A text message describing the problem can be obtained calling
sl@0
   249
						RSqlDatabase::LastErrorMessage();
sl@0
   250
		Other system-wide error codes or SQL errors of ESqlDbError type.
sl@0
   251
sl@0
   252
@panic SqlDb 4 In _DEBUG mode if aSqlStmt is not zero-terminated string
sl@0
   253
@panic SqlDb 7 In _DEBUG mode if aDbHandle is NULL
sl@0
   254
sl@0
   255
@internalComponent
sl@0
   256
*/
sl@0
   257
TInt DbExecStmt8(sqlite3* aDbHandle, const TDesC8& aSqlStmt)
sl@0
   258
	{
sl@0
   259
    __SQLTRACE_INTERNALSEXPR(TPtrC8 sqlprnptr(aSqlStmt.Left(aSqlStmt.Length() - 1)));
sl@0
   260
	__SQLTRACE_INTERNALSVAR(TBuf<100> des16prnbuf);
sl@0
   261
    SQL_TRACE_INTERNALS(OstTraceExt2(TRACE_INTERNALS, DBEXECSTMT8_ENTRY, "Entry;0x%X;DbExecStmt8;sql=%s", (TUint)aDbHandle, __SQLPRNSTR8(sqlprnptr, des16prnbuf)));
sl@0
   262
    
sl@0
   263
	__ASSERT_DEBUG(aDbHandle != NULL, __SQLPANIC2(ESqlPanicInternalError));
sl@0
   264
	__ASSERT_DEBUG(aSqlStmt.Length() > 0 ? (TInt)aSqlStmt[aSqlStmt.Length() - 1] == 0: ETrue, __SQLPANIC2(ESqlPanicBadArgument));
sl@0
   265
sl@0
   266
    SQLPROFILER_SQL8_PRINT((TUint)aDbHandle, aSqlStmt.Left(aSqlStmt.Length() - 1), EFalse);
sl@0
   267
	
sl@0
   268
	(void)sqlite3SymbianLastOsError();//clear last OS error
sl@0
   269
	
sl@0
   270
	TInt err = sqlite3_exec(aDbHandle, reinterpret_cast <const char*> (aSqlStmt.Ptr()), NULL, NULL, NULL);
sl@0
   271
sl@0
   272
	err = ::Sql2OsErrCode(err, sqlite3SymbianLastOsError());
sl@0
   273
	__ASSERT_DEBUG(err != KSqlAtEnd, __SQLPANIC2(ESqlPanicInternalError));
sl@0
   274
	SQL_TRACE_INTERNALS(OstTraceExt2(TRACE_INTERNALS, DBEXECSTMT8_EXIT, "Exit;0x%X;DbExecStmt8;err=%d", (TUint)aDbHandle, err));
sl@0
   275
	return err;
sl@0
   276
	}
sl@0
   277
sl@0
   278
///////////////////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
   279
/////                     16-bit and 8-bit SQL statement preparation                                      /////
sl@0
   280
///////////////////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
   281
sl@0
   282
//Prepares 16-bit SQL statement, returning in aStmtHandle the statement handle and
sl@0
   283
//setting aHasTail to true, if aStmt contains more than one sql statements.
sl@0
   284
//aStmt - zero-terminated string.
sl@0
   285
//Returns one of SQLITE error codes.
sl@0
   286
static TInt DoPrepareStmt16(sqlite3* aDbHandle, const TDesC& aStmt, sqlite3_stmt** aStmtHandle, TBool& aHasTail)
sl@0
   287
	{
sl@0
   288
	const void* stmtTail = NULL;
sl@0
   289
    //sqlite3_prepare16_v2() expects parameter #3 to be one of the following:
sl@0
   290
    // - byte length of the sql statement (parameter #2), excluding terminating zero;
sl@0
   291
    // - negative value - the sql statement (parameter #2) is zero-terminated;
sl@0
   292
	TInt err = sqlite3_prepare16_v2(aDbHandle, aStmt.Ptr(), aStmt.Length() * sizeof(TUint16) - sizeof(TUint16), aStmtHandle, &stmtTail);
sl@0
   293
	aHasTail = stmtTail && static_cast <const TUint16*> (stmtTail)[0] != 0;
sl@0
   294
	__ASSERT_DEBUG(err != SQLITE_OK ? !(*aStmtHandle) : ETrue, __SQLPANIC2(ESqlPanicInternalError));
sl@0
   295
	//(*aStmtHandle) is NULL for ";" statements, when err == SQLITE_OK. Since the server should not panic
sl@0
   296
	//that situation is handled later (not inside the assert above)
sl@0
   297
	return err;
sl@0
   298
	}
sl@0
   299
	
sl@0
   300
//Prepares 8-bit SQL statement, returning in aStmtHandle the statement handle and
sl@0
   301
//setting aHasTail to true, if aStmt contains more than one sql statements.
sl@0
   302
//aStmt - zero-terminated string.
sl@0
   303
//Returns one of SQLITE error codes.
sl@0
   304
static TInt DoPrepareStmt8(sqlite3* aDbHandle, const TUint8* aStmt, sqlite3_stmt** aStmtHandle, TBool& aHasTail)
sl@0
   305
	{
sl@0
   306
	const char* stmtTail = NULL;
sl@0
   307
    //sqlite3_prepare_v2() expects parameter #3 to be one of the following:
sl@0
   308
    // - byte length of the sql statement (parameter #2), excluding terminating zero;
sl@0
   309
    // - negative value - the sql statement (parameter #2) is zero-terminated;
sl@0
   310
	TInt err = sqlite3_prepare_v2(aDbHandle, reinterpret_cast <const char*> (aStmt), -1, aStmtHandle, &stmtTail);
sl@0
   311
	aHasTail = stmtTail && stmtTail[0] != 0;
sl@0
   312
	__ASSERT_DEBUG(err != SQLITE_OK ? !(*aStmtHandle) : ETrue, __SQLPANIC2(ESqlPanicInternalError));
sl@0
   313
	//(*aStmtHandle) is NULL for ";" statements, when err == SQLITE_OK. Since the server should not panic
sl@0
   314
	//that situation is handled later (not inside the assert above)
sl@0
   315
	return err;
sl@0
   316
	}
sl@0
   317
sl@0
   318
//This function accepts as arguments the SQLITE error code, 
sl@0
   319
//"aHasTail" boolean flag set to true if the SQL string contains more than one SQL statement 
sl@0
   320
//and the statement handle.
sl@0
   321
//
sl@0
   322
//It checks the arguments and returns an error if:
sl@0
   323
// - aSqliteError != SQLITE_OK;
sl@0
   324
// - aHasTail is true (possibly more than one SQL statement, separated with ";");
sl@0
   325
// - aStmtHandle is NULL;
sl@0
   326
//
sl@0
   327
static TInt ProcessPrepareError(TInt aSqliteError, TBool aHasTail, sqlite3_stmt*& aStmtHandle)
sl@0
   328
	{
sl@0
   329
	if(aSqliteError != SQLITE_OK)
sl@0
   330
		{
sl@0
   331
		return ::Sql2OsErrCode(aSqliteError, sqlite3SymbianLastOsError());
sl@0
   332
		}
sl@0
   333
	else if(aHasTail || !aStmtHandle)
sl@0
   334
		{//Case 1:
sl@0
   335
		 // More than one SQL statement or the SQL string is "" or ";;;" or ";   ;; ;". 
sl@0
   336
		 // Report it as an error, because there is no statement handle.
sl@0
   337
		 //Case 2:
sl@0
   338
		 // Non-null aHasTail. In this case the SQL string contains more than one SQL statement.
sl@0
   339
		 // The statement handle is not null. The statement has to be finialized before reporting the error.
sl@0
   340
		(void)FinalizeStmtHandle(aStmtHandle);
sl@0
   341
		aStmtHandle = NULL;
sl@0
   342
		return KErrArgument;
sl@0
   343
		}
sl@0
   344
	return KErrNone;
sl@0
   345
	}
sl@0
   346
sl@0
   347
//This function accepts as arguments the SQLITE error code, 
sl@0
   348
//"aHasTail" boolean flag set to true if the SQL string contains more than one SQL statement 
sl@0
   349
//and the statement handle.
sl@0
   350
//
sl@0
   351
//It checks the arguments and leaves if:
sl@0
   352
// - aSqliteError != SQLITE_OK;
sl@0
   353
// - aHasTail is true (possibly more than one SQL statement, separated with ";");
sl@0
   354
// - aStmtHandle is NULL;
sl@0
   355
//
sl@0
   356
static void LeaveIfPrepareErrorL(TInt aSqliteError, TBool aHasTail, sqlite3_stmt* aStmtHandle)
sl@0
   357
	{
sl@0
   358
	__SQLLEAVE_IF_ERROR2(ProcessPrepareError(aSqliteError, aHasTail, aStmtHandle));
sl@0
   359
	}
sl@0
   360
sl@0
   361
/**
sl@0
   362
Prepares 16-bit aSqlStmt SQL statement.
sl@0
   363
sl@0
   364
@param aSqlStmt - zero-terminated string.
sl@0
   365
sl@0
   366
@leave KErrNoMemory, if there is no memory;
sl@0
   367
	   KErrArgument, if the SQL string contains more than one SQL statements;
sl@0
   368
	   One of the error codes in [KSqlErrGeneral..KSqlErrNotDb] range.
sl@0
   369
sl@0
   370
@return The prepared SQL statement handle.
sl@0
   371
sl@0
   372
@internalComponent
sl@0
   373
*/
sl@0
   374
sqlite3_stmt* StmtPrepare16L(sqlite3* aDbHandle, const TDesC16& aSqlStmt)
sl@0
   375
	{
sl@0
   376
    __SQLTRACE_INTERNALSEXPR(TPtrC sqlprnptr(aSqlStmt.Left(aSqlStmt.Length() - 1)));
sl@0
   377
    SQL_TRACE_INTERNALS(OstTraceExt2(TRACE_INTERNALS, STMTPREPARE16L_ENTRY, "Entry;0x%X;StmtPrepare16L;sql=%S", (TUint)aDbHandle, __SQLPRNSTR(sqlprnptr)));	
sl@0
   378
	(void)sqlite3SymbianLastOsError();//clear last OS error
sl@0
   379
	TBool hasTail = EFalse;
sl@0
   380
	sqlite3_stmt* stmtHandle = NULL;
sl@0
   381
	TInt err = DoPrepareStmt16(aDbHandle, aSqlStmt, &stmtHandle, hasTail);
sl@0
   382
	LeaveIfPrepareErrorL(err, hasTail, stmtHandle);
sl@0
   383
	SQL_TRACE_INTERNALS(OstTraceExt2(TRACE_INTERNALS, STMTPREPARE16L_EXIT, "Exit;0x%X;StmtPrepare16L;stmtHandle=0x%X", (TUint)aDbHandle, (TUint)stmtHandle)); 
sl@0
   384
	return stmtHandle;
sl@0
   385
	}
sl@0
   386
sl@0
   387
/**
sl@0
   388
Prepares 8-bit aSqlStmt SQL statement.
sl@0
   389
sl@0
   390
@param aSqlStmt - zero-terminated string.
sl@0
   391
sl@0
   392
@leave KErrNoMemory, if there is no memory;
sl@0
   393
	   KErrArgument, if the SQL string contains more than one SQL statements;
sl@0
   394
	   One of the error codes in [KSqlErrGeneral..KSqlErrNotDb] range.
sl@0
   395
sl@0
   396
@return The prepared SQL statement handle.
sl@0
   397
sl@0
   398
@internalComponent
sl@0
   399
*/
sl@0
   400
TInt StmtPrepare8(sqlite3* aDbHandle, const TDesC8& aSqlStmt, sqlite3_stmt*& aStmtHandle)
sl@0
   401
	{
sl@0
   402
    __SQLTRACE_INTERNALSEXPR(TPtrC8 sqlprnptr(aSqlStmt.Left(aSqlStmt.Length() - 1)));
sl@0
   403
	__SQLTRACE_INTERNALSVAR(TBuf<100> des16prnbuf);
sl@0
   404
    SQL_TRACE_INTERNALS(OstTraceExt2(TRACE_INTERNALS, STMTPREPARE8_ENTRY, "Entry;0x%X;StmtPrepare8;sql=%s", (TUint)aDbHandle, __SQLPRNSTR8(sqlprnptr, des16prnbuf))); 
sl@0
   405
	(void)sqlite3SymbianLastOsError();//clear last OS error
sl@0
   406
	TBool hasTail = EFalse;
sl@0
   407
	TInt err = DoPrepareStmt8(aDbHandle, aSqlStmt.Ptr(), &aStmtHandle, hasTail);
sl@0
   408
	err = ProcessPrepareError(err, hasTail, aStmtHandle);
sl@0
   409
	SQL_TRACE_INTERNALS(OstTraceExt3(TRACE_INTERNALS, STMTPREPARE8_EXIT, "Exit;0x%X;StmtPrepare8;aStmtHandle=0x%X;err=%d", (TUint)aDbHandle, (TUint)aStmtHandle, err)); 
sl@0
   410
	return err;
sl@0
   411
	}
sl@0
   412
sl@0
   413
/**
sl@0
   414
Prepares 8-bit aSqlStmt SQL statement.
sl@0
   415
sl@0
   416
@param aSqlStmt - zero-terminated string.
sl@0
   417
sl@0
   418
@leave KErrNoMemory, if there is no memory;
sl@0
   419
	   KErrArgument, if the SQL string contains more than one SQL statements;
sl@0
   420
	   One of the error codes in [KSqlErrGeneral..KSqlErrNotDb] range.
sl@0
   421
sl@0
   422
@return The prepared SQL statement handle.
sl@0
   423
sl@0
   424
@internalComponent
sl@0
   425
*/
sl@0
   426
sqlite3_stmt* StmtPrepare8L(sqlite3* aDbHandle, const TDesC8& aSqlStmt)
sl@0
   427
	{
sl@0
   428
    __SQLTRACE_INTERNALSEXPR(TPtrC8 sqlprnptr(aSqlStmt.Left(aSqlStmt.Length() - 1)));
sl@0
   429
	__SQLTRACE_INTERNALSVAR(TBuf<100> des16prnbuf);
sl@0
   430
    SQL_TRACE_INTERNALS(OstTraceExt2(TRACE_INTERNALS, STMTPREPARE8L_ENTRY, "Entry;0x%X;StmtPrepare8L;sql=%s", (TUint)aDbHandle, __SQLPRNSTR8(sqlprnptr, des16prnbuf))); 
sl@0
   431
	(void)sqlite3SymbianLastOsError();//clear last OS error
sl@0
   432
	TBool hasTail = EFalse;
sl@0
   433
	sqlite3_stmt* stmtHandle = NULL;
sl@0
   434
	TInt err = DoPrepareStmt8(aDbHandle, aSqlStmt.Ptr(), &stmtHandle, hasTail);
sl@0
   435
	LeaveIfPrepareErrorL(err, hasTail, stmtHandle);
sl@0
   436
	SQL_TRACE_INTERNALS(OstTraceExt2(TRACE_INTERNALS, STMTPREPARE8L_EXIT, "Exit;0x%X;StmtPrepare8L;stmtHandle=0x%X", (TUint)aDbHandle, (TUint)stmtHandle)); 
sl@0
   437
	return stmtHandle;
sl@0
   438
	}
sl@0
   439
sl@0
   440
/**
sl@0
   441
Executes upon completion the prepared SQL statement.
sl@0
   442
sl@0
   443
@param aStmtHandle Prepared statement handle
sl@0
   444
sl@0
   445
@return KSqlErrStmtExpired, Statement expired (if new functions or collating sequences are 
sl@0
   446
							registered or if an authorizer function is added or changed);
sl@0
   447
		KErrNoMemory, Out of memory. The statement will be reset;
sl@0
   448
		KErrNone, The reset operation completed successfully.							
sl@0
   449
sl@0
   450
@panic SqlDb 2 In _DEBUG mode. Invalid (NULL) statement handle.
sl@0
   451
@panic SqlDb 7 In _DEBUG mode. SQLITE internal error. (SQLITE_ERROR, followed by a sqlite3_reset(), which returns SQLITE_OK)
sl@0
   452
sl@0
   453
@internalComponent
sl@0
   454
*/	
sl@0
   455
TInt StmtExec(sqlite3_stmt* aStmtHandle)
sl@0
   456
	{
sl@0
   457
	SQL_TRACE_INTERNALS(OstTraceExt2(TRACE_INTERNALS, STMTEXEC_ENTRY, "Entry;0x%X;StmtExec;aStmtHandle=0x%X", (TUint)sqlite3_db_handle(aStmtHandle), (TUint)aStmtHandle));
sl@0
   458
	__ASSERT_DEBUG(aStmtHandle != NULL, __SQLPANIC2(ESqlPanicInvalidObj));
sl@0
   459
	
sl@0
   460
	(void)sqlite3SymbianLastOsError();//clear last OS error
sl@0
   461
sl@0
   462
	TInt err;
sl@0
   463
	while((err = sqlite3_step(aStmtHandle)) == SQLITE_ROW)
sl@0
   464
		{
sl@0
   465
		}
sl@0
   466
		
sl@0
   467
	if(err == SQLITE_ERROR)	//It may be "out of memory" problem
sl@0
   468
		{
sl@0
   469
		err = sqlite3_reset(aStmtHandle);
sl@0
   470
		__ASSERT_DEBUG(err != SQLITE_OK, __SQLPANIC2(ESqlPanicInternalError));
sl@0
   471
		}
sl@0
   472
		
sl@0
   473
	err = ::Sql2OsErrCode(err, sqlite3SymbianLastOsError());
sl@0
   474
	if(err == KSqlAtEnd)
sl@0
   475
		{
sl@0
   476
		err = KErrNone;	
sl@0
   477
		}
sl@0
   478
sl@0
   479
	SQL_TRACE_INTERNALS(OstTraceExt3(TRACE_INTERNALS, STMTEXEC_EXIT, "Exit;0x%X;StmtExec;aStmtHandle=0x%X;err=%d", (TUint)sqlite3_db_handle(aStmtHandle), (TUint)aStmtHandle, err));
sl@0
   480
	return err;
sl@0
   481
	}
sl@0
   482
sl@0
   483
/**
sl@0
   484
Executes the SQL statement moving it to the next row if available.
sl@0
   485
sl@0
   486
@return KSqlErrStmtExpired Statement expired (if new functions or collating sequences are 
sl@0
   487
							registered or if an authorizer function is added or changed)
sl@0
   488
@return KSqlAtRow, The next record data is ready for processing by the caller;
sl@0
   489
		KSqlAtEnd, No more record data;
sl@0
   490
		KSqlErrBusy, Database file is locked;
sl@0
   491
		KSqlErrGeneral, Run-time error. Next() should not be called anymore;
sl@0
   492
		KSqlErrMisuse, Next() called after KSqlAtEnd or KSqlErrGeneral returned by the previous Next() call;
sl@0
   493
		KErrNoMemory, Out of memory. The statement will be reset.
sl@0
   494
sl@0
   495
@panic SqlDb 2 In _DEBUG mode. Invalid (NULL) statement handle.
sl@0
   496
@panic SqlDb 7 In _DEBUG mode. SQLITE internal error. (SQLITE_ERROR, followed by a sqlite3_reset(), which returns SQLITE_OK)
sl@0
   497
sl@0
   498
@internalComponent
sl@0
   499
*/	
sl@0
   500
TInt StmtNext(sqlite3_stmt* aStmtHandle)
sl@0
   501
	{
sl@0
   502
	SQL_TRACE_INTERNALS(OstTraceExt2(TRACE_INTERNALS, STMTNEXT_ENTRY, "Entry;0x%X;StmtNext;aStmtHandle=0x%X", (TUint)sqlite3_db_handle(aStmtHandle), (TUint)aStmtHandle));
sl@0
   503
	__ASSERT_DEBUG(aStmtHandle != NULL, __SQLPANIC2(ESqlPanicInvalidObj));
sl@0
   504
	
sl@0
   505
	(void)sqlite3SymbianLastOsError();//clear last OS error
sl@0
   506
	
sl@0
   507
	TInt err = sqlite3_step(aStmtHandle);
sl@0
   508
	if(err == SQLITE_ERROR)	//It may be "out of memory" problem
sl@0
   509
		{
sl@0
   510
		err = sqlite3_reset(aStmtHandle);
sl@0
   511
		__ASSERT_DEBUG(err != SQLITE_OK, __SQLPANIC2(ESqlPanicInternalError));
sl@0
   512
		}
sl@0
   513
	err = ::Sql2OsErrCode(err, sqlite3SymbianLastOsError());
sl@0
   514
	SQL_TRACE_INTERNALS(OstTraceExt3(TRACE_INTERNALS, STMTNEXT_EXIT, "Exit;0x%X;StmtNext;aStmtHandle=0x%X;err=%d", (TUint)sqlite3_db_handle(aStmtHandle), (TUint)aStmtHandle, err));
sl@0
   515
	return err;
sl@0
   516
	}
sl@0
   517
sl@0
   518
/**
sl@0
   519
Resets the prepared SQL statement to its initial state and makes it ready to be executed again.
sl@0
   520
Any SQL statement parameters that had values bound to them, retain their values.
sl@0
   521
sl@0
   522
@return KSqlErrStmtExpired, Statement expired (if new functions or collating sequences are 
sl@0
   523
							registered or if an authorizer function is added or changed);
sl@0
   524
		KErrNone, The reset operation completed successfully.							
sl@0
   525
sl@0
   526
@panic SqlDb 2 In _DEBUG mode. Invalid (NULL) statement handle.
sl@0
   527
sl@0
   528
@internalComponent
sl@0
   529
*/	
sl@0
   530
TInt StmtReset(sqlite3_stmt* aStmtHandle)
sl@0
   531
	{
sl@0
   532
	SQL_TRACE_INTERNALS(OstTraceExt2(TRACE_INTERNALS, STMTRESET_ENTRY, "Entry;0x%X;StmtReset;aStmtHandle=0x%X", (TUint)sqlite3_db_handle(aStmtHandle), (TUint)aStmtHandle));
sl@0
   533
	__ASSERT_DEBUG(aStmtHandle != NULL, __SQLPANIC2(ESqlPanicInvalidObj));
sl@0
   534
	
sl@0
   535
	(void)sqlite3SymbianLastOsError();//clear last OS error
sl@0
   536
	
sl@0
   537
	TInt err = sqlite3_reset(aStmtHandle);
sl@0
   538
	err = ::Sql2OsErrCode(err, sqlite3SymbianLastOsError());
sl@0
   539
	SQL_TRACE_INTERNALS(OstTraceExt3(TRACE_INTERNALS, STMTRESET_EXIT, "Exit;0x%X;StmtReset;aStmtHandle=0x%X;err=%d", (TUint)sqlite3_db_handle(aStmtHandle), (TUint)aStmtHandle, err));
sl@0
   540
	return err;
sl@0
   541
	}
sl@0
   542
sl@0
   543
/**
sl@0
   544
Prepares and executes PRAGMA statement and moves the statement cursor on the first row.
sl@0
   545
sl@0
   546
@param aDbHandle Database handle
sl@0
   547
@param aDbName Attached database name or KNullDesC for the main database
sl@0
   548
@param aPragmaSql Pragma sql statement
sl@0
   549
@param aStmtHandle An output parameter where the statement handle will be stored
sl@0
   550
sl@0
   551
@panic SqlDb 7 In _DEBUG mode if aDbHandle is NULL
sl@0
   552
@panic SqlDb 4 In _DEBUG mode if aPragmaSql length is 0 or if the statement is not zero-terminated
sl@0
   553
sl@0
   554
@return KErrNone,     Operation completed successfully;
sl@0
   555
		KErrNoMemory, Out of memory;
sl@0
   556
					  Other system-wide error codes or SQL errors of ESqlDbError type.
sl@0
   557
sl@0
   558
@internalComponent
sl@0
   559
*/
sl@0
   560
static TInt PreRetrievePragmaValue(sqlite3* aDbHandle, const TDesC& aDbName, const TDesC8& aPragmaSql, sqlite3_stmt*& aStmtHandle)
sl@0
   561
	{
sl@0
   562
    __SQLTRACE_INTERNALSEXPR(TPtrC8 sqlprnptr(aPragmaSql.Left(aPragmaSql.Length() - 1)));
sl@0
   563
	__SQLTRACE_INTERNALSVAR(TBuf<100> des16prnbuf);
sl@0
   564
    SQL_TRACE_INTERNALS(OstTraceExt3(TRACE_INTERNALS, PRERETRIEVEPRAGMAVALUE_ENTRY, "Entry;0x%X;PreRetrievePragmaValue;aDbName=%S;aPragmaSql=%s", (TUint)aDbHandle, __SQLPRNSTR(aDbName), __SQLPRNSTR8(sqlprnptr, des16prnbuf)));
sl@0
   565
	__ASSERT_DEBUG(aDbHandle != NULL, __SQLPANIC2(ESqlPanicInternalError));
sl@0
   566
	__ASSERT_DEBUG(aPragmaSql.Length() > 0, __SQLPANIC2(ESqlPanicBadArgument));
sl@0
   567
	__ASSERT_DEBUG(aPragmaSql[aPragmaSql.Length() - 1] == 0, __SQLPANIC2(ESqlPanicBadArgument));
sl@0
   568
	TBuf8<KMaxFileName> dbName;
sl@0
   569
	if(!UTF16ToUTF8(aDbName, dbName))
sl@0
   570
		{
sl@0
   571
		SQL_TRACE_INTERNALS(OstTrace1(TRACE_INTERNALS, PRERETRIEVEPRAGMAVALUE_EXIT1, "Exit;0x%X;PreRetrievePragmaValue;err=KErrGeneral", (TUint)aDbHandle));
sl@0
   572
		return KErrGeneral;
sl@0
   573
		}
sl@0
   574
	TBuf8<KMaxFileName + 64> sql;//64 characters is enough for the longest PRAGMA statement
sl@0
   575
	if(dbName == KNullDesC8)
sl@0
   576
		{
sl@0
   577
		sql.Format(aPragmaSql, &KMainDb8);
sl@0
   578
		}
sl@0
   579
	else
sl@0
   580
		{
sl@0
   581
		sql.Format(aPragmaSql, &dbName);
sl@0
   582
		}
sl@0
   583
	aStmtHandle = NULL;
sl@0
   584
	TInt err = ::StmtPrepare8(aDbHandle, sql, aStmtHandle);
sl@0
   585
	if(err == KErrNone)
sl@0
   586
		{
sl@0
   587
		__ASSERT_DEBUG(aStmtHandle != NULL, __SQLPANIC2(ESqlPanicInvalidObj));
sl@0
   588
		err = ::StmtNext(aStmtHandle);
sl@0
   589
		}
sl@0
   590
	SQL_TRACE_INTERNALS(OstTraceExt3(TRACE_INTERNALS, PRERETRIEVEPRAGMAVALUE_EXIT2, "Exit;0x%X;PreRetrievePragmaValue;aStmtHandle=0x%X;err=%d", (TUint)aDbHandle, (TUint)aStmtHandle, err));
sl@0
   591
	return err;
sl@0
   592
	}
sl@0
   593
sl@0
   594
/**
sl@0
   595
Prepares and executes PRAGMA statement and retrieves the value of column 0 (the pragma value).
sl@0
   596
sl@0
   597
@param aDbHandle Database handle
sl@0
   598
@param aDbName Attached database name or KNullDesC for the main database
sl@0
   599
@param aPragmaSql Pragma sql statement
sl@0
   600
@param aPragmaValue An output parameter where the pragma value will be stored
sl@0
   601
sl@0
   602
@panic SqlDb 7 In _DEBUG mode if aDbHandle is NULL
sl@0
   603
@panic SqlDb 4 In _DEBUG mode if aPragmaSql length is 0 or if the statement is not zero-terminated
sl@0
   604
sl@0
   605
@return KErrNone,     Operation completed successfully;
sl@0
   606
		KErrNoMemory, Out of memory;
sl@0
   607
					  Other system-wide error codes or SQL errors of ESqlDbError type.
sl@0
   608
sl@0
   609
@internalComponent
sl@0
   610
*/
sl@0
   611
static TInt RetrievePragmaValue(sqlite3* aDbHandle, const TDesC& aDbName, const TDesC8& aPragmaSql, TInt& aPragmaValue)
sl@0
   612
	{
sl@0
   613
	__ASSERT_DEBUG(aDbHandle != NULL, __SQLPANIC2(ESqlPanicInternalError));
sl@0
   614
	__ASSERT_DEBUG(aPragmaSql.Length() > 0, __SQLPANIC2(ESqlPanicBadArgument));
sl@0
   615
	__ASSERT_DEBUG(aPragmaSql[aPragmaSql.Length() - 1] == 0, __SQLPANIC2(ESqlPanicBadArgument));
sl@0
   616
	sqlite3_stmt* stmtHandle = NULL;
sl@0
   617
	TInt err = PreRetrievePragmaValue(aDbHandle, aDbName, aPragmaSql, stmtHandle);
sl@0
   618
	if(err == KSqlAtRow)
sl@0
   619
		{
sl@0
   620
		aPragmaValue = sqlite3_column_int(stmtHandle, 0);
sl@0
   621
		err = aPragmaValue >= 0 ? KErrNone : KErrCorrupt;
sl@0
   622
		}
sl@0
   623
	(void)FinalizeStmtHandle(stmtHandle);//sqlite3_finalize() fails only if an invalid statement handle is passed.
sl@0
   624
	return err;
sl@0
   625
	}
sl@0
   626
sl@0
   627
/**
sl@0
   628
Prepares and executes PRAGMA statement and retrieves the value of column 0 (the pragma value) as text.
sl@0
   629
sl@0
   630
@param aDbHandle Database handle
sl@0
   631
@param aDbName Attached database name or KNullDesC for the main database
sl@0
   632
@param aPragmaSql Pragma sql statement
sl@0
   633
@param aPragmaValue An output parameter where the pragma value will be stored (as text)
sl@0
   634
sl@0
   635
@panic SqlDb 7 In _DEBUG mode if aDbHandle is NULL
sl@0
   636
@panic SqlDb 4 In _DEBUG mode if aPragmaSql length is 0 or if the statement is not zero-terminated
sl@0
   637
sl@0
   638
@return KErrNone,     Operation completed successfully;
sl@0
   639
		KErrNoMemory, Out of memory;
sl@0
   640
					  Other system-wide error codes or SQL errors of ESqlDbError type.
sl@0
   641
sl@0
   642
@internalComponent
sl@0
   643
*/
sl@0
   644
static TInt RetrievePragmaValue(sqlite3* aDbHandle, const TDesC& aDbName, const TDesC8& aPragmaSql, TDes8& aPragmaValue)
sl@0
   645
	{
sl@0
   646
	__ASSERT_DEBUG(aDbHandle != NULL, __SQLPANIC2(ESqlPanicInternalError));
sl@0
   647
	__ASSERT_DEBUG(aPragmaSql.Length() > 0, __SQLPANIC2(ESqlPanicBadArgument));
sl@0
   648
	__ASSERT_DEBUG(aPragmaSql[aPragmaSql.Length() - 1] == 0, __SQLPANIC2(ESqlPanicBadArgument));
sl@0
   649
	sqlite3_stmt* stmtHandle = NULL;
sl@0
   650
	TInt err = PreRetrievePragmaValue(aDbHandle, aDbName, aPragmaSql, stmtHandle);
sl@0
   651
	if(err == KSqlAtRow)
sl@0
   652
		{
sl@0
   653
		TPtrC8 ptr(sqlite3_column_text(stmtHandle, 0));
sl@0
   654
		aPragmaValue.Copy(ptr);
sl@0
   655
		err = KErrNone;
sl@0
   656
		}
sl@0
   657
	(void)FinalizeStmtHandle(stmtHandle);//sqlite3_finalize() fails only if an invalid statement handle is passed.
sl@0
   658
	return err;
sl@0
   659
	}
sl@0
   660
sl@0
   661
/**
sl@0
   662
Retrieves the database pages count.
sl@0
   663
sl@0
   664
@param aDbHandle Database handle
sl@0
   665
@param aDbName Attached database name or KNullDesC for the main database
sl@0
   666
@param aPageCount An output parameter where the database pages count will be stored
sl@0
   667
sl@0
   668
@return KErrNone,     Operation completed successfully;
sl@0
   669
		KErrNoMemory, Out of memory;
sl@0
   670
					  Other system-wide error codes or SQL errors of ESqlDbError type.
sl@0
   671
sl@0
   672
@panic SqlDb 7 In _DEBUG mode if aDbHandle is NULL
sl@0
   673
sl@0
   674
@internalComponent
sl@0
   675
*/
sl@0
   676
TInt DbPageCount(sqlite3* aDbHandle, const TDesC& aDbName, TInt& aPageCount)
sl@0
   677
	{
sl@0
   678
	__ASSERT_DEBUG(aDbHandle != NULL, __SQLPANIC2(ESqlPanicInternalError));
sl@0
   679
	return RetrievePragmaValue(aDbHandle, aDbName, KPageCountPragma, aPageCount);
sl@0
   680
	}
sl@0
   681
sl@0
   682
/**
sl@0
   683
Retrieves the database page size.
sl@0
   684
sl@0
   685
@param aDbHandle Database handle
sl@0
   686
@param aDbName Attached database name or KNullDesC for the main database
sl@0
   687
@param aPageSize An output parameter where the page size will be stored
sl@0
   688
sl@0
   689
@return KErrNone,     Operation completed successfully;
sl@0
   690
		KErrNoMemory, Out of memory;
sl@0
   691
					  Other system-wide error codes or SQL errors of ESqlDbError type.
sl@0
   692
sl@0
   693
@panic SqlDb 7 In _DEBUG mode if aDbHandle is NULL
sl@0
   694
sl@0
   695
@internalComponent
sl@0
   696
*/
sl@0
   697
TInt DbPageSize(sqlite3* aDbHandle, const TDesC& aDbName, TInt& aPageSize)
sl@0
   698
	{
sl@0
   699
	__ASSERT_DEBUG(aDbHandle != NULL, __SQLPANIC2(ESqlPanicInternalError));
sl@0
   700
	return RetrievePragmaValue(aDbHandle, aDbName, KPageSizePragma, aPageSize);
sl@0
   701
	}
sl@0
   702
sl@0
   703
/**
sl@0
   704
Retrieves the database cache size in pages.
sl@0
   705
sl@0
   706
@param aDbHandle Database handle
sl@0
   707
@param aDbName Attached database name or KNullDesC for the main database
sl@0
   708
@param aCacheSize An output parameter where the cache size will be stored
sl@0
   709
sl@0
   710
@return KErrNone,     Operation completed successfully;
sl@0
   711
		KErrNoMemory, Out of memory;
sl@0
   712
					  Other system-wide error codes or SQL errors of ESqlDbError type.
sl@0
   713
sl@0
   714
@panic SqlDb 7 In _DEBUG mode if aDbHandle is NULL
sl@0
   715
sl@0
   716
@internalComponent
sl@0
   717
*/
sl@0
   718
TInt DbCacheSize(sqlite3* aDbHandle, const TDesC& aDbName, TInt& aCacheSize)
sl@0
   719
	{
sl@0
   720
	__ASSERT_DEBUG(aDbHandle != NULL, __SQLPANIC2(ESqlPanicInternalError));
sl@0
   721
	return RetrievePragmaValue(aDbHandle, aDbName, KCacheSizePragma, aCacheSize);
sl@0
   722
	}
sl@0
   723
sl@0
   724
/**
sl@0
   725
Retrieves the database encoding.
sl@0
   726
sl@0
   727
@param aDbHandle Database handle
sl@0
   728
@param aDbName Attached database name or KNullDesC for the main database
sl@0
   729
@param aEncoding An output parameter where the encoding type will be stored (as text)
sl@0
   730
sl@0
   731
@return KErrNone,     Operation completed successfully;
sl@0
   732
		KErrNoMemory, Out of memory;
sl@0
   733
					  Other system-wide error codes or SQL errors of ESqlDbError type.
sl@0
   734
sl@0
   735
@panic SqlDb 7 In _DEBUG mode if aDbHandle is NULL
sl@0
   736
sl@0
   737
@internalComponent
sl@0
   738
*/
sl@0
   739
TInt DbEncoding(sqlite3* aDbHandle, const TDesC& aDbName, TDes8& aEncoding)
sl@0
   740
	{
sl@0
   741
	__ASSERT_DEBUG(aDbHandle != NULL, __SQLPANIC2(ESqlPanicInternalError));
sl@0
   742
	return RetrievePragmaValue(aDbHandle, aDbName, KEncodingPragma, aEncoding);
sl@0
   743
	}
sl@0
   744
sl@0
   745
/**
sl@0
   746
Retrieves the database free pages count.
sl@0
   747
sl@0
   748
@param aDbHandle Database handle
sl@0
   749
@param aDbName Attached database name or KNullDesC for the main database
sl@0
   750
@param aPageCount An output parameter where the free pages count will be stored
sl@0
   751
sl@0
   752
@return KErrNone,     Operation completed successfully;
sl@0
   753
		KErrNoMemory, Out of memory;
sl@0
   754
					  Other system-wide error codes or SQL errors of ESqlDbError type.
sl@0
   755
sl@0
   756
@panic SqlDb 7 In _DEBUG mode if aDbHandle is NULL
sl@0
   757
sl@0
   758
@internalComponent
sl@0
   759
*/
sl@0
   760
TInt DbFreePageCount(sqlite3* aDbHandle, const TDesC& aDbName, TInt& aPageCount)
sl@0
   761
	{
sl@0
   762
	__ASSERT_DEBUG(aDbHandle != NULL, __SQLPANIC2(ESqlPanicInternalError));
sl@0
   763
	return RetrievePragmaValue(aDbHandle, aDbName, KFreePageCountPragma, aPageCount);
sl@0
   764
	}
sl@0
   765
sl@0
   766
/**
sl@0
   767
Retrieves the current vacuum mode of the database.
sl@0
   768
sl@0
   769
@param aDbHandle Database handle
sl@0
   770
@param aDbName Attached database name or KNullDesC for the main database
sl@0
   771
@param aVacuumMode An output parameter where the current vacuum mode will be stored
sl@0
   772
sl@0
   773
@return KErrNone,     Operation completed successfully;
sl@0
   774
		KErrNoMemory, Out of memory;
sl@0
   775
					  Other system-wide error codes or SQL errors of ESqlDbError type.
sl@0
   776
sl@0
   777
@panic SqlDb 7 In _DEBUG mode if aDbHandle is NULL
sl@0
   778
sl@0
   779
@internalComponent
sl@0
   780
*/
sl@0
   781
TInt DbVacuumMode(sqlite3* aDbHandle, const TDesC& aDbName, TInt& aVacuumMode)
sl@0
   782
	{
sl@0
   783
	__ASSERT_DEBUG(aDbHandle != NULL, __SQLPANIC2(ESqlPanicInternalError));
sl@0
   784
	return RetrievePragmaValue(aDbHandle, aDbName, KVacuumModePragma, aVacuumMode);
sl@0
   785
	}
sl@0
   786
sl@0
   787
static TBool IsCompactTimeLimitReached(TUint32 aStartTicks, TUint32 aCurrTicks, TInt aMaxTime)
sl@0
   788
	{
sl@0
   789
	__ASSERT_DEBUG(aMaxTime > 0, __SQLPANIC2(ESqlPanicBadArgument));
sl@0
   790
	TInt64 tickDiff64 = (TInt64)aCurrTicks - (TInt64)aStartTicks;
sl@0
   791
	if(tickDiff64 < 0)
sl@0
   792
		{
sl@0
   793
		tickDiff64 = KMaxTUint32 + tickDiff64 + 1;
sl@0
   794
		}
sl@0
   795
	static TInt freq = 0;
sl@0
   796
	TInt err = KErrNone;
sl@0
   797
	if(freq == 0)
sl@0
   798
		{
sl@0
   799
		err = HAL::Get(HAL::EFastCounterFrequency, freq);
sl@0
   800
		SQL_TRACE_INTERNALS(OstTraceExt2(TRACE_INTERNALS, ISCOMPACTTIMELIMITREACHED, "0;IsCompactTimeLimitReached;fast counter frequency=%d;err=%d", freq, err));
sl@0
   801
		}
sl@0
   802
	if(err == KErrNone && freq > 0)
sl@0
   803
		{
sl@0
   804
		const TInt KMicroSecIn1Sec = 1000000;
sl@0
   805
		const TInt KMicroSecIn1Ms = 1000;
sl@0
   806
		TInt64 usDiff64 = (tickDiff64 * KMicroSecIn1Sec) / freq;
sl@0
   807
		if(usDiff64 > aMaxTime * KMicroSecIn1Ms)
sl@0
   808
			{
sl@0
   809
			return ETrue;	
sl@0
   810
			}
sl@0
   811
		}
sl@0
   812
	return EFalse;
sl@0
   813
	}
sl@0
   814
sl@0
   815
/**
sl@0
   816
Compacts the database.
sl@0
   817
sl@0
   818
@param aDbHandle 			Database handle.
sl@0
   819
@param aPageCount 			Count of the free database pages to be removed from the file.
sl@0
   820
@param aProcessedPageCount	Output parameter. How many pages actually have been removed from the file.
sl@0
   821
@param aMaxTime				The max allowed time in milliseconds for the compact operation.
sl@0
   822
							If aMaxTime is zero, then aPageCount pages will be removed regardless the time.
sl@0
   823
sl@0
   824
@return KErrNone,     Operation completed successfully;
sl@0
   825
					  Other system-wide error codes or SQL errors of ESqlDbError type.
sl@0
   826
sl@0
   827
@panic SqlDb 4 In _DEBUG mode if aPageCount is negative
sl@0
   828
@panic SqlDb 4 In _DEBUG mode if aMaxTime is negative
sl@0
   829
@panic SqlDb 7 In _DEBUG mode if aDbHandle is NULL
sl@0
   830
sl@0
   831
@internalComponent
sl@0
   832
*/
sl@0
   833
TInt DbCompact(sqlite3* aDbHandle, const TDesC& aDbName, TInt aPageCount, TInt& aProcessedPageCount, TInt aMaxTime)
sl@0
   834
	{
sl@0
   835
	SQL_TRACE_INTERNALS(OstTraceExt4(TRACE_INTERNALS, DBCOMPACT_ENTRY, "Entry;0x%X;DbCompact;aDbName=%S;aPageCount=%d;aMaxTime=%d", (TUint)aDbHandle, __SQLPRNSTR(aDbName), aPageCount, aMaxTime));
sl@0
   836
	__ASSERT_DEBUG(aPageCount >= 0, __SQLPANIC2(ESqlPanicBadArgument));
sl@0
   837
	__ASSERT_DEBUG(aMaxTime >= 0, __SQLPANIC2(ESqlPanicBadArgument));
sl@0
   838
	__ASSERT_DEBUG(aDbHandle != NULL, __SQLPANIC2(ESqlPanicInternalError));
sl@0
   839
	TBuf8<KMaxFileName> dbName;
sl@0
   840
	if(!UTF16ToUTF8(aDbName, dbName))
sl@0
   841
		{
sl@0
   842
		SQL_TRACE_INTERNALS(OstTrace1(TRACE_INTERNALS, DBCOMPACT_EXIT1, "Exit;0x%X;DbCompact;err=KErrGeneral", (TUint)aDbHandle));
sl@0
   843
		return KErrGeneral;
sl@0
   844
		}
sl@0
   845
	TBuf8<KMaxFileName + sizeof(KIncrementalVacuumPragma) + 1> sql;
sl@0
   846
	if(dbName == KNullDesC8)
sl@0
   847
		{
sl@0
   848
		sql.Format(KIncrementalVacuumPragma, &KMainDb8, aPageCount);
sl@0
   849
		}
sl@0
   850
	else
sl@0
   851
		{
sl@0
   852
		sql.Format(KIncrementalVacuumPragma, &dbName, aPageCount);
sl@0
   853
		}
sl@0
   854
	//Currently there is no way to check how many pages have been compacted without executing a "PRAGMA freelist_count"	
sl@0
   855
	//statement, if sqlite3_exec() is used. 
sl@0
   856
	//So, instead of calling sqlite3_exec(), the function prepares and executes the "PRAGMA incremental_vacuum(N)"
sl@0
   857
	//statement using sqlite3_step() and counts the steps, because each step compacts one page.
sl@0
   858
	(void)sqlite3SymbianLastOsError();//clear last OS error
sl@0
   859
	sqlite3_stmt* stmtHandle = NULL;
sl@0
   860
	const char* stmtTail = NULL;
sl@0
   861
	aProcessedPageCount = 0;
sl@0
   862
	//sqlite3_prepare_v2() expects parameter #3 to be one of the following:
sl@0
   863
	// - byte length of the sql statement (parameter #2), excluding terminating zero;
sl@0
   864
	// - negative value - the sql statement (parameter #2) is zero-terminated;
sl@0
   865
	TInt err = sqlite3_prepare_v2(aDbHandle, (const char*)sql.Ptr(), sql.Length() - sizeof(TUint8), &stmtHandle, &stmtTail);
sl@0
   866
	__ASSERT_DEBUG(err == SQLITE_OK ? !stmtTail || User::StringLength((const TUint8*)stmtTail) == 0 : !stmtHandle, __SQLPANIC2(ESqlPanicInternalError));
sl@0
   867
	if(stmtHandle)	//stmtHandle can be NULL for statements like this: ";".
sl@0
   868
		{
sl@0
   869
		if(err == SQLITE_OK)
sl@0
   870
			{
sl@0
   871
			TUint32 startTicks = 0;
sl@0
   872
			if(aMaxTime > 0)
sl@0
   873
				{
sl@0
   874
				startTicks = User::FastCounter();
sl@0
   875
				}
sl@0
   876
			
sl@0
   877
			while((err = sqlite3_step(stmtHandle)) == SQLITE_ROW)
sl@0
   878
                {
sl@0
   879
                ++aProcessedPageCount;
sl@0
   880
                if(aMaxTime > 0 && IsCompactTimeLimitReached(startTicks, User::FastCounter(), aMaxTime))
sl@0
   881
                    {
sl@0
   882
                    err = SQLITE_DONE;//The statement execution did not complete because of the time limit
sl@0
   883
                    break;  
sl@0
   884
                    }
sl@0
   885
                }
sl@0
   886
			
sl@0
   887
			if(err == SQLITE_ERROR)  //It may be "out of memory" problem
sl@0
   888
                {
sl@0
   889
                err = sqlite3_reset(stmtHandle);
sl@0
   890
                __ASSERT_DEBUG(err != SQLITE_OK, __SQLPANIC2(ESqlPanicInternalError));
sl@0
   891
                }
sl@0
   892
			}
sl@0
   893
		(void)sqlite3_finalize(stmtHandle);//sqlite3_finalize() fails only if an invalid statement handle is passed.
sl@0
   894
		}
sl@0
   895
	err = ::Sql2OsErrCode(err, sqlite3SymbianLastOsError());
sl@0
   896
	if(err == KSqlAtEnd)
sl@0
   897
		{
sl@0
   898
		err = KErrNone;	
sl@0
   899
		}
sl@0
   900
	SQL_TRACE_INTERNALS(OstTraceExt3(TRACE_INTERNALS, DBCOMPACT_EXIT2, "Exit;0x%X;DbCompact;aProcessedPageCount=%d;err=%d", (TUint)aDbHandle, aProcessedPageCount, err));
sl@0
   901
	return err;
sl@0
   902
	}
sl@0
   903
sl@0
   904
/**
sl@0
   905
Finalizes the statement handle.
sl@0
   906
Although the function can return an error, it is ok not to check the returned error,
sl@0
   907
because sqlite3_finalize() fails only if invalid statement handle is passed as an argument.
sl@0
   908
sl@0
   909
@return KErrNone,     Operation completed successfully;
sl@0
   910
					  Other system-wide error codes or SQL errors of ESqlDbError type.
sl@0
   911
sl@0
   912
@internalComponent
sl@0
   913
*/
sl@0
   914
TInt FinalizeStmtHandle(sqlite3_stmt* aStmtHandle)
sl@0
   915
	{
sl@0
   916
	SQL_TRACE_INTERNALS(OstTraceExt2(TRACE_INTERNALS, FINALIZESTMTHANDLE_ENTRY, "Entry;0x%X;FinalizeStmtHandle;aStmtHandle=0x%X", (TUint)sqlite3_db_handle(aStmtHandle), (TUint)aStmtHandle));
sl@0
   917
	TInt err = KErrNone;
sl@0
   918
	if(aStmtHandle)
sl@0
   919
		{
sl@0
   920
		(void)sqlite3SymbianLastOsError();//clear last OS error
sl@0
   921
		err = sqlite3_finalize(aStmtHandle);
sl@0
   922
		err = ::Sql2OsErrCode(err, sqlite3SymbianLastOsError());
sl@0
   923
		}
sl@0
   924
	SQL_TRACE_INTERNALS(OstTrace1(TRACE_INTERNALS, FINALIZESTMTHANDLE_EXIT, "Exit;0;FinalizeStmtHandle;err=%d", err));
sl@0
   925
	return err;
sl@0
   926
	}