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.
20 int main(int argc,char *argv[])
22 if (TheProgram.ProcessCommandLine(argc,argv))
24 TheProgram.ExplainUsage();
33 class Program TheProgram;
36 : iFile(NULL),iOptions(0),iBin(&iBinBuf),iErrors(0)
41 void Program::Information()
44 cout << "\nEPOC PermanentFileStore Dump Utility Version " << MajorVersion << '.' \
45 << setw(2) << MinorVersion << "(build " << setw(3) << setfill('0') << Build \
46 << ")\nCopyright (c) Symbian Software Limited 1997-2004. All rights reserved.\n\n" << flush;
51 void Program::ExplainUsage()
54 cout << "Usage: PFSDUMP [options] storefile\n" \
55 " -v verbose: dump contents\n" \
56 " -f generate a frame dump\n" \
57 " -r<n> use nth previous store revision\n" \
58 " -c format output for comparison (not for frame dumps)\n"
59 " -q quiet: suppress logo\n" << flush;
62 int Program::ProcessCommandLine(int argc,char ** argv)
70 istrstream arg(*++argv);
74 while ((c=arg.get())!=EOF)
89 if (arg.fail() || iTocRevision<0)
95 default: // unrecognised option
100 else if (iFile!=NULL)
101 return 1; // two filenames?
113 // The main part of the program
117 StoreFile store(iFile);
122 cout << "Store is empty" << endl;
135 void Program::Abort(char const* aMessage)
137 cerr << aMessage << endl;
141 void Program::Corrupt(char const* aMessage)
143 // terminate after detecting a fatal corruption error
146 cerr << "\nfatal error: " << aMessage << "\ncannot continue\n" << flush;
150 ostream& Program::Error()
152 ++TheProgram.iErrors;
153 return cerr << "error: ";
156 ostream& Program::Warning()
158 ++TheProgram.iErrors;
159 return cerr << "warning: ";
168 int binbuf::overflow(int)
171 int binbuf::underflow()
176 int StoreFile::OutWidth;
178 StoreFile::StoreFile(char const* aFile)
180 #ifdef __MSVCDOTNET__
181 iFile.open(aFile, ios::binary);
182 #else //!__MSVCDOTNET__
183 iFile.open(aFile, ios::binary | ios::nocreate);
184 #endif //__MSVCDOTNET__
186 Program::Abort("Unable to open file or file does not exist");
189 iFile.read((char*)&uid,sizeof(uid));
190 iFile.seekg(0,ios::end);
191 int size=iFile.tellg();
192 if (size<FrameDes::First || uid!=PermanentFileStoreUid)
193 Program::Abort("Not a permanent file store");
196 if (TheProgram.Option(Program::compare))
209 cout << "Dumping " << aFile << "\n\n";
211 iFile.seekg(Header::Offset,ios::beg);
212 iFile.read((char*)&iHeader,Header::Size);
220 StoreFile::~StoreFile()
225 void StoreFile::LoadFrames()
228 int offset=FrameDes::First;
229 int full=FrameDes::First+FrameDes::Interval;
230 int diff=FrameDes::First;
232 while (offset-FrameDes::Size<iSize)
236 full+=FrameDes::Interval;
237 diff+=FrameDes::Size;
241 Program::Warning() << "incomplete link at " << FramePos(offset-diff) << endl;
244 iFile.seekg(offset-FrameDes::Size,ios::beg);
246 iFrames.Add(FramePos(offset-diff),frame);
247 int length=frame.Length();
250 if (full>iSize && offset>iSize)
251 Program::Warning() << "incomplete frame at " << FramePos(offset-diff) << endl;
256 int newoffset=offset+length+FrameDes::Size;
257 if (newoffset>=full || newoffset-FrameDes::Size>iSize)
259 Program::Error() << "bad link at " << FramePos(offset-diff) << ", skipping to next anchor link" << endl;
265 if (full-offset<=FrameDes::Size)
273 void StoreFile::LoadToc()
275 FramePos toc=iHeader.Toc();
276 Stream stream(iFrames,toc);
277 if (!stream.IsGood())
279 Program::Error() << "invalid toc address " << toc << endl;
282 if (stream.Type()!=FrameDes::Toc)
284 Program::Error() << "toc address " << toc << ": refers to non-toc frame"<< endl;
288 // find the requested store revision
289 Frames::Iterator f=stream.Frame();
290 Frames::Iterator const first=iFrames.Begin();
291 for (int rev=TheProgram.TocRevision();rev;--rev)
295 Program::Abort("Store revision not found");
296 } while (f->iDes.Type()!=FrameDes::Toc);
299 iToc.Load(iFile,iFrames,f,iHeader.GetReloc());
301 // verify the Toc stream references
302 Toc::Iterator const end=iToc.End();
303 for (Toc::Iterator iter=iToc.Begin();iter<end;++iter)
305 if (iter->iHandle.IsNull())
306 Program::Error() << "missing entry in toc-delta for index " << (1+iter-iToc.Begin()) << endl;
307 else if (!iter->iHandle.Avail() && iter->Pos().Pos()>=0)
309 f=iFrames.Find(iter->Pos());
311 Program::Error() << "invalid stream reference in toc entry " << iter->iHandle << endl;
312 else if (iter->Pos().Pos()>=toc.Pos())
313 Program::Error() << "virtual stream reference in toc entry " << iter->iHandle << endl;
314 else if (f->iDes.Type()!=FrameDes::Data)
315 Program::Error() << "toc entry " << iter->iHandle << ": refers to non-data frame" << endl;
320 void StoreFile::EmitHeader()
325 void StoreFile::EmitFrames()
327 int verbose=TheProgram.Option(Program::verbose);
328 Frames::Iterator const end=iFrames.End();
329 for (Frames::Iterator iter=iFrames.Begin();iter<end;++iter)
331 FramePos pos=iter->iPos;
332 cout << "Frame at " << pos << ": " << iter->iDes << endl;
337 int length=iter->iDes.Length();
340 if (iter+1==end) // no more
342 length=iter[1].iPos.Pos()-pos.Pos();
345 iFile.seekg(FileOffset(pos).Offset(),ios::beg);
346 hex.Dump(iFile,length);
353 void StoreFile::EmitToc()
358 void StoreFile::EmitStreams()
360 int verbose=TheProgram.Option(Program::verbose);
362 Toc::Iterator const end=iToc.End();
363 for (Toc::Iterator iter=iToc.Begin();iter<end;++iter)
365 if (!iter->iHandle.Avail())
367 cout << "Stream " << iter->iHandle;
368 if (iter->Pos().Pos()==-1)
370 cout << " is empty\n";
373 if (!TheProgram.Option(Program::compare))
374 cout << " at " << iter->Pos();
375 Stream stream(iFrames,iter->Pos());
376 if (!stream.IsGood() || stream.Type()!=FrameDes::Data)
378 cout << " is invalid\n";
381 int len=stream.Length();
382 cout << ", " << dec << len << hex << " byte" << (len==1 ? "\n" : "s\n");
388 Frames::Iterator f=stream.Frame();
391 FramePos pos=f->iPos;
392 int len=f++->iDes.Length();
394 len=f->iPos.Pos()-pos.Pos();
395 iFile.seekg(FileOffset(pos).Offset(),ios::beg);
397 } while (f->iDes.Type()==FrameDes::Continuation);
417 void HexDump::Reset()
419 memset(iChar,' ',HexInterval+3);
420 iChar[HexInterval+3]=0;
424 void HexDump::Dump(istream& aStream,int aBytes)
426 while (--aBytes>=0 && !aStream.eof())
429 cout << Margin << FileOffset(iPos) << ':';
431 cout << ' ' << setw(2) << c;
432 *iPtr++=char(isprint(c) ? c : '.');
433 if (++iByte==HexInterval/2)
438 else if (iByte==HexInterval)
443 void HexDump::Flush()
447 for (int ii=iByte;ii<HexInterval;++ii)
449 if (iByte<HexInterval/2)
451 cout << iChar << endl;
461 : iPos(0), iRep(NULL), iAvail(0)
463 memset(&iHeader,0,sizeof(iHeader));
472 void Toc::Base(const char* aPtr,int aCount)
475 for (int i=1;i<=aCount;++e,++i)
477 e->iHandle=i; // set the index part
478 memcpy((char*)e+Entry::BaseRedundant,aPtr,Entry::BaseSize);
479 aPtr+=Entry::BaseSize;
483 void Toc::Load(istream& aStream,Frames const& aFrames,Frames::Iterator aFrame,Header::Reloc const* aReloc)
488 void* toc = toc1.Load(aStream);
489 const char* p = reinterpret_cast<char*>(toc);
490 memcpy(&iHeader,p,Head::Size);
492 int n = iHeader.iCount;
495 memset(&iHeader,0,Head::Size);
496 Program::Error() << "corrupt toc" << endl;
499 iRep=static_cast<Entry*>(malloc(n*sizeof(Entry)));
501 Program::Abort("Out of memory");
503 if (iHeader.IsDelta())
505 // verify the delta header
506 memcpy(&iDelta,p,DeltaHead::Size);
508 int dn = iDelta.iCount;
509 if (toc1.Length() != Head::Size + DeltaHead::Size + dn * Entry::DeltaSize)
511 memset(&iHeader,0,Head::Size);
512 Program::Error() << "incomplete toc" << endl;
517 FramePos tocbase(iDelta.iBase + Header::tocoffset);
518 if (aReloc && aReloc->iHandle.IsTocBase())
519 tocbase = aReloc->iPos;
520 Stream toc2(aFrames,tocbase);
523 memset(&iHeader,0,Head::Size);
524 Program::Error() << "invalid toc-base address " << tocbase << endl;
527 if (toc2.Type()!=FrameDes::Toc)
529 memset(&iHeader,0,Head::Size);
530 Program::Error() << "toc-base address " << tocbase << ": refers to non-toc frame"<< endl;
534 // validate and load the toc-base
535 void* tocb = toc2.Load(aStream);
536 const char* p2 = reinterpret_cast<char*>(tocb);
538 memcpy(&headbase,p2,Head::Size);
540 if (headbase.IsDelta())
542 memset(&iHeader,0,Head::Size);
543 Program::Error() << "toc-base is a toc-delta"<< endl;
546 int bn = headbase.iCount;
549 memset(&iHeader,0,Head::Size);
550 Program::Error() << "toc-base is larger than toc"<< endl;
556 // validate and update with the toc-delta
561 memcpy(&e,p,Entry::DeltaSize);
563 int ix = e.iHandle.Index();
566 memset(&iHeader,0,Head::Size);
567 Program::Error() << "toc-delta entry " << e.iHandle << " is outside toc"<< endl;
572 memset(&iHeader,0,Head::Size);
573 Program::Error() << "toc-delta entry " << e.iHandle << " is out of order"<< endl;
582 if (toc1.Length() != Head::Size + n * Entry::BaseSize)
584 memset(&iHeader,0,Head::Size);
585 Program::Error() << "incomplete toc" << endl;
592 // apply the relocation
593 if (aReloc && !aReloc->iHandle.IsTocBase())
595 int ix=aReloc->iHandle.Index();
597 Program::Corrupt("invalid index in relocation patch");
600 if (e.iHandle.Generation()!=aReloc->iHandle.Generation())
601 Program::Corrupt("incorrect generation in relocation patch");
602 e.iPos=aReloc->iPos.Pos();
605 // count the available entries
607 for (int i=0;i<n;++i)
609 if (iRep[i].iHandle.Avail())
614 // verify the available list
615 Handle link=iHeader.iAvail;
619 if (!link.Avail() || ix<=0 || ix >iHeader.iCount)
621 eAvail: Program::Error() << "corrupt available link in toc header " << link << endl;
624 Entry const* en=&(*this)[ix];
625 if (en->iHandle!=link)
631 Program::Error() << "corrupt available list, possible circular reference" << endl;
634 Handle next=en->Link();
638 if (!next.Avail() || ix<=0 || ix >iHeader.iCount)
640 eLink: Program::Error() << "corrupt link in toc entry " << link << endl;
644 if (en->iHandle!=next)
650 Program::Error() << "corrupt available list: free index leakage" << endl;
653 ostream& operator<<(ostream& aStream,Toc const& aToc)
655 int all=TheProgram.Option(Program::frame);
657 Toc::Head const& head=aToc.iHeader;
658 if (TheProgram.Option(Program::compare))
661 aStream << "Toc at " << aToc.iPos << " with ";
662 aStream << dec << head.iCount << (head.iCount==1 ? " entry: " : " entries: ") \
663 << head.iCount-aToc.iAvail << " allocated, " << aToc.iAvail << " free\n" << hex;
665 aStream << "root is " << head.Root() << '\n';
668 aStream << "first available is " << head.iAvail << '\n';
669 Toc::Iterator const end=aToc.End();
670 for (Toc::Iterator iter=aToc.Begin();iter<end;++iter)
672 aStream << (iter->iHandle==head.Root() ? "* " : Margin) << iter->iHandle;
673 if (iter->iHandle.Avail())
674 aStream << " free -> " << iter->Link() << '\n';
675 else if (iter->Pos().Pos()==-1)
676 aStream << " empty\n";
678 aStream << " alloc at " << iter->Pos() << '\n';
681 return aStream << flush;
686 int Stream::Length() const
689 Frames::Iterator f=iFrame;
691 int len=f->iDes.Length();
693 len=f[1].iPos.Pos()-f[0].iPos.Pos();
695 } while ((++f)->iDes.Type()==FrameDes::Continuation);
699 void* Stream::Load(istream& aStream) const
702 void* data = malloc(size);
704 Program::Abort("Out of memory");
706 char* read=reinterpret_cast<char*>(data);
707 Frames::Iterator f = iFrame;
710 FramePos pos=f->iPos;
711 int len=f++->iDes.Length();
713 len=f->iPos.Pos()-pos.Pos();
714 aStream.seekg(FileOffset(pos).Offset(),ios::beg);
715 aStream.read(read,len);
717 } while (f->iDes.Type()==FrameDes::Continuation);
725 : iSize(0),iElements(0),iRep(NULL)
733 void Frames::Add(FramePos aPos,FrameDes aDes)
735 if (iElements==iSize)
737 iSize=iSize==0 ? 128 : iSize+iSize;
738 void* rep=realloc(iRep,iSize*sizeof(*iRep));
740 Program::Abort("Out of memory");
743 Element& element=iRep[iElements++];
748 void Frames::Complete()
750 // add a terminating entry
757 Frames::Iterator Frames::Find(FramePos aPos) const
759 return (Element const*)bsearch(&aPos,iRep,iElements,sizeof(*iRep),Compare);
762 int Frames::Compare(void const* aLeft,void const* aRight)
764 int left=static_cast<FramePos const*>(aLeft)->Pos();
765 int right=static_cast<Element const*>(aRight)->iPos.Pos();
775 FramePos Header::Toc() const
777 return tocoffset+(!Dirty() && iToc.iZero==0 ? iToc.iPos : iBackupToc>>backupshift);
780 Header::Reloc const* Header::GetReloc() const
782 return (Dirty() || iToc.iZero==0) ? NULL : reinterpret_cast<Reloc const*>(&iReloc);
785 ostream& operator<<(ostream& aStream,Header const& aHeader)
787 aStream << "Header is " << (aHeader.Dirty() ? "dirty" : "clean");
788 Header::Reloc const* reloc=aHeader.GetReloc();
791 aStream << "\npending relocation of ";
792 if (reloc->iHandle.IsTocBase())
793 aStream << "toc-base";
795 aStream << "stream " << StreamId(reloc->iHandle);
796 aStream << " to " << reloc->iPos;
798 return aStream << "\n\n" << flush;
803 FileOffset::FileOffset(FramePos aPos)
804 // calculate the file offset for a streampos
807 int n=pos>>FrameDes::FullShift;
808 pos+=FrameDes::Size*n+FrameDes::First;
812 FileOffset::operator FramePos() const
814 int pos=iValue-FrameDes::First;
815 int n=pos/FrameDes::Interval;
816 pos-=n*FrameDes::Size;
817 return FramePos(pos);
820 ostream& operator<<(ostream& aStream,FileOffset anOffset)
822 return aStream << setw(StoreFile::OutWidth) << anOffset.iValue;
827 ostream& operator<<(ostream& aStream,Handle aHandle)
829 if (aHandle.IsNull())
832 aStream << setw(6) << aHandle.Index() << ':' << aHandle.Generation();
838 ostream& operator<<(ostream& aStream,FramePos aPos)
840 return aStream << setw(StoreFile::OutWidth) << aPos.iValue << '[' << FileOffset(aPos) << ']';
845 istream& operator>>(istream& aStream,FrameDes& aFrame)
847 return aStream.read((char*)&aFrame,FrameDes::Size);
850 ostream& operator<<(ostream& aStream,FrameDes aFrame)
852 static char const* FrameType[]={"free","data","toc","continuation"};
853 aStream << FrameType[aFrame.Type()] << " (";
854 int length=aFrame.Length();
858 aStream << dec << length << hex;
859 return aStream << ')';