sl@0: // Copyright (c) 1998-2009 Nokia Corporation and/or its subsidiary(-ies). sl@0: // All rights reserved. sl@0: // This component and the accompanying materials are made available sl@0: // under the terms of the License "Eclipse Public License v1.0" sl@0: // which accompanies this distribution, and is available sl@0: // at the URL "http://www.eclipse.org/legal/epl-v10.html". sl@0: // sl@0: // Initial Contributors: sl@0: // Nokia Corporation - initial contribution. sl@0: // sl@0: // Contributors: sl@0: // sl@0: // Description: sl@0: // f32\sfat32\inc\fat_table32.h sl@0: // FAT32 File Allocation Table classes definitions sl@0: // sl@0: // sl@0: sl@0: /** sl@0: @file sl@0: @internalTechnology sl@0: */ sl@0: sl@0: #ifndef FAT_TABLE_32_H sl@0: #define FAT_TABLE_32_H sl@0: sl@0: //--------------------------------------------------------------------------------------------------------------------------------------- sl@0: sl@0: class CFatHelperThreadBase; sl@0: class CFat32ScanThread; sl@0: sl@0: TInt FAT32_ScanThread(TAny* apHostObject); sl@0: sl@0: //--------------------------------------------------------------------------------------------------------------------------------------- sl@0: sl@0: /** sl@0: Fat table abstraction for all media types except RAM. sl@0: Supports FAT12/16/32 sl@0: */ sl@0: class CAtaFatTable : public CFatTable sl@0: { sl@0: sl@0: public: sl@0: sl@0: static CAtaFatTable* NewL(CFatMountCB& aOwner); sl@0: ~CAtaFatTable(); sl@0: sl@0: //-- overrides from th ebase class sl@0: TUint32 ReadL(TUint32 aFatIndex) const; sl@0: void WriteL(TUint32 aFatIndex, TUint32 aValue); sl@0: void MountL(const TMountParams& aMountParam); sl@0: sl@0: TInt64 DataPositionInBytes(TUint32 aCluster) const; sl@0: sl@0: void InitializeL(); sl@0: void Dismount(TBool aDiscardDirtyData); sl@0: void FlushL(); sl@0: sl@0: void InvalidateCacheL(); sl@0: void InvalidateCacheL(TInt64 aPos, TUint32 aLength); sl@0: sl@0: TUint32 FreeClusterHint() const; sl@0: TUint32 NumberOfFreeClusters(TBool aSyncOperation=EFalse) const; sl@0: sl@0: TBool ConsistentState() const; sl@0: sl@0: inline void AcquireLock() const {iDriveInteface.AcquireLock();} sl@0: inline void ReleaseLock() const {iDriveInteface.ReleaseLock();} sl@0: sl@0: inline TDriveInterface& DriveInterface() const; sl@0: inline CFatMountCB* OwnerMount() const; sl@0: sl@0: sl@0: private: sl@0: sl@0: /** internal states of this object */ sl@0: enum TState sl@0: { sl@0: ENotInitialised, ///< 0 initial invalid state sl@0: EInitialised, ///< 1 initialised, i.e. internal objects created, but unusable becasue number of free entries isn't known sl@0: EMounting, ///< 2 mounting started sl@0: EMounted, ///< 3 successfully mounted; number of free entries is known. This is the only consistent state. sl@0: EDismounted, ///< 4 FAT table object is dismounted sl@0: EFreeClustersScan, ///< 5 FAT32 scan thread is currently scanning FAT table for free clusters sl@0: EMountAborted ///< 6 Mounting failed, probably because of FAT32 free clusters scan thread failure. sl@0: }; sl@0: sl@0: public: sl@0: sl@0: /** A helper class used in FAT scanning for free clusters*/ sl@0: class TFatScanParam sl@0: { sl@0: public: sl@0: inline TFatScanParam(); sl@0: sl@0: public: sl@0: TUint32 iEntriesScanned; ///< total number of FAT entries scanned by DoParseFatBuf() sl@0: TUint32 iFirstFree; ///< first free FAT entry found sl@0: TUint32 iCurrFreeEntries; ///< current number of free FAT entries found by DoParseFatBuf() sl@0: TUint32 iCurrOccupiedEntries; ///< current number of non-free FAT entries found by DoParseFatBuf() sl@0: }; sl@0: sl@0: sl@0: private: sl@0: CAtaFatTable(CFatMountCB& aOwner); sl@0: sl@0: void RequestRawWriteAccess(TInt64 aPos, TUint32 aLen) const; sl@0: sl@0: void SetFreeClusters(TUint32 aFreeClusters); sl@0: void SetFreeClusterHint(TUint32 aCluster); sl@0: void CountFreeClustersL(); sl@0: void CreateCacheL(); sl@0: sl@0: virtual void DecrementFreeClusterCount(TUint32 aCount); sl@0: virtual void IncrementFreeClusterCount(TUint32 aCount); sl@0: virtual TUint32 FindClosestFreeClusterL(TUint32 aCluster); sl@0: sl@0: void DoCountFreeClustersL(); sl@0: void DoParseFatBuf(const TPtrC8& aBuf, TFatScanParam& aScanParam) const; sl@0: sl@0: TBool RequestFreeClusters(TUint32 aClustersRequired) const; sl@0: void DoLaunchFat32FreeSpaceScanThreadL(); sl@0: void DestroyHelperThread(); sl@0: sl@0: inline TState State() const; sl@0: inline void SetState(TState aState); sl@0: sl@0: private: sl@0: sl@0: sl@0: CFatCacheBase* iCache; ///< FAT cache, fixed or LRU depending on the FAT type sl@0: TDriveInterface& iDriveInteface; ///< reference to the drive interface sl@0: CFatHelperThreadBase* ipHelperThread; ///< helper thread object pointer. NULL if it is not present sl@0: TState iState; ///< state of this object sl@0: sl@0: //-- friends sl@0: friend TInt FAT32_ScanThread(TAny* apHostObject); sl@0: friend class CFat32ScanThread; sl@0: friend class CFat32FreeSpaceScanner; sl@0: friend class CFat32BitCachePopulator; sl@0: }; sl@0: sl@0: //--------------------------------------------------------------------------------------------------------------------------------------- sl@0: sl@0: /** sl@0: Fat table abstraction for RAM media type. sl@0: Supports FAT16/32 only sl@0: */ sl@0: class CRamFatTable : public CFatTable sl@0: { sl@0: public: sl@0: sl@0: static CRamFatTable* NewL(CFatMountCB& aOwner); sl@0: void InitializeL(); sl@0: void MountL(const TMountParams& aMountParam); sl@0: sl@0: sl@0: TUint32 ReadL(TUint32 aFatIndex) const; sl@0: void WriteL(TUint32 aFatIndex, TUint32 aValue); sl@0: TInt64 DataPositionInBytes(TUint32 aCluster) const; sl@0: void FreeClusterListL(TUint32 aCluster); sl@0: TUint32 AllocateSingleClusterL(TUint32 aNearestCluster); sl@0: void ExtendClusterListL(TUint32 aNumber,TInt& aCluster); sl@0: TUint32 AllocateClusterListL(TUint32 aNumber,TUint32 aNearestCluster); sl@0: sl@0: private: sl@0: CRamFatTable(CFatMountCB& aOwner); sl@0: sl@0: inline TUint8 *RamDiskBase() const; sl@0: inline TInt AllocateClusterNumber(); sl@0: inline void WriteFatTable(TInt aFatIndex,TInt aValue); sl@0: inline void WriteFatTable(TInt aFatIndex,TInt aFatValue,TInt anIndirectionTableValue); sl@0: inline void ReadIndirectionTable(TUint32& aCluster) const; sl@0: inline void WriteIndirectionTable(TInt aFatIndex,TInt aValue); sl@0: inline TUint8* MemCopy(TAny* aTrg,const TAny* aSrc,TInt aLength); sl@0: inline TUint8* MemCopyFillZ(TAny* aTrg, TAny* aSrc, TInt aLength); sl@0: inline void ZeroFillCluster(TInt aCluster); sl@0: sl@0: void UpdateIndirectionTable(TUint32 aStartCluster,TUint32 anEndCluster,TInt aNum); sl@0: sl@0: protected: sl@0: sl@0: TInt iFatTablePos; ///< Current position in the fat table sl@0: TInt iIndirectionTablePos; ///< Current position in indirection table, second fat used for this sl@0: TUint8* iRamDiskBase; ///< Pointer to the Ram disk base sl@0: }; sl@0: sl@0: sl@0: sl@0: //--------------------------------------------------------------------------------------------------------------------------------------- sl@0: sl@0: /** sl@0: Abstract base class for the FAT32 helper threads. sl@0: Provides basic functionality of the helper threads and interface to the owher CAtaFatTable. sl@0: */ sl@0: class CFatHelperThreadBase : public CBase sl@0: { sl@0: public: sl@0: sl@0: /** Possible types of the FAT32 helper threads */ sl@0: enum TFatHelperThreadType sl@0: { sl@0: EInvalidType, ///< invalid type sl@0: EFreeSpaceScanner, ///< Free FAT32 entries counter, see CFat32FreeSpaceScanner sl@0: EBitCachePopulator ///< FAT32 bit supercache populating thread. sl@0: }; sl@0: sl@0: /** this object states, mostly related to the worker thread activity and results */ sl@0: enum TState sl@0: { sl@0: EInvalid, ///< invalid initial state sl@0: ENotStarted, ///< the worker thread hasn't started yet sl@0: EWorking, ///< worker thread is working sl@0: EFinished_OK, ///< worker thread has successfully finished, everything is fine. sl@0: EFailed ///< worker thread failed to finish its job for some reason; see the thread completion status sl@0: }; sl@0: sl@0: public: sl@0: ~CFatHelperThreadBase(); sl@0: sl@0: //-- virtual interface sl@0: virtual TFatHelperThreadType Type() const = 0; sl@0: virtual TInt Launch()=0; sl@0: virtual void RequestFatEntryWriteAccess(TUint32 aFatIndex) const=0; sl@0: sl@0: //-- non-virtual interface for external user only sl@0: void Close(); sl@0: void ForceStop(); sl@0: sl@0: inline TState State() const; sl@0: sl@0: inline void Suspend() const; sl@0: inline void Resume() const; sl@0: sl@0: inline TInt ThreadCompletionCode() const; sl@0: inline TBool ThreadWorking() const; sl@0: sl@0: inline void BoostPriority(TBool aBoost) const; sl@0: inline TBool IsPriorityBoosted() const; sl@0: sl@0: inline TThreadId ThreadId() const; sl@0: sl@0: TInt WaitToFinish() const; sl@0: sl@0: protected: sl@0: CFatHelperThreadBase(CAtaFatTable& aOwner); sl@0: sl@0: //-- outlaws sl@0: CFatHelperThreadBase(); sl@0: CFatHelperThreadBase(const CFatHelperThreadBase&); sl@0: CFatHelperThreadBase& operator=(const CFatHelperThreadBase&); sl@0: sl@0: /** the worker thread priorities values */ sl@0: enum sl@0: { sl@0: EHelperPriorityNormal = EPriorityMuchLess, ///< FAT32 Helper thread normal priority (assigned on start) sl@0: EHelperPriorityBoosted = EPriorityNormal ///< FAT32 Helper thread bosted priority sl@0: }; sl@0: sl@0: TInt DoLaunchThread(TThreadFunction aFunction, TAny* aThreadParameter); sl@0: sl@0: inline void SetState(TState aState); sl@0: inline TBool AllowedToLive() const; sl@0: inline void AllowToLive(TBool aAllow); sl@0: sl@0: protected: sl@0: CAtaFatTable& iOwner; ///< owner, CAtaFatTable sl@0: sl@0: private: sl@0: TState iState; ///< internal state of this object sl@0: RThread iThread; ///< FAT helper thread handle sl@0: mutable TRequestStatus iThreadStatus; ///< helper thread logon status sl@0: sl@0: TBool iAllowedToLive : 1; ///< if EFalse the worker thread must gracefully finish ASAP. sl@0: mutable TBool iPriorityBoosted : 1; ///< ETrue when thread priority is boosted by BoostPriority() call sl@0: sl@0: sl@0: }; sl@0: sl@0: //--------------------------------------------------------------------------------------------------------------------------------------- sl@0: /** sl@0: Abstract base class for the FAT32 helper threads that read FAT by big chunks of data and parse it. sl@0: Provides basic functionality of the helper threads and interface to the owher CAtaFatTable. sl@0: */ sl@0: class CFat32ScanThread : public CFatHelperThreadBase sl@0: { sl@0: public: sl@0: sl@0: virtual TInt Launch(); sl@0: sl@0: protected: sl@0: CFat32ScanThread(CAtaFatTable& aOwner); sl@0: sl@0: //-- virtual private interface for the thread function. sl@0: virtual TInt Thread_Preamble(); sl@0: virtual TInt Thread_Postamble(TInt aResult); sl@0: virtual TInt Thread_ProcessCollectedFreeEntries(const CAtaFatTable::TFatScanParam& aFatScanParam)=0; sl@0: sl@0: friend TInt FAT32_ScanThread(TAny* apHostObject); ///< FAT32 scanner thread function, generic functionality for derived classes sl@0: sl@0: protected: sl@0: sl@0: RBuf8 iFatChunkBuf; ///< a buffer for reading FAT directly from the media sl@0: sl@0: TTime iTimeStart; ///< thread start time, used to measure how long thread worked sl@0: TTime iTimeEnd; ///< thread end time, used to measure how long thread worked sl@0: sl@0: CFatBitCache *ipFatBitCache; ///< interface to the FAT bit supercache (if it is present) sl@0: }; sl@0: sl@0: sl@0: //--------------------------------------------------------------------------------------------------------------------------------------- sl@0: /** sl@0: FAT32 free FAT entries scanner thread. sl@0: Represents transient FAT32 helper thread that can be launched on FAT table object mounting stage and will be sl@0: counting free FAT entries in order to find out free space on the volume. sl@0: */ sl@0: class CFat32FreeSpaceScanner : public CFat32ScanThread sl@0: { sl@0: public: sl@0: sl@0: static CFat32FreeSpaceScanner* NewL(CAtaFatTable& aOwner); sl@0: sl@0: virtual inline TFatHelperThreadType Type() const; sl@0: sl@0: private: sl@0: CFat32FreeSpaceScanner(CAtaFatTable& aOwner); sl@0: sl@0: void RequestFatEntryWriteAccess(TUint32 aFatIndex) const; sl@0: sl@0: //-- virtual private interface for the thread function. sl@0: TInt Thread_Preamble(); sl@0: TInt Thread_Postamble(TInt aResult); sl@0: TInt Thread_ProcessCollectedFreeEntries(const CAtaFatTable::TFatScanParam& aFatScanParam); sl@0: //-- sl@0: sl@0: private: sl@0: sl@0: void SetClustersScanned(TUint32 aClusters); sl@0: TUint32 ClustersScanned() const; sl@0: sl@0: friend TInt FAT32_ScanThread(TAny* apHostObject); ///< FAT32 scanner thread function, generic functionality for CFat32ScanThread derived classes sl@0: sl@0: private: sl@0: sl@0: enum sl@0: { sl@0: KFatChunkBufSize_Small = 16*K1KiloByte, //-- buffer size for reading small FAT tables sl@0: KFatChunkBufSize_Big = 64*K1KiloByte, //-- buffer size for reading large FAT tables sl@0: sl@0: KBigSzFat_Threshold = 2*K1MegaByte, //-- if FAT table size > this value, larger FAT read chunk (KFatChunkBufSize_Big) will be used sl@0: }; sl@0: sl@0: TUint32 iClustersScanned; ///< Number of FAT entries already scanned by the thread. Counts from the beginning of the FAT sl@0: sl@0: //-- volume space treshold in bytes that causes CMountCB::SetDiskSpaceChange() to be called by FAT32 free space scanner thread. sl@0: //-- This thread will be calling CMountCB::SetDiskSpaceChange() after processing number of FAT32 entries corresponding to sl@0: //-- this amount of space in FAT clusters. e.g. after processing amount of FAT32 entries comprising 256MB volume space sl@0: enum {KVolSpaceNotifyThreshold = 256 * K1MegaByte}; sl@0: sl@0: TUint32 iEntriesNotifyThreshold; ///< the value of FAT32 entries need to be counted for CMountCB::SetDiskSpaceChange() call sl@0: TUint32 iNfyThresholdInc; ///< Threshold increment in FAT32 entries. sl@0: sl@0: sl@0: }; sl@0: sl@0: //--------------------------------------------------------------------------------------------------------------------------------------- sl@0: /** sl@0: FAT32 Bit supercache populating thread. sl@0: Represents transient FAT32 helper thread that is populating bit supercache in backgroud. sl@0: */ sl@0: class CFat32BitCachePopulator : public CFat32ScanThread sl@0: { sl@0: public: sl@0: sl@0: static CFat32BitCachePopulator* NewL(CAtaFatTable& aOwner); sl@0: sl@0: virtual inline TFatHelperThreadType Type() const; sl@0: sl@0: private: sl@0: CFat32BitCachePopulator(CAtaFatTable& aOwner); sl@0: sl@0: void RequestFatEntryWriteAccess(TUint32 aFatIndex) const; sl@0: sl@0: //-- virtual private interface for the thread function. sl@0: TInt Thread_Preamble(); sl@0: TInt Thread_Postamble(TInt aResult); sl@0: TInt Thread_ProcessCollectedFreeEntries(const CAtaFatTable::TFatScanParam& aFatScanParam); sl@0: //-- sl@0: sl@0: private: sl@0: friend TInt FAT32_ScanThread(TAny* apHostObject); ///< FAT32 scanner thread function, generic functionality for CFat32ScanThread derived classes sl@0: enum {KFatChunkBufSize = 16*K1KiloByte}; //-- buffer size for FAT reading sl@0: sl@0: private: sl@0: TUint32 iTotalOccupiedFatEntries; ///< total counted number of non-free FAT entries sl@0: sl@0: }; sl@0: sl@0: sl@0: //--------------------------------------------------------------------------------------------------------------------------------------- sl@0: sl@0: #include "fat_table32.inl" sl@0: sl@0: #endif //FAT_TABLE_32_H sl@0: sl@0: sl@0: sl@0: sl@0: sl@0: sl@0: sl@0: sl@0: sl@0: sl@0: sl@0: sl@0: sl@0: sl@0: sl@0: sl@0: sl@0: sl@0: sl@0: sl@0: sl@0: sl@0: