Skip to content

Build your own LXA TAC images and bundles

License

Notifications You must be signed in to change notification settings

a3f/meta-lxatac

 
 

Repository files navigation

Linux Automation GmbH Test Automation Controller Base Image

Main topics:

This set of recipes is used to build the software images and update bundles for the LXATAC provided by the Linux Automation GmbH and can be used to derive customized and pre-configured images for use in your infrastructure.

Building the image as-is

The image building process will compile a lot of software from source code, including the Linux kernel, language interpreters and development tools. This will take some time, use quite a bit of RAM and consume a lot of storage space. You should preferably do this on a powerful build server with a lot of SSD storage instead of on a laptop.

Getting the sources

Obtaining the recipes and required git submodules:

$ git clone --recursive --branch langdale \
  https://github.com/linux-automation/meta-lxatac.git
$ cd meta-lxatac

Generating RAUC signing keys

In order to be able to build RAUC bundles to update your TAC you will need cryptographic keys to sign said bundles. We can, for obvious reasons, not provide you with the keys we use to sign the official bundles, so you will have to generate your own:

$ openssl req -x509 -newkey rsa:4096 -days 36500 -nodes \
  -keyout meta-lxatac-software/files/rauc.key.pem \
  >> meta-lxatac-software/files/rauc.cert.pem
[…]
Country Name (2 letter code) [AU]:DE
State or Province Name (full name) [Some-State]:
Locality Name (eg, city) []:
Organization Name (eg, company) [Internet Widgits Pty Ltd]:Example Project
Organizational Unit Name (eg, section) []:
Common Name (e.g. server FQDN or YOUR name) []:
Email Address []:

You will be asked a few questions about who the keys are generated for. You can however leave most of the fields empty, as shown above.

There is a section later in the README that shows one method to store keys to use during development.

Setting up the build environment

To use bitbake and other commands you will need to source the OpenEmbedded build environment using:

$ source oe-init-build-env

This will set up a build directory and cd into it. Similar to a python venv it will also add new paths to your PATH environment variable, making e.g. the bitbake command available.

If you want to share downloaded files with other Yocto projects you can now specify a DL_DIR in conf/site.conf. You can also add INHERIT += "rm_work" to clean up build artifacts after building a package to save storage space and OPKGBUILDCMD = 'opkg-build -Z gzip -a "-1n"' to use faster compression while packing packages.

Building a bundle

Now you can start the compilation and build the bundle:

$ bitbake lxatac-core-bundle-base

After some time you should end up with with a rauc bundle in:

$ ls tmp/deploy/images/lxatac/lxatac-core-bundle-base-lxatac.raucb
tmp/deploy/images/lxatac/lxatac-core-bundle-base-lxatac.raucb@

To install the rauc bundle you will have to deploy the certificate you have generated to your TAC:

$ scp meta-lxatac-software/files/rauc.cert.pem \
  [IP/HOSTNAME OF YOUR TAC]:/etc/rauc/rauc.cert.pem

The certificates are plain text files, so you can also deploy them by e.g. opening the file in an editor via the debug serial port.

Lastly you can install the bundle on your LXA TAC, using either the web interface or the commandline:

root@lxatac-00010:~ rauc install [PATH_TO_BUNDLE].raucb

Customization

We will explore two approaches to maintaining a custom layer. The former provides a cleaner layer layout and the latter makes it easier to track a specific version of meta-lxatac along with a version of your layer.

Option A - Custom layer inside of meta-lxatac

Create a new layer inside of the already cloned meta-lxatac next to the existing meta-lxatac-bsp and meta-lxatac-software e.g. meta-lxatac-example (a good alternative to example may be your companies name or your own name):

$ ls meta-lxatac-*   
meta-lxatac-bsp:
conf/  recipes-bsp/  recipes-core/  recipes-devtools/ …

meta-lxatac-software:
conf/  files/  recipes-backports/  recipes-core/ …

$ bitbake-layers create-layer meta-lxatac-example

$ ls meta-lxatac-*
meta-lxatac-bsp:
conf/  recipes-bsp/  recipes-core/  recipes-devtools/ …

meta-lxatac-example:
conf/ COPYING.MIT  README  recipes-example/

meta-lxatac-software:
conf/  files/  recipes-backports/  recipes-core/ …

Populate the layer with sample configs to use when first setting up a build directory and remove the config files that were already generated when building the first time:

$ mkdir -p meta-lxatac-example/conf/templates/default
$ cp meta-lxatac-software/conf/templates/default/*.sample \
    meta-lxatac-example/conf/templates/default
# Add "##OEROOT##/../meta-lxatac-example \" to the list of layers:
$ $EDITOR meta-lxatac-example/conf/templates/default/bblayers.conf.sample
$ rm -rv build/conf

And then create a script to set up the build environment including the example layer:

$ sed "s/lxatac-software/lxatac-example/" \
    oe-init-build-env > meta-lxatac-example/oe-init-build-env
$ chmod +x meta-lxatac-example/oe-init-build-env

Every layer called meta-lxatac-* (except for -bsp and -software) is ignored by git via .gitignore, as the intention is for you to maintain your custom layer as a separate git repository. Initialize meta-lxatac-example as its own independent git repository:

$ cd meta-lxatac-example
$ git init
$ git add .
$ git commit -m "Initial Commit"
$ cd ..

Next open a new terminal, where you have not yet sourced the build envionment and use the new script to initialize the environment:

# In a new terminal window
$ source ./meta-lxatac-example/oe-init-build-env

The build of a new bundle should be rather quick, as we did not change anything yet. See below on how to create custom recipes.

# Should be faster than initial build:
$ bitbake lxatac-core-bundle-base

The next section describes another way to maintain your custom layer, if you have followed the instructions above you can skip right over it to learn how to persist your bundle signing keys and write custom recipes.

Option B - meta-lxatac as git submodule in custom layer

Create a new empty meta-lxatac-example layer inside a new project somewhere outside the already cloned meta-lxatac hierarchy.

# We can reuse the artifacts we have already built.
# Just store its location in a envionment variable for now
$ PREV_SSTATE_DIR=$(readlink -e sstate-cache)
$ test -z $PREV_SSTATE_DIR && echo \
  "sstate-cache directory does not exist. " \
  "If you have not built meta-lxatac by itself you can skip the moving. " \
  "Otherwise please set PREV_SSTATE_DIR manually now"

# Do the same for the download directory
$ PREV_DL_DIR=$(readlink -e downloads)
$ test -z $PREV_DL_DIR && echo \
  "downloads directory does not exist. " \
  "If you have not built meta-lxatac by itself you can skip the moving. " \
  "If you have set DL_DIR in your site.conf you should do so again. "
  "Otherwise please set PREV_DL_DIR manually now"

$ cd ${SOMEWHERE_ELSE}
$ mkdir lxatac-yocto
$ cd lxatac-yocto
$ git init

Add meta-lxatac as git submodule and initialize its submodules:

$ git submodule add --branch langdale https://github.com/linux-automation/meta-lxatac.git
$ git submodule update --init --recursive
$ cp meta-lxatac/.gitignore .
$ git add .
$ git commit -m "Initial commit"

Re-use the sstate (build artifacts) from the previous build directory so we do not have to re-build everything, and the download directory so we do not have to re-download when patching an already built package:

$ mkdir build
$ mv ${PREV_SSTATE_DIR} build/sstate-cache
$ mv ${PREV_DL_DIR} build/downloads

Now we will create a new layer for our custom recipes:

$ ls
build/ meta-lxatac/

$ bitbake-layers create-layer meta-lxatac-example

$ ls
build/ meta-lxatac/ meta-lxatac-example/

Populate the layer with sample configs to use when first setting up a build directory:

$ mkdir -p meta-lxatac-example/conf/templates/default
$ cp meta-lxatac/meta-lxatac-software/conf/templates/default/*.sample \
    meta-lxatac-example/conf/templates/default
# Add "##OEROOT##/../../meta-lxatac-example \" to the list of layers:
$ $EDITOR meta-lxatac-example/conf/templates/default/bblayers.conf.sample

And then create a script to set up the build environment including the example layer:

$ sed "s|lxatac-software|lxatac-example|; s|poky|meta-lxatac/poky|"
    meta-lxatac/oe-init-build-env > oe-init-build-env
$ chmod +x oe-init-build-env

Next open a new terminal, where you have not yet sourced the build envionment, and use the new script to initialize the environment:

# In a new terminal window
$ source ./oe-init-build-env

Remember how we crated a set of cryptographic keys to sign the bundle with when first compiling meta-lxatac? We will need to do so again before building a new bundle. But this time we will do so the semi proper way.

RAUC signing keys in your layer

There are different approaches to storing cryptographic keys. Some, like using a hardware security module (HSM), are considered better in terms of security than others. What we are about to do is an approach from the opposite side of the security spectrum, that trades security for ease of development.

First we are going to generate new keys to sign bundles with, like we did in the first step, but this time inside your new custom layer (if you prefer to do so you can also copy over the keys you have generated in the first step):

$ mkdir meta-lxatac-example/files
$ openssl req -x509 -newkey rsa:4096 -days 36500 -nodes \
  -keyout meta-lxatac-example/files/rauc.key.pem \
  -out meta-lxatac-example/files/rauc.cert.pem
[…]
Country Name (2 letter code) [AU]:DE
State or Province Name (full name) [Some-State]:
Locality Name (eg, city) []:
Organization Name (eg, company) [Internet Widgits Pty Ltd]:Example Project
Organizational Unit Name (eg, section) []:
Common Name (e.g. server FQDN or YOUR name) []:
Email Address []:

If you follwed customization Opton A you will also likely want to remove the keys that were generated in the first step, just to make sure that you will not accidently include them in a future meta-lxatac pull request:

$ git restore --staged --worktree meta-lxatac-software/files

If you still want to be able to install official bundles provided by Linux Automation GmbH, you can add the official certificate to the list of allowed certificates using:

$ cat meta-lxatac-software/files/rauc.cert.pem \
  >> meta-lxatac-example/files/rauc.cert.pem

Next you should add the following lines to meta-lxatac-example/conf/layer.conf:

RAUC_KEY_FILE = "${LAYERDIR}/files/rauc.key.pem"
RAUC_CERT_FILE = "${LAYERDIR}/files/rauc.cert.pem"
RAUC_KEYRING_FILE = "${LAYERDIR}/files/rauc.cert.pem"

Now you should be able to build bundles signed using your own keys, that you can share with other developers using git:

$ bitbake lxatac-core-bundle-base

Custom Recipes

What belongs where?

If you want to add a development tool you need or want to change something in the images that could be useful to the general public, you should consider changing it directly in meta-lxatac and submitting a pull request instead of adding a recipe to your custom layer.

A custom layer is mostly useful for software that is not publicly available or configuration that is specific to your deployment.

An example for such a site specific configuration would be to deploy a set of ssh public keys on your LXA TACs:

$ mkdir -p meta-lxatac-example/recipes-core/ssh-keys/files
# Add all public keys that you want to deploy:
$ $EDITOR meta-lxatac-example/recipes-core/ssh-keys/files/authorized_keys

You can now paste the following recipe to meta-lxatac-example/recipes-core/ssh-keys/ssh-keys.bb:

LICENSE = "CLOSED"

inherit allarch

SRC_URI += " authorized_keys "

do_install() {
    install -D -m0600 ${WORKDIR}/authorized_keys ${D}/home/root/.ssh/authorized_keys
}

FILES:${PN} += "/home/root/.ssh/authorized_keys"

Next you want to tell bitbake to build the recipe and install the result to the default image:

$ mkdir -p meta-lxatac-ptx/recipes-core/images
$ cat > meta-lxatac-ptx/recipes-core/images/lxatac-core-image-base.bbappend <<EOF
IMAGE_INSTALL:append = "\
    ssh-keys \
"
EOF

A newly built update bundle should now contain your specified ssh keys:

$ bitbake lxatac-core-bundle-base

Installing Images via USB

This process mirrors the steps we use when we bring up the LXA TAC for the first time, so it should always work, even if the device is completely bricked software-wise and you can no longer deploy any RAUC bundles.

Build the Firmware Files

Flashing the LXA TAC from scratch requires a different set of firmware files than installing a RAUC update bundle. To build these files run:

$ bitbake emmc-image emmc-boot-image tf-a-stm32mp

The required files should then appear in tmp/deploy/images/lxatac/.

Bring the Device into USB Boot Mode

The first step is to bring the LXA TAC into the USB bootmode and load a prelininary bootloader into RAM, which we will use to flash the new image. The USB bootmode is implemented in the SoC bootrom and can thus not be corrupted by software running on the TAC.

Unscrew the four screws holding the frontplate with the display in place. Pay close attention not to break the flat flex cable connecting the display to the mainboard. The display cable can be disconnected by opening the flap on the flat flex connector.

Connect the mainboard to your computer using an USB-C Cable.

Identify the (possibly unpopulated) connector P4 in the lower left corner of the LXA TAC mainboard. It should have pins GND, BT0, BT1 and BT2 marked on the PCB silkscreen. These BT? pins correspond to the boot mode pins of the STM32MP1 and are usually strapped to be LOW, HIGH and LOW respectively, instructing the SoC to boot from the eMMC. Pull pin BT1 to GND by using e.g. a set of tweezers to connect it to GND. This tells the device to boot into the serial update bootrom.

LXA TAC Bootmode Pins BT0-BT2

Power on the device. It should show up in lsusb and dmesg on your host computer. Next you can upload the required pieces of software:

Flashing the software

$ lsusb | grep DFU
Bus 001 Device 038: ID 0483:df11 STMicroelectronics STM Device in DFU Mode

$ cd tmp/deploy/images/lxatac/

# Trusted Firmware-A handles some early hardware setup and then jumps
# into the actual bootloader
$ dfu-util --alt 1 -D tf-a-stm32mp157c-lxa-tac.stm32

# The Barebox bootloader. This command only loads the bootloader into RAM.
# Nothing is being flashed permanently yet.
$ dfu-util --alt 3 -D emmc-boot-image-lxatac.fip

# Exit the USB Boot mode. The TF-A will run and jump into the Barebox in
# RAM.
$ dfu-util --alt 0 -e

# Write the TF-A and bootloader to one of the two redundant eMMC boot
# partitions. These partitions are not partitions in the normal software
# sense but a special eMMC hardware feature to provide atomic updates.
$ fastboot flash bbu-mmc emmc-boot-image-lxatac.img

# Use the Android fastboot support in Barebox to write the firmware image
# (in the form of an Android sparse image (.simg)) to the eMMC storage.
# This will overwrite the data you've stored there.
$ fastboot flash mmc emmc-image-lxatac.simg

You are now done flashing the LXA TAC and can remove the pull-down on BT1 and power cycle it to boot into your newly flashed image.

Troubleshooting

The device should be in bootrom but does not show up on the Host

Disconnect the USB-C cable from the TAC and power cycle the TAC. One of the LEDs on the DUT ethernet port should now blink in quick sucession. The LED is connected to the PA13 pin of the STM32MP1, which is used for debug output from the bootrom. The LED not blinking means that either the wrong boot mode is selected or there is some hardware fault. Check the bootmode pins and the 5V and 3.3V LEDs inside of the device. If you connect the USB-C cable the blinking should stop and the device should show up on the host. Otherwise there may be something wrong with the cable.

Fastboot keeps < waiting for any device >

If fastboot sits idle < waiting for any device >, this can be an error with the permissions on the USB device. Any easy check for this is to run fastboot as root using e.g. sudo.

The TAC disconnects during fastboot

This can happen when the watchdog timer is set up in the Barebox bootloader loaded into RAM via dfu-util. To prevent the TAC from rebooting during fastboot connect to the Barebox console via the debug UART adapter / the USB console provided by Barebox and execute wd -x on the shell. E.g.:

barebox@Linux Automation Test Automation Controller (TAC):/ wd -x
The installed bundle does not start

When flashing the emmc-image only the first of two update partitions is initialized. The second one is set up on first boot and filled for the first time once you install a RAUC bundle on the system for the first time. There is a 50/50 chance that Barebox is currently set up to boot from this non-existing partition, which will fail and reduce its retry counter. If the TAC does not boot after flashing you should power cycle the device about four times in slow succession to make sure the wrong partition is marked as bad and the correct partition is used.

Alternatively you can choose the correct partition using the barebox console via a UART debug adapter:

# state.bootstate.system1.remaining_attempts=0
# state.bootstate.system1.priority=0
# state.bootstate.system0.remaining_attempts=3
# state.bootstate.system0.priority=1
# state -s
# reset

About

Build your own LXA TAC images and bundles

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Languages

  • BitBake 76.4%
  • Shell 11.8%
  • NASL 8.1%
  • HTML 2.9%
  • C++ 0.8%