// /*************************************************************************** // The Disc Image Chef // ---------------------------------------------------------------------------- // // Filename : Atheos.cs // Author(s) : Natalia Portillo // // Component : Atheos filesystem plugin. // // --[ Description ] ---------------------------------------------------------- // // Identifies the Atheos filesystem 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 . // // ---------------------------------------------------------------------------- // Copyright © 2011-2018 Natalia Portillo // ****************************************************************************/ using System; using System.Collections.Generic; using System.Runtime.InteropServices; using System.Text; using DiscImageChef.CommonTypes; using DiscImageChef.DiscImages; using Schemas; namespace DiscImageChef.Filesystems { public class AtheOS : Filesystem { // Little endian constants (that is, as read by .NET :p) const uint AFS_MAGIC1 = 0x41465331; const uint AFS_MAGIC2 = 0xDD121031; const uint AFS_MAGIC3 = 0x15B6830E; // Common constants const uint AFS_SUPERBLOCK_SIZE = 1024; const uint AFS_BOOTBLOCK_SIZE = AFS_SUPERBLOCK_SIZE; public AtheOS() { Name = "AtheOS Filesystem"; PluginUuid = new Guid("AAB2C4F1-DC07-49EE-A948-576CC51B58C5"); CurrentEncoding = Encoding.GetEncoding("iso-8859-15"); } public AtheOS(Encoding encoding) { Name = "AtheOS Filesystem"; PluginUuid = new Guid("AAB2C4F1-DC07-49EE-A948-576CC51B58C5"); CurrentEncoding = encoding ?? Encoding.GetEncoding("iso-8859-15"); } public AtheOS(ImagePlugin imagePlugin, Partition partition, Encoding encoding) { Name = "AtheOS Filesystem"; PluginUuid = new Guid("AAB2C4F1-DC07-49EE-A948-576CC51B58C5"); CurrentEncoding = encoding ?? Encoding.GetEncoding("iso-8859-15"); } public override bool Identify(ImagePlugin imagePlugin, Partition partition) { ulong sector = AFS_BOOTBLOCK_SIZE / imagePlugin.GetSectorSize(); uint offset = AFS_BOOTBLOCK_SIZE % imagePlugin.GetSectorSize(); uint run = 1; if(imagePlugin.GetSectorSize() < AFS_SUPERBLOCK_SIZE) run = AFS_SUPERBLOCK_SIZE / imagePlugin.GetSectorSize(); if(sector + partition.Start >= partition.End) return false; uint magic; byte[] tmp = imagePlugin.ReadSectors(sector + partition.Start, run); byte[] sbSector = new byte[AFS_SUPERBLOCK_SIZE]; Array.Copy(tmp, offset, sbSector, 0, AFS_SUPERBLOCK_SIZE); magic = BitConverter.ToUInt32(sbSector, 0x20); return magic == AFS_MAGIC1; } public override void GetInformation(ImagePlugin imagePlugin, Partition partition, out string information) { information = ""; StringBuilder sb = new StringBuilder(); ulong sector = AFS_BOOTBLOCK_SIZE / imagePlugin.GetSectorSize(); uint offset = AFS_BOOTBLOCK_SIZE % imagePlugin.GetSectorSize(); uint run = 1; if(imagePlugin.GetSectorSize() < AFS_SUPERBLOCK_SIZE) run = AFS_SUPERBLOCK_SIZE / imagePlugin.GetSectorSize(); byte[] tmp = imagePlugin.ReadSectors(sector + partition.Start, run); byte[] sbSector = new byte[AFS_SUPERBLOCK_SIZE]; Array.Copy(tmp, offset, sbSector, 0, AFS_SUPERBLOCK_SIZE); GCHandle handle = GCHandle.Alloc(sbSector, GCHandleType.Pinned); AtheosSuperBlock afsSb = (AtheosSuperBlock)Marshal.PtrToStructure(handle.AddrOfPinnedObject(), typeof(AtheosSuperBlock)); handle.Free(); sb.AppendLine("Atheos filesystem"); if(afsSb.flags == 1) sb.AppendLine("Filesystem is read-only"); sb.AppendFormat("Volume name: {0}", StringHandlers.CToString(afsSb.name, CurrentEncoding)).AppendLine(); sb.AppendFormat("{0} bytes per block", afsSb.block_size).AppendLine(); sb.AppendFormat("{0} blocks in volume ({1} bytes)", afsSb.num_blocks, afsSb.num_blocks * afsSb.block_size).AppendLine(); sb.AppendFormat("{0} used blocks ({1} bytes)", afsSb.used_blocks, afsSb.used_blocks * afsSb.block_size) .AppendLine(); sb.AppendFormat("{0} bytes per i-node", afsSb.inode_size).AppendLine(); sb.AppendFormat("{0} blocks per allocation group ({1} bytes)", afsSb.blocks_per_ag, afsSb.blocks_per_ag * afsSb.block_size).AppendLine(); sb.AppendFormat("{0} allocation groups in volume", afsSb.num_ags).AppendLine(); sb.AppendFormat("Journal resides in block {0} of allocation group {1} and runs for {2} blocks ({3} bytes)", afsSb.log_blocks_start, afsSb.log_blocks_ag, afsSb.log_blocks_len, afsSb.log_blocks_len * afsSb.block_size).AppendLine(); sb.AppendFormat("Journal starts in byte {0} and has {1} bytes in {2} blocks", afsSb.log_start, afsSb.log_size, afsSb.log_valid_blocks).AppendLine(); sb .AppendFormat("Root folder's i-node resides in block {0} of allocation group {1} and runs for {2} blocks ({3} bytes)", afsSb.root_dir_start, afsSb.root_dir_ag, afsSb.root_dir_len, afsSb.root_dir_len * afsSb.block_size).AppendLine(); sb .AppendFormat("Directory containing files scheduled for deletion's i-node resides in block {0} of allocation group {1} and runs for {2} blocks ({3} bytes)", afsSb.deleted_start, afsSb.deleted_ag, afsSb.deleted_len, afsSb.deleted_len * afsSb.block_size).AppendLine(); sb .AppendFormat("Indices' i-node resides in block {0} of allocation group {1} and runs for {2} blocks ({3} bytes)", afsSb.indices_start, afsSb.indices_ag, afsSb.indices_len, afsSb.indices_len * afsSb.block_size).AppendLine(); sb.AppendFormat("{0} blocks for bootloader ({1} bytes)", afsSb.boot_size, afsSb.boot_size * afsSb.block_size).AppendLine(); information = sb.ToString(); XmlFsType = new FileSystemType { Clusters = afsSb.num_blocks, ClusterSize = (int)afsSb.block_size, Dirty = false, FreeClusters = afsSb.num_blocks - afsSb.used_blocks, FreeClustersSpecified = true, Type = "AtheOS filesystem", VolumeName = StringHandlers.CToString(afsSb.name, CurrentEncoding) }; } /// /// Be superblock /// [StructLayout(LayoutKind.Sequential, Pack = 1)] struct AtheosSuperBlock { /// 0x000, Volume name, 32 bytes [MarshalAs(UnmanagedType.ByValArray, SizeConst = 32)] public byte[] name; /// 0x020, "AFS1", 0x41465331 public uint magic1; /// 0x024, "BIGE", 0x42494745 public uint fs_byte_order; /// 0x028, Bytes per block public uint block_size; /// 0x02C, 1 << block_shift == block_size public uint block_shift; /// 0x030, Blocks in volume public long num_blocks; /// 0x038, Used blocks in volume public long used_blocks; /// 0x040, Bytes per inode public int inode_size; /// 0x044, 0xDD121031 public uint magic2; /// 0x048, Blocks per allocation group public int blocks_per_ag; /// 0x04C, 1 << ag_shift == blocks_per_ag public int ag_shift; /// 0x050, Allocation groups in volume public int num_ags; /// 0x054, 0x434c454e if clean, 0x44495254 if dirty public uint flags; /// 0x058, Allocation group of journal public int log_blocks_ag; /// 0x05C, Start block of journal, inside ag public ushort log_blocks_start; /// 0x05E, Length in blocks of journal, inside ag public ushort log_blocks_len; /// 0x060, Start of journal public long log_start; /// 0x068, Valid block logs public int log_valid_blocks; /// 0x06C, Log size public int log_size; /// 0x070, 0x15B6830E public uint magic3; /// 0x074, Allocation group where root folder's i-node resides public int root_dir_ag; /// 0x078, Start in ag of root folder's i-node public ushort root_dir_start; /// 0x07A, As this is part of inode_addr, this is 1 public ushort root_dir_len; /// 0x07C, Allocation group where pending-delete-files' i-node resides public int deleted_ag; /// 0x080, Start in ag of pending-delete-files' i-node public ushort deleted_start; /// 0x082, As this is part of inode_addr, this is 1 public ushort deleted_len; /// 0x084, Allocation group where indices' i-node resides public int indices_ag; /// 0x088, Start in ag of indices' i-node public ushort indices_start; /// 0x08A, As this is part of inode_addr, this is 1 public ushort indices_len; /// 0x08C, Size of bootloader public int boot_size; } public override Errno Mount() { return Errno.NotImplemented; } public override Errno Mount(bool debug) { return Errno.NotImplemented; } public override Errno Unmount() { return Errno.NotImplemented; } public override Errno MapBlock(string path, long fileBlock, ref long deviceBlock) { return Errno.NotImplemented; } public override Errno GetAttributes(string path, ref FileAttributes attributes) { return Errno.NotImplemented; } public override Errno ListXAttr(string path, ref List xattrs) { return Errno.NotImplemented; } public override Errno GetXattr(string path, string xattr, ref byte[] buf) { return Errno.NotImplemented; } public override Errno Read(string path, long offset, long size, ref byte[] buf) { return Errno.NotImplemented; } public override Errno ReadDir(string path, ref List contents) { return Errno.NotImplemented; } public override Errno StatFs(ref FileSystemInfo stat) { return Errno.NotImplemented; } public override Errno Stat(string path, ref FileEntryInfo stat) { return Errno.NotImplemented; } public override Errno ReadLink(string path, ref string dest) { return Errno.NotImplemented; } } }