// /*************************************************************************** // The Disc Image Chef // ---------------------------------------------------------------------------- // // Filename : BFS.cs // Author(s) : Natalia Portillo // // Component : BeOS filesystem plugin. // // --[ Description ] ---------------------------------------------------------- // // Identifies the BeOS 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-2017 Natalia Portillo // ****************************************************************************/ using System; using System.Collections.Generic; using System.Runtime.InteropServices; using System.Text; using DiscImageChef.CommonTypes; 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"); if(encoding == null) CurrentEncoding = Encoding.GetEncoding("iso-8859-15"); else CurrentEncoding = encoding; } public AtheOS(ImagePlugins.ImagePlugin imagePlugin, Partition partition, Encoding encoding) { Name = "AtheOS Filesystem"; PluginUUID = new Guid("AAB2C4F1-DC07-49EE-A948-576CC51B58C5"); if(encoding == null) CurrentEncoding = Encoding.GetEncoding("iso-8859-15"); else CurrentEncoding = encoding; } public override bool Identify(ImagePlugins.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[] sb_sector = new byte[AFS_SUPERBLOCK_SIZE]; Array.Copy(tmp, offset, sb_sector, 0, AFS_SUPERBLOCK_SIZE); magic = BitConverter.ToUInt32(sb_sector, 0x20); return magic == AFS_MAGIC1; } public override void GetInformation(ImagePlugins.ImagePlugin imagePlugin, Partition partition, out string information) { information = ""; StringBuilder sb = new StringBuilder(); AtheosSuperBlock afs_sb = new AtheosSuperBlock(); 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[] sb_sector = new byte[AFS_SUPERBLOCK_SIZE]; Array.Copy(tmp, offset, sb_sector, 0, AFS_SUPERBLOCK_SIZE); GCHandle handle = GCHandle.Alloc(sb_sector, GCHandleType.Pinned); afs_sb = (AtheosSuperBlock)Marshal.PtrToStructure(handle.AddrOfPinnedObject(), typeof(AtheosSuperBlock)); handle.Free(); sb.AppendLine("Atheos filesystem"); if(afs_sb.flags == 1) sb.AppendLine("Filesystem is read-only"); sb.AppendFormat("Volume name: {0}", StringHandlers.CToString(afs_sb.name, CurrentEncoding)).AppendLine(); sb.AppendFormat("{0} bytes per block", afs_sb.block_size).AppendLine(); sb.AppendFormat("{0} blocks in volume ({1} bytes)", afs_sb.num_blocks, afs_sb.num_blocks * afs_sb.block_size).AppendLine(); sb.AppendFormat("{0} used blocks ({1} bytes)", afs_sb.used_blocks, afs_sb.used_blocks * afs_sb.block_size).AppendLine(); sb.AppendFormat("{0} bytes per i-node", afs_sb.inode_size).AppendLine(); sb.AppendFormat("{0} blocks per allocation group ({1} bytes)", afs_sb.blocks_per_ag, afs_sb.blocks_per_ag * afs_sb.block_size).AppendLine(); sb.AppendFormat("{0} allocation groups in volume", afs_sb.num_ags).AppendLine(); sb.AppendFormat("Journal resides in block {0} of allocation group {1} and runs for {2} blocks ({3} bytes)", afs_sb.log_blocks_start, afs_sb.log_blocks_ag, afs_sb.log_blocks_len, afs_sb.log_blocks_len * afs_sb.block_size).AppendLine(); sb.AppendFormat("Journal starts in byte {0} and has {1} bytes in {2} blocks", afs_sb.log_start, afs_sb.log_size, afs_sb.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)", afs_sb.root_dir_start, afs_sb.root_dir_ag, afs_sb.root_dir_len, afs_sb.root_dir_len * afs_sb.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)", afs_sb.deleted_start, afs_sb.deleted_ag, afs_sb.deleted_len, afs_sb.deleted_len * afs_sb.block_size).AppendLine(); sb.AppendFormat("Indices' i-node resides in block {0} of allocation group {1} and runs for {2} blocks ({3} bytes)", afs_sb.indices_start, afs_sb.indices_ag, afs_sb.indices_len, afs_sb.indices_len * afs_sb.block_size).AppendLine(); sb.AppendFormat("{0} blocks for bootloader ({1} bytes)", afs_sb.boot_size, afs_sb.boot_size * afs_sb.block_size).AppendLine(); information = sb.ToString(); xmlFSType = new Schemas.FileSystemType { Clusters = afs_sb.num_blocks, ClusterSize = (int)afs_sb.block_size, Dirty = false, FreeClusters = afs_sb.num_blocks - afs_sb.used_blocks, FreeClustersSpecified = true, Type = "AtheOS filesystem", VolumeName = StringHandlers.CToString(afs_sb.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; } } }