95 lines
2.4 KiB
Zig
95 lines
2.4 KiB
Zig
const std = @import("std");
|
|
|
|
const DecompTypes = @import("decomp/types.zig");
|
|
const Decompressor = @import("decomp.zig");
|
|
const ExtractionOptions = @import("options.zig");
|
|
const Inode = @import("inode.zig");
|
|
|
|
pub const Error = error{
|
|
BadMagic,
|
|
BadBlockLog,
|
|
BadVersion,
|
|
BadCheck,
|
|
};
|
|
|
|
const Archive = @This();
|
|
|
|
super: Superblock,
|
|
|
|
stateless_decomp: Decompressor,
|
|
|
|
/// Create an Archive from a File.
|
|
pub fn init(fil: std.fs.File, offset: u64) !Archive {
|
|
var super: Superblock = undefined;
|
|
var fil_rdr = fil.reader(&[0]u8{});
|
|
if (offset > 0)
|
|
try fil_rdr.seekTo(offset);
|
|
try fil_rdr.interface.readSliceEndian(Superblock, @ptrCast(&super), .little);
|
|
try super.validate();
|
|
|
|
return .{
|
|
.super = super,
|
|
.stateless_decomp = .{ .vtable = &.{ .stateless = try DecompTypes.getStatelessFn(super.compression) } },
|
|
};
|
|
}
|
|
|
|
pub fn extract(self: Archive, alloc: std.mem.Allocator, path: []const u8, options: ExtractionOptions) !void {
|
|
_ = self;
|
|
_ = alloc;
|
|
_ = path;
|
|
_ = options;
|
|
return error.TODO;
|
|
}
|
|
|
|
// Superblock
|
|
|
|
const SQUASHFS_MAGIC: u32 = std.mem.readInt(u32, "hsqs", .little);
|
|
|
|
pub const Superblock = packed struct {
|
|
magic: u32,
|
|
inode_count: u32,
|
|
mod_time: u32,
|
|
block_size: u32,
|
|
frag_count: u32,
|
|
compression: DecompTypes.Enum,
|
|
block_log: u16,
|
|
flags: packed struct {
|
|
inode_uncompressed: bool,
|
|
data_uncompressed: bool,
|
|
check: bool,
|
|
frag_uncompressed: bool,
|
|
fragment_never: bool,
|
|
fragment_always: bool,
|
|
duplicates: bool,
|
|
exportable: bool,
|
|
xattr_uncompressed: bool,
|
|
xattr_never: bool,
|
|
compression_options: bool,
|
|
ids_uncompressed: bool,
|
|
_: u4,
|
|
},
|
|
id_count: u16,
|
|
ver_maj: u16,
|
|
ver_min: u16,
|
|
root_ref: Inode.Ref,
|
|
size: u64,
|
|
id_start: u64,
|
|
xattr_start: u64,
|
|
inode_start: u64,
|
|
dir_start: u64,
|
|
frag_start: u64,
|
|
export_start: u64,
|
|
|
|
/// Validate the Superblock. If an error is returned, it's likely the archive is corrupted or not a squashfs archive.
|
|
fn validate(self: Superblock) !void {
|
|
if (self.magic != SQUASHFS_MAGIC)
|
|
return Error.BadMagic;
|
|
if (self.flags.check)
|
|
return Error.BadCheck;
|
|
if (self.ver_maj != 4 or self.ver_min != 0)
|
|
return Error.BadVersion;
|
|
if (std.math.log2(self.block_size) != self.block_log)
|
|
return Error.BadBlockLog;
|
|
}
|
|
};
|