1.1 --- /dev/null Thu Jan 01 00:00:00 1970 +0000
1.2 +++ b/os/persistentdata/persistentstorage/sql/SRC/Client/IPC/IPCBuf.cpp Fri Jun 15 03:10:57 2012 +0200
1.3 @@ -0,0 +1,364 @@
1.4 +// Copyright (c) 2005-2010 Nokia Corporation and/or its subsidiary(-ies).
1.5 +// All rights reserved.
1.6 +// This component and the accompanying materials are made available
1.7 +// under the terms of "Eclipse Public License v1.0"
1.8 +// which accompanies this distribution, and is available
1.9 +// at the URL "http://www.eclipse.org/legal/epl-v10.html".
1.10 +//
1.11 +// Initial Contributors:
1.12 +// Nokia Corporation - initial contribution.
1.13 +//
1.14 +// Contributors:
1.15 +//
1.16 +// Description:
1.17 +//
1.18 +
1.19 +#include <e32base.h>
1.20 +#include "IPCBuf.h"
1.21 +#include "SqlAssert.h"
1.22 +#include "SqlDbSession.h"
1.23 +#include "OstTraceDefinitions.h"
1.24 +#ifdef OST_TRACE_COMPILER_IN_USE
1.25 +#include "IPCBufTraces.h"
1.26 +#endif
1.27 +#include "SqlTraceDef.h"
1.28 +
1.29 +/**
1.30 +Standard, phase-one HIpcBuf factory method.
1.31 +Created HIpcBuf instance will be used for transfering large text or binary objects from/to SQL server.
1.32 +The created HIpcBuf instance will be placed in the cleanup stack.
1.33 +
1.34 +@param aSession A reference to RSqlDbSession instance.
1.35 +@param aFunction Prepared function code (with all statement handle bits set)
1.36 +@param aArgs A set of IPC arguments to be sent to the SQL server.
1.37 +
1.38 +@return A pointer to the created HIpcBuf instance
1.39 +
1.40 +@leave KErrNoMemory Out of memory.
1.41 +*/
1.42 +HIpcBuf* HIpcBuf::NewLC(RSqlDbSession& aSession, TInt aFunction, TIpcArgs& aArgs)
1.43 + {
1.44 + HIpcBuf* self = new (ELeave) HIpcBuf(aSession);
1.45 + self->PushL();
1.46 + self->ConstructL(aFunction, aArgs);
1.47 + return self;
1.48 + }
1.49 +
1.50 +/**
1.51 +Standard, phase-one HIpcBuf factory method.
1.52 +Created HIpcBuf instance will be used for transfering large text or binary objects from/to SQL server.
1.53 +
1.54 +@param aSession A reference to RSqlDbSession instance.
1.55 +@param aFunction Prepared function code (with all statement handle bits set)
1.56 +@param aArgs A set of IPC arguments to be sent to the SQL server.
1.57 +
1.58 +@return A pointer to the created HIpcBuf instance
1.59 +
1.60 +@leave KErrNoMemory Out of memory.
1.61 +*/
1.62 +HIpcBuf* HIpcBuf::NewL(RSqlDbSession& aSession, TInt aFunction, TIpcArgs& aArgs)
1.63 + {
1.64 + HIpcBuf* self = NewLC(aSession, aFunction, aArgs);
1.65 + CleanupStack::Pop();
1.66 + return self;
1.67 + }
1.68 +
1.69 +/**
1.70 +Standard, phase-two HIpcBuf construction method.
1.71 +
1.72 +@param aFunction The command code which will be sent to the SQL server
1.73 +@param aArgs A set of IPC arguments to be sent to the SQL server.
1.74 +
1.75 +@leave KErrNoMemory Out of memory.
1.76 +
1.77 +Usage of the IPC call arguments:
1.78 +Arg 2: [in/out] IPC buffer
1.79 +iBuf.iExt: [in] stream size in bytes
1.80 +*/
1.81 +void HIpcBuf::ConstructL(TInt aFunction, TIpcArgs& aArgs)
1.82 + {
1.83 + TPckg<TIpcStreamBuf> pckg(iBuf);
1.84 + aArgs.Set(2, &pckg);
1.85 + __SQLLEAVE_IF_ERROR(iHandle = iSession.SendReceive(aFunction, aArgs));
1.86 + SQL_TRACE_INTERNALS(OstTraceExt2(TRACE_INTERNALS, HIPCBUF_CONSTRUCTL, "0x%X;HIpcBuf::ConstructL;iHandle=%d", (TUint)this, iHandle));
1.87 + TUint8* base = iBuf.iData;
1.88 + // if reading we already have one buffer-full of data
1.89 + TInt avail = Max(0, Min(iBuf.iExt, KIpcBufSize));
1.90 + SetBuf(ERead, base, base + avail);
1.91 + SetPos(ERead, avail);
1.92 + SetBuf(EWrite, base, base);
1.93 + SetPos(EWrite, 0);
1.94 + }
1.95 +
1.96 +/**
1.97 +@param aSession A reference to a sesion object.
1.98 +*/
1.99 +HIpcBuf::HIpcBuf(RSqlDbSession& aSession) :
1.100 + iSession(aSession),
1.101 + iHandle(0),
1.102 + iRPos(0),
1.103 + iWPos(0)
1.104 + {
1.105 + iBuf.iExt = -1;
1.106 + }
1.107 +
1.108 +/**
1.109 +*/
1.110 +HIpcBuf::~HIpcBuf()
1.111 + {
1.112 + SQL_TRACE_INTERNALS(OstTraceExt2(TRACE_INTERNALS, HIPCBUF_HIPCBUFL2, "0x%X;HIpcBuf::~HIpcBuf;iHandle=%d", (TUint)this, iHandle));
1.113 + if(iHandle > 0) //iHandle is valid only when > 0.
1.114 + {
1.115 + (void)iSession.SendReceive(::MakeMsgCode(ESqlSrvStreamClose, ESqlSrvStreamHandle, iHandle));
1.116 + }
1.117 + }
1.118 +
1.119 +/**
1.120 +Fill the buffer's read area.
1.121 +*/
1.122 +TInt HIpcBuf::UnderflowL(TInt)
1.123 + {
1.124 + // when handle is null there is no data to read from server
1.125 + if(!iHandle)
1.126 + {
1.127 + return 0;
1.128 + }
1.129 + __ASSERT_DEBUG(Avail(ERead) == 0, __SQLPANIC(ESqlPanicInternalError));
1.130 + TUint8* base=iBuf.iData;
1.131 + IpcWriteL(base,Lag(EWrite));
1.132 + SetBuf(EWrite,base,base);
1.133 +
1.134 + TInt len=IpcReadL(base,iBuf.ESize);
1.135 + SetBuf(ERead,base,base+len);
1.136 + return len;
1.137 + }
1.138 +
1.139 +/**
1.140 +Set up the buffer's write area.
1.141 +*/
1.142 +void HIpcBuf::OverflowL()
1.143 + {
1.144 + __ASSERT_DEBUG(Avail(EWrite) == 0, __SQLPANIC(ESqlPanicInternalError));
1.145 +
1.146 + TUint8* base = iBuf.iData;
1.147 + MovePos(ERead, Lag(ERead));
1.148 + SetBuf(ERead, base, base);
1.149 +
1.150 + IpcWriteL(base, Lag(EWrite));
1.151 + SetBuf(EWrite, base, base + iBuf.ESize);
1.152 + }
1.153 +
1.154 +/**
1.155 +Destroys HIpcBuf instance.
1.156 +*/
1.157 +void HIpcBuf::DoRelease()
1.158 + {
1.159 + delete this;
1.160 + }
1.161 +
1.162 +/**
1.163 +Synchronise this buffer with its file, giving up on outstanding writes in case of failure.
1.164 +*/
1.165 +void HIpcBuf::DoSynchL()
1.166 + {
1.167 + TUint8* base = iBuf.iData;
1.168 + MovePos(ERead, Lag(ERead));
1.169 + TInt lag = Lag(EWrite);
1.170 + SetBuf(ERead | EWrite, base, base);
1.171 + iBuf.iExt = -1;
1.172 + IpcWriteL(base, lag);
1.173 + __SQLLEAVE_IF_ERROR(iSession.SendReceive(::MakeMsgCode(ESqlSrvStreamSynch, ESqlSrvStreamHandle, iHandle)));
1.174 + }
1.175 +
1.176 +/**
1.177 +Read direct from ipc if asked to transfer more than a bufferful.
1.178 +*/
1.179 +TInt HIpcBuf::DoReadL(TAny* aPtr, TInt aMaxLength)
1.180 + {
1.181 + __ASSERT_DEBUG(aMaxLength > 0, __SQLPANIC(ESqlPanicInternalError));
1.182 + TInt avail = Avail(ERead);
1.183 + __ASSERT_DEBUG(avail >= 0 && Avail(EWrite) >= 0, __SQLPANIC(ESqlPanicInternalError));
1.184 + if(avail > 0)
1.185 + {
1.186 + TInt len = Min(aMaxLength, avail);
1.187 + TUint8* ptr = Ptr(ERead);
1.188 + aPtr = Mem::Copy(aPtr, ptr, len);
1.189 + SetPtr(ERead, ptr + len);
1.190 + aMaxLength -= len;
1.191 + if(aMaxLength == 0)
1.192 + return len; // that's it
1.193 + }
1.194 + __ASSERT_DEBUG(Avail(ERead) == 0, __SQLPANIC(ESqlPanicInternalError));
1.195 + if(aMaxLength < iBuf.ESize)
1.196 + return avail + TStreamBuf::DoReadL(aPtr, aMaxLength);
1.197 +
1.198 + // when handle is null there is no more data to read from server
1.199 + if(!iHandle)
1.200 + {
1.201 + return 0;
1.202 + }
1.203 +
1.204 + TUint8* base = iBuf.iData;
1.205 + IpcWriteL(base, Lag(EWrite));
1.206 + SetBuf(ERead | EWrite, base, base);
1.207 + return avail + IpcReadL(aPtr, aMaxLength);
1.208 + }
1.209 +
1.210 +/**
1.211 +Write direct to ipc if asked to transfer more than a bufferful.
1.212 +*/
1.213 +void HIpcBuf::DoWriteL(const TAny* aPtr,TInt aLength)
1.214 + {
1.215 + __ASSERT_DEBUG(aLength > 0, __SQLPANIC(ESqlPanicInternalError));
1.216 + TInt avail = Avail(EWrite);
1.217 + __ASSERT_DEBUG(Avail(ERead) >= 0 && avail >= 0, __SQLPANIC(ESqlPanicInternalError));
1.218 + if(avail > 0)
1.219 + {
1.220 + TInt len = Min(aLength, avail);
1.221 + SetPtr(EWrite, Mem::Copy(Ptr(EWrite), aPtr, len));
1.222 + aLength -= len;
1.223 + if(aLength == 0)
1.224 + return; // done
1.225 +
1.226 + aPtr = (TUint8*)aPtr + len;
1.227 + }
1.228 + __ASSERT_DEBUG(Avail(EWrite) == 0, __SQLPANIC(ESqlPanicInternalError));
1.229 + if(aLength < iBuf.ESize)
1.230 + TStreamBuf::DoWriteL(aPtr, aLength);
1.231 + else
1.232 + {
1.233 + TUint8* base = iBuf.iData;
1.234 + IpcWriteL(base, Lag(EWrite));
1.235 + MovePos(ERead, Lag(ERead));
1.236 + SetBuf(ERead | EWrite, base, base);
1.237 + IpcWriteL(aPtr, aLength);
1.238 + }
1.239 + }
1.240 +
1.241 +/**
1.242 +Position the mark(s) indicated by aMark at aOffset from aLocation.
1.243 +*/
1.244 +TStreamPos HIpcBuf::DoSeekL(TMark aMark, TStreamLocation aLocation, TInt aOffset)
1.245 + {
1.246 + TUint8* base = iBuf.iData;
1.247 + TInt end = EndL();
1.248 +
1.249 + switch(aLocation)
1.250 + {
1.251 + case EStreamBeginning:
1.252 + break;
1.253 + case EStreamMark:
1.254 + switch(aMark)
1.255 + {
1.256 + case ERead:
1.257 + aOffset += Mark(ERead);
1.258 + break;
1.259 + case EWrite:
1.260 + aOffset += Mark(EWrite);
1.261 + break;
1.262 + default:
1.263 + __ASSERT_ALWAYS(0, __SQLPANIC(ESqlPanicStreamMarkInvalid));
1.264 + break;
1.265 + }
1.266 + break;
1.267 + case EStreamEnd:
1.268 + aOffset += end;
1.269 + break;
1.270 + default:
1.271 + __ASSERT_ALWAYS(0, __SQLPANIC(ESqlPanicStreamLocationInvalid));
1.272 + break;
1.273 + }
1.274 + TInt r = KErrNone;
1.275 + if(aOffset < 0)
1.276 + {
1.277 + aOffset = 0;
1.278 + r = KErrEof;
1.279 + }
1.280 + else if(aOffset > end)
1.281 + {
1.282 + aOffset = end;
1.283 + r = KErrEof;
1.284 + }
1.285 +
1.286 + __ASSERT_ALWAYS(!(aMark & ~(ERead | EWrite)), __SQLPANIC(ESqlPanicStreamMarkInvalid));
1.287 + if(aMark & ERead)
1.288 + {
1.289 + TInt lag = aOffset - Pos(ERead);
1.290 + if(lag >= base - End(ERead) && lag <= 0)
1.291 + SetPtr(ERead, End(ERead) + lag);
1.292 + else
1.293 + {
1.294 + SetPos(ERead, aOffset);
1.295 + SetBuf(ERead, base, base);
1.296 + }
1.297 + }
1.298 + if(aMark & EWrite && aOffset != Mark(EWrite))
1.299 + {
1.300 + IpcWriteL(base, Lag(EWrite));
1.301 + SetPos(EWrite, aOffset);
1.302 + SetBuf(EWrite, base, base);
1.303 + }
1.304 + __SQLLEAVE_IF_ERROR(r);
1.305 + return TStreamPos(aOffset);
1.306 + }
1.307 +
1.308 +/**
1.309 +Read from the server at the current read position.
1.310 +Arg 0: not used
1.311 +Arg 1: [out] from which position to read
1.312 +Arg 2: [in/out] IPC buffer
1.313 +Arg 3: [out] max length of the requested data
1.314 +*/
1.315 +TInt HIpcBuf::IpcReadL(TAny* aPtr, TInt aMaxLength)
1.316 + {
1.317 + __ASSERT_DEBUG(aMaxLength >= 0, __SQLPANIC(ESqlPanicInternalError));
1.318 + if(aMaxLength == 0)
1.319 + return 0;
1.320 +
1.321 + TPtr8 des((TUint8*)aPtr, aMaxLength);
1.322 + TInt pos = Pos(ERead);
1.323 +
1.324 + TInt len = __SQLLEAVE_IF_ERROR(iSession.SendReceive(::MakeMsgCode(ESqlSrvStreamRead, ESqlSrvStreamHandle, iHandle), TIpcArgs(0, pos, &des, aMaxLength)));
1.325 + pos += len;
1.326 + if(len < aMaxLength)
1.327 + iBuf.iExt = pos; // end-of-file encountered
1.328 + SetPos(ERead, pos);
1.329 + return len;
1.330 + }
1.331 +
1.332 +/**
1.333 +Write to the server at the current write position.
1.334 +Arg 0: not used
1.335 +Arg 1: [out] from which position to write
1.336 +Arg 2: [in/out] IPC buffer
1.337 +*/
1.338 +void HIpcBuf::IpcWriteL(const TAny* aPtr, TInt aLength)
1.339 + {
1.340 + __ASSERT_DEBUG(aLength >= 0, __SQLPANIC(ESqlPanicInternalError));
1.341 + if(aLength == 0)
1.342 + return;
1.343 +
1.344 + TPtrC8 ptr((TUint8*)aPtr, aLength);
1.345 + TInt ext = iBuf.iExt;
1.346 + iBuf.iExt = -1;
1.347 + TInt pos = Pos(EWrite);
1.348 + __SQLLEAVE_IF_ERROR(iSession.SendReceive(::MakeMsgCode(ESqlSrvStreamWrite, ESqlSrvStreamHandle, iHandle), TIpcArgs(0, pos, &ptr)));
1.349 + pos += aLength;
1.350 + if(ext >=0 && pos > ext)
1.351 + iBuf.iExt = pos;
1.352 + SetPos(EWrite, pos);
1.353 + }
1.354 +
1.355 +/**
1.356 +Determine the end of the stream
1.357 +*/
1.358 +TInt HIpcBuf::EndL()
1.359 + {
1.360 + TInt ext = iBuf.iExt;
1.361 + if(ext < 0)
1.362 + {
1.363 + iBuf.iExt = ext = __SQLLEAVE_IF_ERROR(iSession.SendReceive(::MakeMsgCode(ESqlSrvStreamSize, ESqlSrvStreamHandle, iHandle)));
1.364 + }
1.365 + return Max(ext, Mark(EWrite));
1.366 + }
1.367 +