NTFSVolume.cs 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435
  1. /* Copyright (C) 2014-2018 Tal Aloni <tal.aloni.il@gmail.com>. All rights reserved.
  2. *
  3. * You can redistribute this program and/or modify it under the terms of
  4. * the GNU Lesser Public License as published by the Free Software Foundation,
  5. * either version 3 of the License, or (at your option) any later version.
  6. */
  7. using System;
  8. using System.Collections.Generic;
  9. using System.IO;
  10. using System.Text;
  11. using DiskAccessLibrary;
  12. using Utilities;
  13. namespace DiskAccessLibrary.FileSystems.NTFS
  14. {
  15. /// <summary>
  16. /// Implements the low level NTFS volume logic.
  17. /// This class can be used by higher level implementation that may include
  18. /// functions such as file copy, caching, symbolic links and etc.
  19. /// </summary>
  20. public partial class NTFSVolume : IExtendableFileSystem
  21. {
  22. private Volume m_volume;
  23. private MasterFileTable m_mft;
  24. private ClusterUsageBitmap m_bitmap;
  25. private NTFSBootRecord m_bootRecord; // partition's boot record
  26. public NTFSVolume(Volume volume) : this(volume, false)
  27. {
  28. }
  29. public NTFSVolume(Volume volume, bool useMftMirror)
  30. {
  31. m_volume = volume;
  32. byte[] bootSector = m_volume.ReadSector(0);
  33. m_bootRecord = NTFSBootRecord.ReadRecord(bootSector);
  34. if (m_bootRecord != null)
  35. {
  36. m_mft = new MasterFileTable(this, useMftMirror);
  37. m_bitmap = new ClusterUsageBitmap(this);
  38. }
  39. }
  40. public FileRecord GetFileRecord(string path)
  41. {
  42. if (path != String.Empty && !path.StartsWith(@"\"))
  43. {
  44. throw new ArgumentException("Invalid path");
  45. }
  46. if (path.EndsWith(@"\"))
  47. {
  48. path = path.Substring(0, path.Length - 1);
  49. }
  50. if (path == String.Empty)
  51. {
  52. return m_mft.GetFileRecord(MasterFileTable.RootDirSegmentNumber);
  53. }
  54. string[] components = path.Substring(1).Split('\\');
  55. long directorySegmentNumber = MasterFileTable.RootDirSegmentNumber;
  56. for (int index = 0; index < components.Length; index++)
  57. {
  58. KeyValuePairList<MftSegmentReference, FileNameRecord> records = GetFileNameRecordsInDirectory(directorySegmentNumber);
  59. if (index < components.Length - 1)
  60. {
  61. FileRecord record = FindDirectoryRecord(records, components[index]);
  62. if (record != null)
  63. {
  64. directorySegmentNumber = record.MftSegmentNumber;
  65. }
  66. else
  67. {
  68. return null;
  69. }
  70. }
  71. else // last component
  72. {
  73. return FindRecord(records, components[index]);
  74. }
  75. }
  76. return null;
  77. }
  78. private FileRecord FindDirectoryRecord(KeyValuePairList<MftSegmentReference, FileNameRecord> records, string directoryName)
  79. {
  80. FileRecord directoryRecord = FindRecord(records, directoryName);
  81. if (directoryRecord.IsDirectory)
  82. {
  83. return directoryRecord;
  84. }
  85. return null;
  86. }
  87. private FileRecord FindRecord(KeyValuePairList<MftSegmentReference, FileNameRecord> records, string name)
  88. {
  89. KeyValuePair<MftSegmentReference, FileNameRecord>? nameRecord = FindFileNameRecord(records, name);
  90. if (nameRecord != null)
  91. {
  92. FileRecord record = m_mft.GetFileRecord(nameRecord.Value.Key);
  93. if (record.IsInUse && !record.IsMetaFile)
  94. {
  95. return record;
  96. }
  97. }
  98. return null;
  99. }
  100. private KeyValuePairList<MftSegmentReference, FileNameRecord> GetFileNameRecordsInDirectory(long directoryBaseSegmentNumber)
  101. {
  102. FileRecord record = m_mft.GetFileRecord(directoryBaseSegmentNumber);
  103. KeyValuePairList<MftSegmentReference, FileNameRecord> result = null;
  104. if (record != null && record.IsDirectory)
  105. {
  106. IndexRootRecord indexRoot = (IndexRootRecord)record.GetAttributeRecord(AttributeType.IndexRoot, IndexRootRecord.FileNameIndexName);
  107. IndexAllocationRecord indexAllocation = (IndexAllocationRecord)record.GetAttributeRecord(AttributeType.IndexAllocation, IndexRootRecord.FileNameIndexName);
  108. if (indexRoot.IsLargeIndex)
  109. {
  110. if (indexAllocation != null)
  111. {
  112. result = indexAllocation.GetAllEntries(this, indexRoot);
  113. }
  114. }
  115. else
  116. {
  117. result = indexRoot.GetSmallIndexEntries();
  118. }
  119. if (result != null)
  120. {
  121. for (int index = 0; index < result.Count; index++)
  122. {
  123. if (result[index].Value.Namespace == FilenameNamespace.DOS)
  124. {
  125. // The same FileRecord can have multiple entries, each with it's own namespace
  126. result.RemoveAt(index);
  127. index--;
  128. }
  129. }
  130. }
  131. }
  132. return result;
  133. }
  134. public List<FileRecord> GetFileRecordsInDirectory(long directoryBaseSegmentNumber)
  135. {
  136. return GetFileRecordsInDirectory(directoryBaseSegmentNumber, false);
  137. }
  138. private List<FileRecord> GetFileRecordsInDirectory(long directoryBaseSegmentNumber, bool includeMetaFiles)
  139. {
  140. KeyValuePairList<MftSegmentReference, FileNameRecord> entries = GetFileNameRecordsInDirectory(directoryBaseSegmentNumber);
  141. List<FileRecord> result = new List<FileRecord>();
  142. if (entries != null)
  143. {
  144. List<MftSegmentReference> files = entries.Keys;
  145. foreach (MftSegmentReference reference in files)
  146. {
  147. FileRecord record = m_mft.GetFileRecord(reference);
  148. if (record != null)
  149. {
  150. if (record.IsInUse && (includeMetaFiles || !record.IsMetaFile))
  151. {
  152. result.Add(record);
  153. }
  154. }
  155. }
  156. }
  157. return result;
  158. }
  159. /// <summary>
  160. /// This method is slower and should only be used for recovery purposes.
  161. /// </summary>
  162. public List<FileRecord> GetFileRecordsInDirectoryFromMft(long directoryBaseSegmentNumber)
  163. {
  164. return GetFileRecordsInDirectoryFromMft(directoryBaseSegmentNumber, false);
  165. }
  166. /// <summary>
  167. /// This method is slower and should only be used for recovery purposes.
  168. /// </summary>
  169. private List<FileRecord> GetFileRecordsInDirectoryFromMft(long directoryBaseSegmentNumber, bool includeMetaFiles)
  170. {
  171. long maxNumOfRecords = m_mft.GetMaximumNumberOfSegments();
  172. List<FileRecord> result = new List<FileRecord>();
  173. for (long index = 0; index < maxNumOfRecords; index++)
  174. {
  175. FileRecord record;
  176. try
  177. {
  178. record = m_mft.GetFileRecord(index);
  179. }
  180. catch (InvalidDataException)
  181. {
  182. continue;
  183. }
  184. if (record != null)
  185. {
  186. if (record.ParentDirMftSegmentNumber == directoryBaseSegmentNumber)
  187. {
  188. if (record.IsInUse && (includeMetaFiles || !record.IsMetaFile))
  189. {
  190. result.Add(record);
  191. }
  192. }
  193. }
  194. }
  195. return result;
  196. }
  197. // logical cluster
  198. public byte[] ReadCluster(long clusterLCN)
  199. {
  200. return ReadClusters(clusterLCN, 1);
  201. }
  202. public byte[] ReadClusters(long clusterLCN, int count)
  203. {
  204. long firstSectorIndex = clusterLCN * m_bootRecord.SectorsPerCluster;
  205. int sectorsToRead = m_bootRecord.SectorsPerCluster * count;
  206. byte[] result = m_volume.ReadSectors(firstSectorIndex, sectorsToRead);
  207. return result;
  208. }
  209. public void WriteClusters(long clusterLCN, byte[] data)
  210. {
  211. long firstSectorIndex = clusterLCN * m_bootRecord.SectorsPerCluster;
  212. m_volume.WriteSectors(firstSectorIndex, data);
  213. }
  214. public byte[] ReadSectors(long sectorIndex, int sectorCount)
  215. {
  216. return m_volume.ReadSectors(sectorIndex, sectorCount);
  217. }
  218. public void WriteSectors(long sectorIndex, byte[] data)
  219. {
  220. m_volume.WriteSectors(sectorIndex, data);
  221. }
  222. public VolumeInformationRecord GetVolumeInformationRecord()
  223. {
  224. FileRecord volumeRecord = m_mft.GetVolumeRecord();
  225. if (volumeRecord != null)
  226. {
  227. VolumeInformationRecord volumeInformationRecord = (VolumeInformationRecord)volumeRecord.GetAttributeRecord(AttributeType.VolumeInformation, String.Empty);
  228. return volumeInformationRecord;
  229. }
  230. else
  231. {
  232. throw new InvalidDataException("Invalid NTFS volume record");
  233. }
  234. }
  235. public override string ToString()
  236. {
  237. StringBuilder builder = new StringBuilder();
  238. if (m_bootRecord != null)
  239. {
  240. builder.AppendLine("Bytes Per Sector: " + m_bootRecord.BytesPerSector);
  241. builder.AppendLine("Bytes Per Cluster: " + m_bootRecord.BytesPerCluster);
  242. builder.AppendLine("File Record Length: " + m_bootRecord.FileRecordSegmentLength);
  243. builder.AppendLine("First MFT Cluster (LCN): " + m_bootRecord.MftStartLCN);
  244. builder.AppendLine("First MFT Mirror Cluster (LCN): " + m_bootRecord.MftMirrorStartLCN);
  245. builder.AppendLine("Volume size (bytes): " + this.Size);
  246. builder.AppendLine();
  247. VolumeInformationRecord volumeInformationRecord = GetVolumeInformationRecord();
  248. if (volumeInformationRecord != null)
  249. {
  250. builder.AppendFormat("NTFS Version: {0}.{1}\n", volumeInformationRecord.MajorVersion, volumeInformationRecord.MinorVersion);
  251. builder.AppendLine();
  252. }
  253. FileRecord mftRecord = m_mft.GetMftRecord();
  254. if (mftRecord != null)
  255. {
  256. builder.AppendLine("Number of $MFT Data Runs: " + mftRecord.NonResidentDataRecord.DataRunSequence.Count);
  257. builder.AppendLine("Number of $MFT Clusters: " + mftRecord.NonResidentDataRecord.DataRunSequence.DataClusterCount);
  258. builder.Append(mftRecord.NonResidentDataRecord.DataRunSequence.ToString());
  259. builder.AppendLine("Number of $MFT Attributes: " + mftRecord.Attributes.Count);
  260. builder.AppendLine("Length of $MFT Attributes: " + mftRecord.StoredAttributesLength);
  261. builder.AppendLine();
  262. FileRecord bitmapRecord = m_mft.GetBitmapRecord();
  263. if (bitmapRecord != null)
  264. {
  265. builder.AppendLine("$Bitmap LCN: " + bitmapRecord.NonResidentDataRecord.DataRunSequence.FirstDataRunLCN);
  266. builder.AppendLine("$Bitmap Clusters: " + bitmapRecord.NonResidentDataRecord.DataRunSequence.DataClusterCount);
  267. builder.AppendLine("Number of $Bitmap Attributes: " + bitmapRecord.Attributes.Count);
  268. builder.AppendLine("Length of $Bitmap Attributes: " + bitmapRecord.StoredAttributesLength);
  269. }
  270. }
  271. byte[] bootRecord = ReadSectors(0, 1);
  272. long backupBootSectorIndex = (long)m_bootRecord.TotalSectors;
  273. byte[] backupBootRecord = ReadSectors(backupBootSectorIndex, 1);
  274. builder.AppendLine();
  275. builder.AppendLine("Valid backup boot sector: " + ByteUtils.AreByteArraysEqual(bootRecord, backupBootRecord));
  276. builder.AppendLine("Free space: " + this.FreeSpace);
  277. }
  278. return builder.ToString();
  279. }
  280. public bool IsValid
  281. {
  282. get
  283. {
  284. return (m_bootRecord != null && m_mft.GetMftRecord() != null);
  285. }
  286. }
  287. public bool IsValidAndSupported
  288. {
  289. get
  290. {
  291. return (this.IsValid && MajorVersion == 3 && MinorVersion == 1);
  292. }
  293. }
  294. public long Size
  295. {
  296. get
  297. {
  298. return (long)(m_bootRecord.TotalSectors * m_bootRecord.BytesPerSector);
  299. }
  300. }
  301. public long FreeSpace
  302. {
  303. get
  304. {
  305. return m_bitmap.CountNumberOfFreeClusters() * this.BytesPerCluster;
  306. }
  307. }
  308. public NTFSBootRecord BootRecord
  309. {
  310. get
  311. {
  312. return m_bootRecord;
  313. }
  314. }
  315. public MasterFileTable MasterFileTable
  316. {
  317. get
  318. {
  319. return m_mft;
  320. }
  321. }
  322. public int BytesPerCluster
  323. {
  324. get
  325. {
  326. return m_bootRecord.BytesPerCluster;
  327. }
  328. }
  329. public int BytesPerSector
  330. {
  331. get
  332. {
  333. return m_bootRecord.BytesPerSector;
  334. }
  335. }
  336. public int SectorsPerCluster
  337. {
  338. get
  339. {
  340. return m_bootRecord.SectorsPerCluster;
  341. }
  342. }
  343. public int SectorsPerFileRecordSegment
  344. {
  345. get
  346. {
  347. return m_bootRecord.SectorsPerFileRecordSegment;
  348. }
  349. }
  350. public ushort MajorVersion
  351. {
  352. get
  353. {
  354. VolumeInformationRecord volumeInformationRecord = GetVolumeInformationRecord();
  355. return volumeInformationRecord.MajorVersion;
  356. }
  357. }
  358. public ushort MinorVersion
  359. {
  360. get
  361. {
  362. VolumeInformationRecord volumeInformationRecord = GetVolumeInformationRecord();
  363. return volumeInformationRecord.MinorVersion;
  364. }
  365. }
  366. public KeyValuePairList<ulong, long> AllocateClusters(ulong desiredStartLCN, long numberOfClusters)
  367. {
  368. return m_bitmap.AllocateClusters(desiredStartLCN, numberOfClusters);
  369. }
  370. private static KeyValuePair<MftSegmentReference, FileNameRecord>? FindFileNameRecord(KeyValuePairList<MftSegmentReference, FileNameRecord> records, string name)
  371. {
  372. foreach (KeyValuePair<MftSegmentReference, FileNameRecord> record in records)
  373. {
  374. if (String.Equals(record.Value.FileName, name, StringComparison.InvariantCultureIgnoreCase))
  375. {
  376. return record;
  377. }
  378. }
  379. return null;
  380. }
  381. }
  382. }