userlibandfileserver/fileserver/srom/sr_rom.cpp
changeset 0 a41df078684a
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/userlibandfileserver/fileserver/srom/sr_rom.cpp	Mon Oct 19 15:55:17 2009 +0100
@@ -0,0 +1,806 @@
+// Copyright (c) 1995-2009 Nokia Corporation and/or its subsidiary(-ies).
+// All rights reserved.
+// This component and the accompanying materials are made available
+// under the terms of the License "Eclipse Public License v1.0"
+// which accompanies this distribution, and is available
+// at the URL "http://www.eclipse.org/legal/epl-v10.html".
+//
+// Initial Contributors:
+// Nokia Corporation - initial contribution.
+//
+// Contributors:
+//
+// Description:
+// f32\srom\sr_rom.cpp
+// 
+//
+
+#include "sr_std.h"
+
+#if defined(_UNICODE)
+#define __SIZE(len) ((len)<<1)
+#else
+#define __SIZE(len) (len)
+#endif
+
+const TRomHeader* CRom::iRomHeaderAddress=(TRomHeader*)UserSvr::RomHeaderAddress();
+
+TInt TRomDir::BinarySearch(const TDesC& aName, TInt aLengthLimit, TInt aMode, TBool aDir) const
+	{
+//	RDebug::Print(_L("BinarySearch %S ll=%d m=%d dir=%d"), &aName, aLengthLimit, aMode, aDir);
+	const TRomDirSortInfo* s = SortInfo();
+	TInt l = aDir ? 0 : s->iSubDirCount;
+	TInt r = aDir ? s->iSubDirCount : s->iSubDirCount + s->iFileCount;
+	TBool found = EFalse;
+	while (r>l)
+		{
+		TInt m=(l+r)>>1;
+		const TRomEntry* e = SortedEntry(m);
+		TInt nl = Min(e->iNameLength, aLengthLimit);
+		TPtrC en((const TText*)&e->iName[0], nl);
+		TInt k = CRomMountCB::Compare(aName, en);
+		if (k==0)
+			{
+			if (aMode == EArrayFindMode_Any)
+				{
+//				RDebug::Printf("Found %d", m);
+				return m;
+				}
+			found = ETrue;
+			if (aMode == EArrayFindMode_First)
+				r=m;
+			else
+				l=m+1;
+			}
+		else if (k>0)
+			l=m+1;
+		else
+			r=m;
+		}
+//	RDebug::Printf("Found=%d r=%d", found, r);
+	return found ? r : KErrNotFound;
+	}
+
+// Navigate the path to find the leaf directory, starting at this.
+const TRomDir* TRomDir::FindLeafDir(const TDesC& aPath) const
+	{
+	TLex lex(aPath);
+	TInt r;
+	const TRomDir* d = this;
+	FOREVER
+		{
+		lex.Inc(); // Skip the file separator
+		lex.Mark();
+		r=lex.Remainder().Locate(KPathDelimiter);
+		if (r==KErrNotFound)
+			r=lex.Remainder().Length();
+		if (r==0) // End of the path
+			break;
+		lex.Inc(r); // Set the token length
+		TInt ix = d->BinarySearch(lex.MarkedToken(), KMaxTInt, EArrayFindMode_Any, ETrue);
+		if (ix<0)
+			return NULL;
+		const TRomEntry* e = d->SortedEntry(ix);
+//		if (!(e->iAtt & KEntryAttDir))
+//			return NULL;
+		d = (const TRomDir*)e->iAddressLin;
+		}
+	return d;
+	}
+
+LOCAL_C void Fault(TFault aFault)
+//
+// Report a fault in the rom file system.
+//
+	{
+
+	User::Panic(_L("ROMFILESYS"),aFault);
+	}
+
+CRomMountCB::CRomMountCB(const CRom* aRom)
+//
+// Constructor
+//
+	: iRom(aRom)
+	{
+	}
+
+void CRomMountCB::Dismounted()
+//
+// Dummy implementation of pure virtual function
+//
+	{}
+
+void CRomMountCB::IsFileInRom(const TDesC& aName,TUint8*& aFileStart)
+//
+// Return the address of the file if it is in rom
+//
+	{
+	
+	TLinAddr dir;
+	TLinAddr entry=0;
+	aFileStart=NULL;
+	TRAPD(r,FindEntryL(aName,KEntryAttNormal,ETrue,dir,entry));
+	if (r!=KErrNone)
+		return;
+	aFileStart=(TUint8*)((const TRomEntry*)entry)->iAddressLin;
+	}
+
+TInt CRomMountCB::Compare(const TDesC& aLeft, const TDesC& aRight)
+//
+//Compares two filenames.  Folds ASCII characters to uppercase
+//
+	{
+
+	TInt ll = aLeft.Length();
+	TInt rl = aRight.Length();
+	TInt len = Min(ll, rl);
+	const TText* l = aLeft.Ptr();
+	const TText* r = aRight.Ptr();
+	while (len--)
+		{
+		TText lc = *l++;
+		TText rc = *r++;
+		if (lc >= 'A' && lc <= 'Z')
+			lc += ('a' - 'A');
+		if (rc >= 'A' && rc <= 'Z')
+			rc += ('a' - 'A');
+		TInt x = lc - rc;
+		if (x)
+			return x;
+		}
+	// match up to end of shorter string, now compare lengths
+	return ll - rl;
+	}
+
+void CRomMountCB::FindBinaryL(const TDesC& aName, TUint aAtt, TBool aAttKnown, TLinAddr aDir, TLinAddr& aEntry, TInt aError) const
+//
+//Identical to FindL, but uses binary search for faster performance.
+//However, can't deal with wildcards, whereas FindL can.
+//
+	{
+
+	//Although the value of aEntry is not used, we expect it to be zero,
+	__ASSERT_DEBUG(aEntry==0,Fault(ERomInvalidArgument));
+	const TRomDir* d = (const TRomDir*)aDir;
+	TBool ix;
+	if (aAttKnown)
+		ix = d->BinarySearch(aName, KMaxTInt, EArrayFindMode_Any, aAtt & KEntryAttDir);
+	else
+		{
+		//We don't know whether we're looking for a file or a directory, so
+		//look through both
+		ix = d->BinarySearch(aName, KMaxTInt, EArrayFindMode_Any, EFalse);
+		if (ix<0 || !MatchEntryAtt(d->SortedEntry(ix)->iAtt, aAtt) )
+			ix = d->BinarySearch(aName, KMaxTInt, EArrayFindMode_Any, ETrue);
+		}
+	if (ix>=0)
+		{
+		const TRomEntry* e = d->SortedEntry(ix);
+		if (MatchEntryAtt(e->iAtt, aAtt))
+			{
+			aEntry = (TLinAddr)e;
+			return;
+			}
+		}
+	User::Leave(aError);
+	}
+
+void CRomMountCB::FindL(const TDesC& aName, TUint anAtt, TLinAddr aDir, TLinAddr& anEntry, TInt anError) const
+//
+// Scan from aDir looking for aName.
+// If found return the result in anEntry.
+//
+	{
+	const TRomDir* pD = (const TRomDir*)aDir;
+	const TRomEntry* pE;
+	if (anEntry==0)
+		pE = &pD->iEntry;
+	else
+		{
+		pE = (const TRomEntry*)anEntry;
+		pE = PtrAdd(pE, Align4(__SIZE(pE->iNameLength) + KRomEntrySize));
+		}
+	const TRomEntry* pEnd = PtrAdd(&pD->iEntry, pD->iSize);
+	while (pE<pEnd)
+		{
+		TPtrC name = TPtrC((const TText*)&pE->iName[0], pE->iNameLength);
+		if (name.MatchF(aName)!=KErrNotFound && MatchEntryAtt(pE->iAtt, anAtt))
+			{
+			anEntry = (TLinAddr)pE;
+			return;
+			}
+		pE = PtrAdd(pE, Align4(__SIZE(pE->iNameLength) + KRomEntrySize));
+		}
+	User::Leave(anError);
+	}
+
+void CRomMountCB::FindEntryL(const TDesC& aName, TUint anAtt, TBool aAttKnown, TLinAddr& aDir, TLinAddr& anEntry) const
+//
+// Locate an entry from its full path name.
+//
+	{
+
+	TInt namePos=aName.LocateReverse(KPathDelimiter)+1; // There is always a path delimiter
+	const TRomDir* d = ((const TRomDir*)RomRootDirectory())->FindLeafDir(aName.Left(namePos));
+	if (!d)
+		User::Leave(KErrPathNotFound);
+	anEntry=0;
+	aDir = (TLinAddr)d;
+	FindBinaryL(aName.Mid(namePos),anAtt,aAttKnown,aDir,anEntry,KErrNotFound);
+	}
+
+void CRomMountCB::MountL(TBool /*aForceMount*/)
+//
+// Mount a media. Only allowed to leave with KErrNoMemory,KErrNotReady,KErrCorrupt,KErrUnknown.
+//
+	{
+
+	iUniqueID=0;
+	iSize=(TUint)RomHeader().iUncompressedSize;
+	SetVolumeName(_L("RomDrive").AllocL());
+	}
+
+TInt CRomMountCB::ReMount()
+//
+// Try and remount this media.
+//
+	{
+
+	Fault(ERomReMountNotSupported);
+	return(0);
+	}
+
+void CRomMountCB::VolumeL(TVolumeInfo& aVolume) const
+//
+// Return the volume info.
+//
+	{
+
+	aVolume.iFree=0;
+	}
+
+void CRomMountCB::SetVolumeL(TDes& /*aName*/)
+//
+// Set the volume label.
+//
+	{
+
+	User::Leave(KErrAccessDenied);
+	}
+
+void CRomMountCB::MkDirL(const TDesC& /*aName*/)
+//
+// Make a directory.
+//
+	{
+
+	User::Leave(KErrAccessDenied);
+	}
+
+void CRomMountCB::RmDirL(const TDesC& /*aName*/)
+//
+// Remove a directory.
+//
+	{
+
+	User::Leave(KErrAccessDenied);
+	}
+
+void CRomMountCB::DeleteL(const TDesC& /*aName*/)
+//
+// Delete a file.
+//
+	{
+
+	User::Leave(KErrAccessDenied);
+	}
+
+void CRomMountCB::RenameL(const TDesC& /*anOldName*/,const TDesC& /*aNewName*/)
+//
+// Rename a file or directory.
+//
+	{
+
+	User::Leave(KErrAccessDenied);
+	}
+
+void CRomMountCB::ReplaceL(const TDesC& /*anOldName*/,const TDesC& /*aNewName*/)
+//
+// Atomic replace.
+//
+	{
+
+	User::Leave(KErrAccessDenied);
+	}
+
+void CRomMountCB::EntryL(const TDesC& aName,TEntry& anEntry) const
+//
+// Get entry details.
+//
+	{
+
+	TLinAddr dir;
+	TLinAddr entry;
+	FindEntryL(aName,KEntryAttMaskSupported,EFalse,dir,entry);
+	const TRomEntry* pE = (const TRomEntry*)entry;
+	anEntry.iAtt=pE->iAtt;
+	anEntry.iSize=pE->iSize;
+	anEntry.iModified=RomHeader().iTime;
+	anEntry.iName.Des().Copy((TText*)&pE->iName[0],pE->iNameLength);
+	ReadUidL(pE->iAddressLin,anEntry);
+	}
+
+void CRomMountCB::SetEntryL(const TDesC& /*aName*/,const TTime& /*aTime*/,TUint /*aMask*/,TUint /*aVal*/)
+//
+// Set entry details.
+//
+	{
+
+	User::Leave(KErrAccessDenied);
+	}
+
+void CRomMountCB::FileOpenL(const TDesC& aName,TUint aMode,TFileOpen anOpen,CFileCB* aFile)
+//
+// Open a file on the current mount.
+//
+	{
+
+	if (aMode&EFileWrite)
+		User::Leave(KErrAccessDenied);
+	switch (anOpen)
+		{
+	case EFileCreate:
+	case EFileReplace:
+		User::Leave(KErrAccessDenied);
+	case EFileOpen:
+		break;
+	default:
+		User::Leave(KErrAccessDenied);
+		}
+	TLinAddr dir;
+	TLinAddr entry;
+	FindEntryL(aName,KEntryAttMustBeFile,ETrue,dir,entry);
+	const TRomEntry* pE = (const TRomEntry*)entry;
+	CRomFileCB& file=(*((CRomFileCB*)aFile));
+	file.SetSize(pE->iSize);
+	file.SetAtt(pE->iAtt);
+	file.SetModified(RomHeader().iTime);
+	file.SetBase((const TUint8*)pE->iAddressLin);
+	}
+
+void CRomMountCB::DirOpenL(const TDesC& aName, CDirCB* aDir)
+//
+// Open a file on the current mount.
+//
+	{
+
+    TFileName fileName=aName;
+    TInt namePos=aName.LocateReverse(KPathDelimiter)+1; // Exclude path delimiter
+    if (namePos==aName.Length())
+        fileName+=_L("*");
+	const TRomDir* d = ((const TRomDir*)RomRootDirectory())->FindLeafDir(aName.Left(namePos));
+	if (!d)
+		User::Leave(KErrPathNotFound);
+	CRomDirCB& dirCB = *(CRomDirCB*)aDir;
+	dirCB.SetDir((TLinAddr)d, NULL, fileName.Mid(namePos));
+	}
+
+void CRomMountCB::RawReadL(TInt64 aPos,TInt aLength,const TAny* aDes,TInt anOffset,const RMessagePtr2& aMessage) const
+//
+// Read up to aLength data directly from the ROM
+//
+	{
+
+	TUint romSize=RomHeader().iUncompressedSize;
+	if (I64LOW(aPos)>=romSize)
+		aMessage.WriteL(2,TPtrC8(NULL,0),anOffset);
+	else
+		{
+		TInt len=Min((TInt)(romSize-I64LOW(aPos)),aLength);
+		aMessage.WriteL(2,TPtrC8((TUint8*)RomHeader().iRomBase,len),anOffset);
+		}
+	}
+
+void CRomMountCB::RawWriteL(TInt64 /*aPos*/,TInt /*aLength*/,const TAny* /*aDes*/,TInt /*anOffset*/,const RMessagePtr2& /*aMessage*/)
+//
+// Write aLength data to ROM (?)
+//
+	{
+
+	User::Leave(KErrAccessDenied);
+	}
+
+void CRomMountCB::ReadUidL(TLinAddr anAddr,TEntry& anEntry) const
+//
+// Read a uid if present.
+//
+	{
+	// Need to consider zero-length files (for which anAddr is 0)
+	// and files too small to have a UID
+	if (anEntry.iSize >= (TInt) sizeof(TCheckedUid))
+		{
+		TCheckedUid entryUid(TPtrC8((TUint8*)anAddr, sizeof(TCheckedUid)));
+		anEntry.iType=entryUid.UidType();
+		}
+	else
+		{
+		anEntry.iType=KNullUid;
+		}
+	}
+
+
+
+void CRomMountCB::ReadSectionL(const TDesC& aName,TInt aPos,TAny* aTrg,TInt aLength,const RMessagePtr2& aMessage)
+	{
+
+	__PRINT(Print(_L("CRomMountCB::ReadSectionL")));
+			
+	TLinAddr dir;
+	TLinAddr entry;
+	FindEntryL(aName,KEntryAttMustBeFile,ETrue,dir,entry);
+	const TRomEntry* pE = (TRomEntry*)entry;
+	TInt size=pE->iSize;
+
+	const TText8* baseAddress = (const TText8*)pE->iAddressLin;
+	
+	if (size>=(aPos+aLength)) //|| (size>aPos)
+		{
+		TPtrC8 section((baseAddress+aPos),aLength);
+		aMessage.WriteL(0,section,0);
+		}
+	else if (size>aPos)
+		{
+		aLength=(size-aPos);
+		TPtrC8 section((baseAddress+aPos),aLength);
+		aMessage.WriteL(0,section,0);
+		}	
+	else
+		User::Leave(KErrEof);
+	
+	}
+
+
+
+void CRomMountCB::GetShortNameL(const TDesC& /*aLongName*/,TDes& /*aShortName*/)
+//
+// Return the short name associated with aLongName
+// Assumes all rom names are 8.3
+//
+	{
+
+	User::Leave(KErrNotSupported);
+	}
+
+void CRomMountCB::GetLongNameL(const TDesC& /*aShortName*/,TDes& /*aLongName*/)
+//
+// Return the short name associated with aLongName
+// Assumes all rom names are 8.3
+//
+	{
+
+	User::Leave(KErrNotSupported);
+	}	
+
+TInt CRomMountCB::GetInterface(TInt aInterfaceId,TAny*& aInterface,TAny* aInput) 
+//
+// Access the specified interface; behaviour is interface-specific
+//
+	{
+	switch(aInterfaceId)
+		{
+		case (CMountCB::EFileAccessor):
+			{
+			((CMountCB::MFileAccessor*&) aInterface) = this;
+			return KErrNone;
+			}
+		default:
+			return (CMountCB::GetInterface(aInterfaceId,aInterface,aInput));
+		}
+	}
+
+TInt CRomMountCB::GetFileUniqueId(const TDesC& /* aName */, TInt64& aUniqueId)
+	{
+	// Get unique identifier for the file - for ROM File System will just return zero
+	aUniqueId = MAKE_TINT64(0,0);
+	return KErrNone;
+	}
+
+TInt CRomMountCB::Spare3(TInt /*aVal*/, TAny* /*aPtr1*/, TAny* /*aPtr2*/)
+	{
+	return KErrNotSupported;
+	}
+
+TInt CRomMountCB::Spare2(TInt /*aVal*/, TAny* /*aPtr1*/, TAny* /*aPtr2*/)
+	{
+	return KErrNotSupported;
+	}
+
+TInt CRomMountCB::Spare1(TInt /*aVal*/, TAny* /*aPtr1*/, TAny* /*aPtr2*/)
+	{
+	return KErrNotSupported;
+	}
+
+
+CRomFileCB::CRomFileCB(const CRom* aRom)
+//
+// Constructor
+//
+	: iRom(aRom)
+	{
+	}
+
+void CRomFileCB::ReadL(TInt aPos,TInt& aLength,const TAny* aTrg,const RMessagePtr2& aMessage)
+//
+// Read from the file.
+//
+	{
+
+	__PRINT(Print(_L("CRomFileCB::ReadL")));
+
+	if (aPos>=iSize)
+		{
+        TPtrC8 nullBuf(NULL,0);
+		aMessage.WriteL(0,nullBuf,0);
+		aLength=0;
+		}
+	else
+		{
+		TInt len=Min((iSize-aPos),aLength);
+        TPtrC8 romBuf(iBase+aPos,len);
+//		thread->WriteL(aTrg,romBuf,0);
+		aMessage.WriteL(0,romBuf,0);
+		aLength=len;
+		}
+	}
+
+void CRomFileCB::WriteL(TInt /*aPos*/,TInt& /*aLength*/,const TAny* /*aDes*/,const RMessagePtr2& /*aMessage*/)
+//
+// Write to the file.
+//
+	{
+
+	User::Leave(KErrAccessDenied);
+	}
+
+void CRomFileCB::RenameL(const TDesC& /*aDes*/)
+//
+// Rename the file.
+//
+	{
+
+	User::Leave(KErrAccessDenied);
+	}
+
+void CRomFileCB::SetSizeL(TInt /*aSize*/)
+//
+// Set the file size.
+//
+	{
+
+	User::Leave(KErrAccessDenied);
+	}
+
+void CRomFileCB::SetEntryL(const TTime& /*aTime*/,TUint /*aMask*/,TUint /*aVal*/)
+//
+// Set the entry's attributes and modified time.
+//
+	{
+
+	User::Leave(KErrAccessDenied);
+	}
+
+void CRomFileCB::FlushAllL()
+//
+// Commit any buffered date to the media.
+//
+	{
+
+	User::Leave(KErrAccessDenied);
+	}
+
+void CRomFileCB::FlushDataL()
+//
+// Commit any buffered date to the media.
+//
+	{
+
+	User::Leave(KErrAccessDenied);
+	}
+
+TInt CRomFileCB::Address(TInt& aPos) const
+//
+// Return address of the file at aPos. Default implementation.
+//
+	{
+
+	if (aPos>=iSize)
+		return(KErrEof);
+	aPos=(TInt)(iBase+aPos);
+	return(KErrNone);
+	}
+
+CRomDirCB::CRomDirCB(const CRom* aRom)
+//
+// Constructor
+//
+	: iRom(aRom)
+	{
+	}
+
+CRomDirCB::~CRomDirCB()
+//
+// Destruct
+//
+	{
+
+	delete iMatch;
+	}
+
+TBool CRomDirCB::MatchUid()
+//
+// Match the uid ?
+//
+	{
+
+	if (iUidType[0]!=TUid::Null() || iUidType[1]!=TUid::Null() || iUidType[2]!=TUid::Null())
+		return(ETrue);
+	return(EFalse);
+	}
+
+LOCAL_C TBool CompareUid(const TUidType& aUidTrg, const TUidType& aUidSuitor)
+//
+// Compare the suitor to the target pattern
+//
+	{
+	
+	if (aUidTrg[0]!=TUid::Null() && aUidTrg[0]!=aUidSuitor[0])
+		return(EFalse);
+	if (aUidTrg[1]!=TUid::Null() && aUidTrg[1]!=aUidSuitor[1])
+		return(EFalse);
+	if (aUidTrg[2]!=TUid::Null() && aUidTrg[2]!=aUidSuitor[2])
+		return(EFalse);
+	return(ETrue);
+	}
+
+void CRomDirCB::ReadL(TEntry& anEntry)
+//
+// Read the next entry from the directory.
+//
+	{
+
+	CRomMountCB& mount=(CRomMountCB&)Mount();
+	const TRomEntry* pE;
+	FOREVER
+		{
+		//
+		//May be called with wildcards in the path, so we need
+		//to call the slow, sequential FindL
+		//
+		if (!iPending)
+			mount.FindL(*iMatch,iAtt,iDir,iNext,KErrEof);
+		iPending=EFalse;
+		pE = (const TRomEntry*)iNext;
+		iEntry.iName.Des().Copy((TText*)&pE->iName[0],pE->iNameLength);
+		iEntry.iAtt=pE->iAtt;
+		iEntry.iSize=pE->iSize;
+		iEntry.iModified=RomHeader().iTime;
+		anEntry=iEntry;
+		if (MatchUid())
+			{
+			mount.ReadUidL(pE->iAddressLin,anEntry);
+			if (CompareUid(iUidType,anEntry.iType))
+				break;
+			}
+		else
+			break;
+		}
+	if (iAtt&KEntryAttAllowUid && (anEntry.iAtt&KEntryAttDir)==0 && MatchUid()==EFalse)
+		mount.ReadUidL(pE->iAddressLin,anEntry);
+	}
+
+void CRomDirCB::SetDir(TLinAddr aDir,TLinAddr anEntry,const TDesC& aName)
+//
+// Set the directory and entry that we are on after open.
+//
+	{
+
+    iDir=aDir;
+	iNext=anEntry;
+	iMatch=aName.AllocL();
+	iPending=(anEntry!=NULL);
+	}
+
+CRom::CRom()
+//
+// Constructor
+//
+	{
+	}
+
+CRom::~CRom()
+//
+// Destruct
+//
+	{
+	}
+
+TInt CRom::Install()
+//
+// Install the file system.
+//
+	{
+
+	iVersion=TVersion(KF32MajorVersionNumber,KF32MinorVersionNumber,KF32BuildVersionNumber);
+    TPtrC name=_L("Rom");
+	return(SetName(&name));
+	}
+
+CMountCB* CRom::NewMountL() const
+//
+// Create a new mount control block.
+//
+	{
+
+	return(new(ELeave) CRomMountCB(this));
+	}
+
+CFileCB* CRom::NewFileL() const
+//
+// Create a new file.
+//
+	{
+
+	return(new(ELeave) CRomFileCB(this));
+	}
+
+CDirCB* CRom::NewDirL() const
+//
+// Create a new directory lister.
+//
+	{
+
+	return(new(ELeave) CRomDirCB(this));
+	}
+
+CFormatCB* CRom::NewFormatL() const
+//
+// Create a new media formatter.
+//
+	{
+
+	User::Leave(KErrAccessDenied);
+	return(NULL);
+	}
+
+void CRom::DriveInfo(TDriveInfo& anInfo,TInt /*aDriveNumber*/) const
+//
+// Return the drive info.
+//
+	{
+	anInfo.iDriveAtt=KDriveAttRom|KDriveAttInternal;
+	anInfo.iMediaAtt=KMediaAttWriteProtected;
+	anInfo.iType=EMediaRom;
+	}
+
+GLDEF_C void InstallRomFileSystemL()
+//
+// Create the ROM file system.
+//
+	{
+
+	CFileSystem* pS=new(ELeave) CRom;
+	CleanupStack::PushL(pS);
+	User::LeaveIfError(InstallFileSystem(pS,RLibrary()));
+	CleanupStack::Pop(1, pS);
+	}
+
+GLDEF_C const TRomHeader *RomHeader(CFileSystem *aFsys)
+//
+// Return the ROM header
+//
+	{
+
+	return &((CRom *)aFsys)->RomHeader();
+	}
+