VirtualHardDisk.cs 9.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268
  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.IO;
  9. using System.Collections.Generic;
  10. using Utilities;
  11. using DiskAccessLibrary.VHD;
  12. namespace DiskAccessLibrary
  13. {
  14. public partial class VirtualHardDisk : DiskImage, IDiskGeometry
  15. {
  16. // VHD sector size is set to 512 bytes.
  17. public const int BytesPerDiskSector = 512;
  18. private RawDiskImage m_file;
  19. private VHDFooter m_vhdFooter;
  20. // Dynamic VHD:
  21. private DynamicDiskHeader m_dynamicHeader;
  22. private BlockAllocationTable m_blockAllocationTable;
  23. // CHS:
  24. private long m_cylinders;
  25. private int m_tracksPerCylinder; // a.k.a. heads
  26. private int m_sectorsPerTrack;
  27. /// <exception cref="System.IO.IOException"></exception>
  28. /// <exception cref="System.IO.InvalidDataException"></exception>
  29. /// <exception cref="System.NotImplementedException"></exception>
  30. /// <exception cref="System.UnauthorizedAccessException"></exception>
  31. public VirtualHardDisk(string virtualHardDiskPath) : base(virtualHardDiskPath)
  32. {
  33. // We can't read the VHD footer using this.ReadSector() because it's out of the disk boundaries
  34. m_file = new RawDiskImage(virtualHardDiskPath, BytesPerDiskSector);
  35. byte[] buffer = m_file.ReadSector(m_file.Size / BytesPerDiskSector - 1);
  36. m_vhdFooter = new VHDFooter(buffer);
  37. if (!m_vhdFooter.IsValid)
  38. {
  39. // check to see if a header is present (dynamic VHD) and use it instead
  40. buffer = m_file.ReadSector(0);
  41. m_vhdFooter = new VHDFooter(buffer);
  42. if (!m_vhdFooter.IsValid)
  43. {
  44. throw new InvalidDataException("Invalid VHD footer");
  45. }
  46. }
  47. if (m_vhdFooter.DiskType == VirtualHardDiskType.Fixed)
  48. {
  49. }
  50. else if (m_vhdFooter.DiskType == VirtualHardDiskType.Dynamic)
  51. {
  52. buffer = m_file.ReadSectors(1, 2);
  53. m_dynamicHeader = new DynamicDiskHeader(buffer);
  54. m_blockAllocationTable = BlockAllocationTable.ReadBlockAllocationTable(virtualHardDiskPath, m_dynamicHeader);
  55. }
  56. else
  57. {
  58. throw new NotImplementedException("Differencing VHD is not supported");
  59. }
  60. SetGeometry();
  61. }
  62. private void SetGeometry()
  63. {
  64. byte heads;
  65. byte sectorsPerTrack;
  66. ushort cylinders;
  67. GetDiskGeometry((ulong)this.TotalSectors, out heads, out sectorsPerTrack, out cylinders);
  68. m_cylinders = cylinders;
  69. m_tracksPerCylinder = heads;
  70. m_sectorsPerTrack = sectorsPerTrack;
  71. }
  72. public override bool ExclusiveLock()
  73. {
  74. return m_file.ExclusiveLock();
  75. }
  76. public override bool ReleaseLock()
  77. {
  78. return m_file.ReleaseLock();
  79. }
  80. /// <summary>
  81. /// Sector refers to physical disk sector, we can only read complete sectors
  82. /// </summary>
  83. public override byte[] ReadSectors(long sectorIndex, int sectorCount)
  84. {
  85. CheckBoundaries(sectorIndex, sectorCount);
  86. if (m_vhdFooter.DiskType == VirtualHardDiskType.Fixed)
  87. {
  88. return m_file.ReadSectors(sectorIndex, sectorCount);
  89. }
  90. else // Dynamic VHD
  91. {
  92. return ReadSectorsFromDynamicDisk(sectorIndex, sectorCount);
  93. }
  94. }
  95. public override void WriteSectors(long sectorIndex, byte[] data)
  96. {
  97. CheckBoundaries(sectorIndex, data.Length / BytesPerDiskSector);
  98. if (m_vhdFooter.DiskType == VirtualHardDiskType.Fixed)
  99. {
  100. m_file.WriteSectors(sectorIndex, data);
  101. }
  102. else // Dynamic VHD
  103. {
  104. WriteSectorsToDynamicDisk(sectorIndex, data);
  105. }
  106. }
  107. public override void Extend(long numberOfAdditionalBytes)
  108. {
  109. if (numberOfAdditionalBytes % BytesPerDiskSector > 0)
  110. {
  111. throw new ArgumentException("numberOfAdditionalBytes must be a multiple of sector size");
  112. }
  113. if (m_vhdFooter.DiskType == VirtualHardDiskType.Fixed)
  114. {
  115. m_vhdFooter.CurrentSize += (ulong)numberOfAdditionalBytes;
  116. ExtendFile(numberOfAdditionalBytes);
  117. }
  118. else
  119. {
  120. throw new NotImplementedException();
  121. }
  122. }
  123. private void ExtendFile(long numberOfAdditionalBytes)
  124. {
  125. long footerOffset = m_file.Size - VHDFooter.Length;
  126. m_file.Extend(numberOfAdditionalBytes);
  127. byte[] footerBytes = m_vhdFooter.GetBytes();
  128. if (m_vhdFooter.DiskType != VirtualHardDiskType.Fixed)
  129. {
  130. m_file.WriteSectors(0, footerBytes);
  131. }
  132. m_file.WriteSectors((footerOffset + numberOfAdditionalBytes) / BytesPerDiskSector, footerBytes);
  133. }
  134. public long Cylinders
  135. {
  136. get
  137. {
  138. return m_cylinders;
  139. }
  140. }
  141. public int TracksPerCylinder
  142. {
  143. get
  144. {
  145. return m_tracksPerCylinder;
  146. }
  147. }
  148. public int SectorsPerTrack
  149. {
  150. get
  151. {
  152. return m_sectorsPerTrack;
  153. }
  154. }
  155. public override int BytesPerSector
  156. {
  157. get
  158. {
  159. return BytesPerDiskSector;
  160. }
  161. }
  162. public override long Size
  163. {
  164. get
  165. {
  166. return (long)m_vhdFooter.CurrentSize;
  167. }
  168. }
  169. public VHDFooter Footer
  170. {
  171. get
  172. {
  173. return m_vhdFooter;
  174. }
  175. }
  176. // Taken From VHD format specs (Appendix)
  177. public static void GetDiskGeometry(ulong totalSectors, out byte heads, out byte sectorsPerTrack, out ushort cylinders)
  178. {
  179. int cylindersTimesHeads;
  180. // If more than ~128GB truncate at ~128GB
  181. if (totalSectors > 65535 * 16 * 255)
  182. {
  183. totalSectors = 65535 * 16 * 255;
  184. }
  185. // If more than ~32GB, break partition table compatibility.
  186. // Partition table has max 63 sectors per track. Otherwise
  187. // we're looking for a geometry that's valid for both BIOS
  188. // and ATA.
  189. if (totalSectors >= 65535 * 16 * 63)
  190. {
  191. sectorsPerTrack = 255;
  192. heads = 16;
  193. cylindersTimesHeads = (int)(totalSectors / sectorsPerTrack);
  194. }
  195. else
  196. {
  197. sectorsPerTrack = 17;
  198. cylindersTimesHeads = (int)(totalSectors / sectorsPerTrack);
  199. heads = (byte)((cylindersTimesHeads + 1023) / 1024);
  200. if (heads < 4)
  201. {
  202. heads = 4;
  203. }
  204. if (cylindersTimesHeads >= (heads * 1024) || heads > 16)
  205. {
  206. sectorsPerTrack = 31;
  207. heads = 16;
  208. cylindersTimesHeads = (int)(totalSectors / sectorsPerTrack);
  209. }
  210. if (cylindersTimesHeads >= (heads * 1024))
  211. {
  212. sectorsPerTrack = 63;
  213. heads = 16;
  214. cylindersTimesHeads = (int)(totalSectors / sectorsPerTrack);
  215. }
  216. }
  217. cylinders = (ushort)(cylindersTimesHeads / heads);
  218. }
  219. /// <param name="diskSize">In bytes</param>
  220. /// <exception cref="System.IO.IOException"></exception>
  221. /// <exception cref="System.UnauthorizedAccessException"></exception>
  222. public static VirtualHardDisk CreateFixedDisk(string path, long diskSize)
  223. {
  224. if (diskSize % BytesPerDiskSector > 0)
  225. {
  226. throw new ArgumentException("diskSize must be a multiple of sector size");
  227. }
  228. VHDFooter footer = new VHDFooter();
  229. footer.OriginalSize = (ulong)diskSize;
  230. footer.CurrentSize = (ulong)diskSize;
  231. footer.SetCurrentTimeStamp();
  232. footer.SetDiskGeometry((ulong)diskSize / BytesPerDiskSector);
  233. RawDiskImage diskImage = RawDiskImage.Create(path, diskSize + VHDFooter.Length, BytesPerDiskSector);
  234. diskImage.WriteSectors(diskSize / BytesPerDiskSector, footer.GetBytes());
  235. return new VirtualHardDisk(path);
  236. }
  237. }
  238. }