// /*************************************************************************** // Aaru Data Preservation Suite // ---------------------------------------------------------------------------- // // Filename : Sun.cs // Author(s) : Natalia Portillo // // Component : Partitioning scheme plugins. // // --[ Description ] ---------------------------------------------------------- // // Manages Sun disklabels. // // --[ 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 . // // ---------------------------------------------------------------------------- // Copyright © 2011-2025 Natalia Portillo // ****************************************************************************/ using System; using System.Collections.Generic; using System.Diagnostics.CodeAnalysis; using System.Runtime.InteropServices; using System.Text; using Aaru.CommonTypes; using Aaru.CommonTypes.Attributes; using Aaru.CommonTypes.Enums; using Aaru.CommonTypes.Interfaces; using Aaru.Helpers; using Aaru.Logging; using Marshal = Aaru.Helpers.Marshal; namespace Aaru.Partitions; /// /// Implements decoding of Sun disklabels [SuppressMessage("ReSharper", "InconsistentNaming")] public sealed partial class SunDisklabel : IPartition { /// Sun disklabel magic number const ushort DKL_MAGIC = 0xDABE; /// Sun VTOC magic number const uint VTOC_SANE = 0x600DDEEE; /// Sun disklabel magic number, byte-swapped const ushort DKL_CIGAM = 0xBEDA; /// Sun VTOC magic number, byte-swapped const uint VTOC_ENAS = 0xEEDE0D60; /// # of logical partitions const int NDKMAP = 8; /// # of logical partitions const int NDKMAP16 = 16; /// Disk label size const int DK_LABEL_SIZE = 512; /// Volume label size const int LEN_DKL_ASCII = 128; /// Length of v_volume const int LEN_DKL_VVOL = 8; /// Size of padding in SunOS disk label const int LEN_DKL_PAD = DK_LABEL_SIZE - (LEN_DKL_ASCII + NDKMAP * 8 + 14 * 2); /// Size of padding in Solaris disk label with 8 partitions const int LEN_DKL_PAD8 = DK_LABEL_SIZE - (LEN_DKL_ASCII + 136 + // sizeof(dk_vtoc8) NDKMAP * 8 + 14 * 2 + 2 * 2); const int LEN_DKL_PAD16 = DK_LABEL_SIZE - (456 + // sizeof(dk_vtoc16) 4 * 4 + 12 * 2 + 2 * 2); const string MODULE_NAME = "Sun partitions plugin"; #region IPartition Members /// public string Name => Localization.SunDisklabel_Name; /// public Guid Id => new("50F35CC4-8375-4445-8DCB-1BA550C931A3"); /// public string Author => Authors.NATALIA_PORTILLO; /// public bool GetInformation(IMediaImage imagePlugin, out List partitions, ulong sectorOffset) { partitions = []; if(imagePlugin.Info.SectorSize < 512) return false; if(sectorOffset + 2 >= imagePlugin.Info.Sectors) return false; bool useDkl = false, useDkl8 = false, useDkl16 = false; ErrorNumber errno = imagePlugin.ReadSector(sectorOffset, false, out byte[] sunSector, out _); if(errno != ErrorNumber.NoError) return false; dk_label dkl = Marshal.ByteArrayToStructureLittleEndian(sunSector); dk_label8 dkl8 = Marshal.ByteArrayToStructureLittleEndian(sunSector); dk_label16 dkl16 = Marshal.ByteArrayToStructureLittleEndian(sunSector); AaruLogging.Debug(MODULE_NAME, "dkl.dkl_magic = 0x{0:X4}", dkl.dkl_magic); AaruLogging.Debug(MODULE_NAME, "dkl8.dkl_vtoc.v_sanity = 0x{0:X8}", dkl8.dkl_vtoc.v_sanity); AaruLogging.Debug(MODULE_NAME, "dkl16.dkl_vtoc.v_sanity = 0x{0:X8}", dkl16.dkl_vtoc.v_sanity); if(dkl.dkl_magic is DKL_MAGIC or DKL_CIGAM) { if(dkl16.dkl_vtoc.v_sanity is VTOC_SANE or VTOC_ENAS) useDkl16 = true; else if(dkl8.dkl_vtoc.v_sanity is VTOC_SANE or VTOC_ENAS) useDkl8 = true; else useDkl = true; } if(!useDkl && !useDkl8 && !useDkl16) { errno = imagePlugin.ReadSector(sectorOffset + 1, false, out sunSector, out _); if(errno == ErrorNumber.NoError) { dkl = Marshal.ByteArrayToStructureLittleEndian(sunSector); dkl8 = Marshal.ByteArrayToStructureLittleEndian(sunSector); dkl16 = Marshal.ByteArrayToStructureLittleEndian(sunSector); if(dkl.dkl_magic is DKL_MAGIC or DKL_CIGAM) { if(dkl16.dkl_vtoc.v_sanity is VTOC_SANE or VTOC_ENAS) useDkl16 = true; else if(dkl8.dkl_vtoc.v_sanity is VTOC_SANE or VTOC_ENAS) useDkl8 = true; else useDkl = true; } } } if(!useDkl && !useDkl8 && !useDkl16) return false; if(useDkl16 && dkl16.dkl_magic == DKL_CIGAM) { AaruLogging.Debug(MODULE_NAME, Localization.Swapping_dk_label16); dkl16 = dkl16.SwapEndian(); } else if(useDkl8 && dkl8.dkl_magic == DKL_CIGAM) { AaruLogging.Debug(MODULE_NAME, Localization.Swapping_dk_label8); dkl8 = dkl8.SwapEndian(); } else if(useDkl && dkl.dkl_magic == DKL_CIGAM) { AaruLogging.Debug(MODULE_NAME, Localization.Swapping_dk_label); dkl = dkl.SwapEndian(); } if(useDkl) { var sectorsPerCylinder = (ulong)(dkl.dkl_nsect * dkl.dkl_nhead); AaruLogging.Debug(MODULE_NAME, "dkl.dkl_asciilabel = \"{0}\"", StringHandlers.CToString(dkl.dkl_asciilabel)); AaruLogging.Debug(MODULE_NAME, "dkl.dkl_rpm = {0}", dkl.dkl_rpm); AaruLogging.Debug(MODULE_NAME, "dkl.dkl_pcyl = {0}", dkl.dkl_pcyl); AaruLogging.Debug(MODULE_NAME, "dkl.dkl_apc = {0}", dkl.dkl_apc); AaruLogging.Debug(MODULE_NAME, "dkl.dkl_gap1 = {0}", dkl.dkl_gap1); AaruLogging.Debug(MODULE_NAME, "dkl.dkl_gap2 = {0}", dkl.dkl_gap2); AaruLogging.Debug(MODULE_NAME, "dkl.dkl_intrlv = {0}", dkl.dkl_intrlv); AaruLogging.Debug(MODULE_NAME, "dkl.dkl_ncyl = {0}", dkl.dkl_ncyl); AaruLogging.Debug(MODULE_NAME, "dkl.dkl_acyl = {0}", dkl.dkl_acyl); AaruLogging.Debug(MODULE_NAME, "dkl.dkl_nhead = {0}", dkl.dkl_nhead); AaruLogging.Debug(MODULE_NAME, "dkl.dkl_nsect = {0}", dkl.dkl_nsect); AaruLogging.Debug(MODULE_NAME, "dkl.dkl_bhead = {0}", dkl.dkl_bhead); AaruLogging.Debug(MODULE_NAME, "dkl.dkl_ppart = {0}", dkl.dkl_ppart); for(var i = 0; i < NDKMAP; i++) { AaruLogging.Debug(MODULE_NAME, "dkl.dkl_map[{0}].dkl_cylno = {1}", i, dkl.dkl_map[i].dkl_cylno); AaruLogging.Debug(MODULE_NAME, "dkl.dkl_map[{0}].dkl_nblk = {1}", i, dkl.dkl_map[i].dkl_nblk); } AaruLogging.Debug(MODULE_NAME, "dkl.dkl_magic = 0x{0:X4}", dkl.dkl_magic); AaruLogging.Debug(MODULE_NAME, "dkl.dkl_cksum = 0x{0:X4}", dkl.dkl_cksum); AaruLogging.Debug(MODULE_NAME, "sectorsPerCylinder = {0}", sectorsPerCylinder); for(var i = 0; i < NDKMAP; i++) { if(dkl.dkl_map[i].dkl_cylno <= 0 || dkl.dkl_map[i].dkl_nblk <= 0) continue; var part = new Partition { Size = (ulong)dkl.dkl_map[i].dkl_nblk * DK_LABEL_SIZE, Length = (ulong)(dkl.dkl_map[i].dkl_nblk * DK_LABEL_SIZE / imagePlugin.Info.SectorSize), Sequence = (ulong)i, Offset = ((ulong)dkl.dkl_map[i].dkl_cylno * sectorsPerCylinder + sectorOffset) * DK_LABEL_SIZE, Start = ((ulong)dkl.dkl_map[i].dkl_cylno * sectorsPerCylinder + sectorOffset) * DK_LABEL_SIZE / imagePlugin.Info.SectorSize, Type = Localization.SunOS_partition, Scheme = Name }; if(part.Start < imagePlugin.Info.Sectors && part.End <= imagePlugin.Info.Sectors) partitions.Add(part); } } else if(useDkl8) { var sectorsPerCylinder = (ulong)(dkl8.dkl_nsect * dkl8.dkl_nhead); AaruLogging.Debug(MODULE_NAME, "dkl8.dkl_asciilabel = \"{0}\"", StringHandlers.CToString(dkl8.dkl_asciilabel)); AaruLogging.Debug(MODULE_NAME, "dkl8.dkl_vtoc.v_version = {0}", dkl8.dkl_vtoc.v_version); AaruLogging.Debug(MODULE_NAME, "dkl8.dkl_vtoc.v_volume = \"{0}\"", StringHandlers.CToString(dkl8.dkl_vtoc.v_volume)); AaruLogging.Debug(MODULE_NAME, "dkl8.dkl_vtoc.v_nparts = {0}", dkl8.dkl_vtoc.v_nparts); AaruLogging.Debug(MODULE_NAME, "dkl8.dkl_vtoc.v_sanity = 0x{0:X8}", dkl8.dkl_vtoc.v_sanity); AaruLogging.Debug(MODULE_NAME, "dkl8.dkl_write_reinstruct = {0}", dkl8.dkl_write_reinstruct); AaruLogging.Debug(MODULE_NAME, "dkl8.dkl_read_reinstruct = {0}", dkl8.dkl_read_reinstruct); AaruLogging.Debug(MODULE_NAME, "dkl8.dkl_rpm = {0}", dkl8.dkl_rpm); AaruLogging.Debug(MODULE_NAME, "dkl8.dkl_pcyl = {0}", dkl8.dkl_pcyl); AaruLogging.Debug(MODULE_NAME, "dkl8.dkl_apc = {0}", dkl8.dkl_apc); AaruLogging.Debug(MODULE_NAME, "dkl8.dkl_obs1 = {0}", dkl8.dkl_obs1); AaruLogging.Debug(MODULE_NAME, "dkl8.dkl_obs2 = {0}", dkl8.dkl_obs2); AaruLogging.Debug(MODULE_NAME, "dkl8.dkl_intrlv = {0}", dkl8.dkl_intrlv); AaruLogging.Debug(MODULE_NAME, "dkl8.dkl_ncyl = {0}", dkl8.dkl_ncyl); AaruLogging.Debug(MODULE_NAME, "dkl8.dkl_acyl = {0}", dkl8.dkl_acyl); AaruLogging.Debug(MODULE_NAME, "dkl8.dkl_nhead = {0}", dkl8.dkl_nhead); AaruLogging.Debug(MODULE_NAME, "dkl8.dkl_nsect = {0}", dkl8.dkl_nsect); AaruLogging.Debug(MODULE_NAME, "dkl8.dkl_obs3 = {0}", dkl8.dkl_obs3); AaruLogging.Debug(MODULE_NAME, "dkl8.dkl_obs4 = {0}", dkl8.dkl_obs4); for(var i = 0; i < NDKMAP; i++) { AaruLogging.Debug(MODULE_NAME, "dkl8.dkl_map[{0}].dkl_cylno = {1}", i, dkl8.dkl_map[i].dkl_cylno); AaruLogging.Debug(MODULE_NAME, "dkl8.dkl_map[{0}].dkl_nblk = {1}", i, dkl8.dkl_map[i].dkl_nblk); AaruLogging.Debug(MODULE_NAME, "dkl8.dkl_vtoc.v_part[{0}].p_tag = {1} ({2})", i, dkl8.dkl_vtoc.v_part[i].p_tag, (ushort)dkl8.dkl_vtoc.v_part[i].p_tag); AaruLogging.Debug(MODULE_NAME, "dkl8.dkl_vtoc.v_part[{0}].p_flag = {1} ({2})", i, dkl8.dkl_vtoc.v_part[i].p_flag, (ushort)dkl8.dkl_vtoc.v_part[i].p_flag); AaruLogging.Debug(MODULE_NAME, "dkl8.dkl_vtoc.v_timestamp[{0}] = {1}", i, DateHandlers.UnixToDateTime(dkl8.dkl_vtoc.v_timestamp[i])); } AaruLogging.Debug(MODULE_NAME, "dkl8.dkl_magic = 0x{0:X4}", dkl8.dkl_magic); AaruLogging.Debug(MODULE_NAME, "dkl8.dkl_cksum = 0x{0:X4}", dkl8.dkl_cksum); AaruLogging.Debug(MODULE_NAME, "sectorsPerCylinder = {0}", sectorsPerCylinder); if(dkl8.dkl_vtoc.v_nparts > NDKMAP) return false; for(var i = 0; i < dkl8.dkl_vtoc.v_nparts; i++) { if(dkl8.dkl_map[i].dkl_nblk <= 0 || dkl8.dkl_vtoc.v_part[i].p_tag == SunTag.SunEmpty || dkl8.dkl_vtoc.v_part[i].p_tag == SunTag.SunWholeDisk) continue; var part = new Partition { Description = SunFlagsToString(dkl8.dkl_vtoc.v_part[i].p_flag), Size = (ulong)dkl8.dkl_map[i].dkl_nblk * DK_LABEL_SIZE, Length = (ulong)(dkl8.dkl_map[i].dkl_nblk * DK_LABEL_SIZE / imagePlugin.Info.SectorSize), Sequence = (ulong)i, Offset = ((ulong)dkl8.dkl_map[i].dkl_cylno * sectorsPerCylinder + sectorOffset) * DK_LABEL_SIZE, Start = ((ulong)dkl8.dkl_map[i].dkl_cylno * sectorsPerCylinder + sectorOffset) * DK_LABEL_SIZE / imagePlugin.Info.SectorSize, Type = SunIdToString(dkl8.dkl_vtoc.v_part[i].p_tag), Scheme = Name }; if(dkl8.dkl_vtoc.v_timestamp[i] != 0) { part.Description += "\n" + string.Format(Localization.Partition_timestamped_on_0, DateHandlers.UnixToDateTime(dkl8.dkl_vtoc.v_timestamp[i])); } if(part.Start < imagePlugin.Info.Sectors && part.End <= imagePlugin.Info.Sectors) partitions.Add(part); } } else { AaruLogging.Debug(MODULE_NAME, "dkl16.dkl_vtoc.v_sanity = 0x{0:X8}", dkl16.dkl_vtoc.v_sanity); AaruLogging.Debug(MODULE_NAME, "dkl16.dkl_vtoc.v_version = {0}", dkl16.dkl_vtoc.v_version); AaruLogging.Debug(MODULE_NAME, "dkl16.dkl_vtoc.v_volume = \"{0}\"", StringHandlers.CToString(dkl16.dkl_vtoc.v_volume)); AaruLogging.Debug(MODULE_NAME, "dkl16.dkl_vtoc.v_sectorsz = {0}", dkl16.dkl_vtoc.v_sectorsz); AaruLogging.Debug(MODULE_NAME, "dkl16.dkl_vtoc.v_nparts = {0}", dkl16.dkl_vtoc.v_nparts); AaruLogging.Debug(MODULE_NAME, "dkl16.dkl_vtoc.v_asciilabel = \"{0}\"", StringHandlers.CToString(dkl16.dkl_vtoc.v_asciilabel)); AaruLogging.Debug(MODULE_NAME, "dkl16.dkl_pcyl = {0}", dkl16.dkl_pcyl); AaruLogging.Debug(MODULE_NAME, "dkl16.dkl_ncyl = {0}", dkl16.dkl_ncyl); AaruLogging.Debug(MODULE_NAME, "dkl16.dkl_acyl = {0}", dkl16.dkl_acyl); AaruLogging.Debug(MODULE_NAME, "dkl16.dkl_bcyl = {0}", dkl16.dkl_bcyl); AaruLogging.Debug(MODULE_NAME, "dkl16.dkl_nhead = {0}", dkl16.dkl_nhead); AaruLogging.Debug(MODULE_NAME, "dkl16.dkl_nsect = {0}", dkl16.dkl_nsect); AaruLogging.Debug(MODULE_NAME, "dkl16.dkl_intrlv = {0}", dkl16.dkl_intrlv); AaruLogging.Debug(MODULE_NAME, "dkl16.dkl_skew = {0}", dkl16.dkl_skew); AaruLogging.Debug(MODULE_NAME, "dkl16.dkl_apc = {0}", dkl16.dkl_apc); AaruLogging.Debug(MODULE_NAME, "dkl16.dkl_rpm = {0}", dkl16.dkl_rpm); AaruLogging.Debug(MODULE_NAME, "dkl16.dkl_write_reinstruct = {0}", dkl16.dkl_write_reinstruct); AaruLogging.Debug(MODULE_NAME, "dkl16.dkl_read_reinstruct = {0}", dkl16.dkl_read_reinstruct); for(var i = 0; i < NDKMAP16; i++) { AaruLogging.Debug(MODULE_NAME, "dkl16.dkl_vtoc.v_part[{0}].p_start = {1}", i, dkl16.dkl_vtoc.v_part[i].p_start); AaruLogging.Debug(MODULE_NAME, "dkl16.dkl_vtoc.v_part[{0}].p_size = {1}", i, dkl16.dkl_vtoc.v_part[i].p_size); AaruLogging.Debug(MODULE_NAME, "dkl16.dkl_vtoc.v_part[{0}].p_tag = {1} ({2})", i, dkl16.dkl_vtoc.v_part[i].p_tag, (ushort)dkl16.dkl_vtoc.v_part[i].p_tag); AaruLogging.Debug(MODULE_NAME, "dkl16.dkl_vtoc.v_part[{0}].p_flag = {1} ({2})", i, dkl16.dkl_vtoc.v_part[i].p_flag, (ushort)dkl16.dkl_vtoc.v_part[i].p_flag); AaruLogging.Debug(MODULE_NAME, "dkl16.dkl_vtoc.v_timestamp[{0}] = {1}", i, DateHandlers.UnixToDateTime(dkl16.dkl_vtoc.v_timestamp[i])); } AaruLogging.Debug(MODULE_NAME, "dkl16.dkl_magic = 0x{0:X4}", dkl16.dkl_magic); AaruLogging.Debug(MODULE_NAME, "dkl16.dkl_cksum = 0x{0:X4}", dkl16.dkl_cksum); if(dkl16.dkl_vtoc.v_nparts > NDKMAP16) return false; for(var i = 0; i < dkl16.dkl_vtoc.v_nparts; i++) { if(dkl16.dkl_vtoc.v_part[i].p_size <= 0 || dkl16.dkl_vtoc.v_part[i].p_tag == SunTag.SunEmpty || dkl16.dkl_vtoc.v_part[i].p_tag == SunTag.SunWholeDisk) continue; var part = new Partition { Description = SunFlagsToString(dkl16.dkl_vtoc.v_part[i].p_flag), Size = (ulong)dkl16.dkl_vtoc.v_part[i].p_size * dkl16.dkl_vtoc.v_sectorsz, Length = (ulong)(dkl16.dkl_vtoc.v_part[i].p_size * dkl16.dkl_vtoc.v_sectorsz / imagePlugin.Info.SectorSize), Sequence = (ulong)i, Offset = ((ulong)dkl16.dkl_vtoc.v_part[i].p_start + sectorOffset) * dkl16.dkl_vtoc.v_sectorsz, Start = ((ulong)dkl16.dkl_vtoc.v_part[i].p_start + sectorOffset) * dkl16.dkl_vtoc.v_sectorsz / imagePlugin.Info.SectorSize, Type = SunIdToString(dkl16.dkl_vtoc.v_part[i].p_tag), Scheme = Name }; if(dkl16.dkl_vtoc.v_timestamp[i] != 0) { part.Description += "\n" + string.Format(Localization.Partition_timestamped_on_0, DateHandlers.UnixToDateTime(dkl16.dkl_vtoc.v_timestamp[i])); } if(part.Start < imagePlugin.Info.Sectors && part.End <= imagePlugin.Info.Sectors) partitions.Add(part); } } return partitions.Count > 0; } #endregion static string SunFlagsToString(SunFlags flags) { var sb = new StringBuilder(); if(flags.HasFlag(SunFlags.NoMount)) sb.AppendLine(Localization.Unmountable); if(flags.HasFlag(SunFlags.ReadOnly)) sb.AppendLine(Localization.Read_only); return sb.ToString(); } static string SunIdToString(SunTag id) => id switch { SunTag.Linux => Localization.Linux, SunTag.LinuxRaid => Localization.Linux_RAID, SunTag.LinuxSwap => Localization.Linux_swap, SunTag.LVM => Localization.LVM, SunTag.SunBoot => Localization.Sun_boot, SunTag.SunEmpty => Localization.Empty, SunTag.SunHome => Localization.Sun_home, SunTag.SunRoot => Localization.Sun_root, SunTag.SunStand => Localization.Sun_stand, SunTag.SunSwap => Localization.Sun_swap, SunTag.SunUsr => Localization.Sun_usr, SunTag.SunVar => Localization.Sun_var, SunTag.SunWholeDisk => Localization.Whole_disk, SunTag.SunAlt => Localization.Replacement_sectors, SunTag.SunCache => Localization.Sun_cachefs, SunTag.SunReserved => Localization.Reserved_for_SMI, SunTag.VxVmPublic => Localization.Veritas_public, SunTag.VxVmPrivate => Localization.Veritas_private, SunTag.NetBSD => Localization.NetBSD, SunTag.FreeBSD_Swap => Localization.FreeBSD_swap, SunTag.FreeBSD_UFS => Localization.FreeBSD, SunTag.FreeBSD_Vinum => Localization.Vinum, SunTag.FreeBSD_ZFS => Localization.FreeBSD_ZFS, SunTag.FreeBSD_NANDFS => Localization.FreeBSD_nandfs, _ => Localization.Unknown_partition_type }; #region Nested type: dk_label /// SunOS disk label [StructLayout(LayoutKind.Sequential, Pack = 1)] [SwapEndian] partial struct dk_label { /// Informative string [MarshalAs(UnmanagedType.ByValArray, SizeConst = LEN_DKL_ASCII)] public byte[] dkl_asciilabel; /// Padding [MarshalAs(UnmanagedType.ByValArray, SizeConst = LEN_DKL_PAD)] public byte[] dkl_pad; /// rotations per minute public ushort dkl_rpm; /// # physical cylinders public ushort dkl_pcyl; /// alternates per cylinder public ushort dkl_apc; /// size of gap 1 public ushort dkl_gap1; /// size of gap 2 public ushort dkl_gap2; /// interleave factor public ushort dkl_intrlv; /// # of data cylinders public ushort dkl_ncyl; /// # of alternate cylinders public ushort dkl_acyl; /// # of heads in this partition public ushort dkl_nhead; /// # of 512 byte sectors per track public ushort dkl_nsect; /// identifies proper label location public ushort dkl_bhead; /// physical partition # public ushort dkl_ppart; /// Logical partitions [MarshalAs(UnmanagedType.ByValArray, SizeConst = NDKMAP)] public dk_map[] dkl_map; /// identifies this label format public ushort dkl_magic; /// xor checksum of sector public ushort dkl_cksum; } #endregion #region Nested type: dk_label16 [StructLayout(LayoutKind.Sequential, Pack = 1)] [SwapEndian] partial struct dk_label16 { /// vtoc inclusions from AT&T SVr4 public dk_vtoc16 dkl_vtoc; /// # of physical cylinders public uint dkl_pcyl; /// # of data cylinders public uint dkl_ncyl; /// # of alternate cylinders public ushort dkl_acyl; /// cyl offset (for fixed head area) public ushort dkl_bcyl; /// # of heads public uint dkl_nhead; /// # of data sectors per track public uint dkl_nsect; /// interleave factor public ushort dkl_intrlv; /// skew factor public ushort dkl_skew; /// alternates per cyl (SCSI only) public ushort dkl_apc; /// revolutions per minute public ushort dkl_rpm; /// # sectors to skip, writes public ushort dkl_write_reinstruct; /// # sectors to skip, reads public ushort dkl_read_reinstruct; /// for compatible expansion [MarshalAs(UnmanagedType.ByValArray, SizeConst = 4)] public ushort[] dkl_extra; /// unused part of 512 bytes [MarshalAs(UnmanagedType.ByValArray, SizeConst = LEN_DKL_PAD16)] public byte[] dkl_pad; /// identifies this label format public ushort dkl_magic; /// xor checksum of sector public ushort dkl_cksum; } #endregion #region Nested type: dk_label8 [StructLayout(LayoutKind.Sequential, Pack = 1)] [SwapEndian] partial struct dk_label8 { /// for compatibility [MarshalAs(UnmanagedType.ByValArray, SizeConst = LEN_DKL_ASCII)] public byte[] dkl_asciilabel; /// vtoc inclusions from AT&T SVr4 public dk_vtoc8 dkl_vtoc; /// # sectors to skip, writes public ushort dkl_write_reinstruct; /// # sectors to skip, reads public ushort dkl_read_reinstruct; /// unused part of 512 bytes [MarshalAs(UnmanagedType.ByValArray, SizeConst = LEN_DKL_PAD8)] public byte[] dkl_pad; /// rotations per minute public ushort dkl_rpm; /// # physical cylinders public ushort dkl_pcyl; /// alternates per cylinder public ushort dkl_apc; /// obsolete public ushort dkl_obs1; /// obsolete public ushort dkl_obs2; /// interleave factor public ushort dkl_intrlv; /// # of data cylinders public ushort dkl_ncyl; /// # of alternate cylinders public ushort dkl_acyl; /// # of heads in this partition public ushort dkl_nhead; /// # of 512 byte sectors per track public ushort dkl_nsect; /// obsolete public ushort dkl_obs3; /// obsolete public ushort dkl_obs4; /// logical partition headers [MarshalAs(UnmanagedType.ByValArray, SizeConst = NDKMAP)] public dk_map[] dkl_map; /// identifies this label format public ushort dkl_magic; /// xor checksum of sector public ushort dkl_cksum; } #endregion #region Nested type: dk_map /// SunOS logical partitions [StructLayout(LayoutKind.Sequential, Pack = 1)] [SwapEndian] partial struct dk_map { /// starting cylinder public int dkl_cylno; /// number of blocks public int dkl_nblk; } #endregion #region Nested type: dk_map2 /// Solaris logical partition for small disk label [StructLayout(LayoutKind.Sequential, Pack = 1)] [SwapEndian] partial struct dk_map2 { /// ID tag of partition public SunTag p_tag; /// permission flag public SunFlags p_flag; } #endregion #region Nested type: dk_vtoc16 [StructLayout(LayoutKind.Sequential, Pack = 1)] [SwapEndian] partial struct dk_vtoc16 { /// info needed by mboot [MarshalAs(UnmanagedType.ByValArray, SizeConst = 3)] public uint[] v_bootinfo; /// to verify vtoc sanity public uint v_sanity; /// layout version public uint v_version; /// volume name [MarshalAs(UnmanagedType.ByValArray, SizeConst = LEN_DKL_VVOL)] public byte[] v_volume; /// sector size in bytes public ushort v_sectorsz; /// number of partitions public ushort v_nparts; /// free space [MarshalAs(UnmanagedType.ByValArray, SizeConst = 10)] public uint[] v_reserved; /// partition headers [MarshalAs(UnmanagedType.ByValArray, SizeConst = NDKMAP16)] public dkl_partition[] v_part; /// partition timestamp [MarshalAs(UnmanagedType.ByValArray, SizeConst = NDKMAP16)] public int[] v_timestamp; /// for compatibility [MarshalAs(UnmanagedType.ByValArray, SizeConst = LEN_DKL_ASCII)] public byte[] v_asciilabel; } #endregion #region Nested type: dk_vtoc8 [StructLayout(LayoutKind.Sequential, Pack = 1)] [SwapEndian] partial struct dk_vtoc8 { /// layout version public uint v_version; /// volume name [MarshalAs(UnmanagedType.ByValArray, SizeConst = LEN_DKL_VVOL)] public byte[] v_volume; /// number of partitions public ushort v_nparts; /// partition hdrs, sec 2 [MarshalAs(UnmanagedType.ByValArray, SizeConst = NDKMAP)] public dk_map2[] v_part; /// Alignment public ushort padding; /// info needed by mboot [MarshalAs(UnmanagedType.ByValArray, SizeConst = 3)] public uint[] v_bootinfo; /// to verify vtoc sanity public uint v_sanity; /// free space [MarshalAs(UnmanagedType.ByValArray, SizeConst = 10)] public uint[] v_reserved; /// partition timestamp [MarshalAs(UnmanagedType.ByValArray, SizeConst = NDKMAP)] public int[] v_timestamp; } #endregion #region Nested type: dkl_partition /// Solaris logical partition [StructLayout(LayoutKind.Sequential, Pack = 1)] [SwapEndian] partial struct dkl_partition { /// ID tag of partition public SunTag p_tag; /// permision flags public SunFlags p_flag; /// start sector no of partition public int p_start; /// # of blocks in partition public int p_size; } #endregion #region Nested type: SunFlags [Flags] enum SunFlags : ushort { NoMount = 0x0001, ReadOnly = 0x0010 } #endregion #region Nested type: SunTag enum SunTag : ushort { SunEmpty = 0x0000, SunBoot = 0x0001, SunRoot = 0x0002, SunSwap = 0x0003, SunUsr = 0x0004, SunWholeDisk = 0x0005, SunStand = 0x0006, SunVar = 0x0007, SunHome = 0x0008, SunAlt = 0x0009, SunCache = 0x000A, SunReserved = 0x000B, VxVmPublic = 0x000E, VxVmPrivate = 0x000F, LinuxSwap = 0x0082, Linux = 0x0083, LVM = 0x008E, LinuxRaid = 0x00FD, NetBSD = 0x00FF, FreeBSD_Swap = 0x0901, FreeBSD_UFS = 0x0902, FreeBSD_Vinum = 0x0903, FreeBSD_ZFS = 0x0904, FreeBSD_NANDFS = 0x0905 } #endregion }