Creating a Minimal Linux System for the BeagleBone Black with Buildroot

Embedded Linux Development Lab

If you've been following my earlier blogs, this is a continuation of my independent study course on Embedded Linux. In the previous blogs, I covered building a Linux system from scratch. But now it's time to upgrade to a build system.

I'm going to be using Buildroot in this tutorial. I chose Buildroot as it is one of the less complex and more widely used build systems in the embedded Linux space. Yocto is arguably better for a production scenario but takes much longer to master due to its complexity. Buildroot is simple but lacks many of the bells and whistles that Yocto has.

Much of this blog is based on bootlin's training documentation, so it may be more useful for you to follow along with that as well as this blog.

What is a Build System?

A build system is a lot of things and can look very different depending on the project. A build system in its easiest description is translating source code into binary files to be run on a system.

A simple build system could be a single makefile that compiles a single C program. Example below:

#include <stdio.h>

int main(){
    printf("Hello World!\n");
CC=gcc

hello:
        $(CC) hello.c -o hello

This is likely one of the simplest forms of build systems possible. It might almost be stretching it calling it a 'system' because of how minimal it is.

Buildroot is similar to this (but much much more complex). It does what this simple makefile does, except it compiles the Linux kernel, bootloader, any modifications you have, and many more, all according to the configurations you input.

Embedded Linux build systems automate so much that building from scratch is essentially obsolete except in some very interesting edge cases.

Getting Buildroot

First, as always, clone the git repository:

git clone https://git.buildroot.net/buildroot

Switch into the cloned repo, and checkout the newest build

cd buildroot 
git checkout 2022.11

Configuring Buildroot

Buildroot uses the same menuconfig we saw in earlier blog's that crosstool and the Linux kernel does, so enter that with:

make menuconfig

I'm closely following Buildroot's Training Document here With the largest change being that my BeagleBone is the non-wireless version, and thus uses a different device tree.

Do experiment with these options on your own time, or take some time to see what other options are available in the menuconfig. There are quite a few interesting options that are easy to implement.

  • In Target Options:

    • Select ARM (little endian) as the Target Architecture

    • Select cortex-A8 as the Target Architecture Variant

  • In Toolchain:

    • Set toolchain type to External toolchain . (This will save us a lot on build time).

    • Set the Toolchain under Toolchain External Options as ARM 2021.07 .

  • In System Configuration:

    • Set the System hostname to something of your preference. I used billbeaglebone .

    • Set the System banner as well to anything of interest. This is a line of text that will print every time you turn the system on. Mine is Bill's Embedded Linux!

    • Enable root login with password and set this password.

  • In Kernel:

    • Enable Linux Kernel, the default should be the newest kernel (at the time of reading 6.0), this is fine for our use.

    • Define the Defconfig name as omap2plus.

    • Make sure the Kernel binary format is zImage .

    • Define the In-tree Device Tree Source File name as am335x-boneblack .

    • Enable Needs host OpenSSL

  • In Target Packages:

    • Enable BusyBox.

    • Optional: For a later blog, we will be debugging a program on the board using GDB and Valgrind. In Debugging, profiling and benchmark enable GDB and gdbserver, as well as valgrind .

    • Optional: in Networking applications enable dropbear. This is an SSH client that will be convenient to use later.

    • Optional: For a little fun, in Games enable ascii_invaders .

    • Take a look around here at all the packages you can include so simply with Buildroot. If you followed the last blog series on creating an embedded Linux system from scratch, you might appreciate just how easy this is.

  • In Bootloaders:

    • Enable U-Boot.

    • Define the U-Boot configuration as am335x_evm .

    • Set the Build system Kconfig.

    • Check that the U-Boot Version is set 2022.04 .

    • In U-Boot binary format, enable only u-boot.img .

    • Enable Install U-Boot SPL binary image, and set U-boot SPL/TPL binary image name(s) to MLO .

This finishes our configuration! Go ahead and save before exiting.

Building Your System

For curiosity, Bootlin notes we should save the build log. If you don't care about this, you can just run make. But if you're interested like I am to see the build log, run make 2>&1 | tee build.log which will save the make and terminal logs to a build.log file.

The first time you build will take a while. Skip to 'Prepping the SD Card' in the interest of time.

Once the build is done, and you've dealt with any errors that may have come up (I had to install perl on my system), you should see these output files in the output/images directory.

Prepping the SD Card

You should use a good SD card for this. At first I tried the Kingston CANVAS Select Plus, but experienced issues with the read/write speeds. I upgraded to the SanDisk Extreme and had no issues.

We need to create two partitions on our card. You can use command line tools, or something easy like GParted.

Create two partitions, the first a FAT32 filesystem with 128Mib, and the second a ext filesystem taking up the remaining space on the card.

You should end up with partitions looking similar to this:

Label the partitions as I did for ease of use, and set the boot flag on the boot partition by left-clicking, and selecting Manage flags .

Prepping Our Linux System

Once the build finishes, and you have your SD Card formatted, we are ready to move our files over to the BeagleBone.

Follow the steps in my earlier blog to connect a serial adapter, and install picocom .

Copy the files from the output/images directory onto the boot partition like so:

#On Fedora Linux, the SD card is mounted at /run/media/$USER/partitionname, this may be different depending on your distro.
cp MLO u-boot.img zImage am335x-boneblack.dtb /run/media/$USER/boot/

We can create a file to give U-Boot commands without setting the bootloader manually. Create a folder and file in the boot partition:

mkdir extlinux && touch extlinux/extlinux.conf

Fill this file with the following text:

label buildroot
        kernel /zImage
        devicetree /am335x-boneblack.dtb
        append console=ttyO0,115200 root=/dev/mmcblk0p2 rootwait

If you follow this from my earlier blog, you might notice this is very similar to the commands we gave to U-Boot manually.

Finally, we need to unpack our root file system now to the second partition:

sudo tar -C /run/media/$USER/rootfs/ -xf rootfs.tar

Now our card is ready for the BeagleBone!

Booting Our System

With your serial adapter plugged into your PC, start up a picocom terminal:

picocom -b 115200 /dev/ttyUSB0

Now, unmount the SD card's partitions, and insert the card into the board.

While plugging in the board, hold down the USR button to force the system to boot from the SD card.

You should see U-Boot and kernel output similar to this:

Finally followed with the banner you set in the Buildroot menuconfig.

Login as root with the password you supplied:

And you've now installed a Buildroot Linux system on the board!

If you followed the optional steps in the config step, enjoy some ascii_invaders:

This finishes the tutorial. Thanks for reading!