First public contribution.
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".
8 // Initial Contributors:
9 // Nokia Corporation - initial contribution.
12 // NTT DOCOMO, INC - Fix for defect 1915 "SQL server panics when using long column type strings"
17 #ifndef __SQLSRVSTATEMENT_H__
18 #define __SQLSRVSTATEMENT_H__
21 #include <s32mem.h> //TBufBuf
22 #include "SqlUtil.h" //Sql2OsErrCode()
23 #include "SqlBufFlat.h" //RSqlBufFlat
24 #include "SqlSrvStatementUtil.h" //Global SQL statement related functions
26 //Forward declarations
30 class CSqlSrvStatement;
32 /////////////////////////////////////////////////////////////////////////////////////////////////////////
33 ////////////////////// HSqlSrvStmtParamBuf //////////////////////////////////
34 /////////////////////////////////////////////////////////////////////////////////////////////////////////
37 Buffer for storing binary/text parameter values (an input from client side write stream objects or bound text/binary parameters).
38 Every CSqlSrvStatement object maintains a collection of HSqlSrvStmtParamBuf objects.
42 - The life-time of the HSqlSrvStmtParamBuf objects is controlled by a set of flags:
43 = iStatementFinalized - False by default, set to true if the bound statement has been finalized.
44 When DoRelease() is called, the method checks this poreperty and if the statement has been finalized,
45 then obviously no more parameter binding operations can be performed and it is safe to destroy the
46 current HSqlSrvStmtParamBuf object. Otherwise the HSqlSrvStmtParamBuf object won't be destroyed,
47 it will be marked as "dead" - see the "iAlive" property;
48 = iAlive - True by default, set to false by the DoRelease() call, if the object cannot be destroyed because
49 the bound statement is not finalized yet;
50 = iBufType - EBufIpcStream if the current HSqlSrvStmtParamBuf object is an IPC stream buffer, EBufSimpleBind
51 if the object holds a parameter value, that has been bound not using an IPC stream. This property
52 is checked when the bound statement gets finalized - inside the NotifyStatementFinalized() function.
53 If the current object is not an IPC stream buffer, then it is safe to destroy the object because
54 the bound statement is finalized;
56 All these checks allow keeping the buffer object alive when it is needed and code like the code bellow should work without problems:
58 RSqlStatement stmt; //Prepare a statement with a binary/text parameter
60 RSqlParamWriteStream strm;
61 strm.BindBinary(stmt,...); //Use a stream object to bind the parameter value
63 strm.CommitL(); //The stream commit call will bind the streamed parameter value to the statement
64 strm.Close(); //The stream close operation should not destroy the HSqlSrvStmtParamBuf object, because the parameter
65 //value is kept in the buffer, and when that value was bound to the statement, the
66 //SQLITE_STATIC constant was used, which means - SQLite won't make a copy of the parameter value,
67 //SQLite will keep a pointer to the parameter value and use it for the Exec() operation.
68 stmt.Exec(); //At this point the SQLite library should be able to get the parameter data and use it.
71 - Another reason for keeping the HSqlSrvStmtParamBuf object alive during the life-time of the CSqlSrvStatement object is that
72 the RSqlStatement::Reset() call should not destroy the buffer. The client should be able to resuse
73 the buffered parameter data for the next RSqlStatement::Exec() call. For example:
75 RSqlStatement stmt; //Prepare a statement with two binary parameters
77 RSqlParamWriteStream strm1, strm2;
79 strm1.BindBinary(stmt,...); //Use a stream object to bind the 1st parameter value
84 strm2.BindBinary(stmt,...); //Use a stream object to bind the 2nd parameter value
89 stmt.Exec(); //Do the operation using the bound parameter values
90 stmt.Reset(); //This call should not destroy the HSqlSrvStmtParamBuf objects that keep the parameter values
92 strm2.BindBinary(stmt,...); //Use a stream object to bind the 2nd parameter value
97 stmt.Exec(); //Do the operation using the bound parameter values.
98 //The operation should be able to use the old 1st parameter value without any problems,
99 //and also the new 2nd parameter value.
102 - The HSqlSrvStmtParamBuf object can be synch-ed just once. The reason for that is to prevent the double commit-s,
103 that will occur as a result of MStreamBuf::Close() calls on the client side, where the Close() method will try to sync the buffer.
104 The single sync will prevent the SQL server from crashing, if the order of the RSqlStatement::Close() and
105 RSqlParamWriteStream::Close() calls is reversed, e.g.:
109 RSqlParamWriteStream strm;
110 strm.BindBinary(stmt,...);
118 - For BC reasons, even though the HSqlSrvStmtParamBuf object can be synch-ed just once (the operation is controlled by the iSynchDone
119 flag), the client side still will be able to update the buffer content, because the related HIpcStream stream object keeps a copy
120 of the buffer's base address. Since the parameter is bound using the SQLITE_STATIC constant (SQLite won't make a copy of the
121 parameter value), at the moment of the Exec() call, SQLite will use the give pointer to the parameter value and will pick up
122 the updated buffer even without a commit call;
126 - The iSynchDone flag controls the number of buffer commit operations. It is false by default,
127 set to true by the DoSynchL() call. The HSqlSrvStmtParamBuf object can be synch-ed just once, because DoSynchL()
128 calls sqlite3_bind_text/binary() and the expectation is that the whole buffer with the parameter data is ready for binding;
129 If the buffer is not commited by calling DoSynchL(), the buffer still will be commited by the DoSynchL() call performed from
130 MStreamBuf::Close(). But in a case of a sync failure, no error will be reported to the client side;
133 @see CSqlSrvStatement
137 NONSHARABLE_CLASS(HSqlSrvStmtParamBuf) : public TBufBuf
140 enum TDataType {EBinary, EText16};
141 enum TBufType {EBufSimpleBind, EBufIpcStream};
143 static inline HSqlSrvStmtParamBuf* NewL(CSqlSrvStatement& aStatement, TInt aParamIndex, HSqlSrvStmtParamBuf::TDataType aDataType,
144 HSqlSrvStmtParamBuf::TBufType aBufType);
145 inline void Reset(HSqlSrvStmtParamBuf::TDataType aDataType, HSqlSrvStmtParamBuf::TBufType aBufType);
147 inline const TPtrC8 SetDataL(const TDesC8& aData);
148 inline const TPtrC8 Data() const;
149 inline HSqlSrvStmtParamBuf::TDataType DataType() const;
151 void NotifyStatementFinalized();
152 inline TInt ParamIndex() const;
155 inline HSqlSrvStmtParamBuf(CSqlSrvStatement& aStatement, TInt aParamIndex, HSqlSrvStmtParamBuf::TDataType aDataType,
156 HSqlSrvStmtParamBuf::TBufType aBufType);
157 virtual ~HSqlSrvStmtParamBuf();
158 inline void ConstructL();
159 virtual void DoSynchL();
160 virtual void DoRelease();
163 enum {EExpandSize = 512};
165 CSqlSrvStatement& iStatement; //The bound CSqlSrvStatement object
166 CBufFlat* iBuf; //Parameter buffer - not owned by HSqlSrvStmtParamBuf
167 TInt iParamIndex; //The parameter index
168 TBool iStatementFinalized;//True if the bound statement object has been finalized
169 TBool iAlive; //True if DoRelease() has not been called yet
170 HSqlSrvStmtParamBuf::TDataType iDataType;//The parameter type
171 HSqlSrvStmtParamBuf::TBufType iBufType; //IPC stream buf or a simple "bind param" buf
172 TBool iSynchDone; //True if the buffer data has been bound to the statement
175 /////////////////////////////////////////////////////////////////////////////////////////////////////////
176 ////////////////// CSqlSrvStatement //////////////////////////////////
177 /////////////////////////////////////////////////////////////////////////////////////////////////////////
180 SQL statement handle.
182 CSqlSrvStatement is a server side class which processes the client side requests for:
183 - preparing 16-bit and 8-bit SQL statements;
184 - executing prepared SQL statement;
185 - retrieving parameter names and column names;
186 - binding SQL parameters;
187 - retrieving column values;
188 - reseting the prepared SQL statement;
189 - moving the cursor to the next record;
191 @see CSqlSrvStatement::NewLC()
192 @see CSqlSrvStatement::NewLC()
193 @see CSqlSrvStatement::BindL()
194 @see CSqlSrvStatement::Next()
195 @see CSqlSrvStatement::Reset()
196 @see CSqlSrvStatement::Exec()
197 @see CSqlSrvStatement::ColumnNamesL()
198 @see CSqlSrvStatement::ParamNamesL()
199 @see CSqlSrvStatement::ColumnValuesL()
200 @see CSqlSrvStatement::ColumnSource()
202 @see HSqlSrvStmtParamBuf
206 NONSHARABLE_CLASS(CSqlSrvStatement) : public CBase
209 static CSqlSrvStatement* NewLC(sqlite3* aDbHandle, const TDesC16& aSqlStmt, TInt& aColumnCount, TInt& aParamCount);
210 static CSqlSrvStatement* NewLC(sqlite3* aDbHandle, const TDesC8& aSqlStmt, TInt& aColumnCount, TInt& aParamCount);
211 virtual ~CSqlSrvStatement();
213 void BindL(const RSqlBufFlat& aParamBuf);
218 inline const RSqlBufFlat& BufFlatL(TSqlBufFlatType aWhat) const;
219 const RSqlBufFlat& ColumnNamesL();
220 const RSqlBufFlat& ParamNamesL();
221 const RSqlBufFlat& ColumnValuesL();
222 TInt ColumnSource(TInt aColumnIndex, TPtrC8& aColumnSource) const;
224 TInt ColumnInt(TInt aColIdx) const;
225 TInt64 ColumnInt64(TInt aColIdx) const;
226 TReal ColumnReal(TInt aColIdx) const;
227 TPtrC ColumnTextL(TInt aColIdx) const;
228 TPtrC8 ColumnBinary(TInt aColIdx) const;
230 const RSqlBufFlat& GetDeclColumnTypesL();
232 HSqlSrvStmtParamBuf* GetParamBufL(TInt aParamIndex, HSqlSrvStmtParamBuf::TDataType aDataType, HSqlSrvStmtParamBuf::TBufType aBufType);
233 void BindParamBufL(TInt aParamIndex);
236 inline CSqlSrvStatement();
237 inline void ConstructL(sqlite3* aDbHandle, const TDesC16& aSqlStmt);
238 inline void ConstructL(sqlite3* aDbHandle, const TDesC8& aSqlStmt);
239 void DoCommonConstructL();
240 void DestroyParamBufArray();
241 void ExtendParamBufArrayL(TInt aParamIndex);
242 TPtrC8 CopyAndStoreParamL(TInt aParamIndex, HSqlSrvStmtParamBuf::TDataType aDataType, const TDesC8& aParamValue);
245 sqlite3_stmt* iStmtHandle; //SQL statement handle
248 TSqlBufFlatType iBufFlatType; //What is in iFlatBuf data member
249 RSqlBufFlat iBufFlat; //Flat buffer used for: column names, parameter names, column values
250 RArray<HSqlSrvStmtParamBuf*> iParamBufArray;//An array with pointers to HSqlSrvStmtParamBuf buffers for the text/binary parameters
254 #include "SqlSrvStatement.inl"
256 #endif//__SQLSRVSTATEMENT_H__