1
1
mirror of https://github.com/NixOS/mobile-nixos.git synced 2024-12-15 19:23:01 +03:00
mobile-nixos/modules/initrd.nix
Samuel Dionne-Riel 2a366b2cae initrd: Allow redirecting output to /dev/kmsg
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.
2020-02-03 16:19:10 -05:00

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";
}