os/security/securityanddataprivacytools/securitytools/certapp/store--/s32file.cpp
Update contrib.
2 * Copyright (c) 2008-2009 Nokia Corporation and/or its subsidiary(-ies).
4 * This component and the accompanying materials are made available
5 * under the terms of the License "Eclipse Public License v1.0"
6 * which accompanies this distribution, and is available
7 * at the URL "http://www.eclipse.org/legal/epl-v10.html".
9 * Initial Contributors:
10 * Nokia Corporation - initial contribution.
20 #include "s32file_private.h"
25 #include "filestream.h"
27 using namespace store_private;
29 //The offset of the header of a permanent file store.
30 //Since all permanent file store operations work in their own coordinate system, where physical file offset 32 is
31 //logical offset 0, KPermanentStoreHeaderOffset is set with -16, which means that the physical file offset is 32 - 16 = 16.
32 const TInt KPermanentStoreHeaderOffset=-16;
34 //Permanent file store header length: sizeof(backup TOC ref) + sizeof(handle) + sizeof(TOC ref) + sizeof(crc) = 4 + 4 + 4 + 2 = 14
35 const TInt KPermanentStoreHeaderLength=14;
37 //Backup TOC ref length - 4 bytes
38 const TInt KPermanentStoreBackupLength=4;
40 const TInt KFileStoreStartOffset=sizeof(TCheckedUid);
42 const TInt KBaseOffset=KFileStoreStartOffset-KPermanentStoreHeaderOffset;
45 const TInt KMaskHandleHash=0xff000000;
46 const TInt KHandleInvalid=0x80000000;
47 const TInt KHandleTocBase=0x40000000;
48 const TInt KMaskHandleClear=0x30000000;
49 const TInt KMaskHandleGen=0x0f000000;
50 const TInt KIncHandleGen=0x01000000;
52 const TInt KMaxHandleIndex=0x00ffffff;
53 const TInt KMaskHandleIndex=0x00ffffff;
54 const TInt KSizeHandleIndex=3;
56 const TInt KTocDeltaCap = 64; //up to 64 entries in a delta TOC
57 const TInt KTocDeltaMagic = 2;
58 const TInt KMaxTocDeltaMagic = KMaxTUint16;
59 const TInt KTocDelta=KHandleInvalid;
61 const TInt KOffsetTocHeader=-12;
62 const TInt KSizeTocEntry=5; //base toc entry size is 5 bytes (when stored in the file, 8 bytes when presented in memory)
63 const TInt KSizeTocDeltaEntry=8;//delta toc entry size is 8 bytes
64 const TInt KSizeTocDeltaExtra=7;
65 const TInt KElementsTocBuf=48;
66 const TInt KBackTocBuf=20*KSizeTocEntry;
67 const TInt KSizeTocBuf=KElementsTocBuf*KSizeTocEntry;
73 EFrameDescriptive16=0x8000,
74 EFrameContinuation16=0xc000
76 typedef TUint16 TFrameDes16;
77 const TInt KSizeFrameDes16=sizeof(TFrameDes16);
78 const TInt KShiftFrameDes16=1;
80 const TInt KMaskFrame16=0xffff;
81 const TInt KMaskFrameType16=0xc000;
82 #if defined(__SMALL_FRAME)
83 const TInt KMaskFrameLength16=0xf;
84 const TInt KShiftFrameLength16=4;
86 const TInt KMaskFrameLength16=0x3fff;
87 const TInt KShiftFrameLength16=14;
90 const TInt KFrameOpen16=0;
91 const TInt KFrameMinLength16=1;
92 const TInt KFrameMaxLength16=KMaskFrameLength16;
93 const TInt KFrameFullLength16=KMaskFrameLength16+1;
95 const TInt KFrameNonexistent16=-1;
97 TUint32 frame2File(TUint32 anOffset)
99 // The frame offset appears to be to calculated as an offset into
100 // a series of maximal length frames. Each maximal frame can hold
101 // 0x4000 bytes of data and has a 2 byte header. Therefore a frame
102 // offset of x needs an extra 2 bytes for each maximal frame. The
103 // base is the first byte of the payload of the first frame, so we
104 // do not need to allow for the header of the first frame.
106 // anOffset>>KShiftFrameLength16 gives the number of maximal
107 // frames. The left shift multiples by 2 to give minimum space
108 // required for frame headers. Adding to the original gives the
111 // This logic is still used even if space for the target frame is
112 // really ocupied by a series of smaller frames.
113 anOffset += (anOffset>>KShiftFrameLength16<<KShiftFrameDes16);
115 // Add 32 to allow for file header
116 anOffset += KBaseOffset;
120 TUint32 file2Frame(TUint32 aFileOffset)
122 // subtract 32 to allow for file header
123 aFileOffset -= KBaseOffset;
125 // File offset is to the first byte of payload of a stream. We
126 // need to remove the 2 byte frame header for the (theoretical)
127 // maximal length frames before it.
129 // Calculate the number of (theoretical) maximal length frames before the file position
130 // n = aFileOffset/0x4002
131 TInt n=aFileOffset/(KFrameFullLength16+KSizeFrameDes16);
133 // Subtract the frame header length for each maximal frame
134 aFileOffset -= n<<KShiftFrameDes16;
141 //TPermanentStoreHeader class.
143 //Represents the data kept in the permanent store file header.
145 // - iBackup - "backup TOC reference", 32-bits integer, which keeps the 31-bit file offset of the backup TOC.
146 // Plays important role in the "store commit" procedure.
147 // The LSB is a "dirty" bit. If during the store opening phase the dirty bit is found to be set,
148 // then it means - the previous "store commit" operation has not been completed successfully and
149 // the backup TOC shall be used instead of the TOC;
150 // - iHandle - 32-bit stream handle (MSB - invalid/deleted, 3 bits - unused, 4 bits - generation counter, 24 bits - stream handle).
151 // Plays important role in the "stream relocation" procedure during store compaction.
152 // iHandle keeps the handle of the stream being relocated, so if the commit phase fails, the original stream entry
153 // can be restored at the moment when the store is reopened;
154 // - iRef - Current "TOC reference". Represents a file offset, where the current TOC is;
155 // - iCrc - 16-bit CRC, protecting iBackup, iHandle, iRef;
156 class TPermanentStoreHeader
159 TPermanentStoreHeader() : iBackup(0), iHandle(0), iRef(0), iCrc(0) {}
161 const TUint8* Ptr() const;
162 TBool IsValid() const;
164 TPermanentStoreHeader(TInt aToc);
165 TPermanentStoreHeader(TInt aBackupToc,TInt aHandle,TInt aReference);
167 TBool IsDirty() const;
169 void SetBackupToc(TInt aBackupToc);
170 TInt BackupToc() const;
173 TInt Reference() const;
175 void Set(TInt aBackupToc,TInt aHandle,TInt aReference);
183 void InternalizeL(RReadStream &aStream);
184 void ExternalizeL(RWriteStream &aStream) const;
187 // Class TPermanentStoreHeader
188 TUint8* TPermanentStoreHeader::Ptr()
189 {return REINTERPRET_CAST(TUint8*,&iBackup);}
190 const TUint8* TPermanentStoreHeader::Ptr() const
191 {return REINTERPRET_CAST(const TUint8*,&iBackup);}
194 TPermanentStoreHeader::TPermanentStoreHeader(TInt aToc)
197 TPermanentStoreHeader::TPermanentStoreHeader(TInt aBackupToc,TInt aHandle,TInt aReference)
199 __ASSERT_DEBUG(aHandle!=0,User::Invariant());
200 Set(aBackupToc,aHandle,aReference);
203 TBool TPermanentStoreHeader::IsDirty() const
208 void TPermanentStoreHeader::MarkDirty()
213 void TPermanentStoreHeader::SetBackupToc(TInt aBackupToc)
215 __ASSERT_DEBUG(aBackupToc>=0,User::Invariant());
216 iBackup=TUint32(aBackupToc)<<1;
219 TInt TPermanentStoreHeader::BackupToc() const
224 TInt TPermanentStoreHeader::Handle() const
228 TInt TPermanentStoreHeader::Reference() const
234 TBool TPermanentStoreHeader::IsValid() const
240 Mem::Crc(crc,Ptr(),_FOFF(TPermanentStoreHeader,iCrc));
245 //Sets the "backup TOC ref", "handle" and "TOC ref" in current TPermanentStoreHeader object.
246 //16-bit CRC is calculated, based on the values of the input parameters, and stored together with them in the
247 //TPermanentStoreHeader object.
248 void TPermanentStoreHeader::Set(TInt aBackupToc,TInt aHandle,TInt aReference)
250 iBackup=TUint32(aBackupToc)<<1;
254 Mem::Crc(iCrc,Ptr(),_FOFF(TPermanentStoreHeader,iCrc));
255 __ASSERT_DEBUG(IsValid(),User::Invariant());
259 void TPermanentStoreHeader::InternalizeL(RReadStream &aStream)
267 void TPermanentStoreHeader::ExternalizeL(RWriteStream &aStream) const
280 void ExternalizeL(RWriteStream &aStream) const;
281 void InternalizeL(RReadStream &aStream);
284 void TocHeader::ExternalizeL(RWriteStream &aStream) const
291 void TocHeader::InternalizeL(RReadStream &aStream)
302 void ExternalizeL(RWriteStream &aStream) const;
303 void InternalizeL(RReadStream &aStream);
307 void STocEntry::ExternalizeL(RWriteStream &aStream) const
309 aStream.WriteL((TUint8*)&handle + KSizeHandleIndex, KSizeTocEntry);
312 void STocEntry::InternalizeL(RReadStream &aStream)
314 aStream.ReadL((TUint8*)&handle + KSizeHandleIndex, KSizeTocEntry);
318 CStreamStore::~CStreamStore()
323 TStreamId CPersistentStore::Root()
328 void CPersistentStore::SetRootL(TStreamId anId)
333 void CFileStore::SetTypeL(const TUidType& aType)
335 if(aType != KPermanentFileStoreLayoutUid) FatalError();
339 CFileStore *CPermanentFileStore::ReplaceLC(RFs &aFs, const TDesC &aName, TUint aFileMode)
342 return new CPermanentFileStore(aName, aFileMode);
345 CPermanentFileStore* CPermanentFileStore::OpenLC(RFs& aFs,const TDesC& aName,TUint aFileMode)
348 CPermanentFileStore *pfs = new CPermanentFileStore(aName, aFileMode);
353 CPermanentFileStore::CPermanentFileStore(const TDesC& aName,TUint aFileMode)
354 : iWritable(false), iFileIn(), iFileOut(), iLastStreamIdCreated(0),
357 TBuf8<KMaxName> utf8Name;
358 utf8Name.Copy(aName);
360 std::string name((const char *)utf8Name.Ptr(), utf8Name.Length());
362 if(aFileMode & EFileWrite)
364 iFileOut.reset(new FileWriteStream(name));
369 iFileIn.reset(new FileReadStream(name));
373 CPermanentFileStore::~CPermanentFileStore()
375 while(!iStreams.empty())
377 MemStreamBuf *p = (*iStreams.begin()).second;
378 iStreams.erase(iStreams.begin());
384 MemStreamBuf *CPermanentFileStore::CreateStoreWriteStream()
386 TStreamId id = ++iLastStreamIdCreated;
387 MemStreamBuf *handler = new MemStreamBuf(*this, id);
389 MemStreamBufPtrMap::value_type value(id, handler);
390 std::pair<MemStreamBufPtrMap::iterator, bool> ret = iStreams.insert(value);
402 MemStreamBuf *CPermanentFileStore::CreateStoreReadStream(TStreamId aId) const
404 MemStreamBufPtrMap::const_iterator it = iStreams.find(TUint32(aId));
405 if(it == iStreams.end())
408 dbg << Log::Indent() << "StreamId 0x" << aId << " does not exist in store!" << Log::Endl();
414 bool CPermanentFileStore::Writable() const
421 void CPermanentFileStore::CommitL()
423 if(!iWritable) FatalError();
425 TCheckedUid id(KPermanentFileStoreLayoutUid);
429 // Write an empty header, we will correct the fields later
430 TPermanentStoreHeader tmpheader(0);
431 *iFileOut << tmpheader;
434 // Now write out the streams
436 std::vector<TUint> streamOffsets;
437 for(MemStreamBufPtrMap::const_iterator streamIt = iStreams.begin();
438 streamIt != iStreams.end();
441 const MemStreamBuf::StreamData &streamData((*streamIt).second->GetData());
443 // Record the offset to the first byte of the payload for use in the TOC
444 streamOffsets.push_back(TUint32(iFileOut->iSnk->TellL(MStreamBuf::EWrite))+KSizeFrameDes16);
446 TUint32 dataAvailable = streamData.size();
447 MemStreamBuf::StreamData::const_iterator dataIt = streamData.begin();
448 TUint32 currentType = EFrameData16;
454 TUint32 fileOffset = TUint32(iFileOut->iSnk->TellL(MStreamBuf::EWrite))+KSizeFrameDes16;
455 TUint32 frameOffset = file2Frame(fileOffset);
456 TUint32 nextAnchor = ((frameOffset + KFrameFullLength16) & ~KMaskFrameLength16);
457 if(dataAvailable >= (nextAnchor-frameOffset))
459 // We have enough data to write an "Open" frame to
460 // take the frame offset up to the next multiple of
462 toWrite = (nextAnchor-frameOffset);
463 frame = currentType | KFrameOpen16;
467 // Write a normal frame
468 toWrite = (dataAvailable < TUint32(KMaskFrameLength16)) ? (dataAvailable) : (KMaskFrameLength16);
469 frame = currentType | toWrite;
477 *iFileOut << *dataIt;
481 // Set type of next frame to continuation
482 currentType = EFrameContinuation16;
489 TInt tocLength = sizeof(TocHeader) + (iStreams.size() * KSizeTocEntry);
490 // Check toc frame length is not too long
491 if(tocLength > KMaskFrameLength16)
496 TFrameDes16 frame = tocLength | EFrameDescriptive16;
499 TStreamPos pos = iFileOut->iSnk->TellL(MStreamBuf::EWrite);
502 TUint32 tocFrameOffset = file2Frame(pos);
503 TUint32 tocCookedFrameOffset = tocFrameOffset - KOffsetTocHeader;
506 tocheader.iPrimary = iRoot;
507 tocheader.iAvail = 0;
508 tocheader.iCount = iStreams.size();
509 *iFileOut << tocheader;
511 // Now write the TOC entries
512 for(TUint i = 0; i < streamOffsets.size(); ++i)
514 STocEntry tocEntry = { 0, file2Frame(streamOffsets[i]) };
515 *iFileOut << tocEntry;
520 // Now overwrite the header with one containing the correct master
523 TPermanentStoreHeader realheader(tocCookedFrameOffset);
524 // For some reason the real store code creates a header with a
525 // checksum which assumes iBackup == iRef but the iBackup value on
526 // disk is set to 0...
527 realheader.SetBackupToc(0);
528 iFileOut->iSnk->SeekL(MStreamBuf::EWrite, EStreamBeginning, sizeof(TCheckedUid));
529 *iFileOut << realheader;
535 void CPermanentFileStore::ReadInData()
537 prog << Log::Indent() << "Reading store" << Log::Endl();
540 TUint lastStreamIndex = 0;
545 if(id.UidType() != KPermanentFileStoreLayoutUid)
547 dbg << Log::Indent() << "Invalid store" << Log::Endl();
551 TPermanentStoreHeader header;
555 // - If the dirty bit is set, the backup TOC will be used;
556 // - If the dirty bit is not set, and the backup TOC ref is not the same as the TOC ref,
557 // then it means the the backup TOC ref has not been written successfully, so the TOC ref will be used;
558 TInt toc=header.BackupToc();
559 if (!header.IsDirty())
561 prog << Log::Indent() << "Store header is clean" << Log::Endl();
562 TInt handle=header.Handle();
563 TInt ref=header.Reference();
564 if (handle==0&&toc!=ref)
565 { // toc pointer not backed up, validate as if it was
568 header.SetBackupToc(toc);
570 if (!header.IsValid()) // not a permanent store or damaged beyond recognition
572 dbg << Log::Indent() << "Store header checksum failed check1" << Log::Endl();
573 __LEAVE(KErrNotSupported);
577 if (toc<0||((handle&~KMaskStreamIdValue)!=0&&handle!=KHandleTocBase)||ref<0||(handle!=0&&ref>=toc+KOffsetTocHeader))
579 dbg << Log::Indent() << "Store header fields invalid" << Log::Endl();
580 __LEAVE(KErrCorrupt); // integrity compromised
588 dbg << Log::Indent() << "Store header dirty" << Log::Endl();
592 // if (iToc!=0 && iToc!=toc) // refresh produced a different toc
593 // __LEAVE(KErrCorrupt);
597 if(!header.IsValid())
599 dbg << Log::Indent() << "Store header checksum failed check2" << Log::Endl();
603 // All offset values are to the payload of frame structures (see RFrame16Buf)
604 // Convert to a file offset.
605 TInt tocFileOffset = frame2File(toc + KOffsetTocHeader);
606 prog << Log::Indent() << "TOC file offset 0x" << tocFileOffset << Log::Endl();
608 // nb. We do not validate the length of the "descriptive" stream
609 // which contains the TocHeader (aka STocHead) and the table of stream offsets.
611 iFileIn->iSrc->SeekL(MStreamBuf::ERead, EStreamBeginning, tocFileOffset);
614 if ((h.iPrimary&~(KMaskStreamIdValue|static_cast<TUint>(KTocDelta)))!=0||
615 h.iAvail>0||(h.iAvail&KMaskHandleClear)!=0||(h.iCount&~KMaskHandleIndex)!=0)
618 dbg << Log::Indent() << "Corrupt TOC in input file" << Log::Endl();
621 std::set<TUint32> streamIdsReadExGenField; // IDs of streams already read, EXCLUDING generation field
622 if (h.iPrimary & KTocDelta)
625 prog << Log::Indent() << "Found Delta TOC" << Log::Endl();
628 // Find and read in the delta entries and then continue to read in the base entries
629 // Read offset to base TOC
630 iFileIn->iSrc->ReadL(&toc, sizeof(toc));
631 // All offset values are to the payload of frame structures (see RFrame16Buf)
632 // Convert to a file offset.
633 tocFileOffset = frame2File(toc + KOffsetTocHeader);
635 if((tocFileOffset < 0) || (tocFileOffset>oldToc))
637 dbg << Log::Indent() << "Invalid TOC in delta TOC" << Log::Endl();
638 FatalError(); // Corrupt
641 // Read/discard the magic value - Not sure what this should be
643 iFileIn->iSrc->ReadL(&magic, sizeof(magic));
645 // Read count of delta toc entries
647 iFileIn->iSrc->ReadL(&n, sizeof(n));
649 prog << Log::Indent() << "Delta TOC entries " << int(n) << Log::Endl();
650 for(int deltaTocEntry=1; deltaTocEntry <= n; ++deltaTocEntry)
653 prog << Log::Indent() << "Delta TOC entry " << deltaTocEntry << Log::Endl();
656 // Read delta toc entry.
658 // The main TOC entries are truncated to 5 bytes so only
659 // include the flag bits of the handle and not the stream.
660 // They gain streamids starting from 1.
662 // The delta TOC contains complete stream IDs, and these
663 // entries take precedence over the matching (implicitly
664 // numbered) entry in the base TOC.
665 iFileIn->iSrc->ReadL(&e, sizeof(e));
667 if ((e.handle&KMaskHandleClear)!=0 ||
668 e.handle>=0 && (e.ref<KFrameNonexistent16) ||
669 e.handle<0 && (e.ref>0 || (e.ref&KMaskHandleClear)!=0))
671 dbg << Log::Indent() << "Invalid delta TOC entry" << Log::Endl();
672 __LEAVE(KErrCorrupt);
676 if(e.handle & KHandleInvalid)
678 // Stream marked as deleted, skip it
679 prog << Log::Indent() << "Delta TOC entry marked as deleted" << Log::Endl();
683 // We do not deal with any other special cases correctly
684 if(((e.handle&(KMaskHandleHash-KMaskHandleGen)) != 0) || (e.ref < 0))
686 dbg << Log::Indent() << "store-- does not support handle flags 0x" << e.handle << "in delta toc" << Log::Endl();
690 prog << Log::Indent()
691 << "streamId 0x" << e.handle << " frame offset 0x" << e.ref
692 << " file offset 0x" << frame2File(e.ref) << Log::Endl();
694 if(!streamIdsReadExGenField.insert(e.handle & ~KMaskHandleHash).second)
696 dbg << Log::Indent() << "store-- corrupt input? Multiple entries for stream ID " << (e.handle & ~KMaskHandleHash) << "encountered in delta toc!" << Log::Endl();
701 // Remember where we are...
702 TStreamPos here = iFileIn->iSrc->TellL(MStreamBuf::ERead);
703 // Read the stream frame
704 ReadStreamData(e.handle, e.ref);
705 // restore position to continue reading delta toc
706 iFileIn->iSrc->SeekL(MStreamBuf::ERead, EStreamBeginning, here);
710 iRoot = TInt(h.iPrimary) & ~KTocDelta;
711 prog << Log::Indent() << "Root stream ID is " << iRoot << Log::Endl();
713 prog << Log::Indent() << "Processing main TOC at file offset 0x" << tocFileOffset << Log::Endl();
715 for(TUint tocIndex=0; tocIndex < h.iCount; ++tocIndex)
718 TStreamId baseStreamId = ++lastStreamIndex;
719 prog << Log::Indent() << "Entry " << std::dec << baseStreamId << std::hex << Log::Endl();
722 // Check if we have already read in this stream id from the delta toc
723 // Note we only compare the base of the stream ID excluding the generation field.
724 if(streamIdsReadExGenField.find(baseStreamId) != streamIdsReadExGenField.end())
726 prog << Log::Indent() << "Already read from delta TOC" << Log::Endl();
727 continue; // Already read from delta toc
730 // Seek to the STocEntry for the stream
731 iFileIn->iSrc->SeekL(MStreamBuf::ERead, EStreamBeginning, tocFileOffset+sizeof(TocHeader)+KSizeTocEntry*(tocIndex));
735 // Only the last 5 bytes of the entry are saved to disk.
736 // Because we are little endian this is the MSB of the handle and the whole ref.
739 if ((e.handle&KMaskHandleClear)!=0 ||
740 e.handle>=0 && (e.ref<KFrameNonexistent16) ||
741 e.handle<0 && (e.ref>0 || (e.ref&KMaskHandleClear)!=0))
743 __LEAVE(KErrCorrupt);
747 if(e.handle & KHandleInvalid)
749 // Stream deleted, skip it
750 prog << Log::Indent() << "Marked as deleted" << Log::Endl();
754 // We do not deal with any other special cases correctly
755 if(((e.handle & (KMaskHandleHash-KMaskHandleGen)) != 0) || (e.ref < 0))
757 dbg << Log::Indent() << "store-- does not support handle flags 0x" << e.handle << "in main toc" << Log::Endl();
761 TStreamId streamId = baseStreamId | e.handle;
762 if(!streamIdsReadExGenField.insert(baseStreamId).second)
764 dbg << Log::Indent() << "store-- internal error for stream ID " << baseStreamId << "encountered in delta toc!" << Log::Endl();
768 // Read the stream frame
769 // prog << Log::Indent() << "Reading stream 0x" << streamId << " frame offset 0x" << e.ref << Log::Endl();
770 ReadStreamData(streamId, e.ref);
775 void CPermanentFileStore::ReadStreamData(TStreamId aStreamId, TInt aFrameOffset)
777 TInt fileOffset = frame2File(aFrameOffset);
779 iFileIn->iSrc->SeekL(MStreamBuf::ERead, EStreamBeginning, fileOffset - sizeof(frame));
781 // Frame runs from fileOffset to fileOffset+2+frame
782 std::auto_ptr<MemStreamBuf> apStream(new MemStreamBuf(*this, aStreamId));
785 EFirst, EContinuation, ELast
788 while(state != ELast)
790 iFileIn->iSrc->ReadL(&frame, sizeof(frame));
792 // Store uses offsets to the first byte of the payload of a frame, so that is what we display
793 TUint32 fileOffset = iFileIn->iSrc->TellL(MStreamBuf::ERead);
794 prog << Log::Indent() << "ReadStreamData State " << int(state) << " offset " << file2Frame(fileOffset) << "[" << fileOffset << "]" << Log::Endl();
796 TUint32 frameType = (frame & KMaskFrameType16);
797 if(frameType == EFrameFree16)
799 dbg << Log::Indent() << "Encountered a deleted frame in data stream" << Log::Endl();
803 if((frameType == EFrameData16) && (state != EFirst))
805 dbg << Log::Indent() << "Non-continuation frame encountered in middle of data stream" << Log::Endl();
809 if(frameType == EFrameDescriptive16)
811 dbg << Log::Indent() << "Descriptive frame encountered in data stream" << Log::Endl();
815 if((frameType == EFrameContinuation16) && (state != EContinuation))
817 dbg << Log::Indent() << "Continuation frame encountered at start of stream" << Log::Endl();
821 TInt len = (frame & KMaskFrameLength16);
822 if (len == KFrameOpen16)
824 prog << Log::Indent() << "\"Open\" frame length" << Log::Endl();
825 len = KFrameFullLength16 - file2Frame(fileOffset);
826 state = EContinuation;
833 prog << Log::Indent() << "Type 0x" << frameType << " length " << std::dec << len << std::hex << Log::Endl();
834 apStream->AppendFromFile(iFileIn->iSrc, len);
837 MemStreamBufPtrMap::value_type value(aStreamId, apStream.get());
838 std::pair<MemStreamBufPtrMap::iterator, bool> ret = iStreams.insert(value);
848 RReadStream::RReadStream()
853 void RReadStream::Close()
858 RStoreReadStream::RStoreReadStream()
864 void RStoreReadStream::OpenLC(const CStreamStore& aStore, TStreamId aId)
866 iSrc = aStore.CreateStoreReadStream(aId);
871 RWriteStream::RWriteStream()
880 TStreamId RStoreWriteStream::CreateLC(CStreamStore& aStore)
882 MemStreamBuf *s = aStore.CreateStoreWriteStream();
884 return s->StreamId();
887 EXPORT_C void RReadStream::ReadL(TDes8& aDes)
888 /** Reads sufficient data from this stream to fill the specified 8 bit descriptor up to its maximum length.
889 No other information is read from this read stream.
891 @param aDes A reference to a modifiable descriptor which is to receive the data read from this stream. Passing the build
892 independent type TDes& allows the compiler to choose the appropriate ReadL() variant (i.e the 8 bit or the 16 bit) at build time.*/
894 ReadL(aDes,aDes.MaxLength());
897 EXPORT_C void RReadStream::ReadL(TDes8& aDes,TInt aLength)
898 /** Reads data of specified length from this stream into the specified 8 bit descriptor. No other information is read
901 @param aDes A reference to a modifiable type descriptor which is to receive the data read from this stream.
902 Passing the build independent type TDes& allows the compiler to choose the appropriate ReadL() variant (i.e the 8 bit
903 or the 16 bit) at build time.
904 @param aLength The length of data to be read from this stream. This value must be non-negative and must not be greater than the maximum length of the descriptor otherwise the function raises a USER 11 panic.*/
906 __ASSERT_DEBUG(aLength<=aDes.MaxLength(),Panic(EStreamReadBeyondEnd));
907 aDes.SetLength(aLength);
908 ReadL((TUint8*)aDes.Ptr(),aLength);
912 EXPORT_C void RReadStream::ReadL(TDes8& aDes,TChar aDelim)
913 /** Reads data from this stream into the 8 bit descriptor, until either the specified delimiter is encountered or the descriptor is filled to its maximum length.
914 The resulting data in aDes always includes the delimiter aDelim, if aDes is large enough.
916 @param aDes A reference to a modifiable type descriptor which is to receive the data read from this stream. Passing
917 the build independent type TDes& allows the compiler to choose the appropriate ReadL() variant (i.e the 8 bit or the 16 bit) at build time.
918 @param aDelim The delimiter marking the end of the data in the stream.*/
921 __ASSERT_DEBUG(iSrc!=NULL,Panic(EStreamNotOpen));
922 TUint8* ptr=(TUint8*)aDes.Ptr();
923 TDelimitedInput8 input(ptr,aDes.MaxLength(),aDelim);
927 } while (!input.Done());
928 aDes.SetLength(input.Ptr()-ptr);
932 EXPORT_C void RReadStream::ReadL(TUint8* aPtr,TInt aLength)
933 /** Reads data of specified length from this stream into the location defined by the specified TUint8 pointer.
935 @param aPtr The target location for the streamed in data.
936 @param aLength The length of data to be streamed in.*/
938 __ASSERT_DEBUG(aLength>=0,Panic(EStreamReadLengthNegative));
942 __ASSERT_DEBUG(iSrc!=NULL,Panic(EStreamNotOpen));
943 TInt len=iSrc->ReadL(aPtr,aLength);
944 __ASSERT_DEBUG(len>=0&&len<=aLength,Panic(EStreamReadInBreach));
949 EXPORT_C void RReadStream::ReadL(TInt aLength)
950 /** Discards data of specified length read from this stream.
952 @param aLength The length of data to be discarded from this read stream.*/
954 __ASSERT_DEBUG(aLength>=0,Panic(EStreamReadLengthNegative));
958 __ASSERT_DEBUG(iSrc!=NULL,Panic(EStreamNotOpen));
961 iSrc->SeekL(MStreamBuf::ERead, EStreamMark, aLength);
964 TInt len=iSrc->ReadL(input,aLength);
965 __ASSERT_DEBUG(len>=0&&len<=aLength,Panic(EStreamReadInBreach));
971 EXPORT_C void RReadStream::ReadL(TDes16& aDes)
972 /** Reads sufficient data from this stream to fill the specified 16 bit descriptor up to its maximum length.
973 No other information is read from this read stream.
975 @param aDes A reference to a modifiable type descriptor which is to receive the data read from this stream. Passing
976 the build independent type TDes& allows the compiler to choose the appropriate ReadL() variant (i.e the 8 bit or the 16
977 bit) at build time.*/
979 ReadL(aDes,aDes.MaxLength());
982 EXPORT_C void RReadStream::ReadL(TDes16& aDes,TInt aLength)
983 /** Reads data of specified length from this stream into the specified 16 bit descriptor. No other information is read from this stream.
985 @param aDes A reference to a modifiable type descriptor which is to receive the data read from this stream. Passing the
986 build independent type TDes& allows the compiler to choose the appropriate ReadL() variant (i.e the 8 bit or the 16 bit)
988 @param aLength The length of data to be read from this stream. This value must be non-negative and must not be greater
989 than the maximum length of the descriptor otherwise the function raises a USER 11 panic.*/
991 __ASSERT_DEBUG(aLength<=aDes.MaxLength(),Panic(EStreamReadBeyondEnd));
992 aDes.SetLength(aLength);
993 ReadL((TUint16*)aDes.Ptr(),aLength);
997 EXPORT_C void RReadStream::ReadL(TDes16& aDes,TChar aDelim)
998 /** Reads data from this stream into the 16 bit descriptor, until either the specified delimiter is encountered or
999 the descriptor is filled to its maximum length.
1000 The resulting data in aDes always includes the delimiter aDelim, if aDes is large enough.
1002 @param aDes A reference to a modifiable type descriptor which is to receive the data read from this stream. Passing
1003 the build independent type TDes& allows the compiler to choose the appropriate ReadL() variant (i.e the 8 bit or the 16 bit) at build time.
1004 @param aDelim The delimiter marking the end of the data in the stream.*/
1006 __ASSERT_DEBUG(iSrc!=NULL,Panic(EStreamNotOpen));
1007 TUint16* ptr=(TUint16*)aDes.Ptr();
1008 TDelimitedInput16 input(ptr,aDes.MaxLength(),aDelim);
1012 } while (!input.Done());
1013 aDes.SetLength(input.Ptr()-ptr);
1017 EXPORT_C void RReadStream::ReadL(TUint16* aPtr,TInt aLength)
1018 /** Reads data of specified length from this stream into the specified 16 bit descriptor.
1019 No other information is read from this stream.
1021 @param aDes A reference to a modifiable type descriptor which is to receive the data read from this stream. Passing the build independent type TDes& allows the compiler to choose the appropriate ReadL() variant (i.e the 8 bit or the 16 bit) at build time.
1022 @param aLength The length of data to be read from this stream. This value must be non-negative and must not be greater than the maximum length of the descriptor otherwise the function raises a USER 11 panic.*/
1024 __ASSERT_DEBUG(aLength>=0,Panic(EStreamReadLengthNegative));
1025 ReadL((TUint8*)aPtr,aLength<<1); // platform dependency
1028 EXPORT_C TInt8 RReadStream::ReadInt8L()
1029 /** Internalises a TInt8 value The function reads an 8Â bit value from this stream
1030 and interprets it as a TInt8.
1032 @return The 8 bit value read from this stream. */
1035 ReadL((TUint8*)&val,1); // platform dependency
1039 EXPORT_C TInt16 RReadStream::ReadInt16L()
1040 /** Internalises a TInt16 value. The function reads a 16Â bit value from this stream
1041 and interprets it as a TInt16.
1043 @return The 16 bit value read from this stream. */
1046 ReadL((TUint8*)&val,2); // platform dependency
1050 EXPORT_C TInt32 RReadStream::ReadInt32L()
1051 /** Internalises a TInt32 value. The function reads a 32Â bit value from this stream
1052 and interprets it as a TInt32.
1054 @return The 32Â bit value read from this stream. */
1057 ReadL((TUint8*)&val,4); // platform dependency
1061 EXPORT_C TUint8 RReadStream::ReadUint8L()
1062 /** Internalises a TUint8 value. The function reads an 8Â bit value from this stream
1063 and interprets it as a TUint8.
1065 @return The 8Â bit value read from this stream. */
1072 EXPORT_C TUint16 RReadStream::ReadUint16L()
1073 /** Internalises a TUint16 value. The function reads a 16Â bit value from this
1074 stream and interprets it as a TUint16.
1076 @return The 16Â bit value read from this stream. */
1079 ReadL((TUint8*)&val,2); // platform dependency
1083 EXPORT_C TUint32 RReadStream::ReadUint32L()
1084 /** Internalises a TUint32 value. The function reads a 32Â bit value from this
1085 stream and interprets it as a TUint32.
1087 @return The 32Â bit value read from this stream. */
1090 ReadL((TUint8*)&val,4); // platform dependency
1095 EXPORT_C void RWriteStream::Close()
1096 /** Commits data to the stream before freeing resources used by the stream. This
1097 ensures that any buffered data is written to the stream.
1099 Note that the function cannot leave. Any errors arising from the attempt to
1100 commit data to the stream are ignored. */
1109 EXPORT_C void RWriteStream::Release()
1110 /** Frees resources before abandoning the stream. The function is called after
1111 data has been committed to the stream.
1113 Note that if a cleanup item for the stream was placed on the cleanup stack
1114 when the stream was opened (e.g by a call to RStoreWriteStreamss CreateLC(),
1115 OpenLC(), ReplaceLC() or RDictionaryStores AssignLC() etc), then this function
1116 need not be called explicitly; clean up is implicitly done by CleanupStack::PopAndDestroy(). */
1125 EXPORT_C void RWriteStream::CommitL()
1126 /** Ensures that any buffered data is written to the stream. Once committed, it
1127 is not possible to roll back the newly written data. */
1136 EXPORT_C void RWriteStream::WriteL(const TDesC8& aDes)
1137 /** Writes the content of the 8 bit descriptor to the stream. No other information
1138 is written to this write stream.
1140 @param aDes A reference to a descriptor. Passing the build independent type
1141 TDesC& allows the compiler to choose the appropriate WriteL() variant (i.e
1142 the 8 bit or the 16 bit) at build time. */
1144 WriteL(aDes.Ptr(),aDes.Length());
1147 EXPORT_C void RWriteStream::WriteL(const TDesC8& aDes,TInt aLength)
1148 /** Writes data of the specified length from the 8 bit descriptor to the stream.
1149 No other information is written to this write stream.
1151 @param aDes A reference to a descriptor. Passing the build independent type
1152 TDesC& allows the compiler to choose the appropriate WriteL() variant (i.e
1153 the 8 bit or the 16 bit) at build time.
1154 @param aLength The length of data to be written to this stream. */
1157 __ASSERT_DEBUG(aLength<=aDes.Length(),Panic(EStreamWriteBeyondEnd));
1158 WriteL(aDes.Ptr(),aLength);
1161 EXPORT_C void RWriteStream::WriteL(const TUint8* aPtr,TInt aLength)
1162 /** Writes 8 bit data of the specified length from the specified location to this
1165 @param aPtr The location from where data is to be streamed out.
1166 @param aLength The length of data to be streamed out. */
1169 __ASSERT_DEBUG(aLength>=0,Panic(EStreamWriteLengthNegative));
1173 __ASSERT_DEBUG(iSnk!=NULL,Panic(EStreamNotOpen));
1174 iSnk->WriteL(aPtr,aLength);
1178 EXPORT_C void RWriteStream::WriteL(RReadStream &aStream)
1179 /** Writes the content of the specified read stream to this write stream.
1181 @param aStream A reference to a read stream which is to be written to this
1185 __ASSERT_DEBUG(iSnk!=NULL,Panic(EStreamNotOpen));
1186 TSourceOutput output(aStream.iSrc);
1187 iSnk->WriteL(output);
1192 EXPORT_C void RWriteStream::WriteL(RReadStream& aStream,TInt aLength)
1193 /** Writes data of the specified length from the specified read stream to this
1196 @param aStream A reference to a read stream part of whose content is to be
1197 written to this stream.
1198 @param aLength The length of data from the read stream to be written to this
1202 __ASSERT_DEBUG(aLength>=0,Panic(EStreamWriteLengthNegative));
1206 __ASSERT_DEBUG(iSnk!=NULL,Panic(EStreamNotOpen));
1207 TSourceOutput output(aStream.iSrc);
1208 TInt len=iSnk->WriteL(output,aLength);
1209 __ASSERT_DEBUG(len>=0&&len<=aLength,Panic(EStreamReadInBreach));
1215 EXPORT_C void RWriteStream::WriteL(const TDesC16& aDes)
1216 /** Writes the content of the 16 bit descriptor to the stream. No other information
1217 is written to this write stream.
1219 @param aDes A reference to a descriptor. Passing the build independent type
1220 TDesC& allows the compiler to choose the appropriate WriteL() variant (i.e
1221 the 8 bit or the 16 bit) at build time. */
1224 WriteL(aDes.Ptr(),aDes.Length());
1227 EXPORT_C void RWriteStream::WriteL(const TDesC16& aDes,TInt aLength)
1228 /** Writes data of the specified length from the 16 bit descriptor to the stream.
1229 No other information is written to this write stream.
1231 @param aDes A reference to a descriptor. Passing the build independent type
1232 TDesC& allows the compiler to choose the appropriate WriteL() variant (i.e
1233 the 8 bit or the 16 bit) at build time.
1234 @param aLength The length of data to be written to this stream. */
1237 __ASSERT_DEBUG(aLength<=aDes.Length(),Panic(EStreamWriteBeyondEnd));
1238 WriteL(aDes.Ptr(),aLength);
1241 EXPORT_C void RWriteStream::WriteL(const TUint16* aPtr,TInt aLength)
1242 /** Writes 16 bit data of the specified length from the specified location to this
1245 @param aPtr The location from where data is to be streamed out.
1246 @param aLength The length of data to be streamed out. */
1249 __ASSERT_DEBUG(aLength>=0,Panic(EStreamWriteLengthNegative));
1250 WriteL((const TUint8*)aPtr,aLength<<1); // platform dependency
1253 EXPORT_C void RWriteStream::WriteInt8L(TInt aValue)
1254 /** Writes a TInt value as an 8Â bit value to this stream.
1256 @param aValue The value to be written to this stream. */
1259 WriteL((const TUint8*)&aValue,1); // platform dependency
1262 EXPORT_C void RWriteStream::WriteInt16L(TInt aValue)
1263 /** Writes a TInt value as a 16Â bit value to this stream.
1265 @param aValue The value to be written to this stream. */
1267 WriteL((const TUint8*)&aValue,2); // platform dependency
1270 EXPORT_C void RWriteStream::WriteInt32L(TInt32 aValue)
1271 /** Writes a TInt32 value as a 32 bit value to this stream.
1273 @param aValue The value to be written to this stream. */
1276 WriteL((const TUint8*)&aValue,4); // platform dependency
1279 EXPORT_C void RWriteStream::WriteUint8L(TUint aValue)
1280 /** Writes a TUint value as an 8 bit value to this stream.
1282 @param aValue The value to be written to this stream. */
1284 WriteL((const TUint8*)&aValue,1); // platform dependency
1287 EXPORT_C void RWriteStream::WriteUint16L(TUint aValue)
1288 /** Writes a TUint value as a 16 bit value to this stream.
1290 @param aValue The value to be written to this stream. */
1293 WriteL((const TUint8*)&aValue,2); // platform dependency
1296 EXPORT_C void RWriteStream::WriteUint32L(TUint32 aValue)
1297 /** Writes a TUint32 value as a 32 bit value to this stream.
1299 @param aValue The value to be written to this stream. */
1301 WriteL((const TUint8*)&aValue,4); // platform dependency
1305 namespace store_private
1308 MemStreamBuf::MemStreamBuf(const CStreamStore& aStore, TStreamId aId)
1309 : iStore(&aStore), iStreamId(aId), iCurrentReadIndex(0), iCurrentWriteIndex(0), iData()
1313 MemStreamBuf::MemStreamBuf(const CStreamStore& aStore, TStreamId aId,
1314 MStreamBuf *aStreamIn, std::streamoff aOffset, TUint32 aLength)
1315 : iStore(&aStore), iStreamId(aId), iCurrentReadIndex(0), iCurrentWriteIndex(0), iData()
1319 aStreamIn->SeekL(MStreamBuf::ERead, EStreamBeginning, aOffset);
1323 aStreamIn->ReadL(&data, 1);
1324 iData.push_back(data);
1330 void MemStreamBuf::AppendFromFile(MStreamBuf *aStreamIn, TUint32 aLength)
1335 aStreamIn->ReadL(&data, 1);
1336 iData.push_back(data);
1343 TStreamId MemStreamBuf::StreamId() const
1348 const std::vector<TUint8> &MemStreamBuf::GetData() const
1354 TInt MemStreamBuf::DoReadL(TAny *aPtr,TInt aMaxLength)
1356 char *p = reinterpret_cast<char *>(aPtr);
1357 for(TInt i=0; i < aMaxLength; ++i)
1359 *p++ = iData[iCurrentReadIndex++];
1364 void MemStreamBuf::DoWriteL(const TUint8* aPtr,TInt aLength)
1366 for(TInt i=0; i < aLength; ++i)
1368 if(iCurrentWriteIndex == iData.size())
1370 iData.push_back(aPtr[i]);
1374 iData[iCurrentWriteIndex] = aPtr[i];
1376 ++iCurrentWriteIndex;
1380 TStreamPos MemStreamBuf::DoSeekL(TMark aMark,TStreamLocation aLocation,TInt anOffset)
1382 if(aMark != ERead && aMark != EWrite)
1387 TUint32 *p = (aMark == ERead) ? (&iCurrentReadIndex) : (&iCurrentWriteIndex);
1391 case EStreamBeginning:
1400 *p = iData.size() + anOffset;
1407 if(*p > iData.size())
1409 // nb we allow a seek 1 byte off end of data
1415 void MemStreamBuf::DoRelease()
1419 }; // end of namespace store_private
1421 const TInt KShiftCardinality8=1;
1422 const TInt KShiftCardinality16=2;
1423 const TInt KShiftCardinality32=3;
1425 const TInt KDefaultIoBufSize=0xc00;
1426 const TInt KFilterIoBufSize=0x100;
1428 EXPORT_C void TCardinality::ExternalizeL(RWriteStream& aStream) const
1429 /** Externalises this object to a write stream.
1431 The existence of this function means that the standard templated operator<<()
1432 can be used to externalise objects of this class.
1434 @param aStream Stream to which the object should be externalised.
1435 @see operator<<() */
1438 if (n<=(KMaxTUint8>>KShiftCardinality8))
1439 aStream.WriteUint8L((n<<KShiftCardinality8));
1440 else if (n<=(KMaxTUint16>>KShiftCardinality16))
1441 aStream.WriteUint16L((n<<KShiftCardinality16)+0x1);
1444 __ASSERT_DEBUG(n<=(TUint)KMaxCardinality,Panic(EStreamCardinalityOutOfRange));
1445 aStream.WriteUint32L((n<<KShiftCardinality32)+0x3);
1449 EXPORT_C void TCardinality::InternalizeL(RReadStream& aStream)
1450 /** Internalizes this object from a read stream.
1452 The existence of this function means that the standard templated operator>>()
1453 can be used to internalise objects of this class.
1455 @param aStream Stream store from which the objet is to be internalised.
1456 @see operator>>() */
1458 TUint n=aStream.ReadUint8L();
1460 n>>=KShiftCardinality8;
1461 else if ((n&0x2)==0)
1463 n+=aStream.ReadUint8L()<<8;
1464 n>>=KShiftCardinality16;
1466 else if ((n&0x4)==0)
1468 aStream.ReadL((TUint8*)&iCount,sizeof(TUint32)-sizeof(TUint8));
1469 n+=TUint(iCount)<<8; // platform dependency
1470 n>>=KShiftCardinality32;
1473 __LEAVE(KErrCorrupt);
1475 __ASSERT_DEBUG(n<=(TUint)KMaxCardinality,User::Invariant());
1479 EXPORT_C void TCardinality::__DbgChkRange(TInt aCount)
1481 // Check for a negative count.
1484 if(!(aCount<=KMaxCardinality)) FatalError();
1487 void CleanupStack::PopAndDestroy(RWriteStream *aStream)
1491 void CleanupStack::PopAndDestroy(RReadStream *aStream)
1496 void CleanupStack::PopAndDestroy(CFileStore *aStore)