os/persistentdata/persistentstorage/sql/SRC/Client/SqlStatementImpl.h
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
     1 // Copyright (c) 2005-2009 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 Bug 1915 "SQL server panics when using long column type strings"
    13 //
    14 // Description:
    15 //
    16 
    17 #ifndef __SQLSTATEMENTIMPL_H__
    18 #define __SQLSTATEMENTIMPL_H__
    19 
    20 #include "SqlBufIterator.h"	//TSqlBufFlat, TSqlBufRIterator, TSqlBufWIterator
    21 #include "SqlDatabaseImpl.h"//CSqlDatabaseImpl
    22 #include "SqlStmtSession.h"	//RSqlStatementSession
    23 
    24 //Forward declarations
    25 class MStreamBuf;
    26 class CSqlStatementImpl;
    27 
    28 #ifdef _DEBUG
    29 #define LONGCOL_INVARIANT()	Invariant()
    30 #else
    31 #define LONGCOL_INVARIANT()
    32 #endif
    33 
    34 /**
    35 RSqlLongColumnColl class represents a collection of long text/binary column values.
    36 Once a column value is added to the collection, it is guaranteed that the memory address of the 
    37 column value will never change.
    38 The class was designed to solve one specific problem - to exclude the possibility that SQL clients
    39 may get a dangling pointer to a block of memory which already has been deleted.
    40 Without RSqlLongColumnColl class, it may happen for example, when client prepares a SELECT sql
    41 statement, where the record consists of two long binary/text column values.
    42 The client calls RSqlStatement::Next() to move to a record. Since the column values are long, they will
    43 not be transferred on the client side by the Next() call, so the client side row buffer (CSqlStatementImpl::iColumnValueBuf)
    44 will mark them as "not present". 
    45 Then the client calls RSqlStatement::ColumnBinary(0, TPtrC8&) to get a pointer to the first column value.
    46 The row buffer is reallocated, and the column value is retrieved from the server and copied into the buffer, the client's pointer
    47 is set to point to the place in the row buffer, where the column value is.
    48 While keeping the pointer to the first column value, the client calls again RSqlStatement::ColumnBinary(1, TPtrC8&) to retrieve
    49 the second column value. In which case, the row buffer may get reallocated again, so the first pointer will point to a 
    50 block of memory, which has been deleted.
    51 With the implementation and use of RSqlLongColumnColl class, the long column values will be kept in a different place,
    52 RSqlLongColumnColl's collection, where it is guaranteed that the memory address of the column value is constant.
    53 
    54 @internalComponent
    55 */
    56 NONSHARABLE_CLASS(RSqlLongColumnColl)
    57 	{
    58 public:	
    59 	struct TColumnReader
    60 		{
    61 		virtual TInt Read(TInt aColumnIndex, TDes8& aBuf) = 0;
    62 		};
    63 		
    64 public:	
    65 	inline RSqlLongColumnColl();
    66 	inline void Close();
    67 	inline void Reset();
    68 	TInt Append(TColumnReader& aReader, TInt aColumnIndex, TInt aColumnSize);
    69 	inline TPtrC Text(TInt aColumnIndex) const;
    70 	inline TPtrC8 Binary(TInt aColumnIndex) const;
    71 	inline TBool IsPresent(TInt aColumnIndex) const;
    72 
    73 private:
    74 	inline TInt FindValue(TInt aColumnIndex) const;
    75 #ifdef _DEBUG
    76 	void Invariant() const;	
    77 #endif
    78 	
    79 private:
    80 	struct TData
    81 		{
    82 		static TBool Compare(const TInt* aIndex, const TData& aData);
    83 		inline TData(TInt aIndex, HBufC8* aData);
    84 		TInt	iIndex;
    85 		HBufC8*	iData;
    86 		};
    87 	enum {KLongColumnCollGranularity = 8};
    88 	RArray<TData>	iValues;
    89 	
    90 	};
    91 
    92 template <class DES> TInt SqlCreateStatement(CSqlStatementImpl*& aImpl, CSqlDatabaseImpl& aDatabase, const DES& aSqlStmt);
    93 
    94 /**
    95 CSqlStatementImpl implements RSqlStatement functionality.
    96 CSqlStatementImpl can prepare and execute parametrized SQL statements and SQL statements without parameters.
    97 CSqlStatementImpl cannot prepare and execute SQL strings containing more than one SQL statement.
    98 @internalComponent
    99 */
   100 NONSHARABLE_CLASS(CSqlStatementImpl) : public CBase
   101 	{
   102 	template <class DES> friend TInt SqlCreateStatement(CSqlStatementImpl*& aImpl, CSqlDatabaseImpl& aDatabase, const DES& aSqlStmt);
   103 	
   104 public:
   105 	static inline TInt New(CSqlStatementImpl*& aImpl, CSqlDatabaseImpl& aDatabase, const TDesC16& aSqlStmt);
   106 	static inline TInt New(CSqlStatementImpl*& aImpl, CSqlDatabaseImpl& aDatabase, const TDesC8& aSqlStmt);
   107 	virtual ~CSqlStatementImpl();
   108 	
   109 	inline TBool AtRow() const;
   110 	
   111 	TInt Reset();
   112 	TInt Exec();
   113 	void Exec(TRequestStatus& aStatus);
   114 	TInt Next();
   115 	
   116 	TInt BindNull(TInt aParamIndex);
   117 	TInt BindInt(TInt aParamIndex, TInt aParamValue);
   118 	TInt BindInt64(TInt aParamIndex, TInt64 aParamValue);
   119 	TInt BindReal(TInt aParamIndex, TReal aParamValue);
   120 	TInt BindText(TInt aParamIndex, const TDesC& aParamText);
   121 	TInt BindBinary(TInt aParamIndex, const TDesC8& aParamData);
   122 	TInt BindZeroBlob(TInt aParamIndex, TInt aBlobSize);
   123 
   124 	inline TInt ColumnCount() const;
   125 	TSqlColumnType ColumnType(TInt aColumnIndex);
   126 	TInt DeclaredColumnType(TInt aColumnIndex, TSqlColumnType& aColumnType);
   127 	TInt ColumnSize(TInt aColumnIndex);
   128 	TInt ColumnInt(TInt aColumnIndex);
   129 	TInt64 ColumnInt64(TInt aColumnIndex);
   130 	TReal ColumnReal(TInt aColumnIndex);
   131 	TInt ColumnText(TInt aColumnIndex, TPtrC& aPtr);
   132 	TInt ColumnText(TInt aColumnIndex, TDes& aDest);
   133 	TInt ColumnBinary(TInt aColumnIndex, TPtrC8& aPtr);
   134 	TInt ColumnBinary(TInt aColumnIndex, TDes8& aDest);
   135 
   136 	TInt ColumnName(TInt aColumnIndex, TPtrC& aNameDest);
   137 	TInt ParameterName(TInt aParamIndex, TPtrC& aNameDest);
   138 
   139 	inline TInt ParamIndex(const TDesC& aParamName);
   140 	inline TInt ColumnIndex(const TDesC& aColumnName);
   141 		
   142 	inline MStreamBuf* ColumnSourceL(TInt aColumnIndex);
   143 	inline MStreamBuf* ParamSinkL(TSqlSrvFunction aFunction, TInt aParamIndex);
   144 
   145 private:
   146 	inline CSqlStatementImpl();
   147 	template <class DES> TInt Construct(CSqlDatabaseImpl& aDatabase, const DES& aSqlStmt);
   148 	TInt Name2Index(const TDesC& aName, RSqlBufFlat& aNameBufFlat, TInt aCount, TSqlSrvFunction aFunction, TBool& aPresent);
   149 	TInt Index2Name(TInt aIndex, RSqlBufFlat& aNameBufFlat, TInt aCount, TSqlSrvFunction aFunction, TBool& aPresent, TPtrC& aColName);
   150 	TInt CheckNameBufPresent(TBool& aPresent, RSqlBufFlat& aNameBufFlat, TInt aCount, TSqlSrvFunction aFunction);
   151 	
   152 private:
   153 	RSqlStatementSession	iSqlStmtSession;
   154 	TBool					iBound;
   155 	enum TState				{EUnknown, EAtRow};
   156 	TState					iState;
   157 	
   158 	TInt 					iColumnCnt;
   159 	TBool					iColumnNameBufPresent;
   160 	RSqlBufFlat				iColumnNameBuf;
   161 	RSqlBufFlat				iColumnValueBuf;
   162 	TSqlBufRIterator		iColumnValBufIt;
   163 	
   164 	TInt 					iParamCnt;
   165 	TBool					iParamNameBufPresent;
   166 	RSqlBufFlat				iParamNameBuf;
   167 	RSqlBufFlat				iParamValueBuf;
   168 	TSqlBufWIterator		iParamValBufIt;
   169 	
   170 	RSqlLongColumnColl		iLongColumnColl;
   171 	
   172 	RArray<TSqlColumnType>	iDeclaredColumnTypes;	//Array of declared column types for current statement.
   173 	};
   174 
   175 #include "SqlStatementImpl.inl"
   176 
   177 #endif //__SQLSTATEMENTIMPL_H__