Update contrib.
1 // Copyright (c) 1998-2010 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.
19 //The default media block size, used in the computations, if the file system guarantees atomic "block write" file operations.
20 const TInt KDefaultMediaBlockSize = 512;
22 ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
23 ///////////////////// TPermanentStoreHeader /////////////////////////////////////////////////////
24 ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
27 // - the dirty bit is set (indicates commit phase #2 or #3 is not complete);
28 // - the CRC check fails ("backup TOC ref", "handle" and "TOC ref" are protected by 16-bit CRC in the permanent store file header);
29 TBool TPermanentStoreHeader::IsValid() const
35 Mem::Crc(crc,Ptr(),_FOFF(TPermanentStoreHeader,iCrc));
39 //Sets the "backup TOC ref", "handle" and "TOC ref" in current TPermanentStoreHeader object.
40 //16-bit CRC is calculated, based on the values of the input parameters, and stored together with them in the
41 //TPermanentStoreHeader object.
42 void TPermanentStoreHeader::Set(TInt aBackupToc,TInt aHandle,TInt aReference)
44 iBackup=TUint32(aBackupToc)<<1;
48 Mem::Crc(iCrc,Ptr(),_FOFF(TPermanentStoreHeader,iCrc));
49 __ASSERT_DEBUG(IsValid(),User::Invariant());
52 const TInt KTocGranularity=12;
60 ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
61 ///////////////////// CPermanentStoreToc ////////////////////////////////////////////////////////
62 ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
64 TInt CPermanentStoreToc::TEntry::Compare(const TEntry& aLeft, const TEntry& aRight)
66 // Ordering for Stream handles, return <0,0,>0 depending on comparison
67 // The index part of the handle is only 24 bits, so just subtract the handles for result
70 return (aLeft.handle&KMaskHandleIndex) - (aRight.handle&KMaskHandleIndex);
73 CPermanentStoreToc* CPermanentStoreToc::NewL(TStreamPos aBase,TStreamExchange& aHost,TInt aToc,TInt aBaseReloc)
75 CPermanentStoreToc* table=new(ELeave) CPermanentStoreToc(aBase,aHost);
76 CleanupStack::PushL(table);
77 table->ConstructL(aToc,aBaseReloc);
82 TBool CPermanentStoreToc::IsDelta() const
84 // Report whether the current TOC should be written as a delta, or a base TOC
87 TInt delta = iEntries.Count();
88 if (delta >= KTocDeltaCap)
90 TInt magic = iMagic + delta;
91 if (magic > KMaxTocDeltaMagic)
93 return magic*KSizeTocDeltaEntry <= iCount*KSizeTocEntry - KSizeTocDeltaExtra;
96 void CPermanentStoreToc::Move(TInt aToc,TInt anExtent)
98 // Final stage of compaction, update to address new TOC
101 __ASSERT_DEBUG(anExtent<=iOff&&anExtent>=aToc&&anExtent-(aToc+KOffsetTocHeader)==iExt-iOff,User::Invariant());
104 // Base-toc only, update the base toc position as well
105 TInt window=iWindow-iTocExt;
106 iTocOff=aToc+KOffsetTocHeader;
108 iWindow=anExtent+window;
110 iOff = aToc+KOffsetTocHeader;
114 TInt CPermanentStoreToc::RealizeL(TInt aPrimary,TInt anExtent) const
116 __ASSERT_DEBUG(IsVirtual(),User::Invariant());
117 RFrame16Buf buf(Base());
119 const TBool delta = IsDelta();
120 aPrimary &= KMaskStreamIdValue;
122 aPrimary |= KTocDelta;
124 h.primary=TInt32(aPrimary);
125 h.avail=TInt32(iAvail);
126 h.count=TUint32(iCount);
130 off=DeltaL(buf,anExtent,h);
132 off=RewriteL(buf,anExtent,h);
135 CleanupStack::PopAndDestroy(&buf);
136 return off-KOffsetTocHeader;
139 void CPermanentStoreToc::Adopt(TInt aToc,TInt aPrimary)
141 // Final stage of Commit - all file changes have been successful
142 // Record the new Toc location and reset the changed flag
145 __ASSERT_DEBUG(aToc+KOffsetTocHeader>=iExt,User::Invariant());
149 // Adjust the TOC-delta location
150 TInt c = iEntries.Count();
151 iOff=aToc+KOffsetTocHeader;
152 iExt=aToc + KSizeTocDeltaExtra + KSizeTocDeltaEntry*c;
153 if (c > KTocDeltaMagic)
154 iMagic += c - KTocDeltaMagic;
158 // Adjust all the TOC data and reset the delta-set
159 TInt window = iTocOff>=0 ? iWindow-iTocOff : -KOffsetTocHeader;
160 iTocOff=iOff=aToc+KOffsetTocHeader;
161 iTocExt=iExt=aToc+KSizeTocEntry*iCount;
162 iWindow = iTocOff + window;
167 __ASSERT_DEBUG(!IsVirtual(),User::Invariant());
170 TInt CPermanentStoreToc::AllocL(TInt anOffset)
172 TEntry& entry=DoAllocL();
177 TInt CPermanentStoreToc::AllocL()
179 TEntry& entry=DoAllocL();
180 entry.ref=KFrameNonexistent16;
181 return entry.handle|=KHandleInvalid;
184 void CPermanentStoreToc::Cancel(TInt aHandle)
186 __ASSERT_DEBUG(aHandle<0&&IsVirtual(),User::Invariant());
187 TEntry* entry=Entry(aHandle);
188 __ASSERT_DEBUG((entry!=NULL) && (entry->handle==aHandle),User::Invariant());
193 void CPermanentStoreToc::FreeL(TInt aHandle)
195 __ASSERT_DEBUG(aHandle>0,User::Invariant());
196 TEntry& entry=FetchL(aHandle);
197 aHandle|=KHandleInvalid;
198 entry.handle=aHandle;
204 TInt CPermanentStoreToc::AtL(TInt aHandle) const
206 __ASSERT_DEBUG(aHandle>0,User::Invariant());
208 // check the delta table
209 const TEntry* entry=Entry(aHandle);
212 // if there is an entry in the delta table, but its
213 // not an exact match leave with KErrNotFound
214 // This happens when the delta indicates it's been deleted
215 if (entry->handle!=aHandle)
216 User::Leave(KErrNotFound);
221 // if it's not in the delta table check the base TOC
222 return DoAtL(aHandle);
225 void CPermanentStoreToc::PutL(TInt aHandle, TInt anOffset, TPut aCheck)
227 // Used by compaction to update a single stream reference IN-PLACE in the TOC
228 // We need to check in which TOC to make the update
231 __ASSERT_DEBUG(!IsVirtual(),User::Invariant());
232 __ASSERT_DEBUG(aHandle>0,User::Invariant());
233 __ASSERT_DEBUG(aHandle!=KHandleTocBase || HasDelta(),User::Invariant());
235 if (aHandle == KHandleTocBase)
237 // update the TOC-base link
238 PutTocL(anOffset - KOffsetTocHeader, aCheck);
239 if (iTocOff != anOffset)
241 TInt size = iTocExt - iTocOff;
242 TInt window = iWindow - iTocOff;
244 iTocExt = anOffset + size;
245 iWindow = anOffset + window;
253 if (iEntries.FindInOrder(e,i,&TEntry::Compare)==0)
255 // update TOC-delta entry
256 TEntry& entry=iEntries[i];
257 if (entry.handle==aHandle && entry.ref != anOffset)
259 PutDeltaL(i,aHandle,anOffset);
265 // update TOC-base entry
266 if (aCheck==EWrite || DoAtL(aHandle)!=anOffset)
267 PutBaseL(aHandle,anOffset);
270 TInt CPermanentStoreToc::GetL(TInt aHandle)
272 __ASSERT_DEBUG(aHandle>0,User::Invariant());
273 TEntry& entry=FetchL(aHandle);
277 TInt CPermanentStoreToc::Set(TInt aHandle,TInt anOffset)
279 __ASSERT_DEBUG(aHandle!=0,User::Invariant());
280 TEntry* entry=Entry(aHandle);
281 __ASSERT_DEBUG((entry!=NULL) && (entry->handle==aHandle),User::Invariant());
282 aHandle&=KMaskStreamIdValue;
283 entry->handle=aHandle;
289 CPermanentStoreToc::CPermanentStoreToc(TStreamPos aBase,TStreamExchange& aHost)
290 : /*iPrimary(0),iAvail(0),iCount(0),*/iEntries(KTocGranularity),
291 iBase(aBase),iHost(&aHost),iOff(KFrameNonexistent16)/*,iExt(0)*/,
292 iTocOff(KFrameNonexistent16)/*,iTocExt(0),iWindow(0)*/
295 CPermanentStoreToc::~CPermanentStoreToc()
300 void CPermanentStoreToc::ConstructL(TInt aToc,TInt aBaseReloc)
302 // Read and validate the TOC header at aToc.
303 // aBaseReloc may contain a relocated TOC-base offset, which should override the one in a TOC-delta
308 __ASSERT_DEBUG(aToc>0&&iEntries.Count()==0,User::Invariant());
309 RFrame16Buf buf(Base());
310 aToc+=KOffsetTocHeader;
311 buf.OpenL(Host(),aToc,EFrameDescriptive16|buf.ERead);
313 RReadStream stream(&buf);
315 stream.ReadL((TUint8*)&h,sizeof(STocHead)); // platform dependency
316 if ((h.primary&~(KMaskStreamIdValue|static_cast<TUint>(KTocDelta)))!=0||
317 h.avail>0||(h.avail&KMaskHandleClear)!=0||(h.count&~KMaskHandleIndex)!=0)
318 __LEAVE(KErrCorrupt);
320 iPrimary=TInt(h.primary) & ~KTocDelta;
321 iAvail=TInt(h.avail);
322 iCount=TInt(h.count);
325 if (h.primary & KTocDelta)
327 // This is a TOC-delta
328 aToc = InternalizeL(stream, aBaseReloc) + KOffsetTocHeader;
330 // Now locate and validate the base TOC
332 buf.OpenL(Host(),aToc,EFrameDescriptive16|buf.ERead);
333 stream.ReadL((TUint8*)&h,sizeof(STocHead)); // platform dependency
334 if ((h.primary&~KMaskStreamIdValue)!=0||(h.count&~KMaskHandleIndex)!=0||
335 h.avail>0||(h.avail&KMaskHandleClear)!=0||TInt(h.count)>iCount)
336 __LEAVE(KErrCorrupt);
338 TInt size = KSizeTocEntry*TInt(h.count);
339 // Eagerly load the first few TOC entries into the cache
340 // This is good as it almost certainly lies in the file buffer already
342 stream.ReadL(&iBuf[0],Min(size,TInt(KSizeTocBuf)));
343 size-=KOffsetTocHeader;
344 if (buf.SizeL()!=size)
345 __LEAVE(KErrCorrupt);
347 iWindow=aToc-KOffsetTocHeader;
350 iExt = iTocExt; // set extent for non-delta TOC
352 CleanupStack::PopAndDestroy(&buf);
355 TInt CPermanentStoreToc::InternalizeL(RReadStream& aIn, TInt aBaseReloc)
357 // Load and validate the delta-toc table
360 TInt tocoff = aIn.ReadInt32L();
361 if (aBaseReloc != KFrameNonexistent16)
362 tocoff = aBaseReloc - KOffsetTocHeader;
363 if (tocoff<0||tocoff>=iOff)
364 __LEAVE(KErrCorrupt);
365 iMagic = aIn.ReadUint16L();
367 __LEAVE(KErrCorrupt);
368 TInt n = aIn.ReadUint8L();
369 TInt size = -KOffsetTocHeader + KSizeTocDeltaExtra + KSizeTocDeltaEntry * n;
370 if (aIn.Source()->SizeL()!=size)
371 __LEAVE(KErrCorrupt);
377 aIn.ReadL((TUint8*)&e,KSizeTocDeltaEntry); // platform dependency
378 if ((e.handle&KMaskHandleClear)!=0||e.handle>=0&&(e.ref<KFrameNonexistent16)||
379 e.handle<0&&(e.ref>0||(e.ref&KMaskHandleClear)!=0))
380 __LEAVE(KErrCorrupt);
381 TInt i = e.handle&KMaskHandleIndex;
382 if (i <= last || i > iCount)
383 __LEAVE(KErrCorrupt);
386 entry.handle = TInt(e.handle);
387 entry.ref = TInt(e.ref);
388 User::LeaveIfError(iEntries.Append(entry));
394 TInt CPermanentStoreToc::DeltaL(RFrame16Buf& aBuf,TInt aExtent,const STocHead& aHead) const
396 // Write the TOC-delta
399 TInt off=aBuf.ExtendL(Host(),aExtent,EFrameDescriptive16|aBuf.EWrite);
400 RWriteStream out(&aBuf);
401 out.WriteL((TUint8*)&aHead,sizeof(STocHead)); // platform dependency
402 out.WriteInt32L(iTocOff - KOffsetTocHeader);
403 TInt n = iEntries.Count();
405 if (n > KTocDeltaMagic)
406 magic += n - KTocDeltaMagic;
407 __ASSERT_DEBUG(magic <= KMaxTocDeltaMagic,User::Invariant());
408 __ASSERT_DEBUG(n <= (TInt)KMaxTUint8, User::Invariant());
409 out.WriteUint16L(magic);
411 for (int i = 0; i < n; ++i)
413 const TEntry& entry = iEntries[i];
415 e.handle=TInt32(entry.handle);
416 e.ref=TInt32(entry.ref);
417 out.WriteL((TUint8*)&e,KSizeTocDeltaEntry); // platform dependency
422 TInt CPermanentStoreToc::RewriteL(RFrame16Buf& aBuf,TInt aExtent,const STocHead& aHead) const
424 // Write the TOC-base
427 const TInt KElementsRewrite=304;
428 const TInt KSizeRewriteBuf=KElementsRewrite*KSizeTocEntry;
429 TUint8 toc[KSizeRewriteBuf];
431 RFrame16Buf buf(Base());
437 oldsize = iTocExt-iTocOff+KOffsetTocHeader;
438 window = iWindow + KOffsetTocHeader - iTocOff;
439 if (oldsize <= Min(KSizeTocBuf,KSizeRewriteBuf) && window == 0)
441 // the old TOC is already in the toc-base cache, no need to read it
442 Mem::Copy(&toc[0],&iBuf[0],oldsize);
443 oldsize = 0; // this prevents the read request
447 buf.Set(Host(),iTocOff,iTocExt,EFrameDescriptive16|buf.ERead);
448 buf.SeekL(buf.ERead,TStreamPos(-KOffsetTocHeader));
451 RReadStream in(&buf);
452 // defer the initialisation of the write in roder to improve file buffering performance
453 RWriteStream out(&aBuf);
455 TInt size=iCount*KSizeTocEntry;
457 for (TInt base=0,delta=0;base<size;base+=KSizeRewriteBuf)
459 // fill buffer with old TOC data
461 in.ReadL(&toc[0],Min(KSizeRewriteBuf,oldsize-base));
462 // apply changes to this block
463 for (TInt n=iEntries.Count();delta<n;++delta)
465 const TEntry& entry=iEntries[delta];
466 TInt pos = (entry.handle&KMaskHandleIndex)*KSizeTocEntry - KSizeTocEntry - base;
467 __ASSERT_DEBUG(pos>=0,User::Invariant());
468 if (pos>=KSizeRewriteBuf)
471 e.handle=TInt32(entry.handle);
472 e.ref=TInt32(entry.ref);
473 Mem::Copy(&toc[pos],(const TUint8*)&e+KSizeHandleIndex,KSizeTocEntry);
474 pos += base - window;
475 if (TUint(pos) < TUint(KSizeTocBuf))
476 Mem::Copy(MUTABLE_CAST(TUint8*,&iBuf[pos]),(const TUint8*)&e+KSizeHandleIndex,KSizeTocEntry);
478 // write buffer to file
481 // initialise writing
482 off=aBuf.ExtendL(Host(),aExtent,EFrameDescriptive16|aBuf.EWrite);
483 out.WriteL((TUint8*)&aHead,sizeof(STocHead)); // platform dependency
485 out.WriteL(&toc[0],Min(KSizeRewriteBuf,size-base));
488 CleanupStack::PopAndDestroy(&buf);
492 CPermanentStoreToc::TEntry* CPermanentStoreToc::Entry(TInt aHandle)
497 if (iEntries.FindInOrder(e,i,&TEntry::Compare)!=0)
500 TEntry& entry=iEntries[i];
502 // allow entry to return a pointer even if not an exact match
506 CPermanentStoreToc::TEntry& CPermanentStoreToc::FetchL(TInt aHandle)
511 if (iEntries.FindInOrder(e,i,&TEntry::Compare)!=0)
513 e.ref=DoAtL(aHandle);
514 User::LeaveIfError(iEntries.Insert(e,i));
516 TEntry& entry=iEntries[i];
517 if (entry.handle!=aHandle)
518 __LEAVE(KErrNotFound);
523 CPermanentStoreToc::TEntry& CPermanentStoreToc::DoAllocL()
529 __ASSERT_DEBUG(iEntries.Count()==0||(iEntries[iEntries.Count()-1].handle&KMaskHandleIndex)<=iCount,User::Invariant());
530 User::LeaveIfError(iEntries.Append(TEntry()));
531 entry=&iEntries[iEntries.Count()-1];
536 entry=&FetchL(handle);
537 handle=(handle+KIncHandleGen)&KMaskStreamIdValue;
539 TInt avail=entry->ref;
540 if (avail>0||(avail&KMaskHandleClear)!=0)
541 __LEAVE(KErrCorrupt);
545 entry->handle=handle;
550 TInt CPermanentStoreToc::DoAtL(TInt aHandle) const
554 __LEAVE(KErrNotFound);
556 off+=-KOffsetTocHeader-KSizeTocEntry+KSizeTocEntry*(aHandle&KMaskHandleIndex);
557 __ASSERT_DEBUG(off>=iTocOff-KOffsetTocHeader,User::Invariant());
559 __LEAVE(KErrNotFound);
562 if (off-window<0||off-window>=KSizeTocBuf)
564 TInt len=iTocOff-KOffsetTocHeader;
565 window=Max(len,Min(off-KBackTocBuf,iTocExt-KSizeTocBuf));
566 len=Min(iTocExt-len,TInt(KSizeTocBuf));
568 RFrame16Buf buf(Base());
569 buf.Set(Host(),iTocOff,iTocExt,EFrameDescriptive16|buf.ERead);
571 buf.SeekL(buf.ERead,TStreamPos(window-iTocOff));
572 RReadStream stream(&buf);
573 MUTABLE_CAST(TInt&,iWindow)=iTocExt;
574 stream.ReadL(MUTABLE_CAST(TUint8*,&iBuf[0]),1); // assume half decent read-ahead buffering
575 stream.ReadL(MUTABLE_CAST(TUint8*,&iBuf[1]),len-1);
576 MUTABLE_CAST(TInt&,iWindow)=window;
577 CleanupStack::PopAndDestroy();
581 Mem::Copy((TUint8*)&e+KSizeHandleIndex,&iBuf[off-window],KSizeTocEntry); // platform dependency
582 if ((e.handle&KMaskHandleClear)!=0||e.handle>=0&&(e.ref<KFrameNonexistent16)||
583 e.handle<0&&(e.ref>0||(e.ref&KMaskHandleClear)!=0))
584 __LEAVE(KErrCorrupt);
586 if (TInt(e.handle)!=aHandle)
587 __LEAVE(KErrNotFound);
592 void CPermanentStoreToc::PutBaseL(TInt aHandle,TInt aReference)
594 // Update a TOC-base entry
597 __ASSERT_DEBUG(iTocOff>=0,User::Invariant());
599 TInt pos=-KOffsetTocHeader-KSizeTocEntry+KSizeTocEntry*(aHandle&KMaskHandleIndex);
600 RFrame16Buf buf(Base());
601 buf.Set(Host(),iTocOff,iTocExt,EFrameDescriptive16|buf.EWrite);
603 buf.SeekL(buf.EWrite,TStreamPos(pos));
604 RWriteStream stream(&buf);
608 stream.WriteL((TUint8*)&e+KSizeHandleIndex,KSizeTocEntry); // platform dependency
609 CleanupStack::PopAndDestroy();
610 TInt off=pos+iTocOff-iWindow;
611 if (off>=0&&off<KSizeTocBuf)
612 Mem::Copy(&iBuf[off],(const TUint8*)&e+KSizeHandleIndex,KSizeTocEntry); // platform dependency
615 void CPermanentStoreToc::PutDeltaL(TInt aPos, TInt aHandle, TInt aReference)
617 // Update a single stream reference IN-PLACE in the TOC-delta
620 __ASSERT_DEBUG(HasDelta(),User::Invariant());
621 __ASSERT_DEBUG(iOff>=0,User::Invariant());
622 RFrame16Buf buf(Base());
623 buf.Set(Host(),iOff,iExt,EFrameDescriptive16|buf.EWrite);
625 buf.SeekL(buf.EWrite,TStreamPos(-KOffsetTocHeader + KSizeTocDeltaExtra + KSizeTocDeltaEntry*aPos));
626 RWriteStream stream(&buf);
630 stream.WriteL((TUint8*)&e,KSizeTocDeltaEntry); // platform dependency
631 CleanupStack::PopAndDestroy();
634 void CPermanentStoreToc::PutTocL(TInt aTocBase, TPut aCheck)
636 // Update the base TOC link in the TOC-delta
639 __ASSERT_DEBUG(HasDelta(),User::Invariant());
640 __ASSERT_DEBUG(iOff>=0,User::Invariant());
641 RFrame16Buf buf(Base());
642 buf.Set(Host(),iOff,iExt,EFrameDescriptive16|buf.EWrite|buf.ERead);
644 buf.SeekL(buf.EWrite|buf.ERead,TStreamPos(-KOffsetTocHeader));
645 RReadStream in(&buf);
646 if (aCheck==EWrite || in.ReadInt32L() != aTocBase)
648 RWriteStream out(&buf);
649 out.WriteInt32L(aTocBase);
651 CleanupStack::PopAndDestroy();
654 // Used prior to PutL to determine exactly where in the file the TOC update will be written.
655 // This may be used to eliminate the pre-put write of the relocation information in the file
656 // header in situations where the write is entirely within an atomic file block.
658 // The detailed numbers used to offset into the TOCs are dependant on the TOC formats, as
659 // implicated by other members of this class
660 TInt CPermanentStoreToc::RefSpan(TInt aHandle,TInt& aLength)
662 __ASSERT_DEBUG(!IsVirtual(),User::Invariant());
663 __ASSERT_DEBUG(aHandle>0,User::Invariant());
665 if (aHandle == KHandleTocBase)
666 { // locate the TOC-base link
667 __ASSERT_DEBUG(HasDelta(),User::Invariant());
668 __ASSERT_DEBUG(iOff>=0,User::Invariant());
669 aLength = sizeof(TInt32);
670 return RFrame16Buf::Position(Base(), iOff - KOffsetTocHeader).Offset();
676 if (iEntries.FindInOrder(e, ix, &TEntry::Compare) == 0)
677 { // locate TOC-delta entry
678 TEntry& entry=iEntries[ix];
679 if (entry.handle==aHandle)
681 __ASSERT_DEBUG(HasDelta(),User::Invariant());
682 __ASSERT_DEBUG(iOff>=0,User::Invariant());
683 aLength = KSizeTocDeltaEntry;
684 return RFrame16Buf::Position(Base(),iOff - KOffsetTocHeader + KSizeTocDeltaExtra + KSizeTocDeltaEntry*ix).Offset();
688 // locate the TOC-base entry
689 __ASSERT_DEBUG(iTocOff>=0,User::Invariant());
690 aLength = KSizeTocEntry;
691 return RFrame16Buf::Position(Base(),iTocOff-KOffsetTocHeader-KSizeTocEntry+KSizeTocEntry*(aHandle&KMaskHandleIndex)).Offset();
694 ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
695 ///////////////////// RPermanentStoreTocIter ////////////////////////////////////////////////////
696 ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
698 RPermanentStoreTocIter::RPermanentStoreTocIter(const CPermanentStoreToc& aTable)
699 : iTable(aTable), iBuf(aTable.Base()), iDelta(0), iDeltaEnd(0)
701 TInt off=aTable.iTocOff;
702 __ASSERT_DEBUG(off>=KFrameNonexistent16,User::Invariant());
703 TInt ext=off<0?off:aTable.iTocExt;
704 iBuf.Set(aTable.Host(),off,ext,EFrameDescriptive16|iBuf.ERead);
707 void RPermanentStoreTocIter::Release()
712 void RPermanentStoreTocIter::ResetL()
721 RReadStream stream(&iBuf);
723 stream.ReadL((TUint8*)&h,sizeof(STocHead)); // platform dependency
724 if (h.primary<0||(h.primary&~KMaskStreamIdValue)!=0||
725 h.avail>0||(h.avail&KMaskHandleClear)!=0||(h.count&~KMaskHandleIndex)!=0)
726 __LEAVE(KErrCorrupt);
728 const CPermanentStoreToc& table=iTable;
729 if (table.HasDelta())
731 TInt c = table.iEntries.Count();
734 iDelta = &table.iEntries[0];
735 iDeltaEnd = iDelta + c;
739 iCount = iTable.iCount;
742 TBool RPermanentStoreTocIter::NextL(TEntry& anEntry)
745 __ASSERT_DEBUG(iCount>=0,User::Invariant());
748 // report TOC-base as a 'stream'
749 anEntry.handle = KHandleTocBase;
750 anEntry.ref = iTable.iTocOff;
755 __ASSERT_DEBUG(i>0,User::Invariant());
759 const TEntry* d = iDelta;
760 if (d != iDeltaEnd && (d->handle&KMaskHandleIndex) == i)
768 RReadStream stream(&iBuf);
769 TInt skip = i - iNext;
771 stream.ReadL(KSizeTocEntry * skip);
774 stream.ReadL((TUint8*)&e+KSizeHandleIndex,KSizeTocEntry); // platform dependency
775 if ((e.handle&KMaskHandleClear)!=0||e.handle>=0&&e.ref<KFrameNonexistent16||
776 e.handle<0&&(e.ref>0||(e.ref&KMaskHandleClear)!=0))
777 __LEAVE(KErrCorrupt);
779 iNext = iIndex = i+1;
780 anEntry.handle=TInt(e.handle);
781 anEntry.ref=TInt(e.ref);
785 ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
786 ///////////////////// TPermanentStoreCache //////////////////////////////////////////////////////
787 ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
789 const TPermanentStoreCache::TItem* TPermanentStoreCache::At(TInt aHandle) const
791 const TItem* item=&iItems[0];
794 while (item->handle!=aHandle)
796 __ASSERT_DEBUG(item==&iItems[i-1],User::Invariant());
797 TInt step=(aHandle&bit)?i+1:i;
805 __ASSERT_DEBUG(item==&iItems[i-1],User::Invariant());
809 void TPermanentStoreCache::Relocated(TInt aHandle,TInt anOffset)
811 const TItem* item=At(aHandle);
814 // update the cache item with the new offset
815 // As 'extent' may only be a partial extent of a fragmented frame
816 // which is no longer fragmented, we cannot simply shift this by the same
817 // amount as 'offset'. Simplest to reset this to 'unknown', i.e. 0
818 const_cast<TItem*>(item)->offset = anOffset;
819 const_cast<TItem*>(item)->extent = 0;
823 void TPermanentStoreCache::Put(const TItem* anItem,TInt anOffset,TInt anExtent)
825 TInt handle=anItem->handle;
826 TItem* item=&iItems[0];
829 while (item != anItem)
831 __ASSERT_DEBUG(item==&iItems[i-1],User::Invariant());
832 TInt step=(handle&bit)?i+1:i;
835 __ASSERT_DEBUG(i<=EItems,User::Invariant());
838 __ASSERT_DEBUG(item==&iItems[i-1],User::Invariant());
842 __ASSERT_DEBUG(item==&iItems[i-1],User::Invariant());
843 TInt step=(handle&bit)?i+1:i;
854 item->offset=anOffset;
855 item->extent=anExtent;
856 __ASSERT_DEBUG(At(handle)==item,User::Invariant());
859 void TPermanentStoreCache::Add(TInt aHandle,TInt anOffset,TInt anExtent)
861 TItem* item=&iItems[0];
866 __ASSERT_DEBUG(item==&iItems[i-1]&&item->handle!=aHandle,User::Invariant());
867 TInt step=(aHandle&bit)?i+1:i;
877 item->handle=aHandle;
878 item->offset=anOffset;
879 item->extent=anExtent;
880 __ASSERT_DEBUG(At(aHandle)==item,User::Invariant());
883 void TPermanentStoreCache::Remove(TInt aHandle)
885 TItem* item=&iItems[0];
888 while (item->handle!=aHandle)
890 __ASSERT_DEBUG(item==&iItems[i-1],User::Invariant());
891 TInt step=(aHandle&bit)?i+1:i;
901 while (item!=&iItems[0])
903 __ASSERT_DEBUG(item==&iItems[i-1],User::Invariant());
910 if (((aHandle^item->handle)&mask)==0)
918 __ASSERT_DEBUG(i==1&&At(aHandle)==NULL,User::Invariant());
921 void TPermanentStoreCache::Invalidate()
923 for (TItem* item=&iItems[0],*end=item+EItems;item<end;++item)
927 ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
928 ///////////////////// CPermanentStoreCoord //////////////////////////////////////////////////////
929 ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
931 //Returns the file system type, according to what the "file write" operation guarantees.
932 //The return result could be one of:
933 // - ESimple - "single byte write" operations are atomic;
934 // - EBlockAtomic - "block/sector write" operations are atomic;
935 // - ETransactional - transactional file system;
936 CPermanentStoreCoord::TFileQoS CPermanentStoreCoord::FileQoSL()
938 //Uncomment, if you want FileQoSL() to return always EBlockAtomic/EBlockAtomic.
939 //iFileQos = ETransactional;
940 //iFileQos = EBlockAtomic;
943 if (iFileQos == EUnknown) //get the file sytem type, if iFileQos is not set yet
945 TStreamExchange &se = Host();
946 RFileBuf *sb = static_cast<RFileBuf *>(se.HostL());
947 RFile &f = sb->File();
951 User::LeaveIfError(f.Drive(dn, di));
953 iFileQos = (di.iDriveAtt & KDriveAttTransaction) ? ETransactional : ESimple;
955 if(iFileQos == ESimple && IsBlockAtomicL(dn))
957 iFileQos = EBlockAtomic;
963 //The function returns true, if the file system guarantees atomic "block write" operations on aDriveNo.
964 //It is not the most effective implementation at the moment (creates/closes a file session),
965 //probably TDriveInfo::iType can be used in a more effective implementation.
966 TBool CPermanentStoreCoord::IsBlockAtomicL(TInt aDriveNo) const
968 __ASSERT_DEBUG(aDriveNo >= EDriveA && aDriveNo <= EDriveZ, User::Invariant());
970 CleanupClosePushL(fs);
971 User::LeaveIfError(fs.Connect());
973 TVolumeIOParamInfo volInfo;
974 TInt err = fs.VolumeIOParam(aDriveNo, volInfo);
975 CleanupStack::PopAndDestroy(&fs);
977 //If VolumeIOParam() succeeds, the media block size is >= 512 bytes and the media block size is power of two - report
978 //that the media supports atomic "block write" operations.
979 return err == KErrNone && volInfo.iBlockSize >= KDefaultMediaBlockSize && (volInfo.iBlockSize & (volInfo.iBlockSize - 1)) == 0;
982 TStreamPos CPermanentStoreCoord::LimitL()
985 return RFrame16Buf::Position(Base(),iExt);
988 TStreamId CPermanentStoreCoord::PrimaryL()
990 return TStreamId(TableL().Primary());
993 void CPermanentStoreCoord::ChangedL()
995 ConsolidateL().Changed();
998 TBool CPermanentStoreCoord::CommitL(TStreamId aPrimary)
1000 __ASSERT_DEBUG(IsTrim(),User::Invariant());
1004 CPermanentStoreToc& table=ConsolidateL();
1005 if (!table.IsVirtual())
1009 TInt toc=table.RealizeL(aPrimary.Value(),iExt);
1011 TPermanentStoreHeader header(toc);
1012 header.SetBackupToc(iToc);
1013 MStreamBuf* buf=BeginL(header);
1015 // it's done, wrap things up
1018 table.Adopt(toc,aPrimary.Value());
1021 iExt=table.Extent();
1025 TBool CPermanentStoreCoord::RevertL(TStreamId& aPrimary)
1027 __ASSERT_ALWAYS(iAccess==0,Panic(EStoreInUse));
1028 // can't revert things under people's feet
1029 CPermanentStoreToc* table=iTable;
1031 iCache.Invalidate();
1032 if (table==NULL||!table->IsVirtual())
1034 __ASSERT_DEBUG(table==NULL||aPrimary==TStreamId(table->Primary()),User::Invariant());
1039 aPrimary=TStreamId(table->Primary());
1043 TStreamExchange& host=Host();
1044 MStreamBuf* buf=host.HostL();
1051 TStreamId CPermanentStoreCoord::ExtendL()
1053 return TStreamId(ConsolidateL().AllocL(KFrameNonexistent16));
1056 void CPermanentStoreCoord::DeleteL(TStreamId anId)
1058 TInt handle=anId.Value();
1059 ConsolidateL().FreeL(handle);
1060 iCache.Remove(handle);
1063 CPermanentStoreCoord::CPermanentStoreCoord(TStreamPos aBase,TStreamExchange& aHost)
1064 : iBase(aBase),iHost(&aHost), iFileQos(EUnknown)
1068 // Read and analyse the store header.
1069 // The whole header (14 bytes) is read from the file and:
1070 // - If the dirty bit is set, the backup TOC will be used;
1071 // - If the dirty bit is not set, and the backup TOC ref is not the same as the TOC ref,
1072 // then it means the the backup TOC ref has not been written successfully, so the TOC ref will be used;
1073 void CPermanentStoreCoord::InternalizeL(RReadStream& aStream)
1076 __LEAVE(KErrNotReady);
1079 TPermanentStoreHeader header;
1080 aStream.ReadL(header.Ptr(),KPermanentStoreHeaderLength);
1081 // determine where the toc lives
1082 TInt toc=header.BackupToc();
1083 if (header.IsDirty())
1090 TInt handle=header.Handle();
1091 TInt ref=header.Reference();
1092 if (handle==0&&toc!=ref)
1093 { // toc pointer not backed up, validate as if it was
1096 header.SetBackupToc(toc);
1098 if (!header.IsValid()) // not a permanent store or damaged beyond recognition
1099 __LEAVE(KErrNotSupported);
1101 if (toc<0||((handle&~KMaskStreamIdValue)!=0&&handle!=KHandleTocBase)||ref<0||(handle!=0&&ref>=toc+KOffsetTocHeader))
1102 __LEAVE(KErrCorrupt); // integrity compromised
1108 if (iToc!=0 && iToc!=toc) // refresh produced a different toc
1109 __LEAVE(KErrCorrupt);
1113 CPermanentStoreCoord::~CPermanentStoreCoord()
1115 __ASSERT_ALWAYS(iRefs==0,Panic(EStoreInUse));
1119 void CPermanentStoreCoord::CanExtendL()
1121 __ASSERT_DEBUG(IsTrim(),User::Invariant());
1128 TInt CPermanentStoreCoord::DoCreateL()
1130 __ASSERT_DEBUG(IsTrim()&&iReloc==0&&iExtend==0,User::Invariant());
1131 TInt handle=Table().AllocL();
1138 void CPermanentStoreCoord::DoReplaceL(TInt aHandle)
1140 __ASSERT_DEBUG(IsTrim()&&iReloc==0&&iExtend==0,User::Invariant());
1141 TInt off=Table().GetL(aHandle);
1142 const TItem* item=iCache.At(aHandle);
1143 __ASSERT_DEBUG(item==NULL||iTable!=NULL&&item->handle==aHandle,User::Invariant());
1145 iCache.Add(aHandle,off,Min(off,0));
1151 TInt CPermanentStoreCoord::DoOpenL(TInt& anOffset,TInt aHandle)
1153 const TItem* item=iCache.At(aHandle);
1154 __ASSERT_DEBUG(item==NULL||iTable!=NULL&&item->handle==aHandle,User::Invariant());
1159 off=TableL().AtL(aHandle);
1160 if (iReloc==aHandle)
1168 ext=Min(off,0); // ensures ext==off for empty streams
1169 iCache.Add(aHandle,off,ext);
1182 void CPermanentStoreCoord::DoRelease(TInt aHandle,TInt anOffset,TInt anExtent)
1184 __ASSERT_DEBUG(aHandle!=0,User::Invariant());
1188 { // failed to commit the extending stream
1189 __ASSERT_DEBUG(aHandle==iExtend&&IsTrim(),User::Invariant());
1193 Table().Cancel(aHandle);
1197 const TItem* item=iCache.At(aHandle);
1198 if (item!=NULL&&item->offset==anOffset&&anExtent>item->extent)
1199 iCache.Put(item,anOffset,anExtent);
1203 TInt CPermanentStoreCoord::DoCommit(TInt aHandle,TInt anOffset,TInt anExtent)
1205 __ASSERT_DEBUG(aHandle!=0&&aHandle==iExtend&&(anExtent>=iExt||anOffset==anExtent),User::Invariant());
1206 aHandle=Table().Set(aHandle,anOffset);
1208 iCache.Remove(aHandle);
1212 const TItem* item=iCache.At(aHandle);
1214 iCache.Add(aHandle,anOffset,anExtent);
1216 iCache.Put(item,anOffset,anExtent);
1222 CPermanentStoreToc& CPermanentStoreCoord::TableL()
1224 CPermanentStoreToc* table=iTable;
1227 table=CPermanentStoreToc::NewL(Base(),Host(),iToc,iReloc==KHandleTocBase?iTarget:KFrameNonexistent16);
1228 iExt=table->Extent();
1234 CPermanentStoreToc& CPermanentStoreCoord::ConsolidateL()
1236 CPermanentStoreToc& table=TableL();
1239 table.PutL(iReloc,iTarget,CPermanentStoreToc::ETestBeforeWrite);
1240 iCache.Relocated(iReloc,iTarget);
1246 //After stream relocation, the stream entry in the TOC has to be updated with the new stream position.
1247 //If the file system is not transactional or if the file system is "block atomic", but the stream entry is split
1248 //on a block/sector boundary, then the stream handle will be stored in the permanent file store header, in case
1249 //if the TOC entry update fails.
1250 void CPermanentStoreCoord::RelocateL(TInt aHandle,TInt anOffset)
1252 __ASSERT_DEBUG(!Accessed(),User::Invariant());
1253 __ASSERT_DEBUG(iReloc==0,User::Invariant());
1255 TBool updateStoreHeader = ETrue;
1256 TFileQoS fileQos = FileQoSL();
1257 if(fileQos == ETransactional)
1259 updateStoreHeader = EFalse;
1261 else if(fileQos == EBlockAtomic)
1264 TInt writePos = iTable->RefSpan(aHandle, dataLen);
1265 __ASSERT_DEBUG(writePos >= 0 && dataLen > 0, User::Invariant());
1266 TInt startSectorAddr = writePos & ~(KDefaultMediaBlockSize - 1);
1267 TInt endSectorAddr = (writePos + dataLen - 1) & ~(KDefaultMediaBlockSize - 1);
1268 if(startSectorAddr == endSectorAddr)
1270 updateStoreHeader = EFalse;
1274 if (updateStoreHeader)
1276 TPermanentStoreHeader header(iToc,aHandle,anOffset);
1277 Host().Share(BeginL(header));
1282 iTable->PutL(aHandle,anOffset,CPermanentStoreToc::EWrite);
1283 iCache.Relocated(aHandle,anOffset);
1287 void CPermanentStoreCoord::MoveL(TInt aToc,TInt anExtent)
1289 __ASSERT_DEBUG(iReloc==0,User::Invariant());
1290 CPermanentStoreToc& table=Table();
1291 TPermanentStoreHeader header(aToc);
1292 header.SetBackupToc(iToc);
1293 Host().Share(BeginL(header));
1294 // update data structures but defer the write
1296 TInt ext=table.Extent();
1297 table.Move(aToc,anExtent);
1307 // Starts a pseudo-atomic update of the permanent file store header.
1309 // For the effect to be 'atomic', writes need to meet the following requirements:
1310 // 1. When updating n bytes using a single write, bytes 2 through n remain unchanged unless the first byte also changes.
1311 // 2. Changes associated with successive write requests happen in strict sequence.
1312 // 3. Updating a single byte is atomic.
1314 // Also, a failure to write to a location shall be reported no later than on the next
1315 // write to that or a different location, or on buffer synchronisation.
1317 // The preconditions of the operation are:
1318 // - all stream insert/delete/relocate operations completed, file - updated;
1319 // - the TOC reference in the file store header points to the current (valid) TOC, which does not include the most recent
1320 // changes, since the last commit;
1321 // - the in-memory backup TOC reference updated and made equal to the file stored TOC reference;
1323 // The procedure consists of 3 "file-write" steps:
1324 // -1- write the backup TOC ref (4 bytes) to the permanent file store header.
1325 // if this operation fails, when the store is reopened, the TOC ref will be used;
1326 // -2- set the dirty bit and write the whole file store header (14 bytes).
1327 // If this operation fails, but the dirty bit has been successfully set, the backup TOC ref will be used,
1328 // when the store is reopened;
1329 // -3- clear the dirty bit (1 byte "file write" op). The commit operation has completed successfully;
1330 MStreamBuf* CPermanentStoreCoord::BeginL(TPermanentStoreHeader& aHeader)
1332 __ASSERT_DEBUG(!aHeader.IsDirty() && aHeader.BackupToc() == Toc() && iReloc == 0, User::Invariant());
1333 MStreamBuf& buf=*Host().HostL();
1334 buf.SeekL(buf.EWrite,Base()+KPermanentStoreHeaderOffset);
1335 TFileQoS fileQos = FileQoSL();
1336 if (fileQos<EBlockAtomic)
1339 { // we've yet to write the backup toc, do that before clobbering the header
1340 buf.WriteL(aHeader.Ptr(),KPermanentStoreBackupLength);
1341 buf.SeekL(buf.EWrite,EStreamBeginning);
1342 buf.SeekL(buf.EWrite,Base()+KPermanentStoreHeaderOffset);
1345 // first write--dirty flag set in the first byte, backup toc unchanged, new header otherwise
1346 aHeader.MarkDirty();
1348 Host().Release(); // from this point onwards any failure results in store shutdown
1350 buf.WriteL(aHeader.Ptr(),KPermanentStoreHeaderLength);
1351 if (fileQos<EBlockAtomic)
1353 buf.SeekL(buf.EWrite,EStreamBeginning);
1354 buf.SeekL(buf.EWrite,Base()+KPermanentStoreHeaderOffset);
1355 aHeader.SetBackupToc(iToc);
1356 // second write--single byte write to clear the dirty flag, no change otherwise
1357 buf.WriteL(aHeader.Ptr(),1);
1359 // at this point synchronisation is atomic; ie, if successful the change
1360 // has been recorded, and failure means it will never happen
1364 ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
1365 ///////////////////// HPermanentStoreBuf ////////////////////////////////////////////////////////
1366 ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
1368 HPermanentStoreBuf* HPermanentStoreBuf::CreateL(CPermanentStoreCoord& aCoord,TStreamId& anId,TInt aMode)
1370 HPermanentStoreBuf* buf=ExtendLC(aCoord,aMode);
1371 TInt handle=aCoord.DoCreateL();
1372 buf->iHandle=handle;
1373 CleanupStack::Pop();
1374 anId=TStreamId(handle&KMaskStreamIdValue);
1378 HPermanentStoreBuf* HPermanentStoreBuf::ReplaceL(CPermanentStoreCoord& aCoord,TStreamId anId,TInt aMode)
1380 HPermanentStoreBuf* buf=ExtendLC(aCoord,aMode);
1381 TInt handle=anId.Value();
1382 aCoord.DoReplaceL(handle);
1383 buf->iHandle=handle;
1384 CleanupStack::Pop();
1388 HPermanentStoreBuf* HPermanentStoreBuf::OpenL(CPermanentStoreCoord& aCoord,TStreamId anId,TInt aMode)
1390 HPermanentStoreBuf* buf=NewLC(aCoord);
1391 TInt handle=anId.Value();
1392 TInt off=KFrameNonexistent16;
1393 TInt ext=aCoord.DoOpenL(off,handle);
1394 buf->iHandle=handle;
1396 buf->RFrame16Buf::Set(aCoord.Host(),off,ext,EFrameData16|aMode);
1398 buf->RFrame16Buf::OpenL(aCoord.Host(),off,EFrameData16|aMode);
1399 CleanupStack::Pop();
1403 HPermanentStoreBuf::~HPermanentStoreBuf()
1405 TInt handle=iHandle;
1407 Coord().DoRelease(handle,Offset(),Extent());
1408 RFrame16Buf::DoRelease();
1411 HPermanentStoreBuf* HPermanentStoreBuf::NewLC(CPermanentStoreCoord& aCoord)
1413 HPermanentStoreBuf* buf=new(ELeave) HPermanentStoreBuf(aCoord);
1418 HPermanentStoreBuf* HPermanentStoreBuf::ExtendLC(CPermanentStoreCoord& aCoord,TInt aMode)
1420 aCoord.CanExtendL();
1421 HPermanentStoreBuf* buf=NewLC(aCoord);
1422 buf->RFrame16Buf::ExtendL(aCoord.Host(),aCoord.iExt,EFrameData16|aMode);
1423 __ASSERT_DEBUG(TStreamPos(aCoord.Host().SizeL())==buf->Position(buf->Offset()),User::Invariant());
1427 void HPermanentStoreBuf::DoRelease()
1432 void HPermanentStoreBuf::DoSynchL()
1434 __ASSERT_DEBUG(iHandle!=0,User::Invariant());
1439 iHandle=Coord().DoCommit(iHandle,Offset(),Extent());