// // Copyright (c) 2008-2011, Kenneth Bell // // Permission is hereby granted, free of charge, to any person obtaining a // copy of this software and associated documentation files (the "Software"), // to deal in the Software without restriction, including without limitation // the rights to use, copy, modify, merge, publish, distribute, sublicense, // and/or sell copies of the Software, and to permit persons to whom the // Software is furnished to do so, subject to the following conditions: // // The above copyright notice and this permission notice shall be included in // all copies or substantial portions of the Software. // // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING // FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER // DEALINGS IN THE SOFTWARE. // using System; using DiscUtils.Internal; using DiscUtils.Streams; namespace DiscUtils.Partitions { internal class GptHeader { public const string GptSignature = "EFI PART"; public long AlternateHeaderLba; public byte[] Buffer; public uint Crc; public Guid DiskGuid; public uint EntriesCrc; public long FirstUsable; public long HeaderLba; public int HeaderSize; public long LastUsable; public long PartitionEntriesLba; public uint PartitionEntryCount; public int PartitionEntrySize; public string Signature; public uint Version; public GptHeader(int sectorSize) { Signature = GptSignature; Version = 0x00010000; HeaderSize = 92; Buffer = new byte[sectorSize]; } public GptHeader(GptHeader toCopy) { Signature = toCopy.Signature; Version = toCopy.Version; HeaderSize = toCopy.HeaderSize; Crc = toCopy.Crc; HeaderLba = toCopy.HeaderLba; AlternateHeaderLba = toCopy.AlternateHeaderLba; FirstUsable = toCopy.FirstUsable; LastUsable = toCopy.LastUsable; DiskGuid = toCopy.DiskGuid; PartitionEntriesLba = toCopy.PartitionEntriesLba; PartitionEntryCount = toCopy.PartitionEntryCount; PartitionEntrySize = toCopy.PartitionEntrySize; EntriesCrc = toCopy.EntriesCrc; Buffer = new byte[toCopy.Buffer.Length]; Array.Copy(toCopy.Buffer, Buffer, Buffer.Length); } public bool ReadFrom(byte[] buffer, int offset) { Signature = EndianUtilities.BytesToString(buffer, offset + 0, 8); Version = EndianUtilities.ToUInt32LittleEndian(buffer, offset + 8); HeaderSize = EndianUtilities.ToInt32LittleEndian(buffer, offset + 12); Crc = EndianUtilities.ToUInt32LittleEndian(buffer, offset + 16); HeaderLba = EndianUtilities.ToInt64LittleEndian(buffer, offset + 24); AlternateHeaderLba = EndianUtilities.ToInt64LittleEndian(buffer, offset + 32); FirstUsable = EndianUtilities.ToInt64LittleEndian(buffer, offset + 40); LastUsable = EndianUtilities.ToInt64LittleEndian(buffer, offset + 48); DiskGuid = EndianUtilities.ToGuidLittleEndian(buffer, offset + 56); PartitionEntriesLba = EndianUtilities.ToInt64LittleEndian(buffer, offset + 72); PartitionEntryCount = EndianUtilities.ToUInt32LittleEndian(buffer, offset + 80); PartitionEntrySize = EndianUtilities.ToInt32LittleEndian(buffer, offset + 84); EntriesCrc = EndianUtilities.ToUInt32LittleEndian(buffer, offset + 88); // In case the header has new fields unknown to us, store the entire header // as a byte array Buffer = new byte[HeaderSize]; Array.Copy(buffer, offset, Buffer, 0, HeaderSize); // Reject obviously invalid data if (Signature != GptSignature || HeaderSize == 0) { return false; } return Crc == CalcCrc(buffer, offset, HeaderSize); } public void WriteTo(byte[] buffer, int offset) { // First, copy the cached header to allow for unknown fields Array.Copy(Buffer, 0, buffer, offset, Buffer.Length); // Next, write the fields EndianUtilities.StringToBytes(Signature, buffer, offset + 0, 8); EndianUtilities.WriteBytesLittleEndian(Version, buffer, offset + 8); EndianUtilities.WriteBytesLittleEndian(HeaderSize, buffer, offset + 12); EndianUtilities.WriteBytesLittleEndian((uint)0, buffer, offset + 16); EndianUtilities.WriteBytesLittleEndian(HeaderLba, buffer, offset + 24); EndianUtilities.WriteBytesLittleEndian(AlternateHeaderLba, buffer, offset + 32); EndianUtilities.WriteBytesLittleEndian(FirstUsable, buffer, offset + 40); EndianUtilities.WriteBytesLittleEndian(LastUsable, buffer, offset + 48); EndianUtilities.WriteBytesLittleEndian(DiskGuid, buffer, offset + 56); EndianUtilities.WriteBytesLittleEndian(PartitionEntriesLba, buffer, offset + 72); EndianUtilities.WriteBytesLittleEndian(PartitionEntryCount, buffer, offset + 80); EndianUtilities.WriteBytesLittleEndian(PartitionEntrySize, buffer, offset + 84); EndianUtilities.WriteBytesLittleEndian(EntriesCrc, buffer, offset + 88); // Calculate & write the CRC EndianUtilities.WriteBytesLittleEndian(CalcCrc(buffer, offset, HeaderSize), buffer, offset + 16); // Update the cached copy - re-allocate the buffer to allow for HeaderSize potentially having changed Buffer = new byte[HeaderSize]; Array.Copy(buffer, offset, Buffer, 0, HeaderSize); } internal static uint CalcCrc(byte[] buffer, int offset, int count) { byte[] temp = new byte[count]; Array.Copy(buffer, offset, temp, 0, count); // Reset CRC field EndianUtilities.WriteBytesLittleEndian((uint)0, temp, 16); return Crc32LittleEndian.Compute(Crc32Algorithm.Common, temp, 0, count); } } }