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.
18 #include "SqlAssert.h"
19 #include "SqlDbSession.h"
20 #include "OstTraceDefinitions.h"
21 #ifdef OST_TRACE_COMPILER_IN_USE
22 #include "IPCBufTraces.h"
24 #include "SqlTraceDef.h"
27 Standard, phase-one HIpcBuf factory method.
28 Created HIpcBuf instance will be used for transfering large text or binary objects from/to SQL server.
29 The created HIpcBuf instance will be placed in the cleanup stack.
31 @param aSession A reference to RSqlDbSession instance.
32 @param aFunction Prepared function code (with all statement handle bits set)
33 @param aArgs A set of IPC arguments to be sent to the SQL server.
35 @return A pointer to the created HIpcBuf instance
37 @leave KErrNoMemory Out of memory.
39 HIpcBuf* HIpcBuf::NewLC(RSqlDbSession& aSession, TInt aFunction, TIpcArgs& aArgs)
41 HIpcBuf* self = new (ELeave) HIpcBuf(aSession);
43 self->ConstructL(aFunction, aArgs);
48 Standard, phase-one HIpcBuf factory method.
49 Created HIpcBuf instance will be used for transfering large text or binary objects from/to SQL server.
51 @param aSession A reference to RSqlDbSession instance.
52 @param aFunction Prepared function code (with all statement handle bits set)
53 @param aArgs A set of IPC arguments to be sent to the SQL server.
55 @return A pointer to the created HIpcBuf instance
57 @leave KErrNoMemory Out of memory.
59 HIpcBuf* HIpcBuf::NewL(RSqlDbSession& aSession, TInt aFunction, TIpcArgs& aArgs)
61 HIpcBuf* self = NewLC(aSession, aFunction, aArgs);
67 Standard, phase-two HIpcBuf construction method.
69 @param aFunction The command code which will be sent to the SQL server
70 @param aArgs A set of IPC arguments to be sent to the SQL server.
72 @leave KErrNoMemory Out of memory.
74 Usage of the IPC call arguments:
75 Arg 2: [in/out] IPC buffer
76 iBuf.iExt: [in] stream size in bytes
78 void HIpcBuf::ConstructL(TInt aFunction, TIpcArgs& aArgs)
80 TPckg<TIpcStreamBuf> pckg(iBuf);
82 __SQLLEAVE_IF_ERROR(iHandle = iSession.SendReceive(aFunction, aArgs));
83 SQL_TRACE_INTERNALS(OstTraceExt2(TRACE_INTERNALS, HIPCBUF_CONSTRUCTL, "0x%X;HIpcBuf::ConstructL;iHandle=%d", (TUint)this, iHandle));
84 TUint8* base = iBuf.iData;
85 // if reading we already have one buffer-full of data
86 TInt avail = Max(0, Min(iBuf.iExt, KIpcBufSize));
87 SetBuf(ERead, base, base + avail);
89 SetBuf(EWrite, base, base);
94 @param aSession A reference to a sesion object.
96 HIpcBuf::HIpcBuf(RSqlDbSession& aSession) :
109 SQL_TRACE_INTERNALS(OstTraceExt2(TRACE_INTERNALS, HIPCBUF_HIPCBUFL2, "0x%X;HIpcBuf::~HIpcBuf;iHandle=%d", (TUint)this, iHandle));
110 if(iHandle > 0) //iHandle is valid only when > 0.
112 (void)iSession.SendReceive(::MakeMsgCode(ESqlSrvStreamClose, ESqlSrvStreamHandle, iHandle));
117 Fill the buffer's read area.
119 TInt HIpcBuf::UnderflowL(TInt)
121 // when handle is null there is no data to read from server
126 __ASSERT_DEBUG(Avail(ERead) == 0, __SQLPANIC(ESqlPanicInternalError));
127 TUint8* base=iBuf.iData;
128 IpcWriteL(base,Lag(EWrite));
129 SetBuf(EWrite,base,base);
131 TInt len=IpcReadL(base,iBuf.ESize);
132 SetBuf(ERead,base,base+len);
137 Set up the buffer's write area.
139 void HIpcBuf::OverflowL()
141 __ASSERT_DEBUG(Avail(EWrite) == 0, __SQLPANIC(ESqlPanicInternalError));
143 TUint8* base = iBuf.iData;
144 MovePos(ERead, Lag(ERead));
145 SetBuf(ERead, base, base);
147 IpcWriteL(base, Lag(EWrite));
148 SetBuf(EWrite, base, base + iBuf.ESize);
152 Destroys HIpcBuf instance.
154 void HIpcBuf::DoRelease()
160 Synchronise this buffer with its file, giving up on outstanding writes in case of failure.
162 void HIpcBuf::DoSynchL()
164 TUint8* base = iBuf.iData;
165 MovePos(ERead, Lag(ERead));
166 TInt lag = Lag(EWrite);
167 SetBuf(ERead | EWrite, base, base);
169 IpcWriteL(base, lag);
170 __SQLLEAVE_IF_ERROR(iSession.SendReceive(::MakeMsgCode(ESqlSrvStreamSynch, ESqlSrvStreamHandle, iHandle)));
174 Read direct from ipc if asked to transfer more than a bufferful.
176 TInt HIpcBuf::DoReadL(TAny* aPtr, TInt aMaxLength)
178 __ASSERT_DEBUG(aMaxLength > 0, __SQLPANIC(ESqlPanicInternalError));
179 TInt avail = Avail(ERead);
180 __ASSERT_DEBUG(avail >= 0 && Avail(EWrite) >= 0, __SQLPANIC(ESqlPanicInternalError));
183 TInt len = Min(aMaxLength, avail);
184 TUint8* ptr = Ptr(ERead);
185 aPtr = Mem::Copy(aPtr, ptr, len);
186 SetPtr(ERead, ptr + len);
189 return len; // that's it
191 __ASSERT_DEBUG(Avail(ERead) == 0, __SQLPANIC(ESqlPanicInternalError));
192 if(aMaxLength < iBuf.ESize)
193 return avail + TStreamBuf::DoReadL(aPtr, aMaxLength);
195 // when handle is null there is no more data to read from server
201 TUint8* base = iBuf.iData;
202 IpcWriteL(base, Lag(EWrite));
203 SetBuf(ERead | EWrite, base, base);
204 return avail + IpcReadL(aPtr, aMaxLength);
208 Write direct to ipc if asked to transfer more than a bufferful.
210 void HIpcBuf::DoWriteL(const TAny* aPtr,TInt aLength)
212 __ASSERT_DEBUG(aLength > 0, __SQLPANIC(ESqlPanicInternalError));
213 TInt avail = Avail(EWrite);
214 __ASSERT_DEBUG(Avail(ERead) >= 0 && avail >= 0, __SQLPANIC(ESqlPanicInternalError));
217 TInt len = Min(aLength, avail);
218 SetPtr(EWrite, Mem::Copy(Ptr(EWrite), aPtr, len));
223 aPtr = (TUint8*)aPtr + len;
225 __ASSERT_DEBUG(Avail(EWrite) == 0, __SQLPANIC(ESqlPanicInternalError));
226 if(aLength < iBuf.ESize)
227 TStreamBuf::DoWriteL(aPtr, aLength);
230 TUint8* base = iBuf.iData;
231 IpcWriteL(base, Lag(EWrite));
232 MovePos(ERead, Lag(ERead));
233 SetBuf(ERead | EWrite, base, base);
234 IpcWriteL(aPtr, aLength);
239 Position the mark(s) indicated by aMark at aOffset from aLocation.
241 TStreamPos HIpcBuf::DoSeekL(TMark aMark, TStreamLocation aLocation, TInt aOffset)
243 TUint8* base = iBuf.iData;
248 case EStreamBeginning:
254 aOffset += Mark(ERead);
257 aOffset += Mark(EWrite);
260 __ASSERT_ALWAYS(0, __SQLPANIC(ESqlPanicStreamMarkInvalid));
268 __ASSERT_ALWAYS(0, __SQLPANIC(ESqlPanicStreamLocationInvalid));
277 else if(aOffset > end)
283 __ASSERT_ALWAYS(!(aMark & ~(ERead | EWrite)), __SQLPANIC(ESqlPanicStreamMarkInvalid));
286 TInt lag = aOffset - Pos(ERead);
287 if(lag >= base - End(ERead) && lag <= 0)
288 SetPtr(ERead, End(ERead) + lag);
291 SetPos(ERead, aOffset);
292 SetBuf(ERead, base, base);
295 if(aMark & EWrite && aOffset != Mark(EWrite))
297 IpcWriteL(base, Lag(EWrite));
298 SetPos(EWrite, aOffset);
299 SetBuf(EWrite, base, base);
301 __SQLLEAVE_IF_ERROR(r);
302 return TStreamPos(aOffset);
306 Read from the server at the current read position.
308 Arg 1: [out] from which position to read
309 Arg 2: [in/out] IPC buffer
310 Arg 3: [out] max length of the requested data
312 TInt HIpcBuf::IpcReadL(TAny* aPtr, TInt aMaxLength)
314 __ASSERT_DEBUG(aMaxLength >= 0, __SQLPANIC(ESqlPanicInternalError));
318 TPtr8 des((TUint8*)aPtr, aMaxLength);
319 TInt pos = Pos(ERead);
321 TInt len = __SQLLEAVE_IF_ERROR(iSession.SendReceive(::MakeMsgCode(ESqlSrvStreamRead, ESqlSrvStreamHandle, iHandle), TIpcArgs(0, pos, &des, aMaxLength)));
324 iBuf.iExt = pos; // end-of-file encountered
330 Write to the server at the current write position.
332 Arg 1: [out] from which position to write
333 Arg 2: [in/out] IPC buffer
335 void HIpcBuf::IpcWriteL(const TAny* aPtr, TInt aLength)
337 __ASSERT_DEBUG(aLength >= 0, __SQLPANIC(ESqlPanicInternalError));
341 TPtrC8 ptr((TUint8*)aPtr, aLength);
342 TInt ext = iBuf.iExt;
344 TInt pos = Pos(EWrite);
345 __SQLLEAVE_IF_ERROR(iSession.SendReceive(::MakeMsgCode(ESqlSrvStreamWrite, ESqlSrvStreamHandle, iHandle), TIpcArgs(0, pos, &ptr)));
347 if(ext >=0 && pos > ext)
353 Determine the end of the stream
357 TInt ext = iBuf.iExt;
360 iBuf.iExt = ext = __SQLLEAVE_IF_ERROR(iSession.SendReceive(::MakeMsgCode(ESqlSrvStreamSize, ESqlSrvStreamHandle, iHandle)));
362 return Max(ext, Mark(EWrite));