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