// /*************************************************************************** // Aaru Data Preservation Suite // ---------------------------------------------------------------------------- // // Filename : HPFS.cs // Author(s) : Natalia Portillo // // Component : OS/2 High Performance File System plugin. // // --[ Description ] ---------------------------------------------------------- // // Identifies the OS/2 High Performance File System 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.Checksums; using Aaru.CommonTypes; using Aaru.CommonTypes.Enums; using Aaru.CommonTypes.Interfaces; using Aaru.Helpers; using Schemas; using Marshal = Aaru.Helpers.Marshal; namespace Aaru.Filesystems; // Information from an old unnamed document /// /// Implements detection of IBM's High Performance File System (HPFS) public sealed class HPFS : IFilesystem { const string FS_TYPE = "hpfs"; /// public FileSystemType XmlFsType { get; private set; } /// public Encoding Encoding { get; private set; } /// public string Name => Localization.HPFS_Name; /// public Guid Id => new("33513B2C-f590-4acb-8bf2-0b1d5e19dec5"); /// public string Author => Authors.NataliaPortillo; /// public bool Identify(IMediaImage imagePlugin, Partition partition) { if(16 + partition.Start >= partition.End) return false; ErrorNumber errno = imagePlugin.ReadSector(16 + partition.Start, out byte[] hpfsSbSector); // Seek to superblock, on logical sector 16 if(errno != ErrorNumber.NoError) return false; uint magic1 = BitConverter.ToUInt32(hpfsSbSector, 0x000); uint magic2 = BitConverter.ToUInt32(hpfsSbSector, 0x004); return magic1 == 0xF995E849 && magic2 == 0xFA53E9C5; } /// public void GetInformation(IMediaImage imagePlugin, Partition partition, out string information, Encoding encoding) { Encoding = encoding ?? Encoding.GetEncoding("ibm850"); information = ""; var sb = new StringBuilder(); ErrorNumber errno = imagePlugin.ReadSector(0 + partition.Start, out byte[] hpfsBpbSector); // Seek to BIOS parameter block, on logical sector 0 if(errno != ErrorNumber.NoError) return; errno = imagePlugin.ReadSector(16 + partition.Start, out byte[] hpfsSbSector); // Seek to superblock, on logical sector 16 if(errno != ErrorNumber.NoError) return; errno = imagePlugin.ReadSector(17 + partition.Start, out byte[] hpfsSpSector); // Seek to spareblock, on logical sector 17 if(errno != ErrorNumber.NoError) return; BiosParameterBlock bpb = Marshal.ByteArrayToStructureLittleEndian(hpfsBpbSector); SuperBlock hpfsSb = Marshal.ByteArrayToStructureLittleEndian(hpfsSbSector); SpareBlock sp = Marshal.ByteArrayToStructureLittleEndian(hpfsSpSector); if(StringHandlers.CToString(bpb.fs_type) != "HPFS " || hpfsSb.magic1 != 0xF995E849 || hpfsSb.magic2 != 0xFA53E9C5 || sp.magic1 != 0xF9911849 || sp.magic2 != 0xFA5229C5) { sb.AppendLine(Localization.This_may_not_be_HPFS_following_information_may_be_not_correct); sb.AppendFormat(Localization.File_system_type_0_Should_be_HPFS, bpb.fs_type).AppendLine(); sb.AppendFormat(Localization.Superblock_magic1_0_Should_be_0xF995E849, hpfsSb.magic1).AppendLine(); sb.AppendFormat(Localization.Superblock_magic2_0_Should_be_0xFA53E9C5, hpfsSb.magic2).AppendLine(); sb.AppendFormat(Localization.Spareblock_magic1_0_Should_be_0xF9911849, sp.magic1).AppendLine(); sb.AppendFormat(Localization.Spareblock_magic2_0_Should_be_0xFA5229C5, sp.magic2).AppendLine(); } sb.AppendFormat(Localization.OEM_name_0, StringHandlers.CToString(bpb.oem_name)).AppendLine(); sb.AppendFormat(Localization._0_bytes_per_sector, bpb.bps).AppendLine(); // sb.AppendFormat("{0} sectors per cluster", hpfs_bpb.spc).AppendLine(); // sb.AppendFormat("{0} reserved sectors", hpfs_bpb.rsectors).AppendLine(); // sb.AppendFormat("{0} FATs", hpfs_bpb.fats_no).AppendLine(); // sb.AppendFormat("{0} entries on root directory", hpfs_bpb.root_ent).AppendLine(); // sb.AppendFormat("{0} mini sectors on volume", hpfs_bpb.sectors).AppendLine(); sb.AppendFormat(Localization.Media_descriptor_0, bpb.media).AppendLine(); // sb.AppendFormat("{0} sectors per FAT", hpfs_bpb.spfat).AppendLine(); // sb.AppendFormat("{0} sectors per track", hpfs_bpb.sptrk).AppendLine(); // sb.AppendFormat("{0} heads", hpfs_bpb.heads).AppendLine(); sb.AppendFormat(Localization._0_sectors_hidden_before_BPB, bpb.hsectors).AppendLine(); sb.AppendFormat(Localization._0_sectors_on_volume_1_bytes, hpfsSb.sectors, hpfsSb.sectors * bpb.bps). AppendLine(); // sb.AppendFormat("{0} sectors on volume ({1} bytes)", hpfs_bpb.big_sectors, hpfs_bpb.big_sectors * hpfs_bpb.bps).AppendLine(); sb.AppendFormat(Localization.BIOS_drive_number_0, bpb.drive_no).AppendLine(); sb.AppendFormat(Localization.NT_Flags_0, bpb.nt_flags).AppendLine(); sb.AppendFormat(Localization.Signature_0, bpb.signature).AppendLine(); sb.AppendFormat(Localization.Serial_number_0, bpb.serial_no).AppendLine(); sb.AppendFormat(Localization.Volume_label_0, StringHandlers.CToString(bpb.volume_label, Encoding)).AppendLine(); // sb.AppendFormat("Filesystem type: \"{0}\"", hpfs_bpb.fs_type).AppendLine(); DateTime lastChk = DateHandlers.UnixToDateTime(hpfsSb.last_chkdsk); DateTime lastOptim = DateHandlers.UnixToDateTime(hpfsSb.last_optim); sb.AppendFormat(Localization.HPFS_version_0, hpfsSb.version).AppendLine(); sb.AppendFormat(Localization.Functional_version_0, hpfsSb.func_version).AppendLine(); sb.AppendFormat(Localization.Sector_of_root_directory_FNode_0, hpfsSb.root_fnode).AppendLine(); sb.AppendFormat(Localization._0_sectors_are_marked_bad, hpfsSb.badblocks).AppendLine(); sb.AppendFormat(Localization.Sector_of_free_space_bitmaps_0, hpfsSb.bitmap_lsn).AppendLine(); sb.AppendFormat(Localization.Sector_of_bad_blocks_list_0, hpfsSb.badblock_lsn).AppendLine(); if(hpfsSb.last_chkdsk > 0) sb.AppendFormat(Localization.Date_of_last_integrity_check_0, lastChk).AppendLine(); else sb.AppendLine(Localization.Filesystem_integrity_has_never_been_checked); if(hpfsSb.last_optim > 0) sb.AppendFormat(Localization.Date_of_last_optimization_0, lastOptim).AppendLine(); else sb.AppendLine(Localization.Filesystem_has_never_been_optimized); sb.AppendFormat(Localization.Directory_band_has_0_sectors, hpfsSb.dband_sectors).AppendLine(); sb.AppendFormat(Localization.Directory_band_starts_at_sector_0, hpfsSb.dband_start).AppendLine(); sb.AppendFormat(Localization.Directory_band_ends_at_sector_0, hpfsSb.dband_last).AppendLine(); sb.AppendFormat(Localization.Sector_of_directory_band_bitmap_0, hpfsSb.dband_bitmap).AppendLine(); sb.AppendFormat(Localization.Sector_of_ACL_directory_0, hpfsSb.acl_start).AppendLine(); sb.AppendFormat(Localization.Sector_of_Hotfix_directory_0, sp.hotfix_start).AppendLine(); sb.AppendFormat(Localization._0_used_Hotfix_entries, sp.hotfix_used).AppendLine(); sb.AppendFormat(Localization._0_total_Hotfix_entries, sp.hotfix_entries).AppendLine(); sb.AppendFormat(Localization._0_free_spare_DNodes, sp.spare_dnodes_free).AppendLine(); sb.AppendFormat(Localization._0_total_spare_DNodes, sp.spare_dnodes).AppendLine(); sb.AppendFormat(Localization.Sector_of_codepage_directory_0, sp.codepage_lsn).AppendLine(); sb.AppendFormat(Localization._0_codepages_used_in_the_volume, sp.codepages).AppendLine(); sb.AppendFormat(Localization.SuperBlock_CRC32_0, sp.sb_crc32).AppendLine(); sb.AppendFormat(Localization.SpareBlock_CRC32_0, sp.sp_crc32).AppendLine(); sb.AppendLine(Localization.Flags); sb.AppendLine((sp.flags1 & 0x01) == 0x01 ? Localization.Filesystem_is_dirty : Localization.Filesystem_is_clean); if((sp.flags1 & 0x02) == 0x02) sb.AppendLine(Localization.Spare_directory_blocks_are_in_use); if((sp.flags1 & 0x04) == 0x04) sb.AppendLine(Localization.Hotfixes_are_in_use); if((sp.flags1 & 0x08) == 0x08) sb.AppendLine(Localization.Disk_contains_bad_sectors); if((sp.flags1 & 0x10) == 0x10) sb.AppendLine(Localization.Disk_has_a_bad_bitmap); if((sp.flags1 & 0x20) == 0x20) sb.AppendLine(Localization.Filesystem_was_formatted_fast); if((sp.flags1 & 0x40) == 0x40) sb.AppendLine(Localization.Unknown_flag_0x40_on_flags1_is_active); if((sp.flags1 & 0x80) == 0x80) sb.AppendLine(Localization.Filesystem_has_been_mounted_by_an_old_IFS); if((sp.flags2 & 0x01) == 0x01) sb.AppendLine(Localization.Install_DASD_limits); if((sp.flags2 & 0x02) == 0x02) sb.AppendLine(Localization.Resync_DASD_limits); if((sp.flags2 & 0x04) == 0x04) sb.AppendLine(Localization.DASD_limits_are_operational); if((sp.flags2 & 0x08) == 0x08) sb.AppendLine(Localization.Multimedia_is_active); if((sp.flags2 & 0x10) == 0x10) sb.AppendLine(Localization.DCE_ACLs_are_active); if((sp.flags2 & 0x20) == 0x20) sb.AppendLine(Localization.DASD_limits_are_dirty); if((sp.flags2 & 0x40) == 0x40) sb.AppendLine(Localization.Unknown_flag_0x40_on_flags2_is_active); if((sp.flags2 & 0x80) == 0x80) sb.AppendLine(Localization.Unknown_flag_0x80_on_flags2_is_active); XmlFsType = new FileSystemType(); // Theoretically everything from BPB to SB is boot code, should I hash everything or only the sector loaded by BIOS itself? if(bpb.jump[0] == 0xEB && bpb.jump[1] > 0x3C && bpb.jump[1] < 0x80 && bpb.signature2 == 0xAA55) { XmlFsType.Bootable = true; string bootChk = Sha1Context.Data(bpb.boot_code, out byte[] _); sb.AppendLine(Localization.Volume_is_bootable); sb.AppendFormat(Localization.Boot_code_SHA1_0, bootChk).AppendLine(); } XmlFsType.Dirty |= (sp.flags1 & 0x01) == 0x01; XmlFsType.Clusters = hpfsSb.sectors; XmlFsType.ClusterSize = bpb.bps; XmlFsType.Type = FS_TYPE; XmlFsType.VolumeName = StringHandlers.CToString(bpb.volume_label, Encoding); XmlFsType.VolumeSerial = $"{bpb.serial_no:X8}"; XmlFsType.SystemIdentifier = StringHandlers.CToString(bpb.oem_name); information = sb.ToString(); } /// BIOS Parameter Block, at sector 0 [StructLayout(LayoutKind.Sequential, Pack = 1)] readonly struct BiosParameterBlock { /// 0x000, Jump to boot code [MarshalAs(UnmanagedType.ByValArray, SizeConst = 3)] public readonly byte[] jump; /// 0x003, OEM Name, 8 bytes, space-padded [MarshalAs(UnmanagedType.ByValArray, SizeConst = 8)] public readonly byte[] oem_name; /// 0x00B, Bytes per sector public readonly ushort bps; /// 0x00D, Sectors per cluster public readonly byte spc; /// 0x00E, Reserved sectors between BPB and... does it have sense in HPFS? public readonly ushort rsectors; /// 0x010, Number of FATs... seriously? public readonly byte fats_no; /// 0x011, Number of entries on root directory... ok public readonly ushort root_ent; /// 0x013, Sectors in volume... doubt it public readonly ushort sectors; /// 0x015, Media descriptor public readonly byte media; /// 0x016, Sectors per FAT... again public readonly ushort spfat; /// 0x018, Sectors per track... you're kidding public readonly ushort sptrk; /// 0x01A, Heads... stop! public readonly ushort heads; /// 0x01C, Hidden sectors before BPB public readonly uint hsectors; /// 0x024, Sectors in volume if > 65535... public readonly uint big_sectors; /// 0x028, Drive number public readonly byte drive_no; /// 0x029, Volume flags? public readonly byte nt_flags; /// 0x02A, EPB signature, 0x29 public readonly byte signature; /// 0x02B, Volume serial number public readonly uint serial_no; /// 0x02F, Volume label, 11 bytes, space-padded [MarshalAs(UnmanagedType.ByValArray, SizeConst = 11)] public readonly byte[] volume_label; /// 0x03A, Filesystem type, 8 bytes, space-padded ("HPFS ") [MarshalAs(UnmanagedType.ByValArray, SizeConst = 8)] public readonly byte[] fs_type; /// Boot code. [MarshalAs(UnmanagedType.ByValArray, SizeConst = 448)] public readonly byte[] boot_code; /// 0x1FE, 0xAA55 public readonly ushort signature2; } /// HPFS superblock at sector 16 [StructLayout(LayoutKind.Sequential, Pack = 1)] readonly struct SuperBlock { /// 0x000, 0xF995E849 public readonly uint magic1; /// 0x004, 0xFA53E9C5 public readonly uint magic2; /// 0x008, HPFS version public readonly byte version; /// 0x009, 2 if <= 4 GiB, 3 if > 4 GiB public readonly byte func_version; /// 0x00A, Alignment public readonly ushort dummy; /// 0x00C, LSN pointer to root fnode public readonly uint root_fnode; /// 0x010, Sectors on volume public readonly uint sectors; /// 0x014, Bad blocks on volume public readonly uint badblocks; /// 0x018, LSN pointer to volume bitmap public readonly uint bitmap_lsn; /// 0x01C, 0 public readonly uint zero1; /// 0x020, LSN pointer to badblock directory public readonly uint badblock_lsn; /// 0x024, 0 public readonly uint zero2; /// 0x028, Time of last CHKDSK public readonly int last_chkdsk; /// 0x02C, Time of last optimization public readonly int last_optim; /// 0x030, Sectors of dir band public readonly uint dband_sectors; /// 0x034, Start sector of dir band public readonly uint dband_start; /// 0x038, Last sector of dir band public readonly uint dband_last; /// 0x03C, LSN of free space bitmap public readonly uint dband_bitmap; /// 0x040, Can be used for volume name (32 bytes) public readonly ulong zero3; /// 0x048, ... public readonly ulong zero4; /// 0x04C, ... public readonly ulong zero5; /// 0x050, ...; public readonly ulong zero6; /// 0x058, LSN pointer to ACLs (only HPFS386) public readonly uint acl_start; } /// HPFS spareblock at sector 17 [StructLayout(LayoutKind.Sequential, Pack = 1)] readonly struct SpareBlock { /// 0x000, 0xF9911849 public readonly uint magic1; /// 0x004, 0xFA5229C5 public readonly uint magic2; /// 0x008, HPFS flags public readonly byte flags1; /// 0x009, HPFS386 flags public readonly byte flags2; /// 0x00A, Alignment public readonly ushort dummy; /// 0x00C, LSN of hotfix directory public readonly uint hotfix_start; /// 0x010, Used hotfixes public readonly uint hotfix_used; /// 0x014, Total hotfixes available public readonly uint hotfix_entries; /// 0x018, Unused spare dnodes public readonly uint spare_dnodes_free; /// 0x01C, Length of spare dnodes list public readonly uint spare_dnodes; /// 0x020, LSN of codepage directory public readonly uint codepage_lsn; /// 0x024, Number of codepages used public readonly uint codepages; /// 0x028, SuperBlock CRC32 (only HPFS386) public readonly uint sb_crc32; /// 0x02C, SpareBlock CRC32 (only HPFS386) public readonly uint sp_crc32; } }