// /*************************************************************************** // The Disc Image Chef // ---------------------------------------------------------------------------- // // 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-2018 Natalia Portillo // ****************************************************************************/ using System; using System.Collections.Generic; using System.Diagnostics.CodeAnalysis; using System.Runtime.InteropServices; using System.Text; using DiscImageChef.CommonTypes; using DiscImageChef.Console; using DiscImageChef.DiscImages; namespace DiscImageChef.Partitions { [SuppressMessage("ReSharper", "InconsistentNaming")] public class SunDisklabel : PartitionPlugin { /// 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 SunDisklabel() { Name = "Sun Disklabel"; PluginUuid = new Guid("50F35CC4-8375-4445-8DCB-1BA550C931A3"); } public override bool GetInformation(ImagePlugin imagePlugin, out List partitions, ulong sectorOffset) { partitions = new List(); if(imagePlugin.GetSectorSize() < 512) return false; if(sectorOffset + 2 >= imagePlugin.GetSectors()) return false; bool useDkl = false, useDkl8 = false, useDkl16 = false; byte[] sunSector = imagePlugin.ReadSector(sectorOffset); GCHandle handle = GCHandle.Alloc(sunSector, GCHandleType.Pinned); dk_label dkl = (dk_label)Marshal.PtrToStructure(handle.AddrOfPinnedObject(), typeof(dk_label)); dk_label8 dkl8 = (dk_label8)Marshal.PtrToStructure(handle.AddrOfPinnedObject(), typeof(dk_label8)); dk_label16 dkl16 = (dk_label16)Marshal.PtrToStructure(handle.AddrOfPinnedObject(), typeof(dk_label16)); handle.Free(); DicConsole.DebugWriteLine("Sun plugin", "dkl.dkl_magic = 0x{0:X4}", dkl.dkl_magic); DicConsole.DebugWriteLine("Sun plugin", "dkl8.dkl_vtoc.v_sanity = 0x{0:X8}", dkl8.dkl_vtoc.v_sanity); DicConsole.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) { sunSector = imagePlugin.ReadSector(sectorOffset + 1); handle = GCHandle.Alloc(sunSector, GCHandleType.Pinned); dkl = (dk_label)Marshal.PtrToStructure(handle.AddrOfPinnedObject(), typeof(dk_label)); dkl8 = (dk_label8)Marshal.PtrToStructure(handle.AddrOfPinnedObject(), typeof(dk_label8)); dkl16 = (dk_label16)Marshal.PtrToStructure(handle.AddrOfPinnedObject(), typeof(dk_label16)); handle.Free(); 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); DicConsole.DebugWriteLine("Sun plugin", "dkl.dkl_asciilabel = \"{0}\"", StringHandlers.CToString(dkl.dkl_asciilabel)); DicConsole.DebugWriteLine("Sun plugin", "dkl.dkl_rpm = {0}", dkl.dkl_rpm); DicConsole.DebugWriteLine("Sun plugin", "dkl.dkl_pcyl = {0}", dkl.dkl_pcyl); DicConsole.DebugWriteLine("Sun plugin", "dkl.dkl_apc = {0}", dkl.dkl_apc); DicConsole.DebugWriteLine("Sun plugin", "dkl.dkl_gap1 = {0}", dkl.dkl_gap1); DicConsole.DebugWriteLine("Sun plugin", "dkl.dkl_gap2 = {0}", dkl.dkl_gap2); DicConsole.DebugWriteLine("Sun plugin", "dkl.dkl_intrlv = {0}", dkl.dkl_intrlv); DicConsole.DebugWriteLine("Sun plugin", "dkl.dkl_ncyl = {0}", dkl.dkl_ncyl); DicConsole.DebugWriteLine("Sun plugin", "dkl.dkl_acyl = {0}", dkl.dkl_acyl); DicConsole.DebugWriteLine("Sun plugin", "dkl.dkl_nhead = {0}", dkl.dkl_nhead); DicConsole.DebugWriteLine("Sun plugin", "dkl.dkl_nsect = {0}", dkl.dkl_nsect); DicConsole.DebugWriteLine("Sun plugin", "dkl.dkl_bhead = {0}", dkl.dkl_bhead); DicConsole.DebugWriteLine("Sun plugin", "dkl.dkl_ppart = {0}", dkl.dkl_ppart); for(int i = 0; i < NDKMAP; i++) { DicConsole.DebugWriteLine("Sun plugin", "dkl.dkl_map[{0}].dkl_cylno = {1}", i, dkl.dkl_map[i].dkl_cylno); DicConsole.DebugWriteLine("Sun plugin", "dkl.dkl_map[{0}].dkl_nblk = {1}", i, dkl.dkl_map[i].dkl_nblk); } DicConsole.DebugWriteLine("Sun plugin", "dkl.dkl_magic = 0x{0:X4}", dkl.dkl_magic); DicConsole.DebugWriteLine("Sun plugin", "dkl.dkl_cksum = 0x{0:X4}", dkl.dkl_cksum); DicConsole.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) { Partition 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.GetSectorSize()), 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.GetSectorSize(), Type = "SunOS partition", Scheme = Name }; if(part.Start < imagePlugin.GetSectors() && part.End <= imagePlugin.GetSectors()) partitions.Add(part); } } else if(useDkl8) { ulong sectorsPerCylinder = (ulong)(dkl8.dkl_nsect * dkl8.dkl_nhead); DicConsole.DebugWriteLine("Sun plugin", "dkl8.dkl_asciilabel = \"{0}\"", StringHandlers.CToString(dkl8.dkl_asciilabel)); DicConsole.DebugWriteLine("Sun plugin", "dkl8.dkl_vtoc.v_version = {0}", dkl8.dkl_vtoc.v_version); DicConsole.DebugWriteLine("Sun plugin", "dkl8.dkl_vtoc.v_volume = \"{0}\"", StringHandlers.CToString(dkl8.dkl_vtoc.v_volume)); DicConsole.DebugWriteLine("Sun plugin", "dkl8.dkl_vtoc.v_nparts = {0}", dkl8.dkl_vtoc.v_nparts); DicConsole.DebugWriteLine("Sun plugin", "dkl8.dkl_vtoc.v_sanity = 0x{0:X8}", dkl8.dkl_vtoc.v_sanity); DicConsole.DebugWriteLine("Sun plugin", "dkl8.dkl_write_reinstruct = {0}", dkl8.dkl_write_reinstruct); DicConsole.DebugWriteLine("Sun plugin", "dkl8.dkl_read_reinstruct = {0}", dkl8.dkl_read_reinstruct); DicConsole.DebugWriteLine("Sun plugin", "dkl8.dkl_rpm = {0}", dkl8.dkl_rpm); DicConsole.DebugWriteLine("Sun plugin", "dkl8.dkl_pcyl = {0}", dkl8.dkl_pcyl); DicConsole.DebugWriteLine("Sun plugin", "dkl8.dkl_apc = {0}", dkl8.dkl_apc); DicConsole.DebugWriteLine("Sun plugin", "dkl8.dkl_obs1 = {0}", dkl8.dkl_obs1); DicConsole.DebugWriteLine("Sun plugin", "dkl8.dkl_obs2 = {0}", dkl8.dkl_obs2); DicConsole.DebugWriteLine("Sun plugin", "dkl8.dkl_intrlv = {0}", dkl8.dkl_intrlv); DicConsole.DebugWriteLine("Sun plugin", "dkl8.dkl_ncyl = {0}", dkl8.dkl_ncyl); DicConsole.DebugWriteLine("Sun plugin", "dkl8.dkl_acyl = {0}", dkl8.dkl_acyl); DicConsole.DebugWriteLine("Sun plugin", "dkl8.dkl_nhead = {0}", dkl8.dkl_nhead); DicConsole.DebugWriteLine("Sun plugin", "dkl8.dkl_nsect = {0}", dkl8.dkl_nsect); DicConsole.DebugWriteLine("Sun plugin", "dkl8.dkl_obs3 = {0}", dkl8.dkl_obs3); DicConsole.DebugWriteLine("Sun plugin", "dkl8.dkl_obs4 = {0}", dkl8.dkl_obs4); for(int i = 0; i < NDKMAP; i++) { DicConsole.DebugWriteLine("Sun plugin", "dkl8.dkl_map[{0}].dkl_cylno = {1}", i, dkl8.dkl_map[i].dkl_cylno); DicConsole.DebugWriteLine("Sun plugin", "dkl8.dkl_map[{0}].dkl_nblk = {1}", i, dkl8.dkl_map[i].dkl_nblk); DicConsole.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); DicConsole.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); DicConsole.DebugWriteLine("Sun plugin", "dkl8.dkl_vtoc.v_timestamp[{0}] = {1}", i, DateHandlers.UnixToDateTime(dkl8.dkl_vtoc.v_timestamp[i])); } DicConsole.DebugWriteLine("Sun plugin", "dkl8.dkl_magic = 0x{0:X4}", dkl8.dkl_magic); DicConsole.DebugWriteLine("Sun plugin", "dkl8.dkl_cksum = 0x{0:X4}", dkl8.dkl_cksum); DicConsole.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) { Partition 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.GetSectorSize()), 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.GetSectorSize(), 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.GetSectors() && part.End <= imagePlugin.GetSectors()) partitions.Add(part); } } else { DicConsole.DebugWriteLine("Sun plugin", "dkl16.dkl_vtoc.v_sanity = 0x{0:X8}", dkl16.dkl_vtoc.v_sanity); DicConsole.DebugWriteLine("Sun plugin", "dkl16.dkl_vtoc.v_version = {0}", dkl16.dkl_vtoc.v_version); DicConsole.DebugWriteLine("Sun plugin", "dkl16.dkl_vtoc.v_volume = \"{0}\"", StringHandlers.CToString(dkl16.dkl_vtoc.v_volume)); DicConsole.DebugWriteLine("Sun plugin", "dkl16.dkl_vtoc.v_sectorsz = {0}", dkl16.dkl_vtoc.v_sectorsz); DicConsole.DebugWriteLine("Sun plugin", "dkl16.dkl_vtoc.v_nparts = {0}", dkl16.dkl_vtoc.v_nparts); DicConsole.DebugWriteLine("Sun plugin", "dkl16.dkl_vtoc.v_asciilabel = \"{0}\"", StringHandlers.CToString(dkl16.dkl_vtoc.v_asciilabel)); DicConsole.DebugWriteLine("Sun plugin", "dkl16.dkl_pcyl = {0}", dkl16.dkl_pcyl); DicConsole.DebugWriteLine("Sun plugin", "dkl16.dkl_ncyl = {0}", dkl16.dkl_ncyl); DicConsole.DebugWriteLine("Sun plugin", "dkl16.dkl_acyl = {0}", dkl16.dkl_acyl); DicConsole.DebugWriteLine("Sun plugin", "dkl16.dkl_bcyl = {0}", dkl16.dkl_bcyl); DicConsole.DebugWriteLine("Sun plugin", "dkl16.dkl_nhead = {0}", dkl16.dkl_nhead); DicConsole.DebugWriteLine("Sun plugin", "dkl16.dkl_nsect = {0}", dkl16.dkl_nsect); DicConsole.DebugWriteLine("Sun plugin", "dkl16.dkl_intrlv = {0}", dkl16.dkl_intrlv); DicConsole.DebugWriteLine("Sun plugin", "dkl16.dkl_skew = {0}", dkl16.dkl_skew); DicConsole.DebugWriteLine("Sun plugin", "dkl16.dkl_apc = {0}", dkl16.dkl_apc); DicConsole.DebugWriteLine("Sun plugin", "dkl16.dkl_rpm = {0}", dkl16.dkl_rpm); DicConsole.DebugWriteLine("Sun plugin", "dkl16.dkl_write_reinstruct = {0}", dkl16.dkl_write_reinstruct); DicConsole.DebugWriteLine("Sun plugin", "dkl16.dkl_read_reinstruct = {0}", dkl16.dkl_read_reinstruct); for(int i = 0; i < NDKMAP16; i++) { DicConsole.DebugWriteLine("Sun plugin", "dkl16.dkl_vtoc.v_part[{0}].p_start = {1}", i, dkl16.dkl_vtoc.v_part[i].p_start); DicConsole.DebugWriteLine("Sun plugin", "dkl16.dkl_vtoc.v_part[{0}].p_size = {1}", i, dkl16.dkl_vtoc.v_part[i].p_size); DicConsole.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); DicConsole.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); DicConsole.DebugWriteLine("Sun plugin", "dkl16.dkl_vtoc.v_timestamp[{0}] = {1}", i, DateHandlers.UnixToDateTime(dkl16.dkl_vtoc.v_timestamp[i])); } DicConsole.DebugWriteLine("Sun plugin", "dkl16.dkl_magic = 0x{0:X4}", dkl16.dkl_magic); DicConsole.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) { Partition 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.GetSectorSize()), 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.GetSectorSize(), 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.GetSectors() && part.End <= imagePlugin.GetSectors()) partitions.Add(part); } } return partitions.Count > 0; } static dk_label SwapDiskLabel(dk_label label) { DicConsole.DebugWriteLine("Sun plugin", "Swapping dk_label"); dk_label lebal = BigEndianMarshal.SwapStructureMembersEndian(label); for(int i = 0; i < label.dkl_map.Length; i++) lebal.dkl_map[i] = BigEndianMarshal.SwapStructureMembersEndian(label.dkl_map[i]); return lebal; } static dk_label8 SwapDiskLabel(dk_label8 label) { DicConsole.DebugWriteLine("Sun plugin", "Swapping dk_label8"); dk_label8 lebal = BigEndianMarshal.SwapStructureMembersEndian(label); lebal.dkl_vtoc = BigEndianMarshal.SwapStructureMembersEndian(label.dkl_vtoc); for(int i = 0; i < label.dkl_map.Length; i++) lebal.dkl_map[i] = BigEndianMarshal.SwapStructureMembersEndian(label.dkl_map[i]); for(int i = 0; i < label.dkl_vtoc.v_bootinfo.Length; i++) lebal.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++) { lebal.dkl_vtoc.v_part[i].p_flag = (SunFlags)Swapping.Swap((ushort)label.dkl_vtoc.v_part[i].p_flag); lebal.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++) lebal.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++) lebal.dkl_vtoc.v_reserved[i] = Swapping.Swap(label.dkl_vtoc.v_reserved[i]); return lebal; } static dk_label16 SwapDiskLabel(dk_label16 label) { DicConsole.DebugWriteLine("Sun plugin", "Swapping dk_label16"); dk_label16 lebal = BigEndianMarshal.SwapStructureMembersEndian(label); lebal.dkl_vtoc = BigEndianMarshal.SwapStructureMembersEndian(label.dkl_vtoc); for(int i = 0; i < label.dkl_vtoc.v_bootinfo.Length; i++) lebal.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++) { lebal.dkl_vtoc.v_part[i].p_flag = (SunFlags)Swapping.Swap((ushort)label.dkl_vtoc.v_part[i].p_flag); lebal.dkl_vtoc.v_part[i].p_tag = (SunTag)Swapping.Swap((ushort)label.dkl_vtoc.v_part[i].p_tag); lebal.dkl_vtoc.v_part[i].p_size = Swapping.Swap(label.dkl_vtoc.v_part[i].p_size); lebal.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++) lebal.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++) lebal.dkl_vtoc.v_reserved[i] = Swapping.Swap(label.dkl_vtoc.v_reserved[i]); return lebal; } static string SunFlagsToString(SunFlags flags) { StringBuilder 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 int dkl_cylno; /// number of blocks public int dkl_nblk; } /// SunOS disk label [StructLayout(LayoutKind.Sequential, Pack = 1)] 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; } /// 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 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; } [StructLayout(LayoutKind.Sequential, Pack = 1)] 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; } [StructLayout(LayoutKind.Sequential, Pack = 1)] 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; } [StructLayout(LayoutKind.Sequential, Pack = 1)] 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; } } }