Skip to content

This repository aims to provide a reference implementation for updateable, secure embedded Linux systems that are built with the Yocto build system.

License

Notifications You must be signed in to change notification settings

JSydll/emx-base-stack

Repository files navigation

Embedded Linux Base Stack

Build [scarthgap|raspberrypi]

This repository aims to provide a reference implementation for updateable, secure embedded Linux systems that are built with the Yocto build system.

Current features:

  • Basic stack structure
  • Dual boot (recovery/system update schema)
  • Bootcounter support
  • Creation of update bundles
  • Authentication of updates via certificates
  • Some common configurations (like root password, ssh config, etc)
  • Lightweight factory reset (through overlay clearance)
  • Updates over various sources
    • from media device (USB)
    • via network download (ssh)
    • over-the-air (using hawkbit as remote backend, see backend)

Planned features:

  • Application watchdog to determine system health
  • Differential updates (using casync)
  • Support for update bundle encryption
  • Support for sub-device updates via custom hook scripts

Getting started

To have something ready to install on one of the supported boards, you need to

(1) Install the minimal host system requirements (see below).

(2) Pull the repository

git clone https://github.com/JSydll/emx-base-stack

(3) Optional: Provide custom configure for some features (e.g. by setting up a .env file, see below).

(4) Start the build by running

./run-kas build conf/<machine>.yml <image [default: full-image]>

(5) Locate the *.wic image file under build/tmp/deploy/<machine>/ and flash it on the device/ memory card using a tool like Etcher.

Host system requirements

The host system requirements are reduced as much as possible, yet there are some:

  • git (well, who doesn't need it...)
  • bash
  • Docker

For development, it is recommended to also install (and use!)

  • shellcheck

If you want to run the hawkbit Update Backend Server, you'll also need

  • docker-compose

Branching and version control strategy

The branches in this repo are aligned with the Yocto project releases.

Currently supported releases:

  • scarthgap

No longer maintained releases (no further feature development):

  • kirkstone
  • dunfell

To control how upstream changes enter builds in this repo, kas provides lockfiles that specify the specific revisions to be checked out. They were created using

./run-kas dump --lock --inplace conf/kas.<machine>.yml

If you want to update the revisions, you can do so using the same command again with the added option --update. In case you want to work with latest greatest, simply remove the respective lockfile next to the configuration you want to build.

Supported machines and images

The approach is to gradually expand the support on different platforms, starting with some of the widely used development boards, such as the Raspberry Pi.

Currently supported boards:

  • RaspberryPi 3 B+ [MACHINE=raspberrypi3-multiboot]
  • QEMU [MACHINE=qemux86-64-multiboot]

For these, the following images can be built:

  • full-image - Builds the full image (including recovery & system) with the configured sub-images.

  • system-update-bundle - Builds a system update bundle with the configured sub-image.

  • recovery-update-bundle - Builds a recovery update bundle with the configured sub-image.

  • custom-recovery-image - Builds the recovery image only, with custom applications included.

  • custom-system-image - Builds the system image only, with custom applications included.

Configuring the project

The overall configuration is managed via the kas tool. It's main configuration can be found under ./conf.

Besides, the customizable features can be configured using an .env file. See .env.example for a reference of the syntax.

The board specifc integrations can be selected using the appropriate MACHINE name in the run commands (see above).

Building the project

Like the configuration, the build environment is provided by kas. The ./run-kas script provided in this repo is only a shallow wrapper around the scripts copied from upstream kas, currently at version 4.5, like recommended by the respective documentation.

If you want to stay in the bitbake environment (and not only execute a single build command), you can use

./run-kas shell <machine>

Deploying build artifacts

There are two types of build artifacts: full system images and update bundles.

Full system images contain a fully partitioned representation of the system to be installed on the main memory (e.g. the eMMC). Depending on the hardware, this might require specific tools provided by the board manufacturer. In the case of popular single-board-computers like the RaspberryPi, the external SD card needs to be flashed with the image - which is most easily done with a tool like Etcher.

As soon as the device is initially flashed with the expected partitioning and software, update bundles can be installed by bringing them on the device and simply running

sw-mode-control update /path/to/update-bundle.raucb

Of course, this will only work for bundles to update the currently inactive software mode (i.e. the system mode needs to be running to update the recovery software and vice versa). To switch modes, you can use

sw-mode-control start-system|start-recovery

Automated update flow

There is an update-observer service that watches for update packages under /media. Currently, it does not account for running uploads, though, so be sure to first upload to a different location (for example /tmp) and then atomically mv the bundle into the watched folder.

The service will attempt to detect whether a reboot into the other mode is required, perform this reboot, then install the update and eventually reboot to the system mode.

It uses file names to do that, so be sure to prefix your bundles with system-update or recovery-update respectively.

Emulation with QEMU

As mentioned above, there is a qemuarm based configuration that now supports the full multiboot setup. For using it, simply build the full-image and run the emulator with

scripts/run-qemuarm-multiboot

The emulator is configured to expose SSH on port 2222. For running the above script on host, network isolation of the spawned container is turned off, so you should see the listening port when running ss -tlp. When externally starting a container before calling the script, be sure to configure it appropriately.

Logging into the virtual device should then be as easy as

ssh -p 2222 root@localhost

Note that ssh uses lower case -p while scp upper case -P to set the port.

With this, you can inspect the build result, test new applications or install updates.

Contribution

Feel free to contact me in case you have feature proposals or want to contribute.

About

This repository aims to provide a reference implementation for updateable, secure embedded Linux systems that are built with the Yocto build system.

Resources

License

Stars

Watchers

Forks

Packages

No packages published