Skip to content

Developer's Guide

Please note, by default the root account is disabled in Infix NETCONF builds. Meaning, the only way to access the system is with the admin account, which is created based on credentials found in the VPD area -- for Qemu devices this is emulated using qemu_fw_cfg.

For developers this can be quite frustrating to be blocked from logging in to debug the system. So we recommend enabling the root account in the Buildroot make menuconfig system.

make menuconfig
     -> System configuration
        -> [*]Enable root login with password

Important

Please see the Contributing section, below, for details on how to fork and clone when contributing to Infix.

Cloning

When pre-built releases are not enough, for instance when you want to add or modify some Open Source components, you can clone the Infix tree to your PC:

$ mkdir ~/Projects; cd ~/Projects
$ git clone https://github.com/kernelkit/infix.git
..
$ cd infix/
$ git submodule update --init
..

Customer Builds

Customer builds add product specific device trees, more OSS packages, e.g., Frr and podman, and sometimes integrates proprietary software. What's important to remember, however, is that they are all made by setting up Infix as a GIT submodule, similar to how Infix set up a GIT submodule for Buildroot.

So, in addition to using the customer's specific defconfig(s), one must also make sure to update all submodules, otherwise you will likely end up with a broken build.

$ ...
$ git submodule update --init --recursive
                              ~~~~~~~~~~~

Other caveats should be documented in the customer specific trees.

Building

Tip

For more details, see the Getting Started and System Requirements sections of the excellent Buildroot manual.

Buildroot is almost stand-alone, it needs a few locally installed tools to bootstrap itself. The most common ones are usually part of the base install of the OS, but specific ones for building need the following. The instructions here are for Debian/Ubuntu based systems (YMMV):

$ sudo apt install bc binutils build-essential bzip2 cpio \
                   diffutils file findutils git gzip      \
                   libncurses-dev libssl-dev perl patch   \
                   python3 rsync sed tar unzip wget       \
                   autopoint bison flex autoconf automake

To build an Infix image; select the target and then make:

make x86_64_defconfig
make

Online help is available:

make help

To see available defconfigs for supported targets, use:

make list-defconfigs

Test

Working with the regression test framework, Infamy, a few more tools and services are required on your system:

$ sudo apt install jq graphviz qemu-system-x86 qemu-system-arm \
                   ethtool gdb-multiarch tcpdump tshark
..

To be able to build the test specification you also need:

$ sudo apt-get install python3-graphviz ruby-asciidoctor-pdf
..

Development

Developing with Infix is the same as developing with Buildroot. When working with a package, be it locally kept sources, or when using local.mk, you only want to rebuild the parts you have modified:

make foo-rebuild

or

make foo-reconfigure

or, as a last resort when nothing seems to bite:

make foo-dirclean foo-rebuild

As shown here, you can combine multiple build targets and steps in one go, like this:

make foo-rebuild bar-rebuild all run

This rebuilds (and installs) foo and bar, the all target calls on Buildroot to finalize the target filesystem and generate the images. The final run argument is explained below.

YANG Model

When making changes to the confd and statd services, you will often need to update the YANG models. If you are adding a new YANG module, it's best to follow the structure of an existing one. However, before making any changes, always discuss them with the Infix core team. This helps avoid issues later in development and makes pull request reviews smoother.

confd

The Infix src/confd/ is the engine of the system. Currently it is a plugin for systemd-plugind and contains XPath subscriptions to all the supported YANG models.

There are essentially two ways of adding support for a new YANG model:

  • The sysrepo way, or
  • The Infix way, using libsrx (the lydx_*() functions)

The former is well documented in sysrepo, and the latter is best taught by example, e.g., src/confd/src/infix-dhcp.c. Essentially libsrx is a way of traversing the libyang tree instead of fetching changes by XPath.

When working with confd you likely want to enable full debug mode, this is how you do it:

  1. Open the file package/confd/confd.conf
  2. Uncomment the first line set DEBUG=1
  3. Change the following line to add -v3 at the end
    [S12345] sysrepo-plugind -f -p /run/confd.pid -n -- Configuration daemon
    

to:

[S12345] sysrepo-plugind -f -p /run/confd.pid -n -v3 -- Configuration daemon

Now you can rebuild confd, just as described above, and restart Infix:

make confd-rebuild all run

statd

The Infix status daemon, src/statd, is responsible for populating the sysrepo operational datastore. Like confd, it uses XPath subscriptions, but unlike confd, it relies entirely on yanger, a Python script that gathers data from local linux services and feeds it into sysrepo.

To apply changes, rebuild the image:

make python-statd-rebuild statd-rebuild all

Rebuilding the image and testing on target for every change during development process can be tedious. Instead, yanger allows remote execution, running the script directly on the host system (test container):

infamy0:test # ../src/statd/python/yanger/yanger -x "../utils/ixll -A ssh d3a" ieee802-dot1ab-lldp

ixll is a utility script that lets you run network commands using an interface name instead of a hostname. It makes operations like ssh, scp, and network discovery easier.

Normally, yanger runs commands locally to retrieve data (e.g., lldpcli when handling ieee802-dot1ab-lldp). However, when executed with -x "../utils/ixll -A ssh d3a" it redirects these commands to a remote system connected to the local d3a interface via SSH. This setup is used for running yanger in an interactive test environment. The yanger script runs on the host system, but key commands are executed on the target system.

For debugging or testing, you can capture system command output and replay it later without needing a live system.

To capture:

infamy0:test # ../src/statd/python/yanger/yanger -c /tmp/capture ieee802-dot1ab-lldp

To replay:

infamy0:test # ../src/statd/python/yanger/yanger -r /tmp/capture ieee802-dot1ab-lldp

This is especially useful when working in isolated environments or debugging issues without direct access to the DUT.

Upgrading Packages

Buildroot

The Kernelkit team maintains an internal fork of Buildroot, with branches following the naming scheme YYYY.MM.patch-kkit e.g. 2025.02.1-kkit, which means a new branch should be created whenever Buildroot is updated. These branches should contain only changes to existing packages (but no new patches), modifications to Buildroot itself or upstream backports.

The team tracks the latest Buildroot LTS (Long-Term Support) release and updates. The impact of minor LTS release upgrades is expected to have a very low impact and should be done as soon there is a patch release of a Buildroot LTS available.

Depending on your setup, follow the appropriate steps below.

Repo locally cloned already

  1. Navigate to the Buildroot directory

    cd buildroot/
    
  2. Pull the latest changes from KernelKit

    git pull
    
  3. Fetch the latest tags from upstream

    git fetch upstream --tags
    

No local repo yet

  1. Clone the Kernelkit Buildroot repository

    git clone git@github.com:kernelkit/buildroot.git
    
  2. Add the upstream remote

    git remote add upstream https://gitlab.com/buildroot.org/buildroot.git
    
  3. Checkout old KernelKit branch

    git checkout 2025.02.1-kkit
    

Note

Below, it is not allowed to rebase the branch when bumped in Infix.

Continue Here

  1. Create a new branch based on the previous KernelKit Buildroot release (e.g. 2025.02.1-kkit) and name it according to the naming scheme (e.g. 2025.02.2-kkit)

    git checkout -b 2025.02.2-kkit
    
  2. Rebase the new branch onto the corresponding upstream release

    git rebase 2025.02.2
    
  3. Push the new branch and tags

    git push origin 2025.02.2-kkit --tags
    
  4. In Infix, checkout new branch of Buildroot

    cd buildroot
    git fetch
    git checkout 2025.02.2-kkit
    
  5. Commit and push the changes. Remember to update the ChangeLog!

  6. Create a pull request.

Note

Remember to set the pull request label to ci:main to ensure full CI coverage.

Linux kernel

The KernelKit team maintains an internal fork of Linux kernel, with branches following the naming scheme kkit-linux-[version].y, e.g. kkit-6.12.y, which means a new branch should be created whenever the major kernel version is updated. This branch should contain all kernel patches used by Infix.

The team tracks the latest Linux kernel LTS (Long-Term Support) release and updates. The upgrade of LTS minor releases is expected to have low impact and should be done as soon as a patch release of the LTS Linux kernel is available.

Repo locally cloned already

  1. Navigate to the Linux kernel directory

    cd linux
    
  2. Get latest changes from KernelKit

    git pull
    
  3. Fetch the latest tags from upstream

    git fetch upstream --tags
    

No local repo yet

  1. Clone the KernelKit Linux kernel repository

    git clone git@github.com:kernelkit/linux.git
    
  2. Add the upstream remote

    git remote add upstream git://git.kernel.org/pub/scm/linux/kernel/git/stable/linux.git
    
  3. Checkout correct kernel branch

    git checkout kkit-linux-6.12.y
    

Continue Here

  1. Rebase on the upstream release

    git rebase v6.12.29
    
  2. Push changes and the tags

    git push -f origin kkit-linux-6.12.y --tags
    

Move to your Infix source tree

Note

See help of kernel-refresh.sh script for more information.

  1. Generate patches

    make x86_64_defconfig
    cd output
    ../utils/kernel-refresh.sh -k /path/to/linux -o 6.12.28 -t v6.12.29
    
  2. Commit and push the changes. Remember to update the ChangeLog

  3. Create a pull request

Note

Remember to set the pull request label to ci:main to ensure full CI coverage.

Testing

Manual testing can be done using Qemu by calling make run, see also Infix in Virtual Environments, or on a physical device by upgrading to the latest build or "netbooting" and running the image from RAM. The latter is how most board porting work is done -- much quicker change-load-test cycles.

The Infix automated test suite is built around Qemu and Qeneth, see:

With any new feature added to Infix, it is essential to include relevant test case(s). See the Test Development section for guidance on adding test cases.

Reviewing

While reviewing a pull request, you might find yourself wanting to play around with a VM running that exact version. For such occasions, gh-dl-artifact.sh is your friend in need! It employs the GitHub CLI (gh) to locate a prebuilt image from our CI workflow, download it, and prepare a local output directory from which you can launch both make run instances, and run regression tests with make test and friends.

For example, if you are curious about how PR 666 behaves in some particular situation, you can use gh to switch to that branch, from which gh-dl-artifact.sh can then download and prepare the corresponding image for execution with our normal tooling:

gh pr checkout 666
./utils/gh-dl-artifact.sh
cd x-artifact-a1b2c3d4-x86_64
make run

Note

CI artifacts are built from a merge commit of the source and target branches. Therefore, the version in the Infix banner will not match the SHA of the commit you have checked out.

Contributing

Infix is built from many components, when contributing you need to set up your own fork, create a local branch for your change, push to your fork, and then use GitHub to create a Pull Reqeuest.

For this to work as painlessly as possible for everyone involved:

  1. Fork Infix to your own user or organization1
  2. Fork all the Infix submodules, e.g., kernelkit/buildroot to your own user or organization as well
  3. Clone your fork of Infix to your laptop/workstation
  4. Deactivate the Actions you don't want in your fork
  5. Please read the Contributing Guidelines as well!
$ cd ~/Projects
$ git clone https://github.com/YOUR_USER_NAME/infix.git
...
$ cd infix/
$ git submodule update --init
...

Note

When updating/synchronizing with upstream Infix changes you may have to synchronize your forks as well. GitHub have a Sync fork button in the GUI for your fork for this purpose. A cronjob on your server of choice can do this for you with the GitHub CLI tool.


  1. Organizations should make sure to lock the main (or master) branch of their clones to ensure members do not accidentally merge changes there. Keeping these branches in sync with upstream Infix is highly recommended as a baseline and reference. For integration of local changes another company-specific branch can be used instead.