os/persistentdata/persistentstorage/sql/SRC/Server/SqlSrvStatement.cpp
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
     1 // Copyright (c) 2005-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 // NTT DOCOMO, INC - Fix for defect 1915 "SQL server panics when using long column type strings"
    13 //
    14 // Description:
    15 //
    16 
    17 #include <utf.h>					//CnvUtfConverter
    18 #include <e32math.h>
    19 #include "SqliteSymbian.h"			//sqlite3SymbianLastOsError()
    20 #include "sqlite3.h"
    21 #include "SqlSrvStatement.h"
    22 #include "SqlBufIterator.h"			//TSqlBufRIterator
    23 #include "SqlSrvResourceProfiler.h"
    24 #include "OstTraceDefinitions.h"
    25 #ifdef OST_TRACE_COMPILER_IN_USE
    26 #include "SqlSrvStatementTraces.h"
    27 #endif
    28 #include "SqlTraceDef.h"
    29 
    30 //////////////////////////////////////////////////////////////////////////////////////////////////////
    31 /////////////////////////////      local const data       ////////////////////////////////////////////
    32 //////////////////////////////////////////////////////////////////////////////////////////////////////
    33 
    34 //This is the name prefix which will be given to the nameless parameters.
    35 //For example, if the SQL string is:
    36 //   SELECT * FROM A WHERE ColA1 = ? AND ColA2 = ?
    37 //then the names which will be given to the parameters will be:
    38 //"?0" and "?1"
    39 _LIT(KNamelessParameter, "?");
    40 
    41 
    42 /////////////////////////////////////////////////////////////////////////////////////////////////////////
    43 //////////////////                HSqlSrvStmtParamBuf                  //////////////////////////////////
    44 /////////////////////////////////////////////////////////////////////////////////////////////////////////
    45 
    46 /**
    47 Destroys the parameter buffer.
    48 
    49 Virtual method.
    50 */
    51 HSqlSrvStmtParamBuf::~HSqlSrvStmtParamBuf()
    52 	{
    53 	delete iBuf;
    54 	}
    55 
    56 /**
    57 Binds the parameter value.
    58 The buffer can be synch-ed if:
    59  - this is the first synch operation;
    60  - the bound statement object is still alive (not finalized);
    61  - the current object is alive;
    62  - the current object data is retrieved from an IPC stream;
    63  
    64 If none of the conditions above is true, the synch operation is no-op.
    65 
    66 Virtual method.
    67 */
    68 void HSqlSrvStmtParamBuf::DoSynchL()
    69 	{
    70 	TBool dontSynch = iSynchDone || !iAlive || iStatementFinalized || iBufType != HSqlSrvStmtParamBuf::EBufIpcStream;
    71 	if(dontSynch)
    72 		{
    73 		return;	
    74 		}
    75 	iSynchDone = ETrue;
    76 	TBufBuf::DoSynchL();
    77 	iStatement.BindParamBufL(iParamIndex);
    78 	}
    79 
    80 /**
    81 Destroys the HSqlSrvStmtParamBuf instance.
    82 This method is a no-op if the statement is not finalized yet.
    83 
    84 Virtual method.
    85 */
    86 void HSqlSrvStmtParamBuf::DoRelease()
    87  	{
    88 	iAlive = EFalse;
    89  	if(iStatementFinalized)
    90  		{//The bound statement has been finalized - destroy the current object then.
    91  		delete this;
    92  		}
    93 	}
    94 
    95 /**
    96 This function is called by the bound statement object to notify the current HSqlSrvStmtParamBuf object that the
    97 bound statement is about to be finalized. That means, when the "stream close" operation on the client side 
    98 makes an attempt to synch the HSqlSrvStmtParamBuf object, no attempt should be made to bound the parameter data,
    99 because the statement object is gone.
   100 After this call the bound statement objects ceases to exist.
   101 
   102 Actions, performed by this method:
   103  - if the buffer type is "simple bind", the buffer will be destroyed. No reason to keep it alive, there is no bound IPC
   104    stream object on the client side;
   105  - if the buffer type is an IPC stream buffer and the buffer is alive, that means: the bound statement object is about to be
   106    finalized, but there is a bound client side IPC stream object that is still alive. In this case the buffer won't be destroyed,
   107    but will be "told" that the bound statement is finalized, so when the client side IPC stream is closed, this object will get destroyed;
   108  - if the buffer type is an IPC stream buffer and the buffer is "dead", that means there is no bound IPC stream object on the client
   109    side and it is safe to destroy the buffer;
   110 
   111 */ 	
   112 void HSqlSrvStmtParamBuf::NotifyStatementFinalized()
   113 	{
   114 	iStatementFinalized = ETrue;
   115 	if(iBufType == HSqlSrvStmtParamBuf::EBufSimpleBind || !iAlive)
   116 		{
   117 		DoRelease();	
   118 		}
   119 	}
   120 
   121 //////////////////////////////////////////////////////////////////////////////////////////////////////
   122 /////////////////////////////   CSqlSrvStatement class    ////////////////////////////////////////////
   123 //////////////////////////////////////////////////////////////////////////////////////////////////////
   124 
   125 /**
   126 Creates a new CSqlSrvStatement instance.
   127 
   128 The created CSqlSrvStatement instance will be placed in the cleanup stack.
   129 
   130 @param aDbHandle The database handle
   131 @param aSqlStmt 16-bit SQL statement, zero-terminated string
   132 @param aColumnCount Output parameter. It will be initialized with the column count.
   133 @param aParamCount Output parameter. It will be initialized with the parameter count.
   134 
   135 @return A pointer to the created CSqlSrvStatement instance.
   136 
   137 @leave KErrNoMemory, an out of memory condition has occurred;
   138 	   KErrArgument, bad argument, for example - the SQL string contains more than one SQL statements.
   139                   Note that the function may also leave with some other database specific 
   140                   errors categorised as ESqlDbError.
   141 
   142 @panic SqlDb 4 In _DEBUG mode if aSqlStmt is not zero-terminated string.
   143 */	
   144 CSqlSrvStatement* CSqlSrvStatement::NewLC(sqlite3* aDbHandle, const TDesC16& aSqlStmt, TInt& aColumnCount, TInt& aParamCount)
   145 	{
   146     __SQLTRACE_INTERNALSEXPR(TPtrC sqlprnptr(aSqlStmt.Left(aSqlStmt.Length() - 1)));
   147     SQL_TRACE_INTERNALS(OstTraceExt2(TRACE_INTERNALS, CSQLSRVSTATEMENT_NEWLC_ENTRY, "Entry;0;CSqlSrvStatement::NewLC-16;aDbHandle=0x%X;aSqlStmt=%S", (TUint)aDbHandle, __SQLPRNSTR(sqlprnptr)));
   148 	__ASSERT_DEBUG(aSqlStmt.Length() > 0 ? (TInt)aSqlStmt[aSqlStmt.Length() - 1] == 0 : ETrue, __SQLPANIC2(ESqlPanicBadArgument));
   149 	
   150 	CSqlSrvStatement* self = new (ELeave) CSqlSrvStatement;
   151 	CleanupStack::PushL(self);
   152 	self->ConstructL(aDbHandle, aSqlStmt);
   153 	aColumnCount = self->iColumnCount;
   154 	aParamCount = self->iParamCount;
   155 	SQL_TRACE_INTERNALS(OstTraceExt4(TRACE_INTERNALS, CSQLSRVSTATEMENT_NEWLC_EXIT, "Exit;0x%X;CSqlSrvStatement::NewLC-16;iStmtHandle=0x%X;aColumnCount=%d;aParamCount=%d", (TUint)self, (TUint)self->iStmtHandle, aColumnCount, aParamCount));
   156 	return self;
   157 	}
   158 	
   159 /**
   160 Creates a new CSqlSrvStatement instance.
   161 
   162 The created CSqlSrvStatement instance will be placed in the cleanup stack.
   163 
   164 @param aDbHandle The database handle
   165 @param aSqlStmt 8-bit SQL statement, zero-terminated string
   166 @param aColumnCount Output parameter. It will be initialized with the column count.
   167 @param aParamCount Output parameter. It will be initialized with the parameter count.
   168 
   169 @return A pointer to the created CSqlSrvStatement instance.
   170 
   171 @leave KErrNoMemory, an out of memory condition has occurred;
   172 	   KErrArgument, bad argument, for example - the SQL string contains more than one SQL statements.
   173                   Note that the function may also leave with some other database specific 
   174                   errors categorised as ESqlDbError.
   175 
   176 @panic SqlDb 4 In _DEBUG mode if aSqlStmt is not zero-terminated string.
   177 */	
   178 CSqlSrvStatement* CSqlSrvStatement::NewLC(sqlite3* aDbHandle, const TDesC8& aSqlStmt, TInt& aColumnCount, TInt& aParamCount)
   179 	{
   180     __SQLTRACE_INTERNALSEXPR(TPtrC8 sqlprnptr(aSqlStmt.Left(aSqlStmt.Length() - 1)));
   181 	__SQLTRACE_INTERNALSVAR(TBuf<100> des16prnbuf);
   182     SQL_TRACE_INTERNALS(OstTraceExt2(TRACE_INTERNALS, CSQLSRVSTATEMENT_NEWLC_ENTRY2, "Entry;0;CSqlSrvStatement::NewLC-8;aDbHandle=0x%X;aSqlStmt=%s", (TUint)aDbHandle, __SQLPRNSTR8(sqlprnptr, des16prnbuf)));
   183 	__ASSERT_DEBUG(aSqlStmt.Length() > 0 ? (TInt)aSqlStmt[aSqlStmt.Length() - 1] == 0 : ETrue, __SQLPANIC2(ESqlPanicBadArgument));
   184 
   185 	CSqlSrvStatement* self = new (ELeave) CSqlSrvStatement;
   186 	CleanupStack::PushL(self);
   187 	self->ConstructL(aDbHandle, aSqlStmt);
   188 	aColumnCount = self->iColumnCount;
   189 	aParamCount = self->iParamCount;
   190 	SQL_TRACE_INTERNALS(OstTraceExt4(TRACE_INTERNALS, CSQLSRVSTATEMENT_NEWLC_EXIT2, "Exit;0x%X;CSqlSrvStatement::NewLC-8;iStmtHandle=0x%X;aColumnCount=%d;aParamCount=%d", (TUint)self, (TUint)self->iStmtHandle, aColumnCount, aParamCount));
   191 	return self;
   192 	}
   193 	
   194 /**
   195 Destroys the allocated by CSqlSrvStatement instance memory and other resources.
   196 */	
   197 CSqlSrvStatement::~CSqlSrvStatement()
   198 	{
   199 	SQL_TRACE_INTERNALS(OstTraceExt2(TRACE_INTERNALS, CSQLSRVSTATEMENT_CSQLSRVSTATEMENT2_ENTRY, "Entry;0x%X;CSqlSrvStatement::~CSqlSrvStatement;iStmtHandle=0x%X", (TUint)this, (TUint)iStmtHandle));
   200 	DestroyParamBufArray();
   201 	iBufFlat.Close();
   202 	if(iStmtHandle)
   203 		{
   204 #ifdef SYMBIAN_USE_SQLITE_VERSION_3_6_4
   205 		__SQLTRACE_INTERNALSEXPR(TInt scanCount = sqlite3_stmt_status(iStmtHandle, SQLITE_STMTSTATUS_FULLSCAN_STEP, ETrue));
   206 		__SQLTRACE_INTERNALSEXPR(TInt sortCount = sqlite3_stmt_status(iStmtHandle, SQLITE_STMTSTATUS_SORT, ETrue));
   207 		SQL_TRACE_INTERNALS(OstTraceExt3(TRACE_INTERNALS, CSQLSRVSTATEMENT_CSQLSRVSTATEMENT2, "0x%X;CSqlSrvStatement::~CSqlSrvStatement;scan count=%d;sort count=%d", (TUint)this, scanCount, sortCount));
   208 #endif		
   209 		(void)sqlite3_finalize(iStmtHandle);
   210 		}
   211 	SQL_TRACE_INTERNALS(OstTrace1(TRACE_INTERNALS, CSQLSRVSTATEMENT_CSQLSRVSTATEMENT2_EXIT, "Exit;0x%X;CSqlSrvStatement::~CSqlSrvStatement", (TUint)this));
   212 	}
   213 
   214 /**
   215 Sets SQL statement parameter values.
   216 
   217 Only parameters, whose values are set by the client, will be processed.
   218 
   219 @param aParamBuf Flat buffer with parameter values.
   220 
   221 @leave KErrArgument, unknown parameter type;
   222 	   KSqlErrStmtExpired, statement handle expired.
   223                   Note that the function may also leave with some other database specific 
   224                   errors categorised as ESqlDbError.
   225 
   226 @panic SqlDb 2 In _DEBUG mode. Invalid (not created) CSqlSrvStatement object.
   227 */	
   228 void CSqlSrvStatement::BindL(const RSqlBufFlat& aParamBuf)
   229 	{
   230 	__ASSERT_DEBUG(iStmtHandle != NULL, __SQLPANIC(ESqlPanicInvalidObj));
   231 	
   232 	(void)sqlite3SymbianLastOsError();//clear last OS error
   233 
   234 	TSqlBufRIterator it;
   235 	it.Set(aParamBuf);
   236 	TInt prmIdx = 0;
   237 	
   238 	TInt err = SQLITE_OK;
   239 	while(it.Next() && err == SQLITE_OK)
   240 		{
   241 		++prmIdx;//the first SQLITE parameter index is 1
   242 		if(it.IsPresent())
   243 			{
   244 			switch(it.Type())
   245  				{
   246 				case ESqlInt:
   247 					err = sqlite3_bind_int(iStmtHandle, prmIdx, it.Int());
   248 					break;
   249 				case ESqlInt64:
   250 					err = sqlite3_bind_int64(iStmtHandle, prmIdx, it.Int64());
   251 					break;
   252 				case ESqlReal: 
   253 					err = sqlite3_bind_double(iStmtHandle, prmIdx, it.Real());
   254 					break;
   255 				case ESqlText: 
   256 					//SQLITE_STATIC is used as an argument, because the text data will be kept and can be used by the next bind call
   257 					{
   258 					TPtrC text = it.Text();
   259 					TPtrC8 prmDataCopy(reinterpret_cast <const TUint8*> (text.Ptr()), text.Length() * sizeof(TUint16));
   260 					prmDataCopy.Set(CopyAndStoreParamL(prmIdx - 1, HSqlSrvStmtParamBuf::EText16, prmDataCopy));
   261 					err = sqlite3_bind_text16(iStmtHandle, prmIdx, prmDataCopy.Ptr(), prmDataCopy.Length(), SQLITE_STATIC);
   262 					}
   263 					break;
   264 				case ESqlBinary:
   265 					//SQLITE_STATIC is used as an argument, because the blob data will be kept and can be used by the next bind call
   266 					{
   267 					TPtrC8 prmDataCopy = CopyAndStoreParamL(prmIdx - 1, HSqlSrvStmtParamBuf::EBinary, it.Binary());
   268 					err = sqlite3_bind_blob(iStmtHandle, prmIdx, prmDataCopy.Ptr(), prmDataCopy.Length(), SQLITE_STATIC);
   269 					}
   270 					break;
   271 				case ESqlNull:
   272 					err = sqlite3_bind_null(iStmtHandle, prmIdx);
   273 					break;
   274 				case ESqlZeroBlob:
   275 					err = sqlite3_bind_zeroblob(iStmtHandle, prmIdx, it.Int());
   276 					break;
   277 				default:
   278 					__SQLLEAVE(KErrArgument);//unknown parameter type
   279 					break;
   280 				}
   281 			}//end of - if(it.IsPresent())
   282 		}//end of - while(it.Next() && err == SQLITE_OK)
   283 	err = ::Sql2OsErrCode(err, sqlite3SymbianLastOsError());
   284 	__SQLLEAVE_IF_ERROR(err);
   285 	}
   286 
   287 /**
   288 Collects column names in a flat buffer and returns a reference to the buffer. 
   289 
   290 @return A const reference to a flat buffer containing the column names.
   291 
   292 @leave KErrNoMemory, an out of memory condition has occurred.
   293 
   294 @panic SqlDb 2 In _DEBUG mode. Invalid (not created) CSqlSrvStatement object.
   295 */	
   296 const RSqlBufFlat& CSqlSrvStatement::ColumnNamesL()
   297 	{
   298 	__ASSERT_DEBUG(iStmtHandle != NULL, __SQLPANIC(ESqlPanicInvalidObj));
   299 	iBufFlatType = static_cast <TSqlBufFlatType> (-1);
   300 	__SQLLEAVE_IF_ERROR(iBufFlat.SetCount(iColumnCount));
   301 	TSqlBufWIterator it;
   302 	it.Set(iBufFlat);
   303 	TInt colIdx = -1;
   304 	while(it.Next())
   305 		{
   306 		++colIdx;//the first SQLITE column index is 0
   307 		const TUint16* name = reinterpret_cast <const TUint16*> (__SQLLEAVE_IF_NULL(const_cast <void*> (sqlite3_column_name16(iStmtHandle, colIdx))));
   308 		TPtrC ptr(name, User::StringLength(name));
   309 		__SQLLEAVE_IF_ERROR(it.SetText(ptr));
   310 		}
   311 	iBufFlatType = ESqlColumnNamesBuf;
   312 	SQLPROFILER_REPORT_ALLOC(iBufFlat.MaxSize());
   313 	return iBufFlat;
   314 	}
   315 
   316 /**
   317 Collects parameter names in a flat buffer and returns a reference to the buffer. 
   318 
   319 @return A const reference to a flat buffer containing the parameter names.
   320 
   321 @leave KErrNoMemory, an out of memory condition has occurred.
   322 
   323 @panic SqlDb 2 In _DEBUG mode. Invalid (not created) CSqlSrvStatement object.
   324 */	
   325 const RSqlBufFlat& CSqlSrvStatement::ParamNamesL()
   326 	{
   327 	__ASSERT_DEBUG(iStmtHandle != NULL, __SQLPANIC(ESqlPanicInvalidObj));
   328 	iBufFlatType = static_cast <TSqlBufFlatType> (-1);
   329 	__SQLLEAVE_IF_ERROR(iBufFlat.SetCount(iParamCount));
   330 	TSqlBufWIterator it;
   331 	it.Set(iBufFlat);
   332 	TInt prmIdx = 0;
   333 	while(it.Next())
   334 		{
   335 		++prmIdx;//the first SQLITE parameter index is 1
   336 		const TUint8* name8 = reinterpret_cast <const TUint8*> (sqlite3_bind_parameter_name(iStmtHandle, prmIdx));
   337 		if(name8)
   338 			{
   339 			HBufC* name = CnvUtfConverter::ConvertToUnicodeFromUtf8L(TPtrC8(name8, User::StringLength(name8)));
   340 			TInt err = it.SetText(name->Des());
   341 			delete name;
   342 			__SQLLEAVE_IF_ERROR(err);
   343 			}
   344 		else //nameless parameter case
   345 			{
   346 			//The parameter name in this case will be formatted as "?<num>", where <num> is the parameter index.
   347 			TBuf<5> prmName;
   348 			prmName.Append(KNamelessParameter);
   349 			prmName.AppendNum((TInt64)(prmIdx - 1));
   350 			__SQLLEAVE_IF_ERROR(it.SetText(prmName));
   351 			}
   352 		}
   353 	iBufFlatType = ESqlParamNamesBuf;
   354 	SQLPROFILER_REPORT_ALLOC(iBufFlat.MaxSize());
   355 	return iBufFlat;
   356 	}
   357 	
   358 /**
   359 Collects the column values in a flat buffer and returns a reference to the buffer. 
   360 
   361 @leave KErrNoMemory, an out of memory condition has occurred.
   362 
   363 @return A const reference to a flat buffer containing the column values.
   364 
   365 @panic SqlDb 2 In _DEBUG mode. Invalid (not created) CSqlSrvStatement object
   366 */	
   367 const RSqlBufFlat& CSqlSrvStatement::ColumnValuesL()
   368 	{
   369 	__ASSERT_DEBUG(iStmtHandle != NULL, __SQLPANIC(ESqlPanicInvalidObj));
   370 	
   371 	iBufFlatType = static_cast <TSqlBufFlatType> (-1);
   372 	iBufFlat.SetCount(iColumnCount);
   373 	TSqlBufWIterator it;
   374 	it.Set(iBufFlat);
   375 	TInt colIdx = -1;
   376 	
   377 	while(it.Next())
   378 		{
   379 		++colIdx;//the first SQLITE column index is 0
   380 		TInt colType = sqlite3_column_type(iStmtHandle, colIdx);
   381 		switch(colType)
   382 			{
   383 			case SQLITE_INTEGER:
   384 				{
   385 				TInt64 val = sqlite3_column_int64(iStmtHandle, colIdx);
   386 				__SQLLEAVE_IF_ERROR(val == TInt64(TInt32(val)) ? it.SetInt(static_cast <TInt> (val)) : it.SetInt64(val));
   387 				}
   388 				break;
   389 			case SQLITE_FLOAT:
   390 				__SQLLEAVE_IF_ERROR(it.SetReal(sqlite3_column_double(iStmtHandle, colIdx)));
   391 				break;
   392 			case SQLITE_TEXT:
   393 				{
   394 				TInt charLength = (TUint)sqlite3_column_bytes16(iStmtHandle, colIdx) / sizeof(TUint16);
   395                 //"charLength == 0" - this might be an indication of an "out of memory" problem, if the column text is in UTF8 format. 
   396                 //(sqlite3_column_bytes16() may allocate memory for UTF8->UTF16 conversion)
   397 				if(charLength == 0 && sqlite3_errcode(sqlite3_db_handle(iStmtHandle)) == SQLITE_NOMEM)
   398                     {
   399                     __SQLLEAVE(KErrNoMemory);
   400                     }
   401 				if(charLength >= KSqlMaxDesLen)
   402 					{
   403 					it.SetAsNotPresent(ESqlText, charLength);
   404 					}
   405 				else
   406 					{//sqlite3_column_bytes16() already allocated the needed memory if a UTF8->UTF16 conversion
   407                      //had to be performed. The sqlite3_column_text16() on the next line is guaranteed to succeed.
   408 					const TUint16* text = reinterpret_cast <const TUint16*> (sqlite3_column_text16(iStmtHandle, colIdx));
   409 					__ASSERT_DEBUG(text != NULL, __SQLPANIC(ESqlPanicInternalError));
   410 					__SQLLEAVE_IF_ERROR(it.SetText(TPtrC16(text, charLength)));
   411 					}
   412 				}
   413 				break;
   414 			case SQLITE_BLOB:
   415 				{
   416 				TInt byteLength = sqlite3_column_bytes(iStmtHandle, colIdx);
   417 				if(byteLength >= KSqlMaxDesLen)
   418 					{
   419 					it.SetAsNotPresent(ESqlBinary, byteLength);
   420 					}
   421 				else
   422 					{
   423 					__SQLLEAVE_IF_ERROR(it.SetBinary(TPtrC8(reinterpret_cast <const TUint8*> (sqlite3_column_blob(iStmtHandle, colIdx)), byteLength)));
   424 					}
   425 				}
   426 				break;
   427 			case SQLITE_NULL:
   428 				it.SetNull();
   429 				break;
   430 			default:
   431 				__ASSERT_DEBUG(EFalse, __SQLPANIC(ESqlPanicInternalError));
   432 				break;
   433 			}//end of switch(...)
   434 		}//end of - while(it.Next())
   435 	iBufFlatType = ESqlColumnValuesBuf;
   436 	SQLPROFILER_REPORT_ALLOC(iBufFlat.MaxSize());
   437 	return iBufFlat;
   438 	}
   439 
   440 /**
   441 This method sets aColumnSource parameter to point to the column data.
   442 
   443 @param aColumnIndex Column Index, zero based. 
   444 @param aColumnSource Output parameter. It is set to point to the column data.
   445 
   446 @return KErrNone, the operation completed successfully;
   447 		KErrArgument, the refered by aColumnIndex index column is not a binary or text column.
   448 
   449 @panic SqlDb 2 In _DEBUG mode. Invalid (not created) CSqlSrvStatement object.
   450 */	
   451 TInt CSqlSrvStatement::ColumnSource(TInt aColumnIndex, TPtrC8& aColumnSource) const
   452 	{
   453 	__ASSERT_DEBUG(iStmtHandle != NULL, __SQLPANIC(ESqlPanicInvalidObj));
   454 	TInt colType = sqlite3_column_type(iStmtHandle, aColumnIndex);
   455 	if(colType == SQLITE_TEXT)
   456         {//Since the first called function after the Next() operation is always CSqlSrvStatement::ColumnValuesL(), then
   457          //sqlite3_column_bytes16() (called from  ColumnValuesL()) already allocated the needed memory if a UTF8->UTF16 
   458          //conversion had to be performed. The sqlite3_column_text16() on the next line is guaranteed to succeed.
   459 		const void* text = sqlite3_column_text16(iStmtHandle, aColumnIndex);
   460 		__ASSERT_DEBUG(text != NULL, __SQLPANIC2(ESqlPanicInternalError));
   461  		TInt length  = sqlite3_column_bytes16(iStmtHandle, aColumnIndex);
   462 		aColumnSource.Set(reinterpret_cast <const TUint8*> (text), length);
   463 		}
   464 	else if(colType == SQLITE_BLOB)
   465 		{
   466 		const void* data = sqlite3_column_blob(iStmtHandle, aColumnIndex);
   467 		TInt length  = sqlite3_column_bytes(iStmtHandle, aColumnIndex);
   468 		aColumnSource.Set(reinterpret_cast <const TUint8*> (data), length);
   469 		}
   470 	else
   471 		{
   472 		return KErrArgument;	
   473 		}
   474 	return KErrNone;
   475 	}
   476 
   477 /**
   478 Retrieves from the SQLITE library columns and parameters count.
   479 
   480 @panic SqlDb 4 In _DEBUG mode. aDbHandle is NULL.
   481 */	
   482 void CSqlSrvStatement::DoCommonConstructL()
   483 	{
   484 	__ASSERT_DEBUG(iStmtHandle != NULL, __SQLPANIC(ESqlPanicInvalidObj));
   485 	iColumnCount = sqlite3_column_count(iStmtHandle);
   486 	iParamCount = sqlite3_bind_parameter_count(iStmtHandle);
   487 	__SQLLEAVE_IF_ERROR(iBufFlat.SetCount(Max(iColumnCount, iParamCount)));
   488 	}
   489 
   490 /**
   491 Destroys the parameter buffer array (used for text or binary parameters).
   492 Before the array destruction, each array member is notified that the statement is about to be finalized.
   493 */
   494 void CSqlSrvStatement::DestroyParamBufArray()
   495 	{
   496 	TInt idx = iParamBufArray.Count();
   497 	while(--idx >= 0)
   498 		{
   499 		if(iParamBufArray[idx])
   500 			{
   501 			iParamBufArray[idx]->NotifyStatementFinalized();	
   502 			}
   503 		}
   504 	iParamBufArray.Close();
   505 	}
   506 	
   507 /**
   508 Binds a streamed text or binary parameter value.
   509 
   510 @param aParamIndex The text/binary parameter index
   511 
   512 @leave KErrNoMemory, an out of memory condition has occurred;
   513 	   KSqlErrStmtExpired, statement handle has expired.
   514                   Note that the function may also leave with some other database specific 
   515                   errors categorised as ESqlDbError.
   516 
   517 @panic SqlDb 2 In _DEBUG mode. Invalid (not created) CSqlSrvStatement object.
   518 @panic SqlDb 4 In _DEBUG mode. No parameter buffer has been created yet for this parameter.
   519 @panic SqlDb 4 In _DEBUG mode. Parameter index out of bounds.
   520 */
   521 void CSqlSrvStatement::BindParamBufL(TInt aParamIndex)
   522 	{
   523 	__ASSERT_DEBUG(iStmtHandle != NULL, __SQLPANIC(ESqlPanicInvalidObj));
   524 	__ASSERT_DEBUG(aParamIndex >= 0 && aParamIndex < sqlite3_bind_parameter_count(iStmtHandle), __SQLPANIC(ESqlPanicBadArgument));
   525 	__ASSERT_DEBUG(aParamIndex < iParamBufArray.Count(), __SQLPANIC(ESqlPanicBadArgument));
   526 	__ASSERT_DEBUG(iParamBufArray[aParamIndex] != NULL, __SQLPANIC(ESqlPanicBadArgument));
   527 	(void)sqlite3SymbianLastOsError();//clear last OS error
   528 	//Bind the parameter value.
   529 	//SQLITE_STATIC is used as an argument, because the text/blob data will be kept and can be used by the next bind call
   530 	HSqlSrvStmtParamBuf& paramBuf = *iParamBufArray[aParamIndex];
   531 	const TPtrC8 paramData(paramBuf.Data());
   532 	SQLPROFILER_REPORT_ALLOC(paramData.Length());
   533 	TInt err = KErrNone;
   534 	++aParamIndex;//SQLite uses positive parameter indexes, the SQL server - parameter indexes begin from 0
   535 	switch(paramBuf.DataType())
   536 		{
   537 		case HSqlSrvStmtParamBuf::EText16:
   538 			//sqlite3_bind_text16() expects 4-th argument to be the bytes count, not the characters count.
   539 			err = sqlite3_bind_text16(iStmtHandle, aParamIndex, paramData.Ptr(), paramData.Length(), SQLITE_STATIC);
   540 			break;
   541 		case HSqlSrvStmtParamBuf::EBinary:
   542 		default:
   543 			err = sqlite3_bind_blob(iStmtHandle, aParamIndex, paramData.Ptr(), paramData.Length(), SQLITE_STATIC);
   544 			break;
   545 		}
   546 	err = ::Sql2OsErrCode(err, sqlite3SymbianLastOsError());
   547 	__SQLLEAVE_IF_ERROR(err);
   548 	}
   549 
   550 /**
   551 @return Represents the content of the column identified by aColIdx as integer value.
   552 		If the current column type does not refer to an integer, then 
   553 		the function will do a data conversion as described in the table which can be found
   554 		in SqlDb.h file.
   555 @see RSqlStatement
   556 
   557 @panic SqlDb 2 In _DEBUG mode. Invalid (not created) CSqlSrvStatement object.
   558 @panic SqlDb 4 In _DEBUG mode. Invalid aColIdx value.
   559 */
   560 TInt CSqlSrvStatement::ColumnInt(TInt aColIdx) const
   561 	{
   562 	__ASSERT_DEBUG(iStmtHandle != NULL, __SQLPANIC(ESqlPanicInvalidObj));
   563 	__ASSERT_DEBUG((TUint)aColIdx < iColumnCount, __SQLPANIC(ESqlPanicBadArgument));
   564 	TInt colType = sqlite3_column_type(iStmtHandle, aColIdx);
   565 	switch(colType)
   566 		{
   567 		case SQLITE_FLOAT:
   568 			{
   569 			TReal roundVal;
   570 			TInt err = Math::Round(roundVal, sqlite3_column_double(iStmtHandle, aColIdx), 0);
   571 			if(err != KErrNone)
   572 				{
   573 				return KMinTInt;
   574 				}
   575 			TRealX val(roundVal);
   576 			return static_cast <TInt> (val);
   577 			}
   578 		case SQLITE_NULL:
   579 		case SQLITE_TEXT:
   580 		case SQLITE_BLOB:
   581 			return 0;
   582 		default:			//int, int64
   583 			{
   584 			TInt64 val = sqlite3_column_int64(iStmtHandle, aColIdx);
   585 			return val == (TInt)val ? (TInt)val : (val < KMinTInt ? KMinTInt : KMaxTInt);
   586 			}
   587 		}
   588 	}
   589 	
   590 /**
   591 @return Represents the content of the column identified by aColIdx as 64-bit integer value.
   592 		If the current column type does not refer to a 64-bit integer, then 
   593 		the function will do a data conversion as described in the table which can be found
   594 		in SqlDb.h file.
   595 @see RSqlStatement
   596 
   597 @panic SqlDb 2 In _DEBUG mode. Invalid (not created) CSqlSrvStatement object.
   598 @panic SqlDb 4 In _DEBUG mode. Invalid aColIdx value.
   599 */
   600 TInt64 CSqlSrvStatement::ColumnInt64(TInt aColIdx) const
   601 	{
   602 	__ASSERT_DEBUG(iStmtHandle != NULL, __SQLPANIC(ESqlPanicInvalidObj));
   603 	__ASSERT_DEBUG((TUint)aColIdx < iColumnCount, __SQLPANIC(ESqlPanicBadArgument));
   604 	TInt colType = sqlite3_column_type(iStmtHandle, aColIdx);
   605 	switch(colType)
   606 		{
   607 		case SQLITE_FLOAT:
   608 			{
   609 			TReal roundVal;
   610 			TInt err = Math::Round(roundVal, sqlite3_column_double(iStmtHandle, aColIdx), 0);
   611 			if(err != KErrNone)
   612 				{
   613 				return KMinTInt64;
   614 				}
   615 			TRealX val(roundVal);
   616 			return static_cast <TInt64> (val);
   617 			}
   618 		case SQLITE_NULL:
   619 		case SQLITE_TEXT:
   620 		case SQLITE_BLOB:
   621 			return 0;
   622 		default:			//int, int64
   623 			return sqlite3_column_int64(iStmtHandle, aColIdx);
   624 		}
   625 	}
   626 	
   627 /**
   628 @return Represents the content of the column identified by aColIdx as real value.
   629 		If the current column type does not refer to a real, then 
   630 		the function will do a data conversion as described in the table which can be found
   631 		in SqlDb.h file.
   632 @see RSqlStatement
   633 
   634 @panic SqlDb 2 In _DEBUG mode. Invalid (not created) CSqlSrvStatement object.
   635 @panic SqlDb 4 In _DEBUG mode. Invalid aColIdx value.
   636 */
   637 TReal CSqlSrvStatement::ColumnReal(TInt aColIdx) const
   638 	{
   639 	__ASSERT_DEBUG(iStmtHandle != NULL, __SQLPANIC(ESqlPanicInvalidObj));
   640 	__ASSERT_DEBUG((TUint)aColIdx < iColumnCount, __SQLPANIC(ESqlPanicBadArgument));
   641 	TInt colType = sqlite3_column_type(iStmtHandle, aColIdx);
   642 	switch(colType)
   643 		{
   644 		case SQLITE_INTEGER:
   645 			{
   646 			TRealX val(sqlite3_column_int64(iStmtHandle, aColIdx));	
   647 			return static_cast <TReal> (val);
   648 			}
   649 		case SQLITE_NULL:
   650 		case SQLITE_TEXT:
   651 		case SQLITE_BLOB:
   652 			return 0.0;
   653 		default:
   654 			return sqlite3_column_double(iStmtHandle, aColIdx);
   655 		}
   656 	}
   657 	
   658 /**
   659 Represents the content of the column identified by aColIdx as text (16 bit) descriptor.
   660 If the current column type does not refer to a text block of data, then 
   661 the function will do a data conversion as described in the table which can be found
   662 in SqlDb.h file.
   663 		
   664 @see RSqlStatement
   665 
   666 @leave KErrNoMemory, an out of memory condition has occurred.
   667 
   668 @panic SqlDb 2 In _DEBUG mode. Invalid (not created) CSqlSrvStatement object.
   669 @panic SqlDb 4 In _DEBUG mode. Invalid aColIdx value.
   670 */
   671 TPtrC CSqlSrvStatement::ColumnTextL(TInt aColIdx) const
   672 	{
   673 	__ASSERT_DEBUG(iStmtHandle != NULL, __SQLPANIC(ESqlPanicInvalidObj));
   674 	__ASSERT_DEBUG((TUint)aColIdx < iColumnCount, __SQLPANIC(ESqlPanicBadArgument));
   675 	TPtrC res;
   676 	TInt colType = sqlite3_column_type(iStmtHandle, aColIdx);
   677 	if(colType == SQLITE_TEXT)
   678 		{
   679 		TInt charLength = (TUint)sqlite3_column_bytes16(iStmtHandle, aColIdx) / sizeof(TUint16);
   680         //"charLength == 0" - this might be an indication of an "out of memory" problem, if the column text is in UTF8 format. 
   681         //(sqlite3_column_bytes16() may allocate memory for UTF8->UTF16 conversion)
   682         if(charLength == 0 && sqlite3_errcode(sqlite3_db_handle(iStmtHandle)) == SQLITE_NOMEM)
   683             {
   684             __SQLLEAVE(KErrNoMemory);
   685             }
   686         //sqlite3_column_bytes16() already allocated the needed memory if a UTF8->UTF16 conversion
   687         //had to be performed. The sqlite3_column_text16() on the next line is guaranteed to succeed.
   688         const TUint16* text = reinterpret_cast <const TUint16*> (sqlite3_column_text16(iStmtHandle, aColIdx));
   689         __ASSERT_DEBUG(text != NULL, __SQLPANIC(ESqlPanicInternalError));
   690  		res.Set(text, charLength);
   691 		}
   692 	return res;
   693 	}
   694 	
   695 /**
   696 Represents the content of the column identified by aColIdx as binary (8 bit) descriptor.
   697 If the current column type does not refer to a binary block of data, then 
   698 the function will do a data conversion as described in the table which can be found
   699 in SqlDb.h file.
   700                       
   701 @see RSqlStatement
   702 
   703 @panic SqlDb 2 In _DEBUG mode. Invalid (not created) CSqlSrvStatement object.
   704 @panic SqlDb 4 In _DEBUG mode. Invalid aColIdx value.
   705 */
   706 TPtrC8 CSqlSrvStatement::ColumnBinary(TInt aColIdx) const
   707 	{
   708 	__ASSERT_DEBUG(iStmtHandle != NULL, __SQLPANIC(ESqlPanicInvalidObj));
   709 	__ASSERT_DEBUG((TUint)aColIdx < iColumnCount, __SQLPANIC(ESqlPanicBadArgument));
   710 	TPtrC8 res;
   711 	TInt colType = sqlite3_column_type(iStmtHandle, aColIdx);
   712 	if(colType == SQLITE_BLOB)
   713 		{
   714 		TInt byteLength = sqlite3_column_bytes(iStmtHandle, aColIdx);
   715 		res.Set(reinterpret_cast <const TUint8*> (sqlite3_column_blob(iStmtHandle, aColIdx)), byteLength);
   716 		}
   717 	return res;
   718 	}
   719 
   720 /**
   721 Retrieves the declared column types using the SQLITE library storing in a 
   722 flat buffer and returns a reference to the buffer. 
   723 
   724 @return A const reference to a flat buffer containing the declared column type names.
   725 
   726 @leave KErrNoMemory, an out of memory condition has occurred;
   727 
   728 @panic SqlDb 2 In _DEBUG mode. Invalid (not created) CSqlSrvStatement object.
   729 */
   730 const RSqlBufFlat& CSqlSrvStatement::GetDeclColumnTypesL()
   731 	{
   732     __ASSERT_DEBUG(iStmtHandle != NULL, __SQLPANIC(ESqlPanicInvalidObj));
   733  	iBufFlatType = static_cast <TSqlBufFlatType> (-1);
   734 	__SQLLEAVE_IF_ERROR(iBufFlat.SetCount(iColumnCount));
   735 	TSqlBufWIterator it;
   736 	it.Set(iBufFlat);
   737 	TInt colIdx = -1;
   738 	while(it.Next())
   739 		{
   740 		++colIdx;//the first SQLITE column index is 0
   741 		const TUint16* declTypeTxt = reinterpret_cast <const TUint16*> (sqlite3_column_decltype16(iStmtHandle, colIdx));
   742 		TPtrC ptr(KNullDesC);
   743         if(declTypeTxt)
   744             {
   745             ptr.Set(declTypeTxt, User::StringLength(declTypeTxt));
   746             }
   747         else
   748             {
   749             //If sqlite3_column_decltype16() returns NULL but sqlite3_column_decltype() doesn't, then it is an "out of memory" condition 
   750             if(sqlite3_column_decltype(iStmtHandle, colIdx))
   751                 {
   752                 __SQLLEAVE(KErrNoMemory);
   753                 }
   754             }
   755 		__SQLLEAVE_IF_ERROR(it.SetText(ptr));
   756 		}
   757 	iBufFlatType = ESqlDeclColumnTypesBuf;
   758 	return iBufFlat;	
   759 	}
   760 
   761 
   762 /**
   763 Creates a new HSqlSrvStmtParamBuf object for the parameter with index "aParamIndex" or
   764 reuses the existing one.
   765 
   766 @param aParamIndex Parameter index, zero based.
   767 @param aDataType Parameter value type - binary, text8 or text16.
   768 @param aIsStreamBuf True if the param data will be retrieved from an IPC stream
   769 
   770 @return A pointer to the created HSqlSrvStmtParamBuf instance.
   771 
   772 @leave KErrNoMemory, an out of memory condition has occurred;
   773 */
   774 HSqlSrvStmtParamBuf* CSqlSrvStatement::GetParamBufL(TInt aParamIndex, HSqlSrvStmtParamBuf::TDataType aDataType, 
   775 													HSqlSrvStmtParamBuf::TBufType aBufType)
   776 	{
   777 	__ASSERT_DEBUG(aParamIndex >= 0 && aParamIndex < sqlite3_bind_parameter_count(iStmtHandle), __SQLPANIC(ESqlPanicBadArgument));
   778 	ExtendParamBufArrayL(aParamIndex);
   779 	HSqlSrvStmtParamBuf*& paramBuf = iParamBufArray[aParamIndex];
   780 	if(paramBuf)
   781 		{//Reset and reuse the existing buffer
   782 		__ASSERT_DEBUG(paramBuf->ParamIndex() == aParamIndex, __SQLPANIC(ESqlPanicInternalError));
   783 		paramBuf->Reset(aDataType, aBufType);	
   784 		}
   785 	else
   786 		{
   787 		paramBuf = HSqlSrvStmtParamBuf::NewL(*this, aParamIndex, aDataType, aBufType);
   788 		}
   789 	return paramBuf;
   790 	}
   791 
   792 /**
   793 This function will extend the iParamBufArray array (filling the new array items with NULL), if it is needed - 
   794 to ensure that there is enough place in the buffer for the parameter identified by aParamIndex.
   795 
   796 @param aParamIndex The parameter index
   797 
   798 @leave KErrNoMemory, an out of memory condition has occurred;
   799 
   800 @panic SqlDb 2 In _DEBUG mode. Invalid (not created) CSqlSrvStatement object.
   801 @panic SqlDb 4 In _DEBUG mode. Parameter index out of bounds.
   802 */
   803 void CSqlSrvStatement::ExtendParamBufArrayL(TInt aParamIndex)
   804 	{
   805 	__ASSERT_DEBUG(iStmtHandle != NULL, __SQLPANIC(ESqlPanicInvalidObj));
   806 	__ASSERT_DEBUG(aParamIndex >= 0 && aParamIndex < sqlite3_bind_parameter_count(iStmtHandle), __SQLPANIC(ESqlPanicBadArgument));
   807 	TInt ext = aParamIndex - iParamBufArray.Count() + 1;
   808 	while(ext-- > 0)
   809 		{
   810 		__SQLLEAVE_IF_ERROR(iParamBufArray.Append(NULL));
   811 		}
   812 	}
   813 
   814 /**
   815 This function will create a copy of the aParamValue and store it in the iParamBufArray array for later use.
   816 The reason: once bound, the parameter value can be used multiple times by the SQLite if it is not set explicitly again.
   817 
   818 @param aParamIndex The parameter index
   819 @param aDataType Parameter value type - binary, text8 or text16.
   820 @param aParamValue The parameter value
   821 
   822 @return 8-bit descriptor to the stored parameter value
   823 
   824 @leave KErrNoMemory, an out of memory condition has occurred;
   825                      
   826 @panic SqlDb 2 In _DEBUG mode. Invalid (not created) CSqlSrvStatement object.
   827 @panic SqlDb 4 In _DEBUG mode. Parameter index out of bounds.
   828 */
   829 TPtrC8 CSqlSrvStatement::CopyAndStoreParamL(TInt aParamIndex, HSqlSrvStmtParamBuf::TDataType aDataType, const TDesC8& aParamValue)
   830 	{
   831 	__ASSERT_DEBUG(iStmtHandle != NULL, __SQLPANIC(ESqlPanicInvalidObj));
   832 	__ASSERT_DEBUG(aParamIndex >= 0 && aParamIndex < sqlite3_bind_parameter_count(iStmtHandle), __SQLPANIC(ESqlPanicBadArgument));
   833 	HSqlSrvStmtParamBuf* paramBuf = GetParamBufL(aParamIndex, aDataType, HSqlSrvStmtParamBuf::EBufSimpleBind);
   834 	return paramBuf->SetDataL(aParamValue);
   835 	}