mirror of
https://github.com/ReneLergner/WPinternals.git
synced 2026-06-14 19:36:41 +10:00
220 lines
10 KiB
C#
220 lines
10 KiB
C#
//
|
|
// 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.
|
|
//
|
|
|
|
namespace DiscUtils.Ntfs
|
|
{
|
|
using System;
|
|
using System.Globalization;
|
|
using System.IO;
|
|
|
|
internal class BiosParameterBlock
|
|
{
|
|
public string OemId;
|
|
public ushort BytesPerSector;
|
|
public byte SectorsPerCluster;
|
|
public ushort ReservedSectors; // Must be 0
|
|
public byte NumFats; // Must be 0
|
|
public ushort FatRootEntriesCount; // Must be 0
|
|
public ushort TotalSectors16; // Must be 0
|
|
public byte Media; // Must be 0xF8
|
|
public ushort FatSize16; // Must be 0
|
|
public ushort SectorsPerTrack; // Value: 0x3F 0x00
|
|
public ushort NumHeads; // Value: 0xFF 0x00
|
|
public uint HiddenSectors; // Value: 0x3F 0x00 0x00 0x00
|
|
public uint TotalSectors32; // Must be 0
|
|
public byte BiosDriveNumber; // Value: 0x80 (first hard disk)
|
|
public byte ChkDskFlags; // Value: 0x00
|
|
public byte SignatureByte; // Value: 0x80
|
|
public byte PaddingByte; // Value: 0x00
|
|
public long TotalSectors64;
|
|
public long MftCluster;
|
|
public long MftMirrorCluster;
|
|
public byte RawMftRecordSize;
|
|
public byte RawIndexBufferSize;
|
|
public ulong VolumeSerialNumber;
|
|
|
|
public int MftRecordSize
|
|
{
|
|
get { return CalcRecordSize(RawMftRecordSize); }
|
|
}
|
|
|
|
public int IndexBufferSize
|
|
{
|
|
get { return CalcRecordSize(RawIndexBufferSize); }
|
|
}
|
|
|
|
public int BytesPerCluster
|
|
{
|
|
get { return ((int)BytesPerSector) * ((int)SectorsPerCluster); }
|
|
}
|
|
|
|
public void Dump(TextWriter writer, string linePrefix)
|
|
{
|
|
writer.WriteLine(linePrefix + "BIOS PARAMETER BLOCK (BPB)");
|
|
writer.WriteLine(linePrefix + " OEM ID: " + OemId);
|
|
writer.WriteLine(linePrefix + " Bytes per Sector: " + BytesPerSector);
|
|
writer.WriteLine(linePrefix + " Sectors per Cluster: " + SectorsPerCluster);
|
|
writer.WriteLine(linePrefix + " Reserved Sectors: " + ReservedSectors);
|
|
writer.WriteLine(linePrefix + " # FATs: " + NumFats);
|
|
writer.WriteLine(linePrefix + " # FAT Root Entries: " + FatRootEntriesCount);
|
|
writer.WriteLine(linePrefix + " Total Sectors (16b): " + TotalSectors16);
|
|
writer.WriteLine(linePrefix + " Media: " + Media.ToString("X", CultureInfo.InvariantCulture) + "h");
|
|
writer.WriteLine(linePrefix + " FAT size (16b): " + FatSize16);
|
|
writer.WriteLine(linePrefix + " Sectors per Track: " + SectorsPerTrack);
|
|
writer.WriteLine(linePrefix + " # Heads: " + NumHeads);
|
|
writer.WriteLine(linePrefix + " Hidden Sectors: " + HiddenSectors);
|
|
writer.WriteLine(linePrefix + " Total Sectors (32b): " + TotalSectors32);
|
|
writer.WriteLine(linePrefix + " BIOS Drive Number: " + BiosDriveNumber);
|
|
writer.WriteLine(linePrefix + " Chkdsk Flags: " + ChkDskFlags);
|
|
writer.WriteLine(linePrefix + " Signature Byte: " + SignatureByte);
|
|
writer.WriteLine(linePrefix + " Total Sectors (64b): " + TotalSectors64);
|
|
writer.WriteLine(linePrefix + " MFT Record Size: " + RawMftRecordSize);
|
|
writer.WriteLine(linePrefix + " Index Buffer Size: " + RawIndexBufferSize);
|
|
writer.WriteLine(linePrefix + " Volume Serial Number: " + VolumeSerialNumber);
|
|
}
|
|
|
|
internal static BiosParameterBlock Initialized(Geometry diskGeometry, int clusterSize, uint partitionStartLba, long partitionSizeLba, int mftRecordSize, int indexBufferSize)
|
|
{
|
|
BiosParameterBlock bpb = new BiosParameterBlock();
|
|
bpb.OemId = "NTFS ";
|
|
bpb.BytesPerSector = Sizes.Sector;
|
|
bpb.SectorsPerCluster = (byte)(clusterSize / bpb.BytesPerSector);
|
|
bpb.ReservedSectors = 0;
|
|
bpb.NumFats = 0;
|
|
bpb.FatRootEntriesCount = 0;
|
|
bpb.TotalSectors16 = 0;
|
|
bpb.Media = 0xF8;
|
|
bpb.FatSize16 = 0;
|
|
bpb.SectorsPerTrack = (ushort)diskGeometry.SectorsPerTrack;
|
|
bpb.NumHeads = (ushort)diskGeometry.HeadsPerCylinder;
|
|
bpb.HiddenSectors = partitionStartLba;
|
|
bpb.TotalSectors32 = 0;
|
|
bpb.BiosDriveNumber = 0x80;
|
|
bpb.ChkDskFlags = 0;
|
|
bpb.SignatureByte = 0x80;
|
|
bpb.PaddingByte = 0;
|
|
bpb.TotalSectors64 = partitionSizeLba - 1;
|
|
bpb.RawMftRecordSize = bpb.CodeRecordSize(mftRecordSize);
|
|
bpb.RawIndexBufferSize = bpb.CodeRecordSize(indexBufferSize);
|
|
bpb.VolumeSerialNumber = GenSerialNumber();
|
|
|
|
return bpb;
|
|
}
|
|
|
|
internal static BiosParameterBlock FromBytes(byte[] bytes, int offset)
|
|
{
|
|
BiosParameterBlock bpb = new BiosParameterBlock();
|
|
bpb.OemId = Utilities.BytesToString(bytes, offset + 0x03, 8);
|
|
bpb.BytesPerSector = Utilities.ToUInt16LittleEndian(bytes, offset + 0x0B);
|
|
bpb.SectorsPerCluster = bytes[offset + 0x0D];
|
|
bpb.ReservedSectors = Utilities.ToUInt16LittleEndian(bytes, offset + 0x0E);
|
|
bpb.NumFats = bytes[offset + 0x10];
|
|
bpb.FatRootEntriesCount = Utilities.ToUInt16LittleEndian(bytes, offset + 0x11);
|
|
bpb.TotalSectors16 = Utilities.ToUInt16LittleEndian(bytes, offset + 0x13);
|
|
bpb.Media = bytes[offset + 0x15];
|
|
bpb.FatSize16 = Utilities.ToUInt16LittleEndian(bytes, offset + 0x16);
|
|
bpb.SectorsPerTrack = Utilities.ToUInt16LittleEndian(bytes, offset + 0x18);
|
|
bpb.NumHeads = Utilities.ToUInt16LittleEndian(bytes, offset + 0x1A);
|
|
bpb.HiddenSectors = Utilities.ToUInt32LittleEndian(bytes, offset + 0x1C);
|
|
bpb.TotalSectors32 = Utilities.ToUInt32LittleEndian(bytes, offset + 0x20);
|
|
bpb.BiosDriveNumber = bytes[offset + 0x24];
|
|
bpb.ChkDskFlags = bytes[offset + 0x25];
|
|
bpb.SignatureByte = bytes[offset + 0x26];
|
|
bpb.PaddingByte = bytes[offset + 0x27];
|
|
bpb.TotalSectors64 = Utilities.ToInt64LittleEndian(bytes, offset + 0x28);
|
|
bpb.MftCluster = Utilities.ToInt64LittleEndian(bytes, offset + 0x30);
|
|
bpb.MftMirrorCluster = Utilities.ToInt64LittleEndian(bytes, offset + 0x38);
|
|
bpb.RawMftRecordSize = bytes[offset + 0x40];
|
|
bpb.RawIndexBufferSize = bytes[offset + 0x44];
|
|
bpb.VolumeSerialNumber = Utilities.ToUInt64LittleEndian(bytes, offset + 0x48);
|
|
|
|
return bpb;
|
|
}
|
|
|
|
internal void ToBytes(byte[] buffer, int offset)
|
|
{
|
|
Utilities.StringToBytes(OemId, buffer, offset + 0x03, 8);
|
|
Utilities.WriteBytesLittleEndian(BytesPerSector, buffer, offset + 0x0B);
|
|
buffer[offset + 0x0D] = SectorsPerCluster;
|
|
Utilities.WriteBytesLittleEndian(ReservedSectors, buffer, offset + 0x0E);
|
|
buffer[offset + 0x10] = NumFats;
|
|
Utilities.WriteBytesLittleEndian(FatRootEntriesCount, buffer, offset + 0x11);
|
|
Utilities.WriteBytesLittleEndian(TotalSectors16, buffer, offset + 0x13);
|
|
buffer[offset + 0x15] = Media;
|
|
Utilities.WriteBytesLittleEndian(FatSize16, buffer, offset + 0x16);
|
|
Utilities.WriteBytesLittleEndian(SectorsPerTrack, buffer, offset + 0x18);
|
|
Utilities.WriteBytesLittleEndian(NumHeads, buffer, offset + 0x1A);
|
|
Utilities.WriteBytesLittleEndian(HiddenSectors, buffer, offset + 0x1C);
|
|
Utilities.WriteBytesLittleEndian(TotalSectors32, buffer, offset + 0x20);
|
|
buffer[offset + 0x24] = BiosDriveNumber;
|
|
buffer[offset + 0x25] = ChkDskFlags;
|
|
buffer[offset + 0x26] = SignatureByte;
|
|
buffer[offset + 0x27] = PaddingByte;
|
|
Utilities.WriteBytesLittleEndian(TotalSectors64, buffer, offset + 0x28);
|
|
Utilities.WriteBytesLittleEndian(MftCluster, buffer, offset + 0x30);
|
|
Utilities.WriteBytesLittleEndian(MftMirrorCluster, buffer, offset + 0x38);
|
|
buffer[offset + 0x40] = RawMftRecordSize;
|
|
buffer[offset + 0x44] = RawIndexBufferSize;
|
|
Utilities.WriteBytesLittleEndian(VolumeSerialNumber, buffer, offset + 0x48);
|
|
}
|
|
|
|
internal int CalcRecordSize(byte rawSize)
|
|
{
|
|
if ((rawSize & 0x80) != 0)
|
|
{
|
|
return 1 << (-(sbyte)rawSize);
|
|
}
|
|
else
|
|
{
|
|
return rawSize * SectorsPerCluster * BytesPerSector;
|
|
}
|
|
}
|
|
|
|
private static ulong GenSerialNumber()
|
|
{
|
|
byte[] buffer = new byte[8];
|
|
Random rng = new Random();
|
|
rng.NextBytes(buffer);
|
|
return Utilities.ToUInt64LittleEndian(buffer, 0);
|
|
}
|
|
|
|
private byte CodeRecordSize(int size)
|
|
{
|
|
if (size >= BytesPerCluster)
|
|
{
|
|
return (byte)(size / BytesPerCluster);
|
|
}
|
|
else
|
|
{
|
|
sbyte val = 0;
|
|
while (size != 1)
|
|
{
|
|
size = (size >> 1) & 0x7FFFFFFF;
|
|
val++;
|
|
}
|
|
|
|
return (byte)-val;
|
|
}
|
|
}
|
|
}
|
|
}
|