os/persistentdata/persistentstorage/store/USTOR/UT_PERM.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) 1998-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 <s32file.h>
sl@0
    17
#include "UT_STD.H"
sl@0
    18
sl@0
    19
//The default media block size, used in the computations, if the file system guarantees atomic "block write" file operations.
sl@0
    20
const TInt KDefaultMediaBlockSize = 512;
sl@0
    21
sl@0
    22
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
    23
/////////////////////              TPermanentStoreHeader               /////////////////////////////////////////////////////
sl@0
    24
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
    25
sl@0
    26
//Returns false, if:
sl@0
    27
// - the dirty bit is set (indicates commit phase #2 or #3 is not complete);
sl@0
    28
// - the CRC check fails ("backup TOC ref", "handle" and "TOC ref" are protected by 16-bit CRC in the permanent store file header);
sl@0
    29
TBool TPermanentStoreHeader::IsValid() const
sl@0
    30
	{
sl@0
    31
	if (IsDirty())
sl@0
    32
		return EFalse;
sl@0
    33
//
sl@0
    34
	TUint16 crc=0;
sl@0
    35
	Mem::Crc(crc,Ptr(),_FOFF(TPermanentStoreHeader,iCrc));
sl@0
    36
	return crc==iCrc;
sl@0
    37
	}
sl@0
    38
sl@0
    39
//Sets the "backup TOC ref", "handle" and "TOC ref" in current TPermanentStoreHeader object.
sl@0
    40
//16-bit CRC is calculated, based on the values of the input parameters, and stored together with them in the 
sl@0
    41
//TPermanentStoreHeader object.
sl@0
    42
void TPermanentStoreHeader::Set(TInt aBackupToc,TInt aHandle,TInt aReference)
sl@0
    43
	{
sl@0
    44
	iBackup=TUint32(aBackupToc)<<1;
sl@0
    45
	iHandle=aHandle;
sl@0
    46
	iRef=aReference;
sl@0
    47
	iCrc=0;
sl@0
    48
	Mem::Crc(iCrc,Ptr(),_FOFF(TPermanentStoreHeader,iCrc));
sl@0
    49
	__ASSERT_DEBUG(IsValid(),User::Invariant());
sl@0
    50
	}
sl@0
    51
sl@0
    52
const TInt KTocGranularity=12;
sl@0
    53
sl@0
    54
struct STocEntry
sl@0
    55
	{
sl@0
    56
	TInt32 handle;
sl@0
    57
	TInt32 ref;
sl@0
    58
	};
sl@0
    59
sl@0
    60
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
    61
/////////////////////              CPermanentStoreToc               ////////////////////////////////////////////////////////
sl@0
    62
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
    63
sl@0
    64
TInt CPermanentStoreToc::TEntry::Compare(const TEntry& aLeft, const TEntry& aRight)
sl@0
    65
//
sl@0
    66
// Ordering for Stream handles, return <0,0,>0 depending on comparison
sl@0
    67
// The index part of the handle is only 24 bits, so just subtract the handles for result
sl@0
    68
//
sl@0
    69
	{
sl@0
    70
	return (aLeft.handle&KMaskHandleIndex) - (aRight.handle&KMaskHandleIndex);
sl@0
    71
	}
sl@0
    72
sl@0
    73
CPermanentStoreToc* CPermanentStoreToc::NewL(TStreamPos aBase,TStreamExchange& aHost,TInt aToc,TInt aBaseReloc)
sl@0
    74
	{
sl@0
    75
	CPermanentStoreToc* table=new(ELeave) CPermanentStoreToc(aBase,aHost);
sl@0
    76
	CleanupStack::PushL(table);
sl@0
    77
	table->ConstructL(aToc,aBaseReloc);
sl@0
    78
	CleanupStack::Pop();
sl@0
    79
	return table;
sl@0
    80
	}
sl@0
    81
sl@0
    82
TBool CPermanentStoreToc::IsDelta() const
sl@0
    83
//
sl@0
    84
// Report whether the current TOC should be written as a delta, or a base TOC
sl@0
    85
//
sl@0
    86
	{
sl@0
    87
	TInt delta = iEntries.Count();
sl@0
    88
	if (delta >= KTocDeltaCap)
sl@0
    89
		return EFalse;
sl@0
    90
	TInt magic = iMagic + delta;
sl@0
    91
	if (magic > KMaxTocDeltaMagic)
sl@0
    92
		return EFalse;
sl@0
    93
	return magic*KSizeTocDeltaEntry <= iCount*KSizeTocEntry - KSizeTocDeltaExtra;
sl@0
    94
	}
sl@0
    95
sl@0
    96
void CPermanentStoreToc::Move(TInt aToc,TInt anExtent)
sl@0
    97
//
sl@0
    98
// Final stage of compaction, update to address new TOC
sl@0
    99
//
sl@0
   100
	{
sl@0
   101
	__ASSERT_DEBUG(anExtent<=iOff&&anExtent>=aToc&&anExtent-(aToc+KOffsetTocHeader)==iExt-iOff,User::Invariant());
sl@0
   102
	if (!HasDelta())
sl@0
   103
		{
sl@0
   104
		// Base-toc only, update the base toc position as well
sl@0
   105
		TInt window=iWindow-iTocExt;
sl@0
   106
		iTocOff=aToc+KOffsetTocHeader;
sl@0
   107
		iTocExt=anExtent;
sl@0
   108
		iWindow=anExtent+window;
sl@0
   109
		}
sl@0
   110
	iOff = aToc+KOffsetTocHeader;
sl@0
   111
	iExt = anExtent;
sl@0
   112
	}
sl@0
   113
sl@0
   114
TInt CPermanentStoreToc::RealizeL(TInt aPrimary,TInt anExtent) const
sl@0
   115
	{
sl@0
   116
	__ASSERT_DEBUG(IsVirtual(),User::Invariant());
sl@0
   117
	RFrame16Buf buf(Base());
sl@0
   118
	buf.PushL();
sl@0
   119
	const TBool delta = IsDelta();
sl@0
   120
	aPrimary &= KMaskStreamIdValue;
sl@0
   121
	if (delta)
sl@0
   122
		aPrimary |= KTocDelta;
sl@0
   123
	STocHead h;
sl@0
   124
	h.primary=TInt32(aPrimary);
sl@0
   125
	h.avail=TInt32(iAvail);
sl@0
   126
	h.count=TUint32(iCount);
sl@0
   127
//
sl@0
   128
	TInt off;
sl@0
   129
	if (delta)
sl@0
   130
		off=DeltaL(buf,anExtent,h);
sl@0
   131
	else
sl@0
   132
		off=RewriteL(buf,anExtent,h);
sl@0
   133
//
sl@0
   134
	buf.SynchL();
sl@0
   135
	CleanupStack::PopAndDestroy(&buf);
sl@0
   136
	return off-KOffsetTocHeader;
sl@0
   137
	}
sl@0
   138
sl@0
   139
void CPermanentStoreToc::Adopt(TInt aToc,TInt aPrimary)
sl@0
   140
//
sl@0
   141
// Final stage of Commit - all file changes have been successful
sl@0
   142
// Record the new Toc location and reset the changed flag
sl@0
   143
//
sl@0
   144
	{
sl@0
   145
	__ASSERT_DEBUG(aToc+KOffsetTocHeader>=iExt,User::Invariant());
sl@0
   146
sl@0
   147
	if (IsDelta())
sl@0
   148
		{
sl@0
   149
		// Adjust the TOC-delta location
sl@0
   150
		TInt c = iEntries.Count();
sl@0
   151
		iOff=aToc+KOffsetTocHeader;
sl@0
   152
		iExt=aToc + KSizeTocDeltaExtra + KSizeTocDeltaEntry*c;
sl@0
   153
		if (c > KTocDeltaMagic)
sl@0
   154
			iMagic += c - KTocDeltaMagic;
sl@0
   155
		}
sl@0
   156
	else
sl@0
   157
		{
sl@0
   158
		// Adjust all the TOC data and reset the delta-set
sl@0
   159
		TInt window = iTocOff>=0 ? iWindow-iTocOff : -KOffsetTocHeader;
sl@0
   160
		iTocOff=iOff=aToc+KOffsetTocHeader;
sl@0
   161
		iTocExt=iExt=aToc+KSizeTocEntry*iCount;
sl@0
   162
		iWindow = iTocOff + window;
sl@0
   163
		iEntries.Reset();
sl@0
   164
		iMagic = 0;
sl@0
   165
		}
sl@0
   166
	iPrimary=aPrimary;
sl@0
   167
	__ASSERT_DEBUG(!IsVirtual(),User::Invariant());
sl@0
   168
	}
sl@0
   169
sl@0
   170
TInt CPermanentStoreToc::AllocL(TInt anOffset)
sl@0
   171
	{
sl@0
   172
	TEntry& entry=DoAllocL();
sl@0
   173
	entry.ref=anOffset;
sl@0
   174
	return entry.handle;
sl@0
   175
	}
sl@0
   176
sl@0
   177
TInt CPermanentStoreToc::AllocL()
sl@0
   178
	{
sl@0
   179
	TEntry& entry=DoAllocL();
sl@0
   180
	entry.ref=KFrameNonexistent16;
sl@0
   181
	return entry.handle|=KHandleInvalid;
sl@0
   182
	}
sl@0
   183
sl@0
   184
void CPermanentStoreToc::Cancel(TInt aHandle)
sl@0
   185
	{
sl@0
   186
	__ASSERT_DEBUG(aHandle<0&&IsVirtual(),User::Invariant());
sl@0
   187
	TEntry* entry=Entry(aHandle);
sl@0
   188
	__ASSERT_DEBUG((entry!=NULL) && (entry->handle==aHandle),User::Invariant());
sl@0
   189
	entry->ref=iAvail;
sl@0
   190
	iAvail=aHandle;
sl@0
   191
	}
sl@0
   192
sl@0
   193
void CPermanentStoreToc::FreeL(TInt aHandle)
sl@0
   194
	{
sl@0
   195
	__ASSERT_DEBUG(aHandle>0,User::Invariant());
sl@0
   196
	TEntry& entry=FetchL(aHandle);
sl@0
   197
	aHandle|=KHandleInvalid;
sl@0
   198
	entry.handle=aHandle;
sl@0
   199
	entry.ref=iAvail;
sl@0
   200
	iAvail=aHandle;
sl@0
   201
	Changed();
sl@0
   202
	}
sl@0
   203
sl@0
   204
TInt CPermanentStoreToc::AtL(TInt aHandle) const
sl@0
   205
	{
sl@0
   206
	__ASSERT_DEBUG(aHandle>0,User::Invariant());
sl@0
   207
sl@0
   208
	// check the delta table
sl@0
   209
	const TEntry* entry=Entry(aHandle);
sl@0
   210
	if (entry!=NULL )
sl@0
   211
		{
sl@0
   212
		// if there is an entry in the delta table, but its
sl@0
   213
		// not an exact match leave with KErrNotFound
sl@0
   214
		// This happens when the delta indicates it's been deleted
sl@0
   215
		if (entry->handle!=aHandle)
sl@0
   216
			User::Leave(KErrNotFound);
sl@0
   217
		
sl@0
   218
		return entry->ref;
sl@0
   219
		}
sl@0
   220
//
sl@0
   221
	// if it's not in the delta table check the base TOC
sl@0
   222
	return DoAtL(aHandle);
sl@0
   223
	}
sl@0
   224
sl@0
   225
void CPermanentStoreToc::PutL(TInt aHandle, TInt anOffset, TPut aCheck)
sl@0
   226
//
sl@0
   227
// Used by compaction to update a single stream reference IN-PLACE in the TOC
sl@0
   228
// We need to check in which TOC to make the update
sl@0
   229
//
sl@0
   230
	{
sl@0
   231
	__ASSERT_DEBUG(!IsVirtual(),User::Invariant());
sl@0
   232
	__ASSERT_DEBUG(aHandle>0,User::Invariant());
sl@0
   233
	__ASSERT_DEBUG(aHandle!=KHandleTocBase || HasDelta(),User::Invariant());
sl@0
   234
sl@0
   235
	if (aHandle == KHandleTocBase)
sl@0
   236
		{
sl@0
   237
		// update the TOC-base link
sl@0
   238
		PutTocL(anOffset - KOffsetTocHeader, aCheck);
sl@0
   239
		if (iTocOff != anOffset)
sl@0
   240
			{
sl@0
   241
			TInt size = iTocExt - iTocOff;
sl@0
   242
			TInt window = iWindow - iTocOff;
sl@0
   243
			iTocOff = anOffset;
sl@0
   244
			iTocExt = anOffset + size;
sl@0
   245
			iWindow = anOffset + window;
sl@0
   246
			}
sl@0
   247
		return;
sl@0
   248
		}
sl@0
   249
sl@0
   250
	TEntry e;
sl@0
   251
	e.handle=aHandle;
sl@0
   252
	TInt i;
sl@0
   253
	if (iEntries.FindInOrder(e,i,&TEntry::Compare)==0)
sl@0
   254
		{
sl@0
   255
		// update TOC-delta entry
sl@0
   256
		TEntry& entry=iEntries[i];
sl@0
   257
		if (entry.handle==aHandle && entry.ref != anOffset)
sl@0
   258
			{
sl@0
   259
			PutDeltaL(i,aHandle,anOffset);
sl@0
   260
			entry.ref=anOffset;
sl@0
   261
			}
sl@0
   262
		return;
sl@0
   263
		}
sl@0
   264
sl@0
   265
	// update TOC-base entry
sl@0
   266
	if (aCheck==EWrite || DoAtL(aHandle)!=anOffset)
sl@0
   267
		PutBaseL(aHandle,anOffset);
sl@0
   268
	}
sl@0
   269
sl@0
   270
TInt CPermanentStoreToc::GetL(TInt aHandle)
sl@0
   271
	{
sl@0
   272
	__ASSERT_DEBUG(aHandle>0,User::Invariant());
sl@0
   273
	TEntry& entry=FetchL(aHandle);
sl@0
   274
	return entry.ref;
sl@0
   275
	}
sl@0
   276
sl@0
   277
TInt CPermanentStoreToc::Set(TInt aHandle,TInt anOffset)
sl@0
   278
	{
sl@0
   279
	__ASSERT_DEBUG(aHandle!=0,User::Invariant());
sl@0
   280
	TEntry* entry=Entry(aHandle);
sl@0
   281
	__ASSERT_DEBUG((entry!=NULL) && (entry->handle==aHandle),User::Invariant());
sl@0
   282
	aHandle&=KMaskStreamIdValue;
sl@0
   283
	entry->handle=aHandle;
sl@0
   284
	entry->ref=anOffset;
sl@0
   285
	Changed();
sl@0
   286
	return aHandle;
sl@0
   287
	}
sl@0
   288
sl@0
   289
CPermanentStoreToc::CPermanentStoreToc(TStreamPos aBase,TStreamExchange& aHost)
sl@0
   290
	: /*iPrimary(0),iAvail(0),iCount(0),*/iEntries(KTocGranularity),
sl@0
   291
		iBase(aBase),iHost(&aHost),iOff(KFrameNonexistent16)/*,iExt(0)*/,
sl@0
   292
		iTocOff(KFrameNonexistent16)/*,iTocExt(0),iWindow(0)*/
sl@0
   293
	{}
sl@0
   294
sl@0
   295
CPermanentStoreToc::~CPermanentStoreToc()
sl@0
   296
	{
sl@0
   297
	iEntries.Close();
sl@0
   298
	}
sl@0
   299
sl@0
   300
void CPermanentStoreToc::ConstructL(TInt aToc,TInt aBaseReloc)
sl@0
   301
//
sl@0
   302
// Read and validate the TOC header at aToc.
sl@0
   303
// aBaseReloc may contain a relocated TOC-base offset, which should override the one in a TOC-delta
sl@0
   304
//
sl@0
   305
	{
sl@0
   306
	if (aToc==0)
sl@0
   307
		return;
sl@0
   308
	__ASSERT_DEBUG(aToc>0&&iEntries.Count()==0,User::Invariant());
sl@0
   309
	RFrame16Buf buf(Base());
sl@0
   310
	aToc+=KOffsetTocHeader;
sl@0
   311
	buf.OpenL(Host(),aToc,EFrameDescriptive16|buf.ERead);
sl@0
   312
	buf.PushL();
sl@0
   313
	RReadStream stream(&buf);
sl@0
   314
	STocHead h;
sl@0
   315
	stream.ReadL((TUint8*)&h,sizeof(STocHead)); // platform dependency
sl@0
   316
	if ((h.primary&~(KMaskStreamIdValue|static_cast<TUint>(KTocDelta)))!=0||
sl@0
   317
		h.avail>0||(h.avail&KMaskHandleClear)!=0||(h.count&~KMaskHandleIndex)!=0)
sl@0
   318
		__LEAVE(KErrCorrupt);
sl@0
   319
//
sl@0
   320
	iPrimary=TInt(h.primary) & ~KTocDelta;
sl@0
   321
	iAvail=TInt(h.avail);
sl@0
   322
	iCount=TInt(h.count);
sl@0
   323
	iOff = aToc;
sl@0
   324
//
sl@0
   325
	if (h.primary & KTocDelta)
sl@0
   326
		{
sl@0
   327
		// This is a TOC-delta
sl@0
   328
		aToc = InternalizeL(stream, aBaseReloc) + KOffsetTocHeader;
sl@0
   329
sl@0
   330
		// Now locate and validate the base TOC
sl@0
   331
		buf.Release();
sl@0
   332
		buf.OpenL(Host(),aToc,EFrameDescriptive16|buf.ERead);
sl@0
   333
		stream.ReadL((TUint8*)&h,sizeof(STocHead)); // platform dependency
sl@0
   334
		if ((h.primary&~KMaskStreamIdValue)!=0||(h.count&~KMaskHandleIndex)!=0||
sl@0
   335
			h.avail>0||(h.avail&KMaskHandleClear)!=0||TInt(h.count)>iCount)
sl@0
   336
			__LEAVE(KErrCorrupt);
sl@0
   337
		}
sl@0
   338
	TInt size = KSizeTocEntry*TInt(h.count);
sl@0
   339
// Eagerly load the first few TOC entries into the cache
sl@0
   340
// This is good as it almost certainly lies in the file buffer already
sl@0
   341
	if (size>0)
sl@0
   342
		stream.ReadL(&iBuf[0],Min(size,TInt(KSizeTocBuf)));
sl@0
   343
	size-=KOffsetTocHeader;
sl@0
   344
	if (buf.SizeL()!=size)
sl@0
   345
		__LEAVE(KErrCorrupt);
sl@0
   346
	iTocOff=aToc;
sl@0
   347
	iWindow=aToc-KOffsetTocHeader;
sl@0
   348
	iTocExt=aToc+size;
sl@0
   349
	if (iExt == 0)
sl@0
   350
		iExt = iTocExt;		// set extent for non-delta TOC
sl@0
   351
//
sl@0
   352
	CleanupStack::PopAndDestroy(&buf);
sl@0
   353
	}
sl@0
   354
sl@0
   355
TInt CPermanentStoreToc::InternalizeL(RReadStream& aIn, TInt aBaseReloc)
sl@0
   356
//
sl@0
   357
// Load and validate the delta-toc table
sl@0
   358
//
sl@0
   359
	{
sl@0
   360
	TInt tocoff = aIn.ReadInt32L();
sl@0
   361
	if (aBaseReloc != KFrameNonexistent16)
sl@0
   362
		tocoff = aBaseReloc - KOffsetTocHeader;
sl@0
   363
	if (tocoff<0||tocoff>=iOff)
sl@0
   364
		__LEAVE(KErrCorrupt);
sl@0
   365
	iMagic = aIn.ReadUint16L();
sl@0
   366
	if (!IsDelta())
sl@0
   367
		__LEAVE(KErrCorrupt);
sl@0
   368
	TInt n = aIn.ReadUint8L();
sl@0
   369
	TInt size = -KOffsetTocHeader + KSizeTocDeltaExtra + KSizeTocDeltaEntry * n;
sl@0
   370
	if (aIn.Source()->SizeL()!=size)
sl@0
   371
		__LEAVE(KErrCorrupt);
sl@0
   372
	iExt = iOff + size;
sl@0
   373
	TInt last = 0;
sl@0
   374
	while (--n >= 0)
sl@0
   375
		{
sl@0
   376
		STocEntry e;
sl@0
   377
		aIn.ReadL((TUint8*)&e,KSizeTocDeltaEntry);	// platform dependency
sl@0
   378
		if ((e.handle&KMaskHandleClear)!=0||e.handle>=0&&(e.ref<KFrameNonexistent16)||
sl@0
   379
			e.handle<0&&(e.ref>0||(e.ref&KMaskHandleClear)!=0))
sl@0
   380
			__LEAVE(KErrCorrupt);
sl@0
   381
		TInt i = e.handle&KMaskHandleIndex;
sl@0
   382
		if (i <= last || i > iCount)
sl@0
   383
			__LEAVE(KErrCorrupt);
sl@0
   384
		last = i;
sl@0
   385
		TEntry entry;
sl@0
   386
		entry.handle = TInt(e.handle);
sl@0
   387
		entry.ref = TInt(e.ref);
sl@0
   388
		User::LeaveIfError(iEntries.Append(entry));
sl@0
   389
		}
sl@0
   390
sl@0
   391
	return tocoff;
sl@0
   392
	}
sl@0
   393
sl@0
   394
TInt CPermanentStoreToc::DeltaL(RFrame16Buf& aBuf,TInt aExtent,const STocHead& aHead) const
sl@0
   395
//
sl@0
   396
// Write the TOC-delta
sl@0
   397
//
sl@0
   398
	{
sl@0
   399
	TInt off=aBuf.ExtendL(Host(),aExtent,EFrameDescriptive16|aBuf.EWrite);
sl@0
   400
	RWriteStream out(&aBuf);
sl@0
   401
	out.WriteL((TUint8*)&aHead,sizeof(STocHead)); // platform dependency
sl@0
   402
	out.WriteInt32L(iTocOff - KOffsetTocHeader);
sl@0
   403
	TInt n = iEntries.Count();
sl@0
   404
	TInt magic = iMagic;
sl@0
   405
	if (n > KTocDeltaMagic)
sl@0
   406
		magic += n - KTocDeltaMagic;
sl@0
   407
	__ASSERT_DEBUG(magic <= KMaxTocDeltaMagic,User::Invariant());
sl@0
   408
	__ASSERT_DEBUG(n <= (TInt)KMaxTUint8, User::Invariant());
sl@0
   409
	out.WriteUint16L(magic);
sl@0
   410
	out.WriteUint8L(n);
sl@0
   411
	for (int i = 0; i < n; ++i)
sl@0
   412
		{
sl@0
   413
		const TEntry& entry = iEntries[i];
sl@0
   414
		STocEntry e;
sl@0
   415
		e.handle=TInt32(entry.handle);
sl@0
   416
		e.ref=TInt32(entry.ref);
sl@0
   417
		out.WriteL((TUint8*)&e,KSizeTocDeltaEntry); // platform dependency
sl@0
   418
		}
sl@0
   419
	return off;
sl@0
   420
	}
sl@0
   421
sl@0
   422
TInt CPermanentStoreToc::RewriteL(RFrame16Buf& aBuf,TInt aExtent,const STocHead& aHead) const
sl@0
   423
//
sl@0
   424
// Write the TOC-base
sl@0
   425
//
sl@0
   426
	{
sl@0
   427
	const TInt KElementsRewrite=304;
sl@0
   428
	const TInt KSizeRewriteBuf=KElementsRewrite*KSizeTocEntry;
sl@0
   429
	TUint8 toc[KSizeRewriteBuf];
sl@0
   430
sl@0
   431
	RFrame16Buf buf(Base());
sl@0
   432
	buf.PushL();
sl@0
   433
	TInt oldsize = 0;
sl@0
   434
	TInt window = 0;
sl@0
   435
	if (iTocOff>=0)
sl@0
   436
		{
sl@0
   437
		oldsize = iTocExt-iTocOff+KOffsetTocHeader;
sl@0
   438
		window = iWindow + KOffsetTocHeader - iTocOff;
sl@0
   439
		if (oldsize <= Min(KSizeTocBuf,KSizeRewriteBuf) && window == 0)
sl@0
   440
			{
sl@0
   441
			// the old TOC is already in the toc-base cache, no need to read it
sl@0
   442
			Mem::Copy(&toc[0],&iBuf[0],oldsize);
sl@0
   443
			oldsize = 0;	// this prevents the read request
sl@0
   444
			}
sl@0
   445
		else
sl@0
   446
			{
sl@0
   447
			buf.Set(Host(),iTocOff,iTocExt,EFrameDescriptive16|buf.ERead);
sl@0
   448
			buf.SeekL(buf.ERead,TStreamPos(-KOffsetTocHeader));
sl@0
   449
			}
sl@0
   450
		}
sl@0
   451
	RReadStream in(&buf);
sl@0
   452
	// defer the initialisation of the write in roder to improve file buffering performance
sl@0
   453
	RWriteStream out(&aBuf);
sl@0
   454
	TInt off=-1;
sl@0
   455
	TInt size=iCount*KSizeTocEntry;
sl@0
   456
sl@0
   457
	for (TInt base=0,delta=0;base<size;base+=KSizeRewriteBuf)
sl@0
   458
		{
sl@0
   459
		// fill buffer with old TOC data
sl@0
   460
		if (base < oldsize)
sl@0
   461
			in.ReadL(&toc[0],Min(KSizeRewriteBuf,oldsize-base));
sl@0
   462
		// apply changes to this block
sl@0
   463
		for (TInt n=iEntries.Count();delta<n;++delta)
sl@0
   464
			{
sl@0
   465
			const TEntry& entry=iEntries[delta];
sl@0
   466
			TInt pos = (entry.handle&KMaskHandleIndex)*KSizeTocEntry - KSizeTocEntry - base;
sl@0
   467
			__ASSERT_DEBUG(pos>=0,User::Invariant());
sl@0
   468
			if (pos>=KSizeRewriteBuf)
sl@0
   469
				break;
sl@0
   470
			STocEntry e;
sl@0
   471
			e.handle=TInt32(entry.handle);
sl@0
   472
			e.ref=TInt32(entry.ref);
sl@0
   473
			Mem::Copy(&toc[pos],(const TUint8*)&e+KSizeHandleIndex,KSizeTocEntry);
sl@0
   474
			pos += base - window;
sl@0
   475
			if (TUint(pos) < TUint(KSizeTocBuf))
sl@0
   476
				Mem::Copy(MUTABLE_CAST(TUint8*,&iBuf[pos]),(const TUint8*)&e+KSizeHandleIndex,KSizeTocEntry);
sl@0
   477
			}
sl@0
   478
		// write buffer to file
sl@0
   479
		if (off == -1)
sl@0
   480
			{
sl@0
   481
			// initialise writing
sl@0
   482
			off=aBuf.ExtendL(Host(),aExtent,EFrameDescriptive16|aBuf.EWrite);
sl@0
   483
			out.WriteL((TUint8*)&aHead,sizeof(STocHead)); // platform dependency
sl@0
   484
			}
sl@0
   485
		out.WriteL(&toc[0],Min(KSizeRewriteBuf,size-base));
sl@0
   486
		}
sl@0
   487
//
sl@0
   488
	CleanupStack::PopAndDestroy(&buf);
sl@0
   489
	return off;
sl@0
   490
	}
sl@0
   491
sl@0
   492
CPermanentStoreToc::TEntry* CPermanentStoreToc::Entry(TInt aHandle)
sl@0
   493
	{
sl@0
   494
	TEntry e;
sl@0
   495
	e.handle=aHandle;
sl@0
   496
	TInt i;
sl@0
   497
	if (iEntries.FindInOrder(e,i,&TEntry::Compare)!=0)
sl@0
   498
		return NULL;
sl@0
   499
//
sl@0
   500
	TEntry& entry=iEntries[i];
sl@0
   501
	
sl@0
   502
	// allow entry to return a pointer even if not an exact match
sl@0
   503
	return &entry;
sl@0
   504
	}
sl@0
   505
sl@0
   506
CPermanentStoreToc::TEntry& CPermanentStoreToc::FetchL(TInt aHandle)
sl@0
   507
	{
sl@0
   508
	TEntry e;
sl@0
   509
	e.handle=aHandle;
sl@0
   510
	TInt i;
sl@0
   511
	if (iEntries.FindInOrder(e,i,&TEntry::Compare)!=0)
sl@0
   512
		{
sl@0
   513
		e.ref=DoAtL(aHandle);
sl@0
   514
		User::LeaveIfError(iEntries.Insert(e,i));
sl@0
   515
		}
sl@0
   516
	TEntry& entry=iEntries[i];
sl@0
   517
	if (entry.handle!=aHandle)
sl@0
   518
		__LEAVE(KErrNotFound);
sl@0
   519
//
sl@0
   520
	return entry;
sl@0
   521
	}
sl@0
   522
sl@0
   523
CPermanentStoreToc::TEntry& CPermanentStoreToc::DoAllocL()
sl@0
   524
	{
sl@0
   525
	TInt handle=iAvail;
sl@0
   526
	TEntry* entry;
sl@0
   527
	if (handle==0)
sl@0
   528
		{
sl@0
   529
		__ASSERT_DEBUG(iEntries.Count()==0||(iEntries[iEntries.Count()-1].handle&KMaskHandleIndex)<=iCount,User::Invariant());
sl@0
   530
		User::LeaveIfError(iEntries.Append(TEntry()));
sl@0
   531
		entry=&iEntries[iEntries.Count()-1];
sl@0
   532
		handle=++iCount;
sl@0
   533
		}
sl@0
   534
	else
sl@0
   535
		{
sl@0
   536
		entry=&FetchL(handle);
sl@0
   537
		handle=(handle+KIncHandleGen)&KMaskStreamIdValue;
sl@0
   538
//
sl@0
   539
		TInt avail=entry->ref;
sl@0
   540
		if (avail>0||(avail&KMaskHandleClear)!=0)
sl@0
   541
			__LEAVE(KErrCorrupt);
sl@0
   542
//
sl@0
   543
		iAvail=avail;
sl@0
   544
		}
sl@0
   545
	entry->handle=handle;
sl@0
   546
	Changed();
sl@0
   547
	return *entry;
sl@0
   548
	}
sl@0
   549
sl@0
   550
TInt CPermanentStoreToc::DoAtL(TInt aHandle) const
sl@0
   551
	{
sl@0
   552
	TInt off=iTocOff;
sl@0
   553
	if (off<0)
sl@0
   554
		__LEAVE(KErrNotFound);
sl@0
   555
//
sl@0
   556
	off+=-KOffsetTocHeader-KSizeTocEntry+KSizeTocEntry*(aHandle&KMaskHandleIndex);
sl@0
   557
	__ASSERT_DEBUG(off>=iTocOff-KOffsetTocHeader,User::Invariant());
sl@0
   558
	if (off>=iTocExt)
sl@0
   559
		__LEAVE(KErrNotFound);
sl@0
   560
//
sl@0
   561
	TInt window=iWindow;
sl@0
   562
	if (off-window<0||off-window>=KSizeTocBuf)
sl@0
   563
		{
sl@0
   564
		TInt len=iTocOff-KOffsetTocHeader;
sl@0
   565
		window=Max(len,Min(off-KBackTocBuf,iTocExt-KSizeTocBuf));
sl@0
   566
		len=Min(iTocExt-len,TInt(KSizeTocBuf));
sl@0
   567
//
sl@0
   568
		RFrame16Buf buf(Base());
sl@0
   569
		buf.Set(Host(),iTocOff,iTocExt,EFrameDescriptive16|buf.ERead);
sl@0
   570
		buf.PushL();
sl@0
   571
		buf.SeekL(buf.ERead,TStreamPos(window-iTocOff));
sl@0
   572
		RReadStream stream(&buf);
sl@0
   573
		MUTABLE_CAST(TInt&,iWindow)=iTocExt;
sl@0
   574
		stream.ReadL(MUTABLE_CAST(TUint8*,&iBuf[0]),1); // assume half decent read-ahead buffering
sl@0
   575
		stream.ReadL(MUTABLE_CAST(TUint8*,&iBuf[1]),len-1);
sl@0
   576
		MUTABLE_CAST(TInt&,iWindow)=window;
sl@0
   577
		CleanupStack::PopAndDestroy();
sl@0
   578
		}
sl@0
   579
	STocEntry e;
sl@0
   580
	e.handle=aHandle;
sl@0
   581
	Mem::Copy((TUint8*)&e+KSizeHandleIndex,&iBuf[off-window],KSizeTocEntry); // platform dependency
sl@0
   582
	if ((e.handle&KMaskHandleClear)!=0||e.handle>=0&&(e.ref<KFrameNonexistent16)||
sl@0
   583
		e.handle<0&&(e.ref>0||(e.ref&KMaskHandleClear)!=0))
sl@0
   584
		__LEAVE(KErrCorrupt);
sl@0
   585
//
sl@0
   586
	if (TInt(e.handle)!=aHandle)
sl@0
   587
		__LEAVE(KErrNotFound);
sl@0
   588
//
sl@0
   589
	return TInt(e.ref);
sl@0
   590
	}
sl@0
   591
sl@0
   592
void CPermanentStoreToc::PutBaseL(TInt aHandle,TInt aReference)
sl@0
   593
//
sl@0
   594
// Update a TOC-base entry
sl@0
   595
//
sl@0
   596
	{
sl@0
   597
	__ASSERT_DEBUG(iTocOff>=0,User::Invariant());
sl@0
   598
sl@0
   599
	TInt pos=-KOffsetTocHeader-KSizeTocEntry+KSizeTocEntry*(aHandle&KMaskHandleIndex);
sl@0
   600
	RFrame16Buf buf(Base());
sl@0
   601
	buf.Set(Host(),iTocOff,iTocExt,EFrameDescriptive16|buf.EWrite);
sl@0
   602
	buf.PushL();
sl@0
   603
	buf.SeekL(buf.EWrite,TStreamPos(pos));
sl@0
   604
	RWriteStream stream(&buf);
sl@0
   605
	STocEntry e;
sl@0
   606
	e.handle=aHandle;
sl@0
   607
	e.ref=aReference;
sl@0
   608
	stream.WriteL((TUint8*)&e+KSizeHandleIndex,KSizeTocEntry); // platform dependency
sl@0
   609
	CleanupStack::PopAndDestroy();
sl@0
   610
	TInt off=pos+iTocOff-iWindow;
sl@0
   611
	if (off>=0&&off<KSizeTocBuf)
sl@0
   612
		Mem::Copy(&iBuf[off],(const TUint8*)&e+KSizeHandleIndex,KSizeTocEntry); // platform dependency
sl@0
   613
	}
sl@0
   614
sl@0
   615
void CPermanentStoreToc::PutDeltaL(TInt aPos, TInt aHandle, TInt aReference)
sl@0
   616
//
sl@0
   617
// Update a single stream reference IN-PLACE in the TOC-delta
sl@0
   618
//
sl@0
   619
	{
sl@0
   620
	__ASSERT_DEBUG(HasDelta(),User::Invariant());
sl@0
   621
	__ASSERT_DEBUG(iOff>=0,User::Invariant());
sl@0
   622
	RFrame16Buf buf(Base());
sl@0
   623
	buf.Set(Host(),iOff,iExt,EFrameDescriptive16|buf.EWrite);
sl@0
   624
	buf.PushL();
sl@0
   625
	buf.SeekL(buf.EWrite,TStreamPos(-KOffsetTocHeader + KSizeTocDeltaExtra + KSizeTocDeltaEntry*aPos));
sl@0
   626
	RWriteStream stream(&buf);
sl@0
   627
	STocEntry e;
sl@0
   628
	e.handle=aHandle;
sl@0
   629
	e.ref=aReference;
sl@0
   630
	stream.WriteL((TUint8*)&e,KSizeTocDeltaEntry); // platform dependency
sl@0
   631
	CleanupStack::PopAndDestroy();
sl@0
   632
	}
sl@0
   633
sl@0
   634
void CPermanentStoreToc::PutTocL(TInt aTocBase, TPut aCheck)
sl@0
   635
//
sl@0
   636
// Update the base TOC link in the TOC-delta
sl@0
   637
//
sl@0
   638
	{
sl@0
   639
	__ASSERT_DEBUG(HasDelta(),User::Invariant());
sl@0
   640
	__ASSERT_DEBUG(iOff>=0,User::Invariant());
sl@0
   641
	RFrame16Buf buf(Base());
sl@0
   642
	buf.Set(Host(),iOff,iExt,EFrameDescriptive16|buf.EWrite|buf.ERead);
sl@0
   643
	buf.PushL();
sl@0
   644
	buf.SeekL(buf.EWrite|buf.ERead,TStreamPos(-KOffsetTocHeader));
sl@0
   645
	RReadStream in(&buf);
sl@0
   646
	if (aCheck==EWrite || in.ReadInt32L() != aTocBase)
sl@0
   647
		{
sl@0
   648
		RWriteStream out(&buf);
sl@0
   649
		out.WriteInt32L(aTocBase);
sl@0
   650
		}
sl@0
   651
	CleanupStack::PopAndDestroy();
sl@0
   652
	}
sl@0
   653
sl@0
   654
// Used prior to PutL to determine exactly where in the file the TOC update will be written.
sl@0
   655
// This may be used to eliminate the pre-put write of the relocation information in the file
sl@0
   656
// header in situations where the write is entirely within an atomic file block.
sl@0
   657
//
sl@0
   658
// The detailed numbers used to offset into the TOCs are dependant on the TOC formats, as
sl@0
   659
// implicated by other members of this class
sl@0
   660
TInt CPermanentStoreToc::RefSpan(TInt aHandle,TInt& aLength)
sl@0
   661
    {
sl@0
   662
    __ASSERT_DEBUG(!IsVirtual(),User::Invariant());
sl@0
   663
    __ASSERT_DEBUG(aHandle>0,User::Invariant());
sl@0
   664
sl@0
   665
    if (aHandle == KHandleTocBase)
sl@0
   666
        {    // locate the TOC-base link
sl@0
   667
        __ASSERT_DEBUG(HasDelta(),User::Invariant());
sl@0
   668
        __ASSERT_DEBUG(iOff>=0,User::Invariant());
sl@0
   669
        aLength = sizeof(TInt32);
sl@0
   670
        return RFrame16Buf::Position(Base(), iOff - KOffsetTocHeader).Offset();
sl@0
   671
        }
sl@0
   672
sl@0
   673
    TEntry e;
sl@0
   674
    e.handle=aHandle;
sl@0
   675
    TInt ix;
sl@0
   676
    if (iEntries.FindInOrder(e, ix, &TEntry::Compare) == 0)
sl@0
   677
        {    // locate TOC-delta entry
sl@0
   678
        TEntry& entry=iEntries[ix];
sl@0
   679
        if (entry.handle==aHandle)
sl@0
   680
            {
sl@0
   681
            __ASSERT_DEBUG(HasDelta(),User::Invariant());
sl@0
   682
            __ASSERT_DEBUG(iOff>=0,User::Invariant());
sl@0
   683
            aLength = KSizeTocDeltaEntry;
sl@0
   684
            return RFrame16Buf::Position(Base(),iOff - KOffsetTocHeader + KSizeTocDeltaExtra + KSizeTocDeltaEntry*ix).Offset();
sl@0
   685
            }
sl@0
   686
        }
sl@0
   687
sl@0
   688
    // locate the TOC-base entry
sl@0
   689
    __ASSERT_DEBUG(iTocOff>=0,User::Invariant());
sl@0
   690
    aLength = KSizeTocEntry;
sl@0
   691
    return RFrame16Buf::Position(Base(),iTocOff-KOffsetTocHeader-KSizeTocEntry+KSizeTocEntry*(aHandle&KMaskHandleIndex)).Offset();
sl@0
   692
    }
sl@0
   693
sl@0
   694
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
   695
/////////////////////              RPermanentStoreTocIter               ////////////////////////////////////////////////////
sl@0
   696
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
   697
sl@0
   698
RPermanentStoreTocIter::RPermanentStoreTocIter(const CPermanentStoreToc& aTable)
sl@0
   699
	: iTable(aTable), iBuf(aTable.Base()), iDelta(0), iDeltaEnd(0)
sl@0
   700
	{
sl@0
   701
	TInt off=aTable.iTocOff;
sl@0
   702
	__ASSERT_DEBUG(off>=KFrameNonexistent16,User::Invariant());
sl@0
   703
	TInt ext=off<0?off:aTable.iTocExt;
sl@0
   704
	iBuf.Set(aTable.Host(),off,ext,EFrameDescriptive16|iBuf.ERead);
sl@0
   705
	}
sl@0
   706
sl@0
   707
void RPermanentStoreTocIter::Release()
sl@0
   708
	{
sl@0
   709
	iBuf.Release();
sl@0
   710
	}
sl@0
   711
sl@0
   712
void RPermanentStoreTocIter::ResetL()
sl@0
   713
	{
sl@0
   714
	iNext = iIndex = 1;
sl@0
   715
	if (iBuf.Offset()<0)
sl@0
   716
		{
sl@0
   717
		iCount=0;
sl@0
   718
		return;
sl@0
   719
		}
sl@0
   720
//
sl@0
   721
	RReadStream stream(&iBuf);
sl@0
   722
	STocHead h;
sl@0
   723
	stream.ReadL((TUint8*)&h,sizeof(STocHead)); // platform dependency
sl@0
   724
	if (h.primary<0||(h.primary&~KMaskStreamIdValue)!=0||
sl@0
   725
		h.avail>0||(h.avail&KMaskHandleClear)!=0||(h.count&~KMaskHandleIndex)!=0)
sl@0
   726
		__LEAVE(KErrCorrupt);
sl@0
   727
//
sl@0
   728
	const CPermanentStoreToc& table=iTable;
sl@0
   729
	if (table.HasDelta())
sl@0
   730
		{
sl@0
   731
		TInt c = table.iEntries.Count();
sl@0
   732
		if (c)
sl@0
   733
			{
sl@0
   734
			iDelta = &table.iEntries[0];
sl@0
   735
			iDeltaEnd = iDelta + c;
sl@0
   736
			}
sl@0
   737
		iIndex = 0;
sl@0
   738
		}
sl@0
   739
	iCount = iTable.iCount;
sl@0
   740
	}
sl@0
   741
sl@0
   742
TBool RPermanentStoreTocIter::NextL(TEntry& anEntry)
sl@0
   743
	{
sl@0
   744
	TInt i=iIndex;
sl@0
   745
	__ASSERT_DEBUG(iCount>=0,User::Invariant());
sl@0
   746
	if (i == 0)
sl@0
   747
		{
sl@0
   748
		// report TOC-base as a 'stream'
sl@0
   749
		anEntry.handle = KHandleTocBase;
sl@0
   750
		anEntry.ref = iTable.iTocOff;
sl@0
   751
		iIndex = 1;
sl@0
   752
		return ETrue;
sl@0
   753
		}
sl@0
   754
sl@0
   755
	__ASSERT_DEBUG(i>0,User::Invariant());
sl@0
   756
	if (i>iCount)
sl@0
   757
		return EFalse;
sl@0
   758
//
sl@0
   759
	const TEntry* d = iDelta;
sl@0
   760
	if (d != iDeltaEnd && (d->handle&KMaskHandleIndex) == i)
sl@0
   761
		{
sl@0
   762
		anEntry = *d;
sl@0
   763
		iDelta = d+1;
sl@0
   764
		iIndex = i+1;
sl@0
   765
		return ETrue;
sl@0
   766
		}
sl@0
   767
//
sl@0
   768
	RReadStream stream(&iBuf);
sl@0
   769
	TInt skip = i - iNext;
sl@0
   770
	if (skip > 0)
sl@0
   771
		stream.ReadL(KSizeTocEntry * skip);
sl@0
   772
	STocEntry e;
sl@0
   773
	e.handle=i;
sl@0
   774
	stream.ReadL((TUint8*)&e+KSizeHandleIndex,KSizeTocEntry); // platform dependency
sl@0
   775
	if ((e.handle&KMaskHandleClear)!=0||e.handle>=0&&e.ref<KFrameNonexistent16||
sl@0
   776
		e.handle<0&&(e.ref>0||(e.ref&KMaskHandleClear)!=0))
sl@0
   777
		__LEAVE(KErrCorrupt);
sl@0
   778
//
sl@0
   779
	iNext = iIndex = i+1;
sl@0
   780
	anEntry.handle=TInt(e.handle);
sl@0
   781
	anEntry.ref=TInt(e.ref);
sl@0
   782
	return ETrue;
sl@0
   783
	}
sl@0
   784
sl@0
   785
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
   786
/////////////////////              TPermanentStoreCache               //////////////////////////////////////////////////////
sl@0
   787
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
   788
sl@0
   789
const TPermanentStoreCache::TItem* TPermanentStoreCache::At(TInt aHandle) const
sl@0
   790
	{
sl@0
   791
	const TItem* item=&iItems[0];
sl@0
   792
	TInt i=1;
sl@0
   793
	TInt bit=1;
sl@0
   794
	while (item->handle!=aHandle)
sl@0
   795
		{
sl@0
   796
		__ASSERT_DEBUG(item==&iItems[i-1],User::Invariant());
sl@0
   797
		TInt step=(aHandle&bit)?i+1:i;
sl@0
   798
		bit<<=1;
sl@0
   799
		i+=step;
sl@0
   800
		if (i>EItems)
sl@0
   801
			return NULL;
sl@0
   802
//
sl@0
   803
		item+=step;
sl@0
   804
		}
sl@0
   805
	__ASSERT_DEBUG(item==&iItems[i-1],User::Invariant());
sl@0
   806
	return item;
sl@0
   807
	}
sl@0
   808
sl@0
   809
void TPermanentStoreCache::Relocated(TInt aHandle,TInt anOffset)
sl@0
   810
	{
sl@0
   811
	const TItem* item=At(aHandle);
sl@0
   812
	if (item)
sl@0
   813
		{
sl@0
   814
		// update the cache item with the new offset
sl@0
   815
		// As 'extent' may only be a partial extent of a fragmented frame
sl@0
   816
		// which is no longer fragmented, we cannot simply shift this by the same
sl@0
   817
		// amount as 'offset'. Simplest to reset this to 'unknown', i.e. 0
sl@0
   818
		const_cast<TItem*>(item)->offset = anOffset;
sl@0
   819
		const_cast<TItem*>(item)->extent = 0;
sl@0
   820
		}
sl@0
   821
	}
sl@0
   822
sl@0
   823
void TPermanentStoreCache::Put(const TItem* anItem,TInt anOffset,TInt anExtent)
sl@0
   824
	{
sl@0
   825
	TInt handle=anItem->handle;
sl@0
   826
	TItem* item=&iItems[0];
sl@0
   827
	TInt i=1;
sl@0
   828
	TInt bit=1;
sl@0
   829
	while (item != anItem)
sl@0
   830
		{
sl@0
   831
		__ASSERT_DEBUG(item==&iItems[i-1],User::Invariant());
sl@0
   832
		TInt step=(handle&bit)?i+1:i;
sl@0
   833
		bit<<=1;
sl@0
   834
		i+=step;
sl@0
   835
		__ASSERT_DEBUG(i<=EItems,User::Invariant());
sl@0
   836
		item+=step;
sl@0
   837
		}
sl@0
   838
	__ASSERT_DEBUG(item==&iItems[i-1],User::Invariant());
sl@0
   839
sl@0
   840
	for(;;)
sl@0
   841
		{
sl@0
   842
		__ASSERT_DEBUG(item==&iItems[i-1],User::Invariant());
sl@0
   843
		TInt step=(handle&bit)?i+1:i;
sl@0
   844
		bit<<=1;
sl@0
   845
		i+=step;
sl@0
   846
		if (i>EItems)
sl@0
   847
			break;
sl@0
   848
//
sl@0
   849
		TItem* hole=item;
sl@0
   850
		item+=step;
sl@0
   851
		*hole=*item;
sl@0
   852
		};
sl@0
   853
	item->handle=handle;
sl@0
   854
	item->offset=anOffset;
sl@0
   855
	item->extent=anExtent;
sl@0
   856
	__ASSERT_DEBUG(At(handle)==item,User::Invariant());
sl@0
   857
	}
sl@0
   858
sl@0
   859
void TPermanentStoreCache::Add(TInt aHandle,TInt anOffset,TInt anExtent)
sl@0
   860
	{
sl@0
   861
	TItem* item=&iItems[0];
sl@0
   862
	TInt i=1;
sl@0
   863
	TInt bit=1;
sl@0
   864
	for(;;)
sl@0
   865
		{
sl@0
   866
		__ASSERT_DEBUG(item==&iItems[i-1]&&item->handle!=aHandle,User::Invariant());
sl@0
   867
		TInt step=(aHandle&bit)?i+1:i;
sl@0
   868
		bit<<=1;
sl@0
   869
		i+=step;
sl@0
   870
		if (i>EItems)
sl@0
   871
			break;
sl@0
   872
//
sl@0
   873
		TItem* hole=item;
sl@0
   874
		item+=step;
sl@0
   875
		*hole=*item;
sl@0
   876
		};
sl@0
   877
	item->handle=aHandle;
sl@0
   878
	item->offset=anOffset;
sl@0
   879
	item->extent=anExtent;
sl@0
   880
	__ASSERT_DEBUG(At(aHandle)==item,User::Invariant());
sl@0
   881
	}
sl@0
   882
sl@0
   883
void TPermanentStoreCache::Remove(TInt aHandle)
sl@0
   884
	{
sl@0
   885
	TItem* item=&iItems[0];
sl@0
   886
	TInt i=1;
sl@0
   887
	TInt bit=1;
sl@0
   888
	while (item->handle!=aHandle)
sl@0
   889
		{
sl@0
   890
		__ASSERT_DEBUG(item==&iItems[i-1],User::Invariant());
sl@0
   891
		TInt step=(aHandle&bit)?i+1:i;
sl@0
   892
		bit<<=1;
sl@0
   893
		i+=step;
sl@0
   894
		if (i>EItems)
sl@0
   895
			return;
sl@0
   896
//
sl@0
   897
		item+=step;
sl@0
   898
		}
sl@0
   899
	TItem* hole=item;
sl@0
   900
	TInt mask=bit-1;
sl@0
   901
	while (item!=&iItems[0])
sl@0
   902
		{
sl@0
   903
		__ASSERT_DEBUG(item==&iItems[i-1],User::Invariant());
sl@0
   904
		TInt step=i;
sl@0
   905
		bit>>=1;
sl@0
   906
		i>>=1;
sl@0
   907
		step-=i;
sl@0
   908
		item-=step;
sl@0
   909
//
sl@0
   910
		if (((aHandle^item->handle)&mask)==0)
sl@0
   911
			{
sl@0
   912
			*hole=*item;
sl@0
   913
			hole=item;
sl@0
   914
			mask=bit-1;
sl@0
   915
			}
sl@0
   916
		}
sl@0
   917
	hole->handle=0;
sl@0
   918
	__ASSERT_DEBUG(i==1&&At(aHandle)==NULL,User::Invariant());
sl@0
   919
	}
sl@0
   920
sl@0
   921
void TPermanentStoreCache::Invalidate()
sl@0
   922
	{
sl@0
   923
	for (TItem* item=&iItems[0],*end=item+EItems;item<end;++item)
sl@0
   924
		item->handle=0;
sl@0
   925
	}
sl@0
   926
sl@0
   927
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
   928
/////////////////////              CPermanentStoreCoord               //////////////////////////////////////////////////////
sl@0
   929
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
   930
sl@0
   931
//Returns the file system type, according to what the "file write" operation guarantees.
sl@0
   932
//The return result could be one of:
sl@0
   933
// - ESimple        - "single byte write" operations are atomic;
sl@0
   934
// - EBlockAtomic   - "block/sector write" operations are atomic;
sl@0
   935
// - ETransactional - transactional file system;
sl@0
   936
CPermanentStoreCoord::TFileQoS CPermanentStoreCoord::FileQoSL()
sl@0
   937
	{
sl@0
   938
	//Uncomment, if you want FileQoSL() to return always EBlockAtomic/EBlockAtomic.
sl@0
   939
	//iFileQos = ETransactional;	
sl@0
   940
	//iFileQos = EBlockAtomic;	
sl@0
   941
	//return iFileQos;
sl@0
   942
	//
sl@0
   943
	if (iFileQos == EUnknown) //get the file sytem type, if iFileQos is not set yet
sl@0
   944
		{
sl@0
   945
		TStreamExchange &se = Host();
sl@0
   946
		RFileBuf *sb = static_cast<RFileBuf *>(se.HostL());
sl@0
   947
		RFile &f = sb->File();
sl@0
   948
sl@0
   949
		TInt dn;
sl@0
   950
		TDriveInfo di;
sl@0
   951
		User::LeaveIfError(f.Drive(dn, di));
sl@0
   952
sl@0
   953
		iFileQos = (di.iDriveAtt & KDriveAttTransaction) ? ETransactional : ESimple;
sl@0
   954
		
sl@0
   955
		if(iFileQos == ESimple && IsBlockAtomicL(dn))
sl@0
   956
			{
sl@0
   957
			iFileQos = EBlockAtomic;	
sl@0
   958
			}
sl@0
   959
		}
sl@0
   960
	return iFileQos;
sl@0
   961
	}
sl@0
   962
sl@0
   963
//The function returns true, if the file system guarantees atomic "block write" operations on aDriveNo.
sl@0
   964
//It is not the most effective implementation at the moment (creates/closes a file session), 
sl@0
   965
//probably TDriveInfo::iType can be used in a more effective implementation.
sl@0
   966
TBool CPermanentStoreCoord::IsBlockAtomicL(TInt aDriveNo) const
sl@0
   967
	{
sl@0
   968
	__ASSERT_DEBUG(aDriveNo >= EDriveA && aDriveNo <= EDriveZ, User::Invariant());
sl@0
   969
	RFs fs;
sl@0
   970
	CleanupClosePushL(fs);
sl@0
   971
	User::LeaveIfError(fs.Connect());
sl@0
   972
	
sl@0
   973
	TVolumeIOParamInfo volInfo;
sl@0
   974
	TInt err = fs.VolumeIOParam(aDriveNo, volInfo);
sl@0
   975
	CleanupStack::PopAndDestroy(&fs);
sl@0
   976
	
sl@0
   977
	//If VolumeIOParam() succeeds, the media block size is >= 512 bytes and the media block size is power of two - report
sl@0
   978
	//that the media supports atomic "block write" operations.
sl@0
   979
	return err == KErrNone && volInfo.iBlockSize >= KDefaultMediaBlockSize && (volInfo.iBlockSize & (volInfo.iBlockSize - 1)) == 0;
sl@0
   980
	}
sl@0
   981
sl@0
   982
TStreamPos CPermanentStoreCoord::LimitL()
sl@0
   983
	{
sl@0
   984
	TableL();
sl@0
   985
	return RFrame16Buf::Position(Base(),iExt);
sl@0
   986
	}
sl@0
   987
sl@0
   988
TStreamId CPermanentStoreCoord::PrimaryL()
sl@0
   989
	{
sl@0
   990
	return TStreamId(TableL().Primary());
sl@0
   991
	}
sl@0
   992
sl@0
   993
void CPermanentStoreCoord::ChangedL()
sl@0
   994
	{
sl@0
   995
	ConsolidateL().Changed();
sl@0
   996
	}
sl@0
   997
sl@0
   998
TBool CPermanentStoreCoord::CommitL(TStreamId aPrimary)
sl@0
   999
	{
sl@0
  1000
	__ASSERT_DEBUG(IsTrim(),User::Invariant());
sl@0
  1001
	if (iExtend!=0)
sl@0
  1002
		__LEAVE(KErrInUse);
sl@0
  1003
//
sl@0
  1004
	CPermanentStoreToc& table=ConsolidateL();
sl@0
  1005
	if (!table.IsVirtual())
sl@0
  1006
		return EFalse;
sl@0
  1007
//
sl@0
  1008
	iState|=EClip;
sl@0
  1009
	TInt toc=table.RealizeL(aPrimary.Value(),iExt);
sl@0
  1010
//
sl@0
  1011
	TPermanentStoreHeader header(toc);
sl@0
  1012
	header.SetBackupToc(iToc);
sl@0
  1013
	MStreamBuf* buf=BeginL(header);
sl@0
  1014
	buf->SynchL();
sl@0
  1015
		// it's done, wrap things up
sl@0
  1016
	Host().Share(buf);
sl@0
  1017
	iToc=toc;
sl@0
  1018
	table.Adopt(toc,aPrimary.Value());
sl@0
  1019
	++iGen;
sl@0
  1020
	iState=EBackup;
sl@0
  1021
	iExt=table.Extent();
sl@0
  1022
	return ETrue;
sl@0
  1023
	}
sl@0
  1024
sl@0
  1025
TBool CPermanentStoreCoord::RevertL(TStreamId& aPrimary)
sl@0
  1026
	{
sl@0
  1027
	__ASSERT_ALWAYS(iAccess==0,Panic(EStoreInUse));
sl@0
  1028
		// can't revert things under people's feet
sl@0
  1029
	CPermanentStoreToc* table=iTable;
sl@0
  1030
	iTable=NULL;
sl@0
  1031
	iCache.Invalidate();
sl@0
  1032
	if (table==NULL||!table->IsVirtual())
sl@0
  1033
		{
sl@0
  1034
		__ASSERT_DEBUG(table==NULL||aPrimary==TStreamId(table->Primary()),User::Invariant());
sl@0
  1035
		delete table;
sl@0
  1036
		return EFalse;
sl@0
  1037
		}
sl@0
  1038
//
sl@0
  1039
	aPrimary=TStreamId(table->Primary());
sl@0
  1040
	delete table;
sl@0
  1041
	iState|=EClip;
sl@0
  1042
//
sl@0
  1043
	TStreamExchange& host=Host();
sl@0
  1044
	MStreamBuf* buf=host.HostL();
sl@0
  1045
	host.Release();
sl@0
  1046
	buf->SynchL();
sl@0
  1047
	host.Share(buf);
sl@0
  1048
	return ETrue;
sl@0
  1049
	}
sl@0
  1050
sl@0
  1051
TStreamId CPermanentStoreCoord::ExtendL()
sl@0
  1052
	{
sl@0
  1053
	return TStreamId(ConsolidateL().AllocL(KFrameNonexistent16));
sl@0
  1054
	}
sl@0
  1055
sl@0
  1056
void CPermanentStoreCoord::DeleteL(TStreamId anId)
sl@0
  1057
	{
sl@0
  1058
	TInt handle=anId.Value();
sl@0
  1059
	ConsolidateL().FreeL(handle);
sl@0
  1060
	iCache.Remove(handle);
sl@0
  1061
	}
sl@0
  1062
sl@0
  1063
CPermanentStoreCoord::CPermanentStoreCoord(TStreamPos aBase,TStreamExchange& aHost)
sl@0
  1064
	: iBase(aBase),iHost(&aHost), iFileQos(EUnknown)
sl@0
  1065
	{}
sl@0
  1066
sl@0
  1067
//
sl@0
  1068
// Read and analyse the store header.
sl@0
  1069
// The whole header (14 bytes) is read from the file and:
sl@0
  1070
// - If the dirty bit is set, the backup TOC will be used;
sl@0
  1071
// - If the dirty bit is not set, and the backup TOC ref is not the same as the TOC ref,
sl@0
  1072
//   then it means the the backup TOC ref has not been written successfully, so the TOC ref will be used;
sl@0
  1073
void CPermanentStoreCoord::InternalizeL(RReadStream& aStream)
sl@0
  1074
	{
sl@0
  1075
	if (iTable!=NULL)
sl@0
  1076
		__LEAVE(KErrNotReady);
sl@0
  1077
//
sl@0
  1078
	iState=EClip;
sl@0
  1079
	TPermanentStoreHeader header;
sl@0
  1080
	aStream.ReadL(header.Ptr(),KPermanentStoreHeaderLength);
sl@0
  1081
		// determine where the toc lives
sl@0
  1082
	TInt toc=header.BackupToc();
sl@0
  1083
	if (header.IsDirty())
sl@0
  1084
		{
sl@0
  1085
		iReloc=0;
sl@0
  1086
		iTarget=0;
sl@0
  1087
		}
sl@0
  1088
	else
sl@0
  1089
		{
sl@0
  1090
		TInt handle=header.Handle();
sl@0
  1091
		TInt ref=header.Reference();
sl@0
  1092
		if (handle==0&&toc!=ref)
sl@0
  1093
			{ // toc pointer not backed up, validate as if it was
sl@0
  1094
			toc=ref;
sl@0
  1095
			iState|=EBackup;
sl@0
  1096
			header.SetBackupToc(toc);
sl@0
  1097
			}
sl@0
  1098
		if (!header.IsValid()) // not a permanent store or damaged beyond recognition
sl@0
  1099
			__LEAVE(KErrNotSupported);
sl@0
  1100
	//
sl@0
  1101
		if (toc<0||((handle&~KMaskStreamIdValue)!=0&&handle!=KHandleTocBase)||ref<0||(handle!=0&&ref>=toc+KOffsetTocHeader))
sl@0
  1102
			__LEAVE(KErrCorrupt); // integrity compromised
sl@0
  1103
	//
sl@0
  1104
		iReloc=handle;
sl@0
  1105
		iTarget=ref;
sl@0
  1106
		}
sl@0
  1107
	//
sl@0
  1108
	if (iToc!=0 && iToc!=toc)		// refresh produced a different toc
sl@0
  1109
		__LEAVE(KErrCorrupt);
sl@0
  1110
	iToc=toc;
sl@0
  1111
	}
sl@0
  1112
sl@0
  1113
CPermanentStoreCoord::~CPermanentStoreCoord()
sl@0
  1114
	{
sl@0
  1115
	__ASSERT_ALWAYS(iRefs==0,Panic(EStoreInUse));
sl@0
  1116
	delete iTable;
sl@0
  1117
	}
sl@0
  1118
sl@0
  1119
void CPermanentStoreCoord::CanExtendL()
sl@0
  1120
	{
sl@0
  1121
	__ASSERT_DEBUG(IsTrim(),User::Invariant());
sl@0
  1122
	if (iExtend!=0)
sl@0
  1123
		__LEAVE(KErrInUse);
sl@0
  1124
//
sl@0
  1125
	ConsolidateL();
sl@0
  1126
	}
sl@0
  1127
sl@0
  1128
TInt CPermanentStoreCoord::DoCreateL()
sl@0
  1129
	{
sl@0
  1130
	__ASSERT_DEBUG(IsTrim()&&iReloc==0&&iExtend==0,User::Invariant());
sl@0
  1131
	TInt handle=Table().AllocL();
sl@0
  1132
	iExtend=handle;
sl@0
  1133
	Inc();
sl@0
  1134
	++iAccess;
sl@0
  1135
	return handle;
sl@0
  1136
	}
sl@0
  1137
sl@0
  1138
void CPermanentStoreCoord::DoReplaceL(TInt aHandle)
sl@0
  1139
	{
sl@0
  1140
	__ASSERT_DEBUG(IsTrim()&&iReloc==0&&iExtend==0,User::Invariant());
sl@0
  1141
	TInt off=Table().GetL(aHandle);
sl@0
  1142
	const TItem* item=iCache.At(aHandle);
sl@0
  1143
	__ASSERT_DEBUG(item==NULL||iTable!=NULL&&item->handle==aHandle,User::Invariant());
sl@0
  1144
	if (item==NULL)
sl@0
  1145
		iCache.Add(aHandle,off,Min(off,0));
sl@0
  1146
	iExtend=aHandle;
sl@0
  1147
	Inc();
sl@0
  1148
	++iAccess;
sl@0
  1149
	}
sl@0
  1150
sl@0
  1151
TInt CPermanentStoreCoord::DoOpenL(TInt& anOffset,TInt aHandle)
sl@0
  1152
	{
sl@0
  1153
	const TItem* item=iCache.At(aHandle);
sl@0
  1154
	__ASSERT_DEBUG(item==NULL||iTable!=NULL&&item->handle==aHandle,User::Invariant());
sl@0
  1155
	TInt off;
sl@0
  1156
	TInt ext;
sl@0
  1157
	if (item==NULL)
sl@0
  1158
		{
sl@0
  1159
		off=TableL().AtL(aHandle);
sl@0
  1160
		if (iReloc==aHandle)
sl@0
  1161
			{
sl@0
  1162
			TInt trg=iTarget;
sl@0
  1163
			if (trg==off)
sl@0
  1164
				iReloc=0;
sl@0
  1165
			else
sl@0
  1166
				off=trg;
sl@0
  1167
			}
sl@0
  1168
		ext=Min(off,0); // ensures ext==off for empty streams
sl@0
  1169
		iCache.Add(aHandle,off,ext);
sl@0
  1170
		}
sl@0
  1171
	else
sl@0
  1172
		{
sl@0
  1173
		off=item->offset;
sl@0
  1174
		ext=item->extent;
sl@0
  1175
		}
sl@0
  1176
	Inc();
sl@0
  1177
	++iAccess;
sl@0
  1178
	anOffset=off;
sl@0
  1179
	return ext;
sl@0
  1180
	}
sl@0
  1181
sl@0
  1182
void CPermanentStoreCoord::DoRelease(TInt aHandle,TInt anOffset,TInt anExtent)
sl@0
  1183
	{
sl@0
  1184
	__ASSERT_DEBUG(aHandle!=0,User::Invariant());
sl@0
  1185
	Dec();
sl@0
  1186
	--iAccess;
sl@0
  1187
	if (anExtent==0)
sl@0
  1188
		{ // failed to commit the extending stream
sl@0
  1189
		__ASSERT_DEBUG(aHandle==iExtend&&IsTrim(),User::Invariant());
sl@0
  1190
		iState|=EClip;
sl@0
  1191
		iExtend=0;
sl@0
  1192
		if (aHandle<0)
sl@0
  1193
			Table().Cancel(aHandle);
sl@0
  1194
		}
sl@0
  1195
	else
sl@0
  1196
		{
sl@0
  1197
		const TItem* item=iCache.At(aHandle);
sl@0
  1198
		if (item!=NULL&&item->offset==anOffset&&anExtent>item->extent)
sl@0
  1199
			iCache.Put(item,anOffset,anExtent);
sl@0
  1200
		}
sl@0
  1201
	}
sl@0
  1202
sl@0
  1203
TInt CPermanentStoreCoord::DoCommit(TInt aHandle,TInt anOffset,TInt anExtent)
sl@0
  1204
	{
sl@0
  1205
	__ASSERT_DEBUG(aHandle!=0&&aHandle==iExtend&&(anExtent>=iExt||anOffset==anExtent),User::Invariant());
sl@0
  1206
	aHandle=Table().Set(aHandle,anOffset);
sl@0
  1207
	if (anExtent<0)
sl@0
  1208
		iCache.Remove(aHandle);
sl@0
  1209
	else
sl@0
  1210
		{
sl@0
  1211
		iExt=anExtent;
sl@0
  1212
		const TItem* item=iCache.At(aHandle);
sl@0
  1213
		if (item==NULL)
sl@0
  1214
			iCache.Add(aHandle,anOffset,anExtent);
sl@0
  1215
		else
sl@0
  1216
			iCache.Put(item,anOffset,anExtent);
sl@0
  1217
		}
sl@0
  1218
	iExtend=0;
sl@0
  1219
	return aHandle;
sl@0
  1220
	}
sl@0
  1221
sl@0
  1222
CPermanentStoreToc& CPermanentStoreCoord::TableL()
sl@0
  1223
	{
sl@0
  1224
	CPermanentStoreToc* table=iTable;
sl@0
  1225
	if (table==NULL)
sl@0
  1226
		{
sl@0
  1227
		table=CPermanentStoreToc::NewL(Base(),Host(),iToc,iReloc==KHandleTocBase?iTarget:KFrameNonexistent16);
sl@0
  1228
		iExt=table->Extent();
sl@0
  1229
		iTable=table;
sl@0
  1230
		}
sl@0
  1231
	return *table;
sl@0
  1232
	}
sl@0
  1233
sl@0
  1234
CPermanentStoreToc& CPermanentStoreCoord::ConsolidateL()
sl@0
  1235
	{
sl@0
  1236
	CPermanentStoreToc& table=TableL();
sl@0
  1237
	if (iReloc!=0)
sl@0
  1238
		{
sl@0
  1239
		table.PutL(iReloc,iTarget,CPermanentStoreToc::ETestBeforeWrite);
sl@0
  1240
		iCache.Relocated(iReloc,iTarget);
sl@0
  1241
		iReloc=0;
sl@0
  1242
		}
sl@0
  1243
	return table;
sl@0
  1244
	}
sl@0
  1245
sl@0
  1246
//After stream relocation, the stream entry in the TOC has to be updated with the new stream position.
sl@0
  1247
//If the file system is not transactional or if the file system is "block atomic", but the stream entry is split
sl@0
  1248
//on a block/sector boundary, then the stream handle will be stored in the permanent file store header, in case
sl@0
  1249
//if the TOC entry update fails.
sl@0
  1250
void CPermanentStoreCoord::RelocateL(TInt aHandle,TInt anOffset)
sl@0
  1251
	{
sl@0
  1252
	__ASSERT_DEBUG(!Accessed(),User::Invariant());
sl@0
  1253
	__ASSERT_DEBUG(iReloc==0,User::Invariant());
sl@0
  1254
sl@0
  1255
	TBool updateStoreHeader = ETrue;	
sl@0
  1256
	TFileQoS fileQos = FileQoSL();
sl@0
  1257
	if(fileQos == ETransactional)
sl@0
  1258
		{
sl@0
  1259
		updateStoreHeader = EFalse;	
sl@0
  1260
		}
sl@0
  1261
	else if(fileQos == EBlockAtomic)
sl@0
  1262
		{
sl@0
  1263
		TInt dataLen = 0;
sl@0
  1264
		TInt writePos = iTable->RefSpan(aHandle, dataLen);
sl@0
  1265
		__ASSERT_DEBUG(writePos >= 0 && dataLen > 0, User::Invariant());
sl@0
  1266
		TInt startSectorAddr = writePos & ~(KDefaultMediaBlockSize - 1);
sl@0
  1267
		TInt endSectorAddr = (writePos + dataLen - 1) & ~(KDefaultMediaBlockSize - 1);
sl@0
  1268
		if(startSectorAddr == endSectorAddr)
sl@0
  1269
			{
sl@0
  1270
			updateStoreHeader = EFalse;	
sl@0
  1271
			}
sl@0
  1272
		}
sl@0
  1273
	
sl@0
  1274
	if (updateStoreHeader)
sl@0
  1275
		{
sl@0
  1276
		TPermanentStoreHeader header(iToc,aHandle,anOffset);
sl@0
  1277
		Host().Share(BeginL(header));
sl@0
  1278
		iReloc=aHandle;
sl@0
  1279
		iTarget=anOffset;
sl@0
  1280
		}
sl@0
  1281
	++iGen;
sl@0
  1282
	iTable->PutL(aHandle,anOffset,CPermanentStoreToc::EWrite);
sl@0
  1283
	iCache.Relocated(aHandle,anOffset);
sl@0
  1284
	iReloc=0;
sl@0
  1285
	}
sl@0
  1286
sl@0
  1287
void CPermanentStoreCoord::MoveL(TInt aToc,TInt anExtent)
sl@0
  1288
	{
sl@0
  1289
	__ASSERT_DEBUG(iReloc==0,User::Invariant());
sl@0
  1290
	CPermanentStoreToc& table=Table();
sl@0
  1291
	TPermanentStoreHeader header(aToc);
sl@0
  1292
	header.SetBackupToc(iToc);
sl@0
  1293
	Host().Share(BeginL(header));
sl@0
  1294
		// update data structures but defer the write
sl@0
  1295
	iToc=aToc;
sl@0
  1296
	TInt ext=table.Extent();
sl@0
  1297
	table.Move(aToc,anExtent);
sl@0
  1298
	iState|=EBackup;
sl@0
  1299
	if (iExt==ext)
sl@0
  1300
		{
sl@0
  1301
		iExt=anExtent;
sl@0
  1302
		iState|=EClip;
sl@0
  1303
		}
sl@0
  1304
	}
sl@0
  1305
sl@0
  1306
//
sl@0
  1307
// Starts a pseudo-atomic update of the permanent file store header.
sl@0
  1308
//
sl@0
  1309
// For the effect to be 'atomic', writes need to meet the following requirements:
sl@0
  1310
// 1. When updating n bytes using a single write, bytes 2 through n remain unchanged unless the first byte also changes.
sl@0
  1311
// 2. Changes associated with successive write requests happen in strict sequence.
sl@0
  1312
// 3. Updating a single byte is atomic.
sl@0
  1313
//
sl@0
  1314
// Also, a failure to write to a location shall be reported no later than on the next
sl@0
  1315
// write to that or a different location, or on buffer synchronisation.
sl@0
  1316
//
sl@0
  1317
// The preconditions of the operation are:
sl@0
  1318
// - all stream insert/delete/relocate operations completed, file - updated;
sl@0
  1319
// - the TOC reference in the file store header points to the current (valid) TOC, which does not include the most recent
sl@0
  1320
//   changes, since the last commit;
sl@0
  1321
// - the in-memory backup TOC reference updated and made equal to the file stored TOC reference;
sl@0
  1322
//
sl@0
  1323
// The procedure consists of 3 "file-write" steps:
sl@0
  1324
// -1- write the backup TOC ref (4 bytes) to the permanent file store header.
sl@0
  1325
//     if this operation fails, when the store is reopened, the TOC ref will be used;
sl@0
  1326
// -2- set the dirty bit and write the whole file store header (14 bytes).
sl@0
  1327
//     If this operation fails, but the dirty bit has been successfully set, the backup TOC ref will be used,
sl@0
  1328
//     when the store is reopened;
sl@0
  1329
// -3- clear the dirty bit (1 byte "file write" op). The commit operation has completed successfully;
sl@0
  1330
MStreamBuf* CPermanentStoreCoord::BeginL(TPermanentStoreHeader& aHeader)
sl@0
  1331
	{
sl@0
  1332
	__ASSERT_DEBUG(!aHeader.IsDirty() && aHeader.BackupToc() == Toc() && iReloc == 0, User::Invariant());
sl@0
  1333
	MStreamBuf& buf=*Host().HostL();
sl@0
  1334
	buf.SeekL(buf.EWrite,Base()+KPermanentStoreHeaderOffset);
sl@0
  1335
	TFileQoS fileQos = FileQoSL();
sl@0
  1336
	if (fileQos<EBlockAtomic)
sl@0
  1337
		{
sl@0
  1338
		if (iState&EBackup)
sl@0
  1339
			{ // we've yet to write the backup toc, do that before clobbering the header
sl@0
  1340
			buf.WriteL(aHeader.Ptr(),KPermanentStoreBackupLength);
sl@0
  1341
			buf.SeekL(buf.EWrite,EStreamBeginning);
sl@0
  1342
			buf.SeekL(buf.EWrite,Base()+KPermanentStoreHeaderOffset);
sl@0
  1343
			iState&=~EBackup;
sl@0
  1344
			}
sl@0
  1345
		// first write--dirty flag set in the first byte, backup toc unchanged, new header otherwise
sl@0
  1346
		aHeader.MarkDirty();
sl@0
  1347
		}
sl@0
  1348
	Host().Release(); // from this point onwards any failure results in store shutdown
sl@0
  1349
//
sl@0
  1350
	buf.WriteL(aHeader.Ptr(),KPermanentStoreHeaderLength);
sl@0
  1351
	if (fileQos<EBlockAtomic)
sl@0
  1352
		{
sl@0
  1353
		buf.SeekL(buf.EWrite,EStreamBeginning);
sl@0
  1354
		buf.SeekL(buf.EWrite,Base()+KPermanentStoreHeaderOffset);
sl@0
  1355
		aHeader.SetBackupToc(iToc);
sl@0
  1356
			// second write--single byte write to clear the dirty flag, no change otherwise
sl@0
  1357
		buf.WriteL(aHeader.Ptr(),1);
sl@0
  1358
		}
sl@0
  1359
		// at this point synchronisation is atomic; ie, if successful the change
sl@0
  1360
		// has been recorded, and failure means it will never happen
sl@0
  1361
	return &buf;
sl@0
  1362
	}
sl@0
  1363
sl@0
  1364
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
  1365
/////////////////////              HPermanentStoreBuf               ////////////////////////////////////////////////////////
sl@0
  1366
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
  1367
sl@0
  1368
HPermanentStoreBuf* HPermanentStoreBuf::CreateL(CPermanentStoreCoord& aCoord,TStreamId& anId,TInt aMode)
sl@0
  1369
	{
sl@0
  1370
	HPermanentStoreBuf* buf=ExtendLC(aCoord,aMode);
sl@0
  1371
	TInt handle=aCoord.DoCreateL();
sl@0
  1372
	buf->iHandle=handle;
sl@0
  1373
	CleanupStack::Pop();
sl@0
  1374
	anId=TStreamId(handle&KMaskStreamIdValue);
sl@0
  1375
	return buf;
sl@0
  1376
	}
sl@0
  1377
sl@0
  1378
HPermanentStoreBuf* HPermanentStoreBuf::ReplaceL(CPermanentStoreCoord& aCoord,TStreamId anId,TInt aMode)
sl@0
  1379
	{
sl@0
  1380
	HPermanentStoreBuf* buf=ExtendLC(aCoord,aMode);
sl@0
  1381
	TInt handle=anId.Value();
sl@0
  1382
	aCoord.DoReplaceL(handle);
sl@0
  1383
	buf->iHandle=handle;
sl@0
  1384
	CleanupStack::Pop();
sl@0
  1385
	return buf;
sl@0
  1386
	}
sl@0
  1387
sl@0
  1388
HPermanentStoreBuf* HPermanentStoreBuf::OpenL(CPermanentStoreCoord& aCoord,TStreamId anId,TInt aMode)
sl@0
  1389
	{
sl@0
  1390
	HPermanentStoreBuf* buf=NewLC(aCoord);
sl@0
  1391
	TInt handle=anId.Value();
sl@0
  1392
	TInt off=KFrameNonexistent16;
sl@0
  1393
	TInt ext=aCoord.DoOpenL(off,handle);
sl@0
  1394
	buf->iHandle=handle;
sl@0
  1395
	if (ext!=0)
sl@0
  1396
		buf->RFrame16Buf::Set(aCoord.Host(),off,ext,EFrameData16|aMode);
sl@0
  1397
	else
sl@0
  1398
		buf->RFrame16Buf::OpenL(aCoord.Host(),off,EFrameData16|aMode);
sl@0
  1399
	CleanupStack::Pop();
sl@0
  1400
	return buf;
sl@0
  1401
	}
sl@0
  1402
sl@0
  1403
HPermanentStoreBuf::~HPermanentStoreBuf()
sl@0
  1404
	{
sl@0
  1405
	TInt handle=iHandle;
sl@0
  1406
	if (handle!=0)
sl@0
  1407
		Coord().DoRelease(handle,Offset(),Extent());
sl@0
  1408
	RFrame16Buf::DoRelease();
sl@0
  1409
	}
sl@0
  1410
sl@0
  1411
HPermanentStoreBuf* HPermanentStoreBuf::NewLC(CPermanentStoreCoord& aCoord)
sl@0
  1412
	{
sl@0
  1413
	HPermanentStoreBuf* buf=new(ELeave) HPermanentStoreBuf(aCoord);
sl@0
  1414
	buf->PushL();
sl@0
  1415
	return buf;
sl@0
  1416
	}
sl@0
  1417
sl@0
  1418
HPermanentStoreBuf* HPermanentStoreBuf::ExtendLC(CPermanentStoreCoord& aCoord,TInt aMode)
sl@0
  1419
	{
sl@0
  1420
	aCoord.CanExtendL();
sl@0
  1421
	HPermanentStoreBuf* buf=NewLC(aCoord);
sl@0
  1422
	buf->RFrame16Buf::ExtendL(aCoord.Host(),aCoord.iExt,EFrameData16|aMode);
sl@0
  1423
	__ASSERT_DEBUG(TStreamPos(aCoord.Host().SizeL())==buf->Position(buf->Offset()),User::Invariant());
sl@0
  1424
	return buf;
sl@0
  1425
	}
sl@0
  1426
sl@0
  1427
void HPermanentStoreBuf::DoRelease()
sl@0
  1428
	{
sl@0
  1429
	delete this;
sl@0
  1430
	}
sl@0
  1431
sl@0
  1432
void HPermanentStoreBuf::DoSynchL()
sl@0
  1433
	{
sl@0
  1434
	__ASSERT_DEBUG(iHandle!=0,User::Invariant());
sl@0
  1435
	if (IsCommitted())
sl@0
  1436
		return;
sl@0
  1437
//
sl@0
  1438
	CommitL();
sl@0
  1439
	iHandle=Coord().DoCommit(iHandle,Offset(),Extent());
sl@0
  1440
	}
sl@0
  1441