Architecture

architecture
Figure: Bern RTOS architecture; red: high-priority components, blue: lower priority components, green: not part of the Bern RTOS.

The Bern RTOS architecture in the figure illustrates a clear separation of the kernel, which is based on the CPU, the hardware abstraction of the microcontroller peripherals and the isolation of the application.

The segmentation into these three parts serves as a basis for memory protection. Overall, the Bern RTOS applies thread privileges on memory segments and not on every single variable or GPIO.

Kernel

Bern RTOS comprises the kernel, drivers, services and tools. The core part is the kernel, which is responsible for scheduling and communication.

Kernel API

All public kernel functions a thread is allowed to access are present in the kernel application programming interface (API). The API also sets a barrier between threads running in user mode and the kernel in kernel mode. All kernel objects are protected and only accessible through the API.

Kernel Components

The kernel is built in a modular design, so that features can be added, configured by the user and modules can be replaced if needed. The kernel handles scheduling, inter-process communication (IPC), memory management and protection as well as logging. The scope of the features planned for the kernel is similar to FreeRTOS and μC/OS-III.

Architecture Dependent Code

Most parts of a real-time kernel run on any CPU, e.g., scheduling the next thread. Thread switching and memory protection, on the other hand, are different for every CPU architecture. CPUs differ in the number of core registers, exception context, memory protection hardware and instruction set.

In order for the kernel to be portable, all architecture-dependent code is put into one crate (module). Because the kernel only uses the CPU and no microcontroller peripherals, this is only a small part of the kernel.

There is a large variety of microcontroller specialized in different use cases, but most of them use the same CPU architecture (e.g. Arm Cortex-M0+/3/4/7, RISC-V). Thus, once the kernel runs on one CPU, it is compatible with loads of microcontrollers.

Hardware Abstraction Layer

Device drivers and business logic can access the microcontroller peripherals via memory-mapped register. The issue here is that the registers are different for every microcontroller. A hardware abstraction layer (HAL) generalizes peripheral access with a common set of functions for a entire line of microcontrollers.

The Rust community is taking the HAL approach one step further with traits (similar to interfaces in C++) for common peripherals. The traits are then implemented for any microcontroller supported by the Rust community (embedded-hal [31]).

Bern RTOS will not implement a HAL because of two reasons:

  1. A HAL is implemented for every microcontroller. This is tedious work and leads to many half finished implementations instead of one commonly used HAL.
  2. Preexisting code using some HAL should be reusable. You should for example be able to start by writing a bare-metal application and then switch to an RTOS with minimal code changes.

An application can access the HAL functions directly. Access control is enforced by enabling or locking the memory sections of memory-mapped registers.

Applications

An application compromises of a set of threads and communication. Typically, there is just one application running on a microcontroller. With memory isolation, Bern RTOS also allows multiple applications to run in parallel.

An application runs in user mode and has fewer privileges than the kernel.

Also planned to be part of the application space are stacks and services, e.g., TCP/IP, Bluetooth Low Energy (BLE) or FAT file system. These could be specific to Bern RTOS or third-party implementations.