PrivateHeader.cs 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330
  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.Text;
  10. using DiskAccessLibrary;
  11. using Utilities;
  12. namespace DiskAccessLibrary.LogicalDiskManager
  13. {
  14. [Flags]
  15. // as reported by DMDiag
  16. public enum PrivateHeaderFlags : uint
  17. {
  18. Shared = 0x01, // private when the flag is absent
  19. NoAutoImport = 0x02, // autoimport when the flag is absent
  20. }
  21. public class PrivateHeader
  22. {
  23. public const int Length = 512;
  24. public const int PrivateHeaderSectorIndex = 6;
  25. public const string PrivateHeaderSignature = "PRIVHEAD";
  26. public string Signature = PrivateHeaderSignature;
  27. //private uint Checksum;
  28. // Versions: 2.11 for Windows XP \ Server 2003
  29. // 2.12 for Windows 7 \ Server 2008
  30. // 2.12 for Veritas Storage Foundation 4.0 (regardless of whether Windows Disk Management compatible group is checked or not)
  31. public ushort MajorVersion;
  32. public ushort MinorVersion;
  33. public DateTime LastUpdateDT; // Set when UpdateSequenceNumber is being updated
  34. public ulong UpdateSequenceNumber; // as reported by DMDiag, kept in sync with UpdateSequenceNumber of the most recent TOCBlock
  35. public ulong PrimaryPrivateHeaderLBA; // Within the private region
  36. public ulong SecondaryPrivateHeaderLBA; // Within the private region
  37. public string DiskGuidString;
  38. public string HostGuidString;
  39. public string DiskGroupGuidString;
  40. public string DiskGroupName; // Veritas will limit DiskGroupName to 18 characters, Windows will use the NetBIOS name (limited to 15 characters) and will append 'Dg0'
  41. public uint BytesPerSector; // a.k.a. iosize
  42. public PrivateHeaderFlags Flags;
  43. public ushort PublicRegionSliceNumber;
  44. public ushort PrivateRegionSliceNumber;
  45. /// <summary>
  46. /// MBR: Windows XP / Server 2003 will ignore this value and will use the first sector of the second disk track (usually sector number 63, as there are usually 63 sectors per track [0-62])
  47. /// </summary>
  48. public ulong PublicRegionStartLBA;
  49. public ulong PublicRegionSizeLBA;
  50. public ulong PrivateRegionStartLBA;
  51. public ulong PrivateRegionSizeLBA;
  52. /// <summary>
  53. /// PrimaryTocLBA / SecondaryTocLBA: on write operation, the updated TOC will be written to PreviousPrimaryTocLBA / PreviousSecondaryTocLBA,
  54. /// and then PrimaryTocLBA / SecondaryTocLBA will be updated to point to it.
  55. /// </summary>
  56. public ulong PrimaryTocLBA; // Note: We have the previous TOC (update-sequence-number wise) adjacent, see PreviousPrimaryTocLBA
  57. public ulong SecondaryTocLBA; // Note: We have the previous TOC adjacent, see PreviousSecondaryTocLBA
  58. public uint NumberOfConfigs; // in private region?
  59. public uint NumberOfLogs; // in private region?
  60. public ulong ConfigSizeLBA; // all config regions in private region in total?
  61. public ulong LogSizeLBA; // all log regions in private region in total?
  62. public uint DiskSignature;
  63. public Guid DiskSetGuid;
  64. public Guid DiskSetGuidRepeat;
  65. private bool m_isChecksumValid;
  66. public PrivateHeader(byte[] buffer)
  67. {
  68. Signature = ByteReader.ReadAnsiString(buffer, 0x00, 8);
  69. uint checksum = BigEndianConverter.ToUInt32(buffer, 0x08);
  70. MajorVersion = BigEndianConverter.ToUInt16(buffer, 0x0C);
  71. MinorVersion = BigEndianConverter.ToUInt16(buffer, 0x0E);
  72. LastUpdateDT = DateTime.FromFileTimeUtc(BigEndianConverter.ToInt64(buffer, 0x10));
  73. UpdateSequenceNumber = BigEndianConverter.ToUInt64(buffer, 0x18);
  74. PrimaryPrivateHeaderLBA = BigEndianConverter.ToUInt64(buffer, 0x20);
  75. SecondaryPrivateHeaderLBA = BigEndianConverter.ToUInt64(buffer, 0x28);
  76. DiskGuidString = ByteReader.ReadAnsiString(buffer, 0x30, 0x40).Trim('\0');
  77. HostGuidString = ByteReader.ReadAnsiString(buffer, 0x70, 0x40).Trim('\0');
  78. DiskGroupGuidString = ByteReader.ReadAnsiString(buffer, 0xB0, 0x40).Trim('\0');
  79. DiskGroupName = ByteReader.ReadAnsiString(buffer, 0xF0, 31).Trim('\0');
  80. BytesPerSector = BigEndianConverter.ToUInt32(buffer, 0x10F);
  81. Flags = (PrivateHeaderFlags)BigEndianConverter.ToUInt32(buffer, 0x113);
  82. PublicRegionSliceNumber = BigEndianConverter.ToUInt16(buffer, 0x117);
  83. PrivateRegionSliceNumber = BigEndianConverter.ToUInt16(buffer, 0x119);
  84. PublicRegionStartLBA = BigEndianConverter.ToUInt64(buffer, 0x11B);
  85. PublicRegionSizeLBA = BigEndianConverter.ToUInt64(buffer, 0x123);
  86. PrivateRegionStartLBA = BigEndianConverter.ToUInt64(buffer, 0x12B);
  87. PrivateRegionSizeLBA = BigEndianConverter.ToUInt64(buffer, 0x133);
  88. PrimaryTocLBA = BigEndianConverter.ToUInt64(buffer, 0x13B);
  89. SecondaryTocLBA = BigEndianConverter.ToUInt64(buffer, 0x143);
  90. NumberOfConfigs = BigEndianConverter.ToUInt32(buffer, 0x14B);
  91. NumberOfLogs = BigEndianConverter.ToUInt32(buffer, 0x14F);
  92. ConfigSizeLBA = BigEndianConverter.ToUInt64(buffer, 0x153);
  93. LogSizeLBA = BigEndianConverter.ToUInt64(buffer, 0x15B);
  94. DiskSignature = BigEndianConverter.ToUInt32(buffer, 0x163);
  95. DiskSetGuid = BigEndianConverter.ToGuid(buffer, 0x167);
  96. DiskSetGuidRepeat = BigEndianConverter.ToGuid(buffer, 0x177);
  97. BigEndianWriter.WriteUInt32(buffer, 0x08, (uint)0); // we exclude the checksum field from checksum calculations
  98. m_isChecksumValid = (checksum == CalculateChecksum(buffer));
  99. }
  100. public byte[] GetBytes()
  101. {
  102. byte[] buffer = new byte[Length];
  103. ByteWriter.WriteAnsiString(buffer, 0x00, Signature, 8);
  104. // we'll write the checksum later
  105. BigEndianWriter.WriteUInt16(buffer, 0x0C, MajorVersion);
  106. BigEndianWriter.WriteUInt16(buffer, 0x0E, MinorVersion);
  107. BigEndianWriter.WriteInt64(buffer, 0x10, LastUpdateDT.ToFileTimeUtc());
  108. BigEndianWriter.WriteUInt64(buffer, 0x18, UpdateSequenceNumber);
  109. BigEndianWriter.WriteUInt64(buffer, 0x20, PrimaryPrivateHeaderLBA);
  110. BigEndianWriter.WriteUInt64(buffer, 0x28, SecondaryPrivateHeaderLBA);
  111. ByteWriter.WriteAnsiString(buffer, 0x30, DiskGuidString, 0x40);
  112. ByteWriter.WriteAnsiString(buffer, 0x70, HostGuidString, 0x40);
  113. ByteWriter.WriteAnsiString(buffer, 0xB0, DiskGroupGuidString, 0x40);
  114. ByteWriter.WriteAnsiString(buffer, 0xF0, DiskGroupName, 31);
  115. BigEndianWriter.WriteUInt32(buffer, 0x10F, BytesPerSector);
  116. BigEndianWriter.WriteUInt32(buffer, 0x113, (uint)Flags);
  117. BigEndianWriter.WriteUInt16(buffer, 0x117, PublicRegionSliceNumber);
  118. BigEndianWriter.WriteUInt16(buffer, 0x119, PrivateRegionSliceNumber);
  119. BigEndianWriter.WriteUInt64(buffer, 0x11B, PublicRegionStartLBA);
  120. BigEndianWriter.WriteUInt64(buffer, 0x123, PublicRegionSizeLBA);
  121. BigEndianWriter.WriteUInt64(buffer, 0x12B, PrivateRegionStartLBA);
  122. BigEndianWriter.WriteUInt64(buffer, 0x133, PrivateRegionSizeLBA);
  123. BigEndianWriter.WriteUInt64(buffer, 0x13B, PrimaryTocLBA);
  124. BigEndianWriter.WriteUInt64(buffer, 0x143, SecondaryTocLBA);
  125. BigEndianWriter.WriteUInt32(buffer, 0x14B, NumberOfConfigs);
  126. BigEndianWriter.WriteUInt32(buffer, 0x14F, NumberOfLogs);
  127. BigEndianWriter.WriteUInt64(buffer, 0x153, ConfigSizeLBA);
  128. BigEndianWriter.WriteUInt64(buffer, 0x15B, LogSizeLBA);
  129. BigEndianWriter.WriteUInt32(buffer, 0x163, DiskSignature);
  130. BigEndianWriter.WriteGuidBytes(buffer, 0x167, DiskSetGuid);
  131. BigEndianWriter.WriteGuidBytes(buffer, 0x177, DiskSetGuidRepeat);
  132. uint checksum = CalculateChecksum(buffer);
  133. BigEndianWriter.WriteUInt32(buffer, 0x08, checksum);
  134. return buffer;
  135. }
  136. public static PrivateHeader ReadFromDisk(Disk disk)
  137. {
  138. MasterBootRecord mbr = MasterBootRecord.ReadFromDisk(disk);
  139. if (mbr.IsGPTBasedDisk)
  140. {
  141. return ReadFromGPTBasedDisk(disk);
  142. }
  143. else
  144. {
  145. return ReadFromMBRBasedDisk(disk);
  146. }
  147. }
  148. public static PrivateHeader ReadFromMBRBasedDisk(Disk disk)
  149. {
  150. // check for private header at the last sector of the disk
  151. PrivateHeader privateHeader = ReadFromDiskEnd(disk, true);
  152. if (privateHeader != null)
  153. {
  154. if (privateHeader.IsChecksumValid)
  155. {
  156. return privateHeader;
  157. }
  158. else
  159. {
  160. // primary has invalid checksum, try secondary private header
  161. long sectorIndex = (long)(privateHeader.PrivateRegionStartLBA + privateHeader.SecondaryPrivateHeaderLBA);
  162. return ReadFromDisk(disk, sectorIndex, false);
  163. }
  164. }
  165. else
  166. {
  167. // maybe the disk was cloned to a bigger disk, check sector 6
  168. return ReadFromDiskStart(disk);
  169. }
  170. }
  171. public static PrivateHeader ReadFromGPTBasedDisk(Disk disk)
  172. {
  173. List<GuidPartitionEntry> entries = GuidPartitionTable.ReadEntriesFromDisk(disk);
  174. int index = GuidPartitionEntryCollection.GetIndexOfPartitionTypeGuid(entries, GPTPartition.PrivateRegionPartitionTypeGuid);
  175. // the private header will be located at the last sector of the private region
  176. PrivateHeader privateHeader = PrivateHeader.ReadFromDisk(disk, (long)entries[index].LastLBA, true);
  177. if (privateHeader != null)
  178. {
  179. if (privateHeader.IsChecksumValid)
  180. {
  181. return privateHeader;
  182. }
  183. else
  184. {
  185. // primary has invalid checksum, try secondary private header
  186. long sectorIndex = (long)(privateHeader.PrivateRegionStartLBA + privateHeader.SecondaryPrivateHeaderLBA);
  187. return ReadFromDisk(disk, sectorIndex, false);
  188. }
  189. }
  190. return null;
  191. }
  192. public static PrivateHeader ReadFromDiskEnd(Disk disk, bool returnPrivateHeaderWithInvalidChecksum)
  193. {
  194. long sectorCount = disk.Size / disk.BytesPerSector;
  195. long sectorIndex = sectorCount - 1;
  196. return ReadFromDisk(disk, sectorIndex, returnPrivateHeaderWithInvalidChecksum);
  197. }
  198. public static PrivateHeader ReadFromDiskStart(Disk disk)
  199. {
  200. return ReadFromDisk(disk, PrivateHeaderSectorIndex, false);
  201. }
  202. public static PrivateHeader ReadFromDisk(Disk disk, long sectorIndex, bool returnPrivateHeaderWithInvalidChecksum)
  203. {
  204. byte[] sector = disk.ReadSector(sectorIndex);
  205. string signature = ByteReader.ReadAnsiString(sector, 0x00, 8);
  206. if (signature == PrivateHeaderSignature)
  207. {
  208. PrivateHeader privateHeader = new PrivateHeader(sector);
  209. if (privateHeader.IsChecksumValid || returnPrivateHeaderWithInvalidChecksum)
  210. {
  211. return privateHeader;
  212. }
  213. }
  214. return null;
  215. }
  216. public static void WriteToDisk(Disk disk, PrivateHeader privateHeader)
  217. {
  218. byte[] bytes = privateHeader.GetBytes();
  219. disk.WriteSectors((long)(privateHeader.PrivateRegionStartLBA + privateHeader.PrimaryPrivateHeaderLBA), bytes);
  220. disk.WriteSectors((long)(privateHeader.PrivateRegionStartLBA + privateHeader.SecondaryPrivateHeaderLBA), bytes);
  221. // update sector 6 if a Private Header is already present there
  222. byte[] sector = disk.ReadSector(PrivateHeaderSectorIndex);
  223. string signature = ByteReader.ReadAnsiString(sector, 0x00, 8);
  224. if (signature == PrivateHeaderSignature)
  225. {
  226. disk.WriteSectors(PrivateHeaderSectorIndex, bytes);
  227. }
  228. }
  229. public Guid DiskGuid
  230. {
  231. get
  232. {
  233. return new Guid(this.DiskGuidString);
  234. }
  235. set
  236. {
  237. this.DiskGuidString = value.ToString();
  238. }
  239. }
  240. public Guid DiskGroupGuid
  241. {
  242. get
  243. {
  244. return new Guid(this.DiskGroupGuidString);
  245. }
  246. set
  247. {
  248. this.DiskGroupGuidString = value.ToString();
  249. }
  250. }
  251. // I'm not aware of any parameter that will tell us where the previous TOCs are,
  252. // I'm assuming it is given that there will be TOCs (in the private region) at Sectors 1,2, PrivateRegionSizeLBA - 2 and PrivateRegionSizeLBA - 1,
  253. // And so PrimaryTocLBA, SecondaryTocLBA simply point to the ones being used
  254. public ulong PreviousPrimaryTocLBA
  255. {
  256. get
  257. {
  258. if (PrimaryTocLBA == 1)
  259. {
  260. return PrimaryTocLBA + 1;
  261. }
  262. else
  263. {
  264. return PrimaryTocLBA - 1;
  265. }
  266. }
  267. }
  268. public ulong PreviousSecondaryTocLBA
  269. {
  270. get
  271. {
  272. if (PrimaryTocLBA == 1)
  273. {
  274. return SecondaryTocLBA - 1;
  275. }
  276. else
  277. {
  278. return SecondaryTocLBA + 1;
  279. }
  280. }
  281. }
  282. public bool IsChecksumValid
  283. {
  284. get
  285. {
  286. return m_isChecksumValid;
  287. }
  288. }
  289. public static uint CalculateChecksum(byte[] bytes)
  290. {
  291. uint result = 0;
  292. for (int index = 0; index < bytes.Length; index++)
  293. {
  294. result += bytes[index];
  295. }
  296. return result;
  297. }
  298. }
  299. }