os/kernelhwsrv/userlibandfileserver/fileserver/sfat32/inc/sl_scandrv.h
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
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 the License "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".
     7 //
     8 // Initial Contributors:
     9 // Nokia Corporation - initial contribution.
    10 //
    11 // Contributors:
    12 //
    13 // Description:
    14 // f32\sfat\inc\sl_scandrv.h
    15 // 
    16 //
    17 
    18 /**
    19  @file
    20  @internalTechnology
    21 */
    22 
    23 #ifndef SL_SCANDRV_H
    24 #define SL_SCANDRV_H
    25 
    26 #include "sl_std.h"
    27 
    28 //---------------------------------------------------------------------------------------------------------------------------------
    29 
    30 const TUint KMaxMatchingEntries		= 2;        ///< Maximum number of matching directory entries scan drive can fix. Any more indicates a fault in the file system
    31 const TUint KMaxArrayDepth			= 6;        ///< Maximum array depth for cluster storage when KMaxScanDepth is reached
    32 
    33 /** Data structure used to store the location of a partial VFat entry */
    34 struct TPartVFatEntry
    35 	{
    36 	TEntryPos    iEntryPos; ///< The position of the partial VFat entry
    37 	TFatDirEntry iEntry;    ///< The Dos entry The VFat entries belong with
    38 	};
    39 
    40 
    41 /** Data structure used to store the locations of entries with matching start cluster numbers. */
    42 struct TMatchingStartCluster
    43 	{
    44 	TEntryPos   iEntries[KMaxMatchingEntries]; ///< The positions of the matching entries
    45 	TUint       iCount;         ///< Count of matching entries
    46 	TUint       iStartCluster;  ///< The matching cluster number found in more than one entry
    47 	};
    48 
    49 
    50 //---------------------------------------------------------------------------------------------------------------------------------
    51 /**
    52     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.
    53     It can operate in 2 modes:
    54     
    55     1. "ScanDrive" mode, scan whole volume for possible Rugged FAT artefacts and fix them if possible. 
    56         1.1 If there was no problem at all, then StartL() finishes normally and ProblemsDiscovered() returns ENoErrors.
    57         1.2 If there was Rugged FAT artefact and it had been successfully fixed, StartL() finishes normally and ProblemsDiscovered() returns EScanDriveDirError.
    58             In this case the client may perform volum remounting, because FAT is very likely to have been changed.
    59         1.3 If there was a fatal error, like media failure or unfixable FS problem, StartL() will leave with some generic error code.
    60 
    61     2.  "CheckDisk" mode. check file system for known artefacts and return an error if _any_ problem discovered.
    62         In this case StartL() _may_ leave with something like KErrCorrupt if there was a media failure or scan has stumbled across unknown FS error, 
    63         ProblemsDiscovered() _may_ return some code describing the problem. If StartL() did not leave, but ProblemsDiscovered() returns a code different 
    64         from ENoErrors, this means that there is FS corruption. 
    65 */
    66 class CScanDrive : public CBase
    67 	{
    68 
    69 public:
    70 
    71 	~CScanDrive();
    72 	static CScanDrive* NewL(CFatMountCB* aMount);
    73 	void ConstructL(CFatMountCB* aMount);
    74 
    75 public:
    76 
    77     /** description of known problems that this scanned can deal with. Mostly used in "CheckDisk " mode */
    78     enum TGenericError
    79         {
    80         ENoErrors = 0,          ///< 0  no errors discovered
    81         EBadClusterNumber,      ///< 1  cluster number that doesn't correspond to the max. amount of clusters on the volume
    82         EClusterAlreadyInUse,   ///< 2  cross-linked cluster chain
    83         EBadClusterValue,       ///< 3  also means "lost cluster"
    84         EInvalidEntrySize,      ///< 4  size of file/directory does not correspond to the cluster chain length
    85         
    86         EUnknownError = 95,     ///< unknown error
    87 
    88         EScanDriveDirError=100  ///< 100 ScanDrive error
    89         };
    90 
    91     TGenericError ProblemsDiscovered() const;
    92 
    93 	/** CScanDrive mode of operation */
    94     enum TScanDriveMode
    95         {
    96         EScanAndFix, ///< "ScanDrive" mode, scan whole volume for possible Rugged FAT artefacts and fix them
    97         ECheckDisk,  ///< "CheckDisk" mode. check file system for known artefacts and return an error if _any_ problem discovered
    98         };
    99     
   100     void StartL(TScanDriveMode aMode);
   101 
   102 private:
   103 	void PrintErrors();
   104 	void CompareFatsL(TBool aStopOnFirstErrorFound) ;
   105 	void CompareAndFixFatsL();
   106 
   107     void FixupDirErrorL();
   108 
   109 	void ReadMediaFatL();
   110     void DoParseFatL();
   111     void DoParseFat32L();
   112     void DoParseFat32Buf(const TPtrC8& aBuf, TUint32& aCurrFatEntry);
   113 
   114 	TBool IsClusterUsedL(TUint aCluster);
   115 	void MarkClusterUsedL(TUint aCluster);
   116 
   117 	TUint32 ReadFatL(TUint aClusterNum) ;
   118 	void FindSameStartClusterL();
   119 	TInt FindStartClusterL(TInt aDirCluster);
   120 	void CheckDirStructureL();
   121 	void CheckDirL(TUint32 aCluster);
   122 	void ProcessEntryL(const TFatDirEntry& aEntry);
   123 	TInt CheckEntryClusterL(const TFatDirEntry& aEntry, const TEntryPos& aEntryPos);
   124 	void RecordClusterChainL(TInt aCluster,TUint aSizeInBytes);
   125 	TBool MoveToVFatEndL(TEntryPos& aPos,TFatDirEntry& aEntry,TInt& aDirLength);
   126 	TBool IsValidVFatEntry(const TFatDirEntry& aEntry,TInt prevNum)const;
   127 	TBool IsDosEntry(const TFatDirEntry& aEntry)const;
   128 	void AddPartialVFatL(const TEntryPos& aStartPos, const TFatDirEntry& aEntry);
   129 	TBool AddMatchingEntryL(const TEntryPos& aEntryPos);
   130 	TInt GetReservedidL(const TEntryPos aVFatPos);
   131 	
   132 	void FixPartEntryL();
   133 	void FixMatchingEntryL();
   134 	void MovePastEntriesL(TEntryPos& aEntryPos,TFatDirEntry& aEntry,TInt aToMove,TInt& aDirEntries);
   135 	void AddToClusterListL(TInt aCluster);
   136 	inline TBool AlreadyExistsL(TInt aCluster)const;
   137 	inline TBool IsEndOfRootDir(const TEntryPos& aPos)const;
   138 	inline TBool IsEofF(TInt aVal)const;
   139 	inline TBool IsDirError()const;
   140 	void MoveToNextEntryL(TEntryPos& aPos);
   141 	void ReadDirEntryL(const TEntryPos& aPos,TFatDirEntry& aDirEntry);
   142 
   143     inline void IndicateErrorsFound(TGenericError aError);
   144     inline TUint32 MaxClusters() const;
   145     inline TBool CheckDiskMode() const;
   146 
   147 protected:
   148 	
   149     /**
   150     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.
   151     Othersise the FS is considered to be corrupted
   152     */
   153     enum TDirError 
   154         {
   155         ENoDirError= 0,         ///< no errors found
   156         EScanMatchingEntry=1,   ///< Two entries pointing to the same cluster chain; Rugged FAT rename/replace artefact
   157         EScanPartEntry,         ///< Deleted DOS entry and orphaned VFAT ones from the same entryset; Rugged FAT 'file/dir delete' artefact
   158         };
   159 
   160 
   161 private:
   162 	CFatMountCB*            iMount;             ///< The owning Fat mount
   163 	
   164     TPartVFatEntry          iPartEntry;         ///< Storage for a partial VFat entry set error, see EScanPartEntry
   165 	TMatchingStartCluster   iMatching;          ///< Storage for Matching start cluster error, see EScanMatchingEntry
   166 	
   167     TDirError               iDirError;          ///< Indicates the error tpye found also used to indicate if an error has occured
   168     TInt                    iDirsChecked;       ///< Count of the number of directories checked
   169 	TInt                    iRecursiveDepth;    ///< Depth of recursion the scan has reached
   170 	RArray<TInt>*           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
   171 	
   172     TUint                   iListArrayIndex;    ///< Current position into cluster list array
   173 	TUint32                 iTruncationCluster; ///< Cluster at which cluster chain truncation should take place, used for truncation errors
   174 	TUint32                 iMaxClusters;       ///< Max. amount of clusters on the volume
   175 
   176     RBitVector              iMediaFatBits;      ///< Storage for bit packed Fat read from media 
   177     RBitVector              iScanFatBits;       ///< Storage for bit packed Fat built up by the scan
   178 
   179     TGenericError           iGenericError;      ///< FS error that is discovered by scanning in any mode  
   180     TScanDriveMode          iScanDriveMode;     ///< mode of operation
   181 	};
   182 
   183 
   184 
   185 #endif //SL_SCANDRV_H
   186 
   187