1.1 --- /dev/null Thu Jan 01 00:00:00 1970 +0000
1.2 +++ b/os/persistentdata/persistentstorage/dbms/sdbms/SD_BUF.CPP Fri Jun 15 03:10:57 2012 +0200
1.3 @@ -0,0 +1,368 @@
1.4 +// Copyright (c) 1998-2009 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 +// DBMS server stream buffer classes
1.18 +//
1.19 +//
1.20 +
1.21 +
1.22 +
1.23 +#include "SD_STD.H"
1.24 +
1.25 +HDbsBuf* HDbsBuf::NewLC(const RDbsObject& aObject,TDbsFunction aFunction,TIpcArgs& aArgs)
1.26 + {
1.27 + HDbsBuf* self=new(ELeave) HDbsBuf;
1.28 + self->PushL();
1.29 + self->ConstructL(aObject,aFunction,aArgs);
1.30 + return self;
1.31 + }
1.32 +
1.33 +HDbsBuf* HDbsBuf::NewL(const RDbsObject& aObject,TDbsFunction aFunction,TIpcArgs& aArgs)
1.34 + {
1.35 + HDbsBuf* self=NewLC(aObject,aFunction,aArgs);
1.36 + CleanupStack::Pop();
1.37 + return self;
1.38 + }
1.39 +
1.40 +void HDbsBuf::ConstructL(const RDbsObject& aObject,TDbsFunction aFunction,TIpcArgs& aArgs)
1.41 + {
1.42 + TPckg<TDbsStreamBuf> pckg(iBuf);
1.43 + aArgs.Set(3,&pckg);
1.44 + iIpc.OpenL(aObject,aFunction,aArgs);
1.45 + TUint8* base=iBuf.iData;
1.46 + // if reading we already have one buffer-full of data
1.47 + TInt avail=Max(0,Min(iBuf.iExt,KDbsStreamBufSize));
1.48 + SetBuf(ERead,base,base+avail);
1.49 + SetPos(ERead,avail);
1.50 + SetBuf(EWrite,base,base);
1.51 + SetPos(EWrite,0);
1.52 + }
1.53 +
1.54 +TInt HDbsBuf::UnderflowL(TInt)
1.55 +//
1.56 +// Fill the buffer's read area.
1.57 +//
1.58 + {
1.59 + // when handle is null there is no data to read from server
1.60 + if(!iIpc.Handle())
1.61 + return 0;
1.62 +
1.63 + __ASSERT(Avail(ERead)==0);
1.64 + TUint8* base=iBuf.iData;
1.65 + IpcWriteL(base,Lag(EWrite));
1.66 + SetBuf(EWrite,base,base);
1.67 +//
1.68 + TInt len=IpcReadL(base,iBuf.ESize);
1.69 + SetBuf(ERead,base,base+len);
1.70 + return len;
1.71 + }
1.72 +
1.73 +void HDbsBuf::OverflowL()
1.74 +//
1.75 +// Set up the buffer's write area.
1.76 +//
1.77 + {
1.78 + __ASSERT(Avail(EWrite)==0);
1.79 + TUint8* base=iBuf.iData;
1.80 + MovePos(ERead,Lag(ERead));
1.81 + SetBuf(ERead,base,base);
1.82 +//
1.83 + IpcWriteL(base,Lag(EWrite));
1.84 + SetBuf(EWrite,base,base+iBuf.ESize);
1.85 + }
1.86 +
1.87 +void HDbsBuf::DoRelease()
1.88 + {
1.89 + delete this;
1.90 + }
1.91 +
1.92 +void HDbsBuf::DoSynchL()
1.93 +//
1.94 +// Synchronise this buffer with its file, giving up on outstanding writes in case of failure.
1.95 +//
1.96 + {
1.97 + TUint8* base=iBuf.iData;
1.98 + MovePos(ERead,Lag(ERead));
1.99 + TInt lag=Lag(EWrite);
1.100 + SetBuf(ERead|EWrite,base,base);
1.101 + iBuf.iExt=-1;
1.102 + IpcWriteL(base,lag);
1.103 + iIpc.SendReceiveL(EDbsStreamSynch);
1.104 + }
1.105 +
1.106 +TInt HDbsBuf::DoReadL(TAny* aPtr,TInt aMaxLength)
1.107 +//
1.108 +// Read direct from ipc if asked to transfer more than a bufferful.
1.109 +//
1.110 + {
1.111 + __ASSERT(aMaxLength>=0);
1.112 + __ASSERT(aMaxLength>0);
1.113 + TInt avail=Avail(ERead);
1.114 + __ASSERT(avail>=0&&Avail(EWrite)>=0);
1.115 + if (avail>0)
1.116 + {
1.117 + TInt len=Min(aMaxLength,avail);
1.118 + TUint8* ptr=Ptr(ERead);
1.119 + aPtr=Mem::Copy(aPtr,ptr,len);
1.120 + SetPtr(ERead,ptr+len);
1.121 + aMaxLength-=len;
1.122 + if (aMaxLength==0)
1.123 + return len; // that's it
1.124 + }
1.125 + __ASSERT(Avail(ERead)==0);
1.126 + if (aMaxLength<iBuf.ESize)
1.127 + return avail+TStreamBuf::DoReadL(aPtr,aMaxLength);
1.128 +//
1.129 + // when handle is null there is no more data to read from server
1.130 + if(!iIpc.Handle())
1.131 + return avail;
1.132 +
1.133 + TUint8* base=iBuf.iData;
1.134 + IpcWriteL(base,Lag(EWrite));
1.135 + SetBuf(ERead|EWrite,base,base);
1.136 + return avail+IpcReadL(aPtr,aMaxLength);
1.137 + }
1.138 +
1.139 +void HDbsBuf::DoWriteL(const TAny* aPtr,TInt aLength)
1.140 +//
1.141 +// Write direct to ipc if asked to transfer more than a bufferful.
1.142 +//
1.143 + {
1.144 + __ASSERT(aLength>=0);
1.145 + __ASSERT(aLength>0);
1.146 + TInt avail=Avail(EWrite);
1.147 + __ASSERT(Avail(ERead)>=0&&avail>=0);
1.148 + if (avail>0)
1.149 + {
1.150 + TInt len=Min(aLength,avail);
1.151 + SetPtr(EWrite,Mem::Copy(Ptr(EWrite),aPtr,len));
1.152 + aLength-=len;
1.153 + if (aLength==0)
1.154 + return; // done
1.155 +//
1.156 + aPtr=(TUint8*)aPtr+len;
1.157 + }
1.158 + __ASSERT(Avail(EWrite)==0);
1.159 + if (aLength<iBuf.ESize)
1.160 + TStreamBuf::DoWriteL(aPtr,aLength);
1.161 + else
1.162 + {
1.163 + TUint8* base=iBuf.iData;
1.164 + IpcWriteL(base,Lag(EWrite));
1.165 + MovePos(ERead,Lag(ERead));
1.166 + SetBuf(ERead|EWrite,base,base);
1.167 + IpcWriteL(aPtr,aLength);
1.168 + }
1.169 + }
1.170 +
1.171 +TStreamPos HDbsBuf::DoSeekL(TMark aMark,TStreamLocation aLocation,TInt anOffset)
1.172 +//
1.173 +// Position the mark(s) indicated by aMark at anOffset from aLocation.
1.174 +//
1.175 + {
1.176 + TUint8* base=iBuf.iData;
1.177 + TInt end=EndL();
1.178 +//
1.179 + switch (aLocation)
1.180 + {
1.181 + case EStreamBeginning:
1.182 + break;
1.183 + case EStreamMark:
1.184 + switch (aMark)
1.185 + {
1.186 + case ERead:
1.187 + anOffset+=Mark(ERead);
1.188 + break;
1.189 + case EWrite:
1.190 + anOffset+=Mark(EWrite);
1.191 + break;
1.192 + default:
1.193 + Panic(EDbsStreamMarkInvalid);
1.194 + break;
1.195 + }
1.196 + break;
1.197 + case EStreamEnd:
1.198 + anOffset+=end;
1.199 + break;
1.200 + default:
1.201 + Panic(EDbsStreamLocationInvalid);
1.202 + break;
1.203 + }
1.204 + TInt r=KErrNone;
1.205 + if (anOffset<0)
1.206 + {
1.207 + anOffset=0;
1.208 + r=KErrEof;
1.209 + }
1.210 + else if (anOffset>end)
1.211 + {
1.212 + anOffset=end;
1.213 + r=KErrEof;
1.214 + }
1.215 +//
1.216 + __ASSERT_ALWAYS(!(aMark&~(ERead|EWrite)),Panic(EDbsStreamMarkInvalid));
1.217 + if (aMark&ERead)
1.218 + {
1.219 + TInt lag=anOffset-Pos(ERead);
1.220 + if (lag>=base-End(ERead)&&lag<=0)
1.221 + SetPtr(ERead,End(ERead)+lag);
1.222 + else
1.223 + {
1.224 + SetPos(ERead,anOffset);
1.225 + SetBuf(ERead,base,base);
1.226 + }
1.227 + }
1.228 + if (aMark&EWrite&&anOffset!=Mark(EWrite))
1.229 + {
1.230 + IpcWriteL(base,Lag(EWrite));
1.231 + SetPos(EWrite,anOffset);
1.232 + SetBuf(EWrite,base,base);
1.233 + }
1.234 + __LEAVE_IF_ERROR(r);
1.235 + return TStreamPos(anOffset);
1.236 + }
1.237 +
1.238 +TInt HDbsBuf::IpcReadL(TAny* aPtr,TInt aMaxLength)
1.239 +//
1.240 +// Read from the server at the current read position.
1.241 +//
1.242 + {
1.243 + __ASSERT(aMaxLength>=0);
1.244 + if (aMaxLength==0)
1.245 + return 0;
1.246 +//
1.247 + TPtr8 des((TUint8*)aPtr,aMaxLength);
1.248 + TInt pos=Pos(ERead);
1.249 +
1.250 + TInt len=iIpc.SendReceiveL(EDbsStreamRead,TIpcArgs(pos,&des,aMaxLength));
1.251 + pos+=len;
1.252 + if (len<aMaxLength)
1.253 + iBuf.iExt=pos; // end-of-file encountered
1.254 + SetPos(ERead,pos);
1.255 + return len;
1.256 + }
1.257 +
1.258 +void HDbsBuf::IpcWriteL(const TAny* aPtr,TInt aLength)
1.259 +//
1.260 +// Write to the server at the current write position.
1.261 +//
1.262 + {
1.263 + __ASSERT(aLength>=0);
1.264 + if (aLength==0)
1.265 + return;
1.266 +//
1.267 + TPtrC8 ptr((TUint8*)aPtr,aLength);
1.268 + TInt ext=iBuf.iExt;
1.269 + iBuf.iExt=-1;
1.270 + TInt pos=Pos(EWrite);
1.271 + iIpc.SendReceiveL(EDbsStreamWrite,TIpcArgs(pos,&ptr));
1.272 + pos+=aLength;
1.273 + if (ext>=0&&pos>ext)
1.274 + iBuf.iExt=pos;
1.275 + SetPos(EWrite,pos);
1.276 + }
1.277 +
1.278 +TInt HDbsBuf::EndL()
1.279 +//
1.280 +// Determine the end of the stream
1.281 +//
1.282 + {
1.283 + TInt ext=iBuf.iExt;
1.284 + if (ext<0)
1.285 + iBuf.iExt=ext=iIpc.SendReceiveL(EDbsStreamSize);
1.286 + return Max(ext,Mark(EWrite));
1.287 + }
1.288 +
1.289 +// Class HDbsReadBuf
1.290 +
1.291 +inline HDbsReadBuf::HDbsReadBuf(const TDesC8& aDes)
1.292 + {
1.293 + TUint8* ptr=CONST_CAST(TUint8*,aDes.Ptr());
1.294 + Set(ptr,ptr+aDes.Length(),ERead);
1.295 + }
1.296 +
1.297 +HDbsReadBuf* HDbsReadBuf::NewL(const TDesC8& aDes)
1.298 + {
1.299 + return new(ELeave) HDbsReadBuf(aDes);
1.300 + }
1.301 +
1.302 +void HDbsReadBuf::DoRelease()
1.303 + {
1.304 + delete this;
1.305 + }
1.306 +
1.307 +// Class HDbsStream
1.308 +TInt HDbsStream::ReadL(const RMessage2& aMessage)
1.309 + {
1.310 + TInt pos=aMessage.Int0();
1.311 + if (pos!=iRPos)
1.312 + iHost.SeekL(iHost.ERead,EStreamBeginning,pos);
1.313 + iRPos=-1;
1.314 + TInt len=aMessage.Int2();
1.315 + pos+=len;
1.316 + TInt tfr=len;
1.317 + for (;;)
1.318 + {
1.319 + TUint8 buf[KDbsStreamIoSize];
1.320 + TInt read=iHost.ReadL(buf,Min(tfr,KDbsStreamIoSize));
1.321 + if (read==0)
1.322 + break;
1.323 + aMessage.WriteL(1,TPtrC8(buf,read),len-tfr);
1.324 + tfr-=read;
1.325 + if (tfr==0)
1.326 + break;
1.327 + if (read<KDbsStreamIoSize)
1.328 + break;
1.329 + }
1.330 + iRPos=pos-tfr;
1.331 + return len-tfr;
1.332 + }
1.333 +
1.334 +void HDbsStream::WriteL(const RMessage2& aMessage)
1.335 + {
1.336 + TInt pos=aMessage.Int0();
1.337 + if (pos!=iWPos)
1.338 + iHost.SeekL(iHost.EWrite,EStreamBeginning,pos);
1.339 + iWPos=-1;
1.340 + TInt offset=0;
1.341 + TBuf8<KDbsStreamIoSize> buf;
1.342 + for (;;)
1.343 + {
1.344 + aMessage.ReadL(1,buf,offset);
1.345 + TInt len=buf.Length();
1.346 + if (len==0)
1.347 + break;
1.348 + iHost.WriteL(buf.Ptr(),len);
1.349 + offset+=len;
1.350 + if (len<KDbsStreamIoSize)
1.351 + break;
1.352 + }
1.353 + iWPos=pos+offset;
1.354 + }
1.355 +
1.356 +// Class HBufBuf
1.357 +
1.358 +void HBufBuf::DoRelease()
1.359 + {
1.360 + delete this;
1.361 + }
1.362 +
1.363 +HBufBuf* HBufBuf::NewLC()
1.364 + {
1.365 + HBufBuf* self=new(ELeave) HBufBuf;
1.366 + self->PushL();
1.367 + self->iBuf=CBufSeg::NewL(EGranularity);
1.368 + self->Set(*self->iBuf,0,ERead|EWrite);
1.369 + return self;
1.370 + }
1.371 +