// /*************************************************************************** // Aaru Data Preservation Suite // ---------------------------------------------------------------------------- // // Filename : Info.cs // Author(s) : Natalia Portillo // // Component : BSD Fast File System plugin. // // --[ 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.Diagnostics.CodeAnalysis; using System.Linq; using System.Text; using Aaru.CommonTypes; using Aaru.CommonTypes.Enums; using Aaru.CommonTypes.Interfaces; using Aaru.Console; using Aaru.Helpers; using Schemas; using time_t = System.Int32; using ufs_daddr_t = System.Int32; namespace Aaru.Filesystems; // Using information from Linux kernel headers /// /// Implements detection of BSD Fast File System (FFS, aka UNIX File System) [SuppressMessage("ReSharper", "InconsistentNaming")] public sealed partial class FFSPlugin { /// public bool Identify(IMediaImage imagePlugin, Partition partition) { if(2 + partition.Start >= partition.End) return false; uint sbSizeInSectors; if(imagePlugin.Info.SectorSize is 2336 or 2352 or 2448) sbSizeInSectors = block_size / 2048; else sbSizeInSectors = block_size / imagePlugin.Info.SectorSize; ulong[] locations = { sb_start_floppy, sb_start_boot, sb_start_long_boot, sb_start_piggy, sb_start_att_dsdd, 8192 / imagePlugin.Info.SectorSize, 65536 / imagePlugin.Info.SectorSize, 262144 / imagePlugin.Info.SectorSize }; try { foreach(ulong loc in locations.Where(loc => partition.End > partition.Start + loc + sbSizeInSectors)) { ErrorNumber errno = imagePlugin.ReadSectors(partition.Start + loc, sbSizeInSectors, out byte[] ufsSbSectors); if(errno != ErrorNumber.NoError) continue; uint magic = BitConverter.ToUInt32(ufsSbSectors, 0x055C); if(magic is UFS_MAGIC or UFS_CIGAM or UFS_MAGIC_BW or UFS_CIGAM_BW or UFS2_MAGIC or UFS2_CIGAM or UFS_BAD_MAGIC or UFS_BAD_CIGAM) return true; } return false; } catch(Exception) { return false; } } /// public void GetInformation(IMediaImage imagePlugin, Partition partition, out string information, Encoding encoding) { Encoding = encoding ?? Encoding.GetEncoding("iso-8859-15"); information = ""; var sbInformation = new StringBuilder(); uint magic = 0; uint sb_size_in_sectors; byte[] ufs_sb_sectors; ulong sb_offset = partition.Start; bool fs_type_42bsd = false; bool fs_type_43bsd = false; bool fs_type_44bsd = false; bool fs_type_ufs = false; bool fs_type_ufs2 = false; bool fs_type_sun = false; bool fs_type_sun86 = false; if(imagePlugin.Info.SectorSize is 2336 or 2352 or 2448) sb_size_in_sectors = block_size / 2048; else sb_size_in_sectors = block_size / imagePlugin.Info.SectorSize; ulong[] locations = { sb_start_floppy, sb_start_boot, sb_start_long_boot, sb_start_piggy, sb_start_att_dsdd, 8192 / imagePlugin.Info.SectorSize, 65536 / imagePlugin.Info.SectorSize, 262144 / imagePlugin.Info.SectorSize }; ErrorNumber errno; foreach(ulong loc in locations.Where(loc => partition.End > partition.Start + loc + sb_size_in_sectors)) { errno = imagePlugin.ReadSectors(partition.Start + loc, sb_size_in_sectors, out ufs_sb_sectors); if(errno != ErrorNumber.NoError) continue; magic = BitConverter.ToUInt32(ufs_sb_sectors, 0x055C); if(magic is UFS_MAGIC or UFS_CIGAM or UFS_MAGIC_BW or UFS_CIGAM_BW or UFS2_MAGIC or UFS2_CIGAM or UFS_BAD_MAGIC or UFS_BAD_CIGAM) { sb_offset = partition.Start + loc; break; } magic = 0; } if(magic == 0) { information = Localization.Not_a_UFS_filesystem_I_shouldnt_have_arrived_here; return; } XmlFsType = new FileSystemType(); switch(magic) { case UFS_MAGIC: sbInformation.AppendLine(Localization.UFS_filesystem); XmlFsType.Type = FS_TYPE_UFS; break; case UFS_CIGAM: sbInformation.AppendLine(Localization.Big_endian_UFS_filesystem); XmlFsType.Type = FS_TYPE_UFS; break; case UFS_MAGIC_BW: sbInformation.AppendLine(Localization.BorderWare_UFS_filesystem); XmlFsType.Type = FS_TYPE_UFS; break; case UFS_CIGAM_BW: sbInformation.AppendLine(Localization.Big_endian_BorderWare_UFS_filesystem); XmlFsType.Type = FS_TYPE_UFS; break; case UFS2_MAGIC: sbInformation.AppendLine(Localization.UFS2_filesystem); XmlFsType.Type = FS_TYPE_UFS2; break; case UFS2_CIGAM: sbInformation.AppendLine(Localization.Big_endian_UFS2_filesystem); XmlFsType.Type = FS_TYPE_UFS2; break; case UFS_BAD_MAGIC: sbInformation.AppendLine(Localization.Incompletely_initialized_UFS_filesystem); sbInformation.AppendLine(Localization.BEWARE_Following_information_may_be_completely_wrong); XmlFsType.Type = FS_TYPE_UFS; break; case UFS_BAD_CIGAM: sbInformation.AppendLine(Localization.Incompletely_initialized_big_endian_UFS_filesystem); sbInformation.AppendLine(Localization.BEWARE_Following_information_may_be_completely_wrong); XmlFsType.Type = FS_TYPE_UFS; break; } // Fun with seeking follows on superblock reading! errno = imagePlugin.ReadSectors(sb_offset, sb_size_in_sectors, out ufs_sb_sectors); if(errno != ErrorNumber.NoError) return; SuperBlock sb = Marshal.ByteArrayToStructureLittleEndian(ufs_sb_sectors); SuperBlock bs_sfu = Marshal.ByteArrayToStructureBigEndian(ufs_sb_sectors); if((bs_sfu.fs_magic == UFS_MAGIC && sb.fs_magic == UFS_CIGAM) || (bs_sfu.fs_magic == UFS_MAGIC_BW && sb.fs_magic == UFS_CIGAM_BW) || (bs_sfu.fs_magic == UFS2_MAGIC && sb.fs_magic == UFS2_CIGAM) || (bs_sfu.fs_magic == UFS_BAD_MAGIC && sb.fs_magic == UFS_BAD_CIGAM)) { sb = bs_sfu; sb.fs_old_cstotal.cs_nbfree = Swapping.Swap(sb.fs_old_cstotal.cs_nbfree); sb.fs_old_cstotal.cs_ndir = Swapping.Swap(sb.fs_old_cstotal.cs_ndir); sb.fs_old_cstotal.cs_nffree = Swapping.Swap(sb.fs_old_cstotal.cs_nffree); sb.fs_old_cstotal.cs_nifree = Swapping.Swap(sb.fs_old_cstotal.cs_nifree); sb.fs_cstotal.cs_numclusters = Swapping.Swap(sb.fs_cstotal.cs_numclusters); sb.fs_cstotal.cs_nbfree = Swapping.Swap(sb.fs_cstotal.cs_nbfree); sb.fs_cstotal.cs_ndir = Swapping.Swap(sb.fs_cstotal.cs_ndir); sb.fs_cstotal.cs_nffree = Swapping.Swap(sb.fs_cstotal.cs_nffree); sb.fs_cstotal.cs_nifree = Swapping.Swap(sb.fs_cstotal.cs_nifree); sb.fs_cstotal.cs_spare[0] = Swapping.Swap(sb.fs_cstotal.cs_spare[0]); sb.fs_cstotal.cs_spare[1] = Swapping.Swap(sb.fs_cstotal.cs_spare[1]); sb.fs_cstotal.cs_spare[2] = Swapping.Swap(sb.fs_cstotal.cs_spare[2]); } AaruConsole.DebugWriteLine("FFS plugin", "sb offset: 0x{0:X8}", sb_offset); AaruConsole.DebugWriteLine("FFS plugin", "fs_rlink: 0x{0:X8}", sb.fs_rlink); AaruConsole.DebugWriteLine("FFS plugin", "fs_sblkno: 0x{0:X8}", sb.fs_sblkno); AaruConsole.DebugWriteLine("FFS plugin", "fs_cblkno: 0x{0:X8}", sb.fs_cblkno); AaruConsole.DebugWriteLine("FFS plugin", "fs_iblkno: 0x{0:X8}", sb.fs_iblkno); AaruConsole.DebugWriteLine("FFS plugin", "fs_dblkno: 0x{0:X8}", sb.fs_dblkno); AaruConsole.DebugWriteLine("FFS plugin", "fs_size: 0x{0:X8}", sb.fs_size); AaruConsole.DebugWriteLine("FFS plugin", "fs_dsize: 0x{0:X8}", sb.fs_dsize); AaruConsole.DebugWriteLine("FFS plugin", "fs_ncg: 0x{0:X8}", sb.fs_ncg); AaruConsole.DebugWriteLine("FFS plugin", "fs_bsize: 0x{0:X8}", sb.fs_bsize); AaruConsole.DebugWriteLine("FFS plugin", "fs_fsize: 0x{0:X8}", sb.fs_fsize); AaruConsole.DebugWriteLine("FFS plugin", "fs_frag: 0x{0:X8}", sb.fs_frag); AaruConsole.DebugWriteLine("FFS plugin", "fs_minfree: 0x{0:X8}", sb.fs_minfree); AaruConsole.DebugWriteLine("FFS plugin", "fs_bmask: 0x{0:X8}", sb.fs_bmask); AaruConsole.DebugWriteLine("FFS plugin", "fs_fmask: 0x{0:X8}", sb.fs_fmask); AaruConsole.DebugWriteLine("FFS plugin", "fs_bshift: 0x{0:X8}", sb.fs_bshift); AaruConsole.DebugWriteLine("FFS plugin", "fs_fshift: 0x{0:X8}", sb.fs_fshift); AaruConsole.DebugWriteLine("FFS plugin", "fs_maxcontig: 0x{0:X8}", sb.fs_maxcontig); AaruConsole.DebugWriteLine("FFS plugin", "fs_maxbpg: 0x{0:X8}", sb.fs_maxbpg); AaruConsole.DebugWriteLine("FFS plugin", "fs_fragshift: 0x{0:X8}", sb.fs_fragshift); AaruConsole.DebugWriteLine("FFS plugin", "fs_fsbtodb: 0x{0:X8}", sb.fs_fsbtodb); AaruConsole.DebugWriteLine("FFS plugin", "fs_sbsize: 0x{0:X8}", sb.fs_sbsize); AaruConsole.DebugWriteLine("FFS plugin", "fs_csmask: 0x{0:X8}", sb.fs_csmask); AaruConsole.DebugWriteLine("FFS plugin", "fs_csshift: 0x{0:X8}", sb.fs_csshift); AaruConsole.DebugWriteLine("FFS plugin", "fs_nindir: 0x{0:X8}", sb.fs_nindir); AaruConsole.DebugWriteLine("FFS plugin", "fs_inopb: 0x{0:X8}", sb.fs_inopb); AaruConsole.DebugWriteLine("FFS plugin", "fs_optim: 0x{0:X8}", sb.fs_optim); AaruConsole.DebugWriteLine("FFS plugin", "fs_id_1: 0x{0:X8}", sb.fs_id_1); AaruConsole.DebugWriteLine("FFS plugin", "fs_id_2: 0x{0:X8}", sb.fs_id_2); AaruConsole.DebugWriteLine("FFS plugin", "fs_csaddr: 0x{0:X8}", sb.fs_csaddr); AaruConsole.DebugWriteLine("FFS plugin", "fs_cssize: 0x{0:X8}", sb.fs_cssize); AaruConsole.DebugWriteLine("FFS plugin", "fs_cgsize: 0x{0:X8}", sb.fs_cgsize); AaruConsole.DebugWriteLine("FFS plugin", "fs_ipg: 0x{0:X8}", sb.fs_ipg); AaruConsole.DebugWriteLine("FFS plugin", "fs_fpg: 0x{0:X8}", sb.fs_fpg); AaruConsole.DebugWriteLine("FFS plugin", "fs_fmod: 0x{0:X2}", sb.fs_fmod); AaruConsole.DebugWriteLine("FFS plugin", "fs_clean: 0x{0:X2}", sb.fs_clean); AaruConsole.DebugWriteLine("FFS plugin", "fs_ronly: 0x{0:X2}", sb.fs_ronly); AaruConsole.DebugWriteLine("FFS plugin", "fs_flags: 0x{0:X2}", sb.fs_flags); AaruConsole.DebugWriteLine("FFS plugin", "fs_magic: 0x{0:X8}", sb.fs_magic); if(sb.fs_magic == UFS2_MAGIC) fs_type_ufs2 = true; else { const uint SunOSEpoch = 0x1A54C580; // We are supposing there cannot be a Sun's fs created before 1/1/1982 00:00:00 fs_type_43bsd = true; // There is no way of knowing this is the version, but there is of knowing it is not. if(sb.fs_link > 0) { fs_type_42bsd = true; // It was used in 4.2BSD fs_type_43bsd = false; } if((sb.fs_maxfilesize & 0xFFFFFFFF) > SunOSEpoch && DateHandlers.UnixUnsignedToDateTime(sb.fs_maxfilesize & 0xFFFFFFFF) < DateTime.Now) { fs_type_42bsd = false; fs_type_sun = true; fs_type_43bsd = false; } // This is for sure, as it is shared with a sectors/track with non-x86 SunOS, Epoch is absurdly high for that if(sb.fs_old_npsect > SunOSEpoch && DateHandlers.UnixToDateTime(sb.fs_old_npsect) < DateTime.Now) { fs_type_42bsd = false; fs_type_sun86 = true; fs_type_sun = false; fs_type_43bsd = false; } if(sb.fs_cgrotor > 0x00000000 && (uint)sb.fs_cgrotor < 0xFFFFFFFF) { fs_type_42bsd = false; fs_type_sun = false; fs_type_sun86 = false; fs_type_ufs = true; fs_type_43bsd = false; } // 4.3BSD code does not use these fields, they are always set up to 0 fs_type_43bsd &= sb is { fs_id_2: 0, fs_id_1: 0 }; // This is the only 4.4BSD inode format fs_type_44bsd |= sb.fs_old_inodefmt == 2; } if(!fs_type_ufs2) { sbInformation.AppendLine(Localization. There_are_a_lot_of_variants_of_UFS_using_overlapped_values_on_same_fields); sbInformation.AppendLine(Localization. I_will_try_to_guess_which_one_it_is_but_unless_its_UFS2_I_may_be_surely_wrong); } if(fs_type_42bsd) sbInformation.AppendLine(Localization.Guessed_as_42BSD_FFS); if(fs_type_43bsd) sbInformation.AppendLine(Localization.Guessed_as_43BSD_FFS); if(fs_type_44bsd) sbInformation.AppendLine(Localization.Guessed_as_44BSD_FFS); if(fs_type_sun) sbInformation.AppendLine(Localization.Guessed_as_SunOS_FFS); if(fs_type_sun86) sbInformation.AppendLine(Localization.Guessed_as_SunOS_x86_FFS); if(fs_type_ufs) sbInformation.AppendLine(Localization.Guessed_as_UFS); if(fs_type_42bsd) sbInformation.AppendFormat(Localization.Linked_list_of_filesystems_0, sb.fs_link).AppendLine(); sbInformation.AppendFormat(Localization.Superblock_LBA_0, sb.fs_sblkno).AppendLine(); sbInformation.AppendFormat(Localization.Cylinder_block_LBA_0, sb.fs_cblkno).AppendLine(); sbInformation.AppendFormat(Localization.inode_block_LBA_0, sb.fs_iblkno).AppendLine(); sbInformation.AppendFormat(Localization.First_data_block_LBA_0, sb.fs_dblkno).AppendLine(); sbInformation.AppendFormat(Localization.Cylinder_group_offset_in_cylinder_0, sb.fs_old_cgoffset).AppendLine(); sbInformation.AppendFormat(Localization.Volume_last_written_on_0, DateHandlers.UnixToDateTime(sb.fs_old_time)). AppendLine(); XmlFsType.ModificationDate = DateHandlers.UnixToDateTime(sb.fs_old_time); XmlFsType.ModificationDateSpecified = true; sbInformation.AppendFormat(Localization._0_blocks_in_volume_1_bytes, sb.fs_old_size, (long)sb.fs_old_size * sb.fs_fsize).AppendLine(); XmlFsType.Clusters = (ulong)sb.fs_old_size; XmlFsType.ClusterSize = (uint)sb.fs_fsize; sbInformation.AppendFormat(Localization._0_data_blocks_in_volume_1_bytes, sb.fs_old_dsize, (long)sb.fs_old_dsize * sb.fs_fsize).AppendLine(); sbInformation.AppendFormat(Localization._0_cylinder_groups_in_volume, sb.fs_ncg).AppendLine(); sbInformation.AppendFormat(Localization._0_bytes_in_a_basic_block, sb.fs_bsize).AppendLine(); sbInformation.AppendFormat(Localization._0_bytes_in_a_frag_block, sb.fs_fsize).AppendLine(); sbInformation.AppendFormat(Localization._0_frags_in_a_block, sb.fs_frag).AppendLine(); sbInformation.AppendFormat(Localization._0_of_blocks_must_be_free, sb.fs_minfree).AppendLine(); sbInformation.AppendFormat(Localization._0_ms_for_optimal_next_block, sb.fs_old_rotdelay).AppendLine(); sbInformation. AppendFormat(Localization.Disk_rotates_0_times_per_second_1_rpm, sb.fs_old_rps, sb.fs_old_rps * 60). AppendLine(); /* sbInformation.AppendFormat("fs_bmask: 0x{0:X8}", sb.fs_bmask).AppendLine(); sbInformation.AppendFormat("fs_fmask: 0x{0:X8}", sb.fs_fmask).AppendLine(); sbInformation.AppendFormat("fs_bshift: 0x{0:X8}", sb.fs_bshift).AppendLine(); sbInformation.AppendFormat("fs_fshift: 0x{0:X8}", sb.fs_fshift).AppendLine();*/ sbInformation.AppendFormat(Localization._0_contiguous_blocks_at_maximum, sb.fs_maxcontig).AppendLine(); sbInformation.AppendFormat(Localization._0_blocks_per_cylinder_group_at_maximum, sb.fs_maxbpg).AppendLine(); sbInformation.AppendFormat(Localization.Superblock_is_0_bytes, sb.fs_sbsize).AppendLine(); sbInformation.AppendFormat(Localization.NINDIR_0, sb.fs_nindir).AppendLine(); sbInformation.AppendFormat(Localization.INOPB_0, sb.fs_inopb).AppendLine(); sbInformation.AppendFormat(Localization.NSPF_0, sb.fs_old_nspf).AppendLine(); switch(sb.fs_optim) { case 0: sbInformation.AppendLine(Localization.Filesystem_will_minimize_allocation_time); break; case 1: sbInformation.AppendLine(Localization.Filesystem_will_minimize_volume_fragmentation); break; default: sbInformation.AppendFormat(Localization.Unknown_optimization_value_0, sb.fs_optim).AppendLine(); break; } if(fs_type_sun) sbInformation.AppendFormat(Localization._0_sectors_track, sb.fs_old_npsect).AppendLine(); else if(fs_type_sun86) sbInformation.AppendFormat(Localization.Volume_state_on_0, DateHandlers.UnixToDateTime(sb.fs_old_npsect)). AppendLine(); sbInformation.AppendFormat(Localization.Hardware_sector_interleave_0, sb.fs_old_interleave).AppendLine(); sbInformation.AppendFormat(Localization.Sector_zero_skew_0_track, sb.fs_old_trackskew).AppendLine(); switch(fs_type_43bsd) { case false when sb is { fs_id_1: > 0, fs_id_2: > 0 }: sbInformation.AppendFormat(Localization.Volume_ID_0_X8_1_X8, sb.fs_id_1, sb.fs_id_2).AppendLine(); break; case true when sb is { fs_id_1: > 0, fs_id_2: > 0 }: sbInformation.AppendFormat(Localization._0_µsec_for_head_switch, sb.fs_id_1).AppendLine(); sbInformation.AppendFormat(Localization._0_µsec_for_track_to_track_seek, sb.fs_id_2).AppendLine(); break; } sbInformation.AppendFormat(Localization.Cylinder_group_summary_LBA_0, sb.fs_old_csaddr).AppendLine(); sbInformation.AppendFormat(Localization._0_bytes_in_cylinder_group_summary, sb.fs_cssize).AppendLine(); sbInformation.AppendFormat(Localization._0_bytes_in_cylinder_group, sb.fs_cgsize).AppendLine(); sbInformation.AppendFormat(Localization._0_tracks_cylinder, sb.fs_old_ntrak).AppendLine(); sbInformation.AppendFormat(Localization._0_sectors_track, sb.fs_old_nsect).AppendLine(); sbInformation.AppendFormat(Localization._0_sectors_cylinder, sb.fs_old_spc).AppendLine(); sbInformation.AppendFormat(Localization._0_cylinders_in_volume, sb.fs_old_ncyl).AppendLine(); sbInformation.AppendFormat(Localization._0_cylinders_group, sb.fs_old_cpg).AppendLine(); sbInformation.AppendFormat(Localization._0_inodes_per_cylinder_group, sb.fs_ipg).AppendLine(); sbInformation.AppendFormat(Localization._0_blocks_per_group, sb.fs_fpg / sb.fs_frag).AppendLine(); sbInformation.AppendFormat(Localization._0_directories, sb.fs_old_cstotal.cs_ndir).AppendLine(); sbInformation.AppendFormat(Localization._0_free_blocks_1_bytes, sb.fs_old_cstotal.cs_nbfree, (long)sb.fs_old_cstotal.cs_nbfree * sb.fs_fsize).AppendLine(); XmlFsType.FreeClusters = (ulong)sb.fs_old_cstotal.cs_nbfree; XmlFsType.FreeClustersSpecified = true; sbInformation.AppendFormat(Localization._0_free_inodes, sb.fs_old_cstotal.cs_nifree).AppendLine(); sbInformation.AppendFormat(Localization._0_free_frags, sb.fs_old_cstotal.cs_nffree).AppendLine(); if(sb.fs_fmod == 1) { sbInformation.AppendLine(Localization.Superblock_is_under_modification); XmlFsType.Dirty = true; } if(sb.fs_clean == 1) sbInformation.AppendLine(Localization.Volume_is_clean); if(sb.fs_ronly == 1) sbInformation.AppendLine(Localization.Volume_is_read_only); sbInformation.AppendFormat(Localization.Volume_flags_0_X2, sb.fs_flags).AppendLine(); if(fs_type_ufs) sbInformation.AppendFormat(Localization.Volume_last_mounted_at_0, StringHandlers.CToString(sb.fs_fsmnt)). AppendLine(); else if(fs_type_ufs2) { sbInformation.AppendFormat(Localization.Volume_last_mounted_at_0, StringHandlers.CToString(sb.fs_fsmnt)). AppendLine(); sbInformation.AppendFormat(Localization.Volume_name_0, StringHandlers.CToString(sb.fs_volname)). AppendLine(); XmlFsType.VolumeName = StringHandlers.CToString(sb.fs_volname); sbInformation.AppendFormat(Localization.Volume_ID_0_X16, sb.fs_swuid).AppendLine(); //xmlFSType.VolumeSerial = string.Format("{0:X16}", sb.fs_swuid); sbInformation.AppendFormat(Localization.Last_searched_cylinder_group_0, sb.fs_cgrotor).AppendLine(); sbInformation.AppendFormat(Localization._0_contiguously_allocated_directories, sb.fs_contigdirs). AppendLine(); sbInformation.AppendFormat(Localization.Standard_superblock_LBA_0, sb.fs_sblkno).AppendLine(); sbInformation.AppendFormat(Localization._0_directories, sb.fs_cstotal.cs_ndir).AppendLine(); sbInformation.AppendFormat(Localization._0_free_blocks_1_bytes, sb.fs_cstotal.cs_nbfree, sb.fs_cstotal.cs_nbfree * sb.fs_fsize).AppendLine(); XmlFsType.FreeClusters = (ulong)sb.fs_cstotal.cs_nbfree; XmlFsType.FreeClustersSpecified = true; sbInformation.AppendFormat(Localization._0_free_inodes, sb.fs_cstotal.cs_nifree).AppendLine(); sbInformation.AppendFormat(Localization._0_free_frags, sb.fs_cstotal.cs_nffree).AppendLine(); sbInformation.AppendFormat(Localization._0_free_clusters, sb.fs_cstotal.cs_numclusters).AppendLine(); sbInformation.AppendFormat(Localization.Volume_last_written_on_0, DateHandlers.UnixToDateTime(sb.fs_time)). AppendLine(); XmlFsType.ModificationDate = DateHandlers.UnixToDateTime(sb.fs_time); XmlFsType.ModificationDateSpecified = true; sbInformation.AppendFormat(Localization._0_blocks_1_bytes, sb.fs_size, sb.fs_size * sb.fs_fsize). AppendLine(); XmlFsType.Clusters = (ulong)sb.fs_size; sbInformation.AppendFormat(Localization._0_data_blocks_1_bytes, sb.fs_dsize, sb.fs_dsize * sb.fs_fsize). AppendLine(); sbInformation.AppendFormat(Localization.Cylinder_group_summary_area_LBA_0, sb.fs_csaddr).AppendLine(); sbInformation.AppendFormat(Localization._0_blocks_pending_of_being_freed, sb.fs_pendingblocks).AppendLine(); sbInformation.AppendFormat(Localization._0_inodes_pending_of_being_freed, sb.fs_pendinginodes).AppendLine(); } if(fs_type_sun) sbInformation.AppendFormat(Localization.Volume_state_on_0, DateHandlers.UnixToDateTime(sb.fs_old_npsect)). AppendLine(); else if(fs_type_sun86) sbInformation.AppendFormat(Localization._0_sectors_track, sb.fs_state).AppendLine(); else if(fs_type_44bsd) { sbInformation.AppendFormat(Localization._0_blocks_on_cluster_summary_array, sb.fs_contigsumsize). AppendLine(); sbInformation.AppendFormat(Localization.Maximum_length_of_a_symbolic_link_0, sb.fs_maxsymlinklen). AppendLine(); sbInformation.AppendFormat(Localization.A_file_can_be_0_bytes_at_max, sb.fs_maxfilesize).AppendLine(); sbInformation.AppendFormat(Localization.Volume_state_on_0, DateHandlers.UnixToDateTime(sb.fs_state)). AppendLine(); } if(sb.fs_old_nrpos > 0) sbInformation.AppendFormat(Localization._0_rotational_positions, sb.fs_old_nrpos).AppendLine(); if(sb.fs_old_rotbloff > 0) sbInformation.AppendFormat(Localization._0_blocks_per_rotation, sb.fs_old_rotbloff).AppendLine(); information = sbInformation.ToString(); } }