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