FileRecord.cs 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398
  1. /* Copyright (C) 2014 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 Utilities;
  12. namespace DiskAccessLibrary.FileSystems.NTFS
  13. {
  14. public class FileRecord // A collection of base record segment and zero or more file record segments making up this file record
  15. {
  16. List<FileRecordSegment> m_segments;
  17. private List<AttributeRecord> m_attributes;
  18. private DataRecord m_dataRecord;
  19. public FileRecord(FileRecordSegment segment)
  20. {
  21. m_segments = new List<FileRecordSegment>();
  22. m_segments.Add(segment);
  23. }
  24. public FileRecord(List<FileRecordSegment> segments)
  25. {
  26. m_segments = segments;
  27. }
  28. public void UpdateSegments(int maximumSegmentLength, int bytesPerSector, ushort minorNTFSVersion)
  29. {
  30. foreach (FileRecordSegment segment in m_segments)
  31. {
  32. segment.ImmediateAttributes.Clear();
  33. }
  34. int segmentLength = FileRecordSegment.GetFirstAttributeOffset(maximumSegmentLength, minorNTFSVersion);
  35. segmentLength += FileRecordSegment.EndMarkerLength;
  36. foreach (AttributeRecord attribute in this.Attributes)
  37. {
  38. segmentLength += (int)attribute.RecordLength;
  39. }
  40. if (segmentLength <= maximumSegmentLength)
  41. {
  42. // a single record segment is needed
  43. FileRecordSegment baseRecordSegment = m_segments[0];
  44. foreach (AttributeRecord attribute in this.Attributes)
  45. {
  46. baseRecordSegment.ImmediateAttributes.Add(attribute);
  47. }
  48. // free the rest of the segments, if there are any
  49. for (int index = 1; index < m_segments.Count; index++)
  50. {
  51. m_segments[index].IsInUse = false;
  52. }
  53. }
  54. else
  55. {
  56. // we have to check if we can make some data streams non-resident,
  57. // otherwise we have to use child segments and create an attribute list
  58. throw new NotImplementedException();
  59. }
  60. }
  61. public List<AttributeRecord> GetAssembledAttributes()
  62. {
  63. return GetAssembledAttributes(m_segments);
  64. }
  65. public List<FileRecordSegment> Segments
  66. {
  67. get
  68. {
  69. return m_segments;
  70. }
  71. }
  72. public List<AttributeRecord> Attributes
  73. {
  74. get
  75. {
  76. if (m_attributes == null)
  77. {
  78. m_attributes = GetAssembledAttributes();
  79. }
  80. return m_attributes;
  81. }
  82. }
  83. public StandardInformationRecord StandardInformation
  84. {
  85. get
  86. {
  87. foreach (AttributeRecord attribute in this.Attributes)
  88. {
  89. if (attribute is StandardInformationRecord)
  90. {
  91. return (StandardInformationRecord)attribute;
  92. }
  93. }
  94. return null;
  95. }
  96. }
  97. public FileNameRecord GetFileNameRecord(FilenameNamespace filenameNamespace)
  98. {
  99. foreach (AttributeRecord attribute in this.Attributes)
  100. {
  101. if (attribute is FileNameAttributeRecord)
  102. {
  103. FileNameRecord fileNameAttribute = ((FileNameAttributeRecord)attribute).Record;
  104. if (fileNameAttribute.Namespace == filenameNamespace)
  105. {
  106. return fileNameAttribute;
  107. }
  108. }
  109. }
  110. return null;
  111. }
  112. public FileNameRecord LongFileNameRecord
  113. {
  114. get
  115. {
  116. FileNameRecord record = GetFileNameRecord(FilenameNamespace.Win32);
  117. if (record == null)
  118. {
  119. record = GetFileNameRecord(FilenameNamespace.POSIX);
  120. }
  121. return record;
  122. }
  123. }
  124. // 8.3 filename
  125. public FileNameRecord ShortFileNameRecord
  126. {
  127. get
  128. {
  129. FileNameRecord record = GetFileNameRecord(FilenameNamespace.DOS);
  130. if (record == null)
  131. {
  132. // Win32AndDOS means that both the Win32 and the DOS filenames are identical and hence have been saved in this single filename record.
  133. record = GetFileNameRecord(FilenameNamespace.Win32AndDOS);
  134. }
  135. return record;
  136. }
  137. }
  138. public FileNameRecord FileNameRecord
  139. {
  140. get
  141. {
  142. FileNameRecord fileNameRecord = this.LongFileNameRecord;
  143. if (fileNameRecord == null)
  144. {
  145. fileNameRecord = this.ShortFileNameRecord;
  146. }
  147. return fileNameRecord;
  148. }
  149. }
  150. /// <summary>
  151. /// Will return the long filename of the file
  152. /// </summary>
  153. public string FileName
  154. {
  155. get
  156. {
  157. FileNameRecord fileNameRecord = this.FileNameRecord;
  158. if (fileNameRecord != null)
  159. {
  160. return fileNameRecord.FileName;
  161. }
  162. else
  163. {
  164. return String.Empty;
  165. }
  166. }
  167. }
  168. public long ParentDirMftSegmentNumber
  169. {
  170. get
  171. {
  172. FileNameRecord fileNameRecord = this.LongFileNameRecord;
  173. if (fileNameRecord == null)
  174. {
  175. fileNameRecord = this.ShortFileNameRecord;
  176. }
  177. if (fileNameRecord != null)
  178. {
  179. return fileNameRecord.ParentDirectory.SegmentNumber;
  180. }
  181. else
  182. {
  183. return 0;
  184. }
  185. }
  186. }
  187. public AttributeRecord GetAttributeRecord(AttributeType type, string name)
  188. {
  189. foreach (AttributeRecord attribute in this.Attributes)
  190. {
  191. if (attribute.AttributeType == type && attribute.Name == name)
  192. {
  193. return attribute;
  194. }
  195. }
  196. return null;
  197. }
  198. public DataRecord DataRecord
  199. {
  200. get
  201. {
  202. if (m_dataRecord == null)
  203. {
  204. AttributeRecord record = GetAttributeRecord(AttributeType.Data, String.Empty);
  205. if (record != null)
  206. {
  207. m_dataRecord = new DataRecord(record);
  208. }
  209. }
  210. return m_dataRecord;
  211. }
  212. }
  213. public NonResidentAttributeRecord NonResidentDataRecord
  214. {
  215. get
  216. {
  217. if (this.DataRecord.Record is NonResidentAttributeRecord)
  218. {
  219. return (NonResidentAttributeRecord)m_dataRecord.Record;
  220. }
  221. else
  222. {
  223. return null;
  224. }
  225. }
  226. }
  227. /// <summary>
  228. /// Segment number of base record
  229. /// </summary>
  230. public long MftSegmentNumber
  231. {
  232. get
  233. {
  234. return m_segments[0].MftSegmentNumber;
  235. }
  236. }
  237. /// <summary>
  238. /// Sequence number of base record
  239. /// </summary>
  240. public long SequenceNumber
  241. {
  242. get
  243. {
  244. return m_segments[0].SequenceNumber;
  245. }
  246. }
  247. public bool IsInUse
  248. {
  249. get
  250. {
  251. return m_segments[0].IsInUse;
  252. }
  253. }
  254. public bool IsDirectory
  255. {
  256. get
  257. {
  258. return m_segments[0].IsDirectory;
  259. }
  260. }
  261. public int StoredAttributesLength
  262. {
  263. get
  264. {
  265. int length = 0;
  266. foreach (AttributeRecord attribute in this.Attributes)
  267. {
  268. length += (int)attribute.StoredRecordLength;
  269. }
  270. return length;
  271. }
  272. }
  273. public bool IsMetaFile
  274. {
  275. get
  276. {
  277. return (this.MftSegmentNumber <= MasterFileTable.LastReservedMftSegmentNumber);
  278. }
  279. }
  280. public static List<AttributeRecord> GetAssembledAttributes(List<FileRecordSegment> segments)
  281. {
  282. List<AttributeRecord> result = new List<AttributeRecord>();
  283. // we need to assemble fragmented attributes (if there are any)
  284. // if two attributes have the same AttributeType and Name, then we need to assemble them back together.
  285. // Note: only non-resident attributes can be fragmented
  286. // Reference: http://technet.microsoft.com/en-us/library/cc976808.aspx
  287. Dictionary<KeyValuePair<AttributeType, string>, List<NonResidentAttributeRecord>> fragments = new Dictionary<KeyValuePair<AttributeType, string>, List<NonResidentAttributeRecord>>();
  288. foreach (FileRecordSegment segment in segments)
  289. {
  290. foreach (AttributeRecord attribute in segment.ImmediateAttributes)
  291. {
  292. if (attribute is ResidentAttributeRecord)
  293. {
  294. result.Add(attribute);
  295. }
  296. else
  297. {
  298. KeyValuePair<AttributeType, string> key = new KeyValuePair<AttributeType, string>(attribute.AttributeType, attribute.Name);
  299. if (fragments.ContainsKey(key))
  300. {
  301. fragments[key].Add((NonResidentAttributeRecord)attribute);
  302. }
  303. else
  304. {
  305. List<NonResidentAttributeRecord> attributeFragments = new List<NonResidentAttributeRecord>();
  306. attributeFragments.Add((NonResidentAttributeRecord)attribute);
  307. fragments.Add(key, attributeFragments);
  308. }
  309. }
  310. }
  311. }
  312. // assemble all non-resident attributes
  313. foreach (List<NonResidentAttributeRecord> attributeFragments in fragments.Values)
  314. {
  315. // we assume attribute fragments are written to disk sorted by LowestVCN
  316. NonResidentAttributeRecord baseAttribute = attributeFragments[0];
  317. if (baseAttribute.LowestVCN != 0)
  318. {
  319. string message = String.Format("Attribute fragments must be sorted, MftSegmentNumber: {0}, attribute type: {1}",
  320. segments[0].MftSegmentNumber, baseAttribute.AttributeType);
  321. throw new InvalidDataException(message);
  322. }
  323. if (baseAttribute.DataRunSequence.DataClusterCount != baseAttribute.HighestVCN + 1)
  324. {
  325. string message = String.Format("Cannot properly assemble data run sequence 0, expected length: {0}, sequence length: {1}",
  326. baseAttribute.HighestVCN + 1, baseAttribute.DataRunSequence.DataClusterCount);
  327. throw new InvalidDataException(message);
  328. }
  329. for (int index = 1; index < attributeFragments.Count; index++)
  330. {
  331. NonResidentAttributeRecord attributeFragment = attributeFragments[index];
  332. if (attributeFragment.LowestVCN == baseAttribute.HighestVCN + 1)
  333. {
  334. // The DataRunSequence of each additional file record segment starts at absolute LCN,
  335. // so we need to convert it to relative offset before adding it to the base DataRunSequence
  336. long absoluteOffset = attributeFragment.DataRunSequence[0].RunOffset;
  337. long previousLCN = baseAttribute.DataRunSequence.LastDataRunStartLCN;
  338. long relativeOffset = absoluteOffset - previousLCN;
  339. attributeFragment.DataRunSequence[0].RunOffset = relativeOffset;
  340. baseAttribute.DataRunSequence.AddRange(attributeFragment.DataRunSequence);
  341. baseAttribute.HighestVCN = attributeFragment.HighestVCN;
  342. if (baseAttribute.DataRunSequence.DataClusterCount != baseAttribute.HighestVCN + 1)
  343. {
  344. string message = String.Format("Cannot properly assemble data run sequence, expected length: {0}, sequence length: {1}",
  345. baseAttribute.HighestVCN + 1, baseAttribute.DataRunSequence.DataClusterCount);
  346. throw new InvalidDataException(message);
  347. }
  348. }
  349. else
  350. {
  351. throw new InvalidDataException("Invalid attribute fragments order");
  352. }
  353. }
  354. result.Add(baseAttribute);
  355. }
  356. return result;
  357. }
  358. }
  359. }