Skip to content

Commit 7126b27

Browse files
author
Dancin Forever
committed
lindbergh: squashfs support (via an overlay with write access)
1 parent 4e59b07 commit 7126b27

File tree

5 files changed

+78
-30
lines changed

5 files changed

+78
-30
lines changed

batocera-Changelog.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -29,7 +29,7 @@
2929
- Steering wheel support added for :
3030
- Logitech: G923 (Xbox), PRO Racing Wheel
3131
- Speedlink: 4in1 Leather Power Feedback Wheel
32-
- Sega Lindbergh loader
32+
- Sega Lindbergh loader with .squashfs support
3333
- Variable Refresh Rate (VRR) support for modern AMD gpus
3434
- Support of Shanwan Twin USB Joystick (new revision)
3535
- Libretro-PS2 core

package/batocera/core/batocera-configgen/configgen/configgen/emulatorlauncher.py

100644100755
+2-1
Original file line numberDiff line numberDiff line change
@@ -44,7 +44,8 @@
4444
def main(args: argparse.Namespace, maxnbplayers: int) -> int:
4545
# squashfs roms if squashed
4646
if args.rom.suffix == ".squashfs":
47-
with squashfs_rom(args.rom) as rom:
47+
writable_rom = (args.system == "lindbergh")
48+
with squashfs_rom(args.rom, overlay=writable_rom) as rom:
4849
return start_rom(args, maxnbplayers, rom, args.rom)
4950
else:
5051
return start_rom(args, maxnbplayers, args.rom, args.rom)

package/batocera/core/batocera-configgen/configgen/configgen/generators/lindbergh/lindberghGenerator.py

+5
Original file line numberDiff line numberDiff line change
@@ -67,6 +67,11 @@ def getHotkeysContext(self) -> HotkeysContext:
6767
}
6868

6969
def generate(self, system, rom, playersControllers, metadata, guns, wheels, gameResolution):
70+
# .squashfs returns mount point so rewire rom to first .game file found
71+
# https://stackoverflow.com/a/43669828/9983389
72+
if (rom.is_dir()):
73+
rom = next(rom.rglob("*.game"))
74+
7075
romDir = rom.parent
7176
romName = rom.name
7277
_logger.debug("ROM path: %s", romDir)
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,8 @@
11
from __future__ import annotations
22

33
import logging
4+
import os
5+
import shutil
46
import subprocess
57
from contextlib import contextmanager
68
from pathlib import Path
@@ -16,28 +18,63 @@
1618

1719
_SQUASHFS_DIR: Final = Path("/var/run/squashfs/")
1820

21+
def _squashfs_removemount(mount_point: Path):
22+
if mount_point.exists():
23+
if mount_point.is_mount():
24+
return_code = subprocess.call(["umount", mount_point])
25+
if return_code != 0:
26+
_logger.debug("squashfs_rom: unmounting %s failed", mount_point)
27+
raise BatoceraException(f"Unable to unmount the file {mount_point}")
28+
mount_point.rmdir()
29+
30+
def _squashfs_rom_cleanup(mount_point: Path, overlay: bool):
31+
_logger.debug("squashfs_rom: cleaning up %s", mount_point)
32+
os.chdir(_SQUASHFS_DIR)
33+
_squashfs_removemount(mount_point)
34+
35+
if overlay:
36+
overlay_root = mount_point.parent
37+
mount_point_rw = overlay_root / "overlay"
38+
39+
if overlay_root.exists():
40+
_logger.debug("squashfs_rom: cleaning up overlay root %s", overlay_root)
41+
_squashfs_removemount(mount_point_rw)
42+
shutil.rmtree(overlay_root)
1943

2044
@contextmanager
21-
def squashfs_rom(rom: Path, /) -> Iterator[Path]:
45+
def squashfs_rom(rom: Path, overlay: bool) -> Iterator[Path]:
2246
_logger.debug("squashfs_rom(%s)", rom)
23-
mount_point = _SQUASHFS_DIR / rom.stem
2447

2548
mkdir_if_not_exists(_SQUASHFS_DIR)
2649

27-
# first, try to clean an empty remaining directory (for example because of a crash)
28-
if mount_point.exists() and mount_point.is_dir():
29-
_logger.debug("squashfs_rom: %s already exists", mount_point)
30-
# try to remove an empty directory, else, run the directory, ignoring the .squashfs
31-
try:
32-
mount_point.rmdir()
33-
except (FileNotFoundError, OSError):
34-
_logger.debug("squashfs_rom: failed to rmdir %s", mount_point)
35-
yield mount_point
36-
# No cleanup is necessary
37-
return
50+
if overlay:
51+
overlay_root = _SQUASHFS_DIR / rom.stem
52+
mount_point = overlay_root / "rom"
53+
overlay_delta = overlay_root / "delta"
54+
overlay_tmp = overlay_root / "tmp"
55+
mount_point_rw = overlay_root / "overlay"
56+
57+
if overlay_root.exists() and overlay_root.is_dir():
58+
_logger.debug("squashfs_rom: %s overlay root already exists", overlay_root)
59+
_squashfs_rom_cleanup(mount_point, overlay)
60+
else:
61+
mount_point = _SQUASHFS_DIR / rom.stem
62+
63+
# first, try to clean an empty remaining directories (for example because of a crash)
64+
if mount_point.exists() and mount_point.is_dir():
65+
_logger.debug("squashfs_rom: %s already exists", mount_point)
66+
67+
# try to remove an empty directory, else, run the directory, ignoring the .squashfs
68+
try:
69+
mount_point.rmdir()
70+
except (FileNotFoundError, OSError):
71+
_logger.debug("squashfs_rom: failed to rmdir %s", mount_point)
72+
yield mount_point
73+
# No cleanup is necessary
74+
return
3875

3976
# ok, the base directory doesn't exist, let's create it and mount the squashfs on it
40-
mount_point.mkdir()
77+
mount_point.mkdir(parents=True)
4178

4279
return_code = subprocess.call(["mount", rom, mount_point])
4380
if return_code != 0:
@@ -48,28 +85,33 @@ def squashfs_rom(rom: Path, /) -> Iterator[Path]:
4885
pass
4986
raise BatoceraException(f"Unable to mount the file {rom}")
5087

88+
if overlay:
89+
for dir in [overlay_delta, overlay_tmp, mount_point_rw]: dir.mkdir(exist_ok=True)
90+
return_code = subprocess.call(["mount",
91+
"-t", "overlay",
92+
"overlay",
93+
"-o", f"lowerdir={mount_point},upperdir={overlay_delta},workdir={overlay_tmp}",
94+
mount_point_rw])
95+
if return_code != 0:
96+
_logger.debug("squashfs_rom: mounting overlay %s failed", mount_point_rw)
97+
_squashfs_rom_cleanup(mount_point, overlay)
98+
raise BatoceraException(f"Unable to mount the file {rom}")
99+
51100
try:
101+
working_mount_point = mount_point_rw if overlay else mount_point
102+
52103
# if the squashfs contains a single file with the same name, take it as the rom file
53-
rom_single = mount_point / rom.stem
104+
rom_single = working_mount_point / rom.stem
54105
if len(list(mount_point.iterdir())) == 1 and rom_single.exists():
55106
_logger.debug("squashfs: single rom %s", rom_single)
56107
yield rom_single
57108
else:
58109
try:
59-
rom_linked = (mount_point / ".ROM").resolve(strict=True)
110+
rom_linked = (working_mount_point / ".ROM").resolve(strict=True)
60111
except OSError:
61-
yield mount_point
112+
yield working_mount_point
62113
else:
63114
_logger.debug("squashfs: linked rom %s", rom_linked)
64115
yield rom_linked
65116
finally:
66-
_logger.debug("squashfs_rom: cleaning up %s", mount_point)
67-
68-
# unmount
69-
return_code = subprocess.call(["umount", mount_point])
70-
if return_code != 0:
71-
_logger.debug("squashfs_rom: unmounting %s failed", mount_point)
72-
raise BatoceraException(f"Unable to unmount the file {mount_point}")
73-
74-
# cleaning the empty directory
75-
mount_point.rmdir()
117+
_squashfs_rom_cleanup(mount_point, overlay)

package/batocera/emulationstation/batocera-es-system/es_systems.yml

+1-1
Original file line numberDiff line numberDiff line change
@@ -5651,7 +5651,7 @@ lindbergh:
56515651
manufacturer: Sega
56525652
release: 2005
56535653
hardware: arcade
5654-
extensions: [game]
5654+
extensions: [game, squashfs]
56555655
platform: lindbergh, arcade
56565656
emulators:
56575657
lindbergh-loader:

0 commit comments

Comments
 (0)