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