os/security/securityanddataprivacytools/securitytools/certapp/store--/s32file.cpp
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
sl@0
     1
/*
sl@0
     2
* Copyright (c) 2008-2009 Nokia Corporation and/or its subsidiary(-ies).
sl@0
     3
* All rights reserved.
sl@0
     4
* This component and the accompanying materials are made available
sl@0
     5
* under the terms of the License "Eclipse Public License v1.0"
sl@0
     6
* which accompanies this distribution, and is available
sl@0
     7
* at the URL "http://www.eclipse.org/legal/epl-v10.html".
sl@0
     8
*
sl@0
     9
* Initial Contributors:
sl@0
    10
* Nokia Corporation - initial contribution.
sl@0
    11
*
sl@0
    12
* Contributors:
sl@0
    13
*
sl@0
    14
* Description: 
sl@0
    15
*
sl@0
    16
*/
sl@0
    17
sl@0
    18
sl@0
    19
#include <s32file.h>
sl@0
    20
#include "s32file_private.h"
sl@0
    21
#include <e32err.h>
sl@0
    22
#include <fstream>
sl@0
    23
#include <set>
sl@0
    24
#include "us_std.h"
sl@0
    25
#include "filestream.h"
sl@0
    26
#include "logger.h"
sl@0
    27
using namespace store_private;
sl@0
    28
sl@0
    29
//The offset of the header of a permanent file store.
sl@0
    30
//Since all permanent file store operations work in their own coordinate system, where physical file offset 32 is
sl@0
    31
//logical offset 0, KPermanentStoreHeaderOffset is set with -16, which means that the physical file offset is 32 - 16 = 16.
sl@0
    32
const TInt KPermanentStoreHeaderOffset=-16;
sl@0
    33
sl@0
    34
//Permanent file store header length: sizeof(backup TOC ref) + sizeof(handle) + sizeof(TOC ref) + sizeof(crc) = 4 + 4 + 4 + 2 = 14
sl@0
    35
const TInt KPermanentStoreHeaderLength=14;
sl@0
    36
sl@0
    37
//Backup TOC ref length - 4 bytes
sl@0
    38
const TInt KPermanentStoreBackupLength=4;
sl@0
    39
sl@0
    40
const TInt KFileStoreStartOffset=sizeof(TCheckedUid);
sl@0
    41
sl@0
    42
const TInt KBaseOffset=KFileStoreStartOffset-KPermanentStoreHeaderOffset;
sl@0
    43
sl@0
    44
//
sl@0
    45
const TInt KMaskHandleHash=0xff000000;
sl@0
    46
const TInt KHandleInvalid=0x80000000;
sl@0
    47
const TInt KHandleTocBase=0x40000000;
sl@0
    48
const TInt KMaskHandleClear=0x30000000;
sl@0
    49
const TInt KMaskHandleGen=0x0f000000;
sl@0
    50
const TInt KIncHandleGen=0x01000000;
sl@0
    51
//
sl@0
    52
const TInt KMaxHandleIndex=0x00ffffff;
sl@0
    53
const TInt KMaskHandleIndex=0x00ffffff;
sl@0
    54
const TInt KSizeHandleIndex=3;
sl@0
    55
//
sl@0
    56
const TInt KTocDeltaCap = 64;		//up to 64 entries in a delta TOC
sl@0
    57
const TInt KTocDeltaMagic = 2;
sl@0
    58
const TInt KMaxTocDeltaMagic = KMaxTUint16;
sl@0
    59
const TInt KTocDelta=KHandleInvalid;
sl@0
    60
//
sl@0
    61
const TInt KOffsetTocHeader=-12;
sl@0
    62
const TInt KSizeTocEntry=5;		//base toc entry size is 5 bytes (when stored in the file, 8 bytes when presented in memory)
sl@0
    63
const TInt KSizeTocDeltaEntry=8;//delta toc entry size is 8 bytes
sl@0
    64
const TInt KSizeTocDeltaExtra=7;
sl@0
    65
const TInt KElementsTocBuf=48;
sl@0
    66
const TInt KBackTocBuf=20*KSizeTocEntry;
sl@0
    67
const TInt KSizeTocBuf=KElementsTocBuf*KSizeTocEntry;
sl@0
    68
sl@0
    69
enum TFrameType16
sl@0
    70
	{
sl@0
    71
	EFrameFree16=0x0000,
sl@0
    72
	EFrameData16=0x4000,
sl@0
    73
	EFrameDescriptive16=0x8000,
sl@0
    74
	EFrameContinuation16=0xc000
sl@0
    75
	};
sl@0
    76
typedef TUint16 TFrameDes16;
sl@0
    77
const TInt KSizeFrameDes16=sizeof(TFrameDes16);
sl@0
    78
const TInt KShiftFrameDes16=1;
sl@0
    79
//
sl@0
    80
const TInt KMaskFrame16=0xffff;
sl@0
    81
const TInt KMaskFrameType16=0xc000;
sl@0
    82
#if defined(__SMALL_FRAME)
sl@0
    83
const TInt KMaskFrameLength16=0xf;
sl@0
    84
const TInt KShiftFrameLength16=4;
sl@0
    85
#else
sl@0
    86
const TInt KMaskFrameLength16=0x3fff;
sl@0
    87
const TInt KShiftFrameLength16=14;
sl@0
    88
#endif
sl@0
    89
//
sl@0
    90
const TInt KFrameOpen16=0;
sl@0
    91
const TInt KFrameMinLength16=1;
sl@0
    92
const TInt KFrameMaxLength16=KMaskFrameLength16;
sl@0
    93
const TInt KFrameFullLength16=KMaskFrameLength16+1;
sl@0
    94
//
sl@0
    95
const TInt KFrameNonexistent16=-1;
sl@0
    96
sl@0
    97
TUint32 frame2File(TUint32 anOffset)
sl@0
    98
{
sl@0
    99
	// The frame offset appears to be to calculated as an offset into
sl@0
   100
	// a series of maximal length frames.  Each maximal frame can hold
sl@0
   101
	// 0x4000 bytes of data and has a 2 byte header. Therefore a frame
sl@0
   102
	// offset of x needs an extra 2 bytes for each maximal frame. The
sl@0
   103
	// base is the first byte of the payload of the first frame, so we
sl@0
   104
	// do not need to allow for the header of the first frame.
sl@0
   105
	//
sl@0
   106
	// anOffset>>KShiftFrameLength16 gives the number of maximal
sl@0
   107
	// frames. The left shift multiples by 2 to give minimum space
sl@0
   108
	// required for frame headers. Adding to the original gives the
sl@0
   109
	// file offset.
sl@0
   110
	//
sl@0
   111
	// This logic is still used even if space for the target frame is
sl@0
   112
	// really ocupied by a series of smaller frames.
sl@0
   113
	anOffset += (anOffset>>KShiftFrameLength16<<KShiftFrameDes16);
sl@0
   114
sl@0
   115
	// Add 32 to allow for file header
sl@0
   116
	anOffset += KBaseOffset;
sl@0
   117
	return anOffset;
sl@0
   118
}
sl@0
   119
sl@0
   120
TUint32 file2Frame(TUint32 aFileOffset)
sl@0
   121
{
sl@0
   122
	// subtract 32 to allow for file header
sl@0
   123
	aFileOffset -= KBaseOffset;
sl@0
   124
sl@0
   125
	// File offset is to the first byte of payload of a stream. We
sl@0
   126
	// need to remove the 2 byte frame header for the (theoretical)
sl@0
   127
	// maximal length frames before it.
sl@0
   128
sl@0
   129
	// Calculate the number of (theoretical) maximal length frames before the file position
sl@0
   130
	// n = aFileOffset/0x4002
sl@0
   131
	TInt n=aFileOffset/(KFrameFullLength16+KSizeFrameDes16);
sl@0
   132
	
sl@0
   133
	// Subtract the frame header length for each maximal frame
sl@0
   134
	aFileOffset -= n<<KShiftFrameDes16;
sl@0
   135
sl@0
   136
	return aFileOffset;
sl@0
   137
}
sl@0
   138
sl@0
   139
sl@0
   140
sl@0
   141
//TPermanentStoreHeader class.
sl@0
   142
//
sl@0
   143
//Represents the data kept in the permanent store file header.
sl@0
   144
//Data members:
sl@0
   145
// - iBackup - "backup TOC reference", 32-bits integer, which keeps the 31-bit file offset of the backup TOC.
sl@0
   146
//             Plays important role in the "store commit" procedure.
sl@0
   147
//             The LSB is a "dirty" bit. If during the store opening phase the dirty bit is found to be set,
sl@0
   148
//             then it means - the previous "store commit" operation has not been completed successfully and
sl@0
   149
//             the backup TOC shall be used instead of the TOC;
sl@0
   150
// - iHandle - 32-bit stream handle (MSB - invalid/deleted, 3 bits - unused, 4 bits - generation counter, 24 bits - stream handle).
sl@0
   151
//             Plays important role in the "stream relocation" procedure during store compaction.
sl@0
   152
//             iHandle keeps the handle of the stream being relocated, so if the commit phase fails, the original stream entry
sl@0
   153
//             can be restored at the moment when the store is reopened;
sl@0
   154
// - iRef    - Current "TOC reference". Represents a file offset, where the current TOC is;
sl@0
   155
// - iCrc    - 16-bit CRC, protecting iBackup, iHandle, iRef;
sl@0
   156
class TPermanentStoreHeader
sl@0
   157
	{
sl@0
   158
public:
sl@0
   159
	TPermanentStoreHeader() : iBackup(0), iHandle(0), iRef(0), iCrc(0) {}
sl@0
   160
	TUint8* Ptr();
sl@0
   161
	const TUint8* Ptr() const;
sl@0
   162
	TBool IsValid() const;
sl@0
   163
//
sl@0
   164
	TPermanentStoreHeader(TInt aToc);
sl@0
   165
	TPermanentStoreHeader(TInt aBackupToc,TInt aHandle,TInt aReference);
sl@0
   166
//
sl@0
   167
	TBool IsDirty() const;
sl@0
   168
	void MarkDirty();
sl@0
   169
	void SetBackupToc(TInt aBackupToc);
sl@0
   170
	TInt BackupToc() const;
sl@0
   171
//
sl@0
   172
	TInt Handle() const;
sl@0
   173
	TInt Reference() const;
sl@0
   174
private:
sl@0
   175
	void Set(TInt aBackupToc,TInt aHandle,TInt aReference);
sl@0
   176
private:
sl@0
   177
	TUint32 iBackup;
sl@0
   178
	TInt32 iHandle;
sl@0
   179
	TInt32 iRef;
sl@0
   180
	TUint16 iCrc;
sl@0
   181
public:
sl@0
   182
	// Changes for port
sl@0
   183
	void InternalizeL(RReadStream &aStream);
sl@0
   184
	void ExternalizeL(RWriteStream &aStream) const;
sl@0
   185
	};
sl@0
   186
sl@0
   187
// Class TPermanentStoreHeader
sl@0
   188
TUint8* TPermanentStoreHeader::Ptr()
sl@0
   189
	{return REINTERPRET_CAST(TUint8*,&iBackup);}
sl@0
   190
const TUint8* TPermanentStoreHeader::Ptr() const
sl@0
   191
	{return REINTERPRET_CAST(const TUint8*,&iBackup);}
sl@0
   192
sl@0
   193
sl@0
   194
TPermanentStoreHeader::TPermanentStoreHeader(TInt aToc)
sl@0
   195
{Set(aToc,0,aToc);}
sl@0
   196
sl@0
   197
TPermanentStoreHeader::TPermanentStoreHeader(TInt aBackupToc,TInt aHandle,TInt aReference)
sl@0
   198
{
sl@0
   199
	__ASSERT_DEBUG(aHandle!=0,User::Invariant());
sl@0
   200
	Set(aBackupToc,aHandle,aReference);
sl@0
   201
}
sl@0
   202
sl@0
   203
TBool TPermanentStoreHeader::IsDirty() const
sl@0
   204
{
sl@0
   205
	return iBackup&0x1;
sl@0
   206
}
sl@0
   207
sl@0
   208
void TPermanentStoreHeader::MarkDirty()
sl@0
   209
{
sl@0
   210
	iBackup|=0x1;
sl@0
   211
}
sl@0
   212
sl@0
   213
void TPermanentStoreHeader::SetBackupToc(TInt aBackupToc)
sl@0
   214
{
sl@0
   215
	__ASSERT_DEBUG(aBackupToc>=0,User::Invariant());
sl@0
   216
	iBackup=TUint32(aBackupToc)<<1;
sl@0
   217
}
sl@0
   218
sl@0
   219
TInt TPermanentStoreHeader::BackupToc() const
sl@0
   220
{
sl@0
   221
	return iBackup>>1;
sl@0
   222
}
sl@0
   223
sl@0
   224
TInt TPermanentStoreHeader::Handle() const
sl@0
   225
{
sl@0
   226
	return iHandle;
sl@0
   227
}
sl@0
   228
TInt TPermanentStoreHeader::Reference() const
sl@0
   229
{
sl@0
   230
	return iRef;
sl@0
   231
}
sl@0
   232
sl@0
   233
sl@0
   234
TBool TPermanentStoreHeader::IsValid() const
sl@0
   235
{
sl@0
   236
	if (IsDirty())
sl@0
   237
		return EFalse;
sl@0
   238
sl@0
   239
	TUint16 crc=0;
sl@0
   240
	Mem::Crc(crc,Ptr(),_FOFF(TPermanentStoreHeader,iCrc));
sl@0
   241
sl@0
   242
	return crc==iCrc;
sl@0
   243
}
sl@0
   244
sl@0
   245
//Sets the "backup TOC ref", "handle" and "TOC ref" in current TPermanentStoreHeader object.
sl@0
   246
//16-bit CRC is calculated, based on the values of the input parameters, and stored together with them in the 
sl@0
   247
//TPermanentStoreHeader object.
sl@0
   248
void TPermanentStoreHeader::Set(TInt aBackupToc,TInt aHandle,TInt aReference)
sl@0
   249
	{
sl@0
   250
	iBackup=TUint32(aBackupToc)<<1;
sl@0
   251
	iHandle=aHandle;
sl@0
   252
	iRef=aReference;
sl@0
   253
	iCrc=0;
sl@0
   254
	Mem::Crc(iCrc,Ptr(),_FOFF(TPermanentStoreHeader,iCrc));
sl@0
   255
	__ASSERT_DEBUG(IsValid(),User::Invariant());
sl@0
   256
	}
sl@0
   257
sl@0
   258
sl@0
   259
void TPermanentStoreHeader::InternalizeL(RReadStream &aStream)
sl@0
   260
{
sl@0
   261
	aStream >> iBackup;
sl@0
   262
	aStream >> iHandle;
sl@0
   263
	aStream >> iRef;
sl@0
   264
	aStream >> iCrc;
sl@0
   265
}
sl@0
   266
sl@0
   267
void TPermanentStoreHeader::ExternalizeL(RWriteStream &aStream) const
sl@0
   268
{
sl@0
   269
	aStream << iBackup;
sl@0
   270
	aStream << iHandle;
sl@0
   271
	aStream << iRef;
sl@0
   272
	aStream << iCrc;
sl@0
   273
}
sl@0
   274
sl@0
   275
struct TocHeader
sl@0
   276
	{
sl@0
   277
	TInt32 iPrimary;
sl@0
   278
	TInt32 iAvail;
sl@0
   279
	TUint32 iCount;
sl@0
   280
	void ExternalizeL(RWriteStream &aStream) const;
sl@0
   281
	void InternalizeL(RReadStream &aStream);
sl@0
   282
	};
sl@0
   283
sl@0
   284
void TocHeader::ExternalizeL(RWriteStream &aStream) const
sl@0
   285
{
sl@0
   286
	aStream << iPrimary;
sl@0
   287
	aStream << iAvail;
sl@0
   288
	aStream << iCount;
sl@0
   289
}
sl@0
   290
sl@0
   291
void TocHeader::InternalizeL(RReadStream &aStream)
sl@0
   292
{
sl@0
   293
	aStream >> iPrimary;
sl@0
   294
	aStream >> iAvail;
sl@0
   295
	aStream >> iCount;
sl@0
   296
}
sl@0
   297
sl@0
   298
struct STocEntry
sl@0
   299
	{
sl@0
   300
	TInt32 handle;
sl@0
   301
	TInt32 ref;
sl@0
   302
	void ExternalizeL(RWriteStream &aStream) const;
sl@0
   303
	void InternalizeL(RReadStream &aStream);
sl@0
   304
	};
sl@0
   305
sl@0
   306
sl@0
   307
void STocEntry::ExternalizeL(RWriteStream &aStream) const
sl@0
   308
{
sl@0
   309
	aStream.WriteL((TUint8*)&handle + KSizeHandleIndex, KSizeTocEntry);
sl@0
   310
}
sl@0
   311
sl@0
   312
void STocEntry::InternalizeL(RReadStream &aStream)
sl@0
   313
{
sl@0
   314
	aStream.ReadL((TUint8*)&handle + KSizeHandleIndex, KSizeTocEntry);
sl@0
   315
}
sl@0
   316
sl@0
   317
sl@0
   318
CStreamStore::~CStreamStore()
sl@0
   319
{
sl@0
   320
}
sl@0
   321
sl@0
   322
sl@0
   323
TStreamId CPersistentStore::Root()
sl@0
   324
{
sl@0
   325
	return iRoot;
sl@0
   326
}
sl@0
   327
sl@0
   328
void CPersistentStore::SetRootL(TStreamId anId)
sl@0
   329
{
sl@0
   330
	iRoot = anId;
sl@0
   331
}
sl@0
   332
sl@0
   333
void CFileStore::SetTypeL(const TUidType& aType)
sl@0
   334
{
sl@0
   335
	if(aType != KPermanentFileStoreLayoutUid) FatalError();
sl@0
   336
}
sl@0
   337
sl@0
   338
sl@0
   339
CFileStore *CPermanentFileStore::ReplaceLC(RFs &aFs, const TDesC &aName, TUint aFileMode)
sl@0
   340
{
sl@0
   341
	(void) aFs;
sl@0
   342
	return new CPermanentFileStore(aName, aFileMode);
sl@0
   343
}
sl@0
   344
sl@0
   345
CPermanentFileStore* CPermanentFileStore::OpenLC(RFs& aFs,const TDesC& aName,TUint aFileMode)
sl@0
   346
{
sl@0
   347
	(void) aFs;
sl@0
   348
	CPermanentFileStore *pfs = new CPermanentFileStore(aName, aFileMode);
sl@0
   349
	pfs->ReadInData();
sl@0
   350
	return pfs;
sl@0
   351
}
sl@0
   352
sl@0
   353
CPermanentFileStore::CPermanentFileStore(const TDesC& aName,TUint aFileMode)
sl@0
   354
	: iWritable(false), iFileIn(), iFileOut(), iLastStreamIdCreated(0), 
sl@0
   355
	  iStreams()
sl@0
   356
{
sl@0
   357
	TBuf8<KMaxName> utf8Name;
sl@0
   358
	utf8Name.Copy(aName);
sl@0
   359
sl@0
   360
	std::string name((const char *)utf8Name.Ptr(), utf8Name.Length());
sl@0
   361
	
sl@0
   362
	if(aFileMode & EFileWrite)
sl@0
   363
		{
sl@0
   364
		iFileOut.reset(new FileWriteStream(name));
sl@0
   365
		iWritable = true;
sl@0
   366
		}
sl@0
   367
	else
sl@0
   368
		{
sl@0
   369
		iFileIn.reset(new FileReadStream(name));
sl@0
   370
		}
sl@0
   371
}
sl@0
   372
sl@0
   373
CPermanentFileStore::~CPermanentFileStore()
sl@0
   374
{
sl@0
   375
	while(!iStreams.empty())
sl@0
   376
		{
sl@0
   377
		MemStreamBuf *p = (*iStreams.begin()).second;
sl@0
   378
		iStreams.erase(iStreams.begin());
sl@0
   379
		delete p;
sl@0
   380
		}
sl@0
   381
}
sl@0
   382
sl@0
   383
sl@0
   384
MemStreamBuf *CPermanentFileStore::CreateStoreWriteStream()
sl@0
   385
{
sl@0
   386
	TStreamId id = ++iLastStreamIdCreated;
sl@0
   387
	MemStreamBuf *handler = new MemStreamBuf(*this, id);
sl@0
   388
	
sl@0
   389
	MemStreamBufPtrMap::value_type value(id, handler);
sl@0
   390
	std::pair<MemStreamBufPtrMap::iterator, bool> ret = iStreams.insert(value);
sl@0
   391
sl@0
   392
	if(!ret.second)
sl@0
   393
		{
sl@0
   394
		// Insert failed
sl@0
   395
		FatalError();
sl@0
   396
		}
sl@0
   397
	
sl@0
   398
sl@0
   399
	return handler;
sl@0
   400
}
sl@0
   401
sl@0
   402
MemStreamBuf *CPermanentFileStore::CreateStoreReadStream(TStreamId aId) const
sl@0
   403
{
sl@0
   404
	MemStreamBufPtrMap::const_iterator it = iStreams.find(TUint32(aId));
sl@0
   405
	if(it == iStreams.end())
sl@0
   406
		{
sl@0
   407
		// Lookup failed
sl@0
   408
		dbg << Log::Indent() << "StreamId 0x" << aId << " does not exist in store!" << Log::Endl();
sl@0
   409
		FatalError();
sl@0
   410
		}
sl@0
   411
	return (*it).second;
sl@0
   412
}
sl@0
   413
sl@0
   414
bool CPermanentFileStore::Writable() const
sl@0
   415
{
sl@0
   416
	return iWritable;
sl@0
   417
}
sl@0
   418
sl@0
   419
sl@0
   420
sl@0
   421
void CPermanentFileStore::CommitL()
sl@0
   422
{
sl@0
   423
	if(!iWritable) FatalError();	
sl@0
   424
sl@0
   425
	TCheckedUid id(KPermanentFileStoreLayoutUid);
sl@0
   426
sl@0
   427
	*iFileOut << id;
sl@0
   428
sl@0
   429
	// Write an empty header, we will correct the fields later
sl@0
   430
	TPermanentStoreHeader tmpheader(0);
sl@0
   431
	*iFileOut << tmpheader;
sl@0
   432
sl@0
   433
	//
sl@0
   434
	// Now write out the streams
sl@0
   435
	//
sl@0
   436
	std::vector<TUint> streamOffsets;
sl@0
   437
	for(MemStreamBufPtrMap::const_iterator streamIt = iStreams.begin();
sl@0
   438
		streamIt != iStreams.end(); 
sl@0
   439
		++streamIt)
sl@0
   440
		{
sl@0
   441
		const MemStreamBuf::StreamData &streamData((*streamIt).second->GetData());
sl@0
   442
sl@0
   443
		// Record the offset to the first byte of the payload for use in the TOC
sl@0
   444
		streamOffsets.push_back(TUint32(iFileOut->iSnk->TellL(MStreamBuf::EWrite))+KSizeFrameDes16);
sl@0
   445
		
sl@0
   446
		TUint32 dataAvailable = streamData.size();
sl@0
   447
		MemStreamBuf::StreamData::const_iterator dataIt = streamData.begin();
sl@0
   448
		TUint32 currentType = EFrameData16;
sl@0
   449
		while(dataAvailable)
sl@0
   450
			{
sl@0
   451
			TUint32 toWrite;
sl@0
   452
			TFrameDes16 frame;
sl@0
   453
			
sl@0
   454
			TUint32 fileOffset = TUint32(iFileOut->iSnk->TellL(MStreamBuf::EWrite))+KSizeFrameDes16;
sl@0
   455
			TUint32 frameOffset = file2Frame(fileOffset);
sl@0
   456
			TUint32 nextAnchor = ((frameOffset + KFrameFullLength16) & ~KMaskFrameLength16);
sl@0
   457
			if(dataAvailable >= (nextAnchor-frameOffset))
sl@0
   458
				{
sl@0
   459
				// We have enough data to write an "Open" frame to
sl@0
   460
				// take the frame offset up to the next multiple of
sl@0
   461
				// 0x4000
sl@0
   462
				toWrite = (nextAnchor-frameOffset);
sl@0
   463
				frame = currentType | KFrameOpen16;
sl@0
   464
				}
sl@0
   465
			else
sl@0
   466
				{
sl@0
   467
				// Write a normal frame
sl@0
   468
				toWrite = (dataAvailable < TUint32(KMaskFrameLength16)) ? (dataAvailable) : (KMaskFrameLength16);
sl@0
   469
				frame = currentType | toWrite;
sl@0
   470
				}
sl@0
   471
			
sl@0
   472
			// Write frame
sl@0
   473
			*iFileOut << frame;
sl@0
   474
			// Write payload
sl@0
   475
			while(toWrite--)
sl@0
   476
				{
sl@0
   477
				*iFileOut << *dataIt;
sl@0
   478
				++dataIt;
sl@0
   479
				--dataAvailable;
sl@0
   480
				}
sl@0
   481
			// Set type of next frame to continuation
sl@0
   482
			currentType = EFrameContinuation16;
sl@0
   483
			}
sl@0
   484
		}
sl@0
   485
sl@0
   486
	//
sl@0
   487
	// Now write the TOC
sl@0
   488
	//
sl@0
   489
	TInt tocLength = sizeof(TocHeader) + (iStreams.size() * KSizeTocEntry);
sl@0
   490
	// Check toc frame length is not too long
sl@0
   491
	if(tocLength > KMaskFrameLength16)
sl@0
   492
		{
sl@0
   493
		FatalError();
sl@0
   494
		}
sl@0
   495
	// Write TOC frame
sl@0
   496
	TFrameDes16 frame = tocLength | EFrameDescriptive16;
sl@0
   497
	*iFileOut << frame;
sl@0
   498
sl@0
   499
	TStreamPos pos = iFileOut->iSnk->TellL(MStreamBuf::EWrite);	
sl@0
   500
sl@0
   501
	// First the header
sl@0
   502
	TUint32 tocFrameOffset = file2Frame(pos);
sl@0
   503
	TUint32 tocCookedFrameOffset = tocFrameOffset - KOffsetTocHeader;
sl@0
   504
	
sl@0
   505
	TocHeader tocheader;
sl@0
   506
	tocheader.iPrimary = iRoot;
sl@0
   507
	tocheader.iAvail = 0;
sl@0
   508
	tocheader.iCount = iStreams.size();
sl@0
   509
	*iFileOut << tocheader;
sl@0
   510
sl@0
   511
	// Now write the TOC entries
sl@0
   512
	for(TUint i = 0; i < streamOffsets.size(); ++i)
sl@0
   513
		{
sl@0
   514
		STocEntry tocEntry = { 0, file2Frame(streamOffsets[i]) };
sl@0
   515
		*iFileOut << tocEntry;
sl@0
   516
		}
sl@0
   517
	
sl@0
   518
	
sl@0
   519
	//
sl@0
   520
	// Now overwrite the header with one containing the correct master
sl@0
   521
	// TOC offset
sl@0
   522
	//
sl@0
   523
	TPermanentStoreHeader realheader(tocCookedFrameOffset);
sl@0
   524
	// For some reason the real store code creates a header with a
sl@0
   525
	// checksum which assumes iBackup == iRef but the iBackup value on
sl@0
   526
	// disk is set to 0...
sl@0
   527
	realheader.SetBackupToc(0);
sl@0
   528
	iFileOut->iSnk->SeekL(MStreamBuf::EWrite, EStreamBeginning, sizeof(TCheckedUid));
sl@0
   529
	*iFileOut << realheader;
sl@0
   530
sl@0
   531
sl@0
   532
	iFileOut->Close();
sl@0
   533
}
sl@0
   534
sl@0
   535
void CPermanentFileStore::ReadInData()
sl@0
   536
{
sl@0
   537
	prog << Log::Indent() << "Reading store" << Log::Endl();
sl@0
   538
	AutoIndent ai(prog);
sl@0
   539
	
sl@0
   540
	TUint lastStreamIndex = 0;
sl@0
   541
	TCheckedUid id;
sl@0
   542
sl@0
   543
	*iFileIn >> id;
sl@0
   544
sl@0
   545
	if(id.UidType() != KPermanentFileStoreLayoutUid)
sl@0
   546
		{
sl@0
   547
		dbg << Log::Indent() << "Invalid store" << Log::Endl();
sl@0
   548
		FatalError();
sl@0
   549
		}
sl@0
   550
	
sl@0
   551
	TPermanentStoreHeader header;
sl@0
   552
	
sl@0
   553
	*iFileIn >> header;
sl@0
   554
sl@0
   555
	// - If the dirty bit is set, the backup TOC will be used;
sl@0
   556
	// - If the dirty bit is not set, and the backup TOC ref is not the same as the TOC ref,
sl@0
   557
	//   then it means the the backup TOC ref has not been written successfully, so the TOC ref will be used;
sl@0
   558
	TInt toc=header.BackupToc();
sl@0
   559
	if (!header.IsDirty())
sl@0
   560
		{
sl@0
   561
		prog << Log::Indent() << "Store header is clean" << Log::Endl();
sl@0
   562
		TInt handle=header.Handle();
sl@0
   563
		TInt ref=header.Reference();
sl@0
   564
		if (handle==0&&toc!=ref)
sl@0
   565
			{ // toc pointer not backed up, validate as if it was
sl@0
   566
			toc=ref;
sl@0
   567
			//			iState|=EBackup;
sl@0
   568
			header.SetBackupToc(toc);
sl@0
   569
			}
sl@0
   570
		if (!header.IsValid()) // not a permanent store or damaged beyond recognition
sl@0
   571
			{
sl@0
   572
			dbg << Log::Indent() << "Store header checksum failed check1" << Log::Endl();
sl@0
   573
			__LEAVE(KErrNotSupported);
sl@0
   574
			}
sl@0
   575
		
sl@0
   576
		//
sl@0
   577
		if (toc<0||((handle&~KMaskStreamIdValue)!=0&&handle!=KHandleTocBase)||ref<0||(handle!=0&&ref>=toc+KOffsetTocHeader))
sl@0
   578
			{
sl@0
   579
			dbg << Log::Indent() << "Store header fields invalid" << Log::Endl();
sl@0
   580
			__LEAVE(KErrCorrupt); // integrity compromised
sl@0
   581
			}
sl@0
   582
		//
sl@0
   583
		//		iReloc=handle;
sl@0
   584
		//		iTarget=ref;
sl@0
   585
		}
sl@0
   586
	else
sl@0
   587
		{
sl@0
   588
		dbg << Log::Indent() << "Store header dirty" << Log::Endl();
sl@0
   589
		}
sl@0
   590
	
sl@0
   591
	//
sl@0
   592
	//	if (iToc!=0 && iToc!=toc)		// refresh produced a different toc
sl@0
   593
	//		__LEAVE(KErrCorrupt);
sl@0
   594
	//	iToc=toc;
sl@0
   595
sl@0
   596
sl@0
   597
	if(!header.IsValid())
sl@0
   598
		{
sl@0
   599
		dbg << Log::Indent() << "Store header checksum failed check2" << Log::Endl();
sl@0
   600
		FatalError();
sl@0
   601
		}
sl@0
   602
sl@0
   603
	// All offset values are to the payload of frame structures (see RFrame16Buf)
sl@0
   604
	// Convert to a file offset.
sl@0
   605
	TInt tocFileOffset = frame2File(toc + KOffsetTocHeader);
sl@0
   606
	prog << Log::Indent() << "TOC file offset 0x" << tocFileOffset << Log::Endl();
sl@0
   607
sl@0
   608
	// nb. We do not validate the length of the "descriptive" stream
sl@0
   609
	// which contains the TocHeader (aka STocHead) and the table of stream offsets.
sl@0
   610
sl@0
   611
	iFileIn->iSrc->SeekL(MStreamBuf::ERead, EStreamBeginning, tocFileOffset);
sl@0
   612
	TocHeader h;
sl@0
   613
	*iFileIn >> h;
sl@0
   614
	if ((h.iPrimary&~(KMaskStreamIdValue|static_cast<TUint>(KTocDelta)))!=0||
sl@0
   615
		h.iAvail>0||(h.iAvail&KMaskHandleClear)!=0||(h.iCount&~KMaskHandleIndex)!=0)
sl@0
   616
		{
sl@0
   617
		// Corrupt
sl@0
   618
		dbg << Log::Indent() << "Corrupt TOC in input file" << Log::Endl();
sl@0
   619
		FatalError();
sl@0
   620
		}
sl@0
   621
	std::set<TUint32> streamIdsReadExGenField; // IDs of streams already read, EXCLUDING generation field
sl@0
   622
	if (h.iPrimary & KTocDelta)
sl@0
   623
		{
sl@0
   624
		AutoIndent ai(prog);
sl@0
   625
		prog << Log::Indent() << "Found Delta TOC" << Log::Endl();
sl@0
   626
		// Process TOC-delta
sl@0
   627
		TInt oldToc = toc;
sl@0
   628
		// Find and read in the delta entries and then continue to read in the base entries
sl@0
   629
		// Read offset to base TOC
sl@0
   630
		iFileIn->iSrc->ReadL(&toc, sizeof(toc));
sl@0
   631
		// All offset values are to the payload of frame structures (see RFrame16Buf)
sl@0
   632
		// Convert to a file offset.
sl@0
   633
		tocFileOffset = frame2File(toc + KOffsetTocHeader);
sl@0
   634
sl@0
   635
		if((tocFileOffset < 0) || (tocFileOffset>oldToc))
sl@0
   636
			{
sl@0
   637
			dbg << Log::Indent() << "Invalid TOC in delta TOC" << Log::Endl();
sl@0
   638
			FatalError(); // Corrupt
sl@0
   639
			}
sl@0
   640
sl@0
   641
		// Read/discard the magic value - Not sure what this should be
sl@0
   642
		TUint16 magic;
sl@0
   643
		iFileIn->iSrc->ReadL(&magic, sizeof(magic));
sl@0
   644
		
sl@0
   645
		// Read count of delta toc entries
sl@0
   646
		TInt8 n;
sl@0
   647
		iFileIn->iSrc->ReadL(&n, sizeof(n));
sl@0
   648
		
sl@0
   649
		prog << Log::Indent() << "Delta TOC entries " << int(n) << Log::Endl();
sl@0
   650
		for(int deltaTocEntry=1; deltaTocEntry <= n; ++deltaTocEntry)
sl@0
   651
			{
sl@0
   652
			AutoIndent ai(prog);
sl@0
   653
			prog << Log::Indent() << "Delta TOC entry " << deltaTocEntry << Log::Endl();
sl@0
   654
sl@0
   655
			STocEntry e = {0,0};
sl@0
   656
			// Read delta toc entry.
sl@0
   657
			//
sl@0
   658
			// The main TOC entries are truncated to 5 bytes so only
sl@0
   659
			// include the flag bits of the handle and not the stream.
sl@0
   660
			// They gain streamids starting from 1.
sl@0
   661
			//
sl@0
   662
			// The delta TOC contains complete stream IDs, and these
sl@0
   663
			// entries take precedence over the matching (implicitly
sl@0
   664
			// numbered) entry in the base TOC.
sl@0
   665
			iFileIn->iSrc->ReadL(&e, sizeof(e));
sl@0
   666
sl@0
   667
			if ((e.handle&KMaskHandleClear)!=0 ||
sl@0
   668
				e.handle>=0 && (e.ref<KFrameNonexistent16) ||
sl@0
   669
				e.handle<0 && (e.ref>0 || (e.ref&KMaskHandleClear)!=0))
sl@0
   670
				{
sl@0
   671
				dbg << Log::Indent() << "Invalid delta TOC entry" << Log::Endl();
sl@0
   672
				__LEAVE(KErrCorrupt);
sl@0
   673
				FatalError();
sl@0
   674
				}
sl@0
   675
sl@0
   676
			if(e.handle & KHandleInvalid)
sl@0
   677
				{
sl@0
   678
				// Stream marked as deleted, skip it
sl@0
   679
				prog << Log::Indent() << "Delta TOC entry marked as deleted" << Log::Endl();
sl@0
   680
				continue;
sl@0
   681
				}
sl@0
   682
			
sl@0
   683
			// We do not deal with any other special cases correctly
sl@0
   684
			if(((e.handle&(KMaskHandleHash-KMaskHandleGen)) != 0) || (e.ref < 0))
sl@0
   685
				{
sl@0
   686
				dbg << Log::Indent() << "store-- does not support handle flags 0x" << e.handle << "in delta toc" << Log::Endl();
sl@0
   687
				FatalError();
sl@0
   688
				}
sl@0
   689
sl@0
   690
			prog << Log::Indent()
sl@0
   691
				 << "streamId 0x" << e.handle << " frame offset 0x" << e.ref 
sl@0
   692
				 << " file offset 0x" << frame2File(e.ref) << Log::Endl();
sl@0
   693
sl@0
   694
			if(!streamIdsReadExGenField.insert(e.handle & ~KMaskHandleHash).second)
sl@0
   695
				{
sl@0
   696
				dbg << Log::Indent() << "store-- corrupt input? Multiple entries for stream ID " << (e.handle & ~KMaskHandleHash) << "encountered in delta toc!" << Log::Endl();
sl@0
   697
				FatalError();
sl@0
   698
				}
sl@0
   699
			
sl@0
   700
sl@0
   701
			// Remember where we are...
sl@0
   702
			TStreamPos here = iFileIn->iSrc->TellL(MStreamBuf::ERead);
sl@0
   703
			// Read the stream frame
sl@0
   704
			ReadStreamData(e.handle, e.ref);
sl@0
   705
			// restore position to continue reading delta toc
sl@0
   706
			iFileIn->iSrc->SeekL(MStreamBuf::ERead, EStreamBeginning, here);
sl@0
   707
			}
sl@0
   708
		}
sl@0
   709
sl@0
   710
	iRoot = TInt(h.iPrimary) & ~KTocDelta;
sl@0
   711
	prog << Log::Indent() << "Root stream ID is " << iRoot << Log::Endl();
sl@0
   712
sl@0
   713
	prog << Log::Indent() << "Processing main TOC at file offset 0x" << tocFileOffset << Log::Endl();
sl@0
   714
		
sl@0
   715
	for(TUint tocIndex=0; tocIndex < h.iCount; ++tocIndex)
sl@0
   716
		{
sl@0
   717
		// Read in streams
sl@0
   718
		TStreamId baseStreamId = ++lastStreamIndex;
sl@0
   719
		prog << Log::Indent() << "Entry " << std::dec << baseStreamId << std::hex << Log::Endl();
sl@0
   720
		AutoIndent ai(prog);
sl@0
   721
		
sl@0
   722
		// Check if we have already read in this stream id from the delta toc
sl@0
   723
		// Note we only compare the base of the stream ID excluding the generation field.
sl@0
   724
		if(streamIdsReadExGenField.find(baseStreamId) != streamIdsReadExGenField.end())
sl@0
   725
			{
sl@0
   726
			prog << Log::Indent() << "Already read from delta TOC" << Log::Endl();
sl@0
   727
			continue; // Already read from delta toc
sl@0
   728
			}
sl@0
   729
		
sl@0
   730
		// Seek to the STocEntry for the stream
sl@0
   731
		iFileIn->iSrc->SeekL(MStreamBuf::ERead, EStreamBeginning, tocFileOffset+sizeof(TocHeader)+KSizeTocEntry*(tocIndex));
sl@0
   732
sl@0
   733
sl@0
   734
		STocEntry e = {0,0};
sl@0
   735
		// Only the last 5 bytes of the entry are saved to disk.
sl@0
   736
		// Because we are little endian this is the MSB of the handle and the whole ref.
sl@0
   737
		*iFileIn >> e;
sl@0
   738
sl@0
   739
		if ((e.handle&KMaskHandleClear)!=0 ||
sl@0
   740
			e.handle>=0 && (e.ref<KFrameNonexistent16) ||
sl@0
   741
			e.handle<0 && (e.ref>0 || (e.ref&KMaskHandleClear)!=0))
sl@0
   742
			{
sl@0
   743
			__LEAVE(KErrCorrupt);
sl@0
   744
			FatalError();
sl@0
   745
			}
sl@0
   746
sl@0
   747
		if(e.handle & KHandleInvalid)
sl@0
   748
			{
sl@0
   749
			// Stream deleted, skip it
sl@0
   750
			prog << Log::Indent() << "Marked as deleted" << Log::Endl();
sl@0
   751
			continue;
sl@0
   752
			}
sl@0
   753
sl@0
   754
		// We do not deal with any other special cases correctly 
sl@0
   755
		if(((e.handle & (KMaskHandleHash-KMaskHandleGen)) != 0) || (e.ref < 0))
sl@0
   756
			{
sl@0
   757
			dbg << Log::Indent() << "store-- does not support handle flags 0x" << e.handle << "in main toc" << Log::Endl();
sl@0
   758
			FatalError();
sl@0
   759
			}
sl@0
   760
sl@0
   761
		TStreamId streamId = baseStreamId | e.handle;
sl@0
   762
		if(!streamIdsReadExGenField.insert(baseStreamId).second)
sl@0
   763
			{
sl@0
   764
			dbg << Log::Indent() << "store-- internal error for stream ID " << baseStreamId << "encountered in delta toc!" << Log::Endl();
sl@0
   765
				FatalError();
sl@0
   766
			}
sl@0
   767
		
sl@0
   768
		// Read the stream frame
sl@0
   769
		//		prog << Log::Indent() << "Reading stream 0x" << streamId << " frame offset 0x" << e.ref << Log::Endl();
sl@0
   770
		ReadStreamData(streamId, e.ref);
sl@0
   771
		}
sl@0
   772
}
sl@0
   773
sl@0
   774
sl@0
   775
void CPermanentFileStore::ReadStreamData(TStreamId aStreamId, TInt aFrameOffset)
sl@0
   776
{
sl@0
   777
	TInt fileOffset = frame2File(aFrameOffset);
sl@0
   778
	TFrameDes16 frame;
sl@0
   779
	iFileIn->iSrc->SeekL(MStreamBuf::ERead, EStreamBeginning, fileOffset - sizeof(frame));
sl@0
   780
sl@0
   781
	// Frame runs from fileOffset to fileOffset+2+frame
sl@0
   782
	std::auto_ptr<MemStreamBuf> apStream(new MemStreamBuf(*this, aStreamId));
sl@0
   783
sl@0
   784
	enum {
sl@0
   785
		EFirst, EContinuation, ELast
sl@0
   786
	} state = EFirst;
sl@0
   787
	
sl@0
   788
	while(state != ELast) 
sl@0
   789
		{
sl@0
   790
		iFileIn->iSrc->ReadL(&frame, sizeof(frame));
sl@0
   791
sl@0
   792
		// Store uses offsets to the first byte of the payload of a frame, so that is what we display
sl@0
   793
		TUint32 fileOffset = iFileIn->iSrc->TellL(MStreamBuf::ERead);
sl@0
   794
		prog << Log::Indent() << "ReadStreamData State " << int(state) << " offset " << file2Frame(fileOffset) << "[" << fileOffset << "]" << Log::Endl();
sl@0
   795
sl@0
   796
		TUint32 frameType = (frame & KMaskFrameType16);
sl@0
   797
		if(frameType == EFrameFree16)
sl@0
   798
			{
sl@0
   799
			dbg << Log::Indent() << "Encountered a deleted frame in data stream" << Log::Endl();
sl@0
   800
			FatalError();
sl@0
   801
			}
sl@0
   802
sl@0
   803
		if((frameType == EFrameData16) && (state != EFirst))
sl@0
   804
			{
sl@0
   805
			dbg << Log::Indent() << "Non-continuation frame encountered in middle of data stream" << Log::Endl();
sl@0
   806
			FatalError();
sl@0
   807
			}
sl@0
   808
sl@0
   809
		if(frameType == EFrameDescriptive16)
sl@0
   810
			{
sl@0
   811
			dbg << Log::Indent() << "Descriptive frame encountered in data stream" << Log::Endl();
sl@0
   812
			FatalError();
sl@0
   813
			}
sl@0
   814
		
sl@0
   815
		if((frameType == EFrameContinuation16) && (state != EContinuation))
sl@0
   816
			{
sl@0
   817
			dbg << Log::Indent() << "Continuation frame encountered at start of stream" << Log::Endl();
sl@0
   818
			FatalError();
sl@0
   819
			}
sl@0
   820
	
sl@0
   821
		TInt len = (frame & KMaskFrameLength16);
sl@0
   822
		if (len == KFrameOpen16)
sl@0
   823
			{
sl@0
   824
			prog << Log::Indent() << "\"Open\" frame length" << Log::Endl();
sl@0
   825
			len = KFrameFullLength16 - file2Frame(fileOffset);
sl@0
   826
			state = EContinuation;
sl@0
   827
			}
sl@0
   828
		else
sl@0
   829
			{
sl@0
   830
			state = ELast;
sl@0
   831
			}
sl@0
   832
sl@0
   833
		prog << Log::Indent() << "Type 0x" << frameType << " length " << std::dec << len << std::hex << Log::Endl();
sl@0
   834
		apStream->AppendFromFile(iFileIn->iSrc, len);
sl@0
   835
		};
sl@0
   836
sl@0
   837
	MemStreamBufPtrMap::value_type value(aStreamId, apStream.get());
sl@0
   838
	std::pair<MemStreamBufPtrMap::iterator, bool> ret = iStreams.insert(value);
sl@0
   839
	if(!ret.second)
sl@0
   840
		{
sl@0
   841
		// Insert failed
sl@0
   842
		FatalError();
sl@0
   843
		}
sl@0
   844
sl@0
   845
	apStream.release();
sl@0
   846
}
sl@0
   847
sl@0
   848
RReadStream::RReadStream()
sl@0
   849
	: iSrc(0)
sl@0
   850
{
sl@0
   851
}
sl@0
   852
sl@0
   853
void RReadStream::Close()
sl@0
   854
{
sl@0
   855
	iSrc = false;
sl@0
   856
}
sl@0
   857
sl@0
   858
RStoreReadStream::RStoreReadStream()
sl@0
   859
	: RReadStream()
sl@0
   860
{
sl@0
   861
}
sl@0
   862
sl@0
   863
sl@0
   864
void RStoreReadStream::OpenLC(const CStreamStore& aStore, TStreamId aId)
sl@0
   865
{
sl@0
   866
	iSrc = aStore.CreateStoreReadStream(aId);
sl@0
   867
}
sl@0
   868
sl@0
   869
sl@0
   870
sl@0
   871
RWriteStream::RWriteStream()
sl@0
   872
	: iSnk(0)
sl@0
   873
{
sl@0
   874
}
sl@0
   875
sl@0
   876
sl@0
   877
sl@0
   878
sl@0
   879
sl@0
   880
TStreamId RStoreWriteStream::CreateLC(CStreamStore& aStore)
sl@0
   881
{
sl@0
   882
	MemStreamBuf *s = aStore.CreateStoreWriteStream();
sl@0
   883
	iSnk = s;
sl@0
   884
	return s->StreamId();
sl@0
   885
}
sl@0
   886
sl@0
   887
EXPORT_C void RReadStream::ReadL(TDes8& aDes)
sl@0
   888
/** Reads sufficient data from this stream to fill the specified 8 bit descriptor up to its maximum length.
sl@0
   889
No other information is read from this read stream.
sl@0
   890
sl@0
   891
@param aDes A reference to a modifiable descriptor which is to receive the data read from this stream. Passing the build 
sl@0
   892
independent type TDes& allows the compiler to choose the appropriate ReadL() variant (i.e the 8 bit or the 16 bit) at build time.*/
sl@0
   893
	{
sl@0
   894
	ReadL(aDes,aDes.MaxLength());
sl@0
   895
	}
sl@0
   896
sl@0
   897
EXPORT_C void RReadStream::ReadL(TDes8& aDes,TInt aLength)
sl@0
   898
/** Reads data of specified length from this stream into the specified 8 bit descriptor. No other information is read 
sl@0
   899
from this stream.
sl@0
   900
sl@0
   901
@param aDes A reference to a modifiable type descriptor which is to receive the data read from this stream. 
sl@0
   902
Passing the build independent type TDes& allows the compiler to choose the appropriate ReadL() variant (i.e the 8 bit 
sl@0
   903
or the 16 bit) at build time.
sl@0
   904
@param aLength The length of data to be read from this stream. This value must be non-negative and must not be greater than the maximum length of the descriptor otherwise the function raises a USER 11 panic.*/
sl@0
   905
	{
sl@0
   906
	__ASSERT_DEBUG(aLength<=aDes.MaxLength(),Panic(EStreamReadBeyondEnd));
sl@0
   907
	aDes.SetLength(aLength);
sl@0
   908
	ReadL((TUint8*)aDes.Ptr(),aLength);
sl@0
   909
	}
sl@0
   910
sl@0
   911
#if 0
sl@0
   912
EXPORT_C void RReadStream::ReadL(TDes8& aDes,TChar aDelim)
sl@0
   913
/** Reads data from this stream into the 8 bit descriptor, until either the specified delimiter is encountered or the descriptor is filled to its maximum length.
sl@0
   914
The resulting data in aDes always includes the delimiter aDelim, if aDes is large enough.
sl@0
   915
sl@0
   916
@param aDes A reference to a modifiable type descriptor which is to receive the data read from this stream. Passing 
sl@0
   917
the build independent type TDes& allows the compiler to choose the appropriate ReadL() variant (i.e the 8 bit or the 16 bit) at build time.
sl@0
   918
@param aDelim The delimiter marking the end of the data in the stream.*/
sl@0
   919
sl@0
   920
	{
sl@0
   921
	__ASSERT_DEBUG(iSrc!=NULL,Panic(EStreamNotOpen));
sl@0
   922
	TUint8* ptr=(TUint8*)aDes.Ptr();
sl@0
   923
	TDelimitedInput8 input(ptr,aDes.MaxLength(),aDelim);
sl@0
   924
	do
sl@0
   925
		{
sl@0
   926
		iSrc->ReadL(input);
sl@0
   927
		} while (!input.Done());
sl@0
   928
	aDes.SetLength(input.Ptr()-ptr);
sl@0
   929
	}
sl@0
   930
#endif
sl@0
   931
sl@0
   932
EXPORT_C void RReadStream::ReadL(TUint8* aPtr,TInt aLength)
sl@0
   933
/** Reads data of specified length from this stream into the location defined by the specified TUint8 pointer.
sl@0
   934
sl@0
   935
@param aPtr The target location for the streamed in data.
sl@0
   936
@param aLength The length of data to be streamed in.*/
sl@0
   937
	{
sl@0
   938
	__ASSERT_DEBUG(aLength>=0,Panic(EStreamReadLengthNegative));
sl@0
   939
	if (aLength==0)
sl@0
   940
		return;
sl@0
   941
//	void *p = NULL;
sl@0
   942
	__ASSERT_DEBUG(iSrc!=NULL,Panic(EStreamNotOpen));
sl@0
   943
	TInt len=iSrc->ReadL(aPtr,aLength);
sl@0
   944
	__ASSERT_DEBUG(len>=0&&len<=aLength,Panic(EStreamReadInBreach));
sl@0
   945
	if (len<aLength)
sl@0
   946
		__LEAVE(KErrEof);
sl@0
   947
	}
sl@0
   948
sl@0
   949
EXPORT_C void RReadStream::ReadL(TInt aLength)
sl@0
   950
/** Discards data of specified length read from this stream.
sl@0
   951
sl@0
   952
@param aLength The length of data to be discarded from this read stream.*/
sl@0
   953
	{
sl@0
   954
	__ASSERT_DEBUG(aLength>=0,Panic(EStreamReadLengthNegative));
sl@0
   955
	if (aLength==0)
sl@0
   956
		return;
sl@0
   957
//
sl@0
   958
	__ASSERT_DEBUG(iSrc!=NULL,Panic(EStreamNotOpen));
sl@0
   959
sl@0
   960
#if 1
sl@0
   961
	iSrc->SeekL(MStreamBuf::ERead, EStreamMark, aLength);
sl@0
   962
#else
sl@0
   963
	TNullInput input;
sl@0
   964
	TInt len=iSrc->ReadL(input,aLength);
sl@0
   965
	__ASSERT_DEBUG(len>=0&&len<=aLength,Panic(EStreamReadInBreach));
sl@0
   966
	if (len<aLength)
sl@0
   967
		__LEAVE(KErrEof);
sl@0
   968
#endif
sl@0
   969
	}
sl@0
   970
sl@0
   971
EXPORT_C void RReadStream::ReadL(TDes16& aDes)
sl@0
   972
/** Reads sufficient data from this stream to fill the specified 16 bit descriptor up to its maximum length. 
sl@0
   973
No other information is read from this read stream.
sl@0
   974
sl@0
   975
@param aDes A reference to a modifiable type descriptor which is to receive the data read from this stream. Passing 
sl@0
   976
the build independent type TDes& allows the compiler to choose the appropriate ReadL() variant (i.e the 8 bit or the 16 
sl@0
   977
bit) at build time.*/
sl@0
   978
	{
sl@0
   979
	ReadL(aDes,aDes.MaxLength());
sl@0
   980
	}
sl@0
   981
sl@0
   982
EXPORT_C void RReadStream::ReadL(TDes16& aDes,TInt aLength)
sl@0
   983
/** Reads data of specified length from this stream into the specified 16 bit descriptor. No other information is read from this stream.
sl@0
   984
sl@0
   985
@param aDes A reference to a modifiable type descriptor which is to receive the data read from this stream. Passing the 
sl@0
   986
build independent type TDes& allows the compiler to choose the appropriate ReadL() variant (i.e the 8 bit or the 16 bit) 
sl@0
   987
at build time.
sl@0
   988
@param aLength The length of data to be read from this stream. This value must be non-negative and must not be greater
sl@0
   989
than the maximum length of the descriptor otherwise the function raises a USER 11 panic.*/
sl@0
   990
	{
sl@0
   991
	__ASSERT_DEBUG(aLength<=aDes.MaxLength(),Panic(EStreamReadBeyondEnd));
sl@0
   992
	aDes.SetLength(aLength);
sl@0
   993
	ReadL((TUint16*)aDes.Ptr(),aLength);
sl@0
   994
	}
sl@0
   995
sl@0
   996
#if 0
sl@0
   997
EXPORT_C void RReadStream::ReadL(TDes16& aDes,TChar aDelim)
sl@0
   998
/** Reads data from this stream into the 16 bit descriptor, until either the specified delimiter is encountered or 
sl@0
   999
the descriptor is filled to its maximum length.
sl@0
  1000
The resulting data in aDes always includes the delimiter aDelim, if aDes is large enough.
sl@0
  1001
sl@0
  1002
@param aDes A reference to a modifiable type descriptor which is to receive the data read from this stream. Passing 
sl@0
  1003
the build independent type TDes& allows the compiler to choose the appropriate ReadL() variant (i.e the 8 bit or the 16 bit) at build time.
sl@0
  1004
@param aDelim The delimiter marking the end of the data in the stream.*/
sl@0
  1005
	{
sl@0
  1006
	__ASSERT_DEBUG(iSrc!=NULL,Panic(EStreamNotOpen));
sl@0
  1007
	TUint16* ptr=(TUint16*)aDes.Ptr();
sl@0
  1008
	TDelimitedInput16 input(ptr,aDes.MaxLength(),aDelim);
sl@0
  1009
	do
sl@0
  1010
		{
sl@0
  1011
		iSrc->ReadL(input);
sl@0
  1012
		} while (!input.Done());
sl@0
  1013
	aDes.SetLength(input.Ptr()-ptr);
sl@0
  1014
	}
sl@0
  1015
#endif
sl@0
  1016
sl@0
  1017
EXPORT_C void RReadStream::ReadL(TUint16* aPtr,TInt aLength)
sl@0
  1018
/** Reads data of specified length from this stream into the specified 16 bit descriptor.
sl@0
  1019
 No other information is read from this stream.
sl@0
  1020
sl@0
  1021
@param aDes A reference to a modifiable type descriptor which is to receive the data read from this stream. Passing the build independent type TDes& allows the compiler to choose the appropriate ReadL() variant (i.e the 8 bit or the 16 bit) at build time.
sl@0
  1022
@param aLength The length of data to be read from this stream. This value must be non-negative and must not be greater than the maximum length of the descriptor otherwise the function raises a USER 11 panic.*/
sl@0
  1023
	{
sl@0
  1024
	__ASSERT_DEBUG(aLength>=0,Panic(EStreamReadLengthNegative));
sl@0
  1025
	ReadL((TUint8*)aPtr,aLength<<1); // platform dependency
sl@0
  1026
	}
sl@0
  1027
sl@0
  1028
EXPORT_C TInt8 RReadStream::ReadInt8L()
sl@0
  1029
/** Internalises a TInt8 value The function reads an 8 bit value from this stream 
sl@0
  1030
and interprets it as a TInt8.
sl@0
  1031
sl@0
  1032
@return The 8 bit value read from this stream. */
sl@0
  1033
	{
sl@0
  1034
	TInt8 val;
sl@0
  1035
	ReadL((TUint8*)&val,1); // platform dependency
sl@0
  1036
	return val;
sl@0
  1037
	}
sl@0
  1038
sl@0
  1039
EXPORT_C TInt16 RReadStream::ReadInt16L()
sl@0
  1040
/** Internalises a TInt16 value. The function reads a 16 bit value from this stream 
sl@0
  1041
and interprets it as a TInt16.
sl@0
  1042
sl@0
  1043
@return The 16 bit value read from this stream. */
sl@0
  1044
	{
sl@0
  1045
	TInt16 val;
sl@0
  1046
	ReadL((TUint8*)&val,2); // platform dependency
sl@0
  1047
	return val;
sl@0
  1048
	}
sl@0
  1049
sl@0
  1050
EXPORT_C TInt32 RReadStream::ReadInt32L()
sl@0
  1051
/** Internalises a TInt32 value. The function reads a 32 bit value from this stream 
sl@0
  1052
and interprets it as a TInt32.
sl@0
  1053
sl@0
  1054
@return The 32 bit value read from this stream. */
sl@0
  1055
	{
sl@0
  1056
	TInt32 val;
sl@0
  1057
	ReadL((TUint8*)&val,4); // platform dependency
sl@0
  1058
	return val;
sl@0
  1059
	}
sl@0
  1060
sl@0
  1061
EXPORT_C TUint8 RReadStream::ReadUint8L()
sl@0
  1062
/** Internalises a TUint8 value. The function reads an 8 bit value from this stream 
sl@0
  1063
and interprets it as a TUint8.
sl@0
  1064
sl@0
  1065
@return The 8 bit value read from this stream. */
sl@0
  1066
	{
sl@0
  1067
	TUint8 val;
sl@0
  1068
	ReadL(&val,1);
sl@0
  1069
	return val;
sl@0
  1070
	}
sl@0
  1071
sl@0
  1072
EXPORT_C TUint16 RReadStream::ReadUint16L()
sl@0
  1073
/** Internalises a TUint16 value. The function reads a 16 bit value from this 
sl@0
  1074
stream and interprets it as a TUint16.
sl@0
  1075
sl@0
  1076
@return The 16 bit value read from this stream. */
sl@0
  1077
	{
sl@0
  1078
	TUint16 val;
sl@0
  1079
	ReadL((TUint8*)&val,2); // platform dependency
sl@0
  1080
	return val;
sl@0
  1081
	}
sl@0
  1082
sl@0
  1083
EXPORT_C TUint32 RReadStream::ReadUint32L()
sl@0
  1084
/** Internalises a TUint32 value. The function reads a 32 bit value from this 
sl@0
  1085
stream and interprets it as a TUint32.
sl@0
  1086
sl@0
  1087
@return The 32 bit value read from this stream. */
sl@0
  1088
	{
sl@0
  1089
	TUint32 val;
sl@0
  1090
	ReadL((TUint8*)&val,4); // platform dependency
sl@0
  1091
	return val;
sl@0
  1092
	}
sl@0
  1093
sl@0
  1094
sl@0
  1095
EXPORT_C void RWriteStream::Close()
sl@0
  1096
/** Commits data to the stream before freeing resources used by the stream. This 
sl@0
  1097
ensures that any buffered data is written to the stream.
sl@0
  1098
sl@0
  1099
Note that the function cannot leave. Any errors arising from the attempt to 
sl@0
  1100
commit data to the stream are ignored. */
sl@0
  1101
	{
sl@0
  1102
	if (iSnk==NULL)
sl@0
  1103
		return;
sl@0
  1104
//
sl@0
  1105
	iSnk->Close();
sl@0
  1106
	iSnk=NULL;
sl@0
  1107
	}
sl@0
  1108
sl@0
  1109
EXPORT_C void RWriteStream::Release()
sl@0
  1110
/** Frees resources before abandoning the stream. The function is called after 
sl@0
  1111
data has been committed to the stream.
sl@0
  1112
sl@0
  1113
Note that if a cleanup item for the stream was placed on the cleanup stack 
sl@0
  1114
when the stream was opened (e.g by a call to RStoreWriteStreamss CreateLC(), 
sl@0
  1115
OpenLC(), ReplaceLC() or RDictionaryStores AssignLC() etc), then this function 
sl@0
  1116
need not be called explicitly; clean up is implicitly done by CleanupStack::PopAndDestroy(). */
sl@0
  1117
	{
sl@0
  1118
	if (iSnk==NULL)
sl@0
  1119
		return;
sl@0
  1120
//
sl@0
  1121
	iSnk->Release();
sl@0
  1122
	iSnk=NULL;
sl@0
  1123
	}
sl@0
  1124
sl@0
  1125
EXPORT_C void RWriteStream::CommitL()
sl@0
  1126
/** Ensures that any buffered data is written to the stream. Once committed, it 
sl@0
  1127
is not possible to roll back the newly written data. */
sl@0
  1128
	{
sl@0
  1129
	if (iSnk==NULL)
sl@0
  1130
		return;
sl@0
  1131
//
sl@0
  1132
	iSnk->SynchL();
sl@0
  1133
	}
sl@0
  1134
sl@0
  1135
sl@0
  1136
EXPORT_C void RWriteStream::WriteL(const TDesC8& aDes)
sl@0
  1137
/** Writes the content of the 8 bit descriptor to the stream. No other information 
sl@0
  1138
is written to this write stream.
sl@0
  1139
sl@0
  1140
@param aDes A reference to a descriptor. Passing the build independent type 
sl@0
  1141
TDesC& allows the compiler to choose the appropriate WriteL() variant (i.e 
sl@0
  1142
the 8 bit or the 16 bit) at build time. */
sl@0
  1143
	{
sl@0
  1144
	WriteL(aDes.Ptr(),aDes.Length());
sl@0
  1145
	}
sl@0
  1146
sl@0
  1147
EXPORT_C void RWriteStream::WriteL(const TDesC8& aDes,TInt aLength)
sl@0
  1148
/** Writes data of the specified length from the 8 bit descriptor to the stream. 
sl@0
  1149
No other information is written to this write stream.
sl@0
  1150
sl@0
  1151
@param aDes A reference to a descriptor. Passing the build independent type 
sl@0
  1152
TDesC& allows the compiler to choose the appropriate WriteL() variant (i.e 
sl@0
  1153
the 8 bit or the 16 bit) at build time.
sl@0
  1154
@param aLength The length of data to be written to this stream. */
sl@0
  1155
sl@0
  1156
	{
sl@0
  1157
	__ASSERT_DEBUG(aLength<=aDes.Length(),Panic(EStreamWriteBeyondEnd));
sl@0
  1158
	WriteL(aDes.Ptr(),aLength);
sl@0
  1159
	}
sl@0
  1160
sl@0
  1161
EXPORT_C void RWriteStream::WriteL(const TUint8* aPtr,TInt aLength)
sl@0
  1162
/** Writes 8 bit data of the specified length from the specified location to this 
sl@0
  1163
write stream.
sl@0
  1164
sl@0
  1165
@param aPtr The location from where data is to be streamed out.
sl@0
  1166
@param aLength The length of data to be streamed out. */
sl@0
  1167
sl@0
  1168
	{
sl@0
  1169
	__ASSERT_DEBUG(aLength>=0,Panic(EStreamWriteLengthNegative));
sl@0
  1170
	if (aLength==0)
sl@0
  1171
		return;
sl@0
  1172
//
sl@0
  1173
	__ASSERT_DEBUG(iSnk!=NULL,Panic(EStreamNotOpen));
sl@0
  1174
	iSnk->WriteL(aPtr,aLength);
sl@0
  1175
	}
sl@0
  1176
sl@0
  1177
#if 0
sl@0
  1178
EXPORT_C void RWriteStream::WriteL(RReadStream &aStream)
sl@0
  1179
/** Writes the content of the specified read stream to this write stream.
sl@0
  1180
sl@0
  1181
@param aStream A reference to a read stream which is to be written to this 
sl@0
  1182
stream. */
sl@0
  1183
sl@0
  1184
	{
sl@0
  1185
	__ASSERT_DEBUG(iSnk!=NULL,Panic(EStreamNotOpen));
sl@0
  1186
	TSourceOutput output(aStream.iSrc);
sl@0
  1187
	iSnk->WriteL(output);
sl@0
  1188
	}
sl@0
  1189
#endif
sl@0
  1190
sl@0
  1191
#if 0
sl@0
  1192
EXPORT_C void RWriteStream::WriteL(RReadStream& aStream,TInt aLength)
sl@0
  1193
/** Writes data of the specified length from the specified read stream to this 
sl@0
  1194
stream.
sl@0
  1195
sl@0
  1196
@param aStream A reference to a read stream part of whose content is to be 
sl@0
  1197
written to this stream.
sl@0
  1198
@param aLength The length of data from the read stream to be written to this 
sl@0
  1199
write stream. */
sl@0
  1200
sl@0
  1201
	{
sl@0
  1202
	__ASSERT_DEBUG(aLength>=0,Panic(EStreamWriteLengthNegative));
sl@0
  1203
	if (aLength==0)
sl@0
  1204
		return;
sl@0
  1205
//
sl@0
  1206
	__ASSERT_DEBUG(iSnk!=NULL,Panic(EStreamNotOpen));
sl@0
  1207
	TSourceOutput output(aStream.iSrc);
sl@0
  1208
	TInt len=iSnk->WriteL(output,aLength);
sl@0
  1209
	__ASSERT_DEBUG(len>=0&&len<=aLength,Panic(EStreamReadInBreach));
sl@0
  1210
	if (len<aLength)
sl@0
  1211
		__LEAVE(KErrEof);
sl@0
  1212
	}
sl@0
  1213
#endif
sl@0
  1214
sl@0
  1215
EXPORT_C void RWriteStream::WriteL(const TDesC16& aDes)
sl@0
  1216
/** Writes the content of the 16 bit descriptor to the stream. No other information 
sl@0
  1217
is written to this write stream.
sl@0
  1218
sl@0
  1219
@param aDes A reference to a descriptor. Passing the build independent type 
sl@0
  1220
TDesC& allows the compiler to choose the appropriate WriteL() variant (i.e 
sl@0
  1221
the 8 bit or the 16 bit) at build time. */
sl@0
  1222
sl@0
  1223
	{
sl@0
  1224
	WriteL(aDes.Ptr(),aDes.Length());
sl@0
  1225
	}
sl@0
  1226
sl@0
  1227
EXPORT_C void RWriteStream::WriteL(const TDesC16& aDes,TInt aLength)
sl@0
  1228
/** Writes data of the specified length from the 16 bit descriptor to the stream. 
sl@0
  1229
No other information is written to this write stream.
sl@0
  1230
sl@0
  1231
@param aDes A reference to a descriptor. Passing the build independent type 
sl@0
  1232
TDesC& allows the compiler to choose the appropriate WriteL() variant (i.e 
sl@0
  1233
the 8 bit or the 16 bit) at build time.
sl@0
  1234
@param aLength The length of data to be written to this stream. */
sl@0
  1235
sl@0
  1236
	{
sl@0
  1237
	__ASSERT_DEBUG(aLength<=aDes.Length(),Panic(EStreamWriteBeyondEnd));
sl@0
  1238
	WriteL(aDes.Ptr(),aLength);
sl@0
  1239
	}
sl@0
  1240
sl@0
  1241
EXPORT_C void RWriteStream::WriteL(const TUint16* aPtr,TInt aLength)
sl@0
  1242
/** Writes 16 bit data of the specified length from the specified location to this 
sl@0
  1243
write stream.
sl@0
  1244
sl@0
  1245
@param aPtr The location from where data is to be streamed out.
sl@0
  1246
@param aLength The length of data to be streamed out. */
sl@0
  1247
sl@0
  1248
	{
sl@0
  1249
	__ASSERT_DEBUG(aLength>=0,Panic(EStreamWriteLengthNegative));
sl@0
  1250
	WriteL((const TUint8*)aPtr,aLength<<1); // platform dependency
sl@0
  1251
	}
sl@0
  1252
sl@0
  1253
EXPORT_C void RWriteStream::WriteInt8L(TInt aValue)
sl@0
  1254
/** Writes a TInt value as an 8 bit value to this stream.
sl@0
  1255
sl@0
  1256
@param aValue The value to be written to this stream. */
sl@0
  1257
sl@0
  1258
	{
sl@0
  1259
	WriteL((const TUint8*)&aValue,1); // platform dependency
sl@0
  1260
	}
sl@0
  1261
sl@0
  1262
EXPORT_C void RWriteStream::WriteInt16L(TInt aValue)
sl@0
  1263
/** Writes a TInt value as a 16 bit value to this stream.
sl@0
  1264
sl@0
  1265
@param aValue The value to be written to this stream. */
sl@0
  1266
	{
sl@0
  1267
	WriteL((const TUint8*)&aValue,2); // platform dependency
sl@0
  1268
	}
sl@0
  1269
sl@0
  1270
EXPORT_C void RWriteStream::WriteInt32L(TInt32 aValue)
sl@0
  1271
/** Writes a TInt32 value as a 32 bit value to this stream.
sl@0
  1272
sl@0
  1273
@param aValue The value to be written to this stream. */
sl@0
  1274
sl@0
  1275
	{
sl@0
  1276
	WriteL((const TUint8*)&aValue,4); // platform dependency
sl@0
  1277
	}
sl@0
  1278
sl@0
  1279
EXPORT_C void RWriteStream::WriteUint8L(TUint aValue)
sl@0
  1280
/** Writes a TUint value as an 8 bit value to this stream.
sl@0
  1281
sl@0
  1282
@param aValue The value to be written to this stream. */
sl@0
  1283
	{
sl@0
  1284
	WriteL((const TUint8*)&aValue,1); // platform dependency
sl@0
  1285
	}
sl@0
  1286
sl@0
  1287
EXPORT_C void RWriteStream::WriteUint16L(TUint aValue)
sl@0
  1288
/** Writes a TUint value as a 16 bit value to this stream.
sl@0
  1289
sl@0
  1290
@param aValue The value to be written to this stream. */
sl@0
  1291
sl@0
  1292
	{
sl@0
  1293
	WriteL((const TUint8*)&aValue,2); // platform dependency
sl@0
  1294
	}
sl@0
  1295
sl@0
  1296
EXPORT_C void RWriteStream::WriteUint32L(TUint32 aValue)
sl@0
  1297
/** Writes a TUint32 value as a 32 bit value to this stream.
sl@0
  1298
sl@0
  1299
@param aValue The value to be written to this stream. */
sl@0
  1300
	{
sl@0
  1301
	WriteL((const TUint8*)&aValue,4); // platform dependency
sl@0
  1302
	}
sl@0
  1303
sl@0
  1304
sl@0
  1305
namespace store_private
sl@0
  1306
{
sl@0
  1307
sl@0
  1308
MemStreamBuf::MemStreamBuf(const CStreamStore& aStore, TStreamId aId)
sl@0
  1309
	: iStore(&aStore), iStreamId(aId), iCurrentReadIndex(0), iCurrentWriteIndex(0), iData()
sl@0
  1310
{
sl@0
  1311
}
sl@0
  1312
sl@0
  1313
MemStreamBuf::MemStreamBuf(const CStreamStore& aStore, TStreamId aId, 
sl@0
  1314
					 MStreamBuf *aStreamIn, std::streamoff aOffset, TUint32 aLength)
sl@0
  1315
	: iStore(&aStore), iStreamId(aId), iCurrentReadIndex(0), iCurrentWriteIndex(0), iData()
sl@0
  1316
{
sl@0
  1317
	
sl@0
  1318
sl@0
  1319
	aStreamIn->SeekL(MStreamBuf::ERead, EStreamBeginning, aOffset);
sl@0
  1320
	while(aLength)
sl@0
  1321
		{
sl@0
  1322
		TUint8 data;
sl@0
  1323
		aStreamIn->ReadL(&data, 1);
sl@0
  1324
		iData.push_back(data);
sl@0
  1325
		--aLength;
sl@0
  1326
		}
sl@0
  1327
	
sl@0
  1328
}
sl@0
  1329
sl@0
  1330
void MemStreamBuf::AppendFromFile(MStreamBuf *aStreamIn, TUint32 aLength)
sl@0
  1331
{
sl@0
  1332
	while(aLength)
sl@0
  1333
		{
sl@0
  1334
		TUint8 data;
sl@0
  1335
		aStreamIn->ReadL(&data, 1);
sl@0
  1336
		iData.push_back(data);
sl@0
  1337
		--aLength;
sl@0
  1338
		}
sl@0
  1339
}
sl@0
  1340
sl@0
  1341
sl@0
  1342
sl@0
  1343
TStreamId MemStreamBuf::StreamId() const
sl@0
  1344
{
sl@0
  1345
	return iStreamId;
sl@0
  1346
}
sl@0
  1347
sl@0
  1348
const std::vector<TUint8> &MemStreamBuf::GetData() const
sl@0
  1349
{
sl@0
  1350
	return iData;
sl@0
  1351
}
sl@0
  1352
sl@0
  1353
sl@0
  1354
TInt MemStreamBuf::DoReadL(TAny *aPtr,TInt aMaxLength)
sl@0
  1355
{
sl@0
  1356
	char *p = reinterpret_cast<char *>(aPtr);
sl@0
  1357
	for(TInt i=0; i < aMaxLength; ++i)
sl@0
  1358
		{
sl@0
  1359
		*p++ = iData[iCurrentReadIndex++];
sl@0
  1360
		}
sl@0
  1361
	return aMaxLength;
sl@0
  1362
}
sl@0
  1363
sl@0
  1364
void MemStreamBuf::DoWriteL(const TUint8* aPtr,TInt aLength)
sl@0
  1365
{
sl@0
  1366
	for(TInt i=0; i < aLength; ++i)
sl@0
  1367
		{
sl@0
  1368
		if(iCurrentWriteIndex == iData.size())
sl@0
  1369
			{
sl@0
  1370
			iData.push_back(aPtr[i]);
sl@0
  1371
			}
sl@0
  1372
		else
sl@0
  1373
			{
sl@0
  1374
			iData[iCurrentWriteIndex] = aPtr[i];
sl@0
  1375
			}
sl@0
  1376
		++iCurrentWriteIndex;
sl@0
  1377
		}
sl@0
  1378
}
sl@0
  1379
sl@0
  1380
TStreamPos MemStreamBuf::DoSeekL(TMark aMark,TStreamLocation aLocation,TInt anOffset)
sl@0
  1381
{
sl@0
  1382
	if(aMark != ERead && aMark != EWrite)
sl@0
  1383
		{
sl@0
  1384
		FatalError();
sl@0
  1385
		}
sl@0
  1386
	
sl@0
  1387
	TUint32 *p = (aMark == ERead) ? (&iCurrentReadIndex) : (&iCurrentWriteIndex);
sl@0
  1388
sl@0
  1389
	switch(aLocation)
sl@0
  1390
		{
sl@0
  1391
		case EStreamBeginning:
sl@0
  1392
			*p = anOffset;
sl@0
  1393
			break;
sl@0
  1394
			
sl@0
  1395
		case EStreamMark:
sl@0
  1396
			*p += anOffset;
sl@0
  1397
			break;
sl@0
  1398
sl@0
  1399
		case EStreamEnd:
sl@0
  1400
			*p = iData.size() + anOffset;
sl@0
  1401
			break;
sl@0
  1402
sl@0
  1403
		default:
sl@0
  1404
			FatalError();
sl@0
  1405
		}
sl@0
  1406
sl@0
  1407
	if(*p > iData.size())
sl@0
  1408
		{
sl@0
  1409
		// nb we allow a seek 1 byte off end of data
sl@0
  1410
		FatalError();
sl@0
  1411
		}
sl@0
  1412
	return *p;
sl@0
  1413
}
sl@0
  1414
sl@0
  1415
void MemStreamBuf::DoRelease()
sl@0
  1416
{
sl@0
  1417
}
sl@0
  1418
sl@0
  1419
}; // end of namespace store_private 
sl@0
  1420
sl@0
  1421
const TInt KShiftCardinality8=1;
sl@0
  1422
const TInt KShiftCardinality16=2;
sl@0
  1423
const TInt KShiftCardinality32=3;
sl@0
  1424
//
sl@0
  1425
const TInt KDefaultIoBufSize=0xc00;
sl@0
  1426
const TInt KFilterIoBufSize=0x100;
sl@0
  1427
sl@0
  1428
EXPORT_C void TCardinality::ExternalizeL(RWriteStream& aStream) const
sl@0
  1429
/** Externalises this object to a write stream.
sl@0
  1430
sl@0
  1431
The existence of this function means that the standard templated operator<<() 
sl@0
  1432
can be used to externalise objects of this class.
sl@0
  1433
sl@0
  1434
@param aStream Stream to which the object should be externalised.
sl@0
  1435
@see operator<<() */
sl@0
  1436
	{
sl@0
  1437
	TUint n=iCount;
sl@0
  1438
	if (n<=(KMaxTUint8>>KShiftCardinality8))
sl@0
  1439
		aStream.WriteUint8L((n<<KShiftCardinality8));
sl@0
  1440
	else if (n<=(KMaxTUint16>>KShiftCardinality16))
sl@0
  1441
		aStream.WriteUint16L((n<<KShiftCardinality16)+0x1);
sl@0
  1442
	else
sl@0
  1443
		{
sl@0
  1444
		__ASSERT_DEBUG(n<=(TUint)KMaxCardinality,Panic(EStreamCardinalityOutOfRange));
sl@0
  1445
		aStream.WriteUint32L((n<<KShiftCardinality32)+0x3);
sl@0
  1446
		}
sl@0
  1447
	}
sl@0
  1448
sl@0
  1449
EXPORT_C void TCardinality::InternalizeL(RReadStream& aStream)
sl@0
  1450
/** Internalizes this object from a read stream.
sl@0
  1451
sl@0
  1452
The existence of this function means that the standard templated operator>>() 
sl@0
  1453
can be used to internalise objects of this class.
sl@0
  1454
sl@0
  1455
@param aStream Stream store from which the objet is to be internalised.
sl@0
  1456
@see operator>>() */
sl@0
  1457
	{
sl@0
  1458
	TUint n=aStream.ReadUint8L();
sl@0
  1459
	if ((n&0x1)==0)
sl@0
  1460
		n>>=KShiftCardinality8;
sl@0
  1461
	else if ((n&0x2)==0)
sl@0
  1462
		{
sl@0
  1463
		n+=aStream.ReadUint8L()<<8;
sl@0
  1464
		n>>=KShiftCardinality16;
sl@0
  1465
		}
sl@0
  1466
	else if ((n&0x4)==0)
sl@0
  1467
		{
sl@0
  1468
		aStream.ReadL((TUint8*)&iCount,sizeof(TUint32)-sizeof(TUint8));
sl@0
  1469
		n+=TUint(iCount)<<8; // platform dependency
sl@0
  1470
		n>>=KShiftCardinality32;
sl@0
  1471
		}
sl@0
  1472
	else
sl@0
  1473
		__LEAVE(KErrCorrupt);
sl@0
  1474
//
sl@0
  1475
	__ASSERT_DEBUG(n<=(TUint)KMaxCardinality,User::Invariant());
sl@0
  1476
	iCount=n;
sl@0
  1477
	}
sl@0
  1478
sl@0
  1479
EXPORT_C void TCardinality::__DbgChkRange(TInt aCount)
sl@0
  1480
//
sl@0
  1481
// Check for a negative count.
sl@0
  1482
//
sl@0
  1483
	{
sl@0
  1484
		if(!(aCount<=KMaxCardinality)) FatalError();
sl@0
  1485
	}
sl@0
  1486
sl@0
  1487
void CleanupStack::PopAndDestroy(RWriteStream *aStream)
sl@0
  1488
{
sl@0
  1489
	aStream->Close();
sl@0
  1490
}
sl@0
  1491
void CleanupStack::PopAndDestroy(RReadStream *aStream)
sl@0
  1492
{
sl@0
  1493
	aStream->Close();
sl@0
  1494
}
sl@0
  1495
sl@0
  1496
void CleanupStack::PopAndDestroy(CFileStore *aStore)
sl@0
  1497
{
sl@0
  1498
	delete aStore;
sl@0
  1499
}
sl@0
  1500
sl@0
  1501
sl@0
  1502
// End of file