diff -r 000000000000 -r bde4ae8d615e os/persistentdata/persistentstorage/sql/SRC/Server/SqlSrvStatement.cpp --- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/os/persistentdata/persistentstorage/sql/SRC/Server/SqlSrvStatement.cpp Fri Jun 15 03:10:57 2012 +0200 @@ -0,0 +1,835 @@ +// Copyright (c) 2005-2010 Nokia Corporation and/or its subsidiary(-ies). +// All rights reserved. +// This component and the accompanying materials are made available +// under the terms of "Eclipse Public License v1.0" +// which accompanies this distribution, and is available +// at the URL "http://www.eclipse.org/legal/epl-v10.html". +// +// Initial Contributors: +// Nokia Corporation - initial contribution. +// +// Contributors: +// NTT DOCOMO, INC - Fix for defect 1915 "SQL server panics when using long column type strings" +// +// Description: +// + +#include //CnvUtfConverter +#include +#include "SqliteSymbian.h" //sqlite3SymbianLastOsError() +#include "sqlite3.h" +#include "SqlSrvStatement.h" +#include "SqlBufIterator.h" //TSqlBufRIterator +#include "SqlSrvResourceProfiler.h" +#include "OstTraceDefinitions.h" +#ifdef OST_TRACE_COMPILER_IN_USE +#include "SqlSrvStatementTraces.h" +#endif +#include "SqlTraceDef.h" + +////////////////////////////////////////////////////////////////////////////////////////////////////// +///////////////////////////// local const data //////////////////////////////////////////// +////////////////////////////////////////////////////////////////////////////////////////////////////// + +//This is the name prefix which will be given to the nameless parameters. +//For example, if the SQL string is: +// SELECT * FROM A WHERE ColA1 = ? AND ColA2 = ? +//then the names which will be given to the parameters will be: +//"?0" and "?1" +_LIT(KNamelessParameter, "?"); + + +///////////////////////////////////////////////////////////////////////////////////////////////////////// +////////////////// HSqlSrvStmtParamBuf ////////////////////////////////// +///////////////////////////////////////////////////////////////////////////////////////////////////////// + +/** +Destroys the parameter buffer. + +Virtual method. +*/ +HSqlSrvStmtParamBuf::~HSqlSrvStmtParamBuf() + { + delete iBuf; + } + +/** +Binds the parameter value. +The buffer can be synch-ed if: + - this is the first synch operation; + - the bound statement object is still alive (not finalized); + - the current object is alive; + - the current object data is retrieved from an IPC stream; + +If none of the conditions above is true, the synch operation is no-op. + +Virtual method. +*/ +void HSqlSrvStmtParamBuf::DoSynchL() + { + TBool dontSynch = iSynchDone || !iAlive || iStatementFinalized || iBufType != HSqlSrvStmtParamBuf::EBufIpcStream; + if(dontSynch) + { + return; + } + iSynchDone = ETrue; + TBufBuf::DoSynchL(); + iStatement.BindParamBufL(iParamIndex); + } + +/** +Destroys the HSqlSrvStmtParamBuf instance. +This method is a no-op if the statement is not finalized yet. + +Virtual method. +*/ +void HSqlSrvStmtParamBuf::DoRelease() + { + iAlive = EFalse; + if(iStatementFinalized) + {//The bound statement has been finalized - destroy the current object then. + delete this; + } + } + +/** +This function is called by the bound statement object to notify the current HSqlSrvStmtParamBuf object that the +bound statement is about to be finalized. That means, when the "stream close" operation on the client side +makes an attempt to synch the HSqlSrvStmtParamBuf object, no attempt should be made to bound the parameter data, +because the statement object is gone. +After this call the bound statement objects ceases to exist. + +Actions, performed by this method: + - if the buffer type is "simple bind", the buffer will be destroyed. No reason to keep it alive, there is no bound IPC + stream object on the client side; + - if the buffer type is an IPC stream buffer and the buffer is alive, that means: the bound statement object is about to be + finalized, but there is a bound client side IPC stream object that is still alive. In this case the buffer won't be destroyed, + but will be "told" that the bound statement is finalized, so when the client side IPC stream is closed, this object will get destroyed; + - 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 + side and it is safe to destroy the buffer; + +*/ +void HSqlSrvStmtParamBuf::NotifyStatementFinalized() + { + iStatementFinalized = ETrue; + if(iBufType == HSqlSrvStmtParamBuf::EBufSimpleBind || !iAlive) + { + DoRelease(); + } + } + +////////////////////////////////////////////////////////////////////////////////////////////////////// +///////////////////////////// CSqlSrvStatement class //////////////////////////////////////////// +////////////////////////////////////////////////////////////////////////////////////////////////////// + +/** +Creates a new CSqlSrvStatement instance. + +The created CSqlSrvStatement instance will be placed in the cleanup stack. + +@param aDbHandle The database handle +@param aSqlStmt 16-bit SQL statement, zero-terminated string +@param aColumnCount Output parameter. It will be initialized with the column count. +@param aParamCount Output parameter. It will be initialized with the parameter count. + +@return A pointer to the created CSqlSrvStatement instance. + +@leave KErrNoMemory, an out of memory condition has occurred; + KErrArgument, bad argument, for example - the SQL string contains more than one SQL statements. + Note that the function may also leave with some other database specific + errors categorised as ESqlDbError. + +@panic SqlDb 4 In _DEBUG mode if aSqlStmt is not zero-terminated string. +*/ +CSqlSrvStatement* CSqlSrvStatement::NewLC(sqlite3* aDbHandle, const TDesC16& aSqlStmt, TInt& aColumnCount, TInt& aParamCount) + { + __SQLTRACE_INTERNALSEXPR(TPtrC sqlprnptr(aSqlStmt.Left(aSqlStmt.Length() - 1))); + SQL_TRACE_INTERNALS(OstTraceExt2(TRACE_INTERNALS, CSQLSRVSTATEMENT_NEWLC_ENTRY, "Entry;0;CSqlSrvStatement::NewLC-16;aDbHandle=0x%X;aSqlStmt=%S", (TUint)aDbHandle, __SQLPRNSTR(sqlprnptr))); + __ASSERT_DEBUG(aSqlStmt.Length() > 0 ? (TInt)aSqlStmt[aSqlStmt.Length() - 1] == 0 : ETrue, __SQLPANIC2(ESqlPanicBadArgument)); + + CSqlSrvStatement* self = new (ELeave) CSqlSrvStatement; + CleanupStack::PushL(self); + self->ConstructL(aDbHandle, aSqlStmt); + aColumnCount = self->iColumnCount; + aParamCount = self->iParamCount; + 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)); + return self; + } + +/** +Creates a new CSqlSrvStatement instance. + +The created CSqlSrvStatement instance will be placed in the cleanup stack. + +@param aDbHandle The database handle +@param aSqlStmt 8-bit SQL statement, zero-terminated string +@param aColumnCount Output parameter. It will be initialized with the column count. +@param aParamCount Output parameter. It will be initialized with the parameter count. + +@return A pointer to the created CSqlSrvStatement instance. + +@leave KErrNoMemory, an out of memory condition has occurred; + KErrArgument, bad argument, for example - the SQL string contains more than one SQL statements. + Note that the function may also leave with some other database specific + errors categorised as ESqlDbError. + +@panic SqlDb 4 In _DEBUG mode if aSqlStmt is not zero-terminated string. +*/ +CSqlSrvStatement* CSqlSrvStatement::NewLC(sqlite3* aDbHandle, const TDesC8& aSqlStmt, TInt& aColumnCount, TInt& aParamCount) + { + __SQLTRACE_INTERNALSEXPR(TPtrC8 sqlprnptr(aSqlStmt.Left(aSqlStmt.Length() - 1))); + __SQLTRACE_INTERNALSVAR(TBuf<100> des16prnbuf); + SQL_TRACE_INTERNALS(OstTraceExt2(TRACE_INTERNALS, CSQLSRVSTATEMENT_NEWLC_ENTRY2, "Entry;0;CSqlSrvStatement::NewLC-8;aDbHandle=0x%X;aSqlStmt=%s", (TUint)aDbHandle, __SQLPRNSTR8(sqlprnptr, des16prnbuf))); + __ASSERT_DEBUG(aSqlStmt.Length() > 0 ? (TInt)aSqlStmt[aSqlStmt.Length() - 1] == 0 : ETrue, __SQLPANIC2(ESqlPanicBadArgument)); + + CSqlSrvStatement* self = new (ELeave) CSqlSrvStatement; + CleanupStack::PushL(self); + self->ConstructL(aDbHandle, aSqlStmt); + aColumnCount = self->iColumnCount; + aParamCount = self->iParamCount; + 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)); + return self; + } + +/** +Destroys the allocated by CSqlSrvStatement instance memory and other resources. +*/ +CSqlSrvStatement::~CSqlSrvStatement() + { + SQL_TRACE_INTERNALS(OstTraceExt2(TRACE_INTERNALS, CSQLSRVSTATEMENT_CSQLSRVSTATEMENT2_ENTRY, "Entry;0x%X;CSqlSrvStatement::~CSqlSrvStatement;iStmtHandle=0x%X", (TUint)this, (TUint)iStmtHandle)); + DestroyParamBufArray(); + iBufFlat.Close(); + if(iStmtHandle) + { +#ifdef SYMBIAN_USE_SQLITE_VERSION_3_6_4 + __SQLTRACE_INTERNALSEXPR(TInt scanCount = sqlite3_stmt_status(iStmtHandle, SQLITE_STMTSTATUS_FULLSCAN_STEP, ETrue)); + __SQLTRACE_INTERNALSEXPR(TInt sortCount = sqlite3_stmt_status(iStmtHandle, SQLITE_STMTSTATUS_SORT, ETrue)); + SQL_TRACE_INTERNALS(OstTraceExt3(TRACE_INTERNALS, CSQLSRVSTATEMENT_CSQLSRVSTATEMENT2, "0x%X;CSqlSrvStatement::~CSqlSrvStatement;scan count=%d;sort count=%d", (TUint)this, scanCount, sortCount)); +#endif + (void)sqlite3_finalize(iStmtHandle); + } + SQL_TRACE_INTERNALS(OstTrace1(TRACE_INTERNALS, CSQLSRVSTATEMENT_CSQLSRVSTATEMENT2_EXIT, "Exit;0x%X;CSqlSrvStatement::~CSqlSrvStatement", (TUint)this)); + } + +/** +Sets SQL statement parameter values. + +Only parameters, whose values are set by the client, will be processed. + +@param aParamBuf Flat buffer with parameter values. + +@leave KErrArgument, unknown parameter type; + KSqlErrStmtExpired, statement handle expired. + Note that the function may also leave with some other database specific + errors categorised as ESqlDbError. + +@panic SqlDb 2 In _DEBUG mode. Invalid (not created) CSqlSrvStatement object. +*/ +void CSqlSrvStatement::BindL(const RSqlBufFlat& aParamBuf) + { + __ASSERT_DEBUG(iStmtHandle != NULL, __SQLPANIC(ESqlPanicInvalidObj)); + + (void)sqlite3SymbianLastOsError();//clear last OS error + + TSqlBufRIterator it; + it.Set(aParamBuf); + TInt prmIdx = 0; + + TInt err = SQLITE_OK; + while(it.Next() && err == SQLITE_OK) + { + ++prmIdx;//the first SQLITE parameter index is 1 + if(it.IsPresent()) + { + switch(it.Type()) + { + case ESqlInt: + err = sqlite3_bind_int(iStmtHandle, prmIdx, it.Int()); + break; + case ESqlInt64: + err = sqlite3_bind_int64(iStmtHandle, prmIdx, it.Int64()); + break; + case ESqlReal: + err = sqlite3_bind_double(iStmtHandle, prmIdx, it.Real()); + break; + case ESqlText: + //SQLITE_STATIC is used as an argument, because the text data will be kept and can be used by the next bind call + { + TPtrC text = it.Text(); + TPtrC8 prmDataCopy(reinterpret_cast (text.Ptr()), text.Length() * sizeof(TUint16)); + prmDataCopy.Set(CopyAndStoreParamL(prmIdx - 1, HSqlSrvStmtParamBuf::EText16, prmDataCopy)); + err = sqlite3_bind_text16(iStmtHandle, prmIdx, prmDataCopy.Ptr(), prmDataCopy.Length(), SQLITE_STATIC); + } + break; + case ESqlBinary: + //SQLITE_STATIC is used as an argument, because the blob data will be kept and can be used by the next bind call + { + TPtrC8 prmDataCopy = CopyAndStoreParamL(prmIdx - 1, HSqlSrvStmtParamBuf::EBinary, it.Binary()); + err = sqlite3_bind_blob(iStmtHandle, prmIdx, prmDataCopy.Ptr(), prmDataCopy.Length(), SQLITE_STATIC); + } + break; + case ESqlNull: + err = sqlite3_bind_null(iStmtHandle, prmIdx); + break; + case ESqlZeroBlob: + err = sqlite3_bind_zeroblob(iStmtHandle, prmIdx, it.Int()); + break; + default: + __SQLLEAVE(KErrArgument);//unknown parameter type + break; + } + }//end of - if(it.IsPresent()) + }//end of - while(it.Next() && err == SQLITE_OK) + err = ::Sql2OsErrCode(err, sqlite3SymbianLastOsError()); + __SQLLEAVE_IF_ERROR(err); + } + +/** +Collects column names in a flat buffer and returns a reference to the buffer. + +@return A const reference to a flat buffer containing the column names. + +@leave KErrNoMemory, an out of memory condition has occurred. + +@panic SqlDb 2 In _DEBUG mode. Invalid (not created) CSqlSrvStatement object. +*/ +const RSqlBufFlat& CSqlSrvStatement::ColumnNamesL() + { + __ASSERT_DEBUG(iStmtHandle != NULL, __SQLPANIC(ESqlPanicInvalidObj)); + iBufFlatType = static_cast (-1); + __SQLLEAVE_IF_ERROR(iBufFlat.SetCount(iColumnCount)); + TSqlBufWIterator it; + it.Set(iBufFlat); + TInt colIdx = -1; + while(it.Next()) + { + ++colIdx;//the first SQLITE column index is 0 + const TUint16* name = reinterpret_cast (__SQLLEAVE_IF_NULL(const_cast (sqlite3_column_name16(iStmtHandle, colIdx)))); + TPtrC ptr(name, User::StringLength(name)); + __SQLLEAVE_IF_ERROR(it.SetText(ptr)); + } + iBufFlatType = ESqlColumnNamesBuf; + SQLPROFILER_REPORT_ALLOC(iBufFlat.MaxSize()); + return iBufFlat; + } + +/** +Collects parameter names in a flat buffer and returns a reference to the buffer. + +@return A const reference to a flat buffer containing the parameter names. + +@leave KErrNoMemory, an out of memory condition has occurred. + +@panic SqlDb 2 In _DEBUG mode. Invalid (not created) CSqlSrvStatement object. +*/ +const RSqlBufFlat& CSqlSrvStatement::ParamNamesL() + { + __ASSERT_DEBUG(iStmtHandle != NULL, __SQLPANIC(ESqlPanicInvalidObj)); + iBufFlatType = static_cast (-1); + __SQLLEAVE_IF_ERROR(iBufFlat.SetCount(iParamCount)); + TSqlBufWIterator it; + it.Set(iBufFlat); + TInt prmIdx = 0; + while(it.Next()) + { + ++prmIdx;//the first SQLITE parameter index is 1 + const TUint8* name8 = reinterpret_cast (sqlite3_bind_parameter_name(iStmtHandle, prmIdx)); + if(name8) + { + HBufC* name = CnvUtfConverter::ConvertToUnicodeFromUtf8L(TPtrC8(name8, User::StringLength(name8))); + TInt err = it.SetText(name->Des()); + delete name; + __SQLLEAVE_IF_ERROR(err); + } + else //nameless parameter case + { + //The parameter name in this case will be formatted as "?", where is the parameter index. + TBuf<5> prmName; + prmName.Append(KNamelessParameter); + prmName.AppendNum((TInt64)(prmIdx - 1)); + __SQLLEAVE_IF_ERROR(it.SetText(prmName)); + } + } + iBufFlatType = ESqlParamNamesBuf; + SQLPROFILER_REPORT_ALLOC(iBufFlat.MaxSize()); + return iBufFlat; + } + +/** +Collects the column values in a flat buffer and returns a reference to the buffer. + +@leave KErrNoMemory, an out of memory condition has occurred. + +@return A const reference to a flat buffer containing the column values. + +@panic SqlDb 2 In _DEBUG mode. Invalid (not created) CSqlSrvStatement object +*/ +const RSqlBufFlat& CSqlSrvStatement::ColumnValuesL() + { + __ASSERT_DEBUG(iStmtHandle != NULL, __SQLPANIC(ESqlPanicInvalidObj)); + + iBufFlatType = static_cast (-1); + iBufFlat.SetCount(iColumnCount); + TSqlBufWIterator it; + it.Set(iBufFlat); + TInt colIdx = -1; + + while(it.Next()) + { + ++colIdx;//the first SQLITE column index is 0 + TInt colType = sqlite3_column_type(iStmtHandle, colIdx); + switch(colType) + { + case SQLITE_INTEGER: + { + TInt64 val = sqlite3_column_int64(iStmtHandle, colIdx); + __SQLLEAVE_IF_ERROR(val == TInt64(TInt32(val)) ? it.SetInt(static_cast (val)) : it.SetInt64(val)); + } + break; + case SQLITE_FLOAT: + __SQLLEAVE_IF_ERROR(it.SetReal(sqlite3_column_double(iStmtHandle, colIdx))); + break; + case SQLITE_TEXT: + { + TInt charLength = (TUint)sqlite3_column_bytes16(iStmtHandle, colIdx) / sizeof(TUint16); + //"charLength == 0" - this might be an indication of an "out of memory" problem, if the column text is in UTF8 format. + //(sqlite3_column_bytes16() may allocate memory for UTF8->UTF16 conversion) + if(charLength == 0 && sqlite3_errcode(sqlite3_db_handle(iStmtHandle)) == SQLITE_NOMEM) + { + __SQLLEAVE(KErrNoMemory); + } + if(charLength >= KSqlMaxDesLen) + { + it.SetAsNotPresent(ESqlText, charLength); + } + else + {//sqlite3_column_bytes16() already allocated the needed memory if a UTF8->UTF16 conversion + //had to be performed. The sqlite3_column_text16() on the next line is guaranteed to succeed. + const TUint16* text = reinterpret_cast (sqlite3_column_text16(iStmtHandle, colIdx)); + __ASSERT_DEBUG(text != NULL, __SQLPANIC(ESqlPanicInternalError)); + __SQLLEAVE_IF_ERROR(it.SetText(TPtrC16(text, charLength))); + } + } + break; + case SQLITE_BLOB: + { + TInt byteLength = sqlite3_column_bytes(iStmtHandle, colIdx); + if(byteLength >= KSqlMaxDesLen) + { + it.SetAsNotPresent(ESqlBinary, byteLength); + } + else + { + __SQLLEAVE_IF_ERROR(it.SetBinary(TPtrC8(reinterpret_cast (sqlite3_column_blob(iStmtHandle, colIdx)), byteLength))); + } + } + break; + case SQLITE_NULL: + it.SetNull(); + break; + default: + __ASSERT_DEBUG(EFalse, __SQLPANIC(ESqlPanicInternalError)); + break; + }//end of switch(...) + }//end of - while(it.Next()) + iBufFlatType = ESqlColumnValuesBuf; + SQLPROFILER_REPORT_ALLOC(iBufFlat.MaxSize()); + return iBufFlat; + } + +/** +This method sets aColumnSource parameter to point to the column data. + +@param aColumnIndex Column Index, zero based. +@param aColumnSource Output parameter. It is set to point to the column data. + +@return KErrNone, the operation completed successfully; + KErrArgument, the refered by aColumnIndex index column is not a binary or text column. + +@panic SqlDb 2 In _DEBUG mode. Invalid (not created) CSqlSrvStatement object. +*/ +TInt CSqlSrvStatement::ColumnSource(TInt aColumnIndex, TPtrC8& aColumnSource) const + { + __ASSERT_DEBUG(iStmtHandle != NULL, __SQLPANIC(ESqlPanicInvalidObj)); + TInt colType = sqlite3_column_type(iStmtHandle, aColumnIndex); + if(colType == SQLITE_TEXT) + {//Since the first called function after the Next() operation is always CSqlSrvStatement::ColumnValuesL(), then + //sqlite3_column_bytes16() (called from ColumnValuesL()) already allocated the needed memory if a UTF8->UTF16 + //conversion had to be performed. The sqlite3_column_text16() on the next line is guaranteed to succeed. + const void* text = sqlite3_column_text16(iStmtHandle, aColumnIndex); + __ASSERT_DEBUG(text != NULL, __SQLPANIC2(ESqlPanicInternalError)); + TInt length = sqlite3_column_bytes16(iStmtHandle, aColumnIndex); + aColumnSource.Set(reinterpret_cast (text), length); + } + else if(colType == SQLITE_BLOB) + { + const void* data = sqlite3_column_blob(iStmtHandle, aColumnIndex); + TInt length = sqlite3_column_bytes(iStmtHandle, aColumnIndex); + aColumnSource.Set(reinterpret_cast (data), length); + } + else + { + return KErrArgument; + } + return KErrNone; + } + +/** +Retrieves from the SQLITE library columns and parameters count. + +@panic SqlDb 4 In _DEBUG mode. aDbHandle is NULL. +*/ +void CSqlSrvStatement::DoCommonConstructL() + { + __ASSERT_DEBUG(iStmtHandle != NULL, __SQLPANIC(ESqlPanicInvalidObj)); + iColumnCount = sqlite3_column_count(iStmtHandle); + iParamCount = sqlite3_bind_parameter_count(iStmtHandle); + __SQLLEAVE_IF_ERROR(iBufFlat.SetCount(Max(iColumnCount, iParamCount))); + } + +/** +Destroys the parameter buffer array (used for text or binary parameters). +Before the array destruction, each array member is notified that the statement is about to be finalized. +*/ +void CSqlSrvStatement::DestroyParamBufArray() + { + TInt idx = iParamBufArray.Count(); + while(--idx >= 0) + { + if(iParamBufArray[idx]) + { + iParamBufArray[idx]->NotifyStatementFinalized(); + } + } + iParamBufArray.Close(); + } + +/** +Binds a streamed text or binary parameter value. + +@param aParamIndex The text/binary parameter index + +@leave KErrNoMemory, an out of memory condition has occurred; + KSqlErrStmtExpired, statement handle has expired. + Note that the function may also leave with some other database specific + errors categorised as ESqlDbError. + +@panic SqlDb 2 In _DEBUG mode. Invalid (not created) CSqlSrvStatement object. +@panic SqlDb 4 In _DEBUG mode. No parameter buffer has been created yet for this parameter. +@panic SqlDb 4 In _DEBUG mode. Parameter index out of bounds. +*/ +void CSqlSrvStatement::BindParamBufL(TInt aParamIndex) + { + __ASSERT_DEBUG(iStmtHandle != NULL, __SQLPANIC(ESqlPanicInvalidObj)); + __ASSERT_DEBUG(aParamIndex >= 0 && aParamIndex < sqlite3_bind_parameter_count(iStmtHandle), __SQLPANIC(ESqlPanicBadArgument)); + __ASSERT_DEBUG(aParamIndex < iParamBufArray.Count(), __SQLPANIC(ESqlPanicBadArgument)); + __ASSERT_DEBUG(iParamBufArray[aParamIndex] != NULL, __SQLPANIC(ESqlPanicBadArgument)); + (void)sqlite3SymbianLastOsError();//clear last OS error + //Bind the parameter value. + //SQLITE_STATIC is used as an argument, because the text/blob data will be kept and can be used by the next bind call + HSqlSrvStmtParamBuf& paramBuf = *iParamBufArray[aParamIndex]; + const TPtrC8 paramData(paramBuf.Data()); + SQLPROFILER_REPORT_ALLOC(paramData.Length()); + TInt err = KErrNone; + ++aParamIndex;//SQLite uses positive parameter indexes, the SQL server - parameter indexes begin from 0 + switch(paramBuf.DataType()) + { + case HSqlSrvStmtParamBuf::EText16: + //sqlite3_bind_text16() expects 4-th argument to be the bytes count, not the characters count. + err = sqlite3_bind_text16(iStmtHandle, aParamIndex, paramData.Ptr(), paramData.Length(), SQLITE_STATIC); + break; + case HSqlSrvStmtParamBuf::EBinary: + default: + err = sqlite3_bind_blob(iStmtHandle, aParamIndex, paramData.Ptr(), paramData.Length(), SQLITE_STATIC); + break; + } + err = ::Sql2OsErrCode(err, sqlite3SymbianLastOsError()); + __SQLLEAVE_IF_ERROR(err); + } + +/** +@return Represents the content of the column identified by aColIdx as integer value. + If the current column type does not refer to an integer, then + the function will do a data conversion as described in the table which can be found + in SqlDb.h file. +@see RSqlStatement + +@panic SqlDb 2 In _DEBUG mode. Invalid (not created) CSqlSrvStatement object. +@panic SqlDb 4 In _DEBUG mode. Invalid aColIdx value. +*/ +TInt CSqlSrvStatement::ColumnInt(TInt aColIdx) const + { + __ASSERT_DEBUG(iStmtHandle != NULL, __SQLPANIC(ESqlPanicInvalidObj)); + __ASSERT_DEBUG((TUint)aColIdx < iColumnCount, __SQLPANIC(ESqlPanicBadArgument)); + TInt colType = sqlite3_column_type(iStmtHandle, aColIdx); + switch(colType) + { + case SQLITE_FLOAT: + { + TReal roundVal; + TInt err = Math::Round(roundVal, sqlite3_column_double(iStmtHandle, aColIdx), 0); + if(err != KErrNone) + { + return KMinTInt; + } + TRealX val(roundVal); + return static_cast (val); + } + case SQLITE_NULL: + case SQLITE_TEXT: + case SQLITE_BLOB: + return 0; + default: //int, int64 + { + TInt64 val = sqlite3_column_int64(iStmtHandle, aColIdx); + return val == (TInt)val ? (TInt)val : (val < KMinTInt ? KMinTInt : KMaxTInt); + } + } + } + +/** +@return Represents the content of the column identified by aColIdx as 64-bit integer value. + If the current column type does not refer to a 64-bit integer, then + the function will do a data conversion as described in the table which can be found + in SqlDb.h file. +@see RSqlStatement + +@panic SqlDb 2 In _DEBUG mode. Invalid (not created) CSqlSrvStatement object. +@panic SqlDb 4 In _DEBUG mode. Invalid aColIdx value. +*/ +TInt64 CSqlSrvStatement::ColumnInt64(TInt aColIdx) const + { + __ASSERT_DEBUG(iStmtHandle != NULL, __SQLPANIC(ESqlPanicInvalidObj)); + __ASSERT_DEBUG((TUint)aColIdx < iColumnCount, __SQLPANIC(ESqlPanicBadArgument)); + TInt colType = sqlite3_column_type(iStmtHandle, aColIdx); + switch(colType) + { + case SQLITE_FLOAT: + { + TReal roundVal; + TInt err = Math::Round(roundVal, sqlite3_column_double(iStmtHandle, aColIdx), 0); + if(err != KErrNone) + { + return KMinTInt64; + } + TRealX val(roundVal); + return static_cast (val); + } + case SQLITE_NULL: + case SQLITE_TEXT: + case SQLITE_BLOB: + return 0; + default: //int, int64 + return sqlite3_column_int64(iStmtHandle, aColIdx); + } + } + +/** +@return Represents the content of the column identified by aColIdx as real value. + If the current column type does not refer to a real, then + the function will do a data conversion as described in the table which can be found + in SqlDb.h file. +@see RSqlStatement + +@panic SqlDb 2 In _DEBUG mode. Invalid (not created) CSqlSrvStatement object. +@panic SqlDb 4 In _DEBUG mode. Invalid aColIdx value. +*/ +TReal CSqlSrvStatement::ColumnReal(TInt aColIdx) const + { + __ASSERT_DEBUG(iStmtHandle != NULL, __SQLPANIC(ESqlPanicInvalidObj)); + __ASSERT_DEBUG((TUint)aColIdx < iColumnCount, __SQLPANIC(ESqlPanicBadArgument)); + TInt colType = sqlite3_column_type(iStmtHandle, aColIdx); + switch(colType) + { + case SQLITE_INTEGER: + { + TRealX val(sqlite3_column_int64(iStmtHandle, aColIdx)); + return static_cast (val); + } + case SQLITE_NULL: + case SQLITE_TEXT: + case SQLITE_BLOB: + return 0.0; + default: + return sqlite3_column_double(iStmtHandle, aColIdx); + } + } + +/** +Represents the content of the column identified by aColIdx as text (16 bit) descriptor. +If the current column type does not refer to a text block of data, then +the function will do a data conversion as described in the table which can be found +in SqlDb.h file. + +@see RSqlStatement + +@leave KErrNoMemory, an out of memory condition has occurred. + +@panic SqlDb 2 In _DEBUG mode. Invalid (not created) CSqlSrvStatement object. +@panic SqlDb 4 In _DEBUG mode. Invalid aColIdx value. +*/ +TPtrC CSqlSrvStatement::ColumnTextL(TInt aColIdx) const + { + __ASSERT_DEBUG(iStmtHandle != NULL, __SQLPANIC(ESqlPanicInvalidObj)); + __ASSERT_DEBUG((TUint)aColIdx < iColumnCount, __SQLPANIC(ESqlPanicBadArgument)); + TPtrC res; + TInt colType = sqlite3_column_type(iStmtHandle, aColIdx); + if(colType == SQLITE_TEXT) + { + TInt charLength = (TUint)sqlite3_column_bytes16(iStmtHandle, aColIdx) / sizeof(TUint16); + //"charLength == 0" - this might be an indication of an "out of memory" problem, if the column text is in UTF8 format. + //(sqlite3_column_bytes16() may allocate memory for UTF8->UTF16 conversion) + if(charLength == 0 && sqlite3_errcode(sqlite3_db_handle(iStmtHandle)) == SQLITE_NOMEM) + { + __SQLLEAVE(KErrNoMemory); + } + //sqlite3_column_bytes16() already allocated the needed memory if a UTF8->UTF16 conversion + //had to be performed. The sqlite3_column_text16() on the next line is guaranteed to succeed. + const TUint16* text = reinterpret_cast (sqlite3_column_text16(iStmtHandle, aColIdx)); + __ASSERT_DEBUG(text != NULL, __SQLPANIC(ESqlPanicInternalError)); + res.Set(text, charLength); + } + return res; + } + +/** +Represents the content of the column identified by aColIdx as binary (8 bit) descriptor. +If the current column type does not refer to a binary block of data, then +the function will do a data conversion as described in the table which can be found +in SqlDb.h file. + +@see RSqlStatement + +@panic SqlDb 2 In _DEBUG mode. Invalid (not created) CSqlSrvStatement object. +@panic SqlDb 4 In _DEBUG mode. Invalid aColIdx value. +*/ +TPtrC8 CSqlSrvStatement::ColumnBinary(TInt aColIdx) const + { + __ASSERT_DEBUG(iStmtHandle != NULL, __SQLPANIC(ESqlPanicInvalidObj)); + __ASSERT_DEBUG((TUint)aColIdx < iColumnCount, __SQLPANIC(ESqlPanicBadArgument)); + TPtrC8 res; + TInt colType = sqlite3_column_type(iStmtHandle, aColIdx); + if(colType == SQLITE_BLOB) + { + TInt byteLength = sqlite3_column_bytes(iStmtHandle, aColIdx); + res.Set(reinterpret_cast (sqlite3_column_blob(iStmtHandle, aColIdx)), byteLength); + } + return res; + } + +/** +Retrieves the declared column types using the SQLITE library storing in a +flat buffer and returns a reference to the buffer. + +@return A const reference to a flat buffer containing the declared column type names. + +@leave KErrNoMemory, an out of memory condition has occurred; + +@panic SqlDb 2 In _DEBUG mode. Invalid (not created) CSqlSrvStatement object. +*/ +const RSqlBufFlat& CSqlSrvStatement::GetDeclColumnTypesL() + { + __ASSERT_DEBUG(iStmtHandle != NULL, __SQLPANIC(ESqlPanicInvalidObj)); + iBufFlatType = static_cast (-1); + __SQLLEAVE_IF_ERROR(iBufFlat.SetCount(iColumnCount)); + TSqlBufWIterator it; + it.Set(iBufFlat); + TInt colIdx = -1; + while(it.Next()) + { + ++colIdx;//the first SQLITE column index is 0 + const TUint16* declTypeTxt = reinterpret_cast (sqlite3_column_decltype16(iStmtHandle, colIdx)); + TPtrC ptr(KNullDesC); + if(declTypeTxt) + { + ptr.Set(declTypeTxt, User::StringLength(declTypeTxt)); + } + else + { + //If sqlite3_column_decltype16() returns NULL but sqlite3_column_decltype() doesn't, then it is an "out of memory" condition + if(sqlite3_column_decltype(iStmtHandle, colIdx)) + { + __SQLLEAVE(KErrNoMemory); + } + } + __SQLLEAVE_IF_ERROR(it.SetText(ptr)); + } + iBufFlatType = ESqlDeclColumnTypesBuf; + return iBufFlat; + } + + +/** +Creates a new HSqlSrvStmtParamBuf object for the parameter with index "aParamIndex" or +reuses the existing one. + +@param aParamIndex Parameter index, zero based. +@param aDataType Parameter value type - binary, text8 or text16. +@param aIsStreamBuf True if the param data will be retrieved from an IPC stream + +@return A pointer to the created HSqlSrvStmtParamBuf instance. + +@leave KErrNoMemory, an out of memory condition has occurred; +*/ +HSqlSrvStmtParamBuf* CSqlSrvStatement::GetParamBufL(TInt aParamIndex, HSqlSrvStmtParamBuf::TDataType aDataType, + HSqlSrvStmtParamBuf::TBufType aBufType) + { + __ASSERT_DEBUG(aParamIndex >= 0 && aParamIndex < sqlite3_bind_parameter_count(iStmtHandle), __SQLPANIC(ESqlPanicBadArgument)); + ExtendParamBufArrayL(aParamIndex); + HSqlSrvStmtParamBuf*& paramBuf = iParamBufArray[aParamIndex]; + if(paramBuf) + {//Reset and reuse the existing buffer + __ASSERT_DEBUG(paramBuf->ParamIndex() == aParamIndex, __SQLPANIC(ESqlPanicInternalError)); + paramBuf->Reset(aDataType, aBufType); + } + else + { + paramBuf = HSqlSrvStmtParamBuf::NewL(*this, aParamIndex, aDataType, aBufType); + } + return paramBuf; + } + +/** +This function will extend the iParamBufArray array (filling the new array items with NULL), if it is needed - +to ensure that there is enough place in the buffer for the parameter identified by aParamIndex. + +@param aParamIndex The parameter index + +@leave KErrNoMemory, an out of memory condition has occurred; + +@panic SqlDb 2 In _DEBUG mode. Invalid (not created) CSqlSrvStatement object. +@panic SqlDb 4 In _DEBUG mode. Parameter index out of bounds. +*/ +void CSqlSrvStatement::ExtendParamBufArrayL(TInt aParamIndex) + { + __ASSERT_DEBUG(iStmtHandle != NULL, __SQLPANIC(ESqlPanicInvalidObj)); + __ASSERT_DEBUG(aParamIndex >= 0 && aParamIndex < sqlite3_bind_parameter_count(iStmtHandle), __SQLPANIC(ESqlPanicBadArgument)); + TInt ext = aParamIndex - iParamBufArray.Count() + 1; + while(ext-- > 0) + { + __SQLLEAVE_IF_ERROR(iParamBufArray.Append(NULL)); + } + } + +/** +This function will create a copy of the aParamValue and store it in the iParamBufArray array for later use. +The reason: once bound, the parameter value can be used multiple times by the SQLite if it is not set explicitly again. + +@param aParamIndex The parameter index +@param aDataType Parameter value type - binary, text8 or text16. +@param aParamValue The parameter value + +@return 8-bit descriptor to the stored parameter value + +@leave KErrNoMemory, an out of memory condition has occurred; + +@panic SqlDb 2 In _DEBUG mode. Invalid (not created) CSqlSrvStatement object. +@panic SqlDb 4 In _DEBUG mode. Parameter index out of bounds. +*/ +TPtrC8 CSqlSrvStatement::CopyAndStoreParamL(TInt aParamIndex, HSqlSrvStmtParamBuf::TDataType aDataType, const TDesC8& aParamValue) + { + __ASSERT_DEBUG(iStmtHandle != NULL, __SQLPANIC(ESqlPanicInvalidObj)); + __ASSERT_DEBUG(aParamIndex >= 0 && aParamIndex < sqlite3_bind_parameter_count(iStmtHandle), __SQLPANIC(ESqlPanicBadArgument)); + HSqlSrvStmtParamBuf* paramBuf = GetParamBufL(aParamIndex, aDataType, HSqlSrvStmtParamBuf::EBufSimpleBind); + return paramBuf->SetDataL(aParamValue); + }