Skip to content
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

Closed
Xynnn007 opened this issue Mar 4, 2025 · 2 comments · Fixed by #286
Closed

SquashFS does not set the right filesize for directory inode #285

Xynnn007 opened this issue Mar 4, 2025 · 2 comments · Fixed by #286

Comments

@Xynnn007
Copy link
Contributor

Xynnn007 commented Mar 4, 2025

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 than go-diskfs like backhand in Rust.

Let me put an example for this then.

@Xynnn007
Copy link
Contributor Author

Xynnn007 commented Mar 6, 2025

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

src/main.rs

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());
    }
}

Cargo.toml

[package]
name = "test-squashfs"
version = "0.1.0"
edition = "2021"

[dependencies]
backhand = "0.20.0"

The terminal will only say

path: /

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]>
@deitch
Copy link
Collaborator

deitch commented Mar 6, 2025

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
Labels
None yet
Projects
None yet
Development

Successfully merging a pull request may close this issue.

2 participants