os/persistentdata/persistentstorage/sql/SRC/Server/SqlSrvStatement.h
author sl@SLION-WIN7.fritz.box
Fri, 15 Jun 2012 03:10:57 +0200
changeset 0 bde4ae8d615e
permissions -rw-r--r--
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".
     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 #ifndef __SQLSRVSTATEMENT_H__
    18 #define __SQLSRVSTATEMENT_H__
    19 
    20 #include <e32base.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
    25 
    26 //Forward declarations
    27 struct sqlite3;
    28 struct sqlite3_stmt;
    29 class RWriteStream;
    30 class CSqlSrvStatement;
    31 
    32 /////////////////////////////////////////////////////////////////////////////////////////////////////////
    33 //////////////////////                HSqlSrvStmtParamBuf              //////////////////////////////////
    34 /////////////////////////////////////////////////////////////////////////////////////////////////////////
    35 
    36 /**
    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.
    39 
    40 Please, note that:
    41 
    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;
    55    
    56    All these checks allow keeping the buffer object alive when it is needed and code like the code bellow should work without problems:
    57 @code
    58 	RSqlStatement stmt;			//Prepare a statement with a binary/text parameter
    59 	....
    60 	RSqlParamWriteStream strm;
    61 	strm.BindBinary(stmt,...);	//Use a stream object to bind the parameter value
    62 	strm.WriteL(...);
    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.
    69 @endcode
    70 
    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:
    74 @code
    75 	RSqlStatement stmt;			//Prepare a statement with two  binary parameters
    76 	....
    77 	RSqlParamWriteStream strm1, strm2;
    78 	
    79 	strm1.BindBinary(stmt,...);	//Use a stream object to bind the 1st parameter value
    80 	strm1.WriteL(...);
    81 	strm1.CommitL();			
    82 	strm1.Close();				
    83 	
    84 	strm2.BindBinary(stmt,...);	//Use a stream object to bind the 2nd parameter value
    85 	strm2.WriteL(...);
    86 	strm2.CommitL();			
    87 	strm2.Close();				
    88 	
    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
    91 
    92 	strm2.BindBinary(stmt,...);	//Use a stream object to bind the 2nd parameter value
    93 	strm2.WriteL(...);
    94 	strm2.CommitL();			
    95 	strm2.Close();				
    96 
    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.
   100 @endcode
   101 								
   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.:
   106 @code
   107 	RSqlStatement stmt;		
   108 	....
   109 	RSqlParamWriteStream strm;
   110 	strm.BindBinary(stmt,...);
   111 	strm.WriteL(...);
   112 	strm.CommitL();		
   113 	stmt.Exec();
   114 	stmt.Close();
   115 	strm.Close();		
   116 @endcode
   117    
   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;
   123  								
   124 @endcode
   125  
   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;
   131  
   132 @see HIpcStream 
   133 @see CSqlSrvStatement
   134  
   135 @internalComponent
   136 */
   137 NONSHARABLE_CLASS(HSqlSrvStmtParamBuf) : public TBufBuf
   138 	{
   139 public:
   140 	enum TDataType {EBinary, EText16};
   141 	enum TBufType {EBufSimpleBind, EBufIpcStream};
   142 	
   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);
   146 	
   147 	inline const TPtrC8 SetDataL(const TDesC8& aData);
   148 	inline const TPtrC8 Data() const;
   149 	inline HSqlSrvStmtParamBuf::TDataType DataType() const;
   150 	
   151 	void NotifyStatementFinalized();
   152 	inline TInt ParamIndex() const;
   153 	
   154 private:
   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();
   161 	
   162 private:
   163 	enum {EExpandSize = 512};
   164 	
   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
   173 	};
   174 
   175 /////////////////////////////////////////////////////////////////////////////////////////////////////////
   176 //////////////////                CSqlSrvStatement                	   //////////////////////////////////
   177 /////////////////////////////////////////////////////////////////////////////////////////////////////////
   178 
   179 /**
   180 SQL statement handle.
   181 
   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;
   190 
   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()
   201 
   202 @see HSqlSrvStmtParamBuf
   203 
   204 @internalComponent
   205 */
   206 NONSHARABLE_CLASS(CSqlSrvStatement) : public CBase
   207 	{
   208 public:
   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();
   212 
   213 	void BindL(const RSqlBufFlat& aParamBuf);	
   214 	inline TInt Next();
   215 	inline TInt Reset();
   216 	inline TInt Exec();
   217 	
   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;
   223 	
   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;
   229 	
   230 	const RSqlBufFlat& GetDeclColumnTypesL();
   231 	
   232 	HSqlSrvStmtParamBuf* GetParamBufL(TInt aParamIndex, HSqlSrvStmtParamBuf::TDataType aDataType, HSqlSrvStmtParamBuf::TBufType aBufType);
   233 	void BindParamBufL(TInt aParamIndex);
   234 		
   235 private:
   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);
   243 
   244 private:
   245 	sqlite3_stmt*		iStmtHandle;		//SQL statement handle
   246 	TInt 				iColumnCount;		
   247 	TInt 				iParamCount;		
   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
   251 
   252 	};
   253 
   254 #include "SqlSrvStatement.inl"
   255 
   256 #endif//__SQLSRVSTATEMENT_H__