// /*************************************************************************** // Aaru Data Preservation Suite // ---------------------------------------------------------------------------- // // Filename : Fossil.cs // Author(s) : Natalia Portillo // // Component : Fossil filesystem plugin // // --[ Description ] ---------------------------------------------------------- // // Identifies the Fossil 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-2023 Natalia Portillo // ****************************************************************************/ using System; 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 Schemas; using Marshal = Aaru.Helpers.Marshal; namespace Aaru.Filesystems; /// /// Implements detection for the Plan-9 Fossil on-disk filesystem public sealed class Fossil : IFilesystem { const uint FOSSIL_HDR_MAGIC = 0x3776AE89; const uint FOSSIL_SB_MAGIC = 0x2340A3B1; // Fossil header starts at 128KiB const ulong HEADER_POS = 128 * 1024; const string FS_TYPE = "fossil"; /// public FileSystemType XmlFsType { get; private set; } /// public Encoding Encoding { get; private set; } /// public string Name => Localization.Fossil_Name; /// public Guid Id => new("932BF104-43F6-494F-973C-45EF58A51DA9"); /// public string Author => Authors.NataliaPortillo; /// public bool Identify(IMediaImage imagePlugin, Partition partition) { ulong hdrSector = HEADER_POS / imagePlugin.Info.SectorSize; if(partition.Start + hdrSector > imagePlugin.Info.Sectors) return false; ErrorNumber errno = imagePlugin.ReadSector(partition.Start + hdrSector, out byte[] sector); if(errno != ErrorNumber.NoError) return false; Header hdr = Marshal.ByteArrayToStructureBigEndian
(sector); AaruConsole.DebugWriteLine("Fossil plugin", Localization.magic_at_0_expected_1, hdr.magic, FOSSIL_HDR_MAGIC); return hdr.magic == FOSSIL_HDR_MAGIC; } /// public void GetInformation(IMediaImage imagePlugin, Partition partition, out string information, Encoding encoding) { // Technically everything on Plan 9 from Bell Labs is in UTF-8 Encoding = Encoding.UTF8; information = ""; if(imagePlugin.Info.SectorSize < 512) return; ulong hdrSector = HEADER_POS / imagePlugin.Info.SectorSize; ErrorNumber errno = imagePlugin.ReadSector(partition.Start + hdrSector, out byte[] sector); if(errno != ErrorNumber.NoError) return; Header hdr = Marshal.ByteArrayToStructureBigEndian
(sector); AaruConsole.DebugWriteLine("Fossil plugin", Localization.magic_at_0_expected_1, hdr.magic, FOSSIL_HDR_MAGIC); var sb = new StringBuilder(); sb.AppendLine(Localization.Fossil_filesystem); sb.AppendFormat(Localization.Filesystem_version_0, hdr.version).AppendLine(); sb.AppendFormat(Localization._0_bytes_per_block, hdr.blockSize).AppendLine(); sb.AppendFormat(Localization.Superblock_resides_in_block_0, hdr.super).AppendLine(); sb.AppendFormat(Localization.Labels_resides_in_block_0, hdr.label).AppendLine(); sb.AppendFormat(Localization.Data_starts_at_block_0, hdr.data).AppendLine(); sb.AppendFormat(Localization.Volume_has_0_blocks, hdr.end).AppendLine(); ulong sbLocation = (hdr.super * (hdr.blockSize / imagePlugin.Info.SectorSize)) + partition.Start; XmlFsType = new FileSystemType { Type = FS_TYPE, ClusterSize = hdr.blockSize, Clusters = hdr.end }; if(sbLocation <= partition.End) { imagePlugin.ReadSector(sbLocation, out sector); SuperBlock fsb = Marshal.ByteArrayToStructureBigEndian(sector); AaruConsole.DebugWriteLine("Fossil plugin", Localization.magic_0_expected_1, fsb.magic, FOSSIL_SB_MAGIC); if(fsb.magic == FOSSIL_SB_MAGIC) { sb.AppendFormat(Localization.Epoch_low_0, fsb.epochLow).AppendLine(); sb.AppendFormat(Localization.Epoch_high_0, fsb.epochHigh).AppendLine(); sb.AppendFormat(Localization.Next_QID_0, fsb.qid).AppendLine(); sb.AppendFormat(Localization.Active_root_block_0, fsb.active).AppendLine(); sb.AppendFormat(Localization.Next_root_block_0, fsb.next).AppendLine(); sb.AppendFormat(Localization.Current_root_block_0, fsb.current).AppendLine(); sb.AppendFormat(Localization.Volume_label_0, StringHandlers.CToString(fsb.name, Encoding)).AppendLine(); XmlFsType.VolumeName = StringHandlers.CToString(fsb.name, Encoding); } } information = sb.ToString(); } [StructLayout(LayoutKind.Sequential, Pack = 1)] readonly struct Header { /// Magic number public readonly uint magic; /// Header version public readonly ushort version; /// Block size public readonly ushort blockSize; /// Block containing superblock public readonly uint super; /// Block containing labels public readonly uint label; /// Where do data blocks start public readonly uint data; /// How many data blocks does it have public readonly uint end; } [StructLayout(LayoutKind.Sequential, Pack = 1)] readonly struct SuperBlock { /// Magic number public readonly uint magic; /// Header version public readonly ushort version; /// file system low epoch public readonly uint epochLow; /// file system high(active) epoch public readonly uint epochHigh; /// next qid to allocate public readonly ulong qid; /// data block number: root of active file system public readonly int active; /// data block number: root of next file system to archive public readonly int next; /// data block number: root of file system currently being archived public readonly int current; /// Venti score of last successful archive [MarshalAs(UnmanagedType.ByValArray, SizeConst = 20)] public readonly byte[] last; /// name of file system(just a comment) [MarshalAs(UnmanagedType.ByValArray, SizeConst = 128)] public readonly byte[] name; } }