os/persistentdata/persistentstorage/store/USTOR/UT_COLL.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-2009 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 "UT_STD.H"
sl@0
    17
sl@0
    18
// PFS frame-related utilities
sl@0
    19
sl@0
    20
TInt SkipLink(TInt anExtent)
sl@0
    21
//
sl@0
    22
// anExtent is the end-of-stream, where is the next link pos?
sl@0
    23
// : add the size of the next frame link, or skip past anchor link
sl@0
    24
//
sl@0
    25
	{return anExtent+Min(KSizeFrameDes16,(-anExtent)&KMaskFrameLength16);}
sl@0
    26
sl@0
    27
inline TBool SpaceFor(TInt aSpace,TInt anEnd,TInt aLength)
sl@0
    28
// Check if there is space for at least aLength between links at aSpace and anEnd
sl@0
    29
	{return SkipLink(aSpace+aLength)<=anEnd;}
sl@0
    30
sl@0
    31
TBool Fits(TInt aSpace,TInt anEnd,TInt aLength)
sl@0
    32
//
sl@0
    33
// Check that aLength can be relocated into the space
sl@0
    34
// either an exact fit, or leaves space for a free frame of at least one byte
sl@0
    35
//
sl@0
    36
	{
sl@0
    37
	TInt end=SkipLink(aSpace+aLength);
sl@0
    38
	return end==anEnd ? ETrue : SpaceFor(end,anEnd,1);
sl@0
    39
	}
sl@0
    40
sl@0
    41
TInt ExtentL(TStreamExchange& aHost,TStreamMark& aMark,TStreamPos aBase,TInt aStream)
sl@0
    42
//
sl@0
    43
// scan a stream to discover it's extent
sl@0
    44
//
sl@0
    45
	{
sl@0
    46
	__ASSERT_DEBUG(aMark.RelatesTo(aHost),User::Invariant());
sl@0
    47
	__ASSERT_DEBUG(aBase>=KStreamBeginning,User::Invariant());
sl@0
    48
	__ASSERT_DEBUG(aStream>=0,User::Invariant());
sl@0
    49
//
sl@0
    50
	aMark.SeekL(aHost,RFrame16Buf::Position(aBase,aStream)-KSizeFrameDes16);
sl@0
    51
	TFrameDes16 des;
sl@0
    52
	aMark.ReadL(aHost,&des,KSizeFrameDes16);
sl@0
    53
	TInt frame=des;
sl@0
    54
	if ((frame&KMaskFrameType16)!=EFrameData16)
sl@0
    55
		if ((frame&KMaskFrameType16)!=EFrameDescriptive16)
sl@0
    56
		{
sl@0
    57
		aMark.Withdraw(aHost);
sl@0
    58
		__LEAVE(KErrCorrupt);
sl@0
    59
		}
sl@0
    60
//
sl@0
    61
	TInt anchor=((aStream>>KShiftFrameLength16)+1)<<KShiftFrameLength16;
sl@0
    62
	do
sl@0
    63
		{
sl@0
    64
		frame&=KMaskFrameLength16;
sl@0
    65
		TInt lim=anchor-aStream;
sl@0
    66
		if (frame!=KFrameOpen16)
sl@0
    67
			{
sl@0
    68
			if (frame>=lim)
sl@0
    69
				{
sl@0
    70
				aMark.Withdraw(aHost);
sl@0
    71
				__LEAVE(KErrCorrupt);
sl@0
    72
				}
sl@0
    73
			return aStream+frame;
sl@0
    74
			}
sl@0
    75
		aMark.SeekL(aHost,EStreamMark,lim);
sl@0
    76
		aStream=anchor;
sl@0
    77
		anchor+=KFrameFullLength16;
sl@0
    78
		aMark.ReadL(aHost,&des,KSizeFrameDes16);
sl@0
    79
		frame=des;
sl@0
    80
		} while ((frame&KMaskFrameType16)==EFrameContinuation16);
sl@0
    81
	return aStream;
sl@0
    82
	}
sl@0
    83
sl@0
    84
sl@0
    85
sl@0
    86
// Class TRelocatorInput
sl@0
    87
// Used to transfer a frame-set within the file
sl@0
    88
sl@0
    89
const TInt KRelocatorBufferSize=0xc00;
sl@0
    90
sl@0
    91
NONSHARABLE_CLASS(TRelocatorInput) : public MStreamInput
sl@0
    92
	{
sl@0
    93
public:
sl@0
    94
	inline TRelocatorInput(TStreamExchange& aHost,TStreamMark& aMark);
sl@0
    95
	void OpenL(TFrameType16 aType,TStreamPos aBase,TInt aPos,TInt aLength,TInt aTerminator);
sl@0
    96
	void CommitL();
sl@0
    97
private:
sl@0
    98
	TInt PushL(const TAny* aPtr,TInt aMaxLength);
sl@0
    99
	TStreamTransfer ReadFromL(MStreamBuf& aSource,TStreamTransfer aTransfer);
sl@0
   100
	void DesL(TFrameType16 aType);
sl@0
   101
//
sl@0
   102
	inline TStreamExchange& Host() const;
sl@0
   103
	inline TStreamMark& Mark();
sl@0
   104
private:
sl@0
   105
	TStreamExchange* iHost;
sl@0
   106
	TStreamMark* iMark;
sl@0
   107
	TFrameType16 iType;
sl@0
   108
	TInt iRemain;
sl@0
   109
	TInt iAnchor;
sl@0
   110
	TInt iTerminator;
sl@0
   111
	};
sl@0
   112
sl@0
   113
inline TRelocatorInput::TRelocatorInput(TStreamExchange& aHost,TStreamMark& aMark)
sl@0
   114
	: iHost(&aHost),iMark(&aMark)
sl@0
   115
	{}
sl@0
   116
inline TStreamExchange& TRelocatorInput::Host() const
sl@0
   117
	{
sl@0
   118
	__ASSERT_DEBUG(iHost!=NULL,User::Invariant());
sl@0
   119
	return *iHost;
sl@0
   120
	}
sl@0
   121
inline TStreamMark& TRelocatorInput::Mark()
sl@0
   122
	{
sl@0
   123
	__ASSERT_DEBUG(iMark!=NULL,User::Invariant());
sl@0
   124
	return *iMark;
sl@0
   125
	}
sl@0
   126
sl@0
   127
void TRelocatorInput::OpenL(TFrameType16 aType,TStreamPos aBase,TInt aPos,TInt aLength,TInt aTerminator)
sl@0
   128
//
sl@0
   129
// initiate the relocated stream
sl@0
   130
//
sl@0
   131
	{
sl@0
   132
	__ASSERT_DEBUG(aType!=EFrameFree16,User::Invariant());
sl@0
   133
	__ASSERT_DEBUG(aBase>=KStreamBeginning,User::Invariant());
sl@0
   134
	__ASSERT_DEBUG(aPos>=0,User::Invariant());
sl@0
   135
	__ASSERT_DEBUG(aLength>0,User::Invariant());
sl@0
   136
//
sl@0
   137
	iType=aType;
sl@0
   138
	iRemain=aLength;
sl@0
   139
	iTerminator=aTerminator;
sl@0
   140
	iAnchor=KFrameFullLength16-(aPos&KMaskFrameLength16);
sl@0
   141
	Mark().SeekL(Host(),RFrame16Buf::Position(aBase,aPos)-KSizeFrameDes16);
sl@0
   142
	}
sl@0
   143
sl@0
   144
sl@0
   145
void TRelocatorInput::CommitL()
sl@0
   146
//
sl@0
   147
// complete the relocated stream
sl@0
   148
//
sl@0
   149
	{
sl@0
   150
	__ASSERT_DEBUG(iRemain==0,User::Invariant());
sl@0
   151
	__ASSERT_DEBUG(iAnchor>=0,User::Invariant());
sl@0
   152
//
sl@0
   153
	if (iTerminator<0)
sl@0
   154
		return;
sl@0
   155
//
sl@0
   156
	TFrameDes16 des[2];
sl@0
   157
	des[1]=TFrameDes16(iTerminator);
sl@0
   158
	TInt l=KSizeFrameDes16;
sl@0
   159
	if (iAnchor<=KSizeFrameDes16)
sl@0
   160
		l+=iAnchor;
sl@0
   161
	Mark().WriteL(Host(),(const TUint8*)&des[2]-l,l);
sl@0
   162
	}
sl@0
   163
sl@0
   164
TInt TRelocatorInput::PushL(const TAny*,TInt)
sl@0
   165
//
sl@0
   166
// use a passive write through to the host
sl@0
   167
//
sl@0
   168
	{
sl@0
   169
	return 0;
sl@0
   170
	}
sl@0
   171
sl@0
   172
void TRelocatorInput::DesL(TFrameType16 aType)
sl@0
   173
//
sl@0
   174
// Write the next frame descriptor
sl@0
   175
//
sl@0
   176
	{
sl@0
   177
	__ASSERT_DEBUG(aType!=EFrameFree16,User::Invariant());
sl@0
   178
	TFrameDes16 des=TFrameDes16(iRemain<iAnchor ? aType+iRemain : aType+KFrameOpen16);
sl@0
   179
	Mark().WriteL(Host(),&des,KSizeFrameDes16);
sl@0
   180
	}
sl@0
   181
sl@0
   182
TStreamTransfer TRelocatorInput::ReadFromL(MStreamBuf& aSource,TStreamTransfer aTransfer)
sl@0
   183
//
sl@0
   184
// bulk of the transfer happens here
sl@0
   185
//
sl@0
   186
	{
sl@0
   187
	__ASSERT_DEBUG(!(aTransfer>iRemain),User::Invariant());
sl@0
   188
	do
sl@0
   189
		{
sl@0
   190
		TUint8 buf[KRelocatorBufferSize];
sl@0
   191
		TInt len=aSource.ReadL(buf,aTransfer[sizeof(buf)]);
sl@0
   192
		if (len==0)
sl@0
   193
			break;
sl@0
   194
		aTransfer-=len;
sl@0
   195
		const TUint8* p=buf;
sl@0
   196
		if (iType!=EFrameFree16)
sl@0
   197
			{
sl@0
   198
			DesL(iType);
sl@0
   199
			iType=EFrameFree16;
sl@0
   200
			}
sl@0
   201
		for (;;)
sl@0
   202
			{
sl@0
   203
			if (iAnchor==0)
sl@0
   204
				{	// write next anchor
sl@0
   205
				iAnchor=KFrameFullLength16;
sl@0
   206
				DesL(EFrameContinuation16);
sl@0
   207
				}
sl@0
   208
			TInt frame=Min(len,iAnchor);
sl@0
   209
			Mark().WriteL(Host(),p,frame);
sl@0
   210
			iAnchor-=frame;
sl@0
   211
			iRemain-=frame;
sl@0
   212
			len-=frame;
sl@0
   213
			if (len==0)
sl@0
   214
				break;
sl@0
   215
			p+=frame;
sl@0
   216
			}
sl@0
   217
		} while (aTransfer>0);
sl@0
   218
	return aTransfer;
sl@0
   219
	}
sl@0
   220
sl@0
   221
sl@0
   222
sl@0
   223
// Class TPermanentStoreRelocator
sl@0
   224
// used to locate streams for relocation in limited memory
sl@0
   225
sl@0
   226
class TPermanentStoreRelocator
sl@0
   227
	{
sl@0
   228
#if defined(__SMALL_BUNDLE)
sl@0
   229
	enum {EBundleSize=8-1};
sl@0
   230
#else
sl@0
   231
	enum {EBundleSize=64-1};
sl@0
   232
#endif
sl@0
   233
public:
sl@0
   234
	typedef CPermanentStoreCollector::TEntry TEntry;
sl@0
   235
public:
sl@0
   236
	void Reset();
sl@0
   237
	TBool Reset(TInt aPos);
sl@0
   238
	TInt FillL(CPermanentStoreToc& aToc);
sl@0
   239
	void EvaluateLengthsL(TStreamExchange& aHost,TStreamMark& aMark,TStreamPos aBase);
sl@0
   240
//
sl@0
   241
	TBool FindFit(TInt aSpace,TInt anEnd);
sl@0
   242
	inline const TEntry* Current() const;
sl@0
   243
	void Relocated(const TEntry* anEntry);
sl@0
   244
//
sl@0
   245
	TInt Extent(TInt aStream) const;
sl@0
   246
	inline TInt MinLength() const;
sl@0
   247
private:
sl@0
   248
	static void Push(TEntry* aHeap,TEntry* aHole,const TEntry& aValue);
sl@0
   249
	static TEntry PopPush(TEntry* aHeap,TEntry* anEnd,const TEntry& aValue);
sl@0
   250
private:
sl@0
   251
	TEntry* iNext;
sl@0
   252
	const TEntry* iFinish;
sl@0
   253
	TInt iMore;
sl@0
   254
	TInt iPos;
sl@0
   255
	TInt iCurrentMin,iMinLength;
sl@0
   256
	TEntry iTable[EBundleSize];
sl@0
   257
	};
sl@0
   258
sl@0
   259
inline const TPermanentStoreRelocator::TEntry* TPermanentStoreRelocator::Current() const
sl@0
   260
	{__ASSERT_DEBUG(iNext>=iTable&&iNext<iFinish,User::Invariant());return iNext;}
sl@0
   261
inline TInt TPermanentStoreRelocator::MinLength() const
sl@0
   262
	{return iMinLength;}
sl@0
   263
sl@0
   264
void TPermanentStoreRelocator::Reset()
sl@0
   265
//
sl@0
   266
// reset the cached length values
sl@0
   267
//
sl@0
   268
	{
sl@0
   269
	iPos=0;
sl@0
   270
	iMinLength=1;
sl@0
   271
	}
sl@0
   272
sl@0
   273
TBool TPermanentStoreRelocator::Reset(TInt aPos)
sl@0
   274
//
sl@0
   275
// reset the iterator for a new scan from aPos
sl@0
   276
//
sl@0
   277
	{
sl@0
   278
	__ASSERT_DEBUG(aPos>=0,User::Invariant());
sl@0
   279
//
sl@0
   280
	iCurrentMin=KMaxTInt;
sl@0
   281
	TEntry* e=iTable;
sl@0
   282
	if (aPos>=iPos || e==iFinish || aPos<e->entry.ref)
sl@0
   283
		{	// rescan required
sl@0
   284
		iFinish=iNext=NULL;
sl@0
   285
		iMore=-1;
sl@0
   286
		iPos=aPos;
sl@0
   287
		return EFalse;
sl@0
   288
		}
sl@0
   289
sl@0
   290
// can use current data
sl@0
   291
	for (;e->entry.ref!=aPos;++e)
sl@0
   292
		{
sl@0
   293
		__ASSERT_DEBUG(e->entry.ref<aPos,User::Invariant());
sl@0
   294
		__ASSERT_DEBUG(e<iFinish,User::Invariant());
sl@0
   295
		}
sl@0
   296
	__ASSERT_DEBUG(e->entry.handle>=0,User::Invariant());
sl@0
   297
	iNext=e;
sl@0
   298
	return ETrue;
sl@0
   299
	}
sl@0
   300
sl@0
   301
TBool TPermanentStoreRelocator::FillL(CPermanentStoreToc& aToc)
sl@0
   302
//
sl@0
   303
// Fill the table with the next bundle of stream offsets
sl@0
   304
// return if there are more available
sl@0
   305
//
sl@0
   306
	{
sl@0
   307
	__ASSERT_DEBUG(iNext==iFinish,User::Invariant());
sl@0
   308
	if (!iMore)
sl@0
   309
		return EFalse;
sl@0
   310
//
sl@0
   311
	__ASSERT_DEBUG(iPos>=0,User::Invariant());
sl@0
   312
//
sl@0
   313
	RPermanentStoreTocIter iter(aToc);
sl@0
   314
	CleanupReleasePushL(iter);
sl@0
   315
sl@0
   316
	TInt streams=0;
sl@0
   317
	TInt from=iPos;
sl@0
   318
	TEntry* table=iTable;
sl@0
   319
	TEntry* last=table;
sl@0
   320
	TEntry* end=table+EBundleSize;
sl@0
   321
	TEntry entry;
sl@0
   322
	__DEBUG(entry.len=-1);
sl@0
   323
	for (iter.ResetL();iter.NextL(entry.entry);)
sl@0
   324
		{
sl@0
   325
		if (entry.entry.handle<0)
sl@0
   326
			continue;
sl@0
   327
		TInt off=entry.entry.ref;
sl@0
   328
		if (off<from)
sl@0
   329
			continue;
sl@0
   330
		++streams;
sl@0
   331
		if (last<end)
sl@0
   332
			Push(table,last++,entry);	// push onto the bottom
sl@0
   333
		else if (off<table->entry.ref)
sl@0
   334
			PopPush(table,last,entry);	// replace item in heap
sl@0
   335
		}
sl@0
   336
	CleanupStack::PopAndDestroy();
sl@0
   337
sl@0
   338
	iMore=streams+table-last;
sl@0
   339
	if (streams)
sl@0
   340
		iPos=table->entry.ref+1;		// largest offset in table
sl@0
   341
	iNext=table;
sl@0
   342
	iFinish=last;
sl@0
   343
	while (--last>table)
sl@0
   344
		*last=PopPush(table,last,*last);
sl@0
   345
	return streams>0;
sl@0
   346
	}
sl@0
   347
sl@0
   348
void TPermanentStoreRelocator::Push(TEntry* aHeap,TEntry* aHole,const TEntry& aValue)
sl@0
   349
//
sl@0
   350
// push aValue onto the heap (which will grow)
sl@0
   351
//
sl@0
   352
	{
sl@0
   353
	TInt ix=aHole+1-aHeap;
sl@0
   354
	while (aHole>aHeap)
sl@0
   355
		{
sl@0
   356
		TInt link=ix-(ix>>1);
sl@0
   357
		ix-=link;
sl@0
   358
		TEntry* parent=aHole-link;
sl@0
   359
		if (parent->entry.ref>=aValue.entry.ref)
sl@0
   360
			break;
sl@0
   361
		*aHole=*parent;
sl@0
   362
		aHole=parent;
sl@0
   363
		}
sl@0
   364
	*aHole=aValue;
sl@0
   365
	}
sl@0
   366
sl@0
   367
TPermanentStoreRelocator::TEntry TPermanentStoreRelocator::PopPush(TEntry* aHeap,TEntry* anEnd,const TEntry& aValue)
sl@0
   368
//
sl@0
   369
// pop the top and push aValue
sl@0
   370
//
sl@0
   371
	{
sl@0
   372
	TEntry top=*aHeap;
sl@0
   373
	TEntry* hole=aHeap;
sl@0
   374
	--anEnd;
sl@0
   375
	TInt ix=1;
sl@0
   376
	for (;;)
sl@0
   377
		{
sl@0
   378
		TEntry* child=hole+ix;
sl@0
   379
		ix<<=1;
sl@0
   380
		if (child>anEnd)
sl@0
   381
			break;
sl@0
   382
		if (child<anEnd && (child+1)->entry.ref>child->entry.ref)
sl@0
   383
			{
sl@0
   384
			++ix;
sl@0
   385
			++child;
sl@0
   386
			}
sl@0
   387
		if (child->entry.ref<=aValue.entry.ref)
sl@0
   388
			break;
sl@0
   389
		*hole=*child;
sl@0
   390
		hole=child;
sl@0
   391
		}
sl@0
   392
	*hole=aValue;
sl@0
   393
	return top;
sl@0
   394
	}
sl@0
   395
sl@0
   396
void TPermanentStoreRelocator::EvaluateLengthsL(TStreamExchange& aHost,TStreamMark& aMark,TStreamPos aBase)
sl@0
   397
//
sl@0
   398
// Evaluate the lengths for all the entries in the table
sl@0
   399
//
sl@0
   400
	{
sl@0
   401
	const TEntry* end=iFinish;
sl@0
   402
	for (TEntry* e=iTable;e<end;++e)
sl@0
   403
		{
sl@0
   404
		__ASSERT_DEBUG(e->entry.handle>=0,User::Invariant());
sl@0
   405
		__ASSERT_DEBUG(e->entry.ref>=0,User::Invariant());
sl@0
   406
		__ASSERT_DEBUG(e->len==-1,User::Invariant());
sl@0
   407
		TInt pos=e->entry.ref;
sl@0
   408
		e->len=ExtentL(aHost,aMark,aBase,pos)-pos;
sl@0
   409
		}
sl@0
   410
	}
sl@0
   411
sl@0
   412
TBool TPermanentStoreRelocator::FindFit(TInt aSpace,TInt anEnd)
sl@0
   413
//
sl@0
   414
// Find a stream which will fit into the space
sl@0
   415
//
sl@0
   416
	{
sl@0
   417
	const TEntry* end=iFinish;
sl@0
   418
	TEntry* e;
sl@0
   419
	for (e=iNext;e<end;++e)
sl@0
   420
		{
sl@0
   421
		if (e->entry.handle<0)
sl@0
   422
			continue;
sl@0
   423
		TInt len=e->len;
sl@0
   424
		__ASSERT_DEBUG(len>0,User::Invariant());
sl@0
   425
		if (Fits(aSpace,anEnd,len))
sl@0
   426
			{
sl@0
   427
			iNext=e;
sl@0
   428
			return ETrue;
sl@0
   429
			}
sl@0
   430
		// len has been left behind, check for minimum
sl@0
   431
		if (len<iCurrentMin)
sl@0
   432
			iCurrentMin=len;
sl@0
   433
		}
sl@0
   434
// if no more data, use current min as cached value
sl@0
   435
	if (iMore==0)
sl@0
   436
		iMinLength=iCurrentMin;
sl@0
   437
	iNext=e;
sl@0
   438
	return EFalse;
sl@0
   439
	}
sl@0
   440
sl@0
   441
void TPermanentStoreRelocator::Relocated(const TEntry* anEntry)
sl@0
   442
//
sl@0
   443
// Relocation has been successful
sl@0
   444
//
sl@0
   445
	{
sl@0
   446
	__ASSERT_DEBUG(anEntry==iNext,User::Invariant());
sl@0
   447
	TEntry* e=CONST_CAST(TEntry*,anEntry);
sl@0
   448
	e->entry.handle=-1;
sl@0
   449
	iNext=e+1;
sl@0
   450
	}
sl@0
   451
sl@0
   452
TInt TPermanentStoreRelocator::Extent(TInt aStream) const
sl@0
   453
//
sl@0
   454
// Return this stream extent if we know it
sl@0
   455
//
sl@0
   456
	{
sl@0
   457
	const TEntry* e=iTable;
sl@0
   458
	if (aStream>=iPos || e==iFinish || aStream<e->entry.ref)
sl@0
   459
		return -1;	// we don't know it
sl@0
   460
	for (;e->entry.ref!=aStream;++e)
sl@0
   461
		{
sl@0
   462
		__ASSERT_DEBUG(e->entry.ref<aStream,User::Invariant());
sl@0
   463
		__ASSERT_DEBUG(e<iFinish,User::Invariant());
sl@0
   464
		}
sl@0
   465
	__ASSERT_DEBUG(e->entry.handle>=0,User::Invariant());
sl@0
   466
	__ASSERT_DEBUG(e->len>=0,User::Invariant());
sl@0
   467
	return aStream+e->len;
sl@0
   468
	}
sl@0
   469
sl@0
   470
sl@0
   471
sl@0
   472
// class TPermanentStoreStreamIter
sl@0
   473
sl@0
   474
void TPermanentStoreStreamIter::Reset()
sl@0
   475
//
sl@0
   476
// reset the iterator for a new scan
sl@0
   477
//
sl@0
   478
	{
sl@0
   479
	iNext=NULL;
sl@0
   480
	iFinish=NULL;
sl@0
   481
	iMore=-1;
sl@0
   482
	iPos=0;
sl@0
   483
	}
sl@0
   484
sl@0
   485
TInt TPermanentStoreStreamIter::FillL(CPermanentStoreToc& aToc)
sl@0
   486
//
sl@0
   487
// Fill the table with the next bundle of stream offsets
sl@0
   488
// return the total streams left to iterate
sl@0
   489
//
sl@0
   490
	{
sl@0
   491
	__ASSERT_DEBUG(iNext==iFinish,User::Invariant());
sl@0
   492
	if (!iMore)
sl@0
   493
		return 0;
sl@0
   494
//
sl@0
   495
	__ASSERT_DEBUG(iPos>=0,User::Invariant());
sl@0
   496
//
sl@0
   497
	RPermanentStoreTocIter iter(aToc);
sl@0
   498
	CleanupReleasePushL(iter);
sl@0
   499
sl@0
   500
	TInt streams=0;
sl@0
   501
	TInt from=iPos;
sl@0
   502
	TInt* heap=iTable;
sl@0
   503
	TInt* last=heap;
sl@0
   504
	TInt* end=heap+EBundleSize;
sl@0
   505
	RPermanentStoreTocIter::TEntry entry;
sl@0
   506
	for (iter.ResetL();iter.NextL(entry);)
sl@0
   507
		{
sl@0
   508
		if (entry.handle<0)
sl@0
   509
			continue;
sl@0
   510
		TInt off=entry.ref;
sl@0
   511
		if (off<from)
sl@0
   512
			continue;
sl@0
   513
		++streams;
sl@0
   514
		if (last<end)
sl@0
   515
			Push(heap,last++,off);	// push onto the bottom
sl@0
   516
		else if (off<*heap)
sl@0
   517
			PopPush(heap,last,off);	// replace item in heap
sl@0
   518
		}
sl@0
   519
	CleanupStack::PopAndDestroy();
sl@0
   520
sl@0
   521
	iMore=streams+(heap-last);
sl@0
   522
	iPos=*heap+1;		// largest offset in table
sl@0
   523
	iNext=heap;
sl@0
   524
	iFinish=last;
sl@0
   525
	while (--last>heap)
sl@0
   526
		*last=PopPush(heap,last,*last);
sl@0
   527
	return streams;
sl@0
   528
	}
sl@0
   529
sl@0
   530
TInt TPermanentStoreStreamIter::Next()
sl@0
   531
//
sl@0
   532
// return the next offset, or <0 if the table is empty
sl@0
   533
//
sl@0
   534
	{
sl@0
   535
	__ASSERT_DEBUG(iMore>=0,User::Invariant());
sl@0
   536
//
sl@0
   537
	while (iNext<iFinish)
sl@0
   538
		{
sl@0
   539
		TInt off=*iNext++;
sl@0
   540
		if (off>=0)
sl@0
   541
			return off;
sl@0
   542
		}
sl@0
   543
	return -1;
sl@0
   544
	}
sl@0
   545
sl@0
   546
void TPermanentStoreStreamIter::Relocated(TInt aStream)
sl@0
   547
//
sl@0
   548
// aStream has been relocated, mark the table
sl@0
   549
//
sl@0
   550
	{
sl@0
   551
	__ASSERT_DEBUG(iMore>=0,User::Invariant());
sl@0
   552
//
sl@0
   553
	if (aStream>=iPos)
sl@0
   554
		return;
sl@0
   555
	TInt* p=iNext;
sl@0
   556
	for (;;)
sl@0
   557
		{
sl@0
   558
		if (p==iFinish)
sl@0
   559
			return;
sl@0
   560
		if (*p>=0)
sl@0
   561
			break;
sl@0
   562
		++p;
sl@0
   563
		}
sl@0
   564
	if (aStream<*p)
sl@0
   565
		return;
sl@0
   566
//
sl@0
   567
	for (;*p!=aStream;++p)
sl@0
   568
		{
sl@0
   569
		__ASSERT_DEBUG(p<iFinish,User::Invariant());
sl@0
   570
		__ASSERT_DEBUG(*p<aStream,User::Invariant());
sl@0
   571
		}
sl@0
   572
	if (p==iNext)
sl@0
   573
		iNext=p+1;
sl@0
   574
	else
sl@0
   575
		*p=-1;
sl@0
   576
	}
sl@0
   577
sl@0
   578
void TPermanentStoreStreamIter::Push(TInt* aHeap,TInt* aHole,TInt aValue)
sl@0
   579
//
sl@0
   580
// push aValue onto the heap (which will grow)
sl@0
   581
//
sl@0
   582
	{
sl@0
   583
	TInt ix=aHole+1-aHeap;
sl@0
   584
	while (aHole>aHeap)
sl@0
   585
		{
sl@0
   586
		TInt link=ix-(ix>>1);
sl@0
   587
		ix-=link;
sl@0
   588
		TInt* parent=aHole-link;
sl@0
   589
		if (*parent>=aValue)
sl@0
   590
			break;
sl@0
   591
		*aHole=*parent;
sl@0
   592
		aHole=parent;
sl@0
   593
		}
sl@0
   594
	*aHole=aValue;
sl@0
   595
	}
sl@0
   596
sl@0
   597
TInt TPermanentStoreStreamIter::PopPush(TInt* aHeap,TInt* anEnd,TInt aValue)
sl@0
   598
//
sl@0
   599
// pop the top and push aValue
sl@0
   600
//
sl@0
   601
	{
sl@0
   602
	TInt top=*aHeap;
sl@0
   603
	TInt* hole=aHeap;
sl@0
   604
	--anEnd;
sl@0
   605
	TInt ix=1;
sl@0
   606
	for (;;)
sl@0
   607
		{
sl@0
   608
		TInt* child=hole+ix;
sl@0
   609
		ix<<=1;
sl@0
   610
		if (child>anEnd)
sl@0
   611
			break;
sl@0
   612
		if (child<anEnd && *(child+1)>*child)
sl@0
   613
			{
sl@0
   614
			++ix;
sl@0
   615
			++child;
sl@0
   616
			}
sl@0
   617
		if (*child<=aValue)
sl@0
   618
			break;
sl@0
   619
		*hole=*child;
sl@0
   620
		hole=child;
sl@0
   621
		}
sl@0
   622
	*hole=aValue;
sl@0
   623
	return top;
sl@0
   624
	}
sl@0
   625
sl@0
   626
sl@0
   627
sl@0
   628
// Class CPermanentStoreCollector
sl@0
   629
sl@0
   630
CPermanentStoreCollector* CPermanentStoreCollector::CompactorL(CPermanentStoreCoord& aCoord)
sl@0
   631
	{
sl@0
   632
	CPermanentStoreCollector* self=ReclaimerL(aCoord);
sl@0
   633
	CleanupStack::PushL(self);
sl@0
   634
	self->iReloc=new(ELeave) TPermanentStoreRelocator;
sl@0
   635
	CleanupStack::Pop();
sl@0
   636
	return self;
sl@0
   637
	}
sl@0
   638
sl@0
   639
CPermanentStoreCollector* CPermanentStoreCollector::ReclaimerL(CPermanentStoreCoord& aCoord)
sl@0
   640
	{
sl@0
   641
	return new(ELeave) CPermanentStoreCollector(aCoord);
sl@0
   642
	}
sl@0
   643
sl@0
   644
CPermanentStoreCollector::CPermanentStoreCollector(CPermanentStoreCoord& aCoord)
sl@0
   645
	: iCoord(&aCoord),iHost(&aCoord.Host()),iStreams(EGranularity,_FOFF(TEntry,entry.ref))
sl@0
   646
	{
sl@0
   647
	Coord().Inc();
sl@0
   648
	}
sl@0
   649
sl@0
   650
CPermanentStoreCollector::~CPermanentStoreCollector()
sl@0
   651
	{
sl@0
   652
	delete iReloc;
sl@0
   653
	iStreams.Close();
sl@0
   654
	iMark.Withdraw(Host());
sl@0
   655
	Coord().Dec();
sl@0
   656
	}
sl@0
   657
sl@0
   658
void CPermanentStoreCollector::DoRelease()
sl@0
   659
	{
sl@0
   660
	delete this;
sl@0
   661
	}
sl@0
   662
sl@0
   663
void CPermanentStoreCollector::DoResetL(TInt& aCount)
sl@0
   664
	{
sl@0
   665
	iMark.Withdraw(Host());
sl@0
   666
	iMark=KStreamBeginning;
sl@0
   667
	iCoordGen=Coord().Generation();
sl@0
   668
	iFree=0;
sl@0
   669
	TRAPD(r, aCount = FastResetL());
sl@0
   670
	if (r == KErrNone)
sl@0
   671
		return;
sl@0
   672
	if (r != KErrNoMemory)
sl@0
   673
		User::Leave(r);
sl@0
   674
// fall back to fixed memory solution
sl@0
   675
	iState=EGetFree;
sl@0
   676
	if (Compacting())
sl@0
   677
		iReloc->Reset();
sl@0
   678
	iIter.Reset();
sl@0
   679
	TInt streams=iIter.FillL(Coord().ConsolidateL());
sl@0
   680
	aCount=streams+1;
sl@0
   681
	}
sl@0
   682
sl@0
   683
const TInt KMaxStepEffort=9;
sl@0
   684
sl@0
   685
void CPermanentStoreCollector::DoNextL(TInt& aStep,TInt& aTotal)
sl@0
   686
//
sl@0
   687
// Dispatch the next set of operations
sl@0
   688
//
sl@0
   689
	{
sl@0
   690
	if (aStep<=0)
sl@0
   691
		{
sl@0
   692
		__DEBUG(Panic(TStorePanic()));
sl@0
   693
		return;
sl@0
   694
		}
sl@0
   695
//
sl@0
   696
	if (iCoordGen!=Coord().Generation() || Coord().TableL().IsVirtual())
sl@0
   697
		__LEAVE(KErrNotReady);
sl@0
   698
//
sl@0
   699
	TInt effort=0;
sl@0
   700
	do
sl@0
   701
		{
sl@0
   702
		switch (iState)
sl@0
   703
			{
sl@0
   704
		default:
sl@0
   705
			__DEBUG(User::Invariant());
sl@0
   706
		case EGetFree:
sl@0
   707
			effort+=GetFreeL();
sl@0
   708
			break;
sl@0
   709
		case ESkip:
sl@0
   710
			effort+=SkipL(aTotal);
sl@0
   711
			--aStep;
sl@0
   712
			break;
sl@0
   713
		case EInitRelocator:
sl@0
   714
			effort+=InitRelocator();
sl@0
   715
			break;
sl@0
   716
		case EFillRelocator:
sl@0
   717
			effort+=FillRelocatorL();
sl@0
   718
			break;
sl@0
   719
		case EEvalRelocator:
sl@0
   720
			effort+=EvalRelocatorL();
sl@0
   721
			break;
sl@0
   722
		case EScanRelocator:
sl@0
   723
			effort+=ScanRelocator();
sl@0
   724
			break;
sl@0
   725
		case ERelocateStream:
sl@0
   726
			effort+=RelocateStreamL();
sl@0
   727
			--aStep;
sl@0
   728
			break;
sl@0
   729
		case ERelocateToc:
sl@0
   730
			RelocateTocL(aTotal);
sl@0
   731
			iStreams.Close();
sl@0
   732
			__ASSERT_DEBUG(aStep==1,User::Invariant());
sl@0
   733
			aStep=0;
sl@0
   734
			return;
sl@0
   735
		case EFastSort:
sl@0
   736
			FastSort();
sl@0
   737
			--aStep;
sl@0
   738
			return;
sl@0
   739
		case EFastExtent:
sl@0
   740
			FastExtentL(aTotal);
sl@0
   741
			--aStep;
sl@0
   742
			return;
sl@0
   743
		case EFastRelocate:
sl@0
   744
			FastRelocateL(aTotal);
sl@0
   745
			--aStep;
sl@0
   746
			return;
sl@0
   747
			}
sl@0
   748
		} while(effort<KMaxStepEffort);
sl@0
   749
	__ASSERT_DEBUG(aStep>=1,User::Invariant());
sl@0
   750
	}
sl@0
   751
sl@0
   752
TInt CPermanentStoreCollector::GetFreeL()
sl@0
   753
//
sl@0
   754
// find the end of the free space
sl@0
   755
//
sl@0
   756
	{
sl@0
   757
	__ASSERT_DEBUG(iState==EGetFree,User::Invariant());
sl@0
   758
	__ASSERT_DEBUG(iFree>=0,User::Invariant());
sl@0
   759
//
sl@0
   760
	TInt effort=0;
sl@0
   761
	iEnd=iIter.Next();
sl@0
   762
	if (iEnd<0)
sl@0
   763
		{
sl@0
   764
		if (iIter.FillL(Coord().ConsolidateL())==0)
sl@0
   765
			{
sl@0
   766
			iState=ERelocateToc;	// no more streams
sl@0
   767
			return effort;
sl@0
   768
			}
sl@0
   769
		effort=KMaxStepEffort;
sl@0
   770
		iEnd=iIter.Next();
sl@0
   771
		__ASSERT_DEBUG(iEnd>=0,User::Invariant());
sl@0
   772
		}
sl@0
   773
	iState=Compacting() && HaveEnoughSpace() ? EInitRelocator : ESkip;
sl@0
   774
	return effort;
sl@0
   775
	}
sl@0
   776
sl@0
   777
TInt CPermanentStoreCollector::SkipL(TInt& aTotal)
sl@0
   778
//
sl@0
   779
// Find some free space, iEnd was the last stream extracted from concat
sl@0
   780
//
sl@0
   781
	{
sl@0
   782
	__ASSERT_DEBUG(iState==ESkip,User::Invariant());
sl@0
   783
	__ASSERT_DEBUG(iFree>=0&&iFree<=iEnd,User::Invariant());
sl@0
   784
//
sl@0
   785
	aTotal+=iEnd-iFree;
sl@0
   786
	iFree=SkipLink(ExtentL(iEnd));
sl@0
   787
	iState=EGetFree;
sl@0
   788
	return 1;		// effort
sl@0
   789
	}
sl@0
   790
sl@0
   791
TInt CPermanentStoreCollector::InitRelocator()
sl@0
   792
//
sl@0
   793
// initialise the relocator for the free space
sl@0
   794
//
sl@0
   795
	{
sl@0
   796
	__ASSERT_DEBUG(iState==EInitRelocator,User::Invariant());
sl@0
   797
	__ASSERT_DEBUG(Compacting(),User::Invariant());
sl@0
   798
//
sl@0
   799
	iState=iReloc->Reset(iEnd) ? EScanRelocator : EFillRelocator;
sl@0
   800
	return 0;		// no real work at all
sl@0
   801
	}
sl@0
   802
sl@0
   803
TInt CPermanentStoreCollector::FillRelocatorL()
sl@0
   804
//
sl@0
   805
// Fill the relocator table
sl@0
   806
//
sl@0
   807
	{
sl@0
   808
	__ASSERT_DEBUG(iState==EFillRelocator,User::Invariant());
sl@0
   809
	__ASSERT_DEBUG(iFree>=0&&iFree<iEnd,User::Invariant());
sl@0
   810
	__ASSERT_DEBUG(Compacting(),User::Invariant());
sl@0
   811
//
sl@0
   812
	TBool more=iReloc->FillL(Coord().ConsolidateL());
sl@0
   813
	iState=more ? EEvalRelocator : ESkip;
sl@0
   814
	return more ? KMaxStepEffort : 0;
sl@0
   815
	}
sl@0
   816
sl@0
   817
TInt CPermanentStoreCollector::EvalRelocatorL()
sl@0
   818
//
sl@0
   819
// evaluate the extents for the relocator
sl@0
   820
//
sl@0
   821
	{
sl@0
   822
	__ASSERT_DEBUG(iState==EEvalRelocator,User::Invariant());
sl@0
   823
	__ASSERT_DEBUG(Compacting(),User::Invariant());
sl@0
   824
//
sl@0
   825
	iReloc->EvaluateLengthsL(Host(),iMark,Coord().Base());
sl@0
   826
	iState=EScanRelocator;
sl@0
   827
	return KMaxStepEffort;
sl@0
   828
	}
sl@0
   829
sl@0
   830
TInt CPermanentStoreCollector::ScanRelocator()
sl@0
   831
//
sl@0
   832
// find a stream that will fit
sl@0
   833
//
sl@0
   834
	{
sl@0
   835
	__ASSERT_DEBUG(iState==EScanRelocator,User::Invariant());
sl@0
   836
	__ASSERT_DEBUG(iFree>=0&&iFree<iEnd,User::Invariant());
sl@0
   837
	__ASSERT_DEBUG(Compacting(),User::Invariant());
sl@0
   838
//
sl@0
   839
	iState=iReloc->FindFit(iFree,iEnd) ? ERelocateStream : EFillRelocator;
sl@0
   840
	return 0;
sl@0
   841
	}
sl@0
   842
sl@0
   843
TInt CPermanentStoreCollector::RelocateStreamL()
sl@0
   844
//
sl@0
   845
// find and relocate a stream
sl@0
   846
//
sl@0
   847
	{
sl@0
   848
	__ASSERT_DEBUG(iState==ERelocateStream,User::Invariant());
sl@0
   849
	__ASSERT_DEBUG(iFree>=0&&iFree<iEnd,User::Invariant());
sl@0
   850
	__ASSERT_DEBUG(Compacting(),User::Invariant());
sl@0
   851
//
sl@0
   852
	const TEntry& e = *iReloc->Current();
sl@0
   853
	RelocateStreamL(e, iEnd);
sl@0
   854
	iState=e.entry.ref==iEnd ? EGetFree : HaveEnoughSpace() ? EScanRelocator : ESkip;
sl@0
   855
	iIter.Relocated(e.entry.ref);
sl@0
   856
	iReloc->Relocated(&e);
sl@0
   857
	return KMaxStepEffort;
sl@0
   858
	}
sl@0
   859
sl@0
   860
void CPermanentStoreCollector::RelocateTocL(TInt& aTotal)
sl@0
   861
//
sl@0
   862
// relocate the toc - return any wasted bytes
sl@0
   863
//
sl@0
   864
	{
sl@0
   865
	__ASSERT_DEBUG(iState==ERelocateToc,User::Invariant());
sl@0
   866
	__ASSERT_DEBUG(iFree>=0,User::Invariant());
sl@0
   867
//
sl@0
   868
	TInt toc=Coord().iToc+KOffsetTocHeader;
sl@0
   869
	if (toc<0)
sl@0
   870
		return;
sl@0
   871
//
sl@0
   872
	if (Compacting())
sl@0
   873
		{
sl@0
   874
		TInt len=Coord().TableL().Extent()-toc;
sl@0
   875
		if (Fits(iFree,toc,len))
sl@0
   876
			{
sl@0
   877
			RelocateL(toc, len, EFrameDescriptive16, toc);
sl@0
   878
			Coord().MoveL(iFree-KOffsetTocHeader,iFree + len);
sl@0
   879
			return;
sl@0
   880
			}
sl@0
   881
		}
sl@0
   882
	// don't move it
sl@0
   883
	aTotal += toc-iFree;
sl@0
   884
	}
sl@0
   885
sl@0
   886
TBool CPermanentStoreCollector::HaveEnoughSpace() const
sl@0
   887
	{
sl@0
   888
	__ASSERT_DEBUG(Compacting(),User::Invariant());
sl@0
   889
//
sl@0
   890
	return SpaceFor(iFree,iEnd,iReloc->MinLength());
sl@0
   891
	}
sl@0
   892
sl@0
   893
TInt CPermanentStoreCollector::ExtentL(TInt aStream)
sl@0
   894
//
sl@0
   895
// Check if the relocator knows before scanning the file
sl@0
   896
//
sl@0
   897
	{
sl@0
   898
	if (Compacting())
sl@0
   899
		{
sl@0
   900
		TInt ext=iReloc->Extent(aStream);
sl@0
   901
		if (ext>=0)
sl@0
   902
			return ext;
sl@0
   903
		}
sl@0
   904
	return ::ExtentL(Host(),iMark,Coord().Base(),aStream);
sl@0
   905
	}
sl@0
   906
sl@0
   907
/* relocate a stream into [iFree, aExtent)
sl@0
   908
sl@0
   909
During compaction, for each string which is to be moved from position A1 to B1, the sequence of operations is:
sl@0
   910
sl@0
   911
1.  Copy stream S1 content from position A1 to position B1 . The copy never overlaps so the old stream content is still good at this point.
sl@0
   912
2.  Optionally rewrite the file header to state that stream S1 is being relocated to B1 (more about the ‘optional below’)
sl@0
   913
3.  Overwrite the TOC entry for S1 to state that the content is now at B1
sl@0
   914
sl@0
   915
This function completes 3 steps above and will be called again and again for every string to be moved.
sl@0
   916
sl@0
   917
In terms of data consistency, first consider the impact of a mid-write failure in any of these steps (when write caching is disabled):
sl@0
   918
1.  If step #1 only partially completes the file is good as the original content is intact and the new content was being written to otherwise free space
sl@0
   919
2.  If step #2 only partially completes the header CRC fails and only the TOC reference is considered valid (so the corrupt stream relocation record is ignored).
sl@0
   920
		 The TOC will be good because it is being overwritten with the same content.
sl@0
   921
3.  If step #3 only partially completes the entry for S1 in the TOC is corrupt, BUT the relocation record for S1 in the file header is good and will
sl@0
   922
 override the entry in the TOC.
sl@0
   923
sl@0
   924
In all cases the file is never broken by a crash in mid-compaction.
sl@0
   925
sl@0
   926
Step #2 is optional – there are many cases when step #3 cannot fail ‘halfway through’ because the underlying media makes atomic block/page based
sl@0
   927
updates and the write does not cross any block boundaries. In STORE we assume that blocks cannot be smaller than 512 bytes and any flash based
sl@0
   928
media provides the required behavior. Thus 99% of the step #2 writes are eliminated.
sl@0
   929
sl@0
   930
Note that sequencing MATTERS even for just one stream. If the TOC update hits the disk before the content is moved, and then the device fails
sl@0
   931
we will have a broken file: S1 points to B1 which contains garbage.  Equally in the case where step #2 is required (i.e. when step #3 straddles
sl@0
   932
a block boundary and could fail) step 2 has to go before the step 3. Otherwise write #3 could go to disk and fail part way through before write #2 
sl@0
   933
and leave the TOC corrupt with no recovery in the file header.
sl@0
   934
sl@0
   935
Consider the case that step 2 was omitted, so the Store relies on step 3 being completed in order to know that S1 is in location B1;
sl@0
   936
and that no flush is done after step 3. In step 4 the stream S2 is moved – at this point the old space for stream S1 at A1 is considered empty
sl@0
   937
– and suppose it gets moved from A2 to B2 where B2 overlaps/overwrites A1. If the writes in step 3 and step 4 are re-ordered and the step 3
sl@0
   938
write does not happen – then the TOC will claim that S1 is still at A1 but this location in the file has been overwritten with data from S2.
sl@0
   939
A corrupted file.
sl@0
   940
sl@0
   941
Based on the knowledge above, it is strongly recommended to set EFileWriteDirectIO bit when opening the file so that the order is maintained
sl@0
   942
when writing to the file.
sl@0
   943
*/
sl@0
   944
void CPermanentStoreCollector::RelocateStreamL(const CPermanentStoreCollector::TEntry& aReloc, TInt aExtent)
sl@0
   945
	{
sl@0
   946
	if (Coord().Accessed())	// must have exclusive access to relocate the stream
sl@0
   947
		__LEAVE(KErrInUse);
sl@0
   948
//
sl@0
   949
	TInt end=RelocateL(aReloc.entry.ref,aReloc.len,aReloc.entry.handle == KHandleTocBase ? EFrameDescriptive16 : EFrameData16, aExtent);
sl@0
   950
	//Step 1, 4,....
sl@0
   951
	Coord().RelocateL(aReloc.entry.handle, iFree);
sl@0
   952
	// Step 2 & 3, 5 & 6,...
sl@0
   953
	iCoordGen=Coord().Generation();	// changed by relocation
sl@0
   954
	iFree = end;
sl@0
   955
	}
sl@0
   956
sl@0
   957
TInt CPermanentStoreCollector::RelocateL(TInt aStream, TInt aLength, TFrameType16 aType, TInt aExtent)
sl@0
   958
//
sl@0
   959
// Relocate a data stream into [iFree, aExtent)
sl@0
   960
//
sl@0
   961
	{
sl@0
   962
	__ASSERT_DEBUG(Fits(iFree,aExtent,aLength),User::Invariant());
sl@0
   963
	__ASSERT_DEBUG(Compacting(),User::Invariant());
sl@0
   964
//
sl@0
   965
	TInt end=SkipLink(iFree+aLength);
sl@0
   966
	TInt terminator;
sl@0
   967
	if (end==aExtent)
sl@0
   968
		terminator=-1;
sl@0
   969
	else
sl@0
   970
		{
sl@0
   971
		TInt anchor=((end>>KShiftFrameLength16)+1)<<KShiftFrameLength16;
sl@0
   972
		if (aExtent<anchor)
sl@0
   973
			{
sl@0
   974
			__ASSERT_DEBUG(aExtent-KSizeFrameDes16-end>0,User::Invariant());
sl@0
   975
			terminator=EFrameFree16+(aExtent-KSizeFrameDes16-end);
sl@0
   976
			}
sl@0
   977
		else
sl@0
   978
			terminator=EFrameFree16+KFrameOpen16;
sl@0
   979
		}
sl@0
   980
//
sl@0
   981
	RFrame16Buf from(Coord().Base());
sl@0
   982
	from.Set(Host(),aStream,aStream+aLength,MStreamBuf::ERead);
sl@0
   983
	from.PushL();
sl@0
   984
	TRelocatorInput input(Host(),iMark);
sl@0
   985
	input.OpenL(aType,Coord().Base(),iFree,aLength,terminator);
sl@0
   986
	from.ReadL(input,aLength);
sl@0
   987
	input.CommitL();
sl@0
   988
	CleanupStack::PopAndDestroy();
sl@0
   989
//
sl@0
   990
	return end;
sl@0
   991
	}
sl@0
   992
sl@0
   993
sl@0
   994
TInt CPermanentStoreCollector::FastResetL()
sl@0
   995
//
sl@0
   996
// Fill the table with the streams with data in the store
sl@0
   997
//
sl@0
   998
	{
sl@0
   999
	iStreams.Reset();
sl@0
  1000
//
sl@0
  1001
	CleanupClosePushL(iStreams);
sl@0
  1002
	RPermanentStoreTocIter iter(Coord().ConsolidateL());
sl@0
  1003
	CleanupReleasePushL(iter);
sl@0
  1004
	TEntry entry;
sl@0
  1005
	for (iter.ResetL();iter.NextL(entry.entry);)
sl@0
  1006
		{
sl@0
  1007
		if (entry.entry.handle<0)
sl@0
  1008
			continue;
sl@0
  1009
		if (entry.entry.ref<0)
sl@0
  1010
			continue;
sl@0
  1011
		User::LeaveIfError(iStreams.Append(entry));
sl@0
  1012
		}
sl@0
  1013
	CleanupStack::PopAndDestroy(&iter);
sl@0
  1014
	CleanupStack::Pop(&iStreams);
sl@0
  1015
//
sl@0
  1016
	// always have final (toc) step
sl@0
  1017
	iState = ERelocateToc;
sl@0
  1018
	TInt streams = iStreams.Count();
sl@0
  1019
	if (streams == 0)
sl@0
  1020
		return 1;
sl@0
  1021
	iState = EFastSort;
sl@0
  1022
	// ordering is 1 step and evaluating the extents is several more
sl@0
  1023
	TInt count = 2 + (streams + EExtentStep - 1)/EExtentStep;
sl@0
  1024
	if (Compacting())
sl@0
  1025
		count += streams;
sl@0
  1026
	return count;
sl@0
  1027
	}
sl@0
  1028
sl@0
  1029
void CPermanentStoreCollector::FastSort()
sl@0
  1030
	{
sl@0
  1031
	__ASSERT_DEBUG(iState == EFastSort, User::Invariant());
sl@0
  1032
//
sl@0
  1033
	iStreams.SortSigned();
sl@0
  1034
	iNext = &iStreams[0];
sl@0
  1035
	iLast = iNext + iStreams.Count();
sl@0
  1036
	iState = EFastExtent;
sl@0
  1037
	}
sl@0
  1038
sl@0
  1039
void CPermanentStoreCollector::FastExtentL(TInt& aTotal)
sl@0
  1040
//
sl@0
  1041
// Evaluate the lengths for all the streams
sl@0
  1042
// if reclaiming, update aTotal with free space skipped
sl@0
  1043
// return false until we've done the last one
sl@0
  1044
//
sl@0
  1045
	{
sl@0
  1046
	__ASSERT_DEBUG(iState == EFastExtent, User::Invariant());
sl@0
  1047
	__ASSERT_DEBUG(iNext != iLast, User::Invariant());
sl@0
  1048
sl@0
  1049
	TEntry* e = iNext;
sl@0
  1050
	const TEntry* end = Min(iLast, e + EExtentStep);
sl@0
  1051
	do
sl@0
  1052
		{
sl@0
  1053
		TInt ref = e->entry.ref;
sl@0
  1054
		__ASSERT_DEBUG(TUint(iFree)<=TUint(ref),User::Invariant());
sl@0
  1055
		TInt ext = ::ExtentL(Host(), iMark, Coord().Base(), ref);
sl@0
  1056
		e->len = ext - ref;
sl@0
  1057
		if (!Compacting())
sl@0
  1058
			aTotal += ref - iFree;
sl@0
  1059
		iFree = SkipLink(ext);
sl@0
  1060
		} while (++e < end);
sl@0
  1061
	iNext = e;
sl@0
  1062
//
sl@0
  1063
	if (e == iLast)
sl@0
  1064
		{
sl@0
  1065
		if (!Compacting())
sl@0
  1066
			iState = ERelocateToc;
sl@0
  1067
		else
sl@0
  1068
			{
sl@0
  1069
			iNext = &iStreams[0];
sl@0
  1070
			iFree = 0;
sl@0
  1071
			iState = EFastRelocate;
sl@0
  1072
			}
sl@0
  1073
		}
sl@0
  1074
	}
sl@0
  1075
sl@0
  1076
CPermanentStoreCollector::TEntry* CPermanentStoreCollector::BestFit(TInt aPos, TInt aExt, TEntry* aFirst, TEntry* aLast)
sl@0
  1077
//
sl@0
  1078
// [aPos, aExt) is a hole - find the best fit to fill it in [aFirst, aLast)
sl@0
  1079
//
sl@0
  1080
	{
sl@0
  1081
	__ASSERT_DEBUG(aPos <= aExt, User::Invariant());
sl@0
  1082
	TEntry* r = 0;
sl@0
  1083
	if (aPos == aExt)
sl@0
  1084
		return r;
sl@0
  1085
//
sl@0
  1086
	if ((aExt & KMaskFrameLength16) != 0)
sl@0
  1087
		aExt -= KSizeFrameDes16;
sl@0
  1088
	const TInt mingap = Min(KSizeFrameDes16 + 1, (aExt & KMaskFrameLength16));
sl@0
  1089
	TInt rlen = 0;
sl@0
  1090
	TInt avail = aExt - aPos;
sl@0
  1091
	do
sl@0
  1092
		{
sl@0
  1093
		TInt len;
sl@0
  1094
		for (;;)
sl@0
  1095
			{
sl@0
  1096
			len = (--aLast)->len;
sl@0
  1097
			if (len > rlen)
sl@0
  1098
				break;
sl@0
  1099
			if (aFirst == aLast)
sl@0
  1100
				return r;
sl@0
  1101
			}
sl@0
  1102
		TInt d = avail - len;
sl@0
  1103
		if (d < 0)
sl@0
  1104
			continue;
sl@0
  1105
		if (d == 0)			// exact fit
sl@0
  1106
			return aLast;
sl@0
  1107
		if (d < mingap)
sl@0
  1108
			continue;
sl@0
  1109
		r = aLast;
sl@0
  1110
		rlen = len;
sl@0
  1111
		} while (aFirst != aLast);
sl@0
  1112
	return r;
sl@0
  1113
	}
sl@0
  1114
sl@0
  1115
void CPermanentStoreCollector::FastRelocateL(TInt& aTotal)
sl@0
  1116
//
sl@0
  1117
// Look for a stream to move. Either move a stream or fail to find a fit before returning
sl@0
  1118
// fill holes from the front of the file with the biggest block that will fit (inverted current algorithm)
sl@0
  1119
// return true when the last hole has been looked at
sl@0
  1120
//
sl@0
  1121
	{
sl@0
  1122
	__ASSERT_DEBUG(iState == EFastRelocate, User::Invariant());
sl@0
  1123
	__ASSERT_DEBUG(iNext != iLast, User::Invariant());
sl@0
  1124
sl@0
  1125
	TEntry* e = iNext;
sl@0
  1126
	TInt ext = e->entry.ref;
sl@0
  1127
	__ASSERT_DEBUG(ext >= 0, User::Invariant());
sl@0
  1128
sl@0
  1129
	TEntry* r = BestFit(iFree, ext, e, iLast);
sl@0
  1130
	if (!r)
sl@0
  1131
		{
sl@0
  1132
		// Nothing fits, accumulate free space
sl@0
  1133
		aTotal += ext - iFree;
sl@0
  1134
		iFree = SkipLink(ext + e->len);
sl@0
  1135
		}
sl@0
  1136
	else
sl@0
  1137
		{
sl@0
  1138
		// lets move it
sl@0
  1139
		RelocateStreamL(*r, ext);
sl@0
  1140
		if (r != e)
sl@0
  1141
			{
sl@0
  1142
			// relocated a stream other than the one terminating the current hole
sl@0
  1143
			// mark it at moved
sl@0
  1144
			r->entry.ref = -1;
sl@0
  1145
			r->len = -1;
sl@0
  1146
			return;
sl@0
  1147
			}
sl@0
  1148
		}
sl@0
  1149
	// skip through any relocated streams
sl@0
  1150
	while (++e < iLast && e->entry.ref < 0)
sl@0
  1151
		;
sl@0
  1152
	iNext = e;
sl@0
  1153
	if (e == iLast)
sl@0
  1154
		iState = ERelocateToc;
sl@0
  1155
	}