os/persistentdata/persistentstorage/sql/SRC/Client/SqlStmtSession.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.
     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 Bug 1915 "SQL server panics when using long column type strings"
    13 //
    14 // Description:
    15 //
    16 
    17 #include <s32mem.h>
    18 #include "SqlStmtSession.h"		//RSqlStatementSession
    19 #include "OstTraceDefinitions.h"
    20 #ifdef OST_TRACE_COMPILER_IN_USE
    21 #include "SqlStmtSessionTraces.h"
    22 #endif
    23 #include "SqlTraceDef.h"
    24 
    25 /**
    26  Sends a request to the SQL server to prepare 16-bit aSqlStmt statement.
    27  
    28  Usage of the IPC call arguments:
    29  Arg 0: [in/out]        data buffer for the column and parameter count.
    30  Arg 1: [out]       statement length in characters
    31  Arg 2: [out]       16-bit statement
    32  
    33  @param aDbSession A reference to RSqlDbSession instance.
    34  @param aSqlStmt 16-bit SQL statement.
    35  @param aColumnCount Output parameter. Statement column count.
    36  @param aParamCount Output parameter. Statement parameter count.
    37  
    38  @return KErrNoMemory, an out of memory condition has occured;
    39  KErrArgument, bad argument, for example - the SQL string contains more than one SQL statements.
    40  Note that the function may return some database specific errors categorised as 
    41  ESqlDbError or other system-wide error codes;
    42  KErrNone      The operation has completed successfully. 
    43  
    44  @panic SqlDb 7 In _DEBUG mode if the statement handle is 0.
    45 */
    46 TInt RSqlStatementSession::Prepare(RSqlDbSession& aDbSession, const TDesC& aSqlStmt, 
    47                                           TInt& aColumnCount, TInt& aParamCount)
    48     {
    49     iDbSession = &aDbSession;
    50     TSqlIpcData data;
    51     TPckg<TSqlIpcData> pckg(data);
    52     TUint stmtLen = aSqlStmt.Length();
    53     iHandle = iDbSession->SendReceive(ESqlSrvStmtPrepare16, TIpcArgs(&pckg, stmtLen, &aSqlStmt));
    54     __ASSERT_DEBUG(iHandle != 0, __SQLPANIC(ESqlPanicInternalError));   
    55     aColumnCount = static_cast <TInt> (data.iPrm1);
    56     aParamCount = static_cast <TInt> (data.iPrm2);
    57     SQL_TRACE_SESSION(OstTraceExt2(TRACE_INTERNALS, RSQLSTATEMENTSESSION_PREPARE16, "0x%X;RSqlStatementSession::Prepare-16;iHandle=%d", (TUint)this, iHandle));
    58     return iHandle > 0 ? KErrNone : iHandle;
    59     }
    60 
    61 /**
    62 Sends a request to the SQL server to prepare 8-bit aSqlStmt statement.
    63 
    64 Usage of the IPC call arguments:
    65 Arg 0: [in/out]     data buffer for the column and parameter count.
    66 Arg 1: [out]        statement length in characters
    67 Arg 2: [out]        8-bit statement
    68 
    69 @param aDbSession A reference to RSqlDbSession instance.
    70 @param aSqlStmt 8-bit SQL statement.
    71 @param aColumnCount Output parameter. Statement column count.
    72 @param aParamCount Output parameter. Statement parameter count.
    73 
    74 @return KErrNoMemory, an out of memory condition has occured;
    75         KErrArgument, bad argument, for example - the SQL string contains more than one SQL statements.
    76                       Note that the function may return some database specific errors categorised as 
    77                       ESqlDbError or other system-wide error codes;
    78         KErrNone      The operation has completed successfully. 
    79 
    80 @panic SqlDb 7 In _DEBUG mode if the statement handle is 0.
    81 */
    82 TInt RSqlStatementSession::Prepare(RSqlDbSession& aDbSession, const TDesC8& aSqlStmt, 
    83                                           TInt& aColumnCount, TInt& aParamCount)
    84     {
    85     iDbSession = &aDbSession;
    86     TSqlIpcData data;
    87     TPckg<TSqlIpcData> pckg(data);
    88     TUint stmtLen = aSqlStmt.Length();
    89     iHandle = iDbSession->SendReceive(ESqlSrvStmtPrepare8, TIpcArgs(&pckg, stmtLen, &aSqlStmt));
    90     __ASSERT_DEBUG(iHandle != 0, __SQLPANIC(ESqlPanicInternalError));
    91     aColumnCount = static_cast <TInt> (data.iPrm1);
    92     aParamCount = static_cast <TInt> (data.iPrm2);
    93     SQL_TRACE_SESSION(OstTraceExt2(TRACE_INTERNALS, RSQLSTATEMENTSESSION_PREPARE8, "0x%X;RSqlStatementSession::Prepare-8;iHandle=%d", (TUint)this, iHandle));
    94     return iHandle > 0 ? KErrNone : iHandle;
    95     }
    96 
    97 /**
    98 Sends a request to the server to close the statement handle.
    99 Closes the session object.
   100 */
   101 void RSqlStatementSession::Close()
   102 	{
   103 	SQL_TRACE_SESSION(OstTraceExt2(TRACE_INTERNALS, RSQLSTATEMENTSESSION_CLOSE, "0x%X;RSqlStatementSession::Close;iHandle=%d", (TUint)this, iHandle));
   104 	if(iDbSession && iHandle > 0)
   105 		{
   106 		(void)iDbSession->SendReceive(::MakeMsgCode(ESqlSrvStmtClose, ESqlSrvStatementHandle, iHandle));
   107 		}
   108 	iDbSession = NULL;
   109 	iHandle = -1;
   110 	}
   111 
   112 /**
   113 Binds the statement parameters and sends a request to the SQL server to move to the next record which satisfies the 
   114 condition of the prepared SQL statement.
   115 If there is a valid next record, the method transfers the column values from the server.
   116 
   117 @param aParamBuf  It references RSqlBufFlat object where the parameter values are stored.
   118 @param aColumnBuf It references RSqlBufFlat object where the column values will be stored.
   119 
   120 @return KSqlAtRow,      the record data is ready for processing by the caller;
   121         KSqlAtEnd,      there is no more record data;
   122         KSqlErrBusy,    the database file is locked;
   123         KErrNoMemory,   an out of memory condition has occurred - the statement
   124                         will be reset;
   125         KSqlErrGeneral, a run-time error has occured - this function must not
   126                         be called again;        
   127         KSqlErrMisuse,  this function has been called after a previous call
   128                         returned KSqlAtEnd or KSqlErrGeneral.
   129         KSqlErrStmtExpired, the SQL statement has expired (if new functions or
   130                             collating sequences have been registered or if an
   131                             authorizer function has been added or changed);
   132 */
   133 TInt RSqlStatementSession::BindNext(const RSqlBufFlat& aParamBuf, RSqlBufFlat& aColumnBuf)
   134 	{
   135 	TPtrC8 prmData(aParamBuf.BufDes());
   136 	TIpcArgs ipcArgs;
   137 	ipcArgs.Set(0, prmData.Length());
   138 	ipcArgs.Set(1, &prmData);
   139 	return DoBindNext(ESqlSrvStmtBindNext, ipcArgs, aColumnBuf);
   140 	}
   141 
   142 /**
   143 Implements RSqlStatementSession::Next() and RSqlStatementSession::BindNext().
   144 Sends a "Next" command to the server combined with optional "Bind" command.
   145 In a single IPC call the statement parameters will be bound and the current row columns - returned.
   146 If the client side flat buffer is not big enough, a second IPC call will be made after reallocating the buffer.
   147 
   148 Usage of the IPC call arguments:
   149 Arg 0: [out]		parameter buffer length in bytes
   150 Arg 1: [out]		parameter buffer
   151 Arg 2: [out]		column buffer length in bytes
   152 Arg 3: [in/out]		column buffer
   153 
   154 @see RSqlStatementSession::Next()
   155 @see RSqlStatementSession::BindNext()
   156 */
   157 TInt RSqlStatementSession::DoBindNext(TSqlSrvFunction aFunction, TIpcArgs& aIpcArgs, RSqlBufFlat& aColumnBuf)
   158 	{
   159 	aColumnBuf.Reset();
   160 	aIpcArgs.Set(2, aColumnBuf.MaxSize());
   161 	aIpcArgs.Set(3, &aColumnBuf.BufPtr());
   162 	TInt err = DbSession().SendReceive(::MakeMsgCode(aFunction, ESqlSrvStatementHandle, iHandle), aIpcArgs);
   163 	if(err > KSqlClientBufOverflowCode)
   164 		{
   165 		err = Retry(aColumnBuf, err - KSqlClientBufOverflowCode, ESqlColumnValuesBuf);
   166 		if(err == KErrNone)
   167 			{
   168 			err = KSqlAtRow;	
   169 			}
   170 		}
   171 	return err;
   172 	}
   173 
   174 /**
   175 Sends a command to the server for retrieving parameter names or column names.
   176 
   177 Usage of the IPC call arguments:
   178 Arg 0: [out]		buffer length in bytes
   179 Arg 1: [in/out]		buffer
   180 */	
   181 TInt RSqlStatementSession::GetNames(TSqlSrvFunction aFunction, RSqlBufFlat& aNameBuf)
   182 	{
   183 	aNameBuf.Reset();
   184 	TPtr8& ptr = aNameBuf.BufPtr();
   185 	TInt err = DbSession().SendReceive(::MakeMsgCode(aFunction, ESqlSrvStatementHandle, iHandle), TIpcArgs(ptr.MaxLength(), &ptr));
   186 	if(err > KSqlClientBufOverflowCode)
   187 		{
   188 		err = Retry(aNameBuf, err - KSqlClientBufOverflowCode, aFunction == ESqlSrvStmtColumnNames ? ESqlColumnNamesBuf : ESqlParamNamesBuf);
   189 		}
   190 	return err;
   191 	}
   192 
   193 /**
   194 Sends a command to the server for retrieving specified data (aWhat parameter).
   195 
   196 Usage of the IPC call arguments:
   197 Arg 0: [out]		The type of the data to be retrieved
   198 Arg 1: [in/out]		Data buffer
   199 */	
   200 TInt RSqlStatementSession::Retry(RSqlBufFlat& aBufFlat, TInt aSize, TSqlBufFlatType aWhat)
   201 	{
   202 	aBufFlat.Reset();
   203 	TInt err = aBufFlat.ReAlloc(aSize);
   204 	if(err == KErrNone)
   205 		{
   206 		TPtr8& ptr = aBufFlat.BufPtr();
   207 		err = DbSession().SendReceive(::MakeMsgCode(ESqlSrvStmtBufFlat, ESqlSrvStatementHandle, iHandle), TIpcArgs(aWhat, &ptr));
   208 		}
   209 	return err;	
   210 	}
   211 
   212 /**
   213 Sends a command to the server for retrieving the declared types of columns
   214 
   215 Usage of the IPC call arguments:
   216 Arg 0: [out]		buffer length in bytes
   217 Arg 1: [in/out]		buffer
   218 */	 
   219 TInt RSqlStatementSession::GetDeclColumnTypes(RSqlBufFlat& aDeclColumnTypeBuf)
   220 	{
   221 	aDeclColumnTypeBuf.Reset();
   222 	TPtr8& ptr = aDeclColumnTypeBuf.BufPtr();
   223 	TInt err = DbSession().SendReceive(::MakeMsgCode(ESqlSrvStmtDeclColumnTypes, ESqlSrvStatementHandle, iHandle), TIpcArgs(ptr.MaxLength(), &ptr));	
   224 	if(err > KSqlClientBufOverflowCode)
   225 		{
   226 		err = Retry(aDeclColumnTypeBuf, err - KSqlClientBufOverflowCode, ESqlDeclColumnTypesBuf);
   227 		}
   228 	return err;	
   229 	}