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".
8 // Initial Contributors:
9 // Nokia Corporation - initial contribution.
14 // f32\sfat\inc\sl_scandrv.h
28 //---------------------------------------------------------------------------------------------------------------------------------
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
33 /** Data structure used to store the location of a partial VFat entry */
36 TEntryPos iEntryPos; ///< The position of the partial VFat entry
37 TFatDirEntry iEntry; ///< The Dos entry The VFat entries belong with
41 /** Data structure used to store the locations of entries with matching start cluster numbers. */
42 struct TMatchingStartCluster
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
50 //---------------------------------------------------------------------------------------------------------------------------------
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:
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.
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.
66 class CScanDrive : public CBase
72 static CScanDrive* NewL(CFatMountCB* aMount);
73 void ConstructL(CFatMountCB* aMount);
77 /** description of known problems that this scanned can deal with. Mostly used in "CheckDisk " mode */
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
86 EUnknownError = 95, ///< unknown error
88 EScanDriveDirError=100 ///< 100 ScanDrive error
91 TGenericError ProblemsDiscovered() const;
93 /** CScanDrive mode of operation */
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
100 void StartL(TScanDriveMode aMode);
104 void CompareFatsL(TBool aStopOnFirstErrorFound) ;
105 void CompareAndFixFatsL();
107 void FixupDirErrorL();
109 void ReadMediaFatL();
111 void DoParseFat32L();
112 void DoParseFat32Buf(const TPtrC8& aBuf, TUint32& aCurrFatEntry);
114 TBool IsClusterUsedL(TUint aCluster);
115 void MarkClusterUsedL(TUint aCluster);
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);
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);
143 inline void IndicateErrorsFound(TGenericError aError);
144 inline TUint32 MaxClusters() const;
145 inline TBool CheckDiskMode() const;
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
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
162 CFatMountCB* iMount; ///< The owning Fat mount
164 TPartVFatEntry iPartEntry; ///< Storage for a partial VFat entry set error, see EScanPartEntry
165 TMatchingStartCluster iMatching; ///< Storage for Matching start cluster error, see EScanMatchingEntry
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
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
176 RBitVector iMediaFatBits; ///< Storage for bit packed Fat read from media
177 RBitVector iScanFatBits; ///< Storage for bit packed Fat built up by the scan
179 TGenericError iGenericError; ///< FS error that is discovered by scanning in any mode
180 TScanDriveMode iScanDriveMode; ///< mode of operation
185 #endif //SL_SCANDRV_H