// /*************************************************************************** // 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-2021 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.Enums; using Aaru.CommonTypes.Interfaces; using Aaru.Console; using Aaru.Helpers; using Marshal = Aaru.Helpers.Marshal; namespace Aaru.Partitions { /// /// Implements decoding of Sun disklabels [SuppressMessage("ReSharper", "InconsistentNaming")] public sealed 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)); /// public string Name => "Sun Disklabel"; /// public Guid Id => new("50F35CC4-8375-4445-8DCB-1BA550C931A3"); /// public string Author => "Natalia Portillo"; /// public bool GetInformation(IMediaImage imagePlugin, out List partitions, ulong sectorOffset) { partitions = new List(); 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, out byte[] sunSector); if(errno != ErrorNumber.NoError) return false; dk_label dkl = Marshal.ByteArrayToStructureLittleEndian(sunSector); dk_label8 dkl8 = Marshal.ByteArrayToStructureLittleEndian(sunSector); dk_label16 dkl16 = Marshal.ByteArrayToStructureLittleEndian(sunSector); AaruConsole.DebugWriteLine("Sun plugin", "dkl.dkl_magic = 0x{0:X4}", dkl.dkl_magic); AaruConsole.DebugWriteLine("Sun plugin", "dkl8.dkl_vtoc.v_sanity = 0x{0:X8}", dkl8.dkl_vtoc.v_sanity); AaruConsole.DebugWriteLine("Sun plugin", "dkl16.dkl_vtoc.v_sanity = 0x{0:X8}", dkl16.dkl_vtoc.v_sanity); if(dkl.dkl_magic == DKL_MAGIC || dkl.dkl_magic == DKL_CIGAM) if(dkl16.dkl_vtoc.v_sanity == VTOC_SANE || dkl16.dkl_vtoc.v_sanity == VTOC_ENAS) useDkl16 = true; else if(dkl8.dkl_vtoc.v_sanity == VTOC_SANE || dkl8.dkl_vtoc.v_sanity == VTOC_ENAS) useDkl8 = true; else useDkl = true; if(!useDkl && !useDkl8 && !useDkl16) { errno = imagePlugin.ReadSector(sectorOffset + 1, out sunSector); if(errno == ErrorNumber.NoError) { dkl = Marshal.ByteArrayToStructureLittleEndian(sunSector); dkl8 = Marshal.ByteArrayToStructureLittleEndian(sunSector); dkl16 = Marshal.ByteArrayToStructureLittleEndian(sunSector); if(dkl.dkl_magic == DKL_MAGIC || dkl.dkl_magic == DKL_CIGAM) if(dkl16.dkl_vtoc.v_sanity == VTOC_SANE || dkl16.dkl_vtoc.v_sanity == VTOC_ENAS) useDkl16 = true; else if(dkl8.dkl_vtoc.v_sanity == VTOC_SANE || dkl8.dkl_vtoc.v_sanity == VTOC_ENAS) useDkl8 = true; else useDkl = true; } } if(!useDkl && !useDkl8 && !useDkl16) return false; if(useDkl16 && dkl16.dkl_magic == DKL_CIGAM) dkl16 = SwapDiskLabel(dkl16); else if(useDkl8 && dkl8.dkl_magic == DKL_CIGAM) dkl8 = SwapDiskLabel(dkl8); else if(useDkl && dkl.dkl_magic == DKL_CIGAM) dkl = SwapDiskLabel(dkl); if(useDkl) { ulong sectorsPerCylinder = (ulong)(dkl.dkl_nsect * dkl.dkl_nhead); AaruConsole.DebugWriteLine("Sun plugin", "dkl.dkl_asciilabel = \"{0}\"", StringHandlers.CToString(dkl.dkl_asciilabel)); AaruConsole.DebugWriteLine("Sun plugin", "dkl.dkl_rpm = {0}", dkl.dkl_rpm); AaruConsole.DebugWriteLine("Sun plugin", "dkl.dkl_pcyl = {0}", dkl.dkl_pcyl); AaruConsole.DebugWriteLine("Sun plugin", "dkl.dkl_apc = {0}", dkl.dkl_apc); AaruConsole.DebugWriteLine("Sun plugin", "dkl.dkl_gap1 = {0}", dkl.dkl_gap1); AaruConsole.DebugWriteLine("Sun plugin", "dkl.dkl_gap2 = {0}", dkl.dkl_gap2); AaruConsole.DebugWriteLine("Sun plugin", "dkl.dkl_intrlv = {0}", dkl.dkl_intrlv); AaruConsole.DebugWriteLine("Sun plugin", "dkl.dkl_ncyl = {0}", dkl.dkl_ncyl); AaruConsole.DebugWriteLine("Sun plugin", "dkl.dkl_acyl = {0}", dkl.dkl_acyl); AaruConsole.DebugWriteLine("Sun plugin", "dkl.dkl_nhead = {0}", dkl.dkl_nhead); AaruConsole.DebugWriteLine("Sun plugin", "dkl.dkl_nsect = {0}", dkl.dkl_nsect); AaruConsole.DebugWriteLine("Sun plugin", "dkl.dkl_bhead = {0}", dkl.dkl_bhead); AaruConsole.DebugWriteLine("Sun plugin", "dkl.dkl_ppart = {0}", dkl.dkl_ppart); for(int i = 0; i < NDKMAP; i++) { AaruConsole.DebugWriteLine("Sun plugin", "dkl.dkl_map[{0}].dkl_cylno = {1}", i, dkl.dkl_map[i].dkl_cylno); AaruConsole.DebugWriteLine("Sun plugin", "dkl.dkl_map[{0}].dkl_nblk = {1}", i, dkl.dkl_map[i].dkl_nblk); } AaruConsole.DebugWriteLine("Sun plugin", "dkl.dkl_magic = 0x{0:X4}", dkl.dkl_magic); AaruConsole.DebugWriteLine("Sun plugin", "dkl.dkl_cksum = 0x{0:X4}", dkl.dkl_cksum); AaruConsole.DebugWriteLine("Sun plugin", "sectorsPerCylinder = {0}", sectorsPerCylinder); for(int i = 0; i < NDKMAP; i++) if(dkl.dkl_map[i].dkl_cylno > 0 && dkl.dkl_map[i].dkl_nblk > 0) { 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 = "SunOS partition", Scheme = Name }; if(part.Start < imagePlugin.Info.Sectors && part.End <= imagePlugin.Info.Sectors) partitions.Add(part); } } else if(useDkl8) { ulong sectorsPerCylinder = (ulong)(dkl8.dkl_nsect * dkl8.dkl_nhead); AaruConsole.DebugWriteLine("Sun plugin", "dkl8.dkl_asciilabel = \"{0}\"", StringHandlers.CToString(dkl8.dkl_asciilabel)); AaruConsole.DebugWriteLine("Sun plugin", "dkl8.dkl_vtoc.v_version = {0}", dkl8.dkl_vtoc.v_version); AaruConsole.DebugWriteLine("Sun plugin", "dkl8.dkl_vtoc.v_volume = \"{0}\"", StringHandlers.CToString(dkl8.dkl_vtoc.v_volume)); AaruConsole.DebugWriteLine("Sun plugin", "dkl8.dkl_vtoc.v_nparts = {0}", dkl8.dkl_vtoc.v_nparts); AaruConsole.DebugWriteLine("Sun plugin", "dkl8.dkl_vtoc.v_sanity = 0x{0:X8}", dkl8.dkl_vtoc.v_sanity); AaruConsole.DebugWriteLine("Sun plugin", "dkl8.dkl_write_reinstruct = {0}", dkl8.dkl_write_reinstruct); AaruConsole.DebugWriteLine("Sun plugin", "dkl8.dkl_read_reinstruct = {0}", dkl8.dkl_read_reinstruct); AaruConsole.DebugWriteLine("Sun plugin", "dkl8.dkl_rpm = {0}", dkl8.dkl_rpm); AaruConsole.DebugWriteLine("Sun plugin", "dkl8.dkl_pcyl = {0}", dkl8.dkl_pcyl); AaruConsole.DebugWriteLine("Sun plugin", "dkl8.dkl_apc = {0}", dkl8.dkl_apc); AaruConsole.DebugWriteLine("Sun plugin", "dkl8.dkl_obs1 = {0}", dkl8.dkl_obs1); AaruConsole.DebugWriteLine("Sun plugin", "dkl8.dkl_obs2 = {0}", dkl8.dkl_obs2); AaruConsole.DebugWriteLine("Sun plugin", "dkl8.dkl_intrlv = {0}", dkl8.dkl_intrlv); AaruConsole.DebugWriteLine("Sun plugin", "dkl8.dkl_ncyl = {0}", dkl8.dkl_ncyl); AaruConsole.DebugWriteLine("Sun plugin", "dkl8.dkl_acyl = {0}", dkl8.dkl_acyl); AaruConsole.DebugWriteLine("Sun plugin", "dkl8.dkl_nhead = {0}", dkl8.dkl_nhead); AaruConsole.DebugWriteLine("Sun plugin", "dkl8.dkl_nsect = {0}", dkl8.dkl_nsect); AaruConsole.DebugWriteLine("Sun plugin", "dkl8.dkl_obs3 = {0}", dkl8.dkl_obs3); AaruConsole.DebugWriteLine("Sun plugin", "dkl8.dkl_obs4 = {0}", dkl8.dkl_obs4); for(int i = 0; i < NDKMAP; i++) { AaruConsole.DebugWriteLine("Sun plugin", "dkl8.dkl_map[{0}].dkl_cylno = {1}", i, dkl8.dkl_map[i].dkl_cylno); AaruConsole.DebugWriteLine("Sun plugin", "dkl8.dkl_map[{0}].dkl_nblk = {1}", i, dkl8.dkl_map[i].dkl_nblk); AaruConsole.DebugWriteLine("Sun plugin", "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); AaruConsole.DebugWriteLine("Sun plugin", "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); AaruConsole.DebugWriteLine("Sun plugin", "dkl8.dkl_vtoc.v_timestamp[{0}] = {1}", i, DateHandlers.UnixToDateTime(dkl8.dkl_vtoc.v_timestamp[i])); } AaruConsole.DebugWriteLine("Sun plugin", "dkl8.dkl_magic = 0x{0:X4}", dkl8.dkl_magic); AaruConsole.DebugWriteLine("Sun plugin", "dkl8.dkl_cksum = 0x{0:X4}", dkl8.dkl_cksum); AaruConsole.DebugWriteLine("Sun plugin", "sectorsPerCylinder = {0}", sectorsPerCylinder); if(dkl8.dkl_vtoc.v_nparts > NDKMAP) return false; for(int 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) { 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 += $"\nPartition timestamped on {DateHandlers.UnixToDateTime(dkl8.dkl_vtoc.v_timestamp[i])}"; if(part.Start < imagePlugin.Info.Sectors && part.End <= imagePlugin.Info.Sectors) partitions.Add(part); } } else { AaruConsole.DebugWriteLine("Sun plugin", "dkl16.dkl_vtoc.v_sanity = 0x{0:X8}", dkl16.dkl_vtoc.v_sanity); AaruConsole.DebugWriteLine("Sun plugin", "dkl16.dkl_vtoc.v_version = {0}", dkl16.dkl_vtoc.v_version); AaruConsole.DebugWriteLine("Sun plugin", "dkl16.dkl_vtoc.v_volume = \"{0}\"", StringHandlers.CToString(dkl16.dkl_vtoc.v_volume)); AaruConsole.DebugWriteLine("Sun plugin", "dkl16.dkl_vtoc.v_sectorsz = {0}", dkl16.dkl_vtoc.v_sectorsz); AaruConsole.DebugWriteLine("Sun plugin", "dkl16.dkl_vtoc.v_nparts = {0}", dkl16.dkl_vtoc.v_nparts); AaruConsole.DebugWriteLine("Sun plugin", "dkl16.dkl_vtoc.v_asciilabel = \"{0}\"", StringHandlers.CToString(dkl16.dkl_vtoc.v_asciilabel)); AaruConsole.DebugWriteLine("Sun plugin", "dkl16.dkl_pcyl = {0}", dkl16.dkl_pcyl); AaruConsole.DebugWriteLine("Sun plugin", "dkl16.dkl_ncyl = {0}", dkl16.dkl_ncyl); AaruConsole.DebugWriteLine("Sun plugin", "dkl16.dkl_acyl = {0}", dkl16.dkl_acyl); AaruConsole.DebugWriteLine("Sun plugin", "dkl16.dkl_bcyl = {0}", dkl16.dkl_bcyl); AaruConsole.DebugWriteLine("Sun plugin", "dkl16.dkl_nhead = {0}", dkl16.dkl_nhead); AaruConsole.DebugWriteLine("Sun plugin", "dkl16.dkl_nsect = {0}", dkl16.dkl_nsect); AaruConsole.DebugWriteLine("Sun plugin", "dkl16.dkl_intrlv = {0}", dkl16.dkl_intrlv); AaruConsole.DebugWriteLine("Sun plugin", "dkl16.dkl_skew = {0}", dkl16.dkl_skew); AaruConsole.DebugWriteLine("Sun plugin", "dkl16.dkl_apc = {0}", dkl16.dkl_apc); AaruConsole.DebugWriteLine("Sun plugin", "dkl16.dkl_rpm = {0}", dkl16.dkl_rpm); AaruConsole.DebugWriteLine("Sun plugin", "dkl16.dkl_write_reinstruct = {0}", dkl16.dkl_write_reinstruct); AaruConsole.DebugWriteLine("Sun plugin", "dkl16.dkl_read_reinstruct = {0}", dkl16.dkl_read_reinstruct); for(int i = 0; i < NDKMAP16; i++) { AaruConsole.DebugWriteLine("Sun plugin", "dkl16.dkl_vtoc.v_part[{0}].p_start = {1}", i, dkl16.dkl_vtoc.v_part[i].p_start); AaruConsole.DebugWriteLine("Sun plugin", "dkl16.dkl_vtoc.v_part[{0}].p_size = {1}", i, dkl16.dkl_vtoc.v_part[i].p_size); AaruConsole.DebugWriteLine("Sun plugin", "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); AaruConsole.DebugWriteLine("Sun plugin", "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); AaruConsole.DebugWriteLine("Sun plugin", "dkl16.dkl_vtoc.v_timestamp[{0}] = {1}", i, DateHandlers.UnixToDateTime(dkl16.dkl_vtoc.v_timestamp[i])); } AaruConsole.DebugWriteLine("Sun plugin", "dkl16.dkl_magic = 0x{0:X4}", dkl16.dkl_magic); AaruConsole.DebugWriteLine("Sun plugin", "dkl16.dkl_cksum = 0x{0:X4}", dkl16.dkl_cksum); if(dkl16.dkl_vtoc.v_nparts > NDKMAP16) return false; for(int 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) { 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 += $"\nPartition timestamped on {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; } static dk_label SwapDiskLabel(dk_label label) { AaruConsole.DebugWriteLine("Sun plugin", "Swapping dk_label"); label = (dk_label)Marshal.SwapStructureMembersEndian(label); for(int i = 0; i < label.dkl_map.Length; i++) label.dkl_map[i] = (dk_map)Marshal.SwapStructureMembersEndian(label.dkl_map[i]); return label; } static dk_label8 SwapDiskLabel(dk_label8 label) { AaruConsole.DebugWriteLine("Sun plugin", "Swapping dk_label8"); label = (dk_label8)Marshal.SwapStructureMembersEndian(label); for(int i = 0; i < label.dkl_map.Length; i++) label.dkl_map[i] = (dk_map)Marshal.SwapStructureMembersEndian(label.dkl_map[i]); for(int i = 0; i < label.dkl_vtoc.v_bootinfo.Length; i++) label.dkl_vtoc.v_bootinfo[i] = Swapping.Swap(label.dkl_vtoc.v_bootinfo[i]); for(int i = 0; i < label.dkl_vtoc.v_part.Length; i++) { label.dkl_vtoc.v_part[i].p_flag = (SunFlags)Swapping.Swap((ushort)label.dkl_vtoc.v_part[i].p_flag); label.dkl_vtoc.v_part[i].p_tag = (SunTag)Swapping.Swap((ushort)label.dkl_vtoc.v_part[i].p_tag); } for(int i = 0; i < label.dkl_vtoc.v_timestamp.Length; i++) label.dkl_vtoc.v_timestamp[i] = Swapping.Swap(label.dkl_vtoc.v_timestamp[i]); for(int i = 0; i < label.dkl_vtoc.v_reserved.Length; i++) label.dkl_vtoc.v_reserved[i] = Swapping.Swap(label.dkl_vtoc.v_reserved[i]); return label; } static dk_label16 SwapDiskLabel(dk_label16 label) { AaruConsole.DebugWriteLine("Sun plugin", "Swapping dk_label16"); label = (dk_label16)Marshal.SwapStructureMembersEndian(label); for(int i = 0; i < label.dkl_vtoc.v_bootinfo.Length; i++) label.dkl_vtoc.v_bootinfo[i] = Swapping.Swap(label.dkl_vtoc.v_bootinfo[i]); for(int i = 0; i < label.dkl_vtoc.v_part.Length; i++) { label.dkl_vtoc.v_part[i].p_flag = (SunFlags)Swapping.Swap((ushort)label.dkl_vtoc.v_part[i].p_flag); label.dkl_vtoc.v_part[i].p_tag = (SunTag)Swapping.Swap((ushort)label.dkl_vtoc.v_part[i].p_tag); label.dkl_vtoc.v_part[i].p_size = Swapping.Swap(label.dkl_vtoc.v_part[i].p_size); label.dkl_vtoc.v_part[i].p_start = Swapping.Swap(label.dkl_vtoc.v_part[i].p_start); } for(int i = 0; i < label.dkl_vtoc.v_timestamp.Length; i++) label.dkl_vtoc.v_timestamp[i] = Swapping.Swap(label.dkl_vtoc.v_timestamp[i]); for(int i = 0; i < label.dkl_vtoc.v_reserved.Length; i++) label.dkl_vtoc.v_reserved[i] = Swapping.Swap(label.dkl_vtoc.v_reserved[i]); return label; } static string SunFlagsToString(SunFlags flags) { var sb = new StringBuilder(); if(flags.HasFlag(SunFlags.NoMount)) sb.AppendLine("Unmountable"); if(flags.HasFlag(SunFlags.ReadOnly)) sb.AppendLine("Read-only"); return sb.ToString(); } static string SunIdToString(SunTag id) { switch(id) { case SunTag.Linux: return "Linux"; case SunTag.LinuxRaid: return "Linux RAID"; case SunTag.LinuxSwap: return "Linux swap"; case SunTag.LVM: return "LVM"; case SunTag.SunBoot: return "Sun boot"; case SunTag.SunEmpty: return "Empty"; case SunTag.SunHome: return "Sun /home"; case SunTag.SunRoot: return "Sun /"; case SunTag.SunStand: return "Sun /stand"; case SunTag.SunSwap: return "Sun swap"; case SunTag.SunUsr: return "Sun /usr"; case SunTag.SunVar: return "Sun /var"; case SunTag.SunWholeDisk: return "Whole disk"; case SunTag.SunAlt: return "Replacement sectors"; case SunTag.SunCache: return "Sun cachefs"; case SunTag.SunReserved: return "Reserved for SMI"; case SunTag.VxVmPublic: return "Veritas public"; case SunTag.VxVmPrivate: return "Veritas private"; case SunTag.NetBSD: return "NetBSD"; case SunTag.FreeBSD_Swap: return "FreeBSD swap"; case SunTag.FreeBSD_UFS: return "FreeBSD"; case SunTag.FreeBSD_Vinum: return "Vinum"; case SunTag.FreeBSD_ZFS: return "FreeBSD ZFS"; case SunTag.FreeBSD_NANDFS: return "FreeBSD nandfs"; default: return "Unknown"; } } 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 } [Flags] enum SunFlags : ushort { NoMount = 0x0001, ReadOnly = 0x0010 } /// SunOS logical partitions [StructLayout(LayoutKind.Sequential, Pack = 1)] struct dk_map { /// starting cylinder public readonly int dkl_cylno; /// number of blocks public readonly int dkl_nblk; } /// SunOS disk label [StructLayout(LayoutKind.Sequential, Pack = 1)] struct dk_label { /// Informative string [MarshalAs(UnmanagedType.ByValArray, SizeConst = LEN_DKL_ASCII)] public readonly byte[] dkl_asciilabel; /// Padding [MarshalAs(UnmanagedType.ByValArray, SizeConst = LEN_DKL_PAD)] public readonly byte[] dkl_pad; /// rotations per minute public readonly ushort dkl_rpm; /// # physical cylinders public readonly ushort dkl_pcyl; /// alternates per cylinder public readonly ushort dkl_apc; /// size of gap 1 public readonly ushort dkl_gap1; /// size of gap 2 public readonly ushort dkl_gap2; /// interleave factor public readonly ushort dkl_intrlv; /// # of data cylinders public readonly ushort dkl_ncyl; /// # of alternate cylinders public readonly ushort dkl_acyl; /// # of heads in this partition public readonly ushort dkl_nhead; /// # of 512 byte sectors per track public readonly ushort dkl_nsect; /// identifies proper label location public readonly ushort dkl_bhead; /// physical partition # public readonly ushort dkl_ppart; /// Logical partitions [MarshalAs(UnmanagedType.ByValArray, SizeConst = NDKMAP)] public readonly dk_map[] dkl_map; /// identifies this label format public readonly ushort dkl_magic; /// xor checksum of sector public readonly ushort dkl_cksum; } /// Solaris logical partition for small disk label [StructLayout(LayoutKind.Sequential, Pack = 1)] struct dk_map2 { /// ID tag of partition public SunTag p_tag; /// permission flag public SunFlags p_flag; } /// Solaris logical partition [StructLayout(LayoutKind.Sequential, Pack = 1)] 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; } [StructLayout(LayoutKind.Sequential, Pack = 1)] struct dk_vtoc8 { /// layout version public readonly uint v_version; /// volume name [MarshalAs(UnmanagedType.ByValArray, SizeConst = LEN_DKL_VVOL)] public readonly byte[] v_volume; /// number of partitions public readonly ushort v_nparts; /// partition hdrs, sec 2 [MarshalAs(UnmanagedType.ByValArray, SizeConst = NDKMAP)] public readonly dk_map2[] v_part; /// Alignment public readonly ushort padding; /// info needed by mboot [MarshalAs(UnmanagedType.ByValArray, SizeConst = 3)] public readonly uint[] v_bootinfo; /// to verify vtoc sanity public readonly uint v_sanity; /// free space [MarshalAs(UnmanagedType.ByValArray, SizeConst = 10)] public readonly uint[] v_reserved; /// partition timestamp [MarshalAs(UnmanagedType.ByValArray, SizeConst = NDKMAP)] public readonly int[] v_timestamp; } [StructLayout(LayoutKind.Sequential, Pack = 1)] struct dk_vtoc16 { /// info needed by mboot [MarshalAs(UnmanagedType.ByValArray, SizeConst = 3)] public readonly uint[] v_bootinfo; /// to verify vtoc sanity public readonly uint v_sanity; /// layout version public readonly uint v_version; /// volume name [MarshalAs(UnmanagedType.ByValArray, SizeConst = LEN_DKL_VVOL)] public readonly byte[] v_volume; /// sector size in bytes public readonly ushort v_sectorsz; /// number of partitions public readonly ushort v_nparts; /// free space [MarshalAs(UnmanagedType.ByValArray, SizeConst = 10)] public readonly uint[] v_reserved; /// partition headers [MarshalAs(UnmanagedType.ByValArray, SizeConst = NDKMAP16)] public readonly dkl_partition[] v_part; /// partition timestamp [MarshalAs(UnmanagedType.ByValArray, SizeConst = NDKMAP16)] public readonly int[] v_timestamp; /// for compatibility [MarshalAs(UnmanagedType.ByValArray, SizeConst = LEN_DKL_ASCII)] public readonly byte[] v_asciilabel; } [StructLayout(LayoutKind.Sequential, Pack = 1)] struct dk_label8 { /// for compatibility [MarshalAs(UnmanagedType.ByValArray, SizeConst = LEN_DKL_ASCII)] public readonly byte[] dkl_asciilabel; /// vtoc inclusions from AT&T SVr4 public readonly dk_vtoc8 dkl_vtoc; /// # sectors to skip, writes public readonly ushort dkl_write_reinstruct; /// # sectors to skip, reads public readonly ushort dkl_read_reinstruct; /// unused part of 512 bytes [MarshalAs(UnmanagedType.ByValArray, SizeConst = LEN_DKL_PAD8)] public readonly byte[] dkl_pad; /// rotations per minute public readonly ushort dkl_rpm; /// # physical cylinders public readonly ushort dkl_pcyl; /// alternates per cylinder public readonly ushort dkl_apc; /// obsolete public readonly ushort dkl_obs1; /// obsolete public readonly ushort dkl_obs2; /// interleave factor public readonly ushort dkl_intrlv; /// # of data cylinders public readonly ushort dkl_ncyl; /// # of alternate cylinders public readonly ushort dkl_acyl; /// # of heads in this partition public readonly ushort dkl_nhead; /// # of 512 byte sectors per track public readonly ushort dkl_nsect; /// obsolete public readonly ushort dkl_obs3; /// obsolete public readonly ushort dkl_obs4; /// logical partition headers [MarshalAs(UnmanagedType.ByValArray, SizeConst = NDKMAP)] public readonly dk_map[] dkl_map; /// identifies this label format public readonly ushort dkl_magic; /// xor checksum of sector public readonly ushort dkl_cksum; } [StructLayout(LayoutKind.Sequential, Pack = 1)] struct dk_label16 { /// vtoc inclusions from AT&T SVr4 public readonly dk_vtoc16 dkl_vtoc; /// # of physical cylinders public readonly uint dkl_pcyl; /// # of data cylinders public readonly uint dkl_ncyl; /// # of alternate cylinders public readonly ushort dkl_acyl; /// cyl offset (for fixed head area) public readonly ushort dkl_bcyl; /// # of heads public readonly uint dkl_nhead; /// # of data sectors per track public readonly uint dkl_nsect; /// interleave factor public readonly ushort dkl_intrlv; /// skew factor public readonly ushort dkl_skew; /// alternates per cyl (SCSI only) public readonly ushort dkl_apc; /// revolutions per minute public readonly ushort dkl_rpm; /// # sectors to skip, writes public readonly ushort dkl_write_reinstruct; /// # sectors to skip, reads public readonly ushort dkl_read_reinstruct; /// for compatible expansion [MarshalAs(UnmanagedType.ByValArray, SizeConst = 4)] public readonly ushort[] dkl_extra; /// unused part of 512 bytes [MarshalAs(UnmanagedType.ByValArray, SizeConst = LEN_DKL_PAD16)] public readonly byte[] dkl_pad; /// identifies this label format public readonly ushort dkl_magic; /// xor checksum of sector public readonly ushort dkl_cksum; } } }