mirror of
https://github.com/aaru-dps/Aaru.git
synced 2025-12-16 19:24:25 +00:00
560 lines
26 KiB
C#
560 lines
26 KiB
C#
// /***************************************************************************
|
|
// Aaru Data Preservation Suite
|
|
// ----------------------------------------------------------------------------
|
|
//
|
|
// Filename : Info.cs
|
|
// Author(s) : Natalia Portillo <claunia@claunia.com>
|
|
//
|
|
// 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 <http://www.gnu.org/licenses/>.
|
|
//
|
|
// ----------------------------------------------------------------------------
|
|
// Copyright © 2011-2025 Natalia Portillo
|
|
// ****************************************************************************/
|
|
|
|
using System;
|
|
using System.Diagnostics.CodeAnalysis;
|
|
using System.Linq;
|
|
using System.Text;
|
|
using Aaru.CommonTypes.AaruMetadata;
|
|
using Aaru.CommonTypes.Enums;
|
|
using Aaru.CommonTypes.Interfaces;
|
|
using Aaru.Helpers;
|
|
using Aaru.Logging;
|
|
using Sentry;
|
|
using Partition = Aaru.CommonTypes.Partition;
|
|
|
|
namespace Aaru.Filesystems;
|
|
|
|
// Using information from Linux kernel headers
|
|
/// <inheritdoc />
|
|
/// <summary>Implements detection of BSD Fast File System (FFS, aka UNIX File System)</summary>
|
|
[SuppressMessage("ReSharper", "InconsistentNaming")]
|
|
public sealed partial class FFSPlugin
|
|
{
|
|
#region IFilesystem Members
|
|
|
|
/// <inheritdoc />
|
|
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 ex)
|
|
{
|
|
SentrySdk.CaptureException(ex);
|
|
|
|
return false;
|
|
}
|
|
}
|
|
|
|
/// <inheritdoc />
|
|
public void GetInformation(IMediaImage imagePlugin, Partition partition, Encoding encoding, out string information,
|
|
out FileSystem metadata)
|
|
{
|
|
information = "";
|
|
metadata = new FileSystem();
|
|
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;
|
|
}
|
|
|
|
metadata = new FileSystem();
|
|
|
|
switch(magic)
|
|
{
|
|
case UFS_MAGIC:
|
|
sbInformation.AppendLine(Localization.UFS_filesystem);
|
|
metadata.Type = FS_TYPE_UFS;
|
|
|
|
break;
|
|
case UFS_CIGAM:
|
|
sbInformation.AppendLine(Localization.Big_endian_UFS_filesystem);
|
|
metadata.Type = FS_TYPE_UFS;
|
|
|
|
break;
|
|
case UFS_MAGIC_BW:
|
|
sbInformation.AppendLine(Localization.BorderWare_UFS_filesystem);
|
|
metadata.Type = FS_TYPE_UFS;
|
|
|
|
break;
|
|
case UFS_CIGAM_BW:
|
|
sbInformation.AppendLine(Localization.Big_endian_BorderWare_UFS_filesystem);
|
|
metadata.Type = FS_TYPE_UFS;
|
|
|
|
break;
|
|
case UFS2_MAGIC:
|
|
sbInformation.AppendLine(Localization.UFS2_filesystem);
|
|
metadata.Type = FS_TYPE_UFS2;
|
|
|
|
break;
|
|
case UFS2_CIGAM:
|
|
sbInformation.AppendLine(Localization.Big_endian_UFS2_filesystem);
|
|
metadata.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);
|
|
metadata.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);
|
|
metadata.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<SuperBlock>(ufs_sb_sectors);
|
|
|
|
SuperBlock bs_sfu = Marshal.ByteArrayToStructureBigEndian<SuperBlock>(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]);
|
|
}
|
|
|
|
AaruLogging.Debug(MODULE_NAME, "sb offset: 0x{0:X8}", sb_offset);
|
|
AaruLogging.Debug(MODULE_NAME, "fs_rlink: 0x{0:X8}", sb.fs_rlink);
|
|
AaruLogging.Debug(MODULE_NAME, "fs_sblkno: 0x{0:X8}", sb.fs_sblkno);
|
|
AaruLogging.Debug(MODULE_NAME, "fs_cblkno: 0x{0:X8}", sb.fs_cblkno);
|
|
AaruLogging.Debug(MODULE_NAME, "fs_iblkno: 0x{0:X8}", sb.fs_iblkno);
|
|
AaruLogging.Debug(MODULE_NAME, "fs_dblkno: 0x{0:X8}", sb.fs_dblkno);
|
|
AaruLogging.Debug(MODULE_NAME, "fs_size: 0x{0:X8}", sb.fs_size);
|
|
AaruLogging.Debug(MODULE_NAME, "fs_dsize: 0x{0:X8}", sb.fs_dsize);
|
|
AaruLogging.Debug(MODULE_NAME, "fs_ncg: 0x{0:X8}", sb.fs_ncg);
|
|
AaruLogging.Debug(MODULE_NAME, "fs_bsize: 0x{0:X8}", sb.fs_bsize);
|
|
AaruLogging.Debug(MODULE_NAME, "fs_fsize: 0x{0:X8}", sb.fs_fsize);
|
|
AaruLogging.Debug(MODULE_NAME, "fs_frag: 0x{0:X8}", sb.fs_frag);
|
|
AaruLogging.Debug(MODULE_NAME, "fs_minfree: 0x{0:X8}", sb.fs_minfree);
|
|
AaruLogging.Debug(MODULE_NAME, "fs_bmask: 0x{0:X8}", sb.fs_bmask);
|
|
AaruLogging.Debug(MODULE_NAME, "fs_fmask: 0x{0:X8}", sb.fs_fmask);
|
|
AaruLogging.Debug(MODULE_NAME, "fs_bshift: 0x{0:X8}", sb.fs_bshift);
|
|
AaruLogging.Debug(MODULE_NAME, "fs_fshift: 0x{0:X8}", sb.fs_fshift);
|
|
AaruLogging.Debug(MODULE_NAME, "fs_maxcontig: 0x{0:X8}", sb.fs_maxcontig);
|
|
AaruLogging.Debug(MODULE_NAME, "fs_maxbpg: 0x{0:X8}", sb.fs_maxbpg);
|
|
AaruLogging.Debug(MODULE_NAME, "fs_fragshift: 0x{0:X8}", sb.fs_fragshift);
|
|
AaruLogging.Debug(MODULE_NAME, "fs_fsbtodb: 0x{0:X8}", sb.fs_fsbtodb);
|
|
AaruLogging.Debug(MODULE_NAME, "fs_sbsize: 0x{0:X8}", sb.fs_sbsize);
|
|
AaruLogging.Debug(MODULE_NAME, "fs_csmask: 0x{0:X8}", sb.fs_csmask);
|
|
AaruLogging.Debug(MODULE_NAME, "fs_csshift: 0x{0:X8}", sb.fs_csshift);
|
|
AaruLogging.Debug(MODULE_NAME, "fs_nindir: 0x{0:X8}", sb.fs_nindir);
|
|
AaruLogging.Debug(MODULE_NAME, "fs_inopb: 0x{0:X8}", sb.fs_inopb);
|
|
AaruLogging.Debug(MODULE_NAME, "fs_optim: 0x{0:X8}", sb.fs_optim);
|
|
AaruLogging.Debug(MODULE_NAME, "fs_id_1: 0x{0:X8}", sb.fs_id_1);
|
|
AaruLogging.Debug(MODULE_NAME, "fs_id_2: 0x{0:X8}", sb.fs_id_2);
|
|
AaruLogging.Debug(MODULE_NAME, "fs_csaddr: 0x{0:X8}", sb.fs_csaddr);
|
|
AaruLogging.Debug(MODULE_NAME, "fs_cssize: 0x{0:X8}", sb.fs_cssize);
|
|
AaruLogging.Debug(MODULE_NAME, "fs_cgsize: 0x{0:X8}", sb.fs_cgsize);
|
|
AaruLogging.Debug(MODULE_NAME, "fs_ipg: 0x{0:X8}", sb.fs_ipg);
|
|
AaruLogging.Debug(MODULE_NAME, "fs_fpg: 0x{0:X8}", sb.fs_fpg);
|
|
AaruLogging.Debug(MODULE_NAME, "fs_fmod: 0x{0:X2}", sb.fs_fmod);
|
|
AaruLogging.Debug(MODULE_NAME, "fs_clean: 0x{0:X2}", sb.fs_clean);
|
|
AaruLogging.Debug(MODULE_NAME, "fs_ronly: 0x{0:X2}", sb.fs_ronly);
|
|
AaruLogging.Debug(MODULE_NAME, "fs_flags: 0x{0:X2}", sb.fs_flags);
|
|
AaruLogging.Debug(MODULE_NAME, "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();
|
|
|
|
metadata.ModificationDate = DateHandlers.UnixToDateTime(sb.fs_old_time);
|
|
|
|
sbInformation.AppendFormat(Localization._0_blocks_in_volume_1_bytes,
|
|
sb.fs_old_size,
|
|
(long)sb.fs_old_size * sb.fs_fsize)
|
|
.AppendLine();
|
|
|
|
metadata.Clusters = (ulong)sb.fs_old_size;
|
|
metadata.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();
|
|
|
|
metadata.FreeClusters = (ulong)sb.fs_old_cstotal.cs_nbfree;
|
|
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_being_modified);
|
|
metadata.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();
|
|
|
|
metadata.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();
|
|
|
|
metadata.FreeClusters = (ulong)sb.fs_cstotal.cs_nbfree;
|
|
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();
|
|
|
|
metadata.ModificationDate = DateHandlers.UnixToDateTime(sb.fs_time);
|
|
|
|
sbInformation.AppendFormat(Localization._0_blocks_1_bytes, sb.fs_size, sb.fs_size * sb.fs_fsize)
|
|
.AppendLine();
|
|
|
|
metadata.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();
|
|
}
|
|
|
|
#endregion
|
|
} |