os/persistentdata/persistentstorage/sql/SRC/Client/SqlStatementImpl.cpp
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
     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 #include "SqlStatementImpl.h" 	//CSqlStatementImpl
    18 #include "SqlDatabaseImpl.h"	//CSqlDatabaseImpl::Session()
    19 
    20 //Constants
    21 
    22 _LIT(KTextKWD, 	"TEXT");
    23 _LIT(KCharKWD, 	"CHAR");
    24 _LIT(KClobKWD, 	"CLOB");
    25 
    26 _LIT(KBinaryKWD,"BINARY");
    27 _LIT(KBlobKWD,	"BLOB");
    28 
    29 _LIT(KRealKWD, 	"REAL");
    30 _LIT(KFloatKWD, "FLOAT");
    31 _LIT(KDoubleKWD,"DOUBLE");
    32 
    33 /////////////////////////////////////////////////////////////////////////////////////////////////////////////
    34 //////////////////////////             RSqlLongColumnColl        ////////////////////////////////////////////
    35 /////////////////////////////////////////////////////////////////////////////////////////////////////////////
    36 
    37 /**
    38 This "comparison" function is used for searching a particular column value, identified by aIndex, in the long column 
    39 values collection.
    40 @param aIndex An integer pointer to long column index value
    41 @param aData A reference to particular elemt of the long column values collection
    42 @return True of the index of aData item is the same as *aIndex, false otherwise
    43 @panic SqlDb 4 In _DEBUG mode, aIndex is NULL.
    44 @panic SqlDb 4 In _DEBUG mode, *aIndex value is negative.
    45 */
    46 TBool RSqlLongColumnColl::TData::Compare(const TInt* aIndex, const RSqlLongColumnColl::TData& aData)
    47 	{
    48 	__ASSERT_DEBUG(aIndex != NULL, __SQLPANIC2(ESqlPanicBadArgument));
    49 	__ASSERT_DEBUG(*aIndex >= 0, __SQLPANIC2(ESqlPanicBadArgument));
    50 	return *aIndex == aData.iIndex;
    51 	}
    52 
    53 /**
    54 Reads a long column value, identified by aColumnIndex parameter, from the server and stores the value in the collection.
    55 @param aReader A RSqlLongColumnColl::TColumnReader object, which performs the "read column value" operation.
    56 @param aColumnIndex Column index
    57 @param aColumnSize The column size in bytes
    58 @panic SqlDb 4 In _DEBUG mode, aColumnIndex value is negative.
    59 @panic SqlDb 4 In _DEBUG mode, aColumnSize is less than KSqlMaxDesLen (not a long column value)
    60 @return KErrNone The call has completed successfully;
    61 		KErrNoMemory Out of memory;
    62         Note that other system-wide or database specific error codes may also be returned.
    63 */
    64 TInt RSqlLongColumnColl::Append(RSqlLongColumnColl::TColumnReader& aReader, TInt aColumnIndex, TInt aColumnSize)
    65 	{
    66 	__ASSERT_DEBUG(aColumnIndex >= 0, __SQLPANIC(ESqlPanicBadArgument));
    67 	__ASSERT_DEBUG(aColumnSize >= KSqlMaxDesLen, __SQLPANIC(ESqlPanicBadArgument));
    68 	LONGCOL_INVARIANT();
    69 	HBufC8* colBuf = HBufC8::New(aColumnSize);
    70 	if(!colBuf)
    71 		{
    72 		return KErrNoMemory;	
    73 		}
    74 	TPtr8 ptr = colBuf->Des();
    75 	TInt err = aReader.Read(aColumnIndex, ptr);
    76 	if(err == KErrNone)
    77 		{
    78 		err = iValues.Append(RSqlLongColumnColl::TData(aColumnIndex, colBuf));	
    79 		}
    80 	if(err != KErrNone)
    81 		{
    82 		delete colBuf;	
    83 		}
    84 	LONGCOL_INVARIANT();
    85 	return err;
    86 	}
    87 	
    88 #ifdef _DEBUG
    89 /**
    90 RSqlLongColumnColl invariant.
    91 The collection cannot have two column values with the same column index.
    92 The collection cannot have NULL column value, or negative column index.
    93 */
    94 void RSqlLongColumnColl::Invariant() const
    95 	{
    96 	for(TInt i=iValues.Count()-1;i>=0;--i)
    97 		{
    98 		const RSqlLongColumnColl::TData& data = iValues[i];
    99 		__ASSERT_DEBUG(data.iIndex >= 0, __SQLPANIC(ESqlPanicInternalError));
   100 		__ASSERT_DEBUG(data.iData != NULL, __SQLPANIC(ESqlPanicInternalError));
   101 		for(TInt j=i-1;j>=0;--j)
   102 			{
   103 			__ASSERT_DEBUG(data.iIndex != iValues[j].iIndex, __SQLPANIC(ESqlPanicInternalError));
   104 			}
   105 		}
   106 	}
   107 #endif
   108 
   109 /////////////////////////////////////////////////////////////////////////////////////////////////////////////
   110 //////////////////////////           CSqlStatementImpl           ////////////////////////////////////////////
   111 /////////////////////////////////////////////////////////////////////////////////////////////////////////////
   112 
   113 /**
   114 RSqlLongColumnColl::TColumnReader derived class, which implements the pure virtual RSqlLongColumnColl::TColumnReader::Read().
   115 The class is used for retrieving long column values from the server.
   116 @internalComponent
   117 */
   118 NONSHARABLE_STRUCT(TSqlLongColumnReader) : public RSqlLongColumnColl::TColumnReader
   119 	{
   120 	TSqlLongColumnReader(RSqlStatementSession& aStmtSession) :
   121 		iStmtSession(aStmtSession)
   122 		{
   123 		}
   124 	virtual TInt Read(TInt aColumnIndex, TDes8& aBuf)
   125 		{
   126 		return iStmtSession.ReadColumnValue(aColumnIndex, aBuf);
   127 		}
   128 	RSqlStatementSession	iStmtSession;
   129 	};
   130 
   131 /////////////////////////////////////////////////////////////////////////////////////////////////////////////
   132 //////////////////////////           CSqlStatementImpl           ////////////////////////////////////////////
   133 /////////////////////////////////////////////////////////////////////////////////////////////////////////////
   134 
   135 /**
   136 Frees the allocated memory and other resources.
   137 */
   138 CSqlStatementImpl::~CSqlStatementImpl()
   139 	{
   140 	iDeclaredColumnTypes.Close();
   141 	iLongColumnColl.Close();
   142 	iParamValueBuf.Close();
   143 	iParamNameBuf.Close();
   144 	iColumnValueBuf.Close();
   145 	iColumnNameBuf.Close();
   146 	iSqlStmtSession.Close();
   147 	}
   148 
   149 
   150 
   151 /**
   152 Resets the prepared SQL statement back to its initial state and makes it ready for re-execution.
   153 The parameters of the SQL statement retain their values.
   154 If CSqlStatementImpl object processes parametrized SQL statement, the parameter values can be bound after 
   155 the Reset() call.
   156 
   157 The method sets the internal object's state to CSqlStatementImpl::EUnknown. The column data accessors
   158 cannot be used until the next successful Next() call.
   159 
   160 @return KSqlErrStmtExpired, statement expired (if new functions or collating sequences are 
   161 							registered or if an authorizer function is added or changed);
   162 		KErrNone, the operation has completed successfully.
   163 
   164 @see RSqlStatement::Reset()
   165 */
   166 TInt CSqlStatementImpl::Reset()
   167 	{
   168 	iState = CSqlStatementImpl::EUnknown;
   169 	iLongColumnColl.Reset();
   170 	if(iParamCnt > 0)
   171 		{
   172 		iParamValueBuf.ResetAndMinimize();
   173 		}
   174 	return iSqlStmtSession.Reset();
   175 	}
   176 
   177 /**
   178 Executes the prepared DDL/DML SQL statement. 
   179 
   180 If the SQL statement contains parameters, their values will be bound right before the execution.
   181 
   182 The method sets the internal object's state to CSqlStatementImpl::EUnknown. The column data accessors
   183 cannot be used until the next successful Next() call.
   184 
   185 @return >=0, The operation has completed successfully. The number of database rows that were 
   186 			 changed/inserted/deleted by the most recently completed DDL/DML sql statement.
   187 			 Exception: If the executed statement is "DELETE FROM <table>", then the function returns 0 
   188 			 if the operation has completed successfully (disregarding the number of the deleted rows);
   189 	    KSqlErrStmtExpired, statement has expired (if new functions or collating sequences are 
   190 							registered or if an authorizer function is added or changed);
   191 		KErrNoMemory, an out of memory condition has occurred.
   192                       Note that database specific errors categorised as ESqlDbError, and
   193                       other system-wide error codes may also be returned.
   194 
   195 @see RSqlStatement::Exec()
   196 */
   197 TInt CSqlStatementImpl::Exec()
   198 	{
   199 	iState = CSqlStatementImpl::EUnknown;
   200 	TInt rc = 0;
   201 	if(iBound)
   202 		{
   203 		rc = iSqlStmtSession.Exec();	
   204 		}
   205 	else
   206 		{
   207 		rc = iSqlStmtSession.BindExec(iParamValueBuf);
   208 		iBound = ETrue;
   209 		iParamValueBuf.ResetAndMinimize();
   210 		}
   211 	return rc;
   212 	}
   213 
   214 /**
   215 Executes the prepared DDL/DML SQL statement asynchronously. 
   216 
   217 If the SQL statement contains parameters, their values will be bound right before the execution.
   218 
   219 The method sets the internal object's state to CSqlStatementImpl::EUnknown. The column data accessors
   220 cannot be used until the next successful Next() call.
   221 
   222 @param aStatus Completion status of asynchronous request, one of the following:
   223 		 >=0, The operation has completed successfully. The number of database rows that were 
   224 		   	   changed/inserted/deleted by the most recently completed DDL/DML sql statement.
   225 			 Exception: If the executed statement is "DELETE FROM <table>", then the function returns 0 
   226 			 if the operation has completed successfully (disregarding the number of the deleted rows);
   227          KSqlErrStmtExpired, the SQL statement has expired (if new functions or
   228                             collating sequences have been registered or if an
   229                             authorizer function has been added or changed);
   230          KErrNoMemory, an out of memory condition has occurred - the statement
   231                       will be reset.
   232                       Note that aStatus may be set with database specific errors categorised as ESqlDbError, 
   233                       and other system-wide error codes.
   234 
   235 
   236 @see RSqlStatement::Exec()
   237 */
   238 void CSqlStatementImpl::Exec(TRequestStatus& aStatus)
   239 	{
   240 	iState = CSqlStatementImpl::EUnknown;
   241 	if(iBound)
   242 		{
   243 		iSqlStmtSession.Exec(aStatus);
   244 		}
   245 	else
   246 		{
   247 		iSqlStmtSession.BindExec(iParamValueBuf, aStatus);
   248 		iBound = ETrue;
   249 		//Here: it is not possible to reset iParamValueBuf for "safety", because this is an asynchronous call
   250 		//      and the buffer content has to be preserved until the operation completes.
   251 		}
   252 	}
   253 
   254 /**
   255 Retrieves a record.
   256 
   257 If the prepared SQL statement is a "SELECT" statement, and is expected to
   258 return a set of records, then this function can be used to retrieve that record data.
   259 
   260 If the SQL statement contains parameters, then their values must be bound before
   261 this function is called.
   262 
   263 If the call to this function completes successfully, i.e. it returns
   264 with KSqlAtRow, then this CSqlStatementImpl object contains the record data, and 
   265 this data will remain valid for access until another call is made to any
   266 CSqlStatementImpl function.
   267 
   268 The record data can be accessed using the following functions:
   269 - CSqlStatementImpl::ColumnType()
   270 - CSqlStatementImpl::ColumnSize()
   271 - CSqlStatementImpl::Column<Type>()
   272 
   273 @return KSqlAtRow,      the record data is ready for processing by the caller;
   274         KSqlAtEnd,      there is no more record data;
   275         KSqlErrBusy,    the database file is locked;
   276         KErrNoMemory,   an out of memory condition has occurred - the statement
   277                         will be reset;
   278         KSqlErrGeneral, a run-time error has occured - this function must not
   279                         be called again;        
   280         KSqlErrMisuse,  this function has been called after a previous call
   281                         returned KSqlAtEnd or KSqlErrGeneral.
   282         KSqlErrStmtExpired, the SQL statement has expired (if new functions or
   283                             collating sequences have been registered or if an
   284                             authorizer function has been added or changed);
   285 
   286 @see RSqlStatement::Next()
   287 */
   288 TInt CSqlStatementImpl::Next()
   289 	{
   290 	iLongColumnColl.Reset();
   291 	iColumnValueBuf.ResetAndMinimize();
   292 	iState = CSqlStatementImpl::EUnknown;
   293 	TInt err = KErrNone;
   294 	if(iBound)
   295 		{
   296 		err = iSqlStmtSession.Next(iColumnValueBuf);
   297 		}
   298 	else
   299 		{
   300 		err = iSqlStmtSession.BindNext(iParamValueBuf, iColumnValueBuf);
   301 		iBound = ETrue;
   302 		iParamValueBuf.ResetAndMinimize();
   303 		}
   304 	iColumnValBufIt.Set(iColumnValueBuf);
   305 	if(err == KSqlAtRow)
   306 		{
   307 		iState = CSqlStatementImpl::EAtRow;
   308 		}
   309 	return err;
   310 	}
   311 
   312 /**
   313 Implements RSqlStatement::BindNull().
   314 
   315 @panic SqlDb 5 Parameter index out of bounds.
   316 
   317 @see RSqlStatement::BindNull()
   318 */	
   319 TInt CSqlStatementImpl::BindNull(TInt aParamIndex)
   320 	{
   321 	__ASSERT_ALWAYS((TUint)aParamIndex < (TUint)iParamCnt, __SQLPANIC(ESqlPanicBadColumnIndex));
   322 	iBound = EFalse;
   323 	iParamValBufIt.MoveTo(aParamIndex);
   324 	iParamValBufIt.SetNull();
   325 	return KErrNone;
   326 	}
   327 	
   328 /**
   329 Implements RSqlStatement::BindInt().
   330 
   331 @panic SqlDb 5 Parameter index out of bounds.
   332 
   333 @see RSqlStatement::BindInt()
   334 */	
   335 TInt CSqlStatementImpl::BindInt(TInt aParamIndex, TInt aParamValue)
   336 	{
   337 	__ASSERT_ALWAYS((TUint)aParamIndex < (TUint)iParamCnt, __SQLPANIC(ESqlPanicBadColumnIndex));
   338 	iBound = EFalse;
   339 	iParamValBufIt.MoveTo(aParamIndex);
   340 	return iParamValBufIt.SetInt(aParamValue);
   341 	}
   342 	
   343 /**
   344 Implements RSqlStatement::BindInt64().
   345 
   346 @panic SqlDb 5 Parameter index out of bounds.
   347 
   348 @see RSqlStatement::BindInt64()
   349 */	
   350 TInt CSqlStatementImpl::BindInt64(TInt aParamIndex, TInt64 aParamValue)
   351 	{
   352 	__ASSERT_ALWAYS((TUint)aParamIndex < (TUint)iParamCnt, __SQLPANIC(ESqlPanicBadColumnIndex));
   353 	iBound = EFalse;
   354 	iParamValBufIt.MoveTo(aParamIndex);
   355 	return iParamValBufIt.SetInt64(aParamValue);
   356 	}
   357 	
   358 /**
   359 Implements RSqlStatement::BindReal().
   360 
   361 @panic SqlDb 5 Parameter index out of bounds.
   362 
   363 @see RSqlStatement::BindReal()
   364 */	
   365 TInt CSqlStatementImpl::BindReal(TInt aParamIndex, TReal aParamValue)
   366 	{
   367 	__ASSERT_ALWAYS((TUint)aParamIndex < (TUint)iParamCnt, __SQLPANIC(ESqlPanicBadColumnIndex));
   368 	iBound = EFalse;
   369 	iParamValBufIt.MoveTo(aParamIndex);
   370 	return iParamValBufIt.SetReal(aParamValue);
   371 	}
   372 	
   373 /**
   374 Implements RSqlStatement::BindText().
   375 
   376 @panic SqlDb 5 Parameter index out of bounds.
   377 
   378 @see RSqlStatement::BindText()
   379 */	
   380 TInt CSqlStatementImpl::BindText(TInt aParamIndex, const TDesC& aParamText)
   381 	{
   382 	__ASSERT_ALWAYS((TUint)aParamIndex < (TUint)iParamCnt, __SQLPANIC(ESqlPanicBadColumnIndex));
   383 	iBound = EFalse;
   384 	iParamValBufIt.MoveTo(aParamIndex);
   385 	return iParamValBufIt.SetText(aParamText);
   386 	}
   387 	
   388 /**
   389 Implements RSqlStatement::BindBinary().
   390 
   391 @panic SqlDb 5 Parameter index out of bounds.
   392 
   393 @see RSqlStatement::BindBinary()
   394 */	
   395 TInt CSqlStatementImpl::BindBinary(TInt aParamIndex, const TDesC8& aParamData)
   396 	{
   397 	__ASSERT_ALWAYS((TUint)aParamIndex < (TUint)iParamCnt, __SQLPANIC(ESqlPanicBadColumnIndex));
   398 	iBound = EFalse;
   399 	iParamValBufIt.MoveTo(aParamIndex);
   400 	return iParamValBufIt.SetBinary(aParamData);
   401 	}
   402 
   403 /**
   404 Implements RSqlStatement::BindZeroBlob().
   405 
   406 @panic SqlDb 5 Parameter index out of bounds.
   407 
   408 @see RSqlStatement::BindZeroBlob()
   409 */	
   410 TInt CSqlStatementImpl::BindZeroBlob(TInt aParamIndex, TInt aBlobSize)
   411 	{
   412 	__ASSERT_ALWAYS((TUint)aParamIndex < (TUint)iParamCnt, __SQLPANIC(ESqlPanicBadColumnIndex));
   413 	iBound = EFalse;
   414 	iParamValBufIt.MoveTo(aParamIndex);
   415 	return iParamValBufIt.SetZeroBlob(aBlobSize);
   416 	}
   417 
   418 /**
   419 Implements RSqlStatement::ColumnType().
   420 
   421 @panic SqlDb 5 Column index out of bounds.
   422 @panic SqlDb 11 Statement cursor not positioned on a row.
   423 
   424 @see RSqlStatement::ColumnType().
   425 */	
   426 TSqlColumnType CSqlStatementImpl::ColumnType(TInt aColumnIndex)
   427 	{
   428 	__ASSERT_ALWAYS((TUint)aColumnIndex < (TUint)iColumnCnt, __SQLPANIC(ESqlPanicBadColumnIndex));
   429 	__ASSERT_ALWAYS(iState == CSqlStatementImpl::EAtRow, __SQLPANIC(ESqlPanicInvalidRow));
   430 	iColumnValBufIt.MoveTo(aColumnIndex);		
   431 	return static_cast <TSqlColumnType> (iColumnValBufIt.Type());
   432 	}
   433 
   434 /**
   435 Implements RSqlStatement::DeclaredColumnType().
   436 
   437 @param aColumnIndex The index value identifying the column. This is 0 for the first column.
   438 @param aColumnType	Output parameter. Upon completion this will contain the type of the column.
   439 
   440 @return KErrNone, the operation completed successfully;
   441         KErrNoMemory, an out of memory condition has occurred.
   442                   One of the other system-wide error codes may also be returned.
   443 
   444 @panic SqlDb 5 Column index out of bounds.
   445 
   446 @see RSqlStatement::DeclaredColumnType().
   447 */
   448 TInt CSqlStatementImpl::DeclaredColumnType(TInt aColumnIndex, TSqlColumnType& aColumnType)
   449 	{
   450 	__ASSERT_ALWAYS((TUint)aColumnIndex < (TUint)iColumnCnt, __SQLPANIC(ESqlPanicBadColumnIndex));
   451 	if(iDeclaredColumnTypes.Count() == 0) //initialise iDeclaredColumnTypes array if necessary
   452 		{
   453 		RSqlBufFlat declaredColumnTypeBuf;
   454 		TInt err = declaredColumnTypeBuf.SetCount(iColumnCnt);
   455 		if(err != KErrNone)
   456 			{
   457 			declaredColumnTypeBuf.Close();
   458 			return err;
   459 			}
   460 		
   461 		//Resize buffer to minimise the chance that two IPC calls are required to get all the column type name data.
   462 		//Allocates enough space to contain the header cells (already created by calling SetCount()) and a data buffer
   463 		//which assumes an average column type name length of 20 characters plus 8-byte alignment padding
   464 		const TInt KBufSizePerColumn = 48;
   465 		TInt newSize = declaredColumnTypeBuf.Size() + iColumnCnt*KBufSizePerColumn;
   466 
   467 		err = declaredColumnTypeBuf.ReAlloc(newSize);
   468 		if(err != KErrNone)
   469 			{
   470 			declaredColumnTypeBuf.Close();
   471 			return err;	
   472 			}
   473 		
   474 		err = iSqlStmtSession.GetDeclColumnTypes(declaredColumnTypeBuf);
   475 		if(err != KErrNone)
   476 			{
   477 			declaredColumnTypeBuf.Close();
   478 			return err;	
   479 			}
   480 
   481 		err = iDeclaredColumnTypes.Reserve(iColumnCnt);//We know what the array size should be - iColumnCnt
   482 		if(err != KErrNone)
   483 			{
   484 			declaredColumnTypeBuf.Close();
   485 			return err;
   486 			}	
   487 
   488 		//Iterate over the column type names buffer and map each column type name to one of the TSqlColumnType enum item values.
   489 		TSqlBufRIterator declColumnTypeBufIt;
   490 		declColumnTypeBufIt.Set(declaredColumnTypeBuf);
   491 		TInt colIdx = 0;
   492 		while(declColumnTypeBufIt.Next())
   493 			{
   494 			TPtrC colTypeName(declColumnTypeBufIt.Text());
   495 			TSqlColumnType colType = ESqlInt;
   496 			if(colTypeName.FindF(KCharKWD) >= 0 || colTypeName.FindF(KTextKWD) >= 0 || colTypeName.FindF(KClobKWD) >= 0)
   497 				{
   498 				colType = ESqlText;
   499 				}
   500 			else if(colTypeName.FindF(KBinaryKWD) >= 0 || colTypeName.FindF(KBlobKWD) >= 0)
   501 				{
   502 				colType = ESqlBinary;
   503 				}
   504 			else if(colTypeName.FindF(KRealKWD) >= 0 || colTypeName.FindF(KFloatKWD) >= 0 || colTypeName.FindF(KDoubleKWD) >= 0)
   505 				{
   506 				colType = ESqlReal;
   507 				}
   508 			err = iDeclaredColumnTypes.Append(colType);
   509 			__ASSERT_DEBUG(err == KErrNone, __SQLPANIC(ESqlPanicInternalError));//memory for the array elements has been reserved already
   510 			++colIdx;
   511 			} //end of - while(declColumnTypeBufIt.Next())
   512 		__ASSERT_DEBUG(colIdx == iColumnCnt, __SQLPANIC(ESqlPanicInternalError));
   513 		declaredColumnTypeBuf.Close();
   514 		} //end of - if(iDeclaredColumnTypes.Count() == 0 && iColumnCnt > 0)
   515 	aColumnType = iDeclaredColumnTypes[aColumnIndex];
   516 	return KErrNone;
   517 	}
   518 	
   519 /**
   520 Implements RSqlStatement::ColumnSize().
   521 
   522 @panic SqlDb 5 Column index out of bounds.
   523 @panic SqlDb 11 Statement cursor not positioned on a row
   524 
   525 @see RSqlStatement::ColumnSize().
   526 */	
   527 TInt CSqlStatementImpl::ColumnSize(TInt aColumnIndex)
   528 	{
   529 	__ASSERT_ALWAYS((TUint)aColumnIndex < (TUint)iColumnCnt, __SQLPANIC(ESqlPanicBadColumnIndex));
   530 	__ASSERT_ALWAYS(iState == CSqlStatementImpl::EAtRow, __SQLPANIC(ESqlPanicInvalidRow));
   531 	iColumnValBufIt.MoveTo(aColumnIndex);		
   532 	return iColumnValBufIt.Size();
   533 	}
   534 
   535 /**
   536 Implements RSqlStatement::ColumnInt().
   537 
   538 @panic SqlDb 5 Column index out of bounds.
   539 @panic SqlDb 11 Statement cursor not positioned on a row
   540 
   541 @see RSqlStatement::ColumnInt().
   542 */	
   543 TInt CSqlStatementImpl::ColumnInt(TInt aColumnIndex)
   544 	{
   545 	__ASSERT_ALWAYS((TUint)aColumnIndex < (TUint)iColumnCnt, __SQLPANIC(ESqlPanicBadColumnIndex));
   546 	__ASSERT_ALWAYS(iState == CSqlStatementImpl::EAtRow, __SQLPANIC(ESqlPanicInvalidRow));
   547 	iColumnValBufIt.MoveTo(aColumnIndex);		
   548 	__ASSERT_DEBUG(iColumnValBufIt.IsPresent(), __SQLPANIC(ESqlPanicValueNotPresent));
   549 	return iColumnValBufIt.Int();
   550 	}
   551 	
   552 /**
   553 Implements RSqlStatement::ColumnInt64().
   554 
   555 @panic SqlDb 5 Column index out of bounds.
   556 @panic SqlDb 11 Statement cursor not positioned on a row
   557 
   558 @see RSqlStatement::ColumnInt64().
   559 */	
   560 TInt64 CSqlStatementImpl::ColumnInt64(TInt aColumnIndex)
   561 	{
   562 	__ASSERT_ALWAYS((TUint)aColumnIndex < (TUint)iColumnCnt, __SQLPANIC(ESqlPanicBadColumnIndex));
   563 	__ASSERT_ALWAYS(iState == CSqlStatementImpl::EAtRow, __SQLPANIC(ESqlPanicInvalidRow));
   564 	iColumnValBufIt.MoveTo(aColumnIndex);		
   565 	__ASSERT_DEBUG(iColumnValBufIt.IsPresent(), __SQLPANIC(ESqlPanicValueNotPresent));
   566 	return iColumnValBufIt.Int64();
   567 	}
   568 	
   569 /**
   570 Implements RSqlStatement::ColumnReal().
   571 
   572 @panic SqlDb 5 Column index out of bounds.
   573 @panic SqlDb 11 Statement cursor not positioned on a row
   574 
   575 @see RSqlStatement::ColumnReal().
   576 */	
   577 TReal CSqlStatementImpl::ColumnReal(TInt aColumnIndex)
   578 	{
   579 	__ASSERT_ALWAYS((TUint)aColumnIndex < (TUint)iColumnCnt, __SQLPANIC(ESqlPanicBadColumnIndex));
   580 	__ASSERT_ALWAYS(iState == CSqlStatementImpl::EAtRow, __SQLPANIC(ESqlPanicInvalidRow));
   581 	iColumnValBufIt.MoveTo(aColumnIndex);		
   582 	__ASSERT_DEBUG(iColumnValBufIt.IsPresent(), __SQLPANIC(ESqlPanicValueNotPresent));
   583 	return iColumnValBufIt.Real();
   584 	}
   585 
   586 /**
   587 Return a text (16 bit) descriptor to a text column identified by aColumnIndex.
   588 
   589 @param aColumnIndex Column index
   590 @param aPtr An output parameter which will be set to point to the column data.
   591 
   592 @return KErrNone, if the function completes successfully,
   593                   otherwise one of the other system-wide error codes.
   594 
   595 @panic SqlDb 5 Column index out of bounds.
   596 @panic SqlDb 11 Statement cursor not positioned on a row
   597 */	
   598 TInt CSqlStatementImpl::ColumnText(TInt aColumnIndex, TPtrC& aPtr)
   599 	{
   600 	__ASSERT_ALWAYS((TUint)aColumnIndex < (TUint)iColumnCnt, __SQLPANIC(ESqlPanicBadColumnIndex));
   601 	__ASSERT_ALWAYS(iState == CSqlStatementImpl::EAtRow, __SQLPANIC(ESqlPanicInvalidRow));
   602 	iColumnValBufIt.MoveTo(aColumnIndex);		
   603 	if(iColumnValBufIt.IsPresent())
   604 		{
   605 		aPtr.Set(iColumnValBufIt.Text());
   606 		return KErrNone;
   607 		}
   608 	//The text column value has not been transferred to the client side if its length is >= KSqlMaxDesLen characters.
   609 	//In this case an additional call to the server is made to get the column value.
   610 	//The column value is stored in a separate collection (iLongColumnColl), because if iColumnValueBuf gets reallocated,
   611 	//the client can get a dangling pointer to some of the located in iColumnValueBuf text/binary column values.
   612 	if(iColumnValBufIt.Type() != ESqlText)
   613 		{
   614 		aPtr.Set(KNullDesC);
   615 		return KErrNone;
   616 		}
   617 	if(!iLongColumnColl.IsPresent(aColumnIndex))
   618 		{
   619 		TSqlLongColumnReader colReader(iSqlStmtSession);
   620 		TInt err = iLongColumnColl.Append(colReader, aColumnIndex, iColumnValBufIt.Size() * sizeof(TUint16));
   621 		if(err != KErrNone)
   622 			{
   623 			return err;	
   624 			}
   625 		}
   626 	aPtr.Set(iLongColumnColl.Text(aColumnIndex));
   627 	return KErrNone;
   628 	}
   629 
   630 /**
   631 Copies the content of a text column, identified by aColumnIndex, to the place refered by aDest parameter.
   632 
   633 If the destination buffer is not big enough, the function will copy as much data as possible and will
   634 return KErrOverflow.
   635 
   636 @param aColumnIndex Column index
   637 @param aDest Refers to the place where the column data will be copied.
   638 
   639 @return KErrNone, if the function completes successfully,
   640                   otherwise one of the other system-wide error codes.
   641 
   642 @panic SqlDb 5 Column index out of bounds.
   643 @panic SqlDb 11 Statement cursor not positioned on a row
   644 */	
   645 TInt CSqlStatementImpl::ColumnText(TInt aColumnIndex, TDes& aDest)
   646 	{
   647 	__ASSERT_ALWAYS((TUint)aColumnIndex < (TUint)iColumnCnt, __SQLPANIC(ESqlPanicBadColumnIndex));
   648 	__ASSERT_ALWAYS(iState == CSqlStatementImpl::EAtRow, __SQLPANIC(ESqlPanicInvalidRow));
   649 	iColumnValBufIt.MoveTo(aColumnIndex);		
   650 	TInt err = KErrNone;
   651 	//The text column value has not been transferred to the client side if its length is >= KSqlMaxDesLen characters.
   652 	//In this case an additional call to the server is made to get the column value.
   653 	if(!iColumnValBufIt.IsPresent())
   654 		{
   655 		if(iColumnValBufIt.Type() != ESqlText)
   656 			{
   657 			aDest.Zero();
   658 			return err;
   659 			}
   660 		TPtr8 ptr(reinterpret_cast <TUint8*> (const_cast <TUint16*> (aDest.Ptr())), aDest.MaxLength() * sizeof(TUint16));
   661 		err = iSqlStmtSession.ReadColumnValue(aColumnIndex, ptr);
   662 		switch(err)
   663 		    {
   664 	        case KErrNone:
   665 	        case KErrOverflow:
   666 	            aDest.SetLength(ptr.Length() / sizeof(TUint16));
   667 	            break;
   668 	        default:
   669 	            break;
   670 		    }
   671 		}
   672 	else
   673 		{
   674 		TPtrC src = iColumnValBufIt.Text();
   675 		TInt len = src.Length();
   676 		if(len > aDest.MaxLength())
   677 			{
   678 			len = aDest.MaxLength();
   679 			err = KErrOverflow;
   680 			}
   681 		aDest.Copy(src.Ptr(), len);
   682 		}
   683 	return err;
   684 	}
   685 
   686 /**
   687 Return a binary (8 bit) descriptor to a binary column identified by aColumnIndex.
   688 
   689 @param aColumnIndex Column index
   690 @param aPtr An output parameter which will be set to point to the column data.
   691 
   692 @return KErrNone, if the function completes successfully,
   693                   otherwise one of the other system-wide error codes.
   694 
   695 @panic SqlDb 5 Column index out of bounds.
   696 @panic SqlDb 11 Statement cursor not positioned on a row
   697 */	
   698 TInt CSqlStatementImpl::ColumnBinary(TInt aColumnIndex, TPtrC8& aPtr)
   699 	{
   700 	__ASSERT_ALWAYS((TUint)aColumnIndex < (TUint)iColumnCnt, __SQLPANIC(ESqlPanicBadColumnIndex));
   701 	__ASSERT_ALWAYS(iState == CSqlStatementImpl::EAtRow, __SQLPANIC(ESqlPanicInvalidRow));
   702 	iColumnValBufIt.MoveTo(aColumnIndex);		
   703 	if(iColumnValBufIt.IsPresent())
   704 		{
   705 		aPtr.Set(iColumnValBufIt.Binary());
   706 		return KErrNone;
   707 		}
   708 	//The binary column value has not been transferred to the client side if its length is >= KSqlMaxDesLen characters.
   709 	//In this case an additional call to the server is made to get the column value.
   710 	//The column value is stored in a separate collection (iLongColumnColl), because if iColumnValueBuf gets reallocated,
   711 	//the client can get a dangling pointer to some of the located in iColumnValueBuf text/binary column values.
   712 	if(iColumnValBufIt.Type() != ESqlBinary)
   713 		{
   714 		aPtr.Set(KNullDesC8);
   715 		return KErrNone;
   716 		}
   717 	if(!iLongColumnColl.IsPresent(aColumnIndex))
   718 		{
   719 		TSqlLongColumnReader colReader(iSqlStmtSession);
   720 		TInt err = iLongColumnColl.Append(colReader, aColumnIndex, iColumnValBufIt.Size());
   721 		if(err != KErrNone)
   722 			{
   723 			return err;	
   724 			}
   725 		}
   726 	aPtr.Set(iLongColumnColl.Binary(aColumnIndex));
   727 	return KErrNone;
   728 	}
   729 
   730 /**
   731 Copies the content of a binary column, identified by aColumnIndex, to the place refered by aDest parameter.
   732 
   733 If the destination buffer is not big enough, the function will copy as much data as possible and will
   734 return KErrOverflow.
   735 
   736 @param aColumnIndex Column index
   737 @param aDest Refers to the place where the column data will be copied.
   738 
   739 @return KErrNone, if the function completes successfully,
   740                   otherwise one of the other system-wide error codes.
   741 
   742 @panic SqlDb 5 Column index out of bounds.
   743 @panic SqlDb 11 Statement cursor not positioned on a row
   744 */	
   745 TInt CSqlStatementImpl::ColumnBinary(TInt aColumnIndex, TDes8& aDest)
   746 	{
   747 	__ASSERT_ALWAYS((TUint)aColumnIndex < (TUint)iColumnCnt, __SQLPANIC(ESqlPanicBadColumnIndex));
   748 	__ASSERT_ALWAYS(iState == CSqlStatementImpl::EAtRow, __SQLPANIC(ESqlPanicInvalidRow));
   749 	iColumnValBufIt.MoveTo(aColumnIndex);		
   750 	TInt err = KErrNone;
   751 	//The binary column value has not been transferred to the client side if its length is >= KSqlMaxDesLen bytes.
   752 	//In this case an additional call to the server is made to get the column value.
   753 	if(!iColumnValBufIt.IsPresent())
   754 		{
   755 		if(iColumnValBufIt.Type() != ESqlBinary)
   756 			{
   757 			aDest.Zero();
   758 			return err;
   759 			}
   760 		err = iSqlStmtSession.ReadColumnValue(aColumnIndex, aDest);
   761 		}
   762 	else
   763 		{
   764 		TPtrC8 src = iColumnValBufIt.Binary();
   765 		TInt len = src.Length();
   766 		if(len > aDest.MaxLength())
   767 			{
   768 			len = aDest.MaxLength();
   769 			err = KErrOverflow;
   770 			}
   771 		aDest.Copy(src.Ptr(), len);
   772 		}
   773 	return err;
   774 	}
   775 
   776 /**
   777  * Obtain the name of a column after performing a query.
   778  * 
   779  * @param aColumnIndex Column index
   780  * @param aNameDest Descriptor which will be set to column name
   781  * @return KErrNone if successfull or one of the system-wide error codes on error
   782  */
   783 TInt CSqlStatementImpl::ColumnName(TInt aColumnIndex, TPtrC& aNameDest)
   784 	{
   785 	return Index2Name(aColumnIndex, iColumnNameBuf, iColumnCnt, ESqlSrvStmtColumnNames, iColumnNameBufPresent, aNameDest);
   786 	}
   787 
   788 /**
   789  * Obtain the name of a parameter after preparing a DML query.
   790  * 
   791  * @param aParamIndex Parameter index
   792  * @param aNameDest Descriptor which will be set to column name
   793  * @return KErrNone if successfull or one of the system-wide error codes on error
   794  */
   795 TInt CSqlStatementImpl::ParameterName(TInt aParamIndex, TPtrC& aNameDest)
   796 	{
   797 	return Index2Name(aParamIndex, iParamNameBuf, iParamCnt, ESqlSrvStmtParamNames, iParamNameBufPresent, aNameDest);
   798 	}
   799 
   800 
   801 TInt CSqlStatementImpl::Name2Index(const TDesC& aName, RSqlBufFlat& aNameBufFlat, TInt aCount, TSqlSrvFunction aFunction, TBool& aPresent)
   802 	{
   803 	if(aCount == 0)
   804 		{
   805 		return KErrNotFound;	
   806 		}
   807 	TInt err = CheckNameBufPresent(aPresent, aNameBufFlat, aCount, aFunction);
   808 	if ( err != KErrNone ) 
   809 		{
   810 		return err;
   811 		}
   812 	TInt idx = -1;
   813 	aPresent = ETrue;	
   814 	TSqlBufRIterator nameBufIt;
   815 	nameBufIt.Set(aNameBufFlat);
   816 	while(nameBufIt.Next())
   817 		{
   818 		++idx;
   819 		if(::CompareNoCase16(aName, nameBufIt.Text()) == 0)
   820 			{
   821 			return idx;	
   822 			}
   823 		}
   824 	return KErrNotFound;
   825 	}
   826 
   827                    
   828 TInt CSqlStatementImpl::Index2Name(TInt aIndex, RSqlBufFlat& aNameBufFlat, TInt aCount, TSqlSrvFunction aFunction, TBool& aPresent, TPtrC& aColName)
   829 	{
   830 	if( aCount == 0 || aIndex < 0 || aIndex >= aCount ) 
   831 		{
   832 		return KErrNotFound;	
   833 		}
   834 	TInt err = CheckNameBufPresent(aPresent, aNameBufFlat, aCount, aFunction);
   835 	if ( err != KErrNone ) 
   836 		{
   837 		return err;
   838 		}
   839 	aPresent = ETrue;	
   840 	TSqlBufRIterator nameBufIt;
   841 	nameBufIt.Set(aNameBufFlat);
   842 	nameBufIt.MoveTo(aIndex);
   843 	aColName.Set(nameBufIt.Text());
   844 	return KErrNone;
   845 	}
   846 
   847 /**
   848 Ensures that the specified names buffer is present on the client side.
   849 Name buffers are used for transporting parameter and column names.
   850 
   851 @return KErrNone if successfull or one of the system-wide error codes on error
   852 @internalComponent
   853 */
   854 TInt CSqlStatementImpl::CheckNameBufPresent(TBool& aPresent, RSqlBufFlat& aNameBufFlat, TInt aCount, TSqlSrvFunction aFunction)
   855 	{
   856 	if(!aPresent)
   857 		{
   858 		aNameBufFlat.Close();
   859 		TInt err = aNameBufFlat.SetCount(aCount);
   860 		if(err != KErrNone)
   861 			{
   862 			return err;	
   863 			}
   864 		err = iSqlStmtSession.GetNames(aFunction, aNameBufFlat);
   865 		if(err != KErrNone)
   866 			{
   867 			return err;	
   868 			}
   869 		}
   870 	return KErrNone;
   871 	}
   872