Skip to main content
Version: 1.4.2

Building an image

caution

Yocto is the defacto embedded linux distribution toolkit, however it is famous for its steep learning curve. Due to this, it is critical that you follow the instructions, as deviating from them will mostly lead to problems!

This guide walks you through a building a custom image using Yocto. The images produced by Yocto are always for dedicated devices as this allows the size of the images to significantly reduced as only the components which are related to the device are included. For example the images produced in this tutorial are only about 80MB in size. In comparison the "lite" image from Raspberry Pi is about 550MB!

Whilst the methodology described on this page can be extended to support other devices, these instructions focus in the following devices:

  • Raspberry Pi 3 (64 bit)
  • Raspberry Pi 4 (64 bit)

Setting up your build system​

info

The instructions below were tested with the kirkstone release. If you want to use a different project version or check the most recent requirements, visit Official Yocto quick guide. Make sure you select the proper release, as the instructions may vary for different versions.

Before being able to build your own image, you need to prepare an environment to run Yocto. Follow the instructions to get your build system setup.

  1. Create a Virtual Machine running Ubuntu 20.04 (LTS)

    The virtual machine should be created

    • Machine with a x86_64 CPU Architecture
    • Use a ubuntu 20.04 image (You can choose if you want a UI or not)
    • Disk with at least 100 GB, though more is better and generally it hard changing the size afterwards
    • Network enabled
  2. Install essential packages

    sudo apt install file gawk wget git diffstat unzip texinfo gcc build-essential chrpath socat cpio python3 python3-pip python3-pexpect xz-utils debianutils iputils-ping python3-git python3-jinja2 libegl1-mesa libsdl1.2-dev xterm python3-subunit mesa-common-dev zstd liblz4-tool
note

The large disk size might seem like a lot, but Yocto builds everything from scratch, and this takes time and disk space!

Building your image​

The meta-tedge repository is a Yocto layer but it also includes some kas definitions to make it easier to build images. kas supports reusing configuration files to make it easier to configure different image definitions to control what should be included in your image.

Feel free to move the kas files into your own repository where it will reference the meta-tedge Yocto layer, however please always refer back to the original kas project files if you run into any problems (as it may have changed in the meantime).

The project uses different Yocto layers to build an image which contains:

ItemDescription
Yocto Linux DistributionBase operating system
thin-edge.iothin-edge.io and all of its components
thin-edge.io workflowA workflow definition to perform Operation System A/B updates which interacts with the underlying technology, e.g. RAUC or mender
dependenciesDependencies of the underlying firmware update mechanism, e.g. RAUC or mender

Cloning the project​

  1. Clone the project and change directory to the locally checked out folder

    git clone https://github.com/thin-edge/meta-tedge.git
    git checkout kirkstone
    cd meta-tedge/kas
  2. Install just which is used to run different tasks provided by the project

    curl --proto '=https' --tlsv1.2 -sSf https://just.systems/install.sh | sudo bash -s -- --to /usr/bin

    Check out the justfile website for more installation options.

  3. Install kas which is used to managed bitbake projects

    pip3 install kas
    note

    kas is used to managed multiple Yocto bitbake projects.

Configure shared folders​

Using shared folders will help reduce the overall build times when building for different targets as the downloaded files can be reused.

  1. Open a console in the project's root directory, and change to the kas directory

    cd kas
  2. Create the following file with the given contents

    file: .env
    SSTATE_DIR=/data/yocto/sstate-cache
    DL_DIR=/data/yocto/downloads
  3. Create the shared folders which are referenced in the above .env file

    sudo mkdir -p /data/yocto/sstate-cache
    sudo chown -R "$(whoami):$(whoami)" /data/yocto/sstate-cache

    sudo mkdir -p /data/yocto/downloads
    sudo chown -R "$(whoami):$(whoami)" /data/yocto/downloads

Building an image​

The following steps will build an image with thin-edge.io and RAUC installed to perform firmware (OS) updates.

  1. Open a console in the project's root directory, and change to the kas directory

    cd kas
  2. Build an image for your device (machine)

    KAS_MACHINE=raspberrypi3-64 just build-project ./projects/tedge-rauc.yaml

    The KAS_MACHINE is used to select the target device.

    note

    Yocto build take a long time (> 4 hours) to build so be patient. A build time of 12 hours is also not unheard of!

  3. Inspect the built image

    ls -l tmp
  4. Flash the base image using the instructions on the Flashing an image page

tip

Check out the project files under the ./projects directory for other available project examples. If you don't find a project that fits your needs, then just create your own project definition.

note

The tedge-rauc project includes a package manager (apt) so that additional packages can be installed without having to do a full image update. If you don't want the image to include a package manager, you can customize the project yourself and remove the related configuration file.

Tips​

This section contains general tips which can be helpful whilst either getting things setup, or what to do when you encounter an error.

Unexpected/cryptic build errors​

Whilst building device images it is common that at some people you will run into a build error. These errors can be very cryptic and hard to find the root cause. Luckily, more often than not, these errors are caused due to some corrupted state within the temporary build output folder.

So before creating a ticket, make sure you clean the folder using the following command:

just clean

Then rebuild the project that you originally tried to build.