mirror of
https://github.com/NixOS/mobile-nixos.git
synced 2024-12-15 19:23:01 +03:00
2a366b2cae
Though this is a bit buggy and less than ideal. A better solution would be a custom wrapper that only handles redirecting. This is, anyway, something that would be required to redirect all logging to all consoles.
169 lines
5.2 KiB
Nix
169 lines
5.2 KiB
Nix
{ config, pkgs, lib, ... }:
|
|
|
|
let
|
|
inherit (pkgs)
|
|
busybox
|
|
makeInitrd
|
|
mkExtraUtils
|
|
runCommandNoCC
|
|
udev
|
|
;
|
|
inherit (lib) flatten optionals optionalString;
|
|
|
|
initWrapperRealInit = "/actual-init";
|
|
|
|
# TODO: define as an option
|
|
# This is a bit buggy:
|
|
# * fast burst of \n-delimited output will not work as expected
|
|
# * `printk.devkmsg=on` is required on the kernel cmdline for better results
|
|
# A better implementation would be to have a binary who's sole purpose is to
|
|
# duplicate the stdout/stderr to /dev/kmsg while still outputting them to
|
|
# stdout/stderr as they do currently.
|
|
#
|
|
# Reminder: redirecting to kmsg is useful *mainly* for getting data through
|
|
# console_ramoops on devices without serial and without any other means to
|
|
# get the initial data out.
|
|
withKmsg = false;
|
|
|
|
# TODO: define as an option
|
|
withStrace = false;
|
|
|
|
initWrapperEnabled = withKmsg || withStrace;
|
|
|
|
device_config = config.mobile.device;
|
|
device_name = device_config.name;
|
|
|
|
stage-1 = config.mobile.boot.stage-1;
|
|
|
|
mobile-nixos-init = pkgs.pkgsStatic.callPackage ../boot/init {};
|
|
init = "${mobile-nixos-init}/bin/init";
|
|
|
|
shell = "${extraUtils}/bin/sh";
|
|
debugInit = pkgs.writeScript "debug-init" ''
|
|
#!${shell}
|
|
|
|
echo
|
|
echo "***************************************"
|
|
echo "* Mobile NixOS stage-0 script wrapper *"
|
|
echo "***************************************"
|
|
echo
|
|
|
|
PS4="=> "
|
|
set -x
|
|
|
|
export LD_LIBRARY_PATH="${extraUtils}/lib"
|
|
|
|
${optionalString withKmsg ''
|
|
${extraUtils}/bin/mknod /.kmsg c 1 11
|
|
exec > /.kmsg 2>&1
|
|
''}
|
|
|
|
exec ${optionalString withStrace "${extraUtils}/bin/strace -f"} ${initWrapperRealInit}
|
|
'';
|
|
|
|
contents =
|
|
(optionals (stage-1 ? contents) (flatten stage-1.contents))
|
|
++ [
|
|
# Populate /bin/sh to stay POSIXLY compliant.
|
|
# FIXME: Do we care?
|
|
#{ object = "${extraUtils}/bin/sh"; symlink = "/bin/sh"; }
|
|
|
|
# FIXME: udev/udevRules module.
|
|
{ object = udevRules; symlink = "/etc/udev/rules.d"; }
|
|
]
|
|
++ optionals (!initWrapperEnabled) [
|
|
{ object = init; symlink = "/init"; }
|
|
]
|
|
++ optionals initWrapperEnabled [
|
|
{ object = init; symlink = initWrapperRealInit; }
|
|
{ object = debugInit; symlink = "/init"; }
|
|
]
|
|
;
|
|
|
|
udevRules = runCommandNoCC "udev-rules" {
|
|
allowedReferences = [ extraUtils ];
|
|
preferLocalBuild = true;
|
|
} ''
|
|
mkdir -p $out
|
|
|
|
# These 00-env rules are used both by udev to set the environment, and
|
|
# by our bespoke init.
|
|
# This makes it a one-stop-shop for preparing the init environment.
|
|
echo 'ENV{LD_LIBRARY_PATH}="${extraUtils}/lib"' > $out/00-env.rules
|
|
echo 'ENV{PATH}="${extraUtils}/bin"' >> $out/00-env.rules
|
|
|
|
cp -v ${udev}/lib/udev/rules.d/60-cdrom_id.rules $out/
|
|
cp -v ${udev}/lib/udev/rules.d/60-persistent-storage.rules $out/
|
|
cp -v ${udev}/lib/udev/rules.d/80-drivers.rules $out/
|
|
cp -v ${pkgs.lvm2}/lib/udev/rules.d/*.rules $out/
|
|
|
|
for i in $out/*.rules; do
|
|
substituteInPlace $i \
|
|
--replace ata_id ${extraUtils}/bin/ata_id \
|
|
--replace scsi_id ${extraUtils}/bin/scsi_id \
|
|
--replace cdrom_id ${extraUtils}/bin/cdrom_id \
|
|
--replace ${pkgs.coreutils}/bin/basename ${extraUtils}/bin/basename \
|
|
--replace ${pkgs.utillinux}/bin/blkid ${extraUtils}/bin/blkid \
|
|
--replace ${pkgs.lvm2}/sbin ${extraUtils}/bin \
|
|
--replace ${pkgs.mdadm}/sbin ${extraUtils}/sbin \
|
|
--replace ${pkgs.bash}/bin/sh ${extraUtils}/bin/sh \
|
|
--replace ${udev}/bin/udevadm ${extraUtils}/bin/udevadm
|
|
done
|
|
|
|
# Work around a bug in QEMU, which doesn't implement the "READ
|
|
# DISC INFORMATION" SCSI command:
|
|
# https://bugzilla.redhat.com/show_bug.cgi?id=609049
|
|
# As a result, `cdrom_id' doesn't print
|
|
# ID_CDROM_MEDIA_TRACK_COUNT_DATA, which in turn prevents the
|
|
# /dev/disk/by-label symlinks from being created. We need these
|
|
# in the NixOS installation CD, so use ID_CDROM_MEDIA in the
|
|
# corresponding udev rules for now. This was the behaviour in
|
|
# udev <= 154. See also
|
|
# http://www.spinics.net/lists/hotplug/msg03935.html
|
|
substituteInPlace $out/60-persistent-storage.rules \
|
|
--replace ID_CDROM_MEDIA_TRACK_COUNT_DATA ID_CDROM_MEDIA
|
|
''; # */
|
|
|
|
extraUtils = mkExtraUtils {
|
|
name = "${device_name}-extra-utils";
|
|
packages = [
|
|
busybox
|
|
]
|
|
++ optionals (stage-1 ? extraUtils) stage-1.extraUtils
|
|
++ [{
|
|
package = runCommandNoCC "empty" {} "mkdir -p $out";
|
|
extraCommand =
|
|
let
|
|
inherit (pkgs) udev;
|
|
in
|
|
''
|
|
# Copy udev.
|
|
copy_bin_and_libs ${udev}/lib/systemd/systemd-udevd
|
|
copy_bin_and_libs ${udev}/bin/udevadm
|
|
for BIN in ${udev}/lib/udev/*_id; do
|
|
copy_bin_and_libs $BIN
|
|
done
|
|
''
|
|
;
|
|
}]
|
|
++ optionals withStrace [
|
|
{
|
|
package = runCommandNoCC "empty" {} "mkdir -p $out";
|
|
extraCommand = with pkgs; ''
|
|
copy_bin_and_libs ${strace}/bin/strace
|
|
cp -fpv ${glibc.out}/lib/libgcc_s.so* $out/lib
|
|
'';
|
|
}
|
|
]
|
|
;
|
|
};
|
|
|
|
initrd = makeInitrd {
|
|
name = "initrd-${device_config.name}";
|
|
inherit contents;
|
|
};
|
|
in
|
|
{
|
|
system.build.initrd = "${initrd}/initrd";
|
|
}
|