Files
Aaru/Aaru.Filesystems/UDF.cs

580 lines
23 KiB
C#
Raw Normal View History

// /***************************************************************************
2020-02-27 12:31:25 +00:00
// Aaru Data Preservation Suite
// ----------------------------------------------------------------------------
//
// Filename : UDF.cs
// Author(s) : Natalia Portillo <claunia@claunia.com>
//
// Component : Universal Disk Format plugin.
//
// --[ Description ] ----------------------------------------------------------
//
// Identifies the Universal Disk Format and shows information.
//
// --[ License ] --------------------------------------------------------------
//
// This library is free software; you can redistribute it and/or modify
// it under the terms of the GNU Lesser General Public License as
// published by the Free Software Foundation; either version 2.1 of the
// License, or (at your option) any later version.
//
// This library is distributed in the hope that it will be useful, but
// WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
// Lesser General Public License for more details.
//
// You should have received a copy of the GNU Lesser General Public
// License along with this library; if not, see <http://www.gnu.org/licenses/>.
//
// ----------------------------------------------------------------------------
2022-02-18 10:02:53 +00:00
// Copyright © 2011-2022 Natalia Portillo
// ****************************************************************************/
2022-03-07 07:36:44 +00:00
namespace Aaru.Filesystems;
using System;
2020-07-20 07:47:12 +01:00
using System.Diagnostics.CodeAnalysis;
using System.Linq;
using System.Runtime.InteropServices;
using System.Text;
2020-02-27 00:33:26 +00:00
using Aaru.CommonTypes;
using Aaru.CommonTypes.Enums;
2020-02-27 00:33:26 +00:00
using Aaru.CommonTypes.Interfaces;
using Aaru.Console;
2020-07-20 15:43:52 +01:00
using Aaru.Helpers;
2017-12-21 14:30:38 +00:00
using Schemas;
2020-02-27 00:33:26 +00:00
using Marshal = Aaru.Helpers.Marshal;
2022-03-06 13:29:38 +00:00
// TODO: Detect bootable
/// <inheritdoc />
/// <summary>Implements detection of the Universal Disk Format filesystem</summary>
[SuppressMessage("ReSharper", "UnusedMember.Local")]
public sealed class UDF : IFilesystem
{
2022-03-06 13:29:38 +00:00
readonly byte[] _magic =
{
2022-03-06 13:29:38 +00:00
0x2A, 0x4F, 0x53, 0x54, 0x41, 0x20, 0x55, 0x44, 0x46, 0x20, 0x43, 0x6F, 0x6D, 0x70, 0x6C, 0x69, 0x61, 0x6E,
0x74, 0x00, 0x00, 0x00, 0x00
};
2022-03-06 13:29:38 +00:00
/// <inheritdoc />
public FileSystemType XmlFsType { get; private set; }
/// <inheritdoc />
public Encoding Encoding { get; private set; }
/// <inheritdoc />
public string Name => "Universal Disk Format";
/// <inheritdoc />
public Guid Id => new("83976FEC-A91B-464B-9293-56C719461BAB");
/// <inheritdoc />
public string Author => "Natalia Portillo";
2022-03-06 13:29:38 +00:00
/// <inheritdoc />
public bool Identify(IMediaImage imagePlugin, Partition partition)
{
// UDF needs at least that
if(partition.End - partition.Start < 256)
return false;
// UDF needs at least that
if(imagePlugin.Info.SectorSize < 512)
return false;
2020-02-29 18:03:35 +00:00
2022-03-06 13:29:38 +00:00
byte[] sector;
var anchor = new AnchorVolumeDescriptorPointer();
2022-03-06 13:29:38 +00:00
// All positions where anchor may reside, with the ratio between 512 and 2048bps
ulong[][] positions =
{
new ulong[]
2020-02-29 18:03:35 +00:00
{
2022-03-06 13:29:38 +00:00
256, 1
},
new ulong[]
{
512, 1
},
new ulong[]
{
partition.End - 256, 1
},
new ulong[]
{
partition.End, 1
},
new ulong[]
{
1024, 4
},
new ulong[]
{
2048, 4
},
new ulong[]
{
partition.End - 1024, 4
},
new ulong[]
{
partition.End - 4, 4
}
};
2020-02-29 18:03:35 +00:00
2022-03-07 07:36:44 +00:00
var anchorFound = false;
2022-03-06 13:29:38 +00:00
uint ratio = 1;
2022-11-15 01:35:06 +00:00
sector = null;
2022-11-15 01:35:06 +00:00
foreach(ulong[] position in from position in
positions.Where(position =>
position[0] + partition.Start + position[1] <=
partition.End && position[0] < partition.End) let errno =
imagePlugin.ReadSectors(position[0], (uint)position[1], out sector)
where errno == ErrorNumber.NoError select position)
2022-03-06 13:29:38 +00:00
{
anchor = Marshal.ByteArrayToStructureLittleEndian<AnchorVolumeDescriptorPointer>(sector);
2022-03-06 13:29:38 +00:00
AaruConsole.DebugWriteLine("UDF Plugin", "anchor.tag.tagIdentifier = {0}", anchor.tag.tagIdentifier);
2020-02-29 18:03:35 +00:00
2022-03-06 13:29:38 +00:00
AaruConsole.DebugWriteLine("UDF Plugin", "anchor.tag.descriptorVersion = {0}",
anchor.tag.descriptorVersion);
2020-02-29 18:03:35 +00:00
2022-03-06 13:29:38 +00:00
AaruConsole.DebugWriteLine("UDF Plugin", "anchor.tag.tagChecksum = 0x{0:X2}", anchor.tag.tagChecksum);
AaruConsole.DebugWriteLine("UDF Plugin", "anchor.tag.reserved = {0}", anchor.tag.reserved);
2020-02-29 18:03:35 +00:00
2022-03-07 07:36:44 +00:00
AaruConsole.DebugWriteLine("UDF Plugin", "anchor.tag.tagSerialNumber = {0}", anchor.tag.tagSerialNumber);
2020-02-29 18:03:35 +00:00
2022-03-07 07:36:44 +00:00
AaruConsole.DebugWriteLine("UDF Plugin", "anchor.tag.descriptorCrc = 0x{0:X4}", anchor.tag.descriptorCrc);
2020-02-29 18:03:35 +00:00
2022-03-06 13:29:38 +00:00
AaruConsole.DebugWriteLine("UDF Plugin", "anchor.tag.descriptorCrcLength = {0}",
anchor.tag.descriptorCrcLength);
2020-02-29 18:03:35 +00:00
2022-03-06 13:29:38 +00:00
AaruConsole.DebugWriteLine("UDF Plugin", "anchor.tag.tagLocation = {0}", anchor.tag.tagLocation);
2020-02-29 18:03:35 +00:00
2022-03-06 13:29:38 +00:00
AaruConsole.DebugWriteLine("UDF Plugin", "anchor.mainVolumeDescriptorSequenceExtent.length = {0}",
anchor.mainVolumeDescriptorSequenceExtent.length);
2020-02-29 18:03:35 +00:00
2022-03-06 13:29:38 +00:00
AaruConsole.DebugWriteLine("UDF Plugin", "anchor.mainVolumeDescriptorSequenceExtent.location = {0}",
anchor.mainVolumeDescriptorSequenceExtent.location);
2020-02-29 18:03:35 +00:00
2022-03-06 13:29:38 +00:00
AaruConsole.DebugWriteLine("UDF Plugin", "anchor.reserveVolumeDescriptorSequenceExtent.length = {0}",
anchor.reserveVolumeDescriptorSequenceExtent.length);
2020-02-29 18:03:35 +00:00
2022-03-06 13:29:38 +00:00
AaruConsole.DebugWriteLine("UDF Plugin", "anchor.reserveVolumeDescriptorSequenceExtent.location = {0}",
anchor.reserveVolumeDescriptorSequenceExtent.location);
2022-03-06 13:29:38 +00:00
if(anchor.tag.tagIdentifier != TagIdentifier.AnchorVolumeDescriptorPointer ||
2022-03-07 07:36:44 +00:00
anchor.tag.tagLocation != position[0] / position[1] ||
anchor.mainVolumeDescriptorSequenceExtent.location * position[1] + partition.Start >= partition.End)
2022-03-06 13:29:38 +00:00
continue;
2022-03-06 13:29:38 +00:00
anchorFound = true;
ratio = (uint)position[1];
2020-02-29 18:03:35 +00:00
2022-03-06 13:29:38 +00:00
break;
}
2022-03-06 13:29:38 +00:00
if(!anchorFound)
return false;
ulong count = 0;
2022-03-06 13:29:38 +00:00
while(count < 256)
{
ErrorNumber errno =
imagePlugin.
2022-03-07 07:36:44 +00:00
ReadSectors(partition.Start + anchor.mainVolumeDescriptorSequenceExtent.location * ratio + count * ratio,
2022-03-06 13:29:38 +00:00
ratio, out sector);
2022-03-06 13:29:38 +00:00
if(errno != ErrorNumber.NoError)
{
2022-03-06 13:29:38 +00:00
count++;
2022-03-06 13:29:38 +00:00
continue;
}
2022-03-07 07:36:44 +00:00
var tagId = (TagIdentifier)BitConverter.ToUInt16(sector, 0);
var location = BitConverter.ToUInt32(sector, 0x0C);
2020-02-29 18:03:35 +00:00
2022-03-07 07:36:44 +00:00
if(location == partition.Start / ratio + anchor.mainVolumeDescriptorSequenceExtent.location + count)
2022-03-06 13:29:38 +00:00
{
if(tagId == TagIdentifier.TerminatingDescriptor)
break;
2022-03-06 13:29:38 +00:00
if(tagId == TagIdentifier.LogicalVolumeDescriptor)
{
2022-03-06 13:29:38 +00:00
LogicalVolumeDescriptor lvd =
Marshal.ByteArrayToStructureLittleEndian<LogicalVolumeDescriptor>(sector);
2022-03-06 13:29:38 +00:00
return _magic.SequenceEqual(lvd.domainIdentifier.identifier);
}
}
2022-03-06 13:29:38 +00:00
else
break;
2022-03-06 13:29:38 +00:00
count++;
}
2022-03-06 13:29:38 +00:00
return false;
}
2022-03-06 13:29:38 +00:00
/// <inheritdoc />
2022-03-07 07:36:44 +00:00
public void GetInformation(IMediaImage imagePlugin, Partition partition, out string information, Encoding encoding)
2022-03-06 13:29:38 +00:00
{
information = "";
ErrorNumber errno;
// UDF is always UTF-8
Encoding = Encoding.UTF8;
byte[] sector;
2022-03-06 13:29:38 +00:00
var sbInformation = new StringBuilder();
2022-03-06 13:29:38 +00:00
sbInformation.AppendLine("Universal Disk Format");
2022-03-06 13:29:38 +00:00
var anchor = new AnchorVolumeDescriptorPointer();
2020-02-29 18:03:35 +00:00
2022-03-06 13:29:38 +00:00
// All positions where anchor may reside, with the ratio between 512 and 2048bps
ulong[][] positions =
{
new ulong[]
2020-02-29 18:03:35 +00:00
{
2022-03-06 13:29:38 +00:00
256, 1
},
new ulong[]
{
512, 1
},
new ulong[]
{
partition.End - 256, 1
},
new ulong[]
{
partition.End, 1
},
new ulong[]
{
1024, 4
},
new ulong[]
{
2048, 4
},
new ulong[]
{
partition.End - 1024, 4
},
new ulong[]
{
partition.End - 4, 4
}
};
2022-03-06 13:29:38 +00:00
uint ratio = 1;
2022-03-06 13:29:38 +00:00
foreach(ulong[] position in positions)
{
errno = imagePlugin.ReadSectors(position[0], (uint)position[1], out sector);
2022-03-06 13:29:38 +00:00
if(errno != ErrorNumber.NoError)
continue;
2022-03-06 13:29:38 +00:00
anchor = Marshal.ByteArrayToStructureLittleEndian<AnchorVolumeDescriptorPointer>(sector);
2022-03-06 13:29:38 +00:00
if(anchor.tag.tagIdentifier != TagIdentifier.AnchorVolumeDescriptorPointer ||
anchor.tag.tagLocation != position[0] / position[1] ||
anchor.mainVolumeDescriptorSequenceExtent.location + partition.Start >= partition.End)
continue;
2022-03-06 13:29:38 +00:00
ratio = (uint)position[1];
2022-03-06 13:29:38 +00:00
break;
}
2022-03-06 13:29:38 +00:00
ulong count = 0;
var pvd = new PrimaryVolumeDescriptor();
var lvd = new LogicalVolumeDescriptor();
var lvidiu = new LogicalVolumeIntegrityDescriptorImplementationUse();
2022-03-06 13:29:38 +00:00
while(count < 256)
{
errno =
imagePlugin.
2022-03-07 07:36:44 +00:00
ReadSectors(partition.Start + anchor.mainVolumeDescriptorSequenceExtent.location * ratio + count * ratio,
2022-03-06 13:29:38 +00:00
ratio, out sector);
2022-03-06 13:29:38 +00:00
if(errno != ErrorNumber.NoError)
continue;
2020-02-29 18:03:35 +00:00
2022-03-07 07:36:44 +00:00
var tagId = (TagIdentifier)BitConverter.ToUInt16(sector, 0);
var location = BitConverter.ToUInt32(sector, 0x0C);
2022-03-07 07:36:44 +00:00
if(location == partition.Start / ratio + anchor.mainVolumeDescriptorSequenceExtent.location + count)
2022-03-06 13:29:38 +00:00
{
if(tagId == TagIdentifier.TerminatingDescriptor)
break;
switch(tagId)
{
2022-03-06 13:29:38 +00:00
case TagIdentifier.LogicalVolumeDescriptor:
lvd = Marshal.ByteArrayToStructureLittleEndian<LogicalVolumeDescriptor>(sector);
2020-02-29 18:03:35 +00:00
break;
2022-03-06 13:29:38 +00:00
case TagIdentifier.PrimaryVolumeDescriptor:
pvd = Marshal.ByteArrayToStructureLittleEndian<PrimaryVolumeDescriptor>(sector);
2022-03-06 13:29:38 +00:00
break;
}
}
else
break;
2020-02-29 18:03:35 +00:00
2022-03-06 13:29:38 +00:00
count++;
}
2020-02-29 18:03:35 +00:00
2022-03-06 13:29:38 +00:00
errno = imagePlugin.ReadSectors(lvd.integritySequenceExtent.location * ratio, ratio, out sector);
2022-03-06 13:29:38 +00:00
if(errno != ErrorNumber.NoError)
return;
LogicalVolumeIntegrityDescriptor lvid =
Marshal.ByteArrayToStructureLittleEndian<LogicalVolumeIntegrityDescriptor>(sector);
2022-03-06 13:29:38 +00:00
if(lvid.tag.tagIdentifier == TagIdentifier.LogicalVolumeIntegrityDescriptor &&
lvid.tag.tagLocation == lvd.integritySequenceExtent.location)
lvidiu =
Marshal.ByteArrayToStructureLittleEndian<LogicalVolumeIntegrityDescriptorImplementationUse>(sector,
2022-03-07 07:36:44 +00:00
(int)(lvid.numberOfPartitions * 8 + 80),
2022-03-06 13:29:38 +00:00
System.Runtime.InteropServices.Marshal.SizeOf(lvidiu));
else
lvid = new LogicalVolumeIntegrityDescriptor();
2022-03-06 13:29:38 +00:00
sbInformation.AppendFormat("Volume is number {0} of {1}", pvd.volumeSequenceNumber,
pvd.maximumVolumeSequenceNumber).AppendLine();
2022-03-06 13:29:38 +00:00
sbInformation.AppendFormat("Volume set identifier: {0}",
StringHandlers.DecompressUnicode(pvd.volumeSetIdentifier)).AppendLine();
2017-12-19 20:33:03 +00:00
2022-03-07 07:36:44 +00:00
sbInformation.AppendFormat("Volume name: {0}", StringHandlers.DecompressUnicode(lvd.logicalVolumeIdentifier)).
AppendLine();
2020-02-29 18:03:35 +00:00
2022-03-06 13:29:38 +00:00
sbInformation.AppendFormat("Volume uses {0} bytes per block", lvd.logicalBlockSize).AppendLine();
2020-02-29 18:03:35 +00:00
2022-03-06 13:29:38 +00:00
sbInformation.AppendFormat("Volume was last written in {0}", EcmaToDateTime(lvid.recordingDateTime)).
AppendLine();
2020-02-29 18:03:35 +00:00
2022-03-06 13:29:38 +00:00
sbInformation.AppendFormat("Volume contains {0} partitions", lvid.numberOfPartitions).AppendLine();
2020-02-29 18:03:35 +00:00
2022-03-07 07:36:44 +00:00
sbInformation.AppendFormat("Volume contains {0} files and {1} directories", lvidiu.files, lvidiu.directories).
AppendLine();
2020-02-29 18:03:35 +00:00
2022-03-06 13:29:38 +00:00
sbInformation.AppendFormat("Volume conforms to {0}",
2022-03-07 07:36:44 +00:00
Encoding.GetString(lvd.domainIdentifier.identifier).TrimEnd('\u0000')).AppendLine();
2020-02-29 18:03:35 +00:00
2022-03-06 13:29:38 +00:00
sbInformation.AppendFormat("Volume was last written by: {0}",
Encoding.GetString(pvd.implementationIdentifier.identifier).TrimEnd('\u0000')).
AppendLine();
2020-02-29 18:03:35 +00:00
2022-03-06 13:29:38 +00:00
sbInformation.AppendFormat("Volume requires UDF version {0}.{1:X2} to be read",
Convert.ToInt32($"{(lvidiu.minimumReadUDF & 0xFF00) >> 8}", 10),
Convert.ToInt32($"{lvidiu.minimumReadUDF & 0xFF}", 10)).AppendLine();
2020-02-29 18:03:35 +00:00
2022-03-06 13:29:38 +00:00
sbInformation.AppendFormat("Volume requires UDF version {0}.{1:X2} to be written to",
Convert.ToInt32($"{(lvidiu.minimumWriteUDF & 0xFF00) >> 8}", 10),
Convert.ToInt32($"{lvidiu.minimumWriteUDF & 0xFF}", 10)).AppendLine();
2020-02-29 18:03:35 +00:00
2022-03-06 13:29:38 +00:00
sbInformation.AppendFormat("Volume cannot be written by any UDF version higher than {0}.{1:X2}",
Convert.ToInt32($"{(lvidiu.maximumWriteUDF & 0xFF00) >> 8}", 10),
Convert.ToInt32($"{lvidiu.maximumWriteUDF & 0xFF}", 10)).AppendLine();
2020-02-29 18:03:35 +00:00
2022-03-06 13:29:38 +00:00
XmlFsType = new FileSystemType
{
Type =
$"UDF v{Convert.ToInt32($"{(lvidiu.maximumWriteUDF & 0xFF00) >> 8}", 10)}.{Convert.ToInt32($"{lvidiu.maximumWriteUDF & 0xFF}", 10):X2}",
ApplicationIdentifier = Encoding.GetString(pvd.implementationIdentifier.identifier).TrimEnd('\u0000'),
ClusterSize = lvd.logicalBlockSize,
ModificationDate = EcmaToDateTime(lvid.recordingDateTime),
ModificationDateSpecified = true,
Files = lvidiu.files,
FilesSpecified = true,
VolumeName = StringHandlers.DecompressUnicode(lvd.logicalVolumeIdentifier),
VolumeSetIdentifier = StringHandlers.DecompressUnicode(pvd.volumeSetIdentifier),
VolumeSerial = StringHandlers.DecompressUnicode(pvd.volumeSetIdentifier),
SystemIdentifier = Encoding.GetString(pvd.implementationIdentifier.identifier).TrimEnd('\u0000')
};
2020-02-29 18:03:35 +00:00
2022-03-06 13:29:38 +00:00
XmlFsType.Clusters = (partition.End - partition.Start + 1) * imagePlugin.Info.SectorSize /
XmlFsType.ClusterSize;
2022-03-06 13:29:38 +00:00
information = sbInformation.ToString();
}
2022-03-06 13:29:38 +00:00
static DateTime EcmaToDateTime(Timestamp timestamp) => DateHandlers.EcmaToDateTime(timestamp.typeAndZone,
timestamp.year, timestamp.month, timestamp.day, timestamp.hour, timestamp.minute, timestamp.second,
timestamp.centiseconds, timestamp.hundredsMicroseconds, timestamp.microseconds);
2022-03-06 13:29:38 +00:00
[Flags]
enum EntityFlags : byte
{
2022-03-07 07:36:44 +00:00
Dirty = 0x01,
Protected = 0x02
2022-03-06 13:29:38 +00:00
}
2022-03-06 13:29:38 +00:00
[StructLayout(LayoutKind.Sequential, Pack = 1)]
readonly struct EntityIdentifier
{
/// <summary>Entity flags</summary>
public readonly EntityFlags flags;
/// <summary>Structure identifier</summary>
[MarshalAs(UnmanagedType.ByValArray, SizeConst = 23)]
public readonly byte[] identifier;
/// <summary>Structure data</summary>
[MarshalAs(UnmanagedType.ByValArray, SizeConst = 8)]
public readonly byte[] identifierSuffix;
}
2022-03-06 13:29:38 +00:00
[StructLayout(LayoutKind.Sequential, Pack = 1)]
readonly struct Timestamp
{
public readonly ushort typeAndZone;
public readonly short year;
public readonly byte month;
public readonly byte day;
public readonly byte hour;
public readonly byte minute;
public readonly byte second;
public readonly byte centiseconds;
public readonly byte hundredsMicroseconds;
public readonly byte microseconds;
}
2022-03-06 13:29:38 +00:00
enum TagIdentifier : ushort
{
2022-03-07 07:36:44 +00:00
PrimaryVolumeDescriptor = 1,
AnchorVolumeDescriptorPointer = 2,
VolumeDescriptorPointer = 3,
ImplementationUseVolumeDescriptor = 4,
PartitionDescriptor = 5,
LogicalVolumeDescriptor = 6,
UnallocatedSpaceDescriptor = 7,
TerminatingDescriptor = 8,
LogicalVolumeIntegrityDescriptor = 9
2022-03-06 13:29:38 +00:00
}
2022-03-06 13:29:38 +00:00
[StructLayout(LayoutKind.Sequential, Pack = 1)]
readonly struct DescriptorTag
{
public readonly TagIdentifier tagIdentifier;
public readonly ushort descriptorVersion;
public readonly byte tagChecksum;
public readonly byte reserved;
public readonly ushort tagSerialNumber;
public readonly ushort descriptorCrc;
public readonly ushort descriptorCrcLength;
public readonly uint tagLocation;
}
2022-03-06 13:29:38 +00:00
[StructLayout(LayoutKind.Sequential, Pack = 1)]
readonly struct ExtentDescriptor
{
public readonly uint length;
public readonly uint location;
}
2022-03-06 13:29:38 +00:00
[StructLayout(LayoutKind.Sequential, Pack = 1)]
readonly struct CharacterSpecification
{
public readonly byte type;
[MarshalAs(UnmanagedType.ByValArray, SizeConst = 63)]
public readonly byte[] information;
}
2022-03-06 13:29:38 +00:00
[StructLayout(LayoutKind.Sequential, Pack = 1)]
readonly struct AnchorVolumeDescriptorPointer
{
public readonly DescriptorTag tag;
public readonly ExtentDescriptor mainVolumeDescriptorSequenceExtent;
public readonly ExtentDescriptor reserveVolumeDescriptorSequenceExtent;
[MarshalAs(UnmanagedType.ByValArray, SizeConst = 480)]
public readonly byte[] reserved;
}
2022-03-06 13:29:38 +00:00
[StructLayout(LayoutKind.Sequential, Pack = 1)]
readonly struct PrimaryVolumeDescriptor
{
public readonly DescriptorTag tag;
public readonly uint volumeDescriptorSequenceNumber;
public readonly uint primaryVolumeDescriptorNumber;
[MarshalAs(UnmanagedType.ByValArray, SizeConst = 32)]
public readonly byte[] volumeIdentifier;
public readonly ushort volumeSequenceNumber;
public readonly ushort maximumVolumeSequenceNumber;
public readonly ushort interchangeLevel;
public readonly ushort maximumInterchangeLevel;
public readonly uint characterSetList;
public readonly uint maximumCharacterSetList;
[MarshalAs(UnmanagedType.ByValArray, SizeConst = 128)]
public readonly byte[] volumeSetIdentifier;
public readonly CharacterSpecification descriptorCharacterSet;
public readonly CharacterSpecification explanatoryCharacterSet;
public readonly ExtentDescriptor volumeAbstract;
public readonly ExtentDescriptor volumeCopyright;
public readonly EntityIdentifier applicationIdentifier;
public readonly Timestamp recordingDateTime;
public readonly EntityIdentifier implementationIdentifier;
[MarshalAs(UnmanagedType.ByValArray, SizeConst = 64)]
public readonly byte[] implementationUse;
public readonly uint predecessorVolumeDescriptorSequenceLocation;
public readonly ushort flags;
[MarshalAs(UnmanagedType.ByValArray, SizeConst = 22)]
public readonly byte[] reserved;
}
2022-03-06 13:29:38 +00:00
[StructLayout(LayoutKind.Sequential, Pack = 1)]
readonly struct LogicalVolumeDescriptor
{
public readonly DescriptorTag tag;
public readonly uint volumeDescriptorSequenceNumber;
public readonly CharacterSpecification descriptorCharacterSet;
[MarshalAs(UnmanagedType.ByValArray, SizeConst = 128)]
public readonly byte[] logicalVolumeIdentifier;
public readonly uint logicalBlockSize;
public readonly EntityIdentifier domainIdentifier;
[MarshalAs(UnmanagedType.ByValArray, SizeConst = 16)]
public readonly byte[] logicalVolumeContentsUse;
public readonly uint mapTableLength;
public readonly uint numberOfPartitionMaps;
public readonly EntityIdentifier implementationIdentifier;
[MarshalAs(UnmanagedType.ByValArray, SizeConst = 128)]
public readonly byte[] implementationUse;
public readonly ExtentDescriptor integritySequenceExtent;
}
2022-03-06 13:29:38 +00:00
[StructLayout(LayoutKind.Sequential, Pack = 1)]
readonly struct LogicalVolumeIntegrityDescriptor
{
public readonly DescriptorTag tag;
public readonly Timestamp recordingDateTime;
public readonly uint integrityType;
public readonly ExtentDescriptor nextIntegrityExtent;
[MarshalAs(UnmanagedType.ByValArray, SizeConst = 32)]
public readonly byte[] logicalVolumeContentsUse;
public readonly uint numberOfPartitions;
public readonly uint lengthOfImplementationUse;
// Follows uint[numberOfPartitions] freeSpaceTable;
// Follows uint[numberOfPartitions] sizeTable;
// Follows byte[lengthOfImplementationUse] implementationUse;
}
2022-03-06 13:29:38 +00:00
[StructLayout(LayoutKind.Sequential, Pack = 1)]
readonly struct LogicalVolumeIntegrityDescriptorImplementationUse
{
public readonly EntityIdentifier implementationId;
public readonly uint files;
public readonly uint directories;
public readonly ushort minimumReadUDF;
public readonly ushort minimumWriteUDF;
public readonly ushort maximumWriteUDF;
}
}