123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268 |
- /* Copyright (C) 2014-2018 Tal Aloni <tal.aloni.il@gmail.com>. All rights reserved.
- *
- * You can redistribute this program and/or modify it under the terms of
- * the GNU Lesser Public License as published by the Free Software Foundation,
- * either version 3 of the License, or (at your option) any later version.
- */
- using System;
- using System.IO;
- using System.Collections.Generic;
- using Utilities;
- using DiskAccessLibrary.VHD;
- namespace DiskAccessLibrary
- {
- public partial class VirtualHardDisk : DiskImage, IDiskGeometry
- {
- // VHD sector size is set to 512 bytes.
- public const int BytesPerDiskSector = 512;
- private RawDiskImage m_file;
- private VHDFooter m_vhdFooter;
- // Dynamic VHD:
- private DynamicDiskHeader m_dynamicHeader;
- private BlockAllocationTable m_blockAllocationTable;
- // CHS:
- private long m_cylinders;
- private int m_tracksPerCylinder; // a.k.a. heads
- private int m_sectorsPerTrack;
- /// <exception cref="System.IO.IOException"></exception>
- /// <exception cref="System.IO.InvalidDataException"></exception>
- /// <exception cref="System.NotImplementedException"></exception>
- /// <exception cref="System.UnauthorizedAccessException"></exception>
- public VirtualHardDisk(string virtualHardDiskPath) : base(virtualHardDiskPath)
- {
- // We can't read the VHD footer using this.ReadSector() because it's out of the disk boundaries
- m_file = new RawDiskImage(virtualHardDiskPath, BytesPerDiskSector);
- byte[] buffer = m_file.ReadSector(m_file.Size / BytesPerDiskSector - 1);
- m_vhdFooter = new VHDFooter(buffer);
- if (!m_vhdFooter.IsValid)
- {
- // check to see if a header is present (dynamic VHD) and use it instead
- buffer = m_file.ReadSector(0);
- m_vhdFooter = new VHDFooter(buffer);
- if (!m_vhdFooter.IsValid)
- {
- throw new InvalidDataException("Invalid VHD footer");
- }
- }
- if (m_vhdFooter.DiskType == VirtualHardDiskType.Fixed)
- {
- }
- else if (m_vhdFooter.DiskType == VirtualHardDiskType.Dynamic)
- {
- buffer = m_file.ReadSectors(1, 2);
- m_dynamicHeader = new DynamicDiskHeader(buffer);
- m_blockAllocationTable = BlockAllocationTable.ReadBlockAllocationTable(virtualHardDiskPath, m_dynamicHeader);
- }
- else
- {
- throw new NotImplementedException("Differencing VHD is not supported");
- }
- SetGeometry();
- }
- private void SetGeometry()
- {
- byte heads;
- byte sectorsPerTrack;
- ushort cylinders;
- GetDiskGeometry((ulong)this.TotalSectors, out heads, out sectorsPerTrack, out cylinders);
- m_cylinders = cylinders;
- m_tracksPerCylinder = heads;
- m_sectorsPerTrack = sectorsPerTrack;
- }
- public override bool ExclusiveLock()
- {
- return m_file.ExclusiveLock();
- }
- public override bool ReleaseLock()
- {
- return m_file.ReleaseLock();
- }
- /// <summary>
- /// Sector refers to physical disk sector, we can only read complete sectors
- /// </summary>
- public override byte[] ReadSectors(long sectorIndex, int sectorCount)
- {
- CheckBoundaries(sectorIndex, sectorCount);
- if (m_vhdFooter.DiskType == VirtualHardDiskType.Fixed)
- {
- return m_file.ReadSectors(sectorIndex, sectorCount);
- }
- else // Dynamic VHD
- {
- return ReadSectorsFromDynamicDisk(sectorIndex, sectorCount);
- }
- }
- public override void WriteSectors(long sectorIndex, byte[] data)
- {
- CheckBoundaries(sectorIndex, data.Length / BytesPerDiskSector);
- if (m_vhdFooter.DiskType == VirtualHardDiskType.Fixed)
- {
- m_file.WriteSectors(sectorIndex, data);
- }
- else // Dynamic VHD
- {
- WriteSectorsToDynamicDisk(sectorIndex, data);
- }
- }
- public override void Extend(long numberOfAdditionalBytes)
- {
- if (numberOfAdditionalBytes % BytesPerDiskSector > 0)
- {
- throw new ArgumentException("numberOfAdditionalBytes must be a multiple of sector size");
- }
- if (m_vhdFooter.DiskType == VirtualHardDiskType.Fixed)
- {
- m_vhdFooter.CurrentSize += (ulong)numberOfAdditionalBytes;
- ExtendFile(numberOfAdditionalBytes);
- }
- else
- {
- throw new NotImplementedException();
- }
- }
- private void ExtendFile(long numberOfAdditionalBytes)
- {
- long footerOffset = m_file.Size - VHDFooter.Length;
- m_file.Extend(numberOfAdditionalBytes);
- byte[] footerBytes = m_vhdFooter.GetBytes();
- if (m_vhdFooter.DiskType != VirtualHardDiskType.Fixed)
- {
- m_file.WriteSectors(0, footerBytes);
- }
- m_file.WriteSectors((footerOffset + numberOfAdditionalBytes) / BytesPerDiskSector, footerBytes);
- }
- public long Cylinders
- {
- get
- {
- return m_cylinders;
- }
- }
- public int TracksPerCylinder
- {
- get
- {
- return m_tracksPerCylinder;
- }
- }
- public int SectorsPerTrack
- {
- get
- {
- return m_sectorsPerTrack;
- }
- }
- public override int BytesPerSector
- {
- get
- {
- return BytesPerDiskSector;
- }
- }
- public override long Size
- {
- get
- {
- return (long)m_vhdFooter.CurrentSize;
- }
- }
- public VHDFooter Footer
- {
- get
- {
- return m_vhdFooter;
- }
- }
- // Taken From VHD format specs (Appendix)
- public static void GetDiskGeometry(ulong totalSectors, out byte heads, out byte sectorsPerTrack, out ushort cylinders)
- {
- int cylindersTimesHeads;
- // If more than ~128GB truncate at ~128GB
- if (totalSectors > 65535 * 16 * 255)
- {
- totalSectors = 65535 * 16 * 255;
- }
- // If more than ~32GB, break partition table compatibility.
- // Partition table has max 63 sectors per track. Otherwise
- // we're looking for a geometry that's valid for both BIOS
- // and ATA.
- if (totalSectors >= 65535 * 16 * 63)
- {
- sectorsPerTrack = 255;
- heads = 16;
- cylindersTimesHeads = (int)(totalSectors / sectorsPerTrack);
- }
- else
- {
- sectorsPerTrack = 17;
- cylindersTimesHeads = (int)(totalSectors / sectorsPerTrack);
- heads = (byte)((cylindersTimesHeads + 1023) / 1024);
- if (heads < 4)
- {
- heads = 4;
- }
- if (cylindersTimesHeads >= (heads * 1024) || heads > 16)
- {
- sectorsPerTrack = 31;
- heads = 16;
- cylindersTimesHeads = (int)(totalSectors / sectorsPerTrack);
- }
- if (cylindersTimesHeads >= (heads * 1024))
- {
- sectorsPerTrack = 63;
- heads = 16;
- cylindersTimesHeads = (int)(totalSectors / sectorsPerTrack);
- }
- }
- cylinders = (ushort)(cylindersTimesHeads / heads);
- }
- /// <param name="diskSize">In bytes</param>
- /// <exception cref="System.IO.IOException"></exception>
- /// <exception cref="System.UnauthorizedAccessException"></exception>
- public static VirtualHardDisk CreateFixedDisk(string path, long diskSize)
- {
- if (diskSize % BytesPerDiskSector > 0)
- {
- throw new ArgumentException("diskSize must be a multiple of sector size");
- }
- VHDFooter footer = new VHDFooter();
- footer.OriginalSize = (ulong)diskSize;
- footer.CurrentSize = (ulong)diskSize;
- footer.SetCurrentTimeStamp();
- footer.SetDiskGeometry((ulong)diskSize / BytesPerDiskSector);
- RawDiskImage diskImage = RawDiskImage.Create(path, diskSize + VHDFooter.Length, BytesPerDiskSector);
- diskImage.WriteSectors(diskSize / BytesPerDiskSector, footer.GetBytes());
- return new VirtualHardDisk(path);
- }
- }
- }
|