query
On this page

make-disk-image

make-disk-image

Docs pulled from | This Revision | about 1 hour ago


Contribute
Enhance the ecosystem with your expertise! Contribute to fill the gaps in documentation. Your input can make a difference.

Noogle detected

Implementation

The following is the current implementation of this function.

{
  pkgs,
  lib,

  # The NixOS configuration to be installed onto the disk image.
  config,

  # The size of the disk, in MiB (1024*1024 bytes).
  # if "auto" size is calculated based on the contents copied to it and
  #   additionalSpace is taken into account.
  diskSize ? "auto",

  # additional disk space to be added to the image if diskSize "auto"
  # is used
  additionalSpace ? "512M",

  # size of the boot partition, is only used if partitionTableType is
  # either "efi", "hybrid", or "legacy+boot"
  # This will be undersized slightly, as this is actually the offset of
  # the end of the partition. Generally it will be 1MiB smaller.
  bootSize ? "256M",

  # The files and directories to be placed in the target file system.
  # This is a list of attribute sets {source, target, mode, user, group} where
  # `source' is the file system object (regular file or directory) to be
  # grafted in the file system at path `target', `mode' is a string containing
  # the permissions that will be set (ex. "755"), `user' and `group' are the
  # user and group name that will be set as owner of the files.
  # `mode', `user', and `group' are optional.
  # When setting one of `user' or `group', the other needs to be set too.
  contents ? [ ],

  # Type of partition table to use; described in the `Image Partitioning` section above.
  partitionTableType ? "legacy",

  # Whether to invoke `switch-to-configuration boot` during image creation
  installBootLoader ? true,

  # Whether to output have EFIVARS available in $out/efi-vars.fd and use it during disk creation
  touchEFIVars ? false,

  # OVMF firmware derivation
  OVMF ? pkgs.OVMF.fd,

  # EFI firmware
  efiFirmware ? OVMF.firmware,

  # EFI variables
  efiVariables ? OVMF.variables,

  # The root file system type.
  fsType ? "ext4",

  # Filesystem label
  label ? if onlyNixStore then "nix-store" else "nixos",

  # The initial NixOS configuration file to be copied to
  # /etc/nixos/configuration.nix.
  configFile ? null,

  # Shell code executed after the VM has finished.
  postVM ? "",

  # Guest memory size in MiB (1024*1024 bytes)
  memSize ? 1024,

  # Copy the contents of the Nix store to the root of the image and
  # skip further setup. Incompatible with `contents`,
  # `installBootLoader` and `configFile`.
  onlyNixStore ? false,

  name ? "nixos-disk-image",

  # Disk image format, one of qcow2, qcow2-compressed, vdi, vpc, raw.
  format ? "raw",

  # Disk image filename, without any extensions (e.g. `image_1`).
  baseName ? "nixos",

  # Whether to fix:
  #   - GPT Disk Unique Identifier (diskGUID)
  #   - GPT Partition Unique Identifier: depends on the layout, root partition UUID can be controlled through `rootGPUID` option
  #   - GPT Partition Type Identifier: fixed according to the layout, e.g. ESP partition, etc. through `parted` invocation.
  #   - Filesystem Unique Identifier when fsType = ext4 for *root partition*.
  # BIOS/MBR support is "best effort" at the moment.
  # Boot partitions may not be deterministic.
  # Also, to fix last time checked of the ext4 partition if fsType = ext4.
  deterministic ? true,

  # GPT Partition Unique Identifier for root partition.
  rootGPUID ? "F222513B-DED1-49FA-B591-20CE86A2FE7F",
  # When fsType = ext4, this is the root Filesystem Unique Identifier.
  # TODO: support other filesystems someday.
  rootFSUID ? (if fsType == "ext4" then rootGPUID else null),

  # Whether a nix channel based on the current source tree should be
  # made available inside the image. Useful for interactive use of nix
  # utils, but changes the hash of the image when the sources are
  # updated.
  copyChannel ? true,

  # Additional store paths to copy to the image's store.
  additionalPaths ? [ ],
}:

assert (
  lib.assertOneOf "partitionTableType" partitionTableType [
    "legacy"
    "legacy+boot"
    "legacy+gpt"
    "efi"
    "efixbootldr"
    "hybrid"
    "none"
  ]
);
assert (
  lib.assertMsg (fsType == "ext4" && deterministic -> rootFSUID != null)
    "In deterministic mode with a ext4 partition, rootFSUID must be non-null, by default, it is equal to rootGPUID."
);
# We use -E offset=X below, which is only supported by e2fsprogs
assert (
  lib.assertMsg (partitionTableType != "none" -> fsType == "ext4")
    "to produce a partition table, we need to use -E offset flag which is support only for fsType = ext4"
);
assert (
  lib.assertMsg
    (
      touchEFIVars
      ->
        partitionTableType == "hybrid"
        || partitionTableType == "efi"
        || partitionTableType == "efixbootldr"
        || partitionTableType == "legacy+gpt"
    )
    "EFI variables can be used only with a partition table of type: hybrid, efi, efixbootldr, or legacy+gpt."
);
# If only Nix store image, then: contents must be empty, configFile must be unset, and we should no install bootloader.
assert (
  lib.assertMsg (onlyNixStore -> contents == [ ] && configFile == null && !installBootLoader)
    "In a only Nix store image, the contents must be empty, no configuration must be provided and no bootloader should be installed."
);
# Either both or none of {user,group} need to be set
assert (
  lib.assertMsg (lib.all (
    attrs: ((attrs.user or null) == null) == ((attrs.group or null) == null)
  ) contents) "Contents of the disk image should set none of {user, group} or both at the same time."
);

let
  format' = format;
in
let

  format = if format' == "qcow2-compressed" then "qcow2" else format';

  compress = lib.optionalString (format' == "qcow2-compressed") "-c";

  filename =
    "${baseName}."
    + {
      qcow2 = "qcow2";
      vdi = "vdi";
      vpc = "vhd";
      raw = "img";
    }
    .${format} or format;

  rootPartition =
    {
      # switch-case
      legacy = "1";
      "legacy+boot" = "2";
      "legacy+gpt" = "2";
      efi = "2";
      efixbootldr = "3";
      hybrid = "3";
    }
    .${partitionTableType};

  partitionDiskScript =
    {
      # switch-case
      legacy = ''
        parted --script $diskImage -- \
          mklabel msdos \
          mkpart primary ext4 1MiB 100% \
          print
      '';
      "legacy+boot" = ''
        parted --script $diskImage -- \
          mklabel msdos \
          mkpart primary fat32 1MiB $bootSizeMiB \
          set 1 boot on \
          mkpart primary ext4 $bootSizeMiB 100% \
          print
      '';
      "legacy+gpt" = ''
        parted --script $diskImage -- \
          mklabel gpt \
          mkpart no-fs 1MiB 2MiB \
          set 1 bios_grub on \
          mkpart primary ext4 2MiB 100% \
          align-check optimal 2 \
          print
        ${lib.optionalString deterministic ''
          sgdisk \
          --disk-guid=97FD5997-D90B-4AA3-8D16-C1723AEA73C \
          --partition-guid=1:1C06F03B-704E-4657-B9CD-681A087A2FDC \
          --partition-guid=2:970C694F-AFD0-4B99-B750-CDB7A329AB6F \
          --partition-guid=3:${rootGPUID} \
          $diskImage
        ''}
      '';
      efi = ''
        parted --script $diskImage -- \
          mklabel gpt \
          mkpart ESP fat32 8MiB $bootSizeMiB \
          set 1 boot on \
          align-check optimal 1 \
          mkpart primary ext4 $bootSizeMiB 100% \
          align-check optimal 2 \
          print
        ${lib.optionalString deterministic ''
          sgdisk \
          --disk-guid=97FD5997-D90B-4AA3-8D16-C1723AEA73C \
          --partition-guid=1:1C06F03B-704E-4657-B9CD-681A087A2FDC \
          --partition-guid=2:${rootGPUID} \
          $diskImage
        ''}
      '';
      efixbootldr = ''
        parted --script $diskImage -- \
          mklabel gpt \
          mkpart ESP fat32 8MiB 100MiB \
          set 1 boot on \
          align-check optimal 1 \
          mkpart BOOT fat32 100MiB $bootSizeMiB \
          set 2 bls_boot on \
          align-check optimal 2 \
          mkpart ROOT ext4 $bootSizeMiB 100% \
          align-check optimal 3 \
          print
        ${lib.optionalString deterministic ''
          sgdisk \
          --disk-guid=97FD5997-D90B-4AA3-8D16-C1723AEA73C \
          --partition-guid=1:1C06F03B-704E-4657-B9CD-681A087A2FDC  \
          --partition-guid=2:970C694F-AFD0-4B99-B750-CDB7A329AB6F  \
          --partition-guid=3:${rootGPUID} \
          $diskImage
        ''}
      '';
      hybrid = ''
        parted --script $diskImage -- \
          mklabel gpt \
          mkpart ESP fat32 8MiB $bootSizeMiB \
          set 1 boot on \
          align-check optimal 1 \
          mkpart no-fs 0 1024KiB \
          set 2 bios_grub on \
          mkpart primary ext4 $bootSizeMiB 100% \
          align-check optimal 3 \
          print
        ${lib.optionalString deterministic ''
          sgdisk \
          --disk-guid=97FD5997-D90B-4AA3-8D16-C1723AEA73C \
          --partition-guid=1:1C06F03B-704E-4657-B9CD-681A087A2FDC \
          --partition-guid=2:970C694F-AFD0-4B99-B750-CDB7A329AB6F \
          --partition-guid=3:${rootGPUID} \
          $diskImage
        ''}
      '';
      none = "";
    }
    .${partitionTableType};

  useEFIBoot = touchEFIVars;

  nixpkgs = lib.cleanSource pkgs.path;

  # FIXME: merge with channel.nix / make-channel.nix.
  channelSources = pkgs.runCommand "nixos-${config.system.nixos.version}" { } ''
    mkdir -p $out
    cp -prd ${nixpkgs.outPath} $out/nixos
    chmod -R u+w $out/nixos
    if [ ! -e $out/nixos/nixpkgs ]; then
      ln -s . $out/nixos/nixpkgs
    fi
    rm -rf $out/nixos/.git
    echo -n ${config.system.nixos.versionSuffix} > $out/nixos/.version-suffix
  '';

  binPath = lib.makeBinPath (
    with pkgs;
    [
      rsync
      util-linux
      parted
      e2fsprogs
      lkl
      config.system.build.nixos-install
      nixos-enter
      nix
      systemdMinimal
    ]
    ++ lib.optional deterministic gptfdisk
    ++ stdenv.initialPath
  );

  # I'm preserving the line below because I'm going to search for it across nixpkgs to consolidate
  # image building logic. The comment right below this now appears in 4 different places in nixpkgs :)
  # !!! should use XML.
  sources = map (x: x.source) contents;
  targets = map (x: x.target) contents;
  modes = map (x: x.mode or "''") contents;
  users = map (x: x.user or "''") contents;
  groups = map (x: x.group or "''") contents;

  basePaths = [ config.system.build.toplevel ] ++ lib.optional copyChannel channelSources;

  additionalPaths' = lib.subtractLists basePaths additionalPaths;

  closureInfo = pkgs.closureInfo {
    rootPaths = basePaths ++ additionalPaths';
  };

  blockSize = toString (4 * 1024); # ext4fs block size (not block device sector size)

  prepareImage = ''
    export PATH=${binPath}

    # Yes, mkfs.ext4 takes different units in different contexts. Fun.
    sectorsToKilobytes() {
      echo $(( ( "$1" * 512 ) / 1024 ))
    }

    sectorsToBytes() {
      echo $(( "$1" * 512  ))
    }

    # Given lines of numbers, adds them together
    sum_lines() {
      local acc=0
      while read -r number; do
        acc=$((acc+number))
      done
      echo "$acc"
    }

    mebibyte=$(( 1024 * 1024 ))

    # Approximative percentage of reserved space in an ext4 fs over 512MiB.
    # 0.05208587646484375
    #  × 1000, integer part: 52
    compute_fudge() {
      echo $(( $1 * 52 / 1000 ))
    }

    round_to_nearest() {
      echo $(( ( $1 / $2 + 1) * $2 ))
    }

    mkdir $out

    root="$PWD/root"
    mkdir -p $root

    # Copy arbitrary other files into the image
    # Semi-shamelessly copied from make-etc.sh.
    set -f
    sources_=(${lib.concatStringsSep " " sources})
    targets_=(${lib.concatStringsSep " " targets})
    modes_=(${lib.concatStringsSep " " modes})
    set +f

    for ((i = 0; i < ''${#targets_[@]}; i++)); do
      source="''${sources_[$i]}"
      target="''${targets_[$i]}"
      mode="''${modes_[$i]}"

      if [ -n "$mode" ]; then
        rsync_chmod_flags="--chmod=$mode"
      else
        rsync_chmod_flags=""
      fi
      # Unfortunately cptofs only supports modes, not ownership, so we can't use
      # rsync's --chown option. Instead, we change the ownerships in the
      # VM script with chown.
      rsync_flags="-a --no-o --no-g $rsync_chmod_flags"
      if [[ "$source" =~ '*' ]]; then
        # If the source name contains '*', perform globbing.
        mkdir -p $root/$target
        for fn in $source; do
          rsync $rsync_flags "$fn" $root/$target/
        done
      else
        mkdir -p $root/$(dirname $target)
        if [ -e $root/$target ]; then
          echo "duplicate entry $target -> $source"
          exit 1
        elif [ -d $source ]; then
          # Append a slash to the end of source to get rsync to copy the
          # directory _to_ the target instead of _inside_ the target.
          # (See `man rsync`'s note on a trailing slash.)
          rsync $rsync_flags $source/ $root/$target
        else
          rsync $rsync_flags $source $root/$target
        fi
      fi
    done

    export HOME=$TMPDIR

    # Provide a Nix database so that nixos-install can copy closures.
    export NIX_STATE_DIR=$TMPDIR/state
    nix-store --load-db < ${closureInfo}/registration

    chmod 755 "$TMPDIR"
    echo "running nixos-install..."
    nixos-install --root $root --no-bootloader --no-root-passwd \
      --system ${config.system.build.toplevel} \
      ${if copyChannel then "--channel ${channelSources}" else "--no-channel-copy"} \
      --substituters ""

    ${lib.optionalString (additionalPaths' != [ ]) ''
      nix --extra-experimental-features nix-command copy --to $root --no-check-sigs ${lib.concatStringsSep " " additionalPaths'}
    ''}

    diskImage=nixos.raw

    bootSize=$(round_to_nearest $(numfmt --from=iec '${bootSize}') $mebibyte)
    bootSizeMiB=$(( bootSize / 1024 / 1024 ))MiB

    ${
      if diskSize == "auto" then
        ''
          ${
            if
              partitionTableType == "efi" || partitionTableType == "efixbootldr" || partitionTableType == "hybrid"
            then
              ''
                # Add the GPT at the end
                gptSpace=$(( 512 * 34 * 1 ))
                # Normally we'd need to account for alignment and things, if bootSize
                # represented the actual size of the boot partition. But it instead
                # represents the offset at which it ends.
                # So we know bootSize is the reserved space in front of the partition.
                reservedSpace=$(( gptSpace + bootSize ))
              ''
            else if partitionTableType == "legacy+gpt" then
              ''
                # Add the GPT at the end
                gptSpace=$(( 512 * 34 * 1 ))
                # And include the bios_grub partition; the ext4 partition starts at 2MiB exactly.
                reservedSpace=$(( gptSpace + 2 * mebibyte ))
              ''
            else if partitionTableType == "legacy" then
              ''
                # Add the 1MiB aligned reserved space (includes MBR)
                reservedSpace=$(( mebibyte ))
              ''
            else if partitionTableType == "legacy+boot" then
              ''
                # The explanation from the above "efi" case applies here too,
                # but gptSpace is not needed without a GPT.
                reservedSpace=$(( bootSize ))
              ''
            else
              ''
                reservedSpace=0
              ''
          }
          additionalSpace=$(( $(numfmt --from=iec '${additionalSpace}') + reservedSpace ))

          # Compute required space in filesystem blocks
          diskUsage=$(find . ! -type d -print0 | du --files0-from=- --apparent-size --count-links --block-size "${blockSize}" | cut -f1 | sum_lines)
          # Each inode takes space!
          numInodes=$(find . | wc -l)
          # Convert to bytes, inodes take two blocks each!
          diskUsage=$(( (diskUsage + 2 * numInodes) * ${blockSize} ))
          # Then increase the required space to account for the reserved blocks.
          fudge=$(compute_fudge $diskUsage)
          requiredFilesystemSpace=$(( diskUsage + fudge ))

          # Round up to the nearest block size.
          # This ensures whole $blockSize bytes block sizes in the filesystem
          # and helps towards aligning partitions optimally.
          requiredFilesystemSpace=$(round_to_nearest $requiredFilesystemSpace ${blockSize})

          diskSize=$(( requiredFilesystemSpace + additionalSpace ))

          # Round up to the nearest mebibyte.
          # This ensures whole 512 bytes sector sizes in the disk image
          # and helps towards aligning partitions optimally.
          diskSize=$(round_to_nearest $diskSize $mebibyte)

          truncate -s "$diskSize" $diskImage

          printf "Automatic disk size...\n"
          printf "  Closure space use: %d bytes\n" $diskUsage
          printf "  fudge: %d bytes\n" $fudge
          printf "  Filesystem size needed: %d bytes\n" $requiredFilesystemSpace
          printf "  Additional space: %d bytes\n" $additionalSpace
          printf "  Disk image size: %d bytes\n" $diskSize
        ''
      else
        ''
          truncate -s ${toString diskSize}M $diskImage
        ''
    }

    ${partitionDiskScript}

    ${
      if partitionTableType != "none" then
        ''
          # Get start & length of the root partition in sectors to $START and $SECTORS.
          eval $(partx $diskImage -o START,SECTORS --nr ${rootPartition} --pairs)

          mkfs.${fsType} -b ${blockSize} -F -L ${label} $diskImage -E offset=$(sectorsToBytes $START) $(sectorsToKilobytes $SECTORS)K
        ''
      else
        ''
          mkfs.${fsType} -b ${blockSize} -F -L ${label} $diskImage
        ''
    }

    echo "copying staging root to image..."
    cptofs -p ${lib.optionalString (partitionTableType != "none") "-P ${rootPartition}"} \
           -t ${fsType} \
           -i $diskImage \
           $root${lib.optionalString onlyNixStore builtins.storeDir}/* / ||
      (echo >&2 "ERROR: cptofs failed. diskSize might be too small for closure."; exit 1)
  '';

  moveOrConvertImage = ''
    ${
      if format == "raw" then
        ''
          mv $diskImage $out/${filename}
        ''
      else
        ''
          ${pkgs.qemu-utils}/bin/qemu-img convert -f raw -O ${format} ${compress} $diskImage $out/${filename}
        ''
    }
    diskImage=$out/${filename}
  '';

  createEFIVars = ''
    efiVars=$out/efi-vars.fd
    cp ${efiVariables} $efiVars
    chmod 0644 $efiVars
  '';

  createHydraBuildProducts = ''
    mkdir -p $out/nix-support
    echo "file ${format}-image $out/${filename}" >> $out/nix-support/hydra-build-products
  '';

  buildImage = pkgs.vmTools.runInLinuxVM (
    pkgs.runCommand name
      {
        preVM = prepareImage + lib.optionalString touchEFIVars createEFIVars;
        buildInputs = with pkgs; [
          util-linux
          e2fsprogs
          dosfstools
        ];
        postVM = moveOrConvertImage + createHydraBuildProducts + postVM;
        QEMU_OPTS = lib.concatStringsSep " " (
          lib.optional useEFIBoot "-drive if=pflash,format=raw,unit=0,readonly=on,file=${efiFirmware}"
          ++ lib.optionals touchEFIVars [
            "-drive if=pflash,format=raw,unit=1,file=$efiVars"
          ]
          ++ lib.optionals (OVMF.systemManagementModeRequired or false) [
            "-machine"
            "q35,smm=on"
            "-global"
            "driver=cfi.pflash01,property=secure,value=on"
          ]
        );
        inherit memSize;
      }
      ''
        export PATH=${binPath}:$PATH

        rootDisk=${if partitionTableType != "none" then "/dev/vda${rootPartition}" else "/dev/vda"}

        # It is necessary to set root filesystem unique identifier in advance, otherwise
        # bootloader might get the wrong one and fail to boot.
        # At the end, we reset again because we want deterministic timestamps.
        ${lib.optionalString (fsType == "ext4" && deterministic) ''
          tune2fs -T now ${lib.optionalString deterministic "-U ${rootFSUID}"} -c 0 -i 0 $rootDisk
        ''}
        # make systemd-boot find ESP without udev
        mkdir /dev/block
        ln -s /dev/vda1 /dev/block/254:1

        mountPoint=/mnt
        mkdir $mountPoint
        mount $rootDisk $mountPoint

        # Create the ESP and mount it. Unlike e2fsprogs, mkfs.vfat doesn't support an
        # '-E offset=X' option, so we can't do this outside the VM.
        ${lib.optionalString (partitionTableType == "efi" || partitionTableType == "hybrid") ''
          mkdir -p /mnt/boot
          mkfs.vfat -n ESP /dev/vda1
          mount /dev/vda1 /mnt/boot

          ${lib.optionalString touchEFIVars "mount -t efivarfs efivarfs /sys/firmware/efi/efivars"}
        ''}
        ${lib.optionalString (partitionTableType == "efixbootldr") ''
          mkdir -p /mnt/{boot,efi}
          mkfs.vfat -n ESP /dev/vda1
          mkfs.vfat -n BOOT /dev/vda2
          mount /dev/vda1 /mnt/efi
          mount /dev/vda2 /mnt/boot

          ${lib.optionalString touchEFIVars "mount -t efivarfs efivarfs /sys/firmware/efi/efivars"}
        ''}
        ${lib.optionalString (partitionTableType == "legacy+boot") ''
          mkdir -p /mnt/boot
          mkfs.vfat -n BOOT /dev/vda1
          mount /dev/vda1 /mnt/boot
        ''}

        # Install a configuration.nix
        mkdir -p /mnt/etc/nixos
        ${lib.optionalString (configFile != null) ''
          cp ${configFile} /mnt/etc/nixos/configuration.nix
        ''}

        ${lib.optionalString installBootLoader ''
          # In this throwaway resource, we only have /dev/vda, but the actual VM may refer to another disk for bootloader, e.g. /dev/vdb
          # Use this option to create a symlink from vda to any arbitrary device you want.
          ${lib.optionalString (config.boot.loader.grub.enable) (
            lib.concatMapStringsSep " " (
              device:
              lib.optionalString (device != "/dev/vda") ''
                mkdir -p "$(dirname ${device})"
                ln -s /dev/vda ${device}
              ''
            ) config.boot.loader.grub.devices
          )}
          ${
            let
              limine = config.boot.loader.limine;
            in
            lib.optionalString (limine.enable && limine.biosSupport && limine.biosDevice != "/dev/vda") ''
              mkdir -p "$(dirname ${limine.biosDevice})"
              ln -s /dev/vda ${limine.biosDevice}
            ''
          }

          # Set up core system link, bootloader (sd-boot, GRUB, uboot, etc.), etc.

          # NOTE: systemd-boot-builder.py calls nix-env --list-generations which
          # clobbers $HOME/.nix-defexpr/channels/nixos This would cause a  folder
          # /homeless-shelter to show up in the final image which  in turn breaks
          # nix builds in the target image if sandboxing is turned off (through
          # __noChroot for example).
          export HOME=$TMPDIR
          NIXOS_INSTALL_BOOTLOADER=1 nixos-enter --root $mountPoint -- /nix/var/nix/profiles/system/bin/switch-to-configuration boot
        ''}

        # Set the ownerships of the contents. The modes are set in preVM.
        # No globbing on targets, so no need to set -f
        targets_=(${lib.concatStringsSep " " targets})
        users_=(${lib.concatStringsSep " " users})
        groups_=(${lib.concatStringsSep " " groups})
        for ((i = 0; i < ''${#targets_[@]}; i++)); do
          target="''${targets_[$i]}"
          user="''${users_[$i]}"
          group="''${groups_[$i]}"
          if [ -n "$user$group" ]; then
            # We have to nixos-enter since we need to use the user and group of the VM
            nixos-enter --root $mountPoint -- chown -R "$user:$group" "$target"
          fi
        done

        umount -R /mnt

        # Make sure resize2fs works. Note that resize2fs has stricter criteria for resizing than a normal
        # mount, so the `-c 0` and `-i 0` don't affect it. Setting it to `now` doesn't produce deterministic
        # output, of course, but we can fix that when/if we start making images deterministic.
        # In deterministic mode, this is fixed to 1970-01-01 (UNIX timestamp 0).
        # This two-step approach is necessary otherwise `tune2fs` will want a fresher filesystem to perform
        # some changes.
        ${lib.optionalString (fsType == "ext4") ''
          tune2fs -T now ${lib.optionalString deterministic "-U ${rootFSUID}"} -c 0 -i 0 $rootDisk
          ${lib.optionalString deterministic "tune2fs -f -T 19700101 $rootDisk"}
        ''}
      ''
  );
in
if onlyNixStore then
  pkgs.runCommand name { } (prepareImage + moveOrConvertImage + createHydraBuildProducts + postVM)
else
  buildImage