// /***************************************************************************
// The Disc Image Chef
// ----------------------------------------------------------------------------
//
// Filename : Sun.cs
// Version : 1.0
// Author(s) : Natalia Portillo
//
// Component : Component
//
// Revision : $Revision$
// Last change by : $Author$
// Date : $Date$
//
// --[ Description ] ----------------------------------------------------------
//
// Description
//
// --[ License ] --------------------------------------------------------------
//
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License as
// published by the Free Software Foundation, either version 3 of the
// License, or (at your option) any later version.
//
// This program 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 General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with this program. If not, see .
//
// ----------------------------------------------------------------------------
// Copyright (C) 2011-2015 Claunia.com
// ****************************************************************************/
// //$Id$
using System;
using System.Collections.Generic;
using DiscImageChef.Console;
namespace DiscImageChef.PartPlugins
{
class SunDisklabel : PartPlugin
{
const UInt16 SUN_MAGIC = 0xDABE;
const UInt32 VTOC_MAGIC = 0x600DDEEE;
public enum SunTypes : ushort
{
SunEmpty = 0x0000,
SunBoot = 0x0001,
SunRoot = 0x0002,
SunSwap = 0x0003,
SunUsr = 0x0004,
SunWholeDisk = 0x0005,
SunStand = 0x0006,
SunVar = 0x0007,
SunHome = 0x0008,
LinuxSwap = 0x0082,
Linux = 0x0083,
LVM = 0x008E,
LinuxRaid = 0x00FD
}
[Flags]
public enum SunFlags : ushort
{
NoMount = 0x0001,
ReadOnly = 0x0010,
}
public SunDisklabel()
{
Name = "Sun Disklabel";
PluginUUID = new Guid("50F35CC4-8375-4445-8DCB-1BA550C931A3");
}
public override bool GetInformation(ImagePlugins.ImagePlugin imagePlugin, out List partitions)
{
partitions = new List();
byte[] sunSector = imagePlugin.ReadSector(0);
byte[] tmpString;
SunDiskLabel sdl = new SunDiskLabel();
sdl.vtoc = new SunVTOC();
sdl.spare = new byte[148];
sdl.vtoc.infos = new SunInfo[8];
sdl.vtoc.bootinfo = new uint[3];
sdl.vtoc.reserved = new byte[40];
sdl.vtoc.timestamp = new uint[8];
sdl.partitions = new SunPartition[8];
BigEndianBitConverter.IsLittleEndian = BitConverter.IsLittleEndian;
tmpString = new byte[128];
Array.Copy(sunSector, 0, tmpString, 0, 128);
sdl.info = StringHandlers.CToString(tmpString);
sdl.vtoc.version = BigEndianBitConverter.ToUInt32(sunSector, 0x80 + 0x00);
tmpString = new byte[8];
Array.Copy(sunSector, 0x80 + 0x04, tmpString, 0, 8);
sdl.vtoc.volname = StringHandlers.CToString(tmpString);
sdl.vtoc.nparts = BigEndianBitConverter.ToUInt16(sunSector, 0x80 + 0x0C);
for(int i = 0; i < 8; i++)
{
sdl.vtoc.infos[i] = new SunInfo();
sdl.vtoc.infos[i].id = BigEndianBitConverter.ToUInt16(sunSector, 0x80 + 0x0E + i * 4 + 0x00);
sdl.vtoc.infos[i].flags = BigEndianBitConverter.ToUInt16(sunSector, 0x80 + 0x0E + i * 4 + 0x02);
}
sdl.vtoc.padding = BigEndianBitConverter.ToUInt16(sunSector, 0x80 + 0x2E);
for(int i = 0; i < 3; i++)
sdl.vtoc.bootinfo[i] = BigEndianBitConverter.ToUInt32(sunSector, 0x80 + 0x30 + i * 4);
sdl.vtoc.sanity = BigEndianBitConverter.ToUInt32(sunSector, 0x80 + 0x3C);
Array.Copy(sunSector, 0x80 + 0x40, sdl.vtoc.reserved, 0, 40);
for(int i = 0; i < 8; i++)
sdl.vtoc.timestamp[i] = BigEndianBitConverter.ToUInt32(sunSector, 0x80 + 0x68 + i * 4);
sdl.write_reinstruct = BigEndianBitConverter.ToUInt32(sunSector, 0x108);
sdl.read_reinstruct = BigEndianBitConverter.ToUInt32(sunSector, 0x10C);
Array.Copy(sunSector, 0x110, sdl.spare, 0, 148);
sdl.rspeed = BigEndianBitConverter.ToUInt16(sunSector, 0x1A4);
sdl.pcylcount = BigEndianBitConverter.ToUInt16(sunSector, 0x1A6);
sdl.sparecyl = BigEndianBitConverter.ToUInt16(sunSector, 0x1A8);
sdl.gap1 = BigEndianBitConverter.ToUInt16(sunSector, 0x1AA);
sdl.gap2 = BigEndianBitConverter.ToUInt16(sunSector, 0x1AC);
sdl.ilfact = BigEndianBitConverter.ToUInt16(sunSector, 0x1AE);
sdl.ncyl = BigEndianBitConverter.ToUInt16(sunSector, 0x1B0);
sdl.nacyl = BigEndianBitConverter.ToUInt16(sunSector, 0x1B2);
sdl.ntrks = BigEndianBitConverter.ToUInt16(sunSector, 0x1B4);
sdl.nsect = BigEndianBitConverter.ToUInt16(sunSector, 0x1B6);
sdl.bhead = BigEndianBitConverter.ToUInt16(sunSector, 0x1B8);
sdl.ppart = BigEndianBitConverter.ToUInt16(sunSector, 0x1BA);
for(int i = 0; i < 8; i++)
{
sdl.partitions[i] = new SunPartition();
sdl.partitions[i].start_cylinder = BigEndianBitConverter.ToUInt32(sunSector, 0x1BC + i * 8 + 0x00);
sdl.partitions[i].num_sectors = BigEndianBitConverter.ToUInt32(sunSector, 0x1BC + i * 8 + 0x04);
}
sdl.magic = BigEndianBitConverter.ToUInt16(sunSector, 0x1FC);
sdl.csum = BigEndianBitConverter.ToUInt16(sunSector, 0x1FE);
ushort csum = 0;
for(int i = 0; i < 510; i += 2)
csum ^= BigEndianBitConverter.ToUInt16(sunSector, i);
DicConsole.DebugWriteLine("Sun Disklabel plugin", "sdl.info = {0}", sdl.info);
DicConsole.DebugWriteLine("Sun Disklabel plugin", "sdl.vtoc.version = {0}", sdl.vtoc.version);
DicConsole.DebugWriteLine("Sun Disklabel plugin", "sdl.vtoc.volname = {0}", sdl.vtoc.volname);
DicConsole.DebugWriteLine("Sun Disklabel plugin", "sdl.vtoc.nparts = {0}", sdl.vtoc.nparts);
for(int i = 0; i < 8; i++)
{
DicConsole.DebugWriteLine("Sun Disklabel plugin", "sdl.vtoc.infos[{1}].id = 0x{0:X4}", sdl.vtoc.infos[i].id, i);
DicConsole.DebugWriteLine("Sun Disklabel plugin", "sdl.vtoc.infos[{1}].flags = 0x{0:X4}", sdl.vtoc.infos[i].flags, i);
}
DicConsole.DebugWriteLine("Sun Disklabel plugin", "sdl.vtoc.padding = 0x{0:X4}", sdl.vtoc.padding);
for(int i = 0; i < 3; i++)
DicConsole.DebugWriteLine("Sun Disklabel plugin", "sdl.vtoc.bootinfo[{1}].id = 0x{0:X8}", sdl.vtoc.bootinfo[i], i);
DicConsole.DebugWriteLine("Sun Disklabel plugin", "sdl.vtoc.sanity = 0x{0:X8}", sdl.vtoc.sanity);
for(int i = 0; i < 8; i++)
DicConsole.DebugWriteLine("Sun Disklabel plugin", "sdl.vtoc.timestamp[{1}] = 0x{0:X8}", sdl.vtoc.timestamp[i], i);
DicConsole.DebugWriteLine("Sun Disklabel plugin", "sdl.rspeed = {0}", sdl.rspeed);
DicConsole.DebugWriteLine("Sun Disklabel plugin", "sdl.pcylcount = {0}", sdl.pcylcount);
DicConsole.DebugWriteLine("Sun Disklabel plugin", "sdl.sparecyl = {0}", sdl.sparecyl);
DicConsole.DebugWriteLine("Sun Disklabel plugin", "sdl.gap1 = {0}", sdl.gap1);
DicConsole.DebugWriteLine("Sun Disklabel plugin", "sdl.gap2 = {0}", sdl.gap2);
DicConsole.DebugWriteLine("Sun Disklabel plugin", "sdl.ilfact = {0}", sdl.ilfact);
DicConsole.DebugWriteLine("Sun Disklabel plugin", "sdl.ncyl = {0}", sdl.ncyl);
DicConsole.DebugWriteLine("Sun Disklabel plugin", "sdl.nacyl = {0}", sdl.nacyl);
DicConsole.DebugWriteLine("Sun Disklabel plugin", "sdl.ntrks = {0}", sdl.ntrks);
DicConsole.DebugWriteLine("Sun Disklabel plugin", "sdl.nsect = {0}", sdl.nsect);
DicConsole.DebugWriteLine("Sun Disklabel plugin", "sdl.bhead = {0}", sdl.bhead);
DicConsole.DebugWriteLine("Sun Disklabel plugin", "sdl.ppart = {0}", sdl.ppart);
for(int i = 0; i < 8; i++)
{
DicConsole.DebugWriteLine("Sun Disklabel plugin", "sdl.partitions[{1}].start_cylinder = {0}", sdl.partitions[i].start_cylinder, i);
DicConsole.DebugWriteLine("Sun Disklabel plugin", "sdl.partitions[{1}].num_sectors = {0}", sdl.partitions[i].num_sectors, i);
}
DicConsole.DebugWriteLine("Sun Disklabel plugin", "sdl.magic = 0x{0:X4}", sdl.magic);
DicConsole.DebugWriteLine("Sun Disklabel plugin", "sdl.csum = 0x{0:X4}", sdl.csum);
DicConsole.DebugWriteLine("Sun Disklabel plugin", "csum = 0x{0:X4}", csum);
ulong sectorsPerCylinder = (ulong)(sdl.nsect * sdl.ntrks);
if(sectorsPerCylinder == 0 || sectorsPerCylinder * sdl.pcylcount > imagePlugin.GetSectors() || sdl.magic != SUN_MAGIC)
return false;
for(int i = 0; i < 8; i++)
{
if((SunTypes)sdl.vtoc.infos[i].id != SunTypes.SunWholeDisk && sdl.partitions[i].num_sectors > 0)
{
CommonTypes.Partition part = new CommonTypes.Partition();
part.PartitionDescription = SunFlagsToString((SunFlags)sdl.vtoc.infos[i].flags);
part.PartitionLength = (ulong)sdl.partitions[i].num_sectors * (ulong)imagePlugin.GetSectorSize();
part.PartitionName = "";
part.PartitionSectors = sdl.partitions[i].num_sectors;
part.PartitionSequence = (ulong)i;
part.PartitionStart = (ulong)sdl.partitions[i].start_cylinder * (ulong)sectorsPerCylinder * (ulong)imagePlugin.GetSectorSize();
part.PartitionStartSector = sdl.partitions[i].start_cylinder * sectorsPerCylinder;
part.PartitionType = SunIdToString((SunTypes)sdl.vtoc.infos[i].id);
if(part.PartitionStartSector > imagePlugin.GetSectors() || (part.PartitionStartSector + part.PartitionSectors) > imagePlugin.GetSectors())
return false;
partitions.Add(part);
}
}
return true;
}
public static string SunFlagsToString(SunFlags flags)
{
System.Text.StringBuilder sb = new System.Text.StringBuilder();
if(flags.HasFlag(SunFlags.NoMount))
sb.AppendLine("Unmountable");
if(flags.HasFlag(SunFlags.ReadOnly))
sb.AppendLine("Read-only");
return sb.ToString();
}
public static string SunIdToString(SunTypes id)
{
switch(id)
{
case SunTypes.Linux:
return "Linux";
case SunTypes.LinuxRaid:
return "Linux RAID";
case SunTypes.LinuxSwap:
return "Linux swap";
case SunTypes.LVM:
return "LVM";
case SunTypes.SunBoot:
return "Sun boot";
case SunTypes.SunEmpty:
return "Empty";
case SunTypes.SunHome:
return "Sun /home";
case SunTypes.SunRoot:
return "Sun /";
case SunTypes.SunStand:
return "Sun /stand";
case SunTypes.SunSwap:
return "Sun swap";
case SunTypes.SunUsr:
return "Sun /usr";
case SunTypes.SunVar:
return "Sun /var";
case SunTypes.SunWholeDisk:
return "Whole disk";
default:
return "Unknown";
}
}
public struct SunDiskLabel
{
///
/// Offset 0x000: Informative string, 128 bytes
///
public string info;
///
/// Offset 0x080: Volume Table Of Contents
///
public SunVTOC vtoc;
///
/// Offset 0x108: Sectors to skip on writes
///
public uint write_reinstruct;
///
/// Offset 0x10C: Sectors to skip in reads
///
public uint read_reinstruct;
///
/// Offset 0x110: Unused, 148 bytes
///
public byte[] spare;
///
/// Offset 0x1A4: Rotational speed
///
public ushort rspeed;
///
/// Offset 0x1A6: Physical cylinder count
///
public ushort pcylcount;
///
/// Offset 0x1A8: Extra sectors per cylinder
///
public ushort sparecyl;
///
/// Offset 0x1AA: Obsolete, gap
///
public ushort gap1;
///
/// Offset 0x1AC: Obsolete, gap
///
public ushort gap2;
///
/// Offset 0x1AE: Interleave factor
///
public ushort ilfact;
///
/// Offset 0x1B0: Cylinders
///
public ushort ncyl;
///
/// Offset 0x1B2: Alternate cylinders
///
public ushort nacyl;
///
/// Offset 0x1B4: Tracks per cylinder
///
public ushort ntrks;
///
/// Offset 0x1B6: Sectors per track
///
public ushort nsect;
///
/// Offset 0x1B8: Label head offset
///
public ushort bhead;
///
/// Offset 0x1BA: Physical partition
///
public ushort ppart;
///
/// Offset 0x1BC: Partitions
///
public SunPartition[] partitions;
///
/// Offset 0x1FC:
///
public ushort magic;
///
/// Offset 0x1FE: XOR of label
///
public ushort csum;
}
public struct SunVTOC
{
///
/// Offset 0x00: VTOC version
///
public uint version;
///
/// Offset 0x04: Volume name, 8 bytes
///
public string volname;
///
/// Offset 0x0C: Number of partitions
///
public ushort nparts;
///
/// Offset 0x0E: Partition information, 8 entries
///
public SunInfo[] infos;
///
/// Offset 0x2E: Padding
///
public ushort padding;
///
/// Offset 0x30: Information needed by mboot, 3 entries
///
public uint[] bootinfo;
///
/// Offset 0x3C: VTOC magic
///
public uint sanity;
///
/// Offset 0x40: Reserved, 40 bytes
///
public byte[] reserved;
///
/// Offset 0x68: Partition timestamps, 8 entries
///
public uint[] timestamp;
}
public struct SunInfo
{
///
/// Offset 0x00: Partition ID
///
public ushort id;
///
/// Offset 0x02: Partition flags
///
public ushort flags;
}
public struct SunPartition
{
///
/// Offset 0x00: Starting cylinder
///
public uint start_cylinder;
///
/// Offset 0x02: Sectors
///
public uint num_sectors;
}
}
}