Making initramfs image with Yocto

Making initramfs image with Yocto

Embedded systems can have different types of persistent (non-volatile) storage, like eMMC, SD card, NOR or NAND flash memory. Each storage type has advantages and disadvantages with regards to the read/write speed, cost per amount of data that can be stored, number of write cycles, etc.

The non-volatile memory is needed to store the root filesystem, as well as certain configuration data. The root filesystem stored in the memory can be used in different ways. It can be used as read-write, in which case rootfs partition is modified directly and keeps the changes between the boots, but it can also be configured so rootfs contents are always the same on boot and every change that is made does not persist.

In the previous posts the rootfs was placed on the SD card and configured to have read-only and writable parts. Using the SD card image is practical, since it resembles the way development would be performed on the real hardware, and it also can have rather large sizes (we used 1GB and 2GB images, but much larger SD card images can be used).

Sometimes it is useful having a small footprint rootfs that can be stored in smaller capacity memory devices, like NAND or NOR flash memory. In this situation, bundling the minimal rootfs in an initramfs or using rootfs from ramdisk can be useful. Initramfs can also be bundled with the Linux kernel, while ramdisk has to be passed as a separate binary file from U-Boot (unless fitimage format is used).

In this post the steps for making a Linux kernel bundled with initramfs for Cubieboard using Yocto will be shown.

All of the changes that will be described, plus a few more which are required for this to work, are available in the scarthgap branch of meta-mistra repository in Github.

The following items will be covered in this post

Yocto initramfs image

Yocto supports making an initramfs image and bundling it with the Linux kernel, as described in the official documentation. Since initramfs image is bundled with the kernel, once kernel image is loaded, it will also extract and mount the rootfs in RAM memory.

Certain configuration options have to be specified in one of the configuration files, either the global local.conf, or machine or distro specific configuration files.

Most of the instructions usually go for local.conf since it is the easiest, but it is also harder to maintain. Choosing either machine or distro configuration file is a better option, since it allows for fine grained control and configuration.

The configuration options to enable initramfs image creation and bundling with the kernel image are

  • INITRAMFS_IMAGE - specifies image file to use for initramfs
  • INITRAMFS_IMAGE_BUNDLE - can be set to 0 or 1, 1 indicates that the initramfs should be bundled

Additionally, the Linux kernel should support GZip compression for the initramfs, so following options need to be enabled

CONFIG_RD_GZIP=y
CONFIG_INITRAMFS_COMPRESSION_GZIP=y

and U-Boot should be configured to support bigger kernel image (since initramfs is bundled) using

CONFIG_SYS_BOOTM_LEN=0x1000000

The goal with the initramfs is to have a small size rootfs, so certain design choices have to be made.

Init system

In the previous posts, systemd was used as the init system. Systemd is a modern choice, used by many distributions and has various built-in features.

However, those built-in features come with high memory footprint. The systemd recipe in Yocto can be tweaked by removing unused features using PACKAGECONFIG to reduce size. However, it will still use more memory than some of the other init system approaches.

Another approach would be to skip adding the init system completely. The goal with the initramfs is to execute only predefined functions, so init system is not required. This, however, requires more effort from the initramfs designer, since some things that are usually handled by the init system will have to be done manually, like creation and mounting of standard filesystems at boot time, or device discovery using mdev or devtmpfs instead of udev.

In order to configure that no init system is used, following setting needs to be applied to a conf file

INIT_SYSTEM=none

Since we will have two different build variants, with different init systems, it makes sense to use different DISTROs for the regular and initramfs types of build.

New DISTRO configuration

In this post a new DISTRO configuration will be created for the initramfs build. The DISTRO is chosen since we are indeed creating a different configuration for this type of image, but it also comes with certain implications.

The excerpt of the new DISTRO configuration is shown below.

# Mistra Distro for initramfs
require conf/distro/include/mistra-base.inc

INIT_MANAGER = "none"

DISTRO = "mistra-recovery"
DISTRO_NAME = "Mistra Recovery"
DISTROOVERRIDES =. "mistra-recovery:"

# Remove conflicting backends.
DISTRO_FEATURES:remove = "x11 wayland directfb vulkan opengl"
# Add usrmerge to features
DISTRO_FEATURES:append = "\
    usrmerge \
"

# Bundle rootfs image with kernel
INITRAMFS_IMAGE_BUNDLE="1"
INITRAMFS_IMAGE="mistra-initramfs"

The biggest thing to keep in mind when using a new DISTRO is that the new build directory should be used, so more space will be used on the build machine.

Reducing kernel size

Compared to the standard distribution which should support many functionalities and different hardware drivers and functions, the initramfs distribution can fulfill it’s purpose with much less hardware support. Therefore, kernel configuration for the use with initramfs image can be customized.

Yocto provides convenience options for editing the kernel configuration. The following command

bitbake virtual/kernel -c menuconfig

opens the ncurses menuconfig window, same as if the make menuconfig were executed from the kernel source code directory.

After the configuration is ready, the defconfig file which can be included in the layer recipes can be generated using

bitbake virtual/kernel -c savedefconfig

This command will print a path information on where the defconfig file is saved so it can be used later on.

For the purpose of this post, several kernel configuration options are disabled, like Video/Graphics and USB support.

The defconfig file is placed in the recipes-kernel/linux/linux-mainline/mistra-recovery/ directory, so it will be used only when the initramfs image is built.

Custom image

Same as with the kernel, the image recipe that will be used for the initramfs image needs only the necessary items. The recipe is shown here and it is based on the core-image-minimal-initramfs.

require recipes-core/images/core-image-minimal.bb
INITRAMFS_SCRIPTS ?= "\
    initramfs-boot \
"
PACKAGE_INSTALL = " \
    ${INITRAMFS_SCRIPTS} \
    ${VIRTUAL-RUNTIME_base-utils} \
    base-passwd \
    ${ROOTFS_BOOTSTRAP_INSTALL} \
"
# Do not pollute the initrd image with rootfs features
IMAGE_FEATURES = ""
# Don't allow the initramfs to contain a kernel
PACKAGE_EXCLUDE = "kernel-image-*"
IMAGE_NAME_SUFFIX ?= ""
IMAGE_LINGUAS = ""
LICENSE = "MIT"
IMAGE_FSTYPES = "${INITRAMFS_FSTYPES}"
inherit core-image
IMAGE_ROOTFS_SIZE = "8192"
IMAGE_ROOTFS_EXTRA_SPACE = "0"

One thing to note is that the kernel image is removed from the initramfs image, since initramfs will be bundled with the kernel.

Init script

Initramfs requires a init file placed in /init, which will be executed automatically on boot. That file needs to contain commands for different configuration, like creating and mounting filesystems, etc.

In our case we will use the script already provided with Yocto for the simple boot to shell.

#!/bin/sh

PATH=/sbin:/bin:/usr/sbin:/usr/bin

mkdir /proc
mkdir /sys
mount -t proc proc /proc
mount -t sysfs sysfs /sys

exec sh

Essentially, when the system boots it will drop into a shell. There will be no init system nor job control, so exiting the shell will cause kernel panic.

Testing

For testing we are going to build the initramfs and then run it in QEMU. Since new DISTRO is defined, we need to create a new build directory, so the steps for triggering a build would be

. ./setup-environment build_recovery
DISTRO=mistra-recovery MACHINE=cubieboard-ng bitbake mistra-initramfs

Once build is complete, the kernel image with bundled initramfs will be in tmp/deploy/images/mistra-recovery/uImage-initramfs-cubieboard-ng.bin

For testing we just need to use that initramfs bundle as the -kernel parameter and configure console parameter to be passed to the kernel.

qemu-system-arm \
  -M cubieboard \
  -m 1G \
  -kernel uImage-initramfs-cubieboard-ng.bin \
  -dtb sun4i-a10-cubieboard.dtb \
  -nographic \
  -append "console=ttyS0"
[    0.000000] Booting Linux on physical CPU 0x0
[    0.000000] Linux version 6.6.28 (oe-user@oe-host) (arm-mistra-linux-gnueabi-gcc (GCC) 13.3.0, GNU ld (GNU Binutils) 2.42.0.20240620) #1 SMP Wed Apr 17 09:19:38 UTC 2024
[    0.000000] CPU: ARMv7 Processor [410fc080] revision 0 (ARMv7), cr=10c5387d
...
[    0.722588] Freeing unused kernel image (initmem) memory: 3072K
[    0.724355] Run /init as init process
sh: can't access tty; job control turned off
~ # uname -a
Linux (none) 6.6.28 #1 SMP Wed Apr 17 09:19:38 UTC 2024 armv7l GNU/Linux

Summary

In this post the Yocto configuration for building a simple initramfs image is shown. The kernel image with the rootfs is loaded in the QEMU using the -kernel parameter.

The init script that is used is basic, only used to execute the shell. In the real use case the script would do some specific work and could either reboot the system or load a rootfs from another storage.

The initramfs image also takes up only 6MB (1.2MB rootfs and the rest is kernel, which can be further optimized), so it is suitable for use in cases where small amount of memory is available.

Subscribe
If you would like to get information as soon as new content is published, please subscribe to the "MistraSolutions newsletter".