Build a Debian Linux image from scratch
This tutorial has an accompanying YouTube video.
In order to build a Debian linux from scratch for an embedded device, we need three components:
Linux Kernel
Root filesystem (along with an optional initial RAM filesystem)
A device tree
Build the kernel
Let's first create a directory in which we will perform all the work then download and uncompress the kernel into it. We will be using the mainline kernel version 6.6.51 for this tutorial, but you're encouraged to check whether the manufacturer of your particular board supplies their own kernel and use that instead.
For the purpose of making it simple, we won't do any modifications to the kernel configuration, but if you do, you can always run $ make menuconfig
and play around with the settings. Run the following to create a .config
file from the default one (you can find these files in linux-6.6.51/arch/arm64/configs
):
After this is done (it can take a while depending on how much cores and RAM you have), we next need to build the initial RAM filesystem or initramfs.
Build the initramfs
Let's use Busybox for this purpose because once fully built the whole filesystem only takes up a couple of megabytes:
Before we build it, it's recommended to build all files as one static binary, so go into menuconfig by entering $ make menuconfig
and then check Settings --> Build static binary (no shared libs).
This is all we have to do in here, but feel free to look around at all the options and packages you can build into the final image and add anything you might seem necessary. Once you're done, save and exit the configuration and run:
This created a directory _install
which holds almost everything we need for our final initramfs, but we're missing two things - an init script which kernel needs to run as the very first thing once it's done booting, and a couple of directories that it also expects, so let's make both and also copy the whole _install
directory over:
Our initramfs is now complete, so lets compress it into a CPIO image:
Notice the last line/command? We need it in order to make it compatible with u-boot. If you plan to only use it within a FIT image, then this step is not necessary. However, for the sake of completeness, we will do both, boot a FIT image and a regular u-boot booti
boot, so let's now make a directory on our TFTP server and copy all the files we'll need into it:
Notice how I'm also copying over the fsl-ls1046a-rdb.dtb
over? This file is particular to the board I'm working on and was automatically built by the kernel. If you're using your own board, make sure to change this to whatever the documentation that came with your board says.
Make a FIT image
Next, let's create a FIT image recipe file, and make sure to put it into /srv/tftp/embedded-debian/board.its
with the following contents:
Here, you can change all the description fields to your heart's content, but pay close attention to the filenames (all the data
fields) - they need to match the names of the files you've built and copied into this directory. And finally, load
and entry
fields, especially for the kernel, need to be positioned correctly. This should come with the documentation for the CPU, but if you're unsure check what u-boot configuration for your board uses - use whatever the $kernel_addr_r
environment variable is set to - just enter printenv $kernel_addr_r
in u-boot.
Now let's build the FIT image:
To test it out, now go u-boot on your board and enter the following (make sure to change IP addresses for both your board and the TFTP server):
This will unpack the FIT image and if everything is in order, start the kernel and get you into the initramfs image we built earlier.
Build a proper Debian distribution
Because the above will only get you into a really basic filesystem that can't do much and is not permanent, it's now time to set up a proper Debian image that we can flash an SD card with. Make sure to cd embedded-debian
where we will make an empty image by using tools that qemu-img brings to the table:
We've created a 4 GB image file then inside of it, created the two partitions we'll need: first, a boot partition, which will hold our kernel, device trees and the initial RAM filesystem, and second, our root filesystem partition which'll hold the whole Debian root filesystem. However, this is still just a file, so run the following commands:
Let's go over what we just did:
we turned our SD card image file into a loopback device (a fake drive, if you will)
since it has paritions, we use
kpartx
to expose them in/dev/mapper/loop0pX
(X being partition number)then we turn the exposed partitions into
ext4
filesystemsand finally, we create mountpoints for the partitions into which we also mount them
If you check both directories, so /mnt/sdcard
and /mnd/sdcard/boot
, you'll notice there is only one directory in each of them (lost+found
which is an artifact of ext4
filesystem), so let's now install Debian into the second partition:
Debootstrap, as the name suggests, is a utility which installs a Debian base system into whichever directory you want it to, and in our case, we install it directly into the sdcard image which we have mounted earlier. But this is only half of the equation, the files are just copied, but not installed, which is why we need to chroot
into the image and run the second stage which does that.
And while we're in, we also install some basic stuff that we'll need, the most important being network and time management. If you need other tools, feel free to install them here.
And last thing, we also need to set up the root password, otherwise we won't be able to log in once we boot into this image.
Install Kernel
We now have our root filesystem in place, so we need to install Kernel into the /boot
partition on the SD card image. Go back to the embedded-debian
directory and run the following:
This'll install both the kernel itself and all the separate modules that were built along with the kernels. The kernel will get installed into our boot partition while the modules go into our root filesystem (into /lib/modules/
). And while we're at it, we also create a dtb
directory on our boot partition where we will put our device trees. This directory is optional and you're free to name it whatever you like - or even skip it completely, if your board boots with device tree being elsewhere.
The final thing we need to copy over into our boot partition is optional if you don't intend to use it, but it's good to have it in place as a sort of a rescue system and that's the initramfs image we built earlier, and while we're at it also copy the FIT image, which we will likely not need, but doesn't hurt to have it available:
There we go, we now have everything in place to flash our SD card. Take the sdcard.img
file that we've been working on this whole time and use the program Balena Etcher to write it to the SD card. Keep in mind that this image IS NOT directly bootable as it doesn't have any bootloaders. In my case, u-boot already comes with the board, but your mileage may vary.
Clean up
Once you're done building the image, use the following commands to unmount the partitions and detatch the loopback device with the commands below, however, feel free to skip this step in case you discover in any of the next steps that your image doesn't boot correctly (or at all) and you need to make any adjustments.
Boot into our Debian
Once in u-boot, run the following commands:
While the commands should be pretty self-explanatory, let's go over them real quick:
We set the kernel boot arguments to now look at the second partition for the root filesystem. This is because our first partition on the device is the
/boot
partition that holds our kernel and the device tree.We load both, the kernel and the device tree into RAM
We boot the board. Notice the minus character in
booti
command? That's because we're skipping the initial RAM filesystem.
This should hopefully get you to the login screen, but we're not quite done yet. If you check the /boot
directory, you'll notice that it's either empty or you'll get an error. This is perfectly fine if you never intend to use it from within Linux, but for the sake of completeness, let's make sure its automatically mounted at boot. First check the UUID of your partitions, by running $ ls -la /dev/disk/by-uuid/
and copy the UUID of the partition that points to mmcblk0p1
(remember, our first partition is the boot partition).
Then open /etc/fstab
file with your favorite editor and paste the following line into it, and make sure you change the UUID to whatever your UUID of the partition is:
Then commit the above by entering $ mount -a
. The /etc/fstab
file gets read at boot which means Debian will automatically mount this partition every time.
Tips and tricks
If you ever need to boot into initramfs for whatever reason, run the following:
Once you're done with it, continue booting into Debian with the following:
Gentoo has some great documentation about this topic, so go and give it a read!
References
Last updated