os/persistentdata/persistentstorage/sql/SRC/Server/SqlSrvStatement.cpp
author sl@SLION-WIN7.fritz.box
Fri, 15 Jun 2012 03:10:57 +0200
changeset 0 bde4ae8d615e
permissions -rw-r--r--
First public contribution.
sl@0
     1
// Copyright (c) 2005-2010 Nokia Corporation and/or its subsidiary(-ies).
sl@0
     2
// All rights reserved.
sl@0
     3
// This component and the accompanying materials are made available
sl@0
     4
// under the terms of "Eclipse Public License v1.0"
sl@0
     5
// which accompanies this distribution, and is available
sl@0
     6
// at the URL "http://www.eclipse.org/legal/epl-v10.html".
sl@0
     7
//
sl@0
     8
// Initial Contributors:
sl@0
     9
// Nokia Corporation - initial contribution.
sl@0
    10
//
sl@0
    11
// Contributors:
sl@0
    12
// NTT DOCOMO, INC - Fix for defect 1915 "SQL server panics when using long column type strings"
sl@0
    13
//
sl@0
    14
// Description:
sl@0
    15
//
sl@0
    16
sl@0
    17
#include <utf.h>					//CnvUtfConverter
sl@0
    18
#include <e32math.h>
sl@0
    19
#include "SqliteSymbian.h"			//sqlite3SymbianLastOsError()
sl@0
    20
#include "sqlite3.h"
sl@0
    21
#include "SqlSrvStatement.h"
sl@0
    22
#include "SqlBufIterator.h"			//TSqlBufRIterator
sl@0
    23
#include "SqlSrvResourceProfiler.h"
sl@0
    24
#include "OstTraceDefinitions.h"
sl@0
    25
#ifdef OST_TRACE_COMPILER_IN_USE
sl@0
    26
#include "SqlSrvStatementTraces.h"
sl@0
    27
#endif
sl@0
    28
#include "SqlTraceDef.h"
sl@0
    29
sl@0
    30
//////////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
    31
/////////////////////////////      local const data       ////////////////////////////////////////////
sl@0
    32
//////////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
    33
sl@0
    34
//This is the name prefix which will be given to the nameless parameters.
sl@0
    35
//For example, if the SQL string is:
sl@0
    36
//   SELECT * FROM A WHERE ColA1 = ? AND ColA2 = ?
sl@0
    37
//then the names which will be given to the parameters will be:
sl@0
    38
//"?0" and "?1"
sl@0
    39
_LIT(KNamelessParameter, "?");
sl@0
    40
sl@0
    41
sl@0
    42
/////////////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
    43
//////////////////                HSqlSrvStmtParamBuf                  //////////////////////////////////
sl@0
    44
/////////////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
    45
sl@0
    46
/**
sl@0
    47
Destroys the parameter buffer.
sl@0
    48
sl@0
    49
Virtual method.
sl@0
    50
*/
sl@0
    51
HSqlSrvStmtParamBuf::~HSqlSrvStmtParamBuf()
sl@0
    52
	{
sl@0
    53
	delete iBuf;
sl@0
    54
	}
sl@0
    55
sl@0
    56
/**
sl@0
    57
Binds the parameter value.
sl@0
    58
The buffer can be synch-ed if:
sl@0
    59
 - this is the first synch operation;
sl@0
    60
 - the bound statement object is still alive (not finalized);
sl@0
    61
 - the current object is alive;
sl@0
    62
 - the current object data is retrieved from an IPC stream;
sl@0
    63
 
sl@0
    64
If none of the conditions above is true, the synch operation is no-op.
sl@0
    65
sl@0
    66
Virtual method.
sl@0
    67
*/
sl@0
    68
void HSqlSrvStmtParamBuf::DoSynchL()
sl@0
    69
	{
sl@0
    70
	TBool dontSynch = iSynchDone || !iAlive || iStatementFinalized || iBufType != HSqlSrvStmtParamBuf::EBufIpcStream;
sl@0
    71
	if(dontSynch)
sl@0
    72
		{
sl@0
    73
		return;	
sl@0
    74
		}
sl@0
    75
	iSynchDone = ETrue;
sl@0
    76
	TBufBuf::DoSynchL();
sl@0
    77
	iStatement.BindParamBufL(iParamIndex);
sl@0
    78
	}
sl@0
    79
sl@0
    80
/**
sl@0
    81
Destroys the HSqlSrvStmtParamBuf instance.
sl@0
    82
This method is a no-op if the statement is not finalized yet.
sl@0
    83
sl@0
    84
Virtual method.
sl@0
    85
*/
sl@0
    86
void HSqlSrvStmtParamBuf::DoRelease()
sl@0
    87
 	{
sl@0
    88
	iAlive = EFalse;
sl@0
    89
 	if(iStatementFinalized)
sl@0
    90
 		{//The bound statement has been finalized - destroy the current object then.
sl@0
    91
 		delete this;
sl@0
    92
 		}
sl@0
    93
	}
sl@0
    94
sl@0
    95
/**
sl@0
    96
This function is called by the bound statement object to notify the current HSqlSrvStmtParamBuf object that the
sl@0
    97
bound statement is about to be finalized. That means, when the "stream close" operation on the client side 
sl@0
    98
makes an attempt to synch the HSqlSrvStmtParamBuf object, no attempt should be made to bound the parameter data,
sl@0
    99
because the statement object is gone.
sl@0
   100
After this call the bound statement objects ceases to exist.
sl@0
   101
sl@0
   102
Actions, performed by this method:
sl@0
   103
 - if the buffer type is "simple bind", the buffer will be destroyed. No reason to keep it alive, there is no bound IPC
sl@0
   104
   stream object on the client side;
sl@0
   105
 - if the buffer type is an IPC stream buffer and the buffer is alive, that means: the bound statement object is about to be
sl@0
   106
   finalized, but there is a bound client side IPC stream object that is still alive. In this case the buffer won't be destroyed,
sl@0
   107
   but will be "told" that the bound statement is finalized, so when the client side IPC stream is closed, this object will get destroyed;
sl@0
   108
 - if the buffer type is an IPC stream buffer and the buffer is "dead", that means there is no bound IPC stream object on the client
sl@0
   109
   side and it is safe to destroy the buffer;
sl@0
   110
sl@0
   111
*/ 	
sl@0
   112
void HSqlSrvStmtParamBuf::NotifyStatementFinalized()
sl@0
   113
	{
sl@0
   114
	iStatementFinalized = ETrue;
sl@0
   115
	if(iBufType == HSqlSrvStmtParamBuf::EBufSimpleBind || !iAlive)
sl@0
   116
		{
sl@0
   117
		DoRelease();	
sl@0
   118
		}
sl@0
   119
	}
sl@0
   120
sl@0
   121
//////////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
   122
/////////////////////////////   CSqlSrvStatement class    ////////////////////////////////////////////
sl@0
   123
//////////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
   124
sl@0
   125
/**
sl@0
   126
Creates a new CSqlSrvStatement instance.
sl@0
   127
sl@0
   128
The created CSqlSrvStatement instance will be placed in the cleanup stack.
sl@0
   129
sl@0
   130
@param aDbHandle The database handle
sl@0
   131
@param aSqlStmt 16-bit SQL statement, zero-terminated string
sl@0
   132
@param aColumnCount Output parameter. It will be initialized with the column count.
sl@0
   133
@param aParamCount Output parameter. It will be initialized with the parameter count.
sl@0
   134
sl@0
   135
@return A pointer to the created CSqlSrvStatement instance.
sl@0
   136
sl@0
   137
@leave KErrNoMemory, an out of memory condition has occurred;
sl@0
   138
	   KErrArgument, bad argument, for example - the SQL string contains more than one SQL statements.
sl@0
   139
                  Note that the function may also leave with some other database specific 
sl@0
   140
                  errors categorised as ESqlDbError.
sl@0
   141
sl@0
   142
@panic SqlDb 4 In _DEBUG mode if aSqlStmt is not zero-terminated string.
sl@0
   143
*/	
sl@0
   144
CSqlSrvStatement* CSqlSrvStatement::NewLC(sqlite3* aDbHandle, const TDesC16& aSqlStmt, TInt& aColumnCount, TInt& aParamCount)
sl@0
   145
	{
sl@0
   146
    __SQLTRACE_INTERNALSEXPR(TPtrC sqlprnptr(aSqlStmt.Left(aSqlStmt.Length() - 1)));
sl@0
   147
    SQL_TRACE_INTERNALS(OstTraceExt2(TRACE_INTERNALS, CSQLSRVSTATEMENT_NEWLC_ENTRY, "Entry;0;CSqlSrvStatement::NewLC-16;aDbHandle=0x%X;aSqlStmt=%S", (TUint)aDbHandle, __SQLPRNSTR(sqlprnptr)));
sl@0
   148
	__ASSERT_DEBUG(aSqlStmt.Length() > 0 ? (TInt)aSqlStmt[aSqlStmt.Length() - 1] == 0 : ETrue, __SQLPANIC2(ESqlPanicBadArgument));
sl@0
   149
	
sl@0
   150
	CSqlSrvStatement* self = new (ELeave) CSqlSrvStatement;
sl@0
   151
	CleanupStack::PushL(self);
sl@0
   152
	self->ConstructL(aDbHandle, aSqlStmt);
sl@0
   153
	aColumnCount = self->iColumnCount;
sl@0
   154
	aParamCount = self->iParamCount;
sl@0
   155
	SQL_TRACE_INTERNALS(OstTraceExt4(TRACE_INTERNALS, CSQLSRVSTATEMENT_NEWLC_EXIT, "Exit;0x%X;CSqlSrvStatement::NewLC-16;iStmtHandle=0x%X;aColumnCount=%d;aParamCount=%d", (TUint)self, (TUint)self->iStmtHandle, aColumnCount, aParamCount));
sl@0
   156
	return self;
sl@0
   157
	}
sl@0
   158
	
sl@0
   159
/**
sl@0
   160
Creates a new CSqlSrvStatement instance.
sl@0
   161
sl@0
   162
The created CSqlSrvStatement instance will be placed in the cleanup stack.
sl@0
   163
sl@0
   164
@param aDbHandle The database handle
sl@0
   165
@param aSqlStmt 8-bit SQL statement, zero-terminated string
sl@0
   166
@param aColumnCount Output parameter. It will be initialized with the column count.
sl@0
   167
@param aParamCount Output parameter. It will be initialized with the parameter count.
sl@0
   168
sl@0
   169
@return A pointer to the created CSqlSrvStatement instance.
sl@0
   170
sl@0
   171
@leave KErrNoMemory, an out of memory condition has occurred;
sl@0
   172
	   KErrArgument, bad argument, for example - the SQL string contains more than one SQL statements.
sl@0
   173
                  Note that the function may also leave with some other database specific 
sl@0
   174
                  errors categorised as ESqlDbError.
sl@0
   175
sl@0
   176
@panic SqlDb 4 In _DEBUG mode if aSqlStmt is not zero-terminated string.
sl@0
   177
*/	
sl@0
   178
CSqlSrvStatement* CSqlSrvStatement::NewLC(sqlite3* aDbHandle, const TDesC8& aSqlStmt, TInt& aColumnCount, TInt& aParamCount)
sl@0
   179
	{
sl@0
   180
    __SQLTRACE_INTERNALSEXPR(TPtrC8 sqlprnptr(aSqlStmt.Left(aSqlStmt.Length() - 1)));
sl@0
   181
	__SQLTRACE_INTERNALSVAR(TBuf<100> des16prnbuf);
sl@0
   182
    SQL_TRACE_INTERNALS(OstTraceExt2(TRACE_INTERNALS, CSQLSRVSTATEMENT_NEWLC_ENTRY2, "Entry;0;CSqlSrvStatement::NewLC-8;aDbHandle=0x%X;aSqlStmt=%s", (TUint)aDbHandle, __SQLPRNSTR8(sqlprnptr, des16prnbuf)));
sl@0
   183
	__ASSERT_DEBUG(aSqlStmt.Length() > 0 ? (TInt)aSqlStmt[aSqlStmt.Length() - 1] == 0 : ETrue, __SQLPANIC2(ESqlPanicBadArgument));
sl@0
   184
sl@0
   185
	CSqlSrvStatement* self = new (ELeave) CSqlSrvStatement;
sl@0
   186
	CleanupStack::PushL(self);
sl@0
   187
	self->ConstructL(aDbHandle, aSqlStmt);
sl@0
   188
	aColumnCount = self->iColumnCount;
sl@0
   189
	aParamCount = self->iParamCount;
sl@0
   190
	SQL_TRACE_INTERNALS(OstTraceExt4(TRACE_INTERNALS, CSQLSRVSTATEMENT_NEWLC_EXIT2, "Exit;0x%X;CSqlSrvStatement::NewLC-8;iStmtHandle=0x%X;aColumnCount=%d;aParamCount=%d", (TUint)self, (TUint)self->iStmtHandle, aColumnCount, aParamCount));
sl@0
   191
	return self;
sl@0
   192
	}
sl@0
   193
	
sl@0
   194
/**
sl@0
   195
Destroys the allocated by CSqlSrvStatement instance memory and other resources.
sl@0
   196
*/	
sl@0
   197
CSqlSrvStatement::~CSqlSrvStatement()
sl@0
   198
	{
sl@0
   199
	SQL_TRACE_INTERNALS(OstTraceExt2(TRACE_INTERNALS, CSQLSRVSTATEMENT_CSQLSRVSTATEMENT2_ENTRY, "Entry;0x%X;CSqlSrvStatement::~CSqlSrvStatement;iStmtHandle=0x%X", (TUint)this, (TUint)iStmtHandle));
sl@0
   200
	DestroyParamBufArray();
sl@0
   201
	iBufFlat.Close();
sl@0
   202
	if(iStmtHandle)
sl@0
   203
		{
sl@0
   204
#ifdef SYMBIAN_USE_SQLITE_VERSION_3_6_4
sl@0
   205
		__SQLTRACE_INTERNALSEXPR(TInt scanCount = sqlite3_stmt_status(iStmtHandle, SQLITE_STMTSTATUS_FULLSCAN_STEP, ETrue));
sl@0
   206
		__SQLTRACE_INTERNALSEXPR(TInt sortCount = sqlite3_stmt_status(iStmtHandle, SQLITE_STMTSTATUS_SORT, ETrue));
sl@0
   207
		SQL_TRACE_INTERNALS(OstTraceExt3(TRACE_INTERNALS, CSQLSRVSTATEMENT_CSQLSRVSTATEMENT2, "0x%X;CSqlSrvStatement::~CSqlSrvStatement;scan count=%d;sort count=%d", (TUint)this, scanCount, sortCount));
sl@0
   208
#endif		
sl@0
   209
		(void)sqlite3_finalize(iStmtHandle);
sl@0
   210
		}
sl@0
   211
	SQL_TRACE_INTERNALS(OstTrace1(TRACE_INTERNALS, CSQLSRVSTATEMENT_CSQLSRVSTATEMENT2_EXIT, "Exit;0x%X;CSqlSrvStatement::~CSqlSrvStatement", (TUint)this));
sl@0
   212
	}
sl@0
   213
sl@0
   214
/**
sl@0
   215
Sets SQL statement parameter values.
sl@0
   216
sl@0
   217
Only parameters, whose values are set by the client, will be processed.
sl@0
   218
sl@0
   219
@param aParamBuf Flat buffer with parameter values.
sl@0
   220
sl@0
   221
@leave KErrArgument, unknown parameter type;
sl@0
   222
	   KSqlErrStmtExpired, statement handle expired.
sl@0
   223
                  Note that the function may also leave with some other database specific 
sl@0
   224
                  errors categorised as ESqlDbError.
sl@0
   225
sl@0
   226
@panic SqlDb 2 In _DEBUG mode. Invalid (not created) CSqlSrvStatement object.
sl@0
   227
*/	
sl@0
   228
void CSqlSrvStatement::BindL(const RSqlBufFlat& aParamBuf)
sl@0
   229
	{
sl@0
   230
	__ASSERT_DEBUG(iStmtHandle != NULL, __SQLPANIC(ESqlPanicInvalidObj));
sl@0
   231
	
sl@0
   232
	(void)sqlite3SymbianLastOsError();//clear last OS error
sl@0
   233
sl@0
   234
	TSqlBufRIterator it;
sl@0
   235
	it.Set(aParamBuf);
sl@0
   236
	TInt prmIdx = 0;
sl@0
   237
	
sl@0
   238
	TInt err = SQLITE_OK;
sl@0
   239
	while(it.Next() && err == SQLITE_OK)
sl@0
   240
		{
sl@0
   241
		++prmIdx;//the first SQLITE parameter index is 1
sl@0
   242
		if(it.IsPresent())
sl@0
   243
			{
sl@0
   244
			switch(it.Type())
sl@0
   245
 				{
sl@0
   246
				case ESqlInt:
sl@0
   247
					err = sqlite3_bind_int(iStmtHandle, prmIdx, it.Int());
sl@0
   248
					break;
sl@0
   249
				case ESqlInt64:
sl@0
   250
					err = sqlite3_bind_int64(iStmtHandle, prmIdx, it.Int64());
sl@0
   251
					break;
sl@0
   252
				case ESqlReal: 
sl@0
   253
					err = sqlite3_bind_double(iStmtHandle, prmIdx, it.Real());
sl@0
   254
					break;
sl@0
   255
				case ESqlText: 
sl@0
   256
					//SQLITE_STATIC is used as an argument, because the text data will be kept and can be used by the next bind call
sl@0
   257
					{
sl@0
   258
					TPtrC text = it.Text();
sl@0
   259
					TPtrC8 prmDataCopy(reinterpret_cast <const TUint8*> (text.Ptr()), text.Length() * sizeof(TUint16));
sl@0
   260
					prmDataCopy.Set(CopyAndStoreParamL(prmIdx - 1, HSqlSrvStmtParamBuf::EText16, prmDataCopy));
sl@0
   261
					err = sqlite3_bind_text16(iStmtHandle, prmIdx, prmDataCopy.Ptr(), prmDataCopy.Length(), SQLITE_STATIC);
sl@0
   262
					}
sl@0
   263
					break;
sl@0
   264
				case ESqlBinary:
sl@0
   265
					//SQLITE_STATIC is used as an argument, because the blob data will be kept and can be used by the next bind call
sl@0
   266
					{
sl@0
   267
					TPtrC8 prmDataCopy = CopyAndStoreParamL(prmIdx - 1, HSqlSrvStmtParamBuf::EBinary, it.Binary());
sl@0
   268
					err = sqlite3_bind_blob(iStmtHandle, prmIdx, prmDataCopy.Ptr(), prmDataCopy.Length(), SQLITE_STATIC);
sl@0
   269
					}
sl@0
   270
					break;
sl@0
   271
				case ESqlNull:
sl@0
   272
					err = sqlite3_bind_null(iStmtHandle, prmIdx);
sl@0
   273
					break;
sl@0
   274
				case ESqlZeroBlob:
sl@0
   275
					err = sqlite3_bind_zeroblob(iStmtHandle, prmIdx, it.Int());
sl@0
   276
					break;
sl@0
   277
				default:
sl@0
   278
					__SQLLEAVE(KErrArgument);//unknown parameter type
sl@0
   279
					break;
sl@0
   280
				}
sl@0
   281
			}//end of - if(it.IsPresent())
sl@0
   282
		}//end of - while(it.Next() && err == SQLITE_OK)
sl@0
   283
	err = ::Sql2OsErrCode(err, sqlite3SymbianLastOsError());
sl@0
   284
	__SQLLEAVE_IF_ERROR(err);
sl@0
   285
	}
sl@0
   286
sl@0
   287
/**
sl@0
   288
Collects column names in a flat buffer and returns a reference to the buffer. 
sl@0
   289
sl@0
   290
@return A const reference to a flat buffer containing the column names.
sl@0
   291
sl@0
   292
@leave KErrNoMemory, an out of memory condition has occurred.
sl@0
   293
sl@0
   294
@panic SqlDb 2 In _DEBUG mode. Invalid (not created) CSqlSrvStatement object.
sl@0
   295
*/	
sl@0
   296
const RSqlBufFlat& CSqlSrvStatement::ColumnNamesL()
sl@0
   297
	{
sl@0
   298
	__ASSERT_DEBUG(iStmtHandle != NULL, __SQLPANIC(ESqlPanicInvalidObj));
sl@0
   299
	iBufFlatType = static_cast <TSqlBufFlatType> (-1);
sl@0
   300
	__SQLLEAVE_IF_ERROR(iBufFlat.SetCount(iColumnCount));
sl@0
   301
	TSqlBufWIterator it;
sl@0
   302
	it.Set(iBufFlat);
sl@0
   303
	TInt colIdx = -1;
sl@0
   304
	while(it.Next())
sl@0
   305
		{
sl@0
   306
		++colIdx;//the first SQLITE column index is 0
sl@0
   307
		const TUint16* name = reinterpret_cast <const TUint16*> (__SQLLEAVE_IF_NULL(const_cast <void*> (sqlite3_column_name16(iStmtHandle, colIdx))));
sl@0
   308
		TPtrC ptr(name, User::StringLength(name));
sl@0
   309
		__SQLLEAVE_IF_ERROR(it.SetText(ptr));
sl@0
   310
		}
sl@0
   311
	iBufFlatType = ESqlColumnNamesBuf;
sl@0
   312
	SQLPROFILER_REPORT_ALLOC(iBufFlat.MaxSize());
sl@0
   313
	return iBufFlat;
sl@0
   314
	}
sl@0
   315
sl@0
   316
/**
sl@0
   317
Collects parameter names in a flat buffer and returns a reference to the buffer. 
sl@0
   318
sl@0
   319
@return A const reference to a flat buffer containing the parameter names.
sl@0
   320
sl@0
   321
@leave KErrNoMemory, an out of memory condition has occurred.
sl@0
   322
sl@0
   323
@panic SqlDb 2 In _DEBUG mode. Invalid (not created) CSqlSrvStatement object.
sl@0
   324
*/	
sl@0
   325
const RSqlBufFlat& CSqlSrvStatement::ParamNamesL()
sl@0
   326
	{
sl@0
   327
	__ASSERT_DEBUG(iStmtHandle != NULL, __SQLPANIC(ESqlPanicInvalidObj));
sl@0
   328
	iBufFlatType = static_cast <TSqlBufFlatType> (-1);
sl@0
   329
	__SQLLEAVE_IF_ERROR(iBufFlat.SetCount(iParamCount));
sl@0
   330
	TSqlBufWIterator it;
sl@0
   331
	it.Set(iBufFlat);
sl@0
   332
	TInt prmIdx = 0;
sl@0
   333
	while(it.Next())
sl@0
   334
		{
sl@0
   335
		++prmIdx;//the first SQLITE parameter index is 1
sl@0
   336
		const TUint8* name8 = reinterpret_cast <const TUint8*> (sqlite3_bind_parameter_name(iStmtHandle, prmIdx));
sl@0
   337
		if(name8)
sl@0
   338
			{
sl@0
   339
			HBufC* name = CnvUtfConverter::ConvertToUnicodeFromUtf8L(TPtrC8(name8, User::StringLength(name8)));
sl@0
   340
			TInt err = it.SetText(name->Des());
sl@0
   341
			delete name;
sl@0
   342
			__SQLLEAVE_IF_ERROR(err);
sl@0
   343
			}
sl@0
   344
		else //nameless parameter case
sl@0
   345
			{
sl@0
   346
			//The parameter name in this case will be formatted as "?<num>", where <num> is the parameter index.
sl@0
   347
			TBuf<5> prmName;
sl@0
   348
			prmName.Append(KNamelessParameter);
sl@0
   349
			prmName.AppendNum((TInt64)(prmIdx - 1));
sl@0
   350
			__SQLLEAVE_IF_ERROR(it.SetText(prmName));
sl@0
   351
			}
sl@0
   352
		}
sl@0
   353
	iBufFlatType = ESqlParamNamesBuf;
sl@0
   354
	SQLPROFILER_REPORT_ALLOC(iBufFlat.MaxSize());
sl@0
   355
	return iBufFlat;
sl@0
   356
	}
sl@0
   357
	
sl@0
   358
/**
sl@0
   359
Collects the column values in a flat buffer and returns a reference to the buffer. 
sl@0
   360
sl@0
   361
@leave KErrNoMemory, an out of memory condition has occurred.
sl@0
   362
sl@0
   363
@return A const reference to a flat buffer containing the column values.
sl@0
   364
sl@0
   365
@panic SqlDb 2 In _DEBUG mode. Invalid (not created) CSqlSrvStatement object
sl@0
   366
*/	
sl@0
   367
const RSqlBufFlat& CSqlSrvStatement::ColumnValuesL()
sl@0
   368
	{
sl@0
   369
	__ASSERT_DEBUG(iStmtHandle != NULL, __SQLPANIC(ESqlPanicInvalidObj));
sl@0
   370
	
sl@0
   371
	iBufFlatType = static_cast <TSqlBufFlatType> (-1);
sl@0
   372
	iBufFlat.SetCount(iColumnCount);
sl@0
   373
	TSqlBufWIterator it;
sl@0
   374
	it.Set(iBufFlat);
sl@0
   375
	TInt colIdx = -1;
sl@0
   376
	
sl@0
   377
	while(it.Next())
sl@0
   378
		{
sl@0
   379
		++colIdx;//the first SQLITE column index is 0
sl@0
   380
		TInt colType = sqlite3_column_type(iStmtHandle, colIdx);
sl@0
   381
		switch(colType)
sl@0
   382
			{
sl@0
   383
			case SQLITE_INTEGER:
sl@0
   384
				{
sl@0
   385
				TInt64 val = sqlite3_column_int64(iStmtHandle, colIdx);
sl@0
   386
				__SQLLEAVE_IF_ERROR(val == TInt64(TInt32(val)) ? it.SetInt(static_cast <TInt> (val)) : it.SetInt64(val));
sl@0
   387
				}
sl@0
   388
				break;
sl@0
   389
			case SQLITE_FLOAT:
sl@0
   390
				__SQLLEAVE_IF_ERROR(it.SetReal(sqlite3_column_double(iStmtHandle, colIdx)));
sl@0
   391
				break;
sl@0
   392
			case SQLITE_TEXT:
sl@0
   393
				{
sl@0
   394
				TInt charLength = (TUint)sqlite3_column_bytes16(iStmtHandle, colIdx) / sizeof(TUint16);
sl@0
   395
                //"charLength == 0" - this might be an indication of an "out of memory" problem, if the column text is in UTF8 format. 
sl@0
   396
                //(sqlite3_column_bytes16() may allocate memory for UTF8->UTF16 conversion)
sl@0
   397
				if(charLength == 0 && sqlite3_errcode(sqlite3_db_handle(iStmtHandle)) == SQLITE_NOMEM)
sl@0
   398
                    {
sl@0
   399
                    __SQLLEAVE(KErrNoMemory);
sl@0
   400
                    }
sl@0
   401
				if(charLength >= KSqlMaxDesLen)
sl@0
   402
					{
sl@0
   403
					it.SetAsNotPresent(ESqlText, charLength);
sl@0
   404
					}
sl@0
   405
				else
sl@0
   406
					{//sqlite3_column_bytes16() already allocated the needed memory if a UTF8->UTF16 conversion
sl@0
   407
                     //had to be performed. The sqlite3_column_text16() on the next line is guaranteed to succeed.
sl@0
   408
					const TUint16* text = reinterpret_cast <const TUint16*> (sqlite3_column_text16(iStmtHandle, colIdx));
sl@0
   409
					__ASSERT_DEBUG(text != NULL, __SQLPANIC(ESqlPanicInternalError));
sl@0
   410
					__SQLLEAVE_IF_ERROR(it.SetText(TPtrC16(text, charLength)));
sl@0
   411
					}
sl@0
   412
				}
sl@0
   413
				break;
sl@0
   414
			case SQLITE_BLOB:
sl@0
   415
				{
sl@0
   416
				TInt byteLength = sqlite3_column_bytes(iStmtHandle, colIdx);
sl@0
   417
				if(byteLength >= KSqlMaxDesLen)
sl@0
   418
					{
sl@0
   419
					it.SetAsNotPresent(ESqlBinary, byteLength);
sl@0
   420
					}
sl@0
   421
				else
sl@0
   422
					{
sl@0
   423
					__SQLLEAVE_IF_ERROR(it.SetBinary(TPtrC8(reinterpret_cast <const TUint8*> (sqlite3_column_blob(iStmtHandle, colIdx)), byteLength)));
sl@0
   424
					}
sl@0
   425
				}
sl@0
   426
				break;
sl@0
   427
			case SQLITE_NULL:
sl@0
   428
				it.SetNull();
sl@0
   429
				break;
sl@0
   430
			default:
sl@0
   431
				__ASSERT_DEBUG(EFalse, __SQLPANIC(ESqlPanicInternalError));
sl@0
   432
				break;
sl@0
   433
			}//end of switch(...)
sl@0
   434
		}//end of - while(it.Next())
sl@0
   435
	iBufFlatType = ESqlColumnValuesBuf;
sl@0
   436
	SQLPROFILER_REPORT_ALLOC(iBufFlat.MaxSize());
sl@0
   437
	return iBufFlat;
sl@0
   438
	}
sl@0
   439
sl@0
   440
/**
sl@0
   441
This method sets aColumnSource parameter to point to the column data.
sl@0
   442
sl@0
   443
@param aColumnIndex Column Index, zero based. 
sl@0
   444
@param aColumnSource Output parameter. It is set to point to the column data.
sl@0
   445
sl@0
   446
@return KErrNone, the operation completed successfully;
sl@0
   447
		KErrArgument, the refered by aColumnIndex index column is not a binary or text column.
sl@0
   448
sl@0
   449
@panic SqlDb 2 In _DEBUG mode. Invalid (not created) CSqlSrvStatement object.
sl@0
   450
*/	
sl@0
   451
TInt CSqlSrvStatement::ColumnSource(TInt aColumnIndex, TPtrC8& aColumnSource) const
sl@0
   452
	{
sl@0
   453
	__ASSERT_DEBUG(iStmtHandle != NULL, __SQLPANIC(ESqlPanicInvalidObj));
sl@0
   454
	TInt colType = sqlite3_column_type(iStmtHandle, aColumnIndex);
sl@0
   455
	if(colType == SQLITE_TEXT)
sl@0
   456
        {//Since the first called function after the Next() operation is always CSqlSrvStatement::ColumnValuesL(), then
sl@0
   457
         //sqlite3_column_bytes16() (called from  ColumnValuesL()) already allocated the needed memory if a UTF8->UTF16 
sl@0
   458
         //conversion had to be performed. The sqlite3_column_text16() on the next line is guaranteed to succeed.
sl@0
   459
		const void* text = sqlite3_column_text16(iStmtHandle, aColumnIndex);
sl@0
   460
		__ASSERT_DEBUG(text != NULL, __SQLPANIC2(ESqlPanicInternalError));
sl@0
   461
 		TInt length  = sqlite3_column_bytes16(iStmtHandle, aColumnIndex);
sl@0
   462
		aColumnSource.Set(reinterpret_cast <const TUint8*> (text), length);
sl@0
   463
		}
sl@0
   464
	else if(colType == SQLITE_BLOB)
sl@0
   465
		{
sl@0
   466
		const void* data = sqlite3_column_blob(iStmtHandle, aColumnIndex);
sl@0
   467
		TInt length  = sqlite3_column_bytes(iStmtHandle, aColumnIndex);
sl@0
   468
		aColumnSource.Set(reinterpret_cast <const TUint8*> (data), length);
sl@0
   469
		}
sl@0
   470
	else
sl@0
   471
		{
sl@0
   472
		return KErrArgument;	
sl@0
   473
		}
sl@0
   474
	return KErrNone;
sl@0
   475
	}
sl@0
   476
sl@0
   477
/**
sl@0
   478
Retrieves from the SQLITE library columns and parameters count.
sl@0
   479
sl@0
   480
@panic SqlDb 4 In _DEBUG mode. aDbHandle is NULL.
sl@0
   481
*/	
sl@0
   482
void CSqlSrvStatement::DoCommonConstructL()
sl@0
   483
	{
sl@0
   484
	__ASSERT_DEBUG(iStmtHandle != NULL, __SQLPANIC(ESqlPanicInvalidObj));
sl@0
   485
	iColumnCount = sqlite3_column_count(iStmtHandle);
sl@0
   486
	iParamCount = sqlite3_bind_parameter_count(iStmtHandle);
sl@0
   487
	__SQLLEAVE_IF_ERROR(iBufFlat.SetCount(Max(iColumnCount, iParamCount)));
sl@0
   488
	}
sl@0
   489
sl@0
   490
/**
sl@0
   491
Destroys the parameter buffer array (used for text or binary parameters).
sl@0
   492
Before the array destruction, each array member is notified that the statement is about to be finalized.
sl@0
   493
*/
sl@0
   494
void CSqlSrvStatement::DestroyParamBufArray()
sl@0
   495
	{
sl@0
   496
	TInt idx = iParamBufArray.Count();
sl@0
   497
	while(--idx >= 0)
sl@0
   498
		{
sl@0
   499
		if(iParamBufArray[idx])
sl@0
   500
			{
sl@0
   501
			iParamBufArray[idx]->NotifyStatementFinalized();	
sl@0
   502
			}
sl@0
   503
		}
sl@0
   504
	iParamBufArray.Close();
sl@0
   505
	}
sl@0
   506
	
sl@0
   507
/**
sl@0
   508
Binds a streamed text or binary parameter value.
sl@0
   509
sl@0
   510
@param aParamIndex The text/binary parameter index
sl@0
   511
sl@0
   512
@leave KErrNoMemory, an out of memory condition has occurred;
sl@0
   513
	   KSqlErrStmtExpired, statement handle has expired.
sl@0
   514
                  Note that the function may also leave with some other database specific 
sl@0
   515
                  errors categorised as ESqlDbError.
sl@0
   516
sl@0
   517
@panic SqlDb 2 In _DEBUG mode. Invalid (not created) CSqlSrvStatement object.
sl@0
   518
@panic SqlDb 4 In _DEBUG mode. No parameter buffer has been created yet for this parameter.
sl@0
   519
@panic SqlDb 4 In _DEBUG mode. Parameter index out of bounds.
sl@0
   520
*/
sl@0
   521
void CSqlSrvStatement::BindParamBufL(TInt aParamIndex)
sl@0
   522
	{
sl@0
   523
	__ASSERT_DEBUG(iStmtHandle != NULL, __SQLPANIC(ESqlPanicInvalidObj));
sl@0
   524
	__ASSERT_DEBUG(aParamIndex >= 0 && aParamIndex < sqlite3_bind_parameter_count(iStmtHandle), __SQLPANIC(ESqlPanicBadArgument));
sl@0
   525
	__ASSERT_DEBUG(aParamIndex < iParamBufArray.Count(), __SQLPANIC(ESqlPanicBadArgument));
sl@0
   526
	__ASSERT_DEBUG(iParamBufArray[aParamIndex] != NULL, __SQLPANIC(ESqlPanicBadArgument));
sl@0
   527
	(void)sqlite3SymbianLastOsError();//clear last OS error
sl@0
   528
	//Bind the parameter value.
sl@0
   529
	//SQLITE_STATIC is used as an argument, because the text/blob data will be kept and can be used by the next bind call
sl@0
   530
	HSqlSrvStmtParamBuf& paramBuf = *iParamBufArray[aParamIndex];
sl@0
   531
	const TPtrC8 paramData(paramBuf.Data());
sl@0
   532
	SQLPROFILER_REPORT_ALLOC(paramData.Length());
sl@0
   533
	TInt err = KErrNone;
sl@0
   534
	++aParamIndex;//SQLite uses positive parameter indexes, the SQL server - parameter indexes begin from 0
sl@0
   535
	switch(paramBuf.DataType())
sl@0
   536
		{
sl@0
   537
		case HSqlSrvStmtParamBuf::EText16:
sl@0
   538
			//sqlite3_bind_text16() expects 4-th argument to be the bytes count, not the characters count.
sl@0
   539
			err = sqlite3_bind_text16(iStmtHandle, aParamIndex, paramData.Ptr(), paramData.Length(), SQLITE_STATIC);
sl@0
   540
			break;
sl@0
   541
		case HSqlSrvStmtParamBuf::EBinary:
sl@0
   542
		default:
sl@0
   543
			err = sqlite3_bind_blob(iStmtHandle, aParamIndex, paramData.Ptr(), paramData.Length(), SQLITE_STATIC);
sl@0
   544
			break;
sl@0
   545
		}
sl@0
   546
	err = ::Sql2OsErrCode(err, sqlite3SymbianLastOsError());
sl@0
   547
	__SQLLEAVE_IF_ERROR(err);
sl@0
   548
	}
sl@0
   549
sl@0
   550
/**
sl@0
   551
@return Represents the content of the column identified by aColIdx as integer value.
sl@0
   552
		If the current column type does not refer to an integer, then 
sl@0
   553
		the function will do a data conversion as described in the table which can be found
sl@0
   554
		in SqlDb.h file.
sl@0
   555
@see RSqlStatement
sl@0
   556
sl@0
   557
@panic SqlDb 2 In _DEBUG mode. Invalid (not created) CSqlSrvStatement object.
sl@0
   558
@panic SqlDb 4 In _DEBUG mode. Invalid aColIdx value.
sl@0
   559
*/
sl@0
   560
TInt CSqlSrvStatement::ColumnInt(TInt aColIdx) const
sl@0
   561
	{
sl@0
   562
	__ASSERT_DEBUG(iStmtHandle != NULL, __SQLPANIC(ESqlPanicInvalidObj));
sl@0
   563
	__ASSERT_DEBUG((TUint)aColIdx < iColumnCount, __SQLPANIC(ESqlPanicBadArgument));
sl@0
   564
	TInt colType = sqlite3_column_type(iStmtHandle, aColIdx);
sl@0
   565
	switch(colType)
sl@0
   566
		{
sl@0
   567
		case SQLITE_FLOAT:
sl@0
   568
			{
sl@0
   569
			TReal roundVal;
sl@0
   570
			TInt err = Math::Round(roundVal, sqlite3_column_double(iStmtHandle, aColIdx), 0);
sl@0
   571
			if(err != KErrNone)
sl@0
   572
				{
sl@0
   573
				return KMinTInt;
sl@0
   574
				}
sl@0
   575
			TRealX val(roundVal);
sl@0
   576
			return static_cast <TInt> (val);
sl@0
   577
			}
sl@0
   578
		case SQLITE_NULL:
sl@0
   579
		case SQLITE_TEXT:
sl@0
   580
		case SQLITE_BLOB:
sl@0
   581
			return 0;
sl@0
   582
		default:			//int, int64
sl@0
   583
			{
sl@0
   584
			TInt64 val = sqlite3_column_int64(iStmtHandle, aColIdx);
sl@0
   585
			return val == (TInt)val ? (TInt)val : (val < KMinTInt ? KMinTInt : KMaxTInt);
sl@0
   586
			}
sl@0
   587
		}
sl@0
   588
	}
sl@0
   589
	
sl@0
   590
/**
sl@0
   591
@return Represents the content of the column identified by aColIdx as 64-bit integer value.
sl@0
   592
		If the current column type does not refer to a 64-bit integer, then 
sl@0
   593
		the function will do a data conversion as described in the table which can be found
sl@0
   594
		in SqlDb.h file.
sl@0
   595
@see RSqlStatement
sl@0
   596
sl@0
   597
@panic SqlDb 2 In _DEBUG mode. Invalid (not created) CSqlSrvStatement object.
sl@0
   598
@panic SqlDb 4 In _DEBUG mode. Invalid aColIdx value.
sl@0
   599
*/
sl@0
   600
TInt64 CSqlSrvStatement::ColumnInt64(TInt aColIdx) const
sl@0
   601
	{
sl@0
   602
	__ASSERT_DEBUG(iStmtHandle != NULL, __SQLPANIC(ESqlPanicInvalidObj));
sl@0
   603
	__ASSERT_DEBUG((TUint)aColIdx < iColumnCount, __SQLPANIC(ESqlPanicBadArgument));
sl@0
   604
	TInt colType = sqlite3_column_type(iStmtHandle, aColIdx);
sl@0
   605
	switch(colType)
sl@0
   606
		{
sl@0
   607
		case SQLITE_FLOAT:
sl@0
   608
			{
sl@0
   609
			TReal roundVal;
sl@0
   610
			TInt err = Math::Round(roundVal, sqlite3_column_double(iStmtHandle, aColIdx), 0);
sl@0
   611
			if(err != KErrNone)
sl@0
   612
				{
sl@0
   613
				return KMinTInt64;
sl@0
   614
				}
sl@0
   615
			TRealX val(roundVal);
sl@0
   616
			return static_cast <TInt64> (val);
sl@0
   617
			}
sl@0
   618
		case SQLITE_NULL:
sl@0
   619
		case SQLITE_TEXT:
sl@0
   620
		case SQLITE_BLOB:
sl@0
   621
			return 0;
sl@0
   622
		default:			//int, int64
sl@0
   623
			return sqlite3_column_int64(iStmtHandle, aColIdx);
sl@0
   624
		}
sl@0
   625
	}
sl@0
   626
	
sl@0
   627
/**
sl@0
   628
@return Represents the content of the column identified by aColIdx as real value.
sl@0
   629
		If the current column type does not refer to a real, then 
sl@0
   630
		the function will do a data conversion as described in the table which can be found
sl@0
   631
		in SqlDb.h file.
sl@0
   632
@see RSqlStatement
sl@0
   633
sl@0
   634
@panic SqlDb 2 In _DEBUG mode. Invalid (not created) CSqlSrvStatement object.
sl@0
   635
@panic SqlDb 4 In _DEBUG mode. Invalid aColIdx value.
sl@0
   636
*/
sl@0
   637
TReal CSqlSrvStatement::ColumnReal(TInt aColIdx) const
sl@0
   638
	{
sl@0
   639
	__ASSERT_DEBUG(iStmtHandle != NULL, __SQLPANIC(ESqlPanicInvalidObj));
sl@0
   640
	__ASSERT_DEBUG((TUint)aColIdx < iColumnCount, __SQLPANIC(ESqlPanicBadArgument));
sl@0
   641
	TInt colType = sqlite3_column_type(iStmtHandle, aColIdx);
sl@0
   642
	switch(colType)
sl@0
   643
		{
sl@0
   644
		case SQLITE_INTEGER:
sl@0
   645
			{
sl@0
   646
			TRealX val(sqlite3_column_int64(iStmtHandle, aColIdx));	
sl@0
   647
			return static_cast <TReal> (val);
sl@0
   648
			}
sl@0
   649
		case SQLITE_NULL:
sl@0
   650
		case SQLITE_TEXT:
sl@0
   651
		case SQLITE_BLOB:
sl@0
   652
			return 0.0;
sl@0
   653
		default:
sl@0
   654
			return sqlite3_column_double(iStmtHandle, aColIdx);
sl@0
   655
		}
sl@0
   656
	}
sl@0
   657
	
sl@0
   658
/**
sl@0
   659
Represents the content of the column identified by aColIdx as text (16 bit) descriptor.
sl@0
   660
If the current column type does not refer to a text block of data, then 
sl@0
   661
the function will do a data conversion as described in the table which can be found
sl@0
   662
in SqlDb.h file.
sl@0
   663
		
sl@0
   664
@see RSqlStatement
sl@0
   665
sl@0
   666
@leave KErrNoMemory, an out of memory condition has occurred.
sl@0
   667
sl@0
   668
@panic SqlDb 2 In _DEBUG mode. Invalid (not created) CSqlSrvStatement object.
sl@0
   669
@panic SqlDb 4 In _DEBUG mode. Invalid aColIdx value.
sl@0
   670
*/
sl@0
   671
TPtrC CSqlSrvStatement::ColumnTextL(TInt aColIdx) const
sl@0
   672
	{
sl@0
   673
	__ASSERT_DEBUG(iStmtHandle != NULL, __SQLPANIC(ESqlPanicInvalidObj));
sl@0
   674
	__ASSERT_DEBUG((TUint)aColIdx < iColumnCount, __SQLPANIC(ESqlPanicBadArgument));
sl@0
   675
	TPtrC res;
sl@0
   676
	TInt colType = sqlite3_column_type(iStmtHandle, aColIdx);
sl@0
   677
	if(colType == SQLITE_TEXT)
sl@0
   678
		{
sl@0
   679
		TInt charLength = (TUint)sqlite3_column_bytes16(iStmtHandle, aColIdx) / sizeof(TUint16);
sl@0
   680
        //"charLength == 0" - this might be an indication of an "out of memory" problem, if the column text is in UTF8 format. 
sl@0
   681
        //(sqlite3_column_bytes16() may allocate memory for UTF8->UTF16 conversion)
sl@0
   682
        if(charLength == 0 && sqlite3_errcode(sqlite3_db_handle(iStmtHandle)) == SQLITE_NOMEM)
sl@0
   683
            {
sl@0
   684
            __SQLLEAVE(KErrNoMemory);
sl@0
   685
            }
sl@0
   686
        //sqlite3_column_bytes16() already allocated the needed memory if a UTF8->UTF16 conversion
sl@0
   687
        //had to be performed. The sqlite3_column_text16() on the next line is guaranteed to succeed.
sl@0
   688
        const TUint16* text = reinterpret_cast <const TUint16*> (sqlite3_column_text16(iStmtHandle, aColIdx));
sl@0
   689
        __ASSERT_DEBUG(text != NULL, __SQLPANIC(ESqlPanicInternalError));
sl@0
   690
 		res.Set(text, charLength);
sl@0
   691
		}
sl@0
   692
	return res;
sl@0
   693
	}
sl@0
   694
	
sl@0
   695
/**
sl@0
   696
Represents the content of the column identified by aColIdx as binary (8 bit) descriptor.
sl@0
   697
If the current column type does not refer to a binary block of data, then 
sl@0
   698
the function will do a data conversion as described in the table which can be found
sl@0
   699
in SqlDb.h file.
sl@0
   700
                      
sl@0
   701
@see RSqlStatement
sl@0
   702
sl@0
   703
@panic SqlDb 2 In _DEBUG mode. Invalid (not created) CSqlSrvStatement object.
sl@0
   704
@panic SqlDb 4 In _DEBUG mode. Invalid aColIdx value.
sl@0
   705
*/
sl@0
   706
TPtrC8 CSqlSrvStatement::ColumnBinary(TInt aColIdx) const
sl@0
   707
	{
sl@0
   708
	__ASSERT_DEBUG(iStmtHandle != NULL, __SQLPANIC(ESqlPanicInvalidObj));
sl@0
   709
	__ASSERT_DEBUG((TUint)aColIdx < iColumnCount, __SQLPANIC(ESqlPanicBadArgument));
sl@0
   710
	TPtrC8 res;
sl@0
   711
	TInt colType = sqlite3_column_type(iStmtHandle, aColIdx);
sl@0
   712
	if(colType == SQLITE_BLOB)
sl@0
   713
		{
sl@0
   714
		TInt byteLength = sqlite3_column_bytes(iStmtHandle, aColIdx);
sl@0
   715
		res.Set(reinterpret_cast <const TUint8*> (sqlite3_column_blob(iStmtHandle, aColIdx)), byteLength);
sl@0
   716
		}
sl@0
   717
	return res;
sl@0
   718
	}
sl@0
   719
sl@0
   720
/**
sl@0
   721
Retrieves the declared column types using the SQLITE library storing in a 
sl@0
   722
flat buffer and returns a reference to the buffer. 
sl@0
   723
sl@0
   724
@return A const reference to a flat buffer containing the declared column type names.
sl@0
   725
sl@0
   726
@leave KErrNoMemory, an out of memory condition has occurred;
sl@0
   727
sl@0
   728
@panic SqlDb 2 In _DEBUG mode. Invalid (not created) CSqlSrvStatement object.
sl@0
   729
*/
sl@0
   730
const RSqlBufFlat& CSqlSrvStatement::GetDeclColumnTypesL()
sl@0
   731
	{
sl@0
   732
    __ASSERT_DEBUG(iStmtHandle != NULL, __SQLPANIC(ESqlPanicInvalidObj));
sl@0
   733
 	iBufFlatType = static_cast <TSqlBufFlatType> (-1);
sl@0
   734
	__SQLLEAVE_IF_ERROR(iBufFlat.SetCount(iColumnCount));
sl@0
   735
	TSqlBufWIterator it;
sl@0
   736
	it.Set(iBufFlat);
sl@0
   737
	TInt colIdx = -1;
sl@0
   738
	while(it.Next())
sl@0
   739
		{
sl@0
   740
		++colIdx;//the first SQLITE column index is 0
sl@0
   741
		const TUint16* declTypeTxt = reinterpret_cast <const TUint16*> (sqlite3_column_decltype16(iStmtHandle, colIdx));
sl@0
   742
		TPtrC ptr(KNullDesC);
sl@0
   743
        if(declTypeTxt)
sl@0
   744
            {
sl@0
   745
            ptr.Set(declTypeTxt, User::StringLength(declTypeTxt));
sl@0
   746
            }
sl@0
   747
        else
sl@0
   748
            {
sl@0
   749
            //If sqlite3_column_decltype16() returns NULL but sqlite3_column_decltype() doesn't, then it is an "out of memory" condition 
sl@0
   750
            if(sqlite3_column_decltype(iStmtHandle, colIdx))
sl@0
   751
                {
sl@0
   752
                __SQLLEAVE(KErrNoMemory);
sl@0
   753
                }
sl@0
   754
            }
sl@0
   755
		__SQLLEAVE_IF_ERROR(it.SetText(ptr));
sl@0
   756
		}
sl@0
   757
	iBufFlatType = ESqlDeclColumnTypesBuf;
sl@0
   758
	return iBufFlat;	
sl@0
   759
	}
sl@0
   760
sl@0
   761
sl@0
   762
/**
sl@0
   763
Creates a new HSqlSrvStmtParamBuf object for the parameter with index "aParamIndex" or
sl@0
   764
reuses the existing one.
sl@0
   765
sl@0
   766
@param aParamIndex Parameter index, zero based.
sl@0
   767
@param aDataType Parameter value type - binary, text8 or text16.
sl@0
   768
@param aIsStreamBuf True if the param data will be retrieved from an IPC stream
sl@0
   769
sl@0
   770
@return A pointer to the created HSqlSrvStmtParamBuf instance.
sl@0
   771
sl@0
   772
@leave KErrNoMemory, an out of memory condition has occurred;
sl@0
   773
*/
sl@0
   774
HSqlSrvStmtParamBuf* CSqlSrvStatement::GetParamBufL(TInt aParamIndex, HSqlSrvStmtParamBuf::TDataType aDataType, 
sl@0
   775
													HSqlSrvStmtParamBuf::TBufType aBufType)
sl@0
   776
	{
sl@0
   777
	__ASSERT_DEBUG(aParamIndex >= 0 && aParamIndex < sqlite3_bind_parameter_count(iStmtHandle), __SQLPANIC(ESqlPanicBadArgument));
sl@0
   778
	ExtendParamBufArrayL(aParamIndex);
sl@0
   779
	HSqlSrvStmtParamBuf*& paramBuf = iParamBufArray[aParamIndex];
sl@0
   780
	if(paramBuf)
sl@0
   781
		{//Reset and reuse the existing buffer
sl@0
   782
		__ASSERT_DEBUG(paramBuf->ParamIndex() == aParamIndex, __SQLPANIC(ESqlPanicInternalError));
sl@0
   783
		paramBuf->Reset(aDataType, aBufType);	
sl@0
   784
		}
sl@0
   785
	else
sl@0
   786
		{
sl@0
   787
		paramBuf = HSqlSrvStmtParamBuf::NewL(*this, aParamIndex, aDataType, aBufType);
sl@0
   788
		}
sl@0
   789
	return paramBuf;
sl@0
   790
	}
sl@0
   791
sl@0
   792
/**
sl@0
   793
This function will extend the iParamBufArray array (filling the new array items with NULL), if it is needed - 
sl@0
   794
to ensure that there is enough place in the buffer for the parameter identified by aParamIndex.
sl@0
   795
sl@0
   796
@param aParamIndex The parameter index
sl@0
   797
sl@0
   798
@leave KErrNoMemory, an out of memory condition has occurred;
sl@0
   799
sl@0
   800
@panic SqlDb 2 In _DEBUG mode. Invalid (not created) CSqlSrvStatement object.
sl@0
   801
@panic SqlDb 4 In _DEBUG mode. Parameter index out of bounds.
sl@0
   802
*/
sl@0
   803
void CSqlSrvStatement::ExtendParamBufArrayL(TInt aParamIndex)
sl@0
   804
	{
sl@0
   805
	__ASSERT_DEBUG(iStmtHandle != NULL, __SQLPANIC(ESqlPanicInvalidObj));
sl@0
   806
	__ASSERT_DEBUG(aParamIndex >= 0 && aParamIndex < sqlite3_bind_parameter_count(iStmtHandle), __SQLPANIC(ESqlPanicBadArgument));
sl@0
   807
	TInt ext = aParamIndex - iParamBufArray.Count() + 1;
sl@0
   808
	while(ext-- > 0)
sl@0
   809
		{
sl@0
   810
		__SQLLEAVE_IF_ERROR(iParamBufArray.Append(NULL));
sl@0
   811
		}
sl@0
   812
	}
sl@0
   813
sl@0
   814
/**
sl@0
   815
This function will create a copy of the aParamValue and store it in the iParamBufArray array for later use.
sl@0
   816
The reason: once bound, the parameter value can be used multiple times by the SQLite if it is not set explicitly again.
sl@0
   817
sl@0
   818
@param aParamIndex The parameter index
sl@0
   819
@param aDataType Parameter value type - binary, text8 or text16.
sl@0
   820
@param aParamValue The parameter value
sl@0
   821
sl@0
   822
@return 8-bit descriptor to the stored parameter value
sl@0
   823
sl@0
   824
@leave KErrNoMemory, an out of memory condition has occurred;
sl@0
   825
                     
sl@0
   826
@panic SqlDb 2 In _DEBUG mode. Invalid (not created) CSqlSrvStatement object.
sl@0
   827
@panic SqlDb 4 In _DEBUG mode. Parameter index out of bounds.
sl@0
   828
*/
sl@0
   829
TPtrC8 CSqlSrvStatement::CopyAndStoreParamL(TInt aParamIndex, HSqlSrvStmtParamBuf::TDataType aDataType, const TDesC8& aParamValue)
sl@0
   830
	{
sl@0
   831
	__ASSERT_DEBUG(iStmtHandle != NULL, __SQLPANIC(ESqlPanicInvalidObj));
sl@0
   832
	__ASSERT_DEBUG(aParamIndex >= 0 && aParamIndex < sqlite3_bind_parameter_count(iStmtHandle), __SQLPANIC(ESqlPanicBadArgument));
sl@0
   833
	HSqlSrvStmtParamBuf* paramBuf = GetParamBufL(aParamIndex, aDataType, HSqlSrvStmtParamBuf::EBufSimpleBind);
sl@0
   834
	return paramBuf->SetDataL(aParamValue);
sl@0
   835
	}