Booster is a tool to create initramfs images needed at the early stage of Linux boot process. Booster is made with speed and full disk encryption use-case in mind.
Booster advantages:
- Fast image build time and fast boot time.
- Out-of-box support for LUKS-based full disk encryption setup.
- Clevis style data binding. The encrypted filesystem can be bound to TPM2 chip or to a network service. This helps to unlock the drive automatically but only if the TPM2/network service presents.
- Automatically detects and unlocks systemd-cryptenroll (fido2 and tpm2) type of partition encryption.
- Easy to configure.
- Automatic host configuration discovery. This helps to create minimalistic images specific for the current host.
booster generator config file is located at /etc/booster.yaml. Here is a sample config file:
network:
interfaces: enp0s31f2,2e:1d:61:30:a3:63
dhcp: on
# either dhcp above or static configuration below can be used
ip: 10.0.2.15/24
gateway: 10.0.2.255
dns_servers: 192.168.1.1,8.8.8.8
universal: false
modules: -*,hid_apple,kernel/sound/usb/,kernel/fs/btrfs/btrfs.ko,kernel/lib/crc4.ko.xz
compression: zstd
mount_timeout: 5m6s
strip: true
extra_files: vim,/usr/share/vim/vim82/,fsck,fsck.ext4
vconsole: true
enable_lvm: true
enable_mdraid: true
-
networknode, if present, initializes the network at the boot time. It is needed if mounting a root fs requires access to the network (e.g. in case of Tang binding). The network can be either configured dynamically with DHCPv4 or statically within this config. In the former casedhcpis set toon. In the latter case the config allows to specifyip- the machine IP address and its network mask,gateway- default gateway,dns_servers- comma-separated list of DNS servers. Thenetworknode also acceptsinterfacesproperty - a comma-separated list of network interfaces (specified either with name or MAC address) to enable at the boot time. Network names likeenp0s31f6get resolved to MAC addresses at generation time and then passed to init. Ifinterfacesnode is not specified then all the interfaces are activated at boot. -
universalis a boolean flag that tells booster to generate a universal image. By default booster generates a host-specific image that includes kernel modules used at the current host. For example if the host does not have a TPM2 chip then tpm modules are ignored. Universal image includes many kernel modules and tools that might be needed at a broad range of hardware configurations. -
modulesis a comma-separated list of extra modules to add to or remove from the generated image. One can use a module name or a path relative to the modules dir (/usr/lib/modules/$KERNEL_VERSION). The compression algorithm suffix (e.g. ".xz", ".gz) can be omitted from the module filename. If the element starts with a minus sign (-) then it means "do not add it to the image", otherwise modules are added. If the path ends with a slash symbol (/) then it is considered a directory and all modules from this directory need to be added recursively. A special symbol*(star) means all modules. It can be used for example to add all modules or remove all predefined modules from the image. Booster also takes module dependencies into account, all dependencies of the specified modules will be added to the image as well. -
modules_force_loadlist of module names that are forcibly loaded during the boot process before switching into user-space. Any module in this list automatically added to the image so there is no need to duplicate it atmodulesproperty. -
append_all_modaliasesis a boolean flag that instructs booster to add all hosts's module aliases to the booster image. This flag is useful for debugging boot timeout issues when some important modules are missed from the image. Setting the flag totruewill help to print module names for aliases that were requested by kernel but missed in the image. -
compressionis a flag that specifies compression for the output initramfs file. Currently supported algorithms are "zstd", "gzip", "xz", "lz4", "none". If no option specified then "zstd" is used as a default compression. -
mount_timeouttimeout for waiting for the root filesystem to appear. The field format is a decimal number and then unit number. Valid units are "s", "m", "h". If no value specified then default timeout (3 minutes) is used. To disable the timeout completely specify "0s". -
stripis a boolean flag that enables ELF files stripping before adding it to the image. Binaries, shared libraries and kernel modules are examples of ELF files that get processed with strip UNIX tool.This options is not compatible with signed modules. If you see
booster: finit(crc32,generic): key was rejected by serviceboot error please set thestripconfig option tofalse. -
extra_filesis a comma-separated list of extra files to add to the image. If an item starts with slash ("/") then it is considered an absolute path. Otherwise it is a path relative to /usr/bin. If the item is a directory then its content is added recursively. There are a few special cases:- adding
busyboxto the image enables an emergency shell in case of a panic during the boot process. - adding
fsckenables boot time filesystem check. It also requires filesystem specific binary calledfsck.$rootfstypeto be added to the image. Filesystems are corrected automatically and if it fails then boot stops and it is responsibility of the user to fix the root filesystem.
- adding
-
vconsoleis a flag that enables early-user console configuration. If it is set totruethen booster reads configuration from/etc/vconsole.confand/etc/locale.confand adds required keymap and fonts to the generated image. The following config properties are taken into account:KEYMAP,KEYMAP_TOGGLE,FONT,FONT_MAP,FONT_UNIMAP. See also man vconsole.conf. -
enable_lvmis a flag that enables LVM volume assembly at the boot time. This flag also makes sure all the required modules/binaries are added to the image. -
enable_mdraidis a flag that enables MdRaid assembly at the boot time. This flag also makes sure all the required modules/binaries are added to the image. -
enable_zfsis a flag that enables ZFS filesystem as root filesystem. This flag also makes sure all the required modules/binaries are added to the image. Note that if ZFS is enabled thenzfs=boot option must be used instead ofroot=boot option. -
enable_plymouthis a flag that enables Plymouth boot splash support. When enabled, booster bundles the Plymouth daemon, plugins, theme, and fonts into the initramfs. GPU driver must be included inmodules_force_load. Thesplashkernel parameter is also required. Note thatbooster.log=consoleconflicts with Plymouth's graphical display; when console logging is active, Plymouth reverts to the details plugin (text-based fallback).
Once you are done modifying your config file and want to regenerate booster images under /boot please use /usr/lib/booster/regenerate_images.
It is a convenience script that performs the same type of image regeneration as if you installed booster with your package manager.
-v,--verboseEnable verbose output
Build initrd image. Usage: booster [OPTIONS] build [build-OPTIONS] output
-f,--forceOverwrite existing initrd file.--init-binary<default: /usr/lib/booster/init> Booster 'init' binary location.--compression<default: zstd> Output file compression. Possible values: zstd, gzip, xz, lz4, none.--kernel-versionLinux kernel version to generate initramfs for.--config<default: /etc/booster.yaml> Configuration file path.--universalAdd wide range of modules/tools to allow this image boot at different machines.--stripStrip ELF files (binaries, shared libraries and kernel modules) before adding it to the image.
Show content of the file inside the image. Usage: booster [OPTIONS] cat image file-in-image
List content of the image. Usage: booster [OPTIONS] ls image
Unpack image. Usage: booster [OPTIONS] unpack image output-dir
Some parts of booster boot functionality can be modified with kernel boot parameters. These parameters are usually set through bootloader config. Booster boot uses following kernel parameters:
-
root=$devicerefdevice reference to root device. See notes below for how to specify the device reference. Ifroot=points to a LUKS partition then it automatically unlocked as a device/dev/mapper/rootand mounted to root. Booster also supports root partition autodiscovery - if noroot=parameter is specified then booster checks for partitions with specific GPT type and uses it to mount as root. -
rootfstype=$TYPE(e.g. rootfstype=ext4). By default booster tries to detect the root filesystem type. But if the autodetection does not work then this kernel parameter is useful. Also please file a ticket so we can improve the code that detects filetypes. -
rootflags=$OPTIONSmount options for the root filesystem, e.g. rootflags=user_xattr,nobarrier. In partition autodiscovery mode GPT attribute 60 ("read-only") is taken into account. -
rd.luks.uuid=$UUIDUUID of the LUKS partition where the root partition is enclosed. booster will try to unlock this LUKS device. -
rd.luks.name=$UUID=$NAMEsimilar to rd.luks.uuid parameter but also specifies the name used for the LUKS device opening. -
rd.luks.key=$UUID=$PATHabsolute path to a keyfile in the initrd/initramfs which can be used to unlock the device identified by UUID, if this file does not exist or fails to unlock it will fall back to a password request. -
rd.luks.options=opt1,opt2a comma-separated list of LUKS flags. Supported options arediscard,same-cpu-crypt,submit-from-crypt-cpus,no-read-workqueue,no-write-workqueue. Note that booster also supports LUKS v2 persistent flags stored with the partition metadata. Any command-line options are added on top of the persistent flags. -
rd.modules_force_loada comma-separated list of extra kernel modules which should be force loaded. -
resume=$devicerefdevice reference to suspend-to-disk device. -
zfs=$pool/$datasetspecifies what ZFS dataset needs to be used for root partition. This option is only used if ZFS config option is enabled. If ZFS filesystem is enabled thenroot=boot param is ignored. -
booster.logconfigures booster init logging. It accepts a comma separated list of following values:One of the level values (from more verbose to less verbose) -
debug,info,warning,errorornull. The last level ofnulldisables any logging, so⚠️ use it only if know what you are doing⚠️ . If the level is not specified theninfoused by default.console- print booster init logs to console.The debug log is also printed to the kernel kmsg buffer and available for reading either with
dmesgor withjournalctl -b. If debug level is enabled then kmsg throttling gets disabled automatically. -
booster.debugan obsolete option that is equivalent tobooster.log=debug,console. -
quietSet booster init verbosity to minimum. This option is ignored ifbooster.debugorbooster.logis set. -
init=$PATHpath to user-space init binary. If not specified then default value/sbin/initis used.
Device reference is a way to specify a device or partition in kernel parameters. It is labeled as $deviceref above.
Device reference has one of the following values:
/dev/XXXpath to specific device file, it can be either a path to real device/partition like/dev/sda1,/dev/nvme0n1or path to dm-mapper virtual device like/dev/mapper/rootor/dev/vg_mesos/lv_mesos_containers.UUID=$UUIDor/dev/disk/by-uuid/$UUIDreferences device by its filesystem/LUKS UUID. See notes about UUID formatting rules below.LABEL=$LABELor/dev/disk/by-label/$LABELreferences device by its filesystem/LUKS label.PARTUUID=$UUIDor/dev/disk/by-partuuid/$UUIDreferences device by GPT partition UUID.PARTUUID=$UUID/PARTNROFF=$OFFSETreferences device by $OFFSET from a GPT partition specified by $UUID e.g.PARTUUID=fd59d06d-ffa8-473b-94f0-6584cb2b6665/PARTNROFF=2.PARTLABEL=$LABELor/dev/disk/by-partlabel/$LABELreferences device by GPT partition label.HWPATH=$PATHor/dev/disk/by-path/$PATHreferences device by deterministic hardware path e.g.pci-0000:02:00.0-nvme-1-part2.WWID=$IDor/dev/disk/by-id/$IDreferences device by its wwid e.g.nvme-KXG6AZNV256G_TOSHIBA_40SA13GZF6B1-part3
Boot parameters such as root=UUID=$UUID and rd.luks.uuid=$UUID allow you to specify the block device by its UUID.
The UUID format is xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx where x is a hexadecimal symbol either in lower of upper case.
UUID parameter can optionally be enclosed with quote symbol " though it is not recommended. Following examples show correct parameters format:
root=UUID=ac8299a8-91ce-4bf6-a524-55a62844b787, root=UUID="ac8299a8-91ce-4bf6-a524-55a62844b787" (not recommended),
rd.luks.uuid=ac8299a8-91ce-4bf6-a524-55a62844b787, rd.luks.uuid="ac8299a8-91ce-4bf6-a524-55a62844b787" (not recommended).
It is a note to summarize the algorithm that computes what modules are going to end up in the generated booster image.
Initial module list for booster is defaultModulesList - a set of predefined hard-coded modules defined at generator.go.
These are selected modules that most likely cover most system boot needs - disk, filesystem, keyboard, tpm, ethernet, usb drivers.
If the universal config option is set to false (default value) then so-called host mode is used.
I.e. image is generated with the drivers needed for current host hardware only.
To achieve it booster fetches all currently loaded modules from /sys/module/ and computes intersection with the defaultModulesList.
Then booster looks at modules config option, a comma-separated list of elements. It iterates over all the elements left-to-right.
The host mode filtering rule does not apply to this list of manually specified modules.
If the element starts with minus sign - then it removes given modules from the image, otherwise modules are added to the image.
If the element is a module name then this module is added/removed. Note that by convention a kernel module name can be computed from its filename by replacing all dashes to underscore, e.g.
For the module hid-apple.ko.gz name will be hid_apple.
If the element is a path to the module file relative to /usr/lib/modules/$KERNEL_VERSION then the module is added/removed. Note that the compression algorithm suffix can be omitted from the module filename.
If the element ends with the slash symbol / then this element is considered a directory relative to /usr/lib/modules/$KERNEL_VERSION.
Booster goes over this directory recursively and adds/removes the modules to the image. Minus sign can be used with the directories.
Star symbol * is a shortcut for "all modules", it can be used to add all modules or remove all modules from the image.
Next booster moves to the modules_force_load option that consists of module names to load at the boot time.
All these modules are also added to the image.
At the final step booster computes dependency graphs between modules and all required dependencies.
For example if a user manually added ext4 and kernel build system says ext module requires mbcache and jbd2 then both
mbcache and jbd2 automatically added to the image.
A Unified Kernel Image (UKI) is a PE binary that bundles various boot components (e.g. kernel, initrd, and an UEFI boot stub) as a single executable. This allows for booting directly through the firmware (UEFI) as well as authenticating all of the boot components at once for Secure Boot.
To generate UKIs in Booster, please install the systemd UKI generator (systemd-ukify) from your distribution's package manager and use /usr/lib/booster/regenerate_uki.
It is a convenience script that performs the same type of image regeneration as if you installed booster with your package manager, then passes the result to systemd's UKI generator (ukify) as input.
The script only passes a subset of boot components, namely the system's microcode(s), initrd, os-release file, boot splash image and kernel. Kernel command-line entries of the UKI are inherited from /etc/booster.yaml.
Please note that to boot the UKI by default, it may be necessary to configure your system's boot loader configuration file(s) accordingly.
If you have a problem with booster boot tool you can enable debug mode to get more
information about what is going on. Just add booster.log=debug,console kernel parameter and booster
provide additional logs.
In case of a boot failure, when the devices are missing, logs can still be retrieved from busybox using the network.
First, set up a tftp server (port 69) on another machine/VM. For example on Archlinux, pacman -S atftp; systemctl start atftpd.
Then, edit /etc/booster.yaml and add network support and busybox (extra_files: busybox).
Regenerate the initramfs and reboot. Once inside busybox, get the logs and send them to the tftp server:
$ dmesg >boot.log
$ lsmod >mods.log
$ tftp -pl boot.log <server ip>
$ tftp -pl mods.log <server ip>
The logs will be in /srv/atftp on the server.
If you got booster: Timeout waiting for root filesystem error please add append_all_modaliases config flag and rebuild the image. With this flag you'll get a list of modules that were requested by the kernel but absent in the booster image. Some of these modules might be required to boot your system.
Create an initramfs file specific for the current kernel/host. The output file is booster.img:
$ booster build booster.img
Create an universal image with many modules (such as SATA/TPM/NVME/... drivers) included:
$ booster build --universal booster.img
Create an initramfs for kernel version 5.4.91-1-lts and copy it to /boot/booster-lts.img:
$ booster build --kernel-version 5.4.91-1-lts /boot/booster-lts.img
Here is a systemd-boot configuration stored at /boot/loader/entries/booster.conf. In this example e122d09e-87a9-4b35-83f7-2592ef40cefa is a UUID for the LUKS partition and 08684949-bcbb-47bb-1c17-089aaa59e17e is a UUID for the encrypted filesystem (e.g. ext4). Please refer to your bootloader documentation for more info about its configuration.
title Linux with Booster
linux /vmlinuz-linux
initrd /booster-linux.img
options rd.luks.uuid=e122d09e-87a9-4b35-83f7-2592ef40cefa root=UUID=08684949-bcbb-47bb-1c17-089aaa59e17e rw
Users of the Btrfs filesystem with a system installed on a subvolume should add rootflags corresponding to their entry in /etc/fstab. In this example 69bc4dd2-7f6c-4821-aa6b-d80d9c97d470 is a UUID for Btrfs partition, with the system installed on subvolume called root and /etc/fstab looks like this:
UUID=69bc4dd2-7f6c-4821-aa6b-d80d9c97d470 / btrfs rw,relatime,autodefrag,compress=zstd:2,space_cache,subvol=root 0 0
So /boot/loader/entries/booster.conf should looks like this:
title Linux with Booster
linux /vmlinuz-linux
initrd /booster-linux.img
options root=UUID=69bc4dd2-7f6c-4821-aa6b-d80d9c97d470 rw rootflags=relatime,autodefrag,compress=zstd:2,space_cache,subvol=root
Create a Unified Kernel Image and write the result to /boot/EFI/Linux:
$ /usr/lib/booster/regenerate_uki build /boot/EFI/Linux
Booster is Copyright (C) 2020 Anatol Pomazau http://github.com/anatol
Project homepage https://github.com/anatol/booster