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".
8 // Initial Contributors:
9 // Nokia Corporation - initial contribution.
20 #include "SqlSrvStatementUtil.h"
21 #include "SqlAssert.h"
22 #include "SqlSrvUtil.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"
30 #include "SqlTraceDef.h"
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");
41 ///////////////////////////////////////////////////////////////////////////////////////////////////////////////
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)
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));
54 (void)sqlite3_extended_result_codes(aDbHandle, 0);
56 //Get the return error code now, because the next "if" may destroy it.
57 TInt rc = ::Sql2OsErrCode(err, sqlite3SymbianLastOsError());
59 {//Yes, it is possible error code != SQLITE_OK and aDbHandle != NULL.
60 CloseDbHandle(aDbHandle);
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)
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));
77 (void)sqlite3_extended_result_codes(aDbHandle, 0);
79 //Get the return error code now, because the next "if" may destroy it.
80 TInt rc = ::Sql2OsErrCode(err, sqlite3SymbianLastOsError());
82 {//Yes, it is possible error code != SQLITE_OK and aDbHandle != NULL.
83 CloseDbHandle(aDbHandle);
89 ///////////////////////////////////////////////////////////////////////////////////////////////////////////////
90 ///// 16-bit and 8-bit SQL statements execution upon completion ////
91 ///////////////////////////////////////////////////////////////////////////////////////////////////////////////
93 //16-bit SQL statement execution.
95 //aSql - zero-terminated string
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).
101 //The function panics in debug mode (panic code 2) if aDbHandle is NULL.
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)
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: ";".
120 while((err = sqlite3_step(stmtHandle)) == SQLITE_ROW)
123 if(err == SQLITE_ERROR) //It may be "out of memory" problem
125 err = sqlite3_reset(stmtHandle);
126 __ASSERT_DEBUG(err != SQLITE_OK, __SQLPANIC2(ESqlPanicInternalError));
129 (void)sqlite3_finalize(stmtHandle);//sqlite3_finalize() fails only if an invalid statement handle is passed.
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.
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.
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).
146 The function expects aString argument to be zero-terminated.
150 TPtrC GetFirstSqlStmt(TPtr& aString)
152 const TChar KDelimitier(';');
153 TPtr originalStr(aString);
154 TPtr str(const_cast <TUint16*> (aString.Ptr()), aString.Length(), aString.Length());
155 TInt afterDelimitierPos = 0;
157 while((pos = str.Locate(KDelimitier) + 1) > 0 && pos < str.Length())
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;
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);
175 str.Set(const_cast <TUint16*> (str.Ptr()) + pos, str.Length() - pos, str.Length() - pos);
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);
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
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.
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.
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
205 TInt DbExecStmt16(sqlite3* aDbHandle, TDes16& aSqlStmt)
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)));
210 __ASSERT_DEBUG(aDbHandle != NULL, __SQLPANIC2(ESqlPanicInternalError));
211 __ASSERT_DEBUG(aSqlStmt.Length() > 0 ? (TInt)aSqlStmt[aSqlStmt.Length() - 1] == 0: ETrue, __SQLPANIC2(ESqlPanicBadArgument));
213 (void)sqlite3SymbianLastOsError();//clear last OS error
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
223 firstSqlStmt.Set(GetFirstSqlStmt(sql));
224 SQLPROFILER_SQL16_PRINT((TUint)aDbHandle, firstSqlStmt.Left(firstSqlStmt.Length() - 1), EFalse);
225 err = ::DoSingleStmtExec16(aDbHandle, firstSqlStmt);
228 err = ::Sql2OsErrCode(err, sqlite3SymbianLastOsError());
233 SQL_TRACE_INTERNALS(OstTraceExt2(TRACE_INTERNALS, DBEXECSTMT16_EXIT, "Exit;0x%X;DbExecStmt16;err=%d", (TUint)aDbHandle, err));
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
243 @param aDbHandle Database handle. Not NULL.
244 @param aSqlStmt String containing one or more 8-bit SQL statements, separated with ';'. Zero-terminated string.
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.
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
257 TInt DbExecStmt8(sqlite3* aDbHandle, const TDesC8& aSqlStmt)
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)));
263 __ASSERT_DEBUG(aDbHandle != NULL, __SQLPANIC2(ESqlPanicInternalError));
264 __ASSERT_DEBUG(aSqlStmt.Length() > 0 ? (TInt)aSqlStmt[aSqlStmt.Length() - 1] == 0: ETrue, __SQLPANIC2(ESqlPanicBadArgument));
266 SQLPROFILER_SQL8_PRINT((TUint)aDbHandle, aSqlStmt.Left(aSqlStmt.Length() - 1), EFalse);
268 (void)sqlite3SymbianLastOsError();//clear last OS error
270 TInt err = sqlite3_exec(aDbHandle, reinterpret_cast <const char*> (aSqlStmt.Ptr()), NULL, NULL, NULL);
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));
278 ///////////////////////////////////////////////////////////////////////////////////////////////////////////////
279 ///// 16-bit and 8-bit SQL statement preparation /////
280 ///////////////////////////////////////////////////////////////////////////////////////////////////////////////
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)
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)
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)
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)
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.
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;
327 static TInt ProcessPrepareError(TInt aSqliteError, TBool aHasTail, sqlite3_stmt*& aStmtHandle)
329 if(aSqliteError != SQLITE_OK)
331 return ::Sql2OsErrCode(aSqliteError, sqlite3SymbianLastOsError());
333 else if(aHasTail || !aStmtHandle)
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.
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);
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.
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;
356 static void LeaveIfPrepareErrorL(TInt aSqliteError, TBool aHasTail, sqlite3_stmt* aStmtHandle)
358 __SQLLEAVE_IF_ERROR2(ProcessPrepareError(aSqliteError, aHasTail, aStmtHandle));
362 Prepares 16-bit aSqlStmt SQL statement.
364 @param aSqlStmt - zero-terminated string.
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.
370 @return The prepared SQL statement handle.
374 sqlite3_stmt* StmtPrepare16L(sqlite3* aDbHandle, const TDesC16& aSqlStmt)
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));
388 Prepares 8-bit aSqlStmt SQL statement.
390 @param aSqlStmt - zero-terminated string.
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.
396 @return The prepared SQL statement handle.
400 TInt StmtPrepare8(sqlite3* aDbHandle, const TDesC8& aSqlStmt, sqlite3_stmt*& aStmtHandle)
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));
414 Prepares 8-bit aSqlStmt SQL statement.
416 @param aSqlStmt - zero-terminated string.
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.
422 @return The prepared SQL statement handle.
426 sqlite3_stmt* StmtPrepare8L(sqlite3* aDbHandle, const TDesC8& aSqlStmt)
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));
441 Executes upon completion the prepared SQL statement.
443 @param aStmtHandle Prepared statement handle
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.
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)
455 TInt StmtExec(sqlite3_stmt* aStmtHandle)
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));
460 (void)sqlite3SymbianLastOsError();//clear last OS error
463 while((err = sqlite3_step(aStmtHandle)) == SQLITE_ROW)
467 if(err == SQLITE_ERROR) //It may be "out of memory" problem
469 err = sqlite3_reset(aStmtHandle);
470 __ASSERT_DEBUG(err != SQLITE_OK, __SQLPANIC2(ESqlPanicInternalError));
473 err = ::Sql2OsErrCode(err, sqlite3SymbianLastOsError());
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));
484 Executes the SQL statement moving it to the next row if available.
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.
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)
500 TInt StmtNext(sqlite3_stmt* aStmtHandle)
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));
505 (void)sqlite3SymbianLastOsError();//clear last OS error
507 TInt err = sqlite3_step(aStmtHandle);
508 if(err == SQLITE_ERROR) //It may be "out of memory" problem
510 err = sqlite3_reset(aStmtHandle);
511 __ASSERT_DEBUG(err != SQLITE_OK, __SQLPANIC2(ESqlPanicInternalError));
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));
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.
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.
526 @panic SqlDb 2 In _DEBUG mode. Invalid (NULL) statement handle.
530 TInt StmtReset(sqlite3_stmt* aStmtHandle)
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));
535 (void)sqlite3SymbianLastOsError();//clear last OS error
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));
544 Prepares and executes PRAGMA statement and moves the statement cursor on the first row.
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
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
554 @return KErrNone, Operation completed successfully;
555 KErrNoMemory, Out of memory;
556 Other system-wide error codes or SQL errors of ESqlDbError type.
560 static TInt PreRetrievePragmaValue(sqlite3* aDbHandle, const TDesC& aDbName, const TDesC8& aPragmaSql, sqlite3_stmt*& aStmtHandle)
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))
571 SQL_TRACE_INTERNALS(OstTrace1(TRACE_INTERNALS, PRERETRIEVEPRAGMAVALUE_EXIT1, "Exit;0x%X;PreRetrievePragmaValue;err=KErrGeneral", (TUint)aDbHandle));
574 TBuf8<KMaxFileName + 64> sql;//64 characters is enough for the longest PRAGMA statement
575 if(dbName == KNullDesC8)
577 sql.Format(aPragmaSql, &KMainDb8);
581 sql.Format(aPragmaSql, &dbName);
584 TInt err = ::StmtPrepare8(aDbHandle, sql, aStmtHandle);
587 __ASSERT_DEBUG(aStmtHandle != NULL, __SQLPANIC2(ESqlPanicInvalidObj));
588 err = ::StmtNext(aStmtHandle);
590 SQL_TRACE_INTERNALS(OstTraceExt3(TRACE_INTERNALS, PRERETRIEVEPRAGMAVALUE_EXIT2, "Exit;0x%X;PreRetrievePragmaValue;aStmtHandle=0x%X;err=%d", (TUint)aDbHandle, (TUint)aStmtHandle, err));
595 Prepares and executes PRAGMA statement and retrieves the value of column 0 (the pragma value).
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
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
605 @return KErrNone, Operation completed successfully;
606 KErrNoMemory, Out of memory;
607 Other system-wide error codes or SQL errors of ESqlDbError type.
611 static TInt RetrievePragmaValue(sqlite3* aDbHandle, const TDesC& aDbName, const TDesC8& aPragmaSql, TInt& aPragmaValue)
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);
620 aPragmaValue = sqlite3_column_int(stmtHandle, 0);
621 err = aPragmaValue >= 0 ? KErrNone : KErrCorrupt;
623 (void)FinalizeStmtHandle(stmtHandle);//sqlite3_finalize() fails only if an invalid statement handle is passed.
628 Prepares and executes PRAGMA statement and retrieves the value of column 0 (the pragma value) as text.
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)
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
638 @return KErrNone, Operation completed successfully;
639 KErrNoMemory, Out of memory;
640 Other system-wide error codes or SQL errors of ESqlDbError type.
644 static TInt RetrievePragmaValue(sqlite3* aDbHandle, const TDesC& aDbName, const TDesC8& aPragmaSql, TDes8& aPragmaValue)
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);
653 TPtrC8 ptr(sqlite3_column_text(stmtHandle, 0));
654 aPragmaValue.Copy(ptr);
657 (void)FinalizeStmtHandle(stmtHandle);//sqlite3_finalize() fails only if an invalid statement handle is passed.
662 Retrieves the database pages count.
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
668 @return KErrNone, Operation completed successfully;
669 KErrNoMemory, Out of memory;
670 Other system-wide error codes or SQL errors of ESqlDbError type.
672 @panic SqlDb 7 In _DEBUG mode if aDbHandle is NULL
676 TInt DbPageCount(sqlite3* aDbHandle, const TDesC& aDbName, TInt& aPageCount)
678 __ASSERT_DEBUG(aDbHandle != NULL, __SQLPANIC2(ESqlPanicInternalError));
679 return RetrievePragmaValue(aDbHandle, aDbName, KPageCountPragma, aPageCount);
683 Retrieves the database page size.
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
689 @return KErrNone, Operation completed successfully;
690 KErrNoMemory, Out of memory;
691 Other system-wide error codes or SQL errors of ESqlDbError type.
693 @panic SqlDb 7 In _DEBUG mode if aDbHandle is NULL
697 TInt DbPageSize(sqlite3* aDbHandle, const TDesC& aDbName, TInt& aPageSize)
699 __ASSERT_DEBUG(aDbHandle != NULL, __SQLPANIC2(ESqlPanicInternalError));
700 return RetrievePragmaValue(aDbHandle, aDbName, KPageSizePragma, aPageSize);
704 Retrieves the database cache size in pages.
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
710 @return KErrNone, Operation completed successfully;
711 KErrNoMemory, Out of memory;
712 Other system-wide error codes or SQL errors of ESqlDbError type.
714 @panic SqlDb 7 In _DEBUG mode if aDbHandle is NULL
718 TInt DbCacheSize(sqlite3* aDbHandle, const TDesC& aDbName, TInt& aCacheSize)
720 __ASSERT_DEBUG(aDbHandle != NULL, __SQLPANIC2(ESqlPanicInternalError));
721 return RetrievePragmaValue(aDbHandle, aDbName, KCacheSizePragma, aCacheSize);
725 Retrieves the database encoding.
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)
731 @return KErrNone, Operation completed successfully;
732 KErrNoMemory, Out of memory;
733 Other system-wide error codes or SQL errors of ESqlDbError type.
735 @panic SqlDb 7 In _DEBUG mode if aDbHandle is NULL
739 TInt DbEncoding(sqlite3* aDbHandle, const TDesC& aDbName, TDes8& aEncoding)
741 __ASSERT_DEBUG(aDbHandle != NULL, __SQLPANIC2(ESqlPanicInternalError));
742 return RetrievePragmaValue(aDbHandle, aDbName, KEncodingPragma, aEncoding);
746 Retrieves the database free pages count.
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
752 @return KErrNone, Operation completed successfully;
753 KErrNoMemory, Out of memory;
754 Other system-wide error codes or SQL errors of ESqlDbError type.
756 @panic SqlDb 7 In _DEBUG mode if aDbHandle is NULL
760 TInt DbFreePageCount(sqlite3* aDbHandle, const TDesC& aDbName, TInt& aPageCount)
762 __ASSERT_DEBUG(aDbHandle != NULL, __SQLPANIC2(ESqlPanicInternalError));
763 return RetrievePragmaValue(aDbHandle, aDbName, KFreePageCountPragma, aPageCount);
767 Retrieves the current vacuum mode of the database.
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
773 @return KErrNone, Operation completed successfully;
774 KErrNoMemory, Out of memory;
775 Other system-wide error codes or SQL errors of ESqlDbError type.
777 @panic SqlDb 7 In _DEBUG mode if aDbHandle is NULL
781 TInt DbVacuumMode(sqlite3* aDbHandle, const TDesC& aDbName, TInt& aVacuumMode)
783 __ASSERT_DEBUG(aDbHandle != NULL, __SQLPANIC2(ESqlPanicInternalError));
784 return RetrievePragmaValue(aDbHandle, aDbName, KVacuumModePragma, aVacuumMode);
787 static TBool IsCompactTimeLimitReached(TUint32 aStartTicks, TUint32 aCurrTicks, TInt aMaxTime)
789 __ASSERT_DEBUG(aMaxTime > 0, __SQLPANIC2(ESqlPanicBadArgument));
790 TInt64 tickDiff64 = (TInt64)aCurrTicks - (TInt64)aStartTicks;
793 tickDiff64 = KMaxTUint32 + tickDiff64 + 1;
795 static TInt freq = 0;
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));
802 if(err == KErrNone && freq > 0)
804 const TInt KMicroSecIn1Sec = 1000000;
805 const TInt KMicroSecIn1Ms = 1000;
806 TInt64 usDiff64 = (tickDiff64 * KMicroSecIn1Sec) / freq;
807 if(usDiff64 > aMaxTime * KMicroSecIn1Ms)
816 Compacts the database.
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.
824 @return KErrNone, Operation completed successfully;
825 Other system-wide error codes or SQL errors of ESqlDbError type.
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
833 TInt DbCompact(sqlite3* aDbHandle, const TDesC& aDbName, TInt aPageCount, TInt& aProcessedPageCount, TInt aMaxTime)
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))
842 SQL_TRACE_INTERNALS(OstTrace1(TRACE_INTERNALS, DBCOMPACT_EXIT1, "Exit;0x%X;DbCompact;err=KErrGeneral", (TUint)aDbHandle));
845 TBuf8<KMaxFileName + sizeof(KIncrementalVacuumPragma) + 1> sql;
846 if(dbName == KNullDesC8)
848 sql.Format(KIncrementalVacuumPragma, &KMainDb8, aPageCount);
852 sql.Format(KIncrementalVacuumPragma, &dbName, aPageCount);
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: ";".
871 TUint32 startTicks = 0;
874 startTicks = User::FastCounter();
877 while((err = sqlite3_step(stmtHandle)) == SQLITE_ROW)
879 ++aProcessedPageCount;
880 if(aMaxTime > 0 && IsCompactTimeLimitReached(startTicks, User::FastCounter(), aMaxTime))
882 err = SQLITE_DONE;//The statement execution did not complete because of the time limit
887 if(err == SQLITE_ERROR) //It may be "out of memory" problem
889 err = sqlite3_reset(stmtHandle);
890 __ASSERT_DEBUG(err != SQLITE_OK, __SQLPANIC2(ESqlPanicInternalError));
893 (void)sqlite3_finalize(stmtHandle);//sqlite3_finalize() fails only if an invalid statement handle is passed.
895 err = ::Sql2OsErrCode(err, sqlite3SymbianLastOsError());
900 SQL_TRACE_INTERNALS(OstTraceExt3(TRACE_INTERNALS, DBCOMPACT_EXIT2, "Exit;0x%X;DbCompact;aProcessedPageCount=%d;err=%d", (TUint)aDbHandle, aProcessedPageCount, err));
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.
909 @return KErrNone, Operation completed successfully;
910 Other system-wide error codes or SQL errors of ESqlDbError type.
914 TInt FinalizeStmtHandle(sqlite3_stmt* aStmtHandle)
916 SQL_TRACE_INTERNALS(OstTraceExt2(TRACE_INTERNALS, FINALIZESTMTHANDLE_ENTRY, "Entry;0x%X;FinalizeStmtHandle;aStmtHandle=0x%X", (TUint)sqlite3_db_handle(aStmtHandle), (TUint)aStmtHandle));
920 (void)sqlite3SymbianLastOsError();//clear last OS error
921 err = sqlite3_finalize(aStmtHandle);
922 err = ::Sql2OsErrCode(err, sqlite3SymbianLastOsError());
924 SQL_TRACE_INTERNALS(OstTrace1(TRACE_INTERNALS, FINALIZESTMTHANDLE_EXIT, "Exit;0;FinalizeStmtHandle;err=%d", err));