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".
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 #include "SqlStatementImpl.h" //CSqlStatementImpl
18 #include "SqlDatabaseImpl.h" //CSqlDatabaseImpl::Session()
22 _LIT(KTextKWD, "TEXT");
23 _LIT(KCharKWD, "CHAR");
24 _LIT(KClobKWD, "CLOB");
26 _LIT(KBinaryKWD,"BINARY");
27 _LIT(KBlobKWD, "BLOB");
29 _LIT(KRealKWD, "REAL");
30 _LIT(KFloatKWD, "FLOAT");
31 _LIT(KDoubleKWD,"DOUBLE");
33 /////////////////////////////////////////////////////////////////////////////////////////////////////////////
34 ////////////////////////// RSqlLongColumnColl ////////////////////////////////////////////
35 /////////////////////////////////////////////////////////////////////////////////////////////////////////////
38 This "comparison" function is used for searching a particular column value, identified by aIndex, in the long column
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.
46 TBool RSqlLongColumnColl::TData::Compare(const TInt* aIndex, const RSqlLongColumnColl::TData& aData)
48 __ASSERT_DEBUG(aIndex != NULL, __SQLPANIC2(ESqlPanicBadArgument));
49 __ASSERT_DEBUG(*aIndex >= 0, __SQLPANIC2(ESqlPanicBadArgument));
50 return *aIndex == aData.iIndex;
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.
64 TInt RSqlLongColumnColl::Append(RSqlLongColumnColl::TColumnReader& aReader, TInt aColumnIndex, TInt aColumnSize)
66 __ASSERT_DEBUG(aColumnIndex >= 0, __SQLPANIC(ESqlPanicBadArgument));
67 __ASSERT_DEBUG(aColumnSize >= KSqlMaxDesLen, __SQLPANIC(ESqlPanicBadArgument));
69 HBufC8* colBuf = HBufC8::New(aColumnSize);
74 TPtr8 ptr = colBuf->Des();
75 TInt err = aReader.Read(aColumnIndex, ptr);
78 err = iValues.Append(RSqlLongColumnColl::TData(aColumnIndex, colBuf));
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.
94 void RSqlLongColumnColl::Invariant() const
96 for(TInt i=iValues.Count()-1;i>=0;--i)
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)
103 __ASSERT_DEBUG(data.iIndex != iValues[j].iIndex, __SQLPANIC(ESqlPanicInternalError));
109 /////////////////////////////////////////////////////////////////////////////////////////////////////////////
110 ////////////////////////// CSqlStatementImpl ////////////////////////////////////////////
111 /////////////////////////////////////////////////////////////////////////////////////////////////////////////
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.
118 NONSHARABLE_STRUCT(TSqlLongColumnReader) : public RSqlLongColumnColl::TColumnReader
120 TSqlLongColumnReader(RSqlStatementSession& aStmtSession) :
121 iStmtSession(aStmtSession)
124 virtual TInt Read(TInt aColumnIndex, TDes8& aBuf)
126 return iStmtSession.ReadColumnValue(aColumnIndex, aBuf);
128 RSqlStatementSession iStmtSession;
131 /////////////////////////////////////////////////////////////////////////////////////////////////////////////
132 ////////////////////////// CSqlStatementImpl ////////////////////////////////////////////
133 /////////////////////////////////////////////////////////////////////////////////////////////////////////////
136 Frees the allocated memory and other resources.
138 CSqlStatementImpl::~CSqlStatementImpl()
140 iDeclaredColumnTypes.Close();
141 iLongColumnColl.Close();
142 iParamValueBuf.Close();
143 iParamNameBuf.Close();
144 iColumnValueBuf.Close();
145 iColumnNameBuf.Close();
146 iSqlStmtSession.Close();
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
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.
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.
164 @see RSqlStatement::Reset()
166 TInt CSqlStatementImpl::Reset()
168 iState = CSqlStatementImpl::EUnknown;
169 iLongColumnColl.Reset();
172 iParamValueBuf.ResetAndMinimize();
174 return iSqlStmtSession.Reset();
178 Executes the prepared DDL/DML SQL statement.
180 If the SQL statement contains parameters, their values will be bound right before the execution.
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.
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.
195 @see RSqlStatement::Exec()
197 TInt CSqlStatementImpl::Exec()
199 iState = CSqlStatementImpl::EUnknown;
203 rc = iSqlStmtSession.Exec();
207 rc = iSqlStmtSession.BindExec(iParamValueBuf);
209 iParamValueBuf.ResetAndMinimize();
215 Executes the prepared DDL/DML SQL statement asynchronously.
217 If the SQL statement contains parameters, their values will be bound right before the execution.
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.
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
232 Note that aStatus may be set with database specific errors categorised as ESqlDbError,
233 and other system-wide error codes.
236 @see RSqlStatement::Exec()
238 void CSqlStatementImpl::Exec(TRequestStatus& aStatus)
240 iState = CSqlStatementImpl::EUnknown;
243 iSqlStmtSession.Exec(aStatus);
247 iSqlStmtSession.BindExec(iParamValueBuf, aStatus);
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.
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.
260 If the SQL statement contains parameters, then their values must be bound before
261 this function is called.
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.
268 The record data can be accessed using the following functions:
269 - CSqlStatementImpl::ColumnType()
270 - CSqlStatementImpl::ColumnSize()
271 - CSqlStatementImpl::Column<Type>()
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
278 KSqlErrGeneral, a run-time error has occured - this function must not
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);
286 @see RSqlStatement::Next()
288 TInt CSqlStatementImpl::Next()
290 iLongColumnColl.Reset();
291 iColumnValueBuf.ResetAndMinimize();
292 iState = CSqlStatementImpl::EUnknown;
296 err = iSqlStmtSession.Next(iColumnValueBuf);
300 err = iSqlStmtSession.BindNext(iParamValueBuf, iColumnValueBuf);
302 iParamValueBuf.ResetAndMinimize();
304 iColumnValBufIt.Set(iColumnValueBuf);
307 iState = CSqlStatementImpl::EAtRow;
313 Implements RSqlStatement::BindNull().
315 @panic SqlDb 5 Parameter index out of bounds.
317 @see RSqlStatement::BindNull()
319 TInt CSqlStatementImpl::BindNull(TInt aParamIndex)
321 __ASSERT_ALWAYS((TUint)aParamIndex < (TUint)iParamCnt, __SQLPANIC(ESqlPanicBadColumnIndex));
323 iParamValBufIt.MoveTo(aParamIndex);
324 iParamValBufIt.SetNull();
329 Implements RSqlStatement::BindInt().
331 @panic SqlDb 5 Parameter index out of bounds.
333 @see RSqlStatement::BindInt()
335 TInt CSqlStatementImpl::BindInt(TInt aParamIndex, TInt aParamValue)
337 __ASSERT_ALWAYS((TUint)aParamIndex < (TUint)iParamCnt, __SQLPANIC(ESqlPanicBadColumnIndex));
339 iParamValBufIt.MoveTo(aParamIndex);
340 return iParamValBufIt.SetInt(aParamValue);
344 Implements RSqlStatement::BindInt64().
346 @panic SqlDb 5 Parameter index out of bounds.
348 @see RSqlStatement::BindInt64()
350 TInt CSqlStatementImpl::BindInt64(TInt aParamIndex, TInt64 aParamValue)
352 __ASSERT_ALWAYS((TUint)aParamIndex < (TUint)iParamCnt, __SQLPANIC(ESqlPanicBadColumnIndex));
354 iParamValBufIt.MoveTo(aParamIndex);
355 return iParamValBufIt.SetInt64(aParamValue);
359 Implements RSqlStatement::BindReal().
361 @panic SqlDb 5 Parameter index out of bounds.
363 @see RSqlStatement::BindReal()
365 TInt CSqlStatementImpl::BindReal(TInt aParamIndex, TReal aParamValue)
367 __ASSERT_ALWAYS((TUint)aParamIndex < (TUint)iParamCnt, __SQLPANIC(ESqlPanicBadColumnIndex));
369 iParamValBufIt.MoveTo(aParamIndex);
370 return iParamValBufIt.SetReal(aParamValue);
374 Implements RSqlStatement::BindText().
376 @panic SqlDb 5 Parameter index out of bounds.
378 @see RSqlStatement::BindText()
380 TInt CSqlStatementImpl::BindText(TInt aParamIndex, const TDesC& aParamText)
382 __ASSERT_ALWAYS((TUint)aParamIndex < (TUint)iParamCnt, __SQLPANIC(ESqlPanicBadColumnIndex));
384 iParamValBufIt.MoveTo(aParamIndex);
385 return iParamValBufIt.SetText(aParamText);
389 Implements RSqlStatement::BindBinary().
391 @panic SqlDb 5 Parameter index out of bounds.
393 @see RSqlStatement::BindBinary()
395 TInt CSqlStatementImpl::BindBinary(TInt aParamIndex, const TDesC8& aParamData)
397 __ASSERT_ALWAYS((TUint)aParamIndex < (TUint)iParamCnt, __SQLPANIC(ESqlPanicBadColumnIndex));
399 iParamValBufIt.MoveTo(aParamIndex);
400 return iParamValBufIt.SetBinary(aParamData);
404 Implements RSqlStatement::BindZeroBlob().
406 @panic SqlDb 5 Parameter index out of bounds.
408 @see RSqlStatement::BindZeroBlob()
410 TInt CSqlStatementImpl::BindZeroBlob(TInt aParamIndex, TInt aBlobSize)
412 __ASSERT_ALWAYS((TUint)aParamIndex < (TUint)iParamCnt, __SQLPANIC(ESqlPanicBadColumnIndex));
414 iParamValBufIt.MoveTo(aParamIndex);
415 return iParamValBufIt.SetZeroBlob(aBlobSize);
419 Implements RSqlStatement::ColumnType().
421 @panic SqlDb 5 Column index out of bounds.
422 @panic SqlDb 11 Statement cursor not positioned on a row.
424 @see RSqlStatement::ColumnType().
426 TSqlColumnType CSqlStatementImpl::ColumnType(TInt aColumnIndex)
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());
435 Implements RSqlStatement::DeclaredColumnType().
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.
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.
444 @panic SqlDb 5 Column index out of bounds.
446 @see RSqlStatement::DeclaredColumnType().
448 TInt CSqlStatementImpl::DeclaredColumnType(TInt aColumnIndex, TSqlColumnType& aColumnType)
450 __ASSERT_ALWAYS((TUint)aColumnIndex < (TUint)iColumnCnt, __SQLPANIC(ESqlPanicBadColumnIndex));
451 if(iDeclaredColumnTypes.Count() == 0) //initialise iDeclaredColumnTypes array if necessary
453 RSqlBufFlat declaredColumnTypeBuf;
454 TInt err = declaredColumnTypeBuf.SetCount(iColumnCnt);
457 declaredColumnTypeBuf.Close();
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;
467 err = declaredColumnTypeBuf.ReAlloc(newSize);
470 declaredColumnTypeBuf.Close();
474 err = iSqlStmtSession.GetDeclColumnTypes(declaredColumnTypeBuf);
477 declaredColumnTypeBuf.Close();
481 err = iDeclaredColumnTypes.Reserve(iColumnCnt);//We know what the array size should be - iColumnCnt
484 declaredColumnTypeBuf.Close();
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);
492 while(declColumnTypeBufIt.Next())
494 TPtrC colTypeName(declColumnTypeBufIt.Text());
495 TSqlColumnType colType = ESqlInt;
496 if(colTypeName.FindF(KCharKWD) >= 0 || colTypeName.FindF(KTextKWD) >= 0 || colTypeName.FindF(KClobKWD) >= 0)
500 else if(colTypeName.FindF(KBinaryKWD) >= 0 || colTypeName.FindF(KBlobKWD) >= 0)
502 colType = ESqlBinary;
504 else if(colTypeName.FindF(KRealKWD) >= 0 || colTypeName.FindF(KFloatKWD) >= 0 || colTypeName.FindF(KDoubleKWD) >= 0)
508 err = iDeclaredColumnTypes.Append(colType);
509 __ASSERT_DEBUG(err == KErrNone, __SQLPANIC(ESqlPanicInternalError));//memory for the array elements has been reserved already
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];
520 Implements RSqlStatement::ColumnSize().
522 @panic SqlDb 5 Column index out of bounds.
523 @panic SqlDb 11 Statement cursor not positioned on a row
525 @see RSqlStatement::ColumnSize().
527 TInt CSqlStatementImpl::ColumnSize(TInt aColumnIndex)
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();
536 Implements RSqlStatement::ColumnInt().
538 @panic SqlDb 5 Column index out of bounds.
539 @panic SqlDb 11 Statement cursor not positioned on a row
541 @see RSqlStatement::ColumnInt().
543 TInt CSqlStatementImpl::ColumnInt(TInt aColumnIndex)
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();
553 Implements RSqlStatement::ColumnInt64().
555 @panic SqlDb 5 Column index out of bounds.
556 @panic SqlDb 11 Statement cursor not positioned on a row
558 @see RSqlStatement::ColumnInt64().
560 TInt64 CSqlStatementImpl::ColumnInt64(TInt aColumnIndex)
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();
570 Implements RSqlStatement::ColumnReal().
572 @panic SqlDb 5 Column index out of bounds.
573 @panic SqlDb 11 Statement cursor not positioned on a row
575 @see RSqlStatement::ColumnReal().
577 TReal CSqlStatementImpl::ColumnReal(TInt aColumnIndex)
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();
587 Return a text (16 bit) descriptor to a text column identified by aColumnIndex.
589 @param aColumnIndex Column index
590 @param aPtr An output parameter which will be set to point to the column data.
592 @return KErrNone, if the function completes successfully,
593 otherwise one of the other system-wide error codes.
595 @panic SqlDb 5 Column index out of bounds.
596 @panic SqlDb 11 Statement cursor not positioned on a row
598 TInt CSqlStatementImpl::ColumnText(TInt aColumnIndex, TPtrC& aPtr)
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())
605 aPtr.Set(iColumnValBufIt.Text());
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)
617 if(!iLongColumnColl.IsPresent(aColumnIndex))
619 TSqlLongColumnReader colReader(iSqlStmtSession);
620 TInt err = iLongColumnColl.Append(colReader, aColumnIndex, iColumnValBufIt.Size() * sizeof(TUint16));
626 aPtr.Set(iLongColumnColl.Text(aColumnIndex));
631 Copies the content of a text column, identified by aColumnIndex, to the place refered by aDest parameter.
633 If the destination buffer is not big enough, the function will copy as much data as possible and will
636 @param aColumnIndex Column index
637 @param aDest Refers to the place where the column data will be copied.
639 @return KErrNone, if the function completes successfully,
640 otherwise one of the other system-wide error codes.
642 @panic SqlDb 5 Column index out of bounds.
643 @panic SqlDb 11 Statement cursor not positioned on a row
645 TInt CSqlStatementImpl::ColumnText(TInt aColumnIndex, TDes& aDest)
647 __ASSERT_ALWAYS((TUint)aColumnIndex < (TUint)iColumnCnt, __SQLPANIC(ESqlPanicBadColumnIndex));
648 __ASSERT_ALWAYS(iState == CSqlStatementImpl::EAtRow, __SQLPANIC(ESqlPanicInvalidRow));
649 iColumnValBufIt.MoveTo(aColumnIndex);
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())
655 if(iColumnValBufIt.Type() != ESqlText)
660 TPtr8 ptr(reinterpret_cast <TUint8*> (const_cast <TUint16*> (aDest.Ptr())), aDest.MaxLength() * sizeof(TUint16));
661 err = iSqlStmtSession.ReadColumnValue(aColumnIndex, ptr);
666 aDest.SetLength(ptr.Length() / sizeof(TUint16));
674 TPtrC src = iColumnValBufIt.Text();
675 TInt len = src.Length();
676 if(len > aDest.MaxLength())
678 len = aDest.MaxLength();
681 aDest.Copy(src.Ptr(), len);
687 Return a binary (8 bit) descriptor to a binary column identified by aColumnIndex.
689 @param aColumnIndex Column index
690 @param aPtr An output parameter which will be set to point to the column data.
692 @return KErrNone, if the function completes successfully,
693 otherwise one of the other system-wide error codes.
695 @panic SqlDb 5 Column index out of bounds.
696 @panic SqlDb 11 Statement cursor not positioned on a row
698 TInt CSqlStatementImpl::ColumnBinary(TInt aColumnIndex, TPtrC8& aPtr)
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())
705 aPtr.Set(iColumnValBufIt.Binary());
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)
714 aPtr.Set(KNullDesC8);
717 if(!iLongColumnColl.IsPresent(aColumnIndex))
719 TSqlLongColumnReader colReader(iSqlStmtSession);
720 TInt err = iLongColumnColl.Append(colReader, aColumnIndex, iColumnValBufIt.Size());
726 aPtr.Set(iLongColumnColl.Binary(aColumnIndex));
731 Copies the content of a binary column, identified by aColumnIndex, to the place refered by aDest parameter.
733 If the destination buffer is not big enough, the function will copy as much data as possible and will
736 @param aColumnIndex Column index
737 @param aDest Refers to the place where the column data will be copied.
739 @return KErrNone, if the function completes successfully,
740 otherwise one of the other system-wide error codes.
742 @panic SqlDb 5 Column index out of bounds.
743 @panic SqlDb 11 Statement cursor not positioned on a row
745 TInt CSqlStatementImpl::ColumnBinary(TInt aColumnIndex, TDes8& aDest)
747 __ASSERT_ALWAYS((TUint)aColumnIndex < (TUint)iColumnCnt, __SQLPANIC(ESqlPanicBadColumnIndex));
748 __ASSERT_ALWAYS(iState == CSqlStatementImpl::EAtRow, __SQLPANIC(ESqlPanicInvalidRow));
749 iColumnValBufIt.MoveTo(aColumnIndex);
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())
755 if(iColumnValBufIt.Type() != ESqlBinary)
760 err = iSqlStmtSession.ReadColumnValue(aColumnIndex, aDest);
764 TPtrC8 src = iColumnValBufIt.Binary();
765 TInt len = src.Length();
766 if(len > aDest.MaxLength())
768 len = aDest.MaxLength();
771 aDest.Copy(src.Ptr(), len);
777 * Obtain the name of a column after performing a query.
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
783 TInt CSqlStatementImpl::ColumnName(TInt aColumnIndex, TPtrC& aNameDest)
785 return Index2Name(aColumnIndex, iColumnNameBuf, iColumnCnt, ESqlSrvStmtColumnNames, iColumnNameBufPresent, aNameDest);
789 * Obtain the name of a parameter after preparing a DML query.
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
795 TInt CSqlStatementImpl::ParameterName(TInt aParamIndex, TPtrC& aNameDest)
797 return Index2Name(aParamIndex, iParamNameBuf, iParamCnt, ESqlSrvStmtParamNames, iParamNameBufPresent, aNameDest);
801 TInt CSqlStatementImpl::Name2Index(const TDesC& aName, RSqlBufFlat& aNameBufFlat, TInt aCount, TSqlSrvFunction aFunction, TBool& aPresent)
807 TInt err = CheckNameBufPresent(aPresent, aNameBufFlat, aCount, aFunction);
808 if ( err != KErrNone )
814 TSqlBufRIterator nameBufIt;
815 nameBufIt.Set(aNameBufFlat);
816 while(nameBufIt.Next())
819 if(::CompareNoCase16(aName, nameBufIt.Text()) == 0)
828 TInt CSqlStatementImpl::Index2Name(TInt aIndex, RSqlBufFlat& aNameBufFlat, TInt aCount, TSqlSrvFunction aFunction, TBool& aPresent, TPtrC& aColName)
830 if( aCount == 0 || aIndex < 0 || aIndex >= aCount )
834 TInt err = CheckNameBufPresent(aPresent, aNameBufFlat, aCount, aFunction);
835 if ( err != KErrNone )
840 TSqlBufRIterator nameBufIt;
841 nameBufIt.Set(aNameBufFlat);
842 nameBufIt.MoveTo(aIndex);
843 aColName.Set(nameBufIt.Text());
848 Ensures that the specified names buffer is present on the client side.
849 Name buffers are used for transporting parameter and column names.
851 @return KErrNone if successfull or one of the system-wide error codes on error
854 TInt CSqlStatementImpl::CheckNameBufPresent(TBool& aPresent, RSqlBufFlat& aNameBufFlat, TInt aCount, TSqlSrvFunction aFunction)
858 aNameBufFlat.Close();
859 TInt err = aNameBufFlat.SetCount(aCount);
864 err = iSqlStmtSession.GetNames(aFunction, aNameBufFlat);