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