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: #ifndef __SQLSRVSTATEMENT_H__ sl@0: #define __SQLSRVSTATEMENT_H__ sl@0: sl@0: #include sl@0: #include //TBufBuf sl@0: #include "SqlUtil.h" //Sql2OsErrCode() sl@0: #include "SqlBufFlat.h" //RSqlBufFlat sl@0: #include "SqlSrvStatementUtil.h" //Global SQL statement related functions sl@0: sl@0: //Forward declarations sl@0: struct sqlite3; sl@0: struct sqlite3_stmt; sl@0: class RWriteStream; sl@0: class CSqlSrvStatement; sl@0: sl@0: ///////////////////////////////////////////////////////////////////////////////////////////////////////// sl@0: ////////////////////// HSqlSrvStmtParamBuf ////////////////////////////////// sl@0: ///////////////////////////////////////////////////////////////////////////////////////////////////////// sl@0: sl@0: /** sl@0: Buffer for storing binary/text parameter values (an input from client side write stream objects or bound text/binary parameters). sl@0: Every CSqlSrvStatement object maintains a collection of HSqlSrvStmtParamBuf objects. sl@0: sl@0: Please, note that: sl@0: sl@0: - The life-time of the HSqlSrvStmtParamBuf objects is controlled by a set of flags: sl@0: = iStatementFinalized - False by default, set to true if the bound statement has been finalized. sl@0: When DoRelease() is called, the method checks this poreperty and if the statement has been finalized, sl@0: then obviously no more parameter binding operations can be performed and it is safe to destroy the sl@0: current HSqlSrvStmtParamBuf object. Otherwise the HSqlSrvStmtParamBuf object won't be destroyed, sl@0: it will be marked as "dead" - see the "iAlive" property; sl@0: = iAlive - True by default, set to false by the DoRelease() call, if the object cannot be destroyed because sl@0: the bound statement is not finalized yet; sl@0: = iBufType - EBufIpcStream if the current HSqlSrvStmtParamBuf object is an IPC stream buffer, EBufSimpleBind sl@0: if the object holds a parameter value, that has been bound not using an IPC stream. This property sl@0: is checked when the bound statement gets finalized - inside the NotifyStatementFinalized() function. sl@0: If the current object is not an IPC stream buffer, then it is safe to destroy the object because sl@0: the bound statement is finalized; sl@0: sl@0: All these checks allow keeping the buffer object alive when it is needed and code like the code bellow should work without problems: sl@0: @code sl@0: RSqlStatement stmt; //Prepare a statement with a binary/text parameter sl@0: .... sl@0: RSqlParamWriteStream strm; sl@0: strm.BindBinary(stmt,...); //Use a stream object to bind the parameter value sl@0: strm.WriteL(...); sl@0: strm.CommitL(); //The stream commit call will bind the streamed parameter value to the statement sl@0: strm.Close(); //The stream close operation should not destroy the HSqlSrvStmtParamBuf object, because the parameter sl@0: //value is kept in the buffer, and when that value was bound to the statement, the sl@0: //SQLITE_STATIC constant was used, which means - SQLite won't make a copy of the parameter value, sl@0: //SQLite will keep a pointer to the parameter value and use it for the Exec() operation. sl@0: stmt.Exec(); //At this point the SQLite library should be able to get the parameter data and use it. sl@0: @endcode sl@0: sl@0: - Another reason for keeping the HSqlSrvStmtParamBuf object alive during the life-time of the CSqlSrvStatement object is that sl@0: the RSqlStatement::Reset() call should not destroy the buffer. The client should be able to resuse sl@0: the buffered parameter data for the next RSqlStatement::Exec() call. For example: sl@0: @code sl@0: RSqlStatement stmt; //Prepare a statement with two binary parameters sl@0: .... sl@0: RSqlParamWriteStream strm1, strm2; sl@0: sl@0: strm1.BindBinary(stmt,...); //Use a stream object to bind the 1st parameter value sl@0: strm1.WriteL(...); sl@0: strm1.CommitL(); sl@0: strm1.Close(); sl@0: sl@0: strm2.BindBinary(stmt,...); //Use a stream object to bind the 2nd parameter value sl@0: strm2.WriteL(...); sl@0: strm2.CommitL(); sl@0: strm2.Close(); sl@0: sl@0: stmt.Exec(); //Do the operation using the bound parameter values sl@0: stmt.Reset(); //This call should not destroy the HSqlSrvStmtParamBuf objects that keep the parameter values sl@0: sl@0: strm2.BindBinary(stmt,...); //Use a stream object to bind the 2nd parameter value sl@0: strm2.WriteL(...); sl@0: strm2.CommitL(); sl@0: strm2.Close(); sl@0: sl@0: stmt.Exec(); //Do the operation using the bound parameter values. sl@0: //The operation should be able to use the old 1st parameter value without any problems, sl@0: //and also the new 2nd parameter value. sl@0: @endcode sl@0: sl@0: - The HSqlSrvStmtParamBuf object can be synch-ed just once. The reason for that is to prevent the double commit-s, sl@0: that will occur as a result of MStreamBuf::Close() calls on the client side, where the Close() method will try to sync the buffer. sl@0: The single sync will prevent the SQL server from crashing, if the order of the RSqlStatement::Close() and sl@0: RSqlParamWriteStream::Close() calls is reversed, e.g.: sl@0: @code sl@0: RSqlStatement stmt; sl@0: .... sl@0: RSqlParamWriteStream strm; sl@0: strm.BindBinary(stmt,...); sl@0: strm.WriteL(...); sl@0: strm.CommitL(); sl@0: stmt.Exec(); sl@0: stmt.Close(); sl@0: strm.Close(); sl@0: @endcode sl@0: sl@0: - For BC reasons, even though the HSqlSrvStmtParamBuf object can be synch-ed just once (the operation is controlled by the iSynchDone sl@0: flag), the client side still will be able to update the buffer content, because the related HIpcStream stream object keeps a copy sl@0: of the buffer's base address. Since the parameter is bound using the SQLITE_STATIC constant (SQLite won't make a copy of the sl@0: parameter value), at the moment of the Exec() call, SQLite will use the give pointer to the parameter value and will pick up sl@0: the updated buffer even without a commit call; sl@0: sl@0: @endcode sl@0: sl@0: - The iSynchDone flag controls the number of buffer commit operations. It is false by default, sl@0: set to true by the DoSynchL() call. The HSqlSrvStmtParamBuf object can be synch-ed just once, because DoSynchL() sl@0: calls sqlite3_bind_text/binary() and the expectation is that the whole buffer with the parameter data is ready for binding; sl@0: If the buffer is not commited by calling DoSynchL(), the buffer still will be commited by the DoSynchL() call performed from sl@0: MStreamBuf::Close(). But in a case of a sync failure, no error will be reported to the client side; sl@0: sl@0: @see HIpcStream sl@0: @see CSqlSrvStatement sl@0: sl@0: @internalComponent sl@0: */ sl@0: NONSHARABLE_CLASS(HSqlSrvStmtParamBuf) : public TBufBuf sl@0: { sl@0: public: sl@0: enum TDataType {EBinary, EText16}; sl@0: enum TBufType {EBufSimpleBind, EBufIpcStream}; sl@0: sl@0: static inline HSqlSrvStmtParamBuf* NewL(CSqlSrvStatement& aStatement, TInt aParamIndex, HSqlSrvStmtParamBuf::TDataType aDataType, sl@0: HSqlSrvStmtParamBuf::TBufType aBufType); sl@0: inline void Reset(HSqlSrvStmtParamBuf::TDataType aDataType, HSqlSrvStmtParamBuf::TBufType aBufType); sl@0: sl@0: inline const TPtrC8 SetDataL(const TDesC8& aData); sl@0: inline const TPtrC8 Data() const; sl@0: inline HSqlSrvStmtParamBuf::TDataType DataType() const; sl@0: sl@0: void NotifyStatementFinalized(); sl@0: inline TInt ParamIndex() const; sl@0: sl@0: private: sl@0: inline HSqlSrvStmtParamBuf(CSqlSrvStatement& aStatement, TInt aParamIndex, HSqlSrvStmtParamBuf::TDataType aDataType, sl@0: HSqlSrvStmtParamBuf::TBufType aBufType); sl@0: virtual ~HSqlSrvStmtParamBuf(); sl@0: inline void ConstructL(); sl@0: virtual void DoSynchL(); sl@0: virtual void DoRelease(); sl@0: sl@0: private: sl@0: enum {EExpandSize = 512}; sl@0: sl@0: CSqlSrvStatement& iStatement; //The bound CSqlSrvStatement object sl@0: CBufFlat* iBuf; //Parameter buffer - not owned by HSqlSrvStmtParamBuf sl@0: TInt iParamIndex; //The parameter index sl@0: TBool iStatementFinalized;//True if the bound statement object has been finalized sl@0: TBool iAlive; //True if DoRelease() has not been called yet sl@0: HSqlSrvStmtParamBuf::TDataType iDataType;//The parameter type sl@0: HSqlSrvStmtParamBuf::TBufType iBufType; //IPC stream buf or a simple "bind param" buf sl@0: TBool iSynchDone; //True if the buffer data has been bound to the statement sl@0: }; sl@0: sl@0: ///////////////////////////////////////////////////////////////////////////////////////////////////////// sl@0: ////////////////// CSqlSrvStatement ////////////////////////////////// sl@0: ///////////////////////////////////////////////////////////////////////////////////////////////////////// sl@0: sl@0: /** sl@0: SQL statement handle. sl@0: sl@0: CSqlSrvStatement is a server side class which processes the client side requests for: sl@0: - preparing 16-bit and 8-bit SQL statements; sl@0: - executing prepared SQL statement; sl@0: - retrieving parameter names and column names; sl@0: - binding SQL parameters; sl@0: - retrieving column values; sl@0: - reseting the prepared SQL statement; sl@0: - moving the cursor to the next record; sl@0: sl@0: @see CSqlSrvStatement::NewLC() sl@0: @see CSqlSrvStatement::NewLC() sl@0: @see CSqlSrvStatement::BindL() sl@0: @see CSqlSrvStatement::Next() sl@0: @see CSqlSrvStatement::Reset() sl@0: @see CSqlSrvStatement::Exec() sl@0: @see CSqlSrvStatement::ColumnNamesL() sl@0: @see CSqlSrvStatement::ParamNamesL() sl@0: @see CSqlSrvStatement::ColumnValuesL() sl@0: @see CSqlSrvStatement::ColumnSource() sl@0: sl@0: @see HSqlSrvStmtParamBuf sl@0: sl@0: @internalComponent sl@0: */ sl@0: NONSHARABLE_CLASS(CSqlSrvStatement) : public CBase sl@0: { sl@0: public: sl@0: static CSqlSrvStatement* NewLC(sqlite3* aDbHandle, const TDesC16& aSqlStmt, TInt& aColumnCount, TInt& aParamCount); sl@0: static CSqlSrvStatement* NewLC(sqlite3* aDbHandle, const TDesC8& aSqlStmt, TInt& aColumnCount, TInt& aParamCount); sl@0: virtual ~CSqlSrvStatement(); sl@0: sl@0: void BindL(const RSqlBufFlat& aParamBuf); sl@0: inline TInt Next(); sl@0: inline TInt Reset(); sl@0: inline TInt Exec(); sl@0: sl@0: inline const RSqlBufFlat& BufFlatL(TSqlBufFlatType aWhat) const; sl@0: const RSqlBufFlat& ColumnNamesL(); sl@0: const RSqlBufFlat& ParamNamesL(); sl@0: const RSqlBufFlat& ColumnValuesL(); sl@0: TInt ColumnSource(TInt aColumnIndex, TPtrC8& aColumnSource) const; sl@0: sl@0: TInt ColumnInt(TInt aColIdx) const; sl@0: TInt64 ColumnInt64(TInt aColIdx) const; sl@0: TReal ColumnReal(TInt aColIdx) const; sl@0: TPtrC ColumnTextL(TInt aColIdx) const; sl@0: TPtrC8 ColumnBinary(TInt aColIdx) const; sl@0: sl@0: const RSqlBufFlat& GetDeclColumnTypesL(); sl@0: sl@0: HSqlSrvStmtParamBuf* GetParamBufL(TInt aParamIndex, HSqlSrvStmtParamBuf::TDataType aDataType, HSqlSrvStmtParamBuf::TBufType aBufType); sl@0: void BindParamBufL(TInt aParamIndex); sl@0: sl@0: private: sl@0: inline CSqlSrvStatement(); sl@0: inline void ConstructL(sqlite3* aDbHandle, const TDesC16& aSqlStmt); sl@0: inline void ConstructL(sqlite3* aDbHandle, const TDesC8& aSqlStmt); sl@0: void DoCommonConstructL(); sl@0: void DestroyParamBufArray(); sl@0: void ExtendParamBufArrayL(TInt aParamIndex); sl@0: TPtrC8 CopyAndStoreParamL(TInt aParamIndex, HSqlSrvStmtParamBuf::TDataType aDataType, const TDesC8& aParamValue); sl@0: sl@0: private: sl@0: sqlite3_stmt* iStmtHandle; //SQL statement handle sl@0: TInt iColumnCount; sl@0: TInt iParamCount; sl@0: TSqlBufFlatType iBufFlatType; //What is in iFlatBuf data member sl@0: RSqlBufFlat iBufFlat; //Flat buffer used for: column names, parameter names, column values sl@0: RArray iParamBufArray;//An array with pointers to HSqlSrvStmtParamBuf buffers for the text/binary parameters sl@0: sl@0: }; sl@0: sl@0: #include "SqlSrvStatement.inl" sl@0: sl@0: #endif//__SQLSRVSTATEMENT_H__