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\sfat\inc\sl_scandrv.h sl@0: // sl@0: // sl@0: sl@0: /** sl@0: @file sl@0: @internalTechnology sl@0: */ sl@0: sl@0: #ifndef SL_SCANDRV_H sl@0: #define SL_SCANDRV_H sl@0: sl@0: #include "sl_std.h" sl@0: sl@0: //--------------------------------------------------------------------------------------------------------------------------------- sl@0: sl@0: const TUint KMaxMatchingEntries = 2; ///< Maximum number of matching directory entries scan drive can fix. Any more indicates a fault in the file system sl@0: const TUint KMaxArrayDepth = 6; ///< Maximum array depth for cluster storage when KMaxScanDepth is reached sl@0: sl@0: /** Data structure used to store the location of a partial VFat entry */ sl@0: struct TPartVFatEntry sl@0: { sl@0: TEntryPos iEntryPos; ///< The position of the partial VFat entry sl@0: TFatDirEntry iEntry; ///< The Dos entry The VFat entries belong with sl@0: }; sl@0: sl@0: sl@0: /** Data structure used to store the locations of entries with matching start cluster numbers. */ sl@0: struct TMatchingStartCluster sl@0: { sl@0: TEntryPos iEntries[KMaxMatchingEntries]; ///< The positions of the matching entries sl@0: TUint iCount; ///< Count of matching entries sl@0: TUint iStartCluster; ///< The matching cluster number found in more than one entry sl@0: }; sl@0: sl@0: sl@0: //--------------------------------------------------------------------------------------------------------------------------------- sl@0: /** sl@0: This class is used for checking volume for FS errors and fixing a limited set of FS artefacts introduced by Rugged FAT on write failures. sl@0: It can operate in 2 modes: sl@0: sl@0: 1. "ScanDrive" mode, scan whole volume for possible Rugged FAT artefacts and fix them if possible. sl@0: 1.1 If there was no problem at all, then StartL() finishes normally and ProblemsDiscovered() returns ENoErrors. sl@0: 1.2 If there was Rugged FAT artefact and it had been successfully fixed, StartL() finishes normally and ProblemsDiscovered() returns EScanDriveDirError. sl@0: In this case the client may perform volum remounting, because FAT is very likely to have been changed. sl@0: 1.3 If there was a fatal error, like media failure or unfixable FS problem, StartL() will leave with some generic error code. sl@0: sl@0: 2. "CheckDisk" mode. check file system for known artefacts and return an error if _any_ problem discovered. sl@0: In this case StartL() _may_ leave with something like KErrCorrupt if there was a media failure or scan has stumbled across unknown FS error, sl@0: ProblemsDiscovered() _may_ return some code describing the problem. If StartL() did not leave, but ProblemsDiscovered() returns a code different sl@0: from ENoErrors, this means that there is FS corruption. sl@0: */ sl@0: class CScanDrive : public CBase sl@0: { sl@0: sl@0: public: sl@0: sl@0: ~CScanDrive(); sl@0: static CScanDrive* NewL(CFatMountCB* aMount); sl@0: void ConstructL(CFatMountCB* aMount); sl@0: sl@0: public: sl@0: sl@0: /** description of known problems that this scanned can deal with. Mostly used in "CheckDisk " mode */ sl@0: enum TGenericError sl@0: { sl@0: ENoErrors = 0, ///< 0 no errors discovered sl@0: EBadClusterNumber, ///< 1 cluster number that doesn't correspond to the max. amount of clusters on the volume sl@0: EClusterAlreadyInUse, ///< 2 cross-linked cluster chain sl@0: EBadClusterValue, ///< 3 also means "lost cluster" sl@0: EInvalidEntrySize, ///< 4 size of file/directory does not correspond to the cluster chain length sl@0: sl@0: EUnknownError = 95, ///< unknown error sl@0: sl@0: EScanDriveDirError=100 ///< 100 ScanDrive error sl@0: }; sl@0: sl@0: TGenericError ProblemsDiscovered() const; sl@0: sl@0: /** CScanDrive mode of operation */ sl@0: enum TScanDriveMode sl@0: { sl@0: EScanAndFix, ///< "ScanDrive" mode, scan whole volume for possible Rugged FAT artefacts and fix them sl@0: ECheckDisk, ///< "CheckDisk" mode. check file system for known artefacts and return an error if _any_ problem discovered sl@0: }; sl@0: sl@0: void StartL(TScanDriveMode aMode); sl@0: sl@0: private: sl@0: void PrintErrors(); sl@0: void CompareFatsL(TBool aStopOnFirstErrorFound) ; sl@0: void CompareAndFixFatsL(); sl@0: sl@0: void FixupDirErrorL(); sl@0: sl@0: void ReadMediaFatL(); sl@0: void DoParseFatL(); sl@0: void DoParseFat32L(); sl@0: void DoParseFat32Buf(const TPtrC8& aBuf, TUint32& aCurrFatEntry); sl@0: sl@0: TBool IsClusterUsedL(TUint aCluster); sl@0: void MarkClusterUsedL(TUint aCluster); sl@0: sl@0: TUint32 ReadFatL(TUint aClusterNum) ; sl@0: void FindSameStartClusterL(); sl@0: TInt FindStartClusterL(TInt aDirCluster); sl@0: void CheckDirStructureL(); sl@0: void CheckDirL(TUint32 aCluster); sl@0: void ProcessEntryL(const TFatDirEntry& aEntry); sl@0: TInt CheckEntryClusterL(const TFatDirEntry& aEntry, const TEntryPos& aEntryPos); sl@0: void RecordClusterChainL(TInt aCluster,TUint aSizeInBytes); sl@0: TBool MoveToVFatEndL(TEntryPos& aPos,TFatDirEntry& aEntry,TInt& aDirLength); sl@0: TBool IsValidVFatEntry(const TFatDirEntry& aEntry,TInt prevNum)const; sl@0: TBool IsDosEntry(const TFatDirEntry& aEntry)const; sl@0: void AddPartialVFatL(const TEntryPos& aStartPos, const TFatDirEntry& aEntry); sl@0: TBool AddMatchingEntryL(const TEntryPos& aEntryPos); sl@0: TInt GetReservedidL(const TEntryPos aVFatPos); sl@0: sl@0: void FixPartEntryL(); sl@0: void FixMatchingEntryL(); sl@0: void MovePastEntriesL(TEntryPos& aEntryPos,TFatDirEntry& aEntry,TInt aToMove,TInt& aDirEntries); sl@0: void AddToClusterListL(TInt aCluster); sl@0: inline TBool AlreadyExistsL(TInt aCluster)const; sl@0: inline TBool IsEndOfRootDir(const TEntryPos& aPos)const; sl@0: inline TBool IsEofF(TInt aVal)const; sl@0: inline TBool IsDirError()const; sl@0: void MoveToNextEntryL(TEntryPos& aPos); sl@0: void ReadDirEntryL(const TEntryPos& aPos,TFatDirEntry& aDirEntry); sl@0: sl@0: inline void IndicateErrorsFound(TGenericError aError); sl@0: inline TUint32 MaxClusters() const; sl@0: inline TBool CheckDiskMode() const; sl@0: sl@0: protected: sl@0: sl@0: /** sl@0: Internal ScanDrive mode specific errors. In Rugged FAT mode (current implementatio) any type of error of this kind can occur only once and it will be fixed. sl@0: Othersise the FS is considered to be corrupted sl@0: */ sl@0: enum TDirError sl@0: { sl@0: ENoDirError= 0, ///< no errors found sl@0: EScanMatchingEntry=1, ///< Two entries pointing to the same cluster chain; Rugged FAT rename/replace artefact sl@0: EScanPartEntry, ///< Deleted DOS entry and orphaned VFAT ones from the same entryset; Rugged FAT 'file/dir delete' artefact sl@0: }; sl@0: sl@0: sl@0: private: sl@0: CFatMountCB* iMount; ///< The owning Fat mount sl@0: sl@0: TPartVFatEntry iPartEntry; ///< Storage for a partial VFat entry set error, see EScanPartEntry sl@0: TMatchingStartCluster iMatching; ///< Storage for Matching start cluster error, see EScanMatchingEntry sl@0: sl@0: TDirError iDirError; ///< Indicates the error tpye found also used to indicate if an error has occured sl@0: TInt iDirsChecked; ///< Count of the number of directories checked sl@0: TInt iRecursiveDepth; ///< Depth of recursion the scan has reached sl@0: RArray* iClusterListArray[KMaxArrayDepth]; ///< Size in bytes of the bit packed Fat Cluster list array used when maximum depth has been reached so that directory may be re-visited. Avoid stack overflow sl@0: sl@0: TUint iListArrayIndex; ///< Current position into cluster list array sl@0: TUint32 iTruncationCluster; ///< Cluster at which cluster chain truncation should take place, used for truncation errors sl@0: TUint32 iMaxClusters; ///< Max. amount of clusters on the volume sl@0: sl@0: RBitVector iMediaFatBits; ///< Storage for bit packed Fat read from media sl@0: RBitVector iScanFatBits; ///< Storage for bit packed Fat built up by the scan sl@0: sl@0: TGenericError iGenericError; ///< FS error that is discovered by scanning in any mode sl@0: TScanDriveMode iScanDriveMode; ///< mode of operation sl@0: }; sl@0: sl@0: sl@0: sl@0: #endif //SL_SCANDRV_H sl@0: sl@0: