os/persistentdata/persistentstorage/sql/SRC/Client/IPC/IPCBuf.cpp
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
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".
     7 //
     8 // Initial Contributors:
     9 // Nokia Corporation - initial contribution.
    10 //
    11 // Contributors:
    12 //
    13 // Description:
    14 //
    15 
    16 #include <e32base.h>
    17 #include "IPCBuf.h"
    18 #include "SqlAssert.h"
    19 #include "SqlDbSession.h"
    20 #include "OstTraceDefinitions.h"
    21 #ifdef OST_TRACE_COMPILER_IN_USE
    22 #include "IPCBufTraces.h"
    23 #endif
    24 #include "SqlTraceDef.h"
    25 
    26 /**
    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.
    30 
    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.
    34 
    35 @return A pointer to the created HIpcBuf instance
    36 
    37 @leave KErrNoMemory Out of memory.
    38 */
    39 HIpcBuf* HIpcBuf::NewLC(RSqlDbSession& aSession, TInt aFunction, TIpcArgs& aArgs)
    40 	{
    41 	HIpcBuf* self = new (ELeave) HIpcBuf(aSession);
    42 	self->PushL();
    43 	self->ConstructL(aFunction, aArgs);
    44 	return self;
    45 	}
    46 
    47 /**
    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.
    50 
    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.
    54 
    55 @return A pointer to the created HIpcBuf instance
    56 
    57 @leave KErrNoMemory Out of memory.
    58 */
    59 HIpcBuf* HIpcBuf::NewL(RSqlDbSession& aSession, TInt aFunction, TIpcArgs& aArgs)
    60 	{
    61 	HIpcBuf* self = NewLC(aSession, aFunction, aArgs);
    62 	CleanupStack::Pop();
    63 	return self;
    64 	}
    65 
    66 /**
    67 Standard, phase-two HIpcBuf construction method.
    68 
    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.
    71 
    72 @leave KErrNoMemory Out of memory.
    73 
    74 Usage of the IPC call arguments:
    75 Arg 2: [in/out]  IPC buffer
    76 iBuf.iExt: [in]  stream size in bytes
    77 */
    78 void HIpcBuf::ConstructL(TInt aFunction, TIpcArgs& aArgs)
    79 	{
    80 	TPckg<TIpcStreamBuf> pckg(iBuf);
    81 	aArgs.Set(2, &pckg);
    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);
    88 	SetPos(ERead, avail);
    89 	SetBuf(EWrite, base, base);
    90 	SetPos(EWrite, 0);
    91 	}
    92 
    93 /**
    94 @param aSession A reference to a sesion object.
    95 */
    96 HIpcBuf::HIpcBuf(RSqlDbSession& aSession) :
    97 	iSession(aSession),
    98 	iHandle(0),
    99 	iRPos(0),
   100 	iWPos(0)
   101 	{
   102 	iBuf.iExt = -1;
   103 	}
   104 
   105 /**
   106 */
   107 HIpcBuf::~HIpcBuf()
   108 	{
   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.
   111 		{
   112 		(void)iSession.SendReceive(::MakeMsgCode(ESqlSrvStreamClose, ESqlSrvStreamHandle, iHandle));
   113 		}
   114 	}
   115 
   116 /**
   117 Fill the buffer's read area.
   118 */
   119 TInt HIpcBuf::UnderflowL(TInt)
   120 	{
   121 	// when handle is null there is no data to read from server
   122 	if(!iHandle)
   123 		{
   124 		return 0;
   125 		}
   126 	__ASSERT_DEBUG(Avail(ERead) == 0, __SQLPANIC(ESqlPanicInternalError));
   127 	TUint8* base=iBuf.iData;
   128 	IpcWriteL(base,Lag(EWrite));
   129 	SetBuf(EWrite,base,base);
   130 
   131 	TInt len=IpcReadL(base,iBuf.ESize);
   132 	SetBuf(ERead,base,base+len);
   133 	return len;
   134 	}
   135 
   136 /**
   137 Set up the buffer's write area.
   138 */
   139 void HIpcBuf::OverflowL()
   140 	{
   141 	__ASSERT_DEBUG(Avail(EWrite) == 0, __SQLPANIC(ESqlPanicInternalError));
   142 	
   143 	TUint8* base = iBuf.iData;
   144 	MovePos(ERead, Lag(ERead));
   145 	SetBuf(ERead, base, base);
   146 
   147 	IpcWriteL(base, Lag(EWrite));
   148 	SetBuf(EWrite, base, base + iBuf.ESize);
   149 	}
   150 
   151 /**
   152 Destroys HIpcBuf instance.
   153 */
   154 void HIpcBuf::DoRelease()
   155 	{
   156 	delete this;
   157 	}
   158 
   159 /**
   160 Synchronise this buffer with its file, giving up on outstanding writes in case of failure.
   161 */
   162 void HIpcBuf::DoSynchL()
   163 	{
   164 	TUint8* base = iBuf.iData;
   165 	MovePos(ERead, Lag(ERead));
   166 	TInt lag = Lag(EWrite);
   167 	SetBuf(ERead | EWrite, base, base);
   168 	iBuf.iExt = -1;
   169 	IpcWriteL(base, lag);
   170 	__SQLLEAVE_IF_ERROR(iSession.SendReceive(::MakeMsgCode(ESqlSrvStreamSynch, ESqlSrvStreamHandle, iHandle)));
   171 	}
   172 
   173 /**
   174 Read direct from ipc if asked to transfer more than a bufferful.
   175 */
   176 TInt HIpcBuf::DoReadL(TAny* aPtr, TInt aMaxLength)
   177 	{
   178 	__ASSERT_DEBUG(aMaxLength > 0, __SQLPANIC(ESqlPanicInternalError));
   179 	TInt avail = Avail(ERead);
   180 	__ASSERT_DEBUG(avail >= 0 && Avail(EWrite) >= 0, __SQLPANIC(ESqlPanicInternalError));
   181 	if(avail > 0)
   182 		{
   183 		TInt len = Min(aMaxLength, avail);
   184 		TUint8* ptr = Ptr(ERead);
   185 		aPtr = Mem::Copy(aPtr, ptr, len);
   186 		SetPtr(ERead, ptr + len);
   187 		aMaxLength -= len;
   188 		if(aMaxLength == 0)
   189 			return len; // that's it
   190 		}
   191 	__ASSERT_DEBUG(Avail(ERead) == 0, __SQLPANIC(ESqlPanicInternalError));
   192 	if(aMaxLength < iBuf.ESize)
   193 		return avail + TStreamBuf::DoReadL(aPtr, aMaxLength);
   194 
   195 	// when handle is null there is no more data to read from server
   196 	if(!iHandle)
   197 		{
   198 		return 0;
   199 		}
   200 
   201 	TUint8* base = iBuf.iData;
   202 	IpcWriteL(base, Lag(EWrite));
   203 	SetBuf(ERead | EWrite, base, base);
   204 	return avail + IpcReadL(aPtr, aMaxLength);
   205 	}
   206 
   207 /**
   208 Write direct to ipc if asked to transfer more than a bufferful.
   209 */
   210 void HIpcBuf::DoWriteL(const TAny* aPtr,TInt aLength)
   211 	{
   212 	__ASSERT_DEBUG(aLength > 0, __SQLPANIC(ESqlPanicInternalError));
   213 	TInt avail = Avail(EWrite);
   214 	__ASSERT_DEBUG(Avail(ERead) >= 0 && avail >= 0, __SQLPANIC(ESqlPanicInternalError));
   215 	if(avail > 0)
   216 		{
   217 		TInt len = Min(aLength, avail);
   218 		SetPtr(EWrite, Mem::Copy(Ptr(EWrite), aPtr, len));
   219 		aLength -= len;
   220 		if(aLength == 0)
   221 			return; // done
   222 
   223 		aPtr = (TUint8*)aPtr + len;
   224 		}
   225 	__ASSERT_DEBUG(Avail(EWrite) == 0, __SQLPANIC(ESqlPanicInternalError));
   226 	if(aLength < iBuf.ESize)
   227 		TStreamBuf::DoWriteL(aPtr, aLength);
   228 	else
   229 		{
   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);
   235 		}
   236 	}
   237 
   238 /**
   239 Position the mark(s) indicated by aMark at aOffset from aLocation.
   240 */
   241 TStreamPos HIpcBuf::DoSeekL(TMark aMark, TStreamLocation aLocation, TInt aOffset)
   242 	{
   243 	TUint8* base = iBuf.iData;
   244 	TInt end = EndL();
   245 
   246 	switch(aLocation)
   247 		{
   248 	case EStreamBeginning:
   249 		break;
   250 	case EStreamMark:
   251 		switch(aMark)
   252 			{
   253 		case ERead:
   254 			aOffset += Mark(ERead);
   255 			break;
   256 		case EWrite:
   257 			aOffset += Mark(EWrite);
   258 			break;
   259 		default:
   260 		    __ASSERT_ALWAYS(0, __SQLPANIC(ESqlPanicStreamMarkInvalid));
   261 			break;
   262 			}
   263 		break;
   264 	case EStreamEnd:
   265 		aOffset += end;
   266 		break;
   267 	default:
   268 	    __ASSERT_ALWAYS(0, __SQLPANIC(ESqlPanicStreamLocationInvalid));
   269 		break;
   270 		}
   271 	TInt r = KErrNone;
   272 	if(aOffset < 0)
   273 		{
   274 		aOffset = 0;
   275 		r = KErrEof;
   276 		}
   277 	else if(aOffset > end)
   278 		{
   279 		aOffset = end;
   280 		r = KErrEof;
   281 		}
   282 
   283 	__ASSERT_ALWAYS(!(aMark & ~(ERead | EWrite)), __SQLPANIC(ESqlPanicStreamMarkInvalid));
   284 	if(aMark & ERead)
   285 		{
   286 		TInt lag = aOffset - Pos(ERead);
   287 		if(lag >= base - End(ERead) && lag <= 0)
   288 			SetPtr(ERead, End(ERead) + lag);
   289 		else
   290 			{
   291 			SetPos(ERead, aOffset);
   292 			SetBuf(ERead, base, base);
   293 			}
   294 		}
   295 	if(aMark & EWrite && aOffset != Mark(EWrite))
   296 		{
   297 		IpcWriteL(base, Lag(EWrite));
   298 		SetPos(EWrite, aOffset);
   299 		SetBuf(EWrite, base, base);
   300 		}
   301 	__SQLLEAVE_IF_ERROR(r);
   302 	return TStreamPos(aOffset);
   303 	}
   304 
   305 /**
   306 Read from the server at the current read position.
   307 Arg 0:            not used
   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
   311 */
   312 TInt HIpcBuf::IpcReadL(TAny* aPtr, TInt aMaxLength)
   313 	{
   314 	__ASSERT_DEBUG(aMaxLength >= 0, __SQLPANIC(ESqlPanicInternalError));
   315 	if(aMaxLength == 0)
   316 		return 0;
   317 
   318 	TPtr8 des((TUint8*)aPtr, aMaxLength);
   319 	TInt pos = Pos(ERead);
   320 		
   321 	TInt len = __SQLLEAVE_IF_ERROR(iSession.SendReceive(::MakeMsgCode(ESqlSrvStreamRead, ESqlSrvStreamHandle, iHandle), TIpcArgs(0, pos, &des, aMaxLength)));
   322 	pos += len;
   323 	if(len < aMaxLength)
   324 		iBuf.iExt = pos; // end-of-file encountered
   325 	SetPos(ERead, pos);
   326 	return len;
   327 	}
   328 
   329 /**
   330 Write to the server at the current write position.
   331 Arg 0:            not used
   332 Arg 1: [out]      from which position to write
   333 Arg 2: [in/out]   IPC buffer
   334 */
   335 void HIpcBuf::IpcWriteL(const TAny* aPtr, TInt aLength)
   336 	{
   337 	__ASSERT_DEBUG(aLength >= 0, __SQLPANIC(ESqlPanicInternalError));
   338 	if(aLength == 0)
   339 		return;
   340 
   341 	TPtrC8 ptr((TUint8*)aPtr, aLength);
   342 	TInt ext = iBuf.iExt;
   343 	iBuf.iExt = -1;
   344 	TInt pos = Pos(EWrite);
   345 	__SQLLEAVE_IF_ERROR(iSession.SendReceive(::MakeMsgCode(ESqlSrvStreamWrite, ESqlSrvStreamHandle, iHandle), TIpcArgs(0, pos, &ptr)));
   346 	pos += aLength;
   347 	if(ext >=0 && pos > ext)
   348 		iBuf.iExt = pos;
   349 	SetPos(EWrite, pos);
   350 	}
   351 
   352 /**
   353 Determine the end of the stream
   354 */
   355 TInt HIpcBuf::EndL()
   356 	{
   357 	TInt ext = iBuf.iExt;
   358 	if(ext < 0)
   359 		{
   360 		iBuf.iExt = ext = __SQLLEAVE_IF_ERROR(iSession.SendReceive(::MakeMsgCode(ESqlSrvStreamSize, ESqlSrvStreamHandle, iHandle)));
   361 		}
   362 	return Max(ext, Mark(EWrite));
   363 	}
   364