Update contrib.
1 // Copyright (c) 1998-2009 Nokia Corporation and/or its subsidiary(-ies).
2 // All rights reserved.
3 // This component and the accompanying materials are made available
4 // under the terms of "Eclipse Public License v1.0"
5 // which accompanies this distribution, and is available
6 // at the URL "http://www.eclipse.org/legal/epl-v10.html".
8 // Initial Contributors:
9 // Nokia Corporation - initial contribution.
18 // PFS frame-related utilities
20 TInt SkipLink(TInt anExtent)
22 // anExtent is the end-of-stream, where is the next link pos?
23 // : add the size of the next frame link, or skip past anchor link
25 {return anExtent+Min(KSizeFrameDes16,(-anExtent)&KMaskFrameLength16);}
27 inline TBool SpaceFor(TInt aSpace,TInt anEnd,TInt aLength)
28 // Check if there is space for at least aLength between links at aSpace and anEnd
29 {return SkipLink(aSpace+aLength)<=anEnd;}
31 TBool Fits(TInt aSpace,TInt anEnd,TInt aLength)
33 // Check that aLength can be relocated into the space
34 // either an exact fit, or leaves space for a free frame of at least one byte
37 TInt end=SkipLink(aSpace+aLength);
38 return end==anEnd ? ETrue : SpaceFor(end,anEnd,1);
41 TInt ExtentL(TStreamExchange& aHost,TStreamMark& aMark,TStreamPos aBase,TInt aStream)
43 // scan a stream to discover it's extent
46 __ASSERT_DEBUG(aMark.RelatesTo(aHost),User::Invariant());
47 __ASSERT_DEBUG(aBase>=KStreamBeginning,User::Invariant());
48 __ASSERT_DEBUG(aStream>=0,User::Invariant());
50 aMark.SeekL(aHost,RFrame16Buf::Position(aBase,aStream)-KSizeFrameDes16);
52 aMark.ReadL(aHost,&des,KSizeFrameDes16);
54 if ((frame&KMaskFrameType16)!=EFrameData16)
55 if ((frame&KMaskFrameType16)!=EFrameDescriptive16)
57 aMark.Withdraw(aHost);
61 TInt anchor=((aStream>>KShiftFrameLength16)+1)<<KShiftFrameLength16;
64 frame&=KMaskFrameLength16;
65 TInt lim=anchor-aStream;
66 if (frame!=KFrameOpen16)
70 aMark.Withdraw(aHost);
75 aMark.SeekL(aHost,EStreamMark,lim);
77 anchor+=KFrameFullLength16;
78 aMark.ReadL(aHost,&des,KSizeFrameDes16);
80 } while ((frame&KMaskFrameType16)==EFrameContinuation16);
86 // Class TRelocatorInput
87 // Used to transfer a frame-set within the file
89 const TInt KRelocatorBufferSize=0xc00;
91 NONSHARABLE_CLASS(TRelocatorInput) : public MStreamInput
94 inline TRelocatorInput(TStreamExchange& aHost,TStreamMark& aMark);
95 void OpenL(TFrameType16 aType,TStreamPos aBase,TInt aPos,TInt aLength,TInt aTerminator);
98 TInt PushL(const TAny* aPtr,TInt aMaxLength);
99 TStreamTransfer ReadFromL(MStreamBuf& aSource,TStreamTransfer aTransfer);
100 void DesL(TFrameType16 aType);
102 inline TStreamExchange& Host() const;
103 inline TStreamMark& Mark();
105 TStreamExchange* iHost;
113 inline TRelocatorInput::TRelocatorInput(TStreamExchange& aHost,TStreamMark& aMark)
114 : iHost(&aHost),iMark(&aMark)
116 inline TStreamExchange& TRelocatorInput::Host() const
118 __ASSERT_DEBUG(iHost!=NULL,User::Invariant());
121 inline TStreamMark& TRelocatorInput::Mark()
123 __ASSERT_DEBUG(iMark!=NULL,User::Invariant());
127 void TRelocatorInput::OpenL(TFrameType16 aType,TStreamPos aBase,TInt aPos,TInt aLength,TInt aTerminator)
129 // initiate the relocated stream
132 __ASSERT_DEBUG(aType!=EFrameFree16,User::Invariant());
133 __ASSERT_DEBUG(aBase>=KStreamBeginning,User::Invariant());
134 __ASSERT_DEBUG(aPos>=0,User::Invariant());
135 __ASSERT_DEBUG(aLength>0,User::Invariant());
139 iTerminator=aTerminator;
140 iAnchor=KFrameFullLength16-(aPos&KMaskFrameLength16);
141 Mark().SeekL(Host(),RFrame16Buf::Position(aBase,aPos)-KSizeFrameDes16);
145 void TRelocatorInput::CommitL()
147 // complete the relocated stream
150 __ASSERT_DEBUG(iRemain==0,User::Invariant());
151 __ASSERT_DEBUG(iAnchor>=0,User::Invariant());
157 des[1]=TFrameDes16(iTerminator);
158 TInt l=KSizeFrameDes16;
159 if (iAnchor<=KSizeFrameDes16)
161 Mark().WriteL(Host(),(const TUint8*)&des[2]-l,l);
164 TInt TRelocatorInput::PushL(const TAny*,TInt)
166 // use a passive write through to the host
172 void TRelocatorInput::DesL(TFrameType16 aType)
174 // Write the next frame descriptor
177 __ASSERT_DEBUG(aType!=EFrameFree16,User::Invariant());
178 TFrameDes16 des=TFrameDes16(iRemain<iAnchor ? aType+iRemain : aType+KFrameOpen16);
179 Mark().WriteL(Host(),&des,KSizeFrameDes16);
182 TStreamTransfer TRelocatorInput::ReadFromL(MStreamBuf& aSource,TStreamTransfer aTransfer)
184 // bulk of the transfer happens here
187 __ASSERT_DEBUG(!(aTransfer>iRemain),User::Invariant());
190 TUint8 buf[KRelocatorBufferSize];
191 TInt len=aSource.ReadL(buf,aTransfer[sizeof(buf)]);
196 if (iType!=EFrameFree16)
204 { // write next anchor
205 iAnchor=KFrameFullLength16;
206 DesL(EFrameContinuation16);
208 TInt frame=Min(len,iAnchor);
209 Mark().WriteL(Host(),p,frame);
217 } while (aTransfer>0);
223 // Class TPermanentStoreRelocator
224 // used to locate streams for relocation in limited memory
226 class TPermanentStoreRelocator
228 #if defined(__SMALL_BUNDLE)
229 enum {EBundleSize=8-1};
231 enum {EBundleSize=64-1};
234 typedef CPermanentStoreCollector::TEntry TEntry;
237 TBool Reset(TInt aPos);
238 TInt FillL(CPermanentStoreToc& aToc);
239 void EvaluateLengthsL(TStreamExchange& aHost,TStreamMark& aMark,TStreamPos aBase);
241 TBool FindFit(TInt aSpace,TInt anEnd);
242 inline const TEntry* Current() const;
243 void Relocated(const TEntry* anEntry);
245 TInt Extent(TInt aStream) const;
246 inline TInt MinLength() const;
248 static void Push(TEntry* aHeap,TEntry* aHole,const TEntry& aValue);
249 static TEntry PopPush(TEntry* aHeap,TEntry* anEnd,const TEntry& aValue);
252 const TEntry* iFinish;
255 TInt iCurrentMin,iMinLength;
256 TEntry iTable[EBundleSize];
259 inline const TPermanentStoreRelocator::TEntry* TPermanentStoreRelocator::Current() const
260 {__ASSERT_DEBUG(iNext>=iTable&&iNext<iFinish,User::Invariant());return iNext;}
261 inline TInt TPermanentStoreRelocator::MinLength() const
264 void TPermanentStoreRelocator::Reset()
266 // reset the cached length values
273 TBool TPermanentStoreRelocator::Reset(TInt aPos)
275 // reset the iterator for a new scan from aPos
278 __ASSERT_DEBUG(aPos>=0,User::Invariant());
280 iCurrentMin=KMaxTInt;
282 if (aPos>=iPos || e==iFinish || aPos<e->entry.ref)
290 // can use current data
291 for (;e->entry.ref!=aPos;++e)
293 __ASSERT_DEBUG(e->entry.ref<aPos,User::Invariant());
294 __ASSERT_DEBUG(e<iFinish,User::Invariant());
296 __ASSERT_DEBUG(e->entry.handle>=0,User::Invariant());
301 TBool TPermanentStoreRelocator::FillL(CPermanentStoreToc& aToc)
303 // Fill the table with the next bundle of stream offsets
304 // return if there are more available
307 __ASSERT_DEBUG(iNext==iFinish,User::Invariant());
311 __ASSERT_DEBUG(iPos>=0,User::Invariant());
313 RPermanentStoreTocIter iter(aToc);
314 CleanupReleasePushL(iter);
318 TEntry* table=iTable;
320 TEntry* end=table+EBundleSize;
322 __DEBUG(entry.len=-1);
323 for (iter.ResetL();iter.NextL(entry.entry);)
325 if (entry.entry.handle<0)
327 TInt off=entry.entry.ref;
332 Push(table,last++,entry); // push onto the bottom
333 else if (off<table->entry.ref)
334 PopPush(table,last,entry); // replace item in heap
336 CleanupStack::PopAndDestroy();
338 iMore=streams+table-last;
340 iPos=table->entry.ref+1; // largest offset in table
344 *last=PopPush(table,last,*last);
348 void TPermanentStoreRelocator::Push(TEntry* aHeap,TEntry* aHole,const TEntry& aValue)
350 // push aValue onto the heap (which will grow)
353 TInt ix=aHole+1-aHeap;
356 TInt link=ix-(ix>>1);
358 TEntry* parent=aHole-link;
359 if (parent->entry.ref>=aValue.entry.ref)
367 TPermanentStoreRelocator::TEntry TPermanentStoreRelocator::PopPush(TEntry* aHeap,TEntry* anEnd,const TEntry& aValue)
369 // pop the top and push aValue
378 TEntry* child=hole+ix;
382 if (child<anEnd && (child+1)->entry.ref>child->entry.ref)
387 if (child->entry.ref<=aValue.entry.ref)
396 void TPermanentStoreRelocator::EvaluateLengthsL(TStreamExchange& aHost,TStreamMark& aMark,TStreamPos aBase)
398 // Evaluate the lengths for all the entries in the table
401 const TEntry* end=iFinish;
402 for (TEntry* e=iTable;e<end;++e)
404 __ASSERT_DEBUG(e->entry.handle>=0,User::Invariant());
405 __ASSERT_DEBUG(e->entry.ref>=0,User::Invariant());
406 __ASSERT_DEBUG(e->len==-1,User::Invariant());
407 TInt pos=e->entry.ref;
408 e->len=ExtentL(aHost,aMark,aBase,pos)-pos;
412 TBool TPermanentStoreRelocator::FindFit(TInt aSpace,TInt anEnd)
414 // Find a stream which will fit into the space
417 const TEntry* end=iFinish;
419 for (e=iNext;e<end;++e)
421 if (e->entry.handle<0)
424 __ASSERT_DEBUG(len>0,User::Invariant());
425 if (Fits(aSpace,anEnd,len))
430 // len has been left behind, check for minimum
434 // if no more data, use current min as cached value
436 iMinLength=iCurrentMin;
441 void TPermanentStoreRelocator::Relocated(const TEntry* anEntry)
443 // Relocation has been successful
446 __ASSERT_DEBUG(anEntry==iNext,User::Invariant());
447 TEntry* e=CONST_CAST(TEntry*,anEntry);
452 TInt TPermanentStoreRelocator::Extent(TInt aStream) const
454 // Return this stream extent if we know it
457 const TEntry* e=iTable;
458 if (aStream>=iPos || e==iFinish || aStream<e->entry.ref)
459 return -1; // we don't know it
460 for (;e->entry.ref!=aStream;++e)
462 __ASSERT_DEBUG(e->entry.ref<aStream,User::Invariant());
463 __ASSERT_DEBUG(e<iFinish,User::Invariant());
465 __ASSERT_DEBUG(e->entry.handle>=0,User::Invariant());
466 __ASSERT_DEBUG(e->len>=0,User::Invariant());
467 return aStream+e->len;
472 // class TPermanentStoreStreamIter
474 void TPermanentStoreStreamIter::Reset()
476 // reset the iterator for a new scan
485 TInt TPermanentStoreStreamIter::FillL(CPermanentStoreToc& aToc)
487 // Fill the table with the next bundle of stream offsets
488 // return the total streams left to iterate
491 __ASSERT_DEBUG(iNext==iFinish,User::Invariant());
495 __ASSERT_DEBUG(iPos>=0,User::Invariant());
497 RPermanentStoreTocIter iter(aToc);
498 CleanupReleasePushL(iter);
504 TInt* end=heap+EBundleSize;
505 RPermanentStoreTocIter::TEntry entry;
506 for (iter.ResetL();iter.NextL(entry);)
515 Push(heap,last++,off); // push onto the bottom
517 PopPush(heap,last,off); // replace item in heap
519 CleanupStack::PopAndDestroy();
521 iMore=streams+(heap-last);
522 iPos=*heap+1; // largest offset in table
526 *last=PopPush(heap,last,*last);
530 TInt TPermanentStoreStreamIter::Next()
532 // return the next offset, or <0 if the table is empty
535 __ASSERT_DEBUG(iMore>=0,User::Invariant());
537 while (iNext<iFinish)
546 void TPermanentStoreStreamIter::Relocated(TInt aStream)
548 // aStream has been relocated, mark the table
551 __ASSERT_DEBUG(iMore>=0,User::Invariant());
567 for (;*p!=aStream;++p)
569 __ASSERT_DEBUG(p<iFinish,User::Invariant());
570 __ASSERT_DEBUG(*p<aStream,User::Invariant());
578 void TPermanentStoreStreamIter::Push(TInt* aHeap,TInt* aHole,TInt aValue)
580 // push aValue onto the heap (which will grow)
583 TInt ix=aHole+1-aHeap;
586 TInt link=ix-(ix>>1);
588 TInt* parent=aHole-link;
597 TInt TPermanentStoreStreamIter::PopPush(TInt* aHeap,TInt* anEnd,TInt aValue)
599 // pop the top and push aValue
612 if (child<anEnd && *(child+1)>*child)
628 // Class CPermanentStoreCollector
630 CPermanentStoreCollector* CPermanentStoreCollector::CompactorL(CPermanentStoreCoord& aCoord)
632 CPermanentStoreCollector* self=ReclaimerL(aCoord);
633 CleanupStack::PushL(self);
634 self->iReloc=new(ELeave) TPermanentStoreRelocator;
639 CPermanentStoreCollector* CPermanentStoreCollector::ReclaimerL(CPermanentStoreCoord& aCoord)
641 return new(ELeave) CPermanentStoreCollector(aCoord);
644 CPermanentStoreCollector::CPermanentStoreCollector(CPermanentStoreCoord& aCoord)
645 : iCoord(&aCoord),iHost(&aCoord.Host()),iStreams(EGranularity,_FOFF(TEntry,entry.ref))
650 CPermanentStoreCollector::~CPermanentStoreCollector()
654 iMark.Withdraw(Host());
658 void CPermanentStoreCollector::DoRelease()
663 void CPermanentStoreCollector::DoResetL(TInt& aCount)
665 iMark.Withdraw(Host());
666 iMark=KStreamBeginning;
667 iCoordGen=Coord().Generation();
669 TRAPD(r, aCount = FastResetL());
672 if (r != KErrNoMemory)
674 // fall back to fixed memory solution
679 TInt streams=iIter.FillL(Coord().ConsolidateL());
683 const TInt KMaxStepEffort=9;
685 void CPermanentStoreCollector::DoNextL(TInt& aStep,TInt& aTotal)
687 // Dispatch the next set of operations
692 __DEBUG(Panic(TStorePanic()));
696 if (iCoordGen!=Coord().Generation() || Coord().TableL().IsVirtual())
697 __LEAVE(KErrNotReady);
705 __DEBUG(User::Invariant());
710 effort+=SkipL(aTotal);
714 effort+=InitRelocator();
717 effort+=FillRelocatorL();
720 effort+=EvalRelocatorL();
723 effort+=ScanRelocator();
725 case ERelocateStream:
726 effort+=RelocateStreamL();
730 RelocateTocL(aTotal);
732 __ASSERT_DEBUG(aStep==1,User::Invariant());
744 FastRelocateL(aTotal);
748 } while(effort<KMaxStepEffort);
749 __ASSERT_DEBUG(aStep>=1,User::Invariant());
752 TInt CPermanentStoreCollector::GetFreeL()
754 // find the end of the free space
757 __ASSERT_DEBUG(iState==EGetFree,User::Invariant());
758 __ASSERT_DEBUG(iFree>=0,User::Invariant());
764 if (iIter.FillL(Coord().ConsolidateL())==0)
766 iState=ERelocateToc; // no more streams
769 effort=KMaxStepEffort;
771 __ASSERT_DEBUG(iEnd>=0,User::Invariant());
773 iState=Compacting() && HaveEnoughSpace() ? EInitRelocator : ESkip;
777 TInt CPermanentStoreCollector::SkipL(TInt& aTotal)
779 // Find some free space, iEnd was the last stream extracted from concat
782 __ASSERT_DEBUG(iState==ESkip,User::Invariant());
783 __ASSERT_DEBUG(iFree>=0&&iFree<=iEnd,User::Invariant());
786 iFree=SkipLink(ExtentL(iEnd));
791 TInt CPermanentStoreCollector::InitRelocator()
793 // initialise the relocator for the free space
796 __ASSERT_DEBUG(iState==EInitRelocator,User::Invariant());
797 __ASSERT_DEBUG(Compacting(),User::Invariant());
799 iState=iReloc->Reset(iEnd) ? EScanRelocator : EFillRelocator;
800 return 0; // no real work at all
803 TInt CPermanentStoreCollector::FillRelocatorL()
805 // Fill the relocator table
808 __ASSERT_DEBUG(iState==EFillRelocator,User::Invariant());
809 __ASSERT_DEBUG(iFree>=0&&iFree<iEnd,User::Invariant());
810 __ASSERT_DEBUG(Compacting(),User::Invariant());
812 TBool more=iReloc->FillL(Coord().ConsolidateL());
813 iState=more ? EEvalRelocator : ESkip;
814 return more ? KMaxStepEffort : 0;
817 TInt CPermanentStoreCollector::EvalRelocatorL()
819 // evaluate the extents for the relocator
822 __ASSERT_DEBUG(iState==EEvalRelocator,User::Invariant());
823 __ASSERT_DEBUG(Compacting(),User::Invariant());
825 iReloc->EvaluateLengthsL(Host(),iMark,Coord().Base());
826 iState=EScanRelocator;
827 return KMaxStepEffort;
830 TInt CPermanentStoreCollector::ScanRelocator()
832 // find a stream that will fit
835 __ASSERT_DEBUG(iState==EScanRelocator,User::Invariant());
836 __ASSERT_DEBUG(iFree>=0&&iFree<iEnd,User::Invariant());
837 __ASSERT_DEBUG(Compacting(),User::Invariant());
839 iState=iReloc->FindFit(iFree,iEnd) ? ERelocateStream : EFillRelocator;
843 TInt CPermanentStoreCollector::RelocateStreamL()
845 // find and relocate a stream
848 __ASSERT_DEBUG(iState==ERelocateStream,User::Invariant());
849 __ASSERT_DEBUG(iFree>=0&&iFree<iEnd,User::Invariant());
850 __ASSERT_DEBUG(Compacting(),User::Invariant());
852 const TEntry& e = *iReloc->Current();
853 RelocateStreamL(e, iEnd);
854 iState=e.entry.ref==iEnd ? EGetFree : HaveEnoughSpace() ? EScanRelocator : ESkip;
855 iIter.Relocated(e.entry.ref);
856 iReloc->Relocated(&e);
857 return KMaxStepEffort;
860 void CPermanentStoreCollector::RelocateTocL(TInt& aTotal)
862 // relocate the toc - return any wasted bytes
865 __ASSERT_DEBUG(iState==ERelocateToc,User::Invariant());
866 __ASSERT_DEBUG(iFree>=0,User::Invariant());
868 TInt toc=Coord().iToc+KOffsetTocHeader;
874 TInt len=Coord().TableL().Extent()-toc;
875 if (Fits(iFree,toc,len))
877 RelocateL(toc, len, EFrameDescriptive16, toc);
878 Coord().MoveL(iFree-KOffsetTocHeader,iFree + len);
886 TBool CPermanentStoreCollector::HaveEnoughSpace() const
888 __ASSERT_DEBUG(Compacting(),User::Invariant());
890 return SpaceFor(iFree,iEnd,iReloc->MinLength());
893 TInt CPermanentStoreCollector::ExtentL(TInt aStream)
895 // Check if the relocator knows before scanning the file
900 TInt ext=iReloc->Extent(aStream);
904 return ::ExtentL(Host(),iMark,Coord().Base(),aStream);
907 /* relocate a stream into [iFree, aExtent)
909 During compaction, for each string which is to be moved from position A1 to B1, the sequence of operations is:
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.
912 2. Optionally rewrite the file header to state that stream S1 is being relocated to B1 (more about the ‘optional below’)
913 3. Overwrite the TOC entry for S1 to state that the content is now at B1
915 This function completes 3 steps above and will be called again and again for every string to be moved.
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):
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
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).
920 The TOC will be good because it is being overwritten with the same content.
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
922 override the entry in the TOC.
924 In all cases the file is never broken by a crash in mid-compaction.
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
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
928 media provides the required behavior. Thus 99% of the step #2 writes are eliminated.
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
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
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
933 and leave the TOC corrupt with no recovery in the file header.
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;
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
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
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.
941 Based on the knowledge above, it is strongly recommended to set EFileWriteDirectIO bit when opening the file so that the order is maintained
942 when writing to the file.
944 void CPermanentStoreCollector::RelocateStreamL(const CPermanentStoreCollector::TEntry& aReloc, TInt aExtent)
946 if (Coord().Accessed()) // must have exclusive access to relocate the stream
949 TInt end=RelocateL(aReloc.entry.ref,aReloc.len,aReloc.entry.handle == KHandleTocBase ? EFrameDescriptive16 : EFrameData16, aExtent);
951 Coord().RelocateL(aReloc.entry.handle, iFree);
952 // Step 2 & 3, 5 & 6,...
953 iCoordGen=Coord().Generation(); // changed by relocation
957 TInt CPermanentStoreCollector::RelocateL(TInt aStream, TInt aLength, TFrameType16 aType, TInt aExtent)
959 // Relocate a data stream into [iFree, aExtent)
962 __ASSERT_DEBUG(Fits(iFree,aExtent,aLength),User::Invariant());
963 __ASSERT_DEBUG(Compacting(),User::Invariant());
965 TInt end=SkipLink(iFree+aLength);
971 TInt anchor=((end>>KShiftFrameLength16)+1)<<KShiftFrameLength16;
974 __ASSERT_DEBUG(aExtent-KSizeFrameDes16-end>0,User::Invariant());
975 terminator=EFrameFree16+(aExtent-KSizeFrameDes16-end);
978 terminator=EFrameFree16+KFrameOpen16;
981 RFrame16Buf from(Coord().Base());
982 from.Set(Host(),aStream,aStream+aLength,MStreamBuf::ERead);
984 TRelocatorInput input(Host(),iMark);
985 input.OpenL(aType,Coord().Base(),iFree,aLength,terminator);
986 from.ReadL(input,aLength);
988 CleanupStack::PopAndDestroy();
994 TInt CPermanentStoreCollector::FastResetL()
996 // Fill the table with the streams with data in the store
1001 CleanupClosePushL(iStreams);
1002 RPermanentStoreTocIter iter(Coord().ConsolidateL());
1003 CleanupReleasePushL(iter);
1005 for (iter.ResetL();iter.NextL(entry.entry);)
1007 if (entry.entry.handle<0)
1009 if (entry.entry.ref<0)
1011 User::LeaveIfError(iStreams.Append(entry));
1013 CleanupStack::PopAndDestroy(&iter);
1014 CleanupStack::Pop(&iStreams);
1016 // always have final (toc) step
1017 iState = ERelocateToc;
1018 TInt streams = iStreams.Count();
1022 // ordering is 1 step and evaluating the extents is several more
1023 TInt count = 2 + (streams + EExtentStep - 1)/EExtentStep;
1029 void CPermanentStoreCollector::FastSort()
1031 __ASSERT_DEBUG(iState == EFastSort, User::Invariant());
1033 iStreams.SortSigned();
1034 iNext = &iStreams[0];
1035 iLast = iNext + iStreams.Count();
1036 iState = EFastExtent;
1039 void CPermanentStoreCollector::FastExtentL(TInt& aTotal)
1041 // Evaluate the lengths for all the streams
1042 // if reclaiming, update aTotal with free space skipped
1043 // return false until we've done the last one
1046 __ASSERT_DEBUG(iState == EFastExtent, User::Invariant());
1047 __ASSERT_DEBUG(iNext != iLast, User::Invariant());
1050 const TEntry* end = Min(iLast, e + EExtentStep);
1053 TInt ref = e->entry.ref;
1054 __ASSERT_DEBUG(TUint(iFree)<=TUint(ref),User::Invariant());
1055 TInt ext = ::ExtentL(Host(), iMark, Coord().Base(), ref);
1058 aTotal += ref - iFree;
1059 iFree = SkipLink(ext);
1060 } while (++e < end);
1066 iState = ERelocateToc;
1069 iNext = &iStreams[0];
1071 iState = EFastRelocate;
1076 CPermanentStoreCollector::TEntry* CPermanentStoreCollector::BestFit(TInt aPos, TInt aExt, TEntry* aFirst, TEntry* aLast)
1078 // [aPos, aExt) is a hole - find the best fit to fill it in [aFirst, aLast)
1081 __ASSERT_DEBUG(aPos <= aExt, User::Invariant());
1086 if ((aExt & KMaskFrameLength16) != 0)
1087 aExt -= KSizeFrameDes16;
1088 const TInt mingap = Min(KSizeFrameDes16 + 1, (aExt & KMaskFrameLength16));
1090 TInt avail = aExt - aPos;
1096 len = (--aLast)->len;
1099 if (aFirst == aLast)
1102 TInt d = avail - len;
1105 if (d == 0) // exact fit
1111 } while (aFirst != aLast);
1115 void CPermanentStoreCollector::FastRelocateL(TInt& aTotal)
1117 // Look for a stream to move. Either move a stream or fail to find a fit before returning
1118 // fill holes from the front of the file with the biggest block that will fit (inverted current algorithm)
1119 // return true when the last hole has been looked at
1122 __ASSERT_DEBUG(iState == EFastRelocate, User::Invariant());
1123 __ASSERT_DEBUG(iNext != iLast, User::Invariant());
1126 TInt ext = e->entry.ref;
1127 __ASSERT_DEBUG(ext >= 0, User::Invariant());
1129 TEntry* r = BestFit(iFree, ext, e, iLast);
1132 // Nothing fits, accumulate free space
1133 aTotal += ext - iFree;
1134 iFree = SkipLink(ext + e->len);
1139 RelocateStreamL(*r, ext);
1142 // relocated a stream other than the one terminating the current hole
1149 // skip through any relocated streams
1150 while (++e < iLast && e->entry.ref < 0)
1154 iState = ERelocateToc;