-
Notifications
You must be signed in to change notification settings - Fork 121
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
SquashFS does not set the right filesize for directory inode #285
Comments
9 tasks
Use the go file to create squashfs image package main
import (
"errors"
"fmt"
"os"
"github.com/diskfs/go-diskfs"
"github.com/diskfs/go-diskfs/disk"
"github.com/diskfs/go-diskfs/filesystem"
"github.com/diskfs/go-diskfs/filesystem/squashfs"
)
func main() {
CreateFilesystem := func(d *disk.Disk, spec disk.FilesystemSpec) (filesystem.FileSystem, error) {
// find out where the partition starts and ends, or if it is the entire disk
var (
size, start int64
)
switch {
case spec.Partition == 0:
size = d.Size
start = 0
case d.Table == nil:
return nil, fmt.Errorf("cannot create filesystem on a partition without a partition table")
default:
partitions := d.Table.GetPartitions()
// API indexes from 1, but slice from 0
part := spec.Partition - 1
if spec.Partition > len(partitions) {
return nil, fmt.Errorf("cannot create filesystem on partition %d greater than maximum partition %d", spec.Partition, len(partitions))
}
size = partitions[part].GetSize()
start = partitions[part].GetStart()
}
switch spec.FSType {
case filesystem.TypeSquashfs:
return squashfs.Create(d.Backend, size, start, d.LogicalBlocksize)
default:
return nil, errors.New("unknown filesystem type requested")
}
}
// Create squashfs image file
var diskSize int64 = 10 * 1024 * 1024 // 10 MB
initdataImagePath := "/tmp/data.img"
mydisk, err := diskfs.Create(initdataImagePath, diskSize, 131072)
if err != nil {
panic(err)
}
fspec := disk.FilesystemSpec{Partition: 0, FSType: filesystem.TypeSquashfs, VolumeLabel: "label"}
fs, err := CreateFilesystem(mydisk, fspec)
if err != nil {
panic(err)
}
err = fs.Mkdir("/")
if err != nil {
panic(err)
}
err = fs.Mkdir("/123")
if err != nil {
panic(err)
}
err = fs.Mkdir("/foo")
if err != nil {
panic(err)
}
err = fs.Mkdir("/")
if err != nil {
panic(err)
}
rw, err := fs.OpenFile("example.txt", os.O_CREATE|os.O_RDWR)
if err != nil {
panic(err)
}
_, err = rw.Write([]byte("Hello, SquashFS!"))
if err != nil {
panic(err)
}
sqs, ok := fs.(*squashfs.FileSystem)
if !ok {
panic("not a squashfs filesystem")
}
err = sqs.Finalize(squashfs.FinalizeOptions{})
if err != nil {
panic(err)
}
} Using the following rust program to read
use std::io::BufReader;
use backhand::FilesystemReader;
fn main() {
let initdata_devfile = std::fs::File::open("/tmp/data.img").unwrap();
let buf_reader = BufReader::new(initdata_devfile);
let read_filesystem = FilesystemReader::from_reader(buf_reader).unwrap();
for node in read_filesystem.files() {
println!("path: {:?}", node.fullpath.as_path());
}
}
[package]
name = "test-squashfs"
version = "0.1.0"
edition = "2021"
[dependencies]
backhand = "0.20.0" The terminal will only say
|
Xynnn007
added a commit
to Xynnn007/go-diskfs
that referenced
this issue
Mar 6, 2025
Due to SquashFS spec https://dr-emann.github.io/squashfs/squashfs.html#_directory_inodes Section 5.2, it says about directory inode file size ''' This value is 3 bytes larger than the real listing. The Linux kernel creates "." and ".." entries for offsets 0 and 1, and only after 3 looks into the listing, subtracting 3 from the size. ''' In go-disks the size is calculated by serialize the 'directory'. A directory serialization includes multiple 'directoryHeader's and 'directoryEntryRaw's. This aligns with the dir_size calculation logic with well-known squasjfs-tools project https://github.com/plougher/squashfs-tools/blob/master/squashfs-tools/mksquashfs.c#L1549 Note that the file_size is set to 3 larger than the size. Thus here, we set directoryLocation.size to 3 bytes larger and the new value will be used to set basicDirectory.fileSize. Fixes diskfs#285 Signed-off-by: Xynnn007 <[email protected]>
Ah, and it would not be a problem reading, because there it just takes the data as it finds it. Rather nice catch. |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment
Due to SquashFS Spec https://dr-emann.github.io/squashfs/squashfs.html#_directory_inodes Section 5.2, the Directory Inode should have its
file size
3 bytes larger than the real listing.SquashFS implementation does not follow this well. Some bugs would come when using
go-diskfs
to generate a squashfs image with a file in it, but reading it using a different tool rather thango-diskfs
like backhand in Rust.Let me put an example for this then.
The text was updated successfully, but these errors were encountered: