Direkt zum Hauptbereich

Using a bitbake CI - For Free

This time I want to write something about an issue everybody maintaining a git repo might faced already - CI.

In theory every push and every pull request should have been build with all supported layer-versions... well in theory.

The issue

If you have a local setup it's sometimes hard to switch layer-version - I agree the usage of repo is highly recommended here, as it simplifies such work heavily.
Nevertheless you might need multiple work spaces, which all need a lot of disk space.

Roughly calculated you can expect ~50GB of data per architecture/distro-combination without the usage of "rm_work"-bbclass, round about 15GB if you're using it.

So if you decided to support >3 layer versions of YOCTO, it's a lot of space blocked for a lot of redundant data. Not to mention that you need to build everything now and then to get results, if your code is still working or not.

Solution: CI

This is where you pick a CI provider - Jenkins immediately comes into my mind - to do all the work for you.
When you are with a company you normally have some on-premise hardware where to run such a service - I was lacking of some capable hardware, as this was my spare time project.

So I googled around for a service which does CI for me, something not that pricy, which means free would be best.

Travis CI

As my code is in a public repository, my first pick was Travis CI, which is offering CI service for free for all public repositories - Great!

I added a file called ".travis.yml" to my repository root, pushed it and voila a build ran immediately after my push - I went away to have a coffee or two and as I came back the following notification was in my inbox
Build #1 Failed

A larger stack of limitations

Damn - I had a look at the log and it said
The job exceeded the maximum time limit for jobs, and has been terminated.

which reminded myself of reading the manual first ((sic) - who does this???) - and there it was: the free service is limited in the following ways
  • maximum job runtime = 50min (including setup)
  • 2 vCPUs
Yes - this won't work with bitbake/YOCTO, as a typical "from-scratch" build with just 2 CPU
will take at least 4h.
So I continued to read through the manual of Travis CI.

Digging deeper in Travis CI

Next I found that Travis CI offers a caching mechanism where you can save build artifacts for upcoming builds - and I though why not cache the bitbake downloads and sstate cache (like suggested here), auto terminate the build after ~40min without issuing an error (like suggest here) and let the build iterate more than once to get to the result.
So I did - results were not as good as I expected - It would take >15 builds to get just one image.Besides the fact that the sstate-cache is steadily growing, which limits build time even more, as
within the 50min the "fresh" cache needs to be packed and uploaded - very bad.
A few other pitfalls I recognized on the way
  • logfile limit is 4MB at Travis CI, which is easily reached with a typical bitbake build.I worked around that by running "bitbake -q", which disables print out to console.
    Now you need to run each command with a "travis_wait" prefix, as Travis terminates any command which isn't printing out something at least every 10min
  • As bitbake was run with "bitbake -q" there is no way to tell how much work has been performed within the given time.
    I worked around this by appending
    'echo "Tasks performed "$(grep -o "do_.*Succeeded" build/tmp/log/*/*.log | wc -l' to my build pipeline so I got 'Tasks performed 512' in the build console

Right back to start

Next I thought, let's do the minimal compliant solution and lets just check if bitbake is capable of parsing the setup of multiple layer-repo (incl. mine) - so I just added "bitbake -p" as the final step to my pipeline - First step successfully accomplished - I had the automated information from each
push/pull request if the result is still parsable.

First step done - Now for the real work...

Now I just needed to find a way of building the software at least weekly to check if it is all really working.Luckily I was hosting (and by the time of this writing, I still am) on GitHub and read the advertisement that they offering a CI-like service soon - which is available now, if you subscribe to a beta program - it's currently called Github Actions.
It offers (last update: 2019/10/22)
  • 2 vCPUs
  • 14 GB HDD
  • 6h max. job run time
especially the last point sounded promising.The whole setup could be done over the GitHub UI via browser - and I did coded a basic nightly pipeline in a few minutes - and it worked!
After roughly 4.5h I had the result that my build was working with 
latest repo HEAD.
Finally I was more than happy, I had a fully functional CI without any costs, setup and (most important) without wasting days or weeks in hacking together some weird setup
(For some of you that have been doing stuff with ArchLinux AUR, so you might know what I'm talking about - pure PITA).



Final considerations

My personal setup

On each push / pull request


  • Travis CI checks on any possible combination of layer (currently 4 in my case), if the result is still parsable by bitbake
  • Result is presented within in GitHub at the corresponding commit
  • A merge of PR is only possible if the Travis CI build is good
  • Testing by Travis CI takes ~7-10 minutes in total

Twice a week

  • GitHub Actions checks on any possible combination of layer (currently 4 in my case), if the result is buildable by bitbake
  • Result is presented by mail
  • Testing takes around 5h in total, so it's normally done over night

Real world code

If you're now keen on "real world" code, just head over to my static code analysis layer and have a look at 

or have a look at the build information of past builds at

Unfortunately the GitHub-Action results aren't public, so you have to try that yourself.

If you have any questions, remarks or just want to have a chat, feel free to write a comment.

If you liked this post - be sure to read the follow-up


Kommentare

Beliebte Posts aus diesem Blog

Speedup python on embedded systems

Have you ever considered to use python as a scripting language in an embedded system? I've been using this on recent projects although it wasn't my first choice. If I had to choose a scripting language to be used in embedded I always had a strong preference for shell/bash or lua, because they are either builtin or designed to have a significant lower footprint compared to others. Nevertheless the choice was python3 (was out of my hands to decide). When putting together the first builds using YOCTO I realized that there are two sides to python. the starting phase, where the app is initializing the execution phase, where the app just processes new data In the 2nd phase python3 has good tradeoffs between maintainability of code vs. execution speed, so there is nothing to moan about. Startup is the worst But the 1st phase where the python3-interpreter is starting is really bad. So I did some research where is might be coming from. Just to give a comparison of ...

Making go not a no-go

Anyone that dealt with container engines came across go - a wonderful language, that was built to provide a right way of what C++ intended to do. The language itself is pretty straight forward and upstream poky support is given since ages... In the go world one would just run 1 2 go get github.com/foo/bar go build github.com/foo/bar and magically the go ecosystem would pull all the needed sources and build them into an executable. This is where the issues start... In the Openembedded world, one would have  one provider (aka recipe) for each dependency each recipe comes with a (remote) artifact (e.g. tarball, git repo, a.s.o.) which can be archived (so one can build the same software at a later point in time without any online connectivity) dedicated license information all this information is pretty useful when working is an environment (aka company) that has restrictions, such as reproducible builds license compliance security compliance (for instance no unpatched CVE) but whe...

Using bbclass based on conditions

When you are working with YOCTO build system, you might be aware of the construct .bbclass. For those who are not: these are code snippets which will be injected into the recipe itself. Mostly they add new tasks or provide some generalization for things you often. A good example might be pypi.bbclass. In your recipe you write usually something likes this SRC_URI = "https://www.foo.org/foo.tar.gz" When it comes to python packages a way more elegant way is to use pypi.org . And here does the pypi.bbclass provide some magic - you recipe will just look like this inherit pypi   PYPI_PACKAGE = "foo" the bbclass pypi will automatically translate the variable name PYPI_PACKAGE into a valid URL to fetch the package. Also it will set some internal variables such as HOMEPAGE or S to the correct settings. You see life can be convenient when you know how to do it. If you want to apply a bbclass you can either insert inherit foo.bbclass into each recipe or you...