// Copyright (c) 1998-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:
/**
@file
@internalTechnology
*/
#include <e32cmn.h>
#include <f32ver.h>
#include <u32std.h>
#include <f32file.h>
#include "filesystem_automounter.h"
#include "automounter.h"
//-----------------------------------------------------------------------------
void Fault(TFault aFault)
{
_LIT(KPanicName, "AutoMounter_fsy");
User::Panic(KPanicName, aFault);
}
//-----------------------------------------------------------------------------
/**
Factory function, Create a new object of this file system
*/
extern "C"
{
EXPORT_C CFileSystem* CreateFileSystem()
{
return CAutoMounterFileSystem::New();
}
}
//#######################################################################################################################################
//# CAutoMounterFileSystem class implementation
//#######################################################################################################################################
/**
Factory method
*/
CAutoMounterFileSystem* CAutoMounterFileSystem::New()
{
CAutoMounterFileSystem* pThis = new CAutoMounterFileSystem();
return pThis;
}
CAutoMounterFileSystem::CAutoMounterFileSystem()
{
__PRINT1(_L("#<<- CAutoMounterFileSystem::CAutoMounterFileSystem() [0x%x]"), this);
iChildFsForDefFmt = KErrNotFound;
SetState(EInvalid);
}
CAutoMounterFileSystem::~CAutoMounterFileSystem()
{
__PRINT1(_L("#<<- CAutoMounterFileSystem::~CAutoMounterFileSystem() [0x%x]"), this);
}
//-----------------------------------------------------------------------------
/**
Install iand initialise file system.
*/
TInt CAutoMounterFileSystem::Install()
{
SetState(ENotInitialised);
__PRINT1(_L("#<<- CAutoMounterFileSystem::Install() [0x%x]"), this);
iVersion=TVersion(KF32MajorVersionNumber,KF32MinorVersionNumber,KF32BuildVersionNumber);
InitialiseFileSystem();
return SetName(&KFileSystemName_AutoMounter);
}
//-----------------------------------------------------------------------------
/**
Create a new mount control block.
This method migh be called by the file server in some unusual cases, when the actual mount is needed only temporarily to get access to the
corresponding media driver. E.g. TDrive::ForceRemountDrive()
Produce the _default_ file system mount
*/
CMountCB* CAutoMounterFileSystem::NewMountL() const
{
__PRINT1(_L("#<<- CAutoMounterFileSystem::NewMountL() [0x%x]"), this);
ASSERT(State() == EInitialised);
__PRINT1(_L("#<<- producing the _default_ filesystem:%S"), &iFSNames[KDefaultFSNo]);
CFileSystem* pFS = GetChildFileSystem(KDefaultFSNo);
ASSERT(pFS);
return pFS->NewMountL();
}
//-----------------------------------------------------------------------------
/**
Create a new file.
*/
CFileCB* CAutoMounterFileSystem::NewFileL() const
{
__PRINT1(_L("#<<- CAutoMounterFileSystem::NewFileL() [0x%x]"), this);
Fault(EMustNotBeCalled);
return NULL;
}
//-----------------------------------------------------------------------------
/**
Create a new directory object
*/
CDirCB* CAutoMounterFileSystem::NewDirL() const
{
__PRINT1(_L("#<<- CAutoMounterFileSystem::NewDirL() [0x%x]"), this);
Fault(EMustNotBeCalled);
return NULL;
}
//-----------------------------------------------------------------------------
/**
Create a new media formatter
*/
CFormatCB* CAutoMounterFileSystem::NewFormatL() const
{
__PRINT1(_L("#<<- CAutoMounterFileSystem::NewFormatL() [0x%x]"), this);
Fault(EMustNotBeCalled);
return NULL;
}
//-----------------------------------------------------------------------------
#ifdef _DEBUG
/**
Called by File Server before deleting File System object.
*/
TInt CAutoMounterFileSystem::Remove()
{
__PRINT1(_L("#<<- CAutoMounterFileSystem::Remove() [0x%x]"), this);
return CFileSystem::Remove();
}
//-----------------------------------------------------------------------------
/**
*/
TBool CAutoMounterFileSystem::QueryVersionSupported(const TVersion& aVer) const
{
__PRINT1(_L("#<<- CAutoMounterFileSystem::QueryVersionSupported() [0x%x]"), this);
return CFileSystem::QueryVersionSupported(aVer);
}
#endif
//-----------------------------------------------------------------------------
/**
Find out if drive extensions are supported. In order to have consistent behaviour, _all_ child
file systems shall behave the same way.
@return ETrue if drive extensions are supported.
*/
TBool CAutoMounterFileSystem::IsExtensionSupported() const
{
__PRINT1(_L("#<<- CAutoMounterFileSystem::IsExtensionSupported() [0x%x]"), this);
ASSERT(State() == EInitialised && ChildFsNum() >= KMinChildFsNum);
//-- in debug mode check file systems compatibility: ALL childs must support this feature
for(TUint i=0; i<ChildFsNum(); ++i)
{
if( !(GetChildFileSystem(i)->IsExtensionSupported()))
{
DBG_STATEMENT(Fault(EIncompatibleFileSystems));
__PRINT(_L("#<<- ::IsExtensionSupported(): Incompatible file sytems!"));
return EFalse;
}
}
return ETrue;
}
//-----------------------------------------------------------------------------
/**
Return the initial default path.
*/
TInt CAutoMounterFileSystem::DefaultPath(TDes& aPath) const
{
__PRINT1(_L("#<<- CAutoMounterFileSystem::DefaultPath() [0x%x]"), this);
aPath=_L("?:\\");
aPath[0] = (TUint8) RFs::GetSystemDriveChar();
return KErrNone;
}
//-----------------------------------------------------------------------------
/**
Additional interfaces support.
*/
TInt CAutoMounterFileSystem::GetInterface(TInt aInterfaceId, TAny*& aInterface, TAny* aInput)
{
__PRINT2(_L("#<<- CAutoMounterFileSystem::GetInterface(id:%d) [0x%x]"), aInterfaceId, this);
switch(aInterfaceId)
{
//-- It is this filesystem private interface.
case EExtendedFunctionality:
aInterface = (CFileSystem::MFileSystemExtInterface*)this;
return KErrNone;
//-- a special case for child filesystems.
//-- ALL of them must respond to this interface query exactly the same way. I.e. It is impossible
//-- to have some of the child FS supporting it and some not.
case EProxyDriveSupport:
return DoProcessProxyDriveSupport();
default:
//-- This is the request to other (child file system) from the file server
//-- Actually, this part must never be called. File Server shall query the file system interfaces _after_ mounting the concrete FS
//-- calling TDrive::iCurrentMount->FileSystem().GetInterface()
ASSERT(0);
return CFileSystem::GetInterface(aInterfaceId, aInterface, aInput);
}
}
//-----------------------------------------------------------------------------
/**
Find out if _all_ child file systems support the proxy drive. All child file systems shall behave exactly the same way.
This is mostly for the USB host drives support. All child file systems should either support this feature or do not.
Otherwise, it may lead to very obscured cases of susystem misbehaviour.
@return KErrNone if all child file systems support proxy drives, or KErrNotSupported if all of them do not.
*/
TInt CAutoMounterFileSystem::DoProcessProxyDriveSupport()
{
__PRINT1(_L("#<<- CAutoMounterFileSystem::DoProcessProxyDriveSupport[0x%x]"), this);
ASSERT(State() == EInitialised);
const TUint cnt = ChildFsNum();
ASSERT(cnt >= KMinChildFsNum);
//-- query the default filesystem #0
const TBool bRes = GetChildFileSystem(KDefaultFSNo)->IsProxyDriveSupported();
//-- query the rest of child filesystems
for(TUint i=1; i<cnt; ++i)
{
const TBool b = GetChildFileSystem(i)->IsProxyDriveSupported();
if(BoolXOR(b, bRes))
Fault(EIncompatibleFileSystems);
}
return bRes ? KErrNone : KErrNotSupported;
}
//-----------------------------------------------------------------------------
/**
Get the child file system name by its index (enumerator).
@param aFsNumber index of the child FS 0...KMaxTInt
@param aFsName on success the child file system name will be placed into this buffer
@return KErrNone if there is a child FS name with index 'aFsNumber' (child FS 'aFsNumber' is supported by automounter)
KErrNotFound if child FS 'aFsNumber' is not supported
*/
TInt CAutoMounterFileSystem::GetSupportedFileSystemName(TInt aFsNumber, TDes& aFsName) const
{
__PRINT2(_L("#<<- CAutoMounterFileSystem::GetSupportedFileSystemName[0x%x](%d)"), this, aFsNumber);
if(aFsNumber == RFs::KRootFileSystem)
{//-- this is a name query for "root filesystem" or automounter
aFsName = Name(); //-- ourselves
return KErrNone;
}
//-- this is a query for one of the child filesystems
if((TUint)aFsNumber < ChildFsNum())
{
aFsName = iFSNames[aFsNumber];
return KErrNone;
}
return KErrNotFound;
}
//-----------------------------------------------------------------------------
/**
This is the only factory method that can be called by file server for this file system.
In this method the automounter sequentially tries to mount every child and on success produces the corresponding CMountCB object.
@param apDrive pointer to the TDrive, child FS will need this to access media.
@param apFileSystem on return will contain the pointer to the CFileSystem that has produced the proper CMountCB if
one of the child file system has recognised the volume layout.
@param aForceMount if ETrue the appropriate child FS (designated by aFsNameHash) will be forcedly mounted on the volume. for volume formatting purposes.
@param aFsNameHash if !=0 specifies the file system name, see TVolFormatParam::CalcFSNameHash(). 0 means "file system name is not specified"
@return pointer to the constructed CMountCB by one of the child file systems (and pointer to this child FS in apFileSystem)
NULL if it was impossible to produce proper CMountCB object.
*/
CMountCB* CAutoMounterFileSystem::NewMountExL(TDrive* apDrive, CFileSystem** apFileSystem, TBool aForceMount, TUint32 aFsNameHash)
{
__PRINT4(_L("#<<- CAutoMounterFileSystem::NewMountExL[0x%x] drv:%d, ForceMount:%d, FSNameHash:0x%x"), this, apDrive->DriveNumber(), aForceMount, aFsNameHash);
ASSERT(State() == EInitialised && apDrive);
//-- This method is usually called from the appropriate drive thread; this file system is intended to be bound to
//-- removable drives. Having removable drive runnind in the main file server thread means that something is terribly wrongly configured.
if(apDrive->IsSynchronous())
Fault(EWrongDriveAttributes);
if(ChildFsNum() < KMinChildFsNum)
Fault(EWrongConfiguration);
//-- if aForceMount is true, this means that the TDrive tries mount the filesystem by force for formatting because normal mounting has failed before.
//-- in our case it means that the file system on the volume hadn't been recognised by any child FS.
//-- aFsNameHash shall designate the file system to be forcedly mounted. Depending on this the appropriat CMounCB object will be produced.
//-- if aFsNameHash is 0, i.e. not provided, this method will fail with KErrNotFound if it can't to select appropriate child FS to be used.
if(aForceMount)
{
CFileSystem *pFS = NULL; //-- the FS selected to produce CMountCB for media formatting
if(aFsNameHash == 0)
{//-- the file system to mount forcedly is not specified
//-- 1. check if we have only 1 child FS supported. Quite a weird case, but an easy choice.
if(ChildFsNum() == 1)
{//-- use the single known child FS
__PRINT1(_L("#<<- only 1 child FS available for formatting: %S"), &iFSNames[KDefaultFSNo]);
pFS = GetChildFileSystem(KDefaultFSNo);
}
else if(iChildFsForDefFmt >= 0)
{//-- the parameter "child fs for default formatting" is defined. try using this
__PRINT1(_L("#<<- using the default child FS for formatting: %S"), &iFSNames[iChildFsForDefFmt]);
pFS = GetChildFileSystem(iChildFsForDefFmt);
}
else
{
__PRINT(_L("#<<- Unable to select appropriate child FS for formatting!"));
User::Leave(KErrNotFound);
}
}
else //if(aFsNameHash == 0)
{//-- try to find appropriate child FS by its name hash provided by the upper level
pFS = GetChildFileSysteByNameHash(aFsNameHash);
if(!pFS)
{
__PRINT(_L("#<<- no child FS found by its name hash!"));
ASSERT(0);
User::Leave(KErrNotFound);
}
}
//-- ask the selected FS to produce us a mount for media formating
ASSERT(pFS);
CMountCB* pMount = pFS->NewMountL();
ASSERT(pMount);
*apFileSystem = pFS;
return pMount;
}//if(aForceMount)
//-- try instantiate a new CMountCB depending on the file system on the media
ASSERT(!aForceMount);
CMountCB* pMatchedMount;
TInt nRes = TryMountFilesystem(apDrive, &pMatchedMount, apFileSystem);
if(nRes == KErrNone)
{
ASSERT(pMatchedMount);
return pMatchedMount;
}
User::Leave(nRes);
return NULL;
}
//-----------------------------------------------------------------------------
/**
Parse the string with child file system names. These names come from config, which can be either
estart.txt or a text debug property that can override it.
@param aList string descriptor with contents like "fat,exfat" i.e. it is comma separated file system names
*/
void CAutoMounterFileSystem::DoParseChildNames(const TDesC8& aList)
{
ASSERT(State() == EInitialising);
TBuf<0x100> fsName;
fsName.Copy(aList);
__PRINT1(_L("#<<- childFS list:'%S'"), &fsName);
//-- parse CSV config line and fill in the file system names array
const TChar chDelim = ','; //-- token delimiter, comma
TPtrC8 ptrCurrLine(aList);
for(TInt i=0; ;++i)
{
const TInt delimPos = ptrCurrLine.Locate(chDelim);
if(delimPos <= 0)
{
fsName.Copy(ptrCurrLine);
}
else
{
TPtrC8 temp(ptrCurrLine.Ptr(), delimPos);
fsName.Copy(temp);
}
fsName.Trim();
__PRINT2(_L("#<<- child FS[%d]: '%S'"), i, &fsName);
if(fsName.Length() <= 0)
Fault(EPluginInitialise);
//-- check if the FS name being appended is unique
for(TUint j=0; j<iFSNames.Count(); ++j)
{
if(iFSNames[j] == fsName)
{
Fault(EPluginInitialise);
}
}
TInt nRes = iFSNames.Append(fsName);
(void)nRes;
ASSERT(nRes ==KErrNone);
if(delimPos <=0 )
break;
ptrCurrLine.Set(ptrCurrLine.Ptr()+delimPos+1, ptrCurrLine.Length()-delimPos-1);
}
}
//-----------------------------------------------------------------------------
//-- if this macro is defined, the automounter will use a default hard-coded list of child file systems if it is not found in the estart.txt
//-- and this list will consist of one filesystem "FAT".
//-- otherwise automounter will panic if it can't find appropriate config in estart.txt
#define ALLOW_CONFIGLESS
/**
Process Automounter configuration
*/
void CAutoMounterFileSystem::ParseConfig()
{
ASSERT(State() == ENotInitialised);
SetState(EInitialising);
//-- 1. initialise the array of file system names. These names shall be listed in a config string.
//-- the config string is taken from estart.txt usually and its format is like this:
//-- section: [AutoMounter] and property "AM_FSNames fat,exfat"
//-- in debug version a special text property can override the config string. This allows controlling automounter from
//-- the test environment.
TBuf8<0x100> buf(0);
_LIT8(KSection, "AutoMounter"); ///< section name
_LIT8(KKey_ChildFsList, "AM_FSNames"); ///< a key for the CSV list of child file system names
_LIT8(KProp_DefFmtFsIdx, "AM_DefFmtFsIdx");///< a key for the optional parameter that specifies the child file system index, which will be used for formatting unrecognised media
TBool bUseEstart= ETrue;
#ifdef _DEBUG
TInt nRes;
const TUid KSID_Test1={0x10210EB3}; //-- SID of the test that will define and set test property to control volume mounting
const TUint KPropKey = 0; //-- property key
//-- in debug mode the property will override the estart.txt config
if(RProperty::Get(KSID_Test1, KPropKey, buf) == KErrNone)
{
__PRINT(_L("#<<- reading config from the debug propery:"));
TBuf<0x100> buf16(0);
buf16.Copy(buf);
__PRINT(buf16);
bUseEstart = EFalse;
}
#endif //_DEBUG
if(bUseEstart)
{//-- need to read data from the estart.txt, section [AutoMounter]
__PRINT(_L("#<<- reading config from estart.txt..."));
//-- 1. read and parse a CSV list of child file system names
//-- this is a mandatory parameter
TBool bFound = F32Properties::GetString(KSection, KKey_ChildFsList, buf);
if(!bFound)
{
__PRINT(_L("#<<- child FS list isn't found in config !"));
#ifdef ALLOW_CONFIGLESS
//-- let's use a default list of child file systems (consisting of only one: "FAT")
//-- if it is allowed. FAT FS is suggested to be available and already loaded by file server.
//-- otherwise it will panic later
_LIT(KDefChildFS_Config, "fat"); //-- default list
__PRINT1(_L("#<<- Using a default list:'%S'"), &KDefChildFS_Config());
buf.Copy(KDefChildFS_Config);
#else //ALLOW_CONFIGLESS
Fault(EPluginInitialise);
#endif //ALLOW_CONFIGLESS
}
DoParseChildNames(buf);
ASSERT(iFSNames.Count());
//-- 2. read a child FS index that can be used for unrecognisable media default formatting
//-- this is an optional parameter. If we have just 1 Child FS, this parameter doesn't make much sense.
if(iFSNames.Count() > 1)
{
TInt32 nVal;
bFound = F32Properties::GetInt(KSection, KProp_DefFmtFsIdx, nVal);
if(!bFound)
iChildFsForDefFmt = KErrNotFound;
else
{//-- check the index validity
if(nVal <0 || nVal >= (TInt)iFSNames.Count())
{
__PRINT1(_L("#<<- Bad DefFmtFsIdx value:%d"), nVal);
Fault(EPluginInitialise);
}
else
{
iChildFsForDefFmt = nVal;
}
}
}
else//if(iFSNames.Count() > 1)
{//-- it looks like we have a weird case when the automounter is configured with just 1 child file system
iChildFsForDefFmt = KDefaultFSNo;
}
}//if(bUseEstart)
else
{
#ifdef _DEBUG
//-- it looks like there is a test property that overrides the estart.txt settings. This property should contain
//-- a representation of a whole [AutoMounter] section in estart.txt and can have "multiple lines" separated by 0x0d,0x0a
buf.TrimAll();
TInt nPos1;
TInt nPos2;
//-- 1. process "AM_FSNames" key
nPos1 = buf.FindF(KKey_ChildFsList);
if(nPos1 < 0)
{
__PRINT(_L("#<<- child FS list isn't found!"));
Fault(EPluginInitialise);
}
TPtrC8 ptr(buf.Mid(nPos1));
nPos2 = ptr.Locate('\n');
if(nPos2 > 0)
ptr.Set(ptr.Left(nPos2));
TLex8 lex(ptr);
lex.NextToken();
lex.SkipSpace();
ptr.Set(lex.Remainder());
DoParseChildNames(ptr);
ASSERT(iFSNames.Count());
//-- 2. process "DefFmtFsIdx" key
nPos1 = buf.FindF(KProp_DefFmtFsIdx);
if(nPos1 > 0 && iFSNames.Count() > 1)
{
ptr.Set(buf.Mid(nPos1));
nPos2 = ptr.Locate('\n');
if(nPos2 > 0)
ptr.Set(ptr.Left(nPos2));
lex.Assign(ptr);
lex.NextToken();
lex.SkipSpace();
nRes = lex.Val(nPos1);
ASSERT(nRes == KErrNone);
ASSERT(nPos1 >= 0 && nPos1 < (TInt)iFSNames.Count());
iChildFsForDefFmt = nPos1;
}
else
{
iChildFsForDefFmt = KErrNotFound;
}
#endif //_DEBUG
}
}
//-----------------------------------------------------------------------------
/**
Initialise this file system. Reads and processes configuration, fills in file system names container, etc.
*/
void CAutoMounterFileSystem::InitialiseFileSystem()
{
__PRINT1(_L("#<<- CAutoMounterFileSystem::InitialiseFileSystem() [0x%x]"), this);
ASSERT(State() == ENotInitialised);
ParseConfig();
SetState(EInitialised);
//-- 2. check that the file server has all filesystems we need instantiated and stored in a global container
TUint cnt = ChildFsNum();
if(cnt < KMinChildFsNum)
{
__PRINT1(_L("#<<- ::InitialiseFileSystem(): too few File Systems bound: %d!"), cnt);
Fault(EPluginInitialise);
}
while(cnt--)
{
GetChildFileSystem(cnt);
}
}
//-----------------------------------------------------------------------------
/**
Tries to find out if some of the child file systems can be mounted on the given volume.
@param apDrive pointer to the TDrive, child FS will need this to access media.
@param on return will contain the pointer to the CMountCB object if some of the childs has decided that it can be mounted.
@param apFS on return will contain the pointer to the CFileSystem that has produced the proped CMountCB if
@return KErrNone on success, otherwise standard error code.
*/
TInt CAutoMounterFileSystem::TryMountFilesystem(TDrive* apDrive, CMountCB** apMount, CFileSystem** apFS)
{
__PRINT1(_L("#<<- CAutoMounterFileSystem::TryMountFilesystem()[0x%x]"), this);
const TUint KNumFS = ChildFsNum(); //-- number of child file systems supported
ASSERT(State() == EInitialised && (KNumFS >= KMinChildFsNum));
*apMount = NULL;
*apFS = NULL;
TInt nRes;
TUint cntFS;
CMountCB* pMountCB = NULL;
CFileSystem* pMatchedFS = NULL;
for(cntFS=0; cntFS < KNumFS; ++cntFS)
{
__PRINT2(_L("#<<-@@ trying FS[%d]:%S"), cntFS, &iFSNames[cntFS]);
CFileSystem* pFS = GetChildFileSystem(cntFS); //-- Find current filesystem object in the FileServer's global container
//-- 2. create CMountCB instance and set it up
pMountCB = NULL;
TRAP(nRes, pMountCB = pFS->NewMountL());
if(nRes != KErrNone)
{
return nRes;
}
ASSERT(pMountCB);
pMountCB->SetDrive(apDrive);
//-- 2.1 Firstly try using special CMountCB interface to find out if current FS can be mounted on this media
nRes = pMountCB->CheckFileSystemMountable();
if(nRes == KErrNotSupported)
{
//-- file system doesn't support this feature,
//-- 2.2 try to check the file system by mounting and dismounting later. It can result in some long activity, like FAT scanning etc.
TRAP(nRes, pMountCB->MountL(EFalse));
pMountCB->Dismounted(); //-- dismount the mountCB, it will be required in dismounted state anyway
}
//-- 2.3 analyse the result of mounting
if(nRes != KErrNone)
{
if(nRes == KErrLocked)
{//-- this is a special case; The media (SD card for example) is locked.
//-- Pretend that everything is OK and return CMountCB instance that is produced by the _default_ file system.
//-- locked media case will be handled by the file server later.
ASSERT(cntFS == KDefaultFSNo); //-- the default FS is tried first and must detect the locked media
pMatchedFS = pFS;
__PRINT(_L("#<<-@@ The media is LOCKED !"));
break;
}
//-- failed to mount the file system, most likey it is not recognised
pMountCB->Close(); //-- this is a self-destructing object!
pMountCB = NULL;
__PRINT(_L("#<<-@@ Mount FAILED !"));
}
else
{
//-- mounted OK, the file system is recognised
__PRINT(_L("#<<-@@ Mount OK!"));
pMatchedFS = pFS;
break;
}
}//for(cntFS=0; cntFS<KNumFS; ++cntFS)
if(cntFS >= KNumFS)
{//-- no one from the FS factories recognised the file system
__PRINT1(_L("#<<- ::TryMountFilesystem()[0x%x] No file system recognised!"), this);
SetName(&KFileSystemName_AutoMounter);
return KErrCorrupt;
}
ASSERT(pMountCB && pMatchedFS);
*apMount = pMountCB;
*apFS = pMatchedFS;
//-- set this FS name to the name of recognised FS. In this case the automounter "pretends" to be one of the child file systems on
//-- successful mounting. This behaviour was considered to be incorrect.
//TPtrC fsName = pMatchedFS->Name();
//SetName(&fsName);
return KErrNone;
}
//-----------------------------------------------------------------------------
/**
get the child file system object by the index in file child system names container
@param aIndex index in the iFSNames
@return pointer to the FS object if it is found, NULL otherwise
*/
CFileSystem* CAutoMounterFileSystem::GetChildFileSystem(TUint aIndex) const
{
ASSERT(State() == EInitialised && (ChildFsNum() >= KMinChildFsNum) && aIndex < ChildFsNum());
const TDesC& fsName = iFSNames[aIndex]; //-- registered child file system name
CFileSystem* pFS = GetFileSystem(fsName); //-- Find filesystem object in the FileServer's global container
if(!pFS)
{
__PRINT1(_L("#<<- CAutoMounterFileSystem::GetChildFileSystem() FileServer doesn't have FS:%S Added!"), &fsName);
Fault(EFileSysNotAdded);
}
return pFS;
}
//-----------------------------------------------------------------------------
/**
Find the child file system object by the file system name name hash.
@param aFsNameHash FS name hash
@return pointer to the FS object if it is found, NULL otherwise
*/
CFileSystem* CAutoMounterFileSystem::GetChildFileSysteByNameHash(TUint32 aFsNameHash) const
{
ASSERT(State() == EInitialised && (ChildFsNum() >= KMinChildFsNum) && aFsNameHash);
const TUint cnt = ChildFsNum();
for(TUint i=0; i<cnt; ++i)
{
if(aFsNameHash == iFSNames.GetStringHash(i))
{
const TDesC& fsName = iFSNames[i]; //-- registered child file system name
__PRINT2(_L("#<<- ::GetChildFileSysteByNameHash() found FsName:%S by hash:0x%x"), &fsName, aFsNameHash);
CFileSystem* pFS = GetFileSystem(fsName); //-- Find filesystem object in the FileServer's global container
ASSERT(pFS);
return pFS;
}
}
__PRINT1(_L("#<<- ::GetChildFileSysteByNameHash() No FS name found by hash:0x%x"), aFsNameHash);
return NULL;
}
//#######################################################################################################################################
//# XStringArray implementation
//#######################################################################################################################################
XStringArray::XStringArray()
{
}
XStringArray::~XStringArray()
{
Reset();
iStrings.Close();
}
//-----------------------------------------------------------------------------
void XStringArray::Reset()
{
iStrings.ResetAndDestroy();
}
//-----------------------------------------------------------------------------
const TDesC& XStringArray::operator[](TUint aIndex) const
{
if(aIndex >= (TUint)iStrings.Count())
Panic(EIndexOutOfRange);
HBufC* des=iStrings[aIndex];
return *des;
}
//-----------------------------------------------------------------------------
/**
Append file system name to the container. The name is converted to upper case.
@param aString name descriptor.
@return standard error code
*/
TInt XStringArray::Append(const TDesC& aString)
{
HBufC* pBuf = aString.Alloc();
if(!pBuf)
return KErrNoMemory;
//-- string being appended shall be unique
for(TUint i=0; i<Count(); ++i)
{
if(operator[](i).CompareF(aString) == 0)
{
ASSERT(0);
return KErrNone;
}
}
pBuf->Des().UpperCase(); //-- convert the FS name to upper-case in order to correctly calculate hash later
return iStrings.Append(pBuf);
}
//-----------------------------------------------------------------------------
/**
Get child FS name hash by index in the names container.
@param aIndex name index in the container
@return file system name hash (crc32)
*/
TUint32 XStringArray::GetStringHash(TUint aIndex) const
{
const TDesC& des = operator[](aIndex);
return TVolFormatParam::CalcFSNameHash(des);
}
void XStringArray::Panic(TPanicCode aPanicCode) const
{
_LIT(KPanicCat,"XStringArray");
User::Panic(KPanicCat, aPanicCode);
}