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
-
network
node, 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 casedhcp
is 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. Thenetwork
node also acceptsinterfaces
property - a comma-separated list of network interfaces (specified either with name or MAC address) to enable at the boot time. Network names likeenp0s31f6
get resolved to MAC addresses at generation time and then passed to init. Ifinterfaces
node is not specified then all the interfaces are activated at boot. -
universal
is 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. -
modules
is 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_load
list 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 atmodules
property. -
append_all_modaliases
is 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 totrue
will help to print module names for aliases that were requested by kernel but missed in the image. -
compression
is 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_timeout
timeout 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". -
strip
is 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 service
boot error please set thestrip
config option tofalse
. -
extra_files
is 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
busybox
to the image enables an emergency shell in case of a panic during the boot process. - adding
fsck
enables boot time filesystem check. It also requires filesystem specific binary calledfsck.$rootfstype
to 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
-
vconsole
is a flag that enables early-user console configuration. If it is set totrue
then booster reads configuration from/etc/vconsole.conf
and/etc/locale.conf
and 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_lvm
is 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_mdraid
is 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_zfs
is 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.
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
,--verbose
Enable verbose output
Build initrd image. Usage: booster [OPTIONS] build [build-OPTIONS] output
-f
,--force
Overwrite 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-version
Linux kernel version to generate initramfs for.--config
<default: /etc/booster.yaml> Configuration file path.--universal
Add wide range of modules/tools to allow this image boot at different machines.--strip
Strip 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=$deviceref
device 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/root
and 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=$OPTIONS
mount 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=$UUID
UUID of the LUKS partition where the root partition is enclosed. booster will try to unlock this LUKS device. -
rd.luks.name=$UUID=$NAME
similar to rd.luks.uuid parameter but also specifies the name used for the LUKS device opening. -
rd.luks.key=$UUID=$PATH
absolute 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,opt2
a 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_load
a comma-separated list of extra kernel modules which should be force loaded. -
resume=$deviceref
device reference to suspend-to-disk device. -
zfs=$pool/$dataset
specifies 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.log
configures 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
,error
. If the level is not specified theninfo
used 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
dmesg
or withjournalctl -b
. If debug level is enabled then kmsg throttling gets disabled automatically. -
booster.debug
an obsolete option that is equivalent tobooster.log=debug,console
. -
quiet
Set booster init verbosity to minimum. This option is ignored ifbooster.debug
orbooster.log
is set. -
init=$PATH
path to user-space init binary. If not specified then default value/sbin/init
is 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/XXX
path to specific device file, it can be either a path to real device/partition like/dev/sda1
,/dev/nvme0n1
or path to dm-mapper virtual device like/dev/mapper/root
or/dev/vg_mesos/lv_mesos_containers
.UUID=$UUID
or/dev/disk/by-uuid/$UUID
references device by its filesystem/LUKS UUID. See notes about UUID formatting rules below.LABEL=$LABEL
or/dev/disk/by-label/$LABEL
references device by its filesystem/LUKS label.PARTUUID=$UUID
or/dev/disk/by-partuuid/$UUID
references device by GPT partition UUID.PARTUUID=$UUID/PARTNROFF=$OFFSET
references device by $OFFSET from a GPT partition specified by $UUID e.g.PARTUUID=fd59d06d-ffa8-473b-94f0-6584cb2b6665/PARTNROFF=2
.PARTLABEL=$LABEL
or/dev/disk/by-partlabel/$LABEL
references device by GPT partition label.HWPATH=$PATH
or/dev/disk/by-path/$PATH
references device by deterministic hardware path e.g.pci-0000:02:00.0-nvme-1-part2
.WWID=$ID
or/dev/disk/by-id/$ID
references 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.
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
Booster is Copyright (C) 2020 Anatol Pomazau http://github.com/anatol
Project homepage https://github.com/anatol/booster