VirtualHardDisk.Dynamic.cs 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279
  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
  15. {
  16. private byte[] ReadSectorsFromDynamicDisk(long sectorIndex, int sectorCount)
  17. {
  18. byte[] buffer = new byte[sectorCount * BytesPerDiskSector];
  19. int sectorsInBlock = (int)(m_dynamicHeader.BlockSize / BytesPerDiskSector);
  20. int sectorOffset = 0;
  21. while (sectorOffset < sectorCount)
  22. {
  23. uint blockIndex = (uint)((sectorIndex + sectorOffset) * BytesPerDiskSector / m_dynamicHeader.BlockSize);
  24. int sectorOffsetInBlock = (int)(((sectorIndex + sectorOffset) * BytesPerDiskSector % m_dynamicHeader.BlockSize) / BytesPerDiskSector);
  25. int sectorsRemainingInBlock = sectorsInBlock - sectorOffsetInBlock;
  26. int sectorsToRead = Math.Min(sectorCount - sectorOffset, sectorsRemainingInBlock);
  27. uint blockStartSector;
  28. if (m_blockAllocationTable.IsBlockInUse(blockIndex, out blockStartSector))
  29. {
  30. // Each data block has a sector bitmap preceding the data, the bitmap is padded to a 512-byte sector boundary.
  31. int blockBitmapSectorCount = (int)Math.Ceiling((double)sectorsInBlock / (BytesPerDiskSector * 8));
  32. // "All sectors within a block whose corresponding bits in the bitmap are zero must contain 512 bytes of zero on disk"
  33. byte[] temp = m_file.ReadSectors(blockStartSector + blockBitmapSectorCount + sectorOffsetInBlock, sectorsToRead);
  34. ByteWriter.WriteBytes(buffer, sectorOffset * BytesPerDiskSector, temp);
  35. }
  36. sectorOffset += sectorsToRead;
  37. }
  38. return buffer;
  39. }
  40. public bool AreSectorsInUse(long sectorIndex, int sectorCount)
  41. {
  42. if (m_vhdFooter.DiskType != VirtualHardDiskType.Fixed)
  43. {
  44. int sectorsInBlock = (int)(m_dynamicHeader.BlockSize / BytesPerDiskSector);
  45. int sectorOffset = 0;
  46. while (sectorOffset < sectorCount)
  47. {
  48. uint blockIndex = (uint)((sectorIndex + sectorOffset) * BytesPerDiskSector / m_dynamicHeader.BlockSize);
  49. int sectorOffsetInBlock = (int)(((sectorIndex + sectorOffset) * BytesPerDiskSector % m_dynamicHeader.BlockSize) / BytesPerDiskSector);
  50. int sectorsRemainingInBlock = sectorsInBlock - sectorOffsetInBlock;
  51. int sectorsToRead = Math.Min(sectorCount - sectorOffset, sectorsRemainingInBlock);
  52. uint blockStartSector;
  53. if (m_blockAllocationTable.IsBlockInUse(blockIndex, out blockStartSector))
  54. {
  55. byte[] bitmap = ReadBlockUsageBitmap(blockIndex);
  56. if (!AreSectorsInUse(bitmap, sectorOffsetInBlock, sectorsToRead))
  57. {
  58. return false;
  59. }
  60. }
  61. else
  62. {
  63. return false;
  64. }
  65. sectorOffset += sectorsToRead;
  66. }
  67. }
  68. return true;
  69. }
  70. private byte[] ReadBlockUsageBitmap(uint blockIndex)
  71. {
  72. if (m_vhdFooter.DiskType != VirtualHardDiskType.Fixed)
  73. {
  74. uint blockStartSector;
  75. int sectorsInBlock = (int)(m_dynamicHeader.BlockSize / BytesPerDiskSector);
  76. // Each data block has a sector bitmap preceding the data, the bitmap is padded to a 512-byte sector boundary.
  77. int blockBitmapSectorCount = (int)Math.Ceiling((double)sectorsInBlock / (BytesPerDiskSector * 8));
  78. if (m_blockAllocationTable.IsBlockInUse(blockIndex, out blockStartSector))
  79. {
  80. byte[] bitmap = m_file.ReadSectors(blockStartSector, blockBitmapSectorCount);
  81. return bitmap;
  82. }
  83. else
  84. {
  85. return new byte[blockBitmapSectorCount * BytesPerDiskSector];
  86. }
  87. }
  88. else
  89. {
  90. throw new InvalidOperationException("Fixed VHDs do not have a Block Usage Bitmap");
  91. }
  92. }
  93. private void WriteSectorsToDynamicDisk(long sectorIndex, byte[] data)
  94. {
  95. int sectorCount = data.Length / BytesPerDiskSector;
  96. int sectorsInBlock = (int)(m_dynamicHeader.BlockSize / BytesPerDiskSector);
  97. int sectorOffset = 0;
  98. while (sectorOffset < sectorCount)
  99. {
  100. uint blockIndex = (uint)((sectorIndex + sectorOffset) * BytesPerDiskSector / m_dynamicHeader.BlockSize);
  101. int sectorOffsetInBlock = (int)(((sectorIndex + sectorOffset) * BytesPerDiskSector % m_dynamicHeader.BlockSize) / BytesPerDiskSector);
  102. int sectorsRemainingInBlock = sectorsInBlock - sectorOffsetInBlock;
  103. int sectorsToWrite = Math.Min(sectorCount - sectorOffset, sectorsRemainingInBlock);
  104. uint blockStartSector;
  105. if (!m_blockAllocationTable.IsBlockInUse(blockIndex, out blockStartSector))
  106. {
  107. blockStartSector = AllocateDynamicDiskBlock(blockIndex);
  108. }
  109. // Each data block has a sector bitmap preceding the data, the bitmap is padded to a 512-byte sector boundary.
  110. int blockBitmapSectorCount = (int)Math.Ceiling((double)sectorsInBlock / (BytesPerDiskSector * 8));
  111. // "All sectors within a block whose corresponding bits in the bitmap are zero must contain 512 bytes of zero on disk"
  112. int blockBitmapSectorOffset = sectorOffsetInBlock / (BytesPerDiskSector * 8);
  113. int sectorOffsetInBitmap = sectorOffsetInBlock % (BytesPerDiskSector * 8);
  114. int remainingInBitmapSector = (BytesPerDiskSector * 8) - sectorOffsetInBitmap;
  115. int blockBitmapSectorsToUpdate = 1;
  116. if (sectorsToWrite > remainingInBitmapSector)
  117. {
  118. blockBitmapSectorsToUpdate = (int)Math.Ceiling((double)(sectorsToWrite - remainingInBitmapSector) / (BytesPerDiskSector * 8));
  119. }
  120. byte[] bitmap = m_file.ReadSectors(blockStartSector + blockBitmapSectorOffset, blockBitmapSectorsToUpdate);
  121. UpdateBlockUsageBitmap(bitmap, sectorOffsetInBitmap, sectorsToWrite, true);
  122. m_file.WriteSectors(blockStartSector + blockBitmapSectorOffset, bitmap);
  123. byte[] temp = ByteReader.ReadBytes(data, sectorOffset * BytesPerDiskSector, sectorsToWrite * BytesPerDiskSector);
  124. m_file.WriteSectors(blockStartSector + blockBitmapSectorCount + sectorOffsetInBlock, temp);
  125. sectorOffset += sectorsToWrite;
  126. }
  127. }
  128. /// <returns>Block start sector</returns>
  129. private uint AllocateDynamicDiskBlock(uint blockIndex)
  130. {
  131. long footerOffset = m_file.Size - VHDFooter.Length;
  132. uint blockStartSector = (uint)(footerOffset / BytesPerDiskSector);
  133. int sectorsInBlock = (int)(m_dynamicHeader.BlockSize / BytesPerDiskSector);
  134. // Each data block has a sector bitmap preceding the data, the bitmap is padded to a 512-byte sector boundary.
  135. int blockBitmapSectorCount = (int)Math.Ceiling((double)sectorsInBlock / (BytesPerDiskSector * 8));
  136. // Block Size does not include the size of the block bitmap.
  137. ExtendFile(blockBitmapSectorCount * BytesPerDiskSector + m_dynamicHeader.BlockSize);
  138. byte[] bitmap = new byte[blockBitmapSectorCount * BytesPerDiskSector];
  139. m_file.WriteSectors(blockStartSector, bitmap);
  140. // All sectors within a block whose corresponding bits in the bitmap are zero must contain 512 bytes of zero on disk.
  141. byte[] blockSectors = new byte[sectorsInBlock * BytesPerDiskSector];
  142. m_file.WriteSectors(blockStartSector + blockBitmapSectorCount, blockSectors);
  143. // Update the Block Allocation Table
  144. m_blockAllocationTable.SetBlockStartSector(blockIndex, blockStartSector);
  145. byte[] blockAllocationTableBytes = m_blockAllocationTable.GetBytes();
  146. long blockAllocationTableSectorIndex = (long)(m_dynamicHeader.TableOffset / VirtualHardDisk.BytesPerDiskSector);
  147. m_file.WriteSectors(blockAllocationTableSectorIndex, blockAllocationTableBytes);
  148. return blockStartSector;
  149. }
  150. public void AllocateAllDynamicDiskBlocks()
  151. {
  152. if (m_vhdFooter.DiskType != VirtualHardDiskType.Fixed)
  153. {
  154. for (uint blockIndex = 0; blockIndex < m_dynamicHeader.MaxTableEntries; blockIndex++)
  155. {
  156. if (!m_blockAllocationTable.IsBlockInUse(blockIndex))
  157. {
  158. AllocateDynamicDiskBlock(blockIndex);
  159. }
  160. }
  161. }
  162. }
  163. private static bool AreSectorsInUse(byte[] bitmap, int sectorOffsetInBitmap, int sectorCount)
  164. {
  165. int leadingBits = (8 - (sectorOffsetInBitmap % 8)) % 8;
  166. for (int sectorOffset = 0; sectorOffset < leadingBits; sectorOffset++)
  167. {
  168. if (!IsSectorInUse(bitmap, sectorOffsetInBitmap + sectorOffset))
  169. {
  170. return false;
  171. }
  172. }
  173. int byteCount = Math.Max(sectorCount - leadingBits, 0) / 8;
  174. int byteOffsetInBitmap = (sectorOffsetInBitmap + leadingBits) / 8;
  175. for (int byteOffset = 0; byteOffset < byteCount; byteOffset++)
  176. {
  177. if (bitmap[byteOffsetInBitmap + byteOffset] != 0xFF)
  178. {
  179. return false;
  180. }
  181. }
  182. for (int sectorOffset = leadingBits + byteCount * 8; sectorOffset < sectorCount; sectorOffset++)
  183. {
  184. if (!IsSectorInUse(bitmap, sectorOffsetInBitmap + sectorOffset))
  185. {
  186. return false;
  187. }
  188. }
  189. return true;
  190. }
  191. private static bool IsSectorInUse(byte[] bitmap, int sectorOffsetInBitmap)
  192. {
  193. int byteOffset = sectorOffsetInBitmap / 8;
  194. int bitOffset = 7 - sectorOffsetInBitmap % 8;
  195. bool isUsed = (bitmap[byteOffset] & (byte)(0x01 << bitOffset)) > 0;
  196. return isUsed;
  197. }
  198. private static void UpdateBlockUsageBitmap(byte[] bitmap, int sectorOffsetInBitmap, int sectorCount, bool isUsed)
  199. {
  200. for (int offsetInBitmap = 0; offsetInBitmap < sectorCount; offsetInBitmap++)
  201. {
  202. UpdateBlockUsageBitmap(bitmap, sectorOffsetInBitmap + offsetInBitmap, isUsed);
  203. }
  204. }
  205. private static void UpdateBlockUsageBitmap(byte[] bitmap, int sectorOffsetInBitmap, bool isUsed)
  206. {
  207. int byteOffset = sectorOffsetInBitmap / 8;
  208. int bitOffset = 7 - sectorOffsetInBitmap % 8;
  209. if (isUsed)
  210. {
  211. bitmap[byteOffset] |= (byte)(0x01 << bitOffset);
  212. }
  213. else
  214. {
  215. bitmap[byteOffset] &= (byte)(~(0x01 << bitOffset));
  216. }
  217. }
  218. /// <param name="diskSize">In bytes</param>
  219. /// <exception cref="System.IO.IOException"></exception>
  220. /// <exception cref="System.UnauthorizedAccessException"></exception>
  221. public static VirtualHardDisk CreateDynamicDisk(string path, long diskSize)
  222. {
  223. if (diskSize % BytesPerDiskSector > 0)
  224. {
  225. throw new ArgumentException("diskSize must be a multiple of sector size");
  226. }
  227. const int BlockSizeInBytes = 4096 * BytesPerDiskSector;
  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. footer.DiskType = VirtualHardDiskType.Dynamic;
  234. DynamicDiskHeader header = new DynamicDiskHeader();
  235. header.TableOffset = VHDFooter.Length + DynamicDiskHeader.Length;
  236. header.BlockSize = BlockSizeInBytes;
  237. header.MaxTableEntries = (uint)Math.Ceiling((double)diskSize / BlockSizeInBytes);
  238. BlockAllocationTable blockAllocationTable = new BlockAllocationTable(header.MaxTableEntries);
  239. byte[] footerBytes = footer.GetBytes();
  240. byte[] headerBytes = header.GetBytes();
  241. byte[] blockAllocationTableBytes = blockAllocationTable.GetBytes();
  242. int fileSize = VHDFooter.Length + DynamicDiskHeader.Length + blockAllocationTableBytes.Length + VHDFooter.Length;
  243. RawDiskImage diskImage = RawDiskImage.Create(path, fileSize, BytesPerDiskSector);
  244. diskImage.WriteSectors(0, footerBytes);
  245. diskImage.WriteSectors(1, headerBytes);
  246. diskImage.WriteSectors(3, blockAllocationTableBytes);
  247. diskImage.WriteSectors(fileSize / BytesPerDiskSector - 1, footerBytes);
  248. return new VirtualHardDisk(path);
  249. }
  250. }
  251. }