VirtualHardDisk.cs 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292
  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.IO;
  9. using System.Collections.Generic;
  10. using System.Text;
  11. using Utilities;
  12. using DiskAccessLibrary.VHD;
  13. namespace DiskAccessLibrary
  14. {
  15. public partial class VirtualHardDisk : DiskImage, IDiskGeometry
  16. {
  17. private RawDiskImage m_file;
  18. private VHDFooter m_vhdFooter;
  19. // Dynamic VHD:
  20. private DynamicDiskHeader m_dynamicHeader;
  21. private BlockAllocationTable m_blockAllocationTable;
  22. // CHS:
  23. private long m_cylinders;
  24. private int m_tracksPerCylinder; // a.k.a. heads
  25. private int m_sectorsPerTrack;
  26. public VirtualHardDisk(string virtualHardDiskPath) : base(virtualHardDiskPath)
  27. {
  28. // We can't read the VHD footer using this.ReadSector() because it's out of the disk boundaries
  29. m_file = new RawDiskImage(virtualHardDiskPath);
  30. byte[] buffer = m_file.ReadSector(m_file.Size / m_file.BytesPerSector - 1);
  31. m_vhdFooter = new VHDFooter(buffer);
  32. if (!m_vhdFooter.IsValid)
  33. {
  34. // check to see if a header is present (dynamic VHD) and use it instead
  35. buffer = m_file.ReadSector(0);
  36. m_vhdFooter = new VHDFooter(buffer);
  37. if (!m_vhdFooter.IsValid)
  38. {
  39. throw new InvalidDataException("Invalid VHD footer");
  40. }
  41. }
  42. if (m_vhdFooter.DiskType == VirtualHardDiskType.Fixed)
  43. {
  44. }
  45. else if (m_vhdFooter.DiskType == VirtualHardDiskType.Dynamic)
  46. {
  47. buffer = m_file.ReadSectors(1, 2);
  48. m_dynamicHeader = new DynamicDiskHeader(buffer);
  49. m_blockAllocationTable = BlockAllocationTable.ReadBlockAllocationTable(virtualHardDiskPath, m_dynamicHeader);
  50. this.IsReadOnly = true;
  51. }
  52. else
  53. {
  54. throw new NotImplementedException("Differencing VHD is not supported");
  55. }
  56. SetGeometry();
  57. }
  58. private void SetGeometry()
  59. {
  60. byte heads;
  61. byte sectorsPerTrack;
  62. ushort cylinders;
  63. GetDiskGeometry((ulong)this.TotalSectors, out heads, out sectorsPerTrack, out cylinders);
  64. m_cylinders = cylinders;
  65. m_tracksPerCylinder = heads;
  66. m_sectorsPerTrack = sectorsPerTrack;
  67. }
  68. public override bool ExclusiveLock()
  69. {
  70. return m_file.ExclusiveLock();
  71. }
  72. public override bool ReleaseLock()
  73. {
  74. return m_file.ReleaseLock();
  75. }
  76. /// <summary>
  77. /// Sector refers to physical disk sector, we can only read complete sectors
  78. /// </summary>
  79. public override byte[] ReadSectors(long sectorIndex, int sectorCount)
  80. {
  81. CheckBoundaries(sectorIndex, sectorCount);
  82. if (m_vhdFooter.DiskType == VirtualHardDiskType.Fixed)
  83. {
  84. return m_file.ReadSectors(sectorIndex, sectorCount);
  85. }
  86. else // dynamic
  87. {
  88. byte[] buffer = new byte[sectorCount * this.BytesPerSector];
  89. int sectorOffset = 0;
  90. while (sectorOffset < sectorCount)
  91. {
  92. uint blockIndex = (uint)((sectorIndex + sectorOffset) * this.BytesPerSector / m_dynamicHeader.BlockSize);
  93. int sectorOffsetInBlock = (int)(((sectorIndex + sectorOffset) * this.BytesPerSector % m_dynamicHeader.BlockSize) / this.BytesPerSector);
  94. int sectorsInBlock = (int)(m_dynamicHeader.BlockSize / this.BytesPerSector);
  95. int sectorsRemainingInBlock = sectorsInBlock - sectorOffsetInBlock;
  96. int sectorsToRead = Math.Min(sectorCount - sectorOffset, sectorsRemainingInBlock);
  97. if (m_blockAllocationTable.IsBlockInUse(blockIndex))
  98. {
  99. uint blockStartSector = m_blockAllocationTable.Entries[blockIndex];
  100. // each data block has a sector bitmap preceding the data, the bitmap is padded to a 512-byte sector boundary.
  101. int blockBitmapSectorCount = (int)Math.Ceiling((double)sectorsInBlock / (this.BytesPerSector * 8));
  102. // "All sectors within a block whose corresponding bits in the bitmap are zero must contain 512 bytes of zero on disk"
  103. byte[] temp = m_file.ReadSectors(blockStartSector + blockBitmapSectorCount + sectorOffsetInBlock, sectorsToRead);
  104. ByteWriter.WriteBytes(buffer, sectorOffset * this.BytesPerSector, temp);
  105. }
  106. sectorOffset += sectorsToRead;
  107. }
  108. return buffer;
  109. }
  110. }
  111. public override void WriteSectors(long sectorIndex, byte[] data)
  112. {
  113. CheckBoundaries(sectorIndex, data.Length / this.BytesPerSector);
  114. m_file.WriteSectors(sectorIndex, data);
  115. }
  116. public override void Extend(long additionalNumberOfBytes)
  117. {
  118. if (additionalNumberOfBytes % this.BytesPerSector > 0)
  119. {
  120. throw new ArgumentException("additionalNumberOfBytes must be a multiple of BytesPerSector");
  121. }
  122. if (m_vhdFooter.DiskType == VirtualHardDiskType.Fixed)
  123. {
  124. long length = this.Size; // does not include the footer
  125. m_file.Extend(additionalNumberOfBytes);
  126. m_vhdFooter.CurrentSize += (ulong)additionalNumberOfBytes;
  127. byte[] footerBytes = m_vhdFooter.GetBytes();
  128. m_file.WriteSectors((length + additionalNumberOfBytes) / this.BytesPerSector, footerBytes);
  129. }
  130. else
  131. {
  132. throw new NotImplementedException();
  133. }
  134. }
  135. public long Cylinders
  136. {
  137. get
  138. {
  139. return m_cylinders;
  140. }
  141. }
  142. public int TracksPerCylinder
  143. {
  144. get
  145. {
  146. return m_tracksPerCylinder;
  147. }
  148. }
  149. public int SectorsPerTrack
  150. {
  151. get
  152. {
  153. return m_sectorsPerTrack;
  154. }
  155. }
  156. public override long Size
  157. {
  158. get
  159. {
  160. return (long)m_vhdFooter.CurrentSize;
  161. }
  162. }
  163. public VHDFooter Footer
  164. {
  165. get
  166. {
  167. return m_vhdFooter;
  168. }
  169. }
  170. // Taken From VHD format specs (Appendix)
  171. public static void GetDiskGeometry(ulong totalSectors, out byte heads, out byte sectorsPerTrack, out ushort cylinders)
  172. {
  173. int cylindersTimesHeads;
  174. // If more than ~128GB truncate at ~128GB
  175. if (totalSectors > 65535 * 16 * 255)
  176. {
  177. totalSectors = 65535 * 16 * 255;
  178. }
  179. // If more than ~32GB, break partition table compatibility.
  180. // Partition table has max 63 sectors per track. Otherwise
  181. // we're looking for a geometry that's valid for both BIOS
  182. // and ATA.
  183. if (totalSectors >= 65535 * 16 * 63)
  184. {
  185. sectorsPerTrack = 255;
  186. heads = 16;
  187. cylindersTimesHeads = (int)(totalSectors / sectorsPerTrack);
  188. }
  189. else
  190. {
  191. sectorsPerTrack = 17;
  192. cylindersTimesHeads = (int)(totalSectors / sectorsPerTrack);
  193. heads = (byte)((cylindersTimesHeads + 1023) / 1024);
  194. if (heads < 4)
  195. {
  196. heads = 4;
  197. }
  198. if (cylindersTimesHeads >= (heads * 1024) || heads > 16)
  199. {
  200. sectorsPerTrack = 31;
  201. heads = 16;
  202. cylindersTimesHeads = (int)(totalSectors / sectorsPerTrack);
  203. }
  204. if (cylindersTimesHeads >= (heads * 1024))
  205. {
  206. sectorsPerTrack = 63;
  207. heads = 16;
  208. cylindersTimesHeads = (int)(totalSectors / sectorsPerTrack);
  209. }
  210. }
  211. cylinders = (ushort)(cylindersTimesHeads / heads);
  212. }
  213. /// <param name="length">In bytes</param>
  214. /// <exception cref="System.IO.IOException"></exception>
  215. /// <exception cref="System.UnauthorizedAccessException"></exception>
  216. public static VirtualHardDisk Create(string path, long size)
  217. {
  218. #if Win32
  219. // calling AdjustTokenPrivileges and then immediately calling SetFileValidData will sometimes result in ERROR_PRIVILEGE_NOT_HELD.
  220. // We can work around the issue by obtaining the privilege before obtaining the handle.
  221. bool hasManageVolumePrivilege = SecurityUtils.ObtainManageVolumePrivilege();
  222. #endif
  223. FileStream stream = new FileStream(path, FileMode.Create, FileAccess.Write, FileShare.None);
  224. try
  225. {
  226. stream.SetLength(size + 512); // VHD footer is 512 bytes
  227. }
  228. catch (IOException)
  229. {
  230. stream.Close();
  231. try
  232. {
  233. // Delete the incomplete file
  234. File.Delete(path);
  235. }
  236. catch (IOException)
  237. {
  238. }
  239. throw;
  240. }
  241. #if Win32
  242. if (hasManageVolumePrivilege)
  243. {
  244. try
  245. {
  246. FileStreamUtils.SetValidLength(stream, size + 512);
  247. }
  248. catch (IOException)
  249. {
  250. }
  251. }
  252. #endif
  253. VHDFooter footer = new VHDFooter();
  254. footer.OriginalSize = (ulong)size;
  255. footer.CurrentSize = (ulong)size;
  256. footer.SetCurrentTimeStamp();
  257. footer.SetDiskGeometry((ulong)size / DiskImage.BytesPerDiskImageSector);
  258. stream.Seek(size, SeekOrigin.Begin);
  259. stream.Write(footer.GetBytes(), 0, VHDFooter.Length);
  260. stream.Close();
  261. return new VirtualHardDisk(path);
  262. }
  263. }
  264. }