2024-11-28 08:36:58 -07:00
{ lib , config , pkgs , modulesPath , . . . }: {
options . ps4 = with lib ; {
enable = mkEnableOption " B u i l d N i x O S f o r S o n y P l a y S t a t i o n 4 . " ;
usbImage = {
imageName = mkOption {
type = types . str ;
default = " ${ config . ps4 . usbImage . imageBaseName } - ${ config . system . nixos . label } . i m g " ;
description = " N a m e o f t h e g e n e r a t e d i m a g e f i l e . " ;
} ;
imageBaseName = mkOption {
type = types . str ;
default = " p s 4 n i x o s " ;
description = " P r e f i x o f t h e n a m e o f t h e g e n e r a t e d i m a g e f i l e . " ;
} ;
compressImage = mkOption {
type = types . bool ;
default = false ;
description = " C o m p r e s s t h e r e s u l t i n g i m a g e w i t h z s t d . " ;
} ;
2024-11-29 06:19:46 -07:00
# TODO this is unnecessary. just make the partition big enough to fit the boot files automatically.
2024-11-28 08:36:58 -07:00
bootPartitionSize = mkOption {
type = types . int ;
2024-11-29 06:19:46 -07:00
default = 64 ;
2024-11-28 08:36:58 -07:00
description = " S i z e o f t h e b o o t p a r t i t i o n , i n m e b i b y t e s ( 1 0 2 4 x 1 0 2 4 b y t e s ) . " ;
} ;
bootPartitionLabel = mkOption {
type = types . str ;
default = " n i x o s - b o o t " ;
description = " L a b e l t o g i v e t h e b o o t p a r t i t i o n o f t h e i m a g e . " ;
} ;
rootPartitionLabel = mkOption {
type = types . str ;
default = " n i x o s - r o o t " ;
description = " L a b e l t o g i v e t h e r o o t p a r t i t i o n o f t h e i m a g e . " ;
} ;
expandOnBoot = mkOption {
type = types . bool ;
default = true ;
description = " W h e t h e r t h e i m a g e s h o u l d e x p a n d t h e r o o t p a r t i t i o n o n f i r s t b o o t . " ;
} ;
firstbootFile = mkOption {
type = types . str ;
default = " / f i r s t b o o t " ;
description = ''
Location of the file that allows commands to be run on first boot .
If overriding fileSystems . " / " then you should to set this to the root mount + /firstboot
'' ;
} ;
} ;
} ;
config = let cfg = config . ps4 ; in lib . mkIf cfg . enable {
2024-11-29 06:19:46 -07:00
# TODO write definitions for the other southbridges
boot . kernelPackages = pkgs . recurseIntoAttrs ( pkgs . linuxPackagesFor ( pkgs . callPackage ../kernel/kernel-aeolia-5.3.nix { } ) ) ;
2024-11-28 08:36:58 -07:00
fileSystems = {
" / b o o t " = {
device = " / d e v / d i s k / b y - l a b e l / ${ cfg . usbImage . bootPartitionLabel } " ;
fsType = " v f a t " ;
options = [ " n o f a i l " " n o a u t o " ] ;
} ;
" / " = {
device = " / d e v / d i s k / b y - l a b e l / ${ cfg . usbImage . rootPartitionLabel } " ;
fsType = " e x t 4 " ;
} ;
} ;
boot . loader . grub . enable = lib . mkForce false ;
2024-11-29 06:19:46 -07:00
boot . loader . initScript . enable = lib . mkForce true ;
2024-11-28 08:36:58 -07:00
# from https://github.com/NixOS/nixpkgs/blob/e405f30513169feedb64b5c25e7b00242010af58/nixos/modules/installer/sd-card/sd-image.nix#L267
boot . postBootCommands = let
expandOnBoot = lib . optionalString cfg . usbImage . expandOnBoot ''
# Figure out device names for the boot device and root filesystem.
rootPart = $ ( $ { pkgs . util-linux } /bin/findmnt - n - o SOURCE / )
bootDevice = $ ( lsblk - npo PKNAME $ rootPart )
partNum = $ ( lsblk - npo MAJ:MIN $ rootPart | $ { pkgs . gawk } /bin/awk - F : ' { print $ 2 } ' )
# Resize the root partition and the filesystem to fit the disk
echo " , + , " | sfdisk - N $ partNum - - no-reread $ bootDevice
$ { pkgs . parted } /bin/partprobe
$ { pkgs . e2fsprogs } /bin/resize2fs $ rootPart
'' ;
inherit ( cfg . usbImage ) firstbootFile ;
in ''
# On the first boot do some maintenance tasks
if [ - f $ { firstbootFile } ] ; then
set - euo pipefail
set - x
$ { expandOnBoot }
# TODO what does all this do?
# Register the contents of the initial Nix store
# ${config.nix.package.out}/bin/nix-store --load-db < ${firstbootFile}
# nixos-rebuild also requires a "system" profile and an /etc/NIXOS tag.
# touch /etc/NIXOS
$ { config . nix . package . out } /bin/nix-env - p /nix/var/nix/profiles/system - - set /run/current-system
# Prevents this from running on later boots.
rm - f $ { firstbootFile }
fi
'' ;
2024-11-29 06:19:46 -07:00
system . build . rootfsImage = pkgs . callPackage ( modulesPath + " / . . / l i b / m a k e - e x t 4 - f s . n i x " ) {
2024-11-28 08:36:58 -07:00
storePaths = [ config . system . build . toplevel ] ;
volumeLabel = cfg . usbImage . rootPartitionLabel ;
populateImageCommands = ''
cp $ { config . system . build . toplevel } /init ./files/init
touch ./files/firstboot
'' ;
} ;
2024-11-29 06:19:46 -07:00
# slightly modified version of https://github.com/NixOS/nixpkgs/blob/e405f30513169feedb64b5c25e7b00242010af58/nixos/modules/installer/sd-card/sd-image.nix#L182
system . build . usbImage = pkgs . stdenv . mkDerivation {
name = cfg . usbImage . imageName ;
nativeBuildInputs = with pkgs ; [ dosfstools parted mtools e2fsprogs util-linux ]
++ lib . optional cfg . usbImage . compressImage zstd ;
inherit ( cfg . usbImage ) compressImage ;
buildCommand = ''
# mkdir -p $out/usb-image
img = $ out
rootfs = $ { config . system . build . rootfsImage }
# Create the image to fit rootfs + bootfs + 1 MiB
rootfsSizeBlocks = $ ( du - B 512 - - apparent-size $ rootfs | awk ' { print $ 1 } ' )
bootfsSizeBlocks = $ ( ( $ { toString cfg . usbImage . bootPartitionSize } * 1024 * 1024 / 512 ) )
imageSize = $ ( ( rootfsSizeBlocks * 512 + bootfsSizeBlocks * 512 + 1 * 1024 * 1024 ) )
truncate - s $ imageSize $ img
# Partition the image
parted - s $ img - - mklabel msdos
parted - s $ img - - mkpart primary fat32 1 M $ { toString ( cfg . usbImage . bootPartitionSize + 1 ) } M
parted - s $ img - - mkpart primary ext4 $ { toString ( cfg . usbImage . bootPartitionSize + 1 ) } M 100 %
# Copy the rootfs to the image
eval $ ( partx $ img - o START , SECTORS - - nr 2 - - pairs )
dd conv = notrunc if = $ rootfs of = $ img seek = $ START count = $ SECTORS
# Create a temporary image for the boot partition
eval $ ( partx $ img - o START , SECTORS - - nr 1 - - pairs )
truncate - s $ ( ( SECTORS * 512 ) ) boot . img
mkfs . vfat - F 32 - n $ { cfg . usbImage . bootPartitionLabel } boot . img
# Copy the files that the PS4 Linux Loader payload expects to the boot partition
mcopy - i boot . img $ { config . system . build . kernel } /bzImage : : /bzImage
mcopy - i boot . img $ { config . system . build . initialRamdisk } /initrd.gz : : /initramfs.cpio.gz
# Verify the boot partition before copying it
fsck . vfat - vn boot . img
dd conv = notrunc if = boot . img of = $ img seek = $ START count = $ SECTORS
if test - n " $ c o m p r e s s I m a g e " ; then
zstd - T $ NIX_BUILD_CORES - - rm $ img
fi
'' ;
} ;
2024-11-28 08:36:58 -07:00
} ;
}