// /*************************************************************************** // The Disc Image Chef // ---------------------------------------------------------------------------- // // Filename : Squash.cs // Author(s) : Natalia Portillo // // Component : Squash file system plugin. // // --[ Description ] ---------------------------------------------------------- // // Identifies the Squash 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-2018 Natalia Portillo // ****************************************************************************/ using System; using System.Collections.Generic; using System.Runtime.InteropServices; using System.Text; using DiscImageChef.CommonTypes; using DiscImageChef.DiscImages; using Schemas; namespace DiscImageChef.Filesystems { public class Squash : IFilesystem { /// /// Identifier for Squash /// const uint SQUASH_MAGIC = 0x73717368; const uint SQUASH_CIGAM = 0x68737173; Encoding currentEncoding; FileSystemType xmlFsType; public FileSystemType XmlFsType => xmlFsType; public Encoding Encoding => currentEncoding; public string Name => "Squash filesystem"; public Guid Id => new Guid("F8F6E46F-7A2A-48E3-9C0A-46AF4DC29E09"); public bool Identify(IMediaImage imagePlugin, Partition partition) { if(partition.Start >= partition.End) return false; byte[] sector = imagePlugin.ReadSector(partition.Start); uint magic = BitConverter.ToUInt32(sector, 0x00); return magic == SQUASH_MAGIC || magic == SQUASH_CIGAM; } public void GetInformation(IMediaImage imagePlugin, Partition partition, out string information, Encoding encoding) { currentEncoding = encoding ?? Encoding.UTF8; byte[] sector = imagePlugin.ReadSector(partition.Start); uint magic = BitConverter.ToUInt32(sector, 0x00); SquashSuperBlock sqSb = new SquashSuperBlock(); bool littleEndian = true; switch(magic) { case SQUASH_MAGIC: IntPtr sqSbPtr = Marshal.AllocHGlobal(Marshal.SizeOf(sqSb)); Marshal.Copy(sector, 0, sqSbPtr, Marshal.SizeOf(sqSb)); sqSb = (SquashSuperBlock)Marshal.PtrToStructure(sqSbPtr, typeof(SquashSuperBlock)); Marshal.FreeHGlobal(sqSbPtr); break; case SQUASH_CIGAM: sqSb = BigEndianMarshal.ByteArrayToStructureBigEndian(sector); littleEndian = false; break; } StringBuilder sbInformation = new StringBuilder(); sbInformation.AppendLine("Squash file system"); sbInformation.AppendLine(littleEndian ? "Little-endian" : "Big-endian"); sbInformation.AppendFormat("Volume version {0}.{1}", sqSb.s_major, sqSb.s_minor).AppendLine(); sbInformation.AppendFormat("Volume has {0} bytes", sqSb.bytes_used).AppendLine(); sbInformation.AppendFormat("Volume has {0} bytes per block", sqSb.block_size).AppendLine(); sbInformation.AppendFormat("Volume created on {0}", DateHandlers.UnixUnsignedToDateTime(sqSb.mkfs_time)) .AppendLine(); sbInformation.AppendFormat("Volume has {0} inodes", sqSb.inodes).AppendLine(); switch(sqSb.compression) { case (ushort)SquashCompression.Lz4: sbInformation.AppendLine("Volume is compressed using LZ4"); break; case (ushort)SquashCompression.Lzo: sbInformation.AppendLine("Volume is compressed using LZO"); break; case (ushort)SquashCompression.Lzma: sbInformation.AppendLine("Volume is compressed using LZMA"); break; case (ushort)SquashCompression.Xz: sbInformation.AppendLine("Volume is compressed using XZ"); break; case (ushort)SquashCompression.Zlib: sbInformation.AppendLine("Volume is compressed using GZIP"); break; case (ushort)SquashCompression.Zstd: sbInformation.AppendLine("Volume is compressed using Zstandard"); break; default: sbInformation.AppendFormat("Volume is compressed using unknown algorithm {0}", sqSb.compression) .AppendLine(); break; } information = sbInformation.ToString(); xmlFsType = new FileSystemType { Type = "Squash file system", CreationDate = DateHandlers.UnixUnsignedToDateTime(sqSb.mkfs_time), CreationDateSpecified = true, Clusters = (long)((partition.End - partition.Start + 1) * imagePlugin.Info.SectorSize / sqSb.block_size), ClusterSize = (int)sqSb.block_size, Files = sqSb.inodes, FilesSpecified = true, FreeClusters = 0, FreeClustersSpecified = true }; } enum SquashCompression : ushort { Zlib = 1, Lzma = 2, Lzo = 3, Xz = 4, Lz4 = 5, Zstd = 6 } [StructLayout(LayoutKind.Sequential, Pack = 1)] struct SquashSuperBlock { public uint magic; public uint inodes; public uint mkfs_time; public uint block_size; public uint fragments; public ushort compression; public ushort block_log; public ushort flags; public ushort no_ids; public ushort s_major; public ushort s_minor; public ulong root_inode; public ulong bytes_used; public ulong id_table_start; public ulong xattr_id_table_start; public ulong inode_table_start; public ulong directory_table_start; public ulong fragment_table_start; public ulong lookup_table_start; } } }