1 unstable release
0.1.0 | Feb 21, 2024 |
---|
#26 in #hard
2KB
SCARS
SCARS is a Real-Time Operating System (RTOS) designed for hard real-time applications, inspired by the Ada Ravenscar profile. It is a traditional style RTOS with fixed priority preemptive scheduling, and built-in async executor. It is still a work-in-progress, and is not recommended for production systems. Currently only RISC-V and ARM Cortex-M without FPU, as well as pthreads based simulator are implemented. FPU support is planned.
-
It implements the immediate priority ceiling protocol, and provides an unified priority model for threads and interrupts, which allows developers to use the same APIs in threads and interrupts. Mutex protected data can be accessed from interrupt handlers.
-
All threads and interrupt handlers are statically allocated, yet started at runtime, allowing threads and interrupt handlers to capture variables from the environment that they are started in. No need for global static variables.
-
All scheduling operations that depend on the number or order of threads, are executed within a preemption lock that allows interrupts, minimizing the maximum interrupt latency. Interrupt latency is independent from scheduling and number of threads.
-
Any priority based synchronization primitives that are below a priority ceiling, will not prevent higher priority threads or interrupts from executing.
-
Built-in async executor allows associating async tasks to interrupt handlers. Executor is polled on-interrupt, and by the kernel when needed.
Threads
The entry thread is defined with entry
attribute for a function. There can be only
one entry thread. Threads can never exit.
#[scars::entry(name = "main", priority = 1, stack_size = 2048)]
fn main() {
loop {}
}
Other threads are defined with a the thread
attribute macro, which statically
allocates the thread state, including the thread stack. The thread is created by calling
the thread function, and started with the start
method.
const THREAD_PRIO: Priority = Priority::thread(2);
const THREAD_STACK_SIZE: usize = 1024;
const CEILING_PRIO: Priority = THREAD_PRIO;
const CHANNEL_CAPACITY: usize = 1;
#[scars::thread(name = "thread", priority = THREAD_PRIO, stack_size = THREAD_STACK_SIZE)]
fn thread(sender: Sender<u32, CEILING_PRIO>) {
let mut counter: u32 = 0;
loop {
sender.send(counter);
counter += 1;
}
}
#[scars::entry(name = "main", priority = 1, stack_size = 2048)]
fn main() {
let (sender, receiver) = make_channel!(u32, CHANNEL_CAPACITY, CEILING_PRIO);
thread(sender).start();
loop {
let value = receiver.recv();
println!("Received {:?}", value);
}
}
Idle thread
The kernel creates an internal idle thread to be run when no other thread is ready. The idle thread is special in that it does not follow the priority ceiling protocol, and can run even when higher priority threads hold locks.
To run custom code in the idle thread, the user can define the idle thread hook with
the idle_thread_hook
attribute:
#[scars::idle_thread_hook]
fn idle_thread_hook() {
// do something when idle
}
The hook-function can use InterruptLock
and PreemptLock
; however, due to being exempt
from the priority ceiling protocol, it may not acquire a CeilingLock
. An attempt to
acquire a CeilingLock
from the idle thread will result in a runtime error.
Interrupt Handlers
Interrupt handlers are created similarly to threads with interrupt_handler
attribute
macro.
use scars::khal::{Interrupt, Peripherals, pac::EXTI};
use scars::sync::channel::Sender;
const INTERRUPT_PRIO: Priority = Priority::interrupt(2);
const CEILING_PRIO: Priority = INTERRUPT_PRIO;
const CHANNEL_CAPACITY: usize = 10;
#[scars::interrupt_handler(interrupt = Interrupt::EXTI0, priority = INTERRUPT_PRIO)]
fn exti0_handler(sender: Sender<u32, CEILING_PRIO>, mut counter: u32, exti: EXTI) {
// Interrupt handler closure captures `sender` and `counter`
counter += 1;
// Must use non-blocking `try_send` in interrupt
let _ = sender.try_send(counter);
// Clear EXTI0 interrupt flag
exti.pr.write(|w| w.pr0().set_bit());
}
#[scars::entry(name = "main", priority = 1, stack_size = 2048)]
fn main() {
let Peripherals { SYSCFG, EXTI, .. } = Peripherals::take().unwrap();
// Source EXTI0 interrupt from PA0 GPIO
SYSCFG.exticr1.write(|w| unsafe { w.exti0().bits(0) });
// Enable EXTI0 interrupt in EXTI
EXTI.imr.write(|w| w.mr0().set_bit());
// Trigger interrupt from rising edge
EXTI.rtsr.write(|w| w.tr0().set_bit());
let (sender, receiver) = make_channel!(u32, CHANNEL_CAPACITY, CEILING_PRIO);
let counter: u32 = 0;
let exti0 = exti0_handler(sender, counter, EXTI);
exti0.enable();
loop {
let value = receiver.recv();
println!("Handled {:?} interrupts", value);
}
}
Immediate Priority Ceiling Protocol
threads and interrupt handlers have fixed base priorities, and all priority based locking primitives have a priority ceiling, which is the maximum priority of any thread or an interrupt that can acquire the lock. Acquiring a priority based lock immediately raises the active priority of the thread or an interrupt to the ceiling priority of that lock, preventing other threads or interrupts from acquiring the lock. The priority based locking implements mutual exclusion in the scheduler ready queue.
If the ceiling of the lock is at interrupt priorities, then interrupts are blocked up to that priority. Higher priority interrupts are allowed to execute, but since they are above the ceiling, they are never allowed to acquire the lock. The benefit from this, is that whenever interrupt handler executes, it may access priority lock protected data; if a thread or lower priority interrupt handler is holding the lock, the interrupt handler is not allowed to execute in the first place. The downside is that all locking primitives such as mutexes and condition variables, must have the ceiling priority specified by the developer.
Violating the priority ceiling protocol by trying to acquire a lock from a thread or an interrupt handler with higher priority than the lock ceiling priority, will lead to a runtime error.
Synchronization primitives
- Locks:
InterruptLock
,PreemptLock
andCeilingLock
Mutex
Condvar
Channel
- others TBD
Simulator
The scars-khal-sim
crate implements a pthreads based simulator on host. Interrupt
simulation has not yet been implemented.
Running tests
khal-e310x
Configuration in .cargo/config.toml is for QEMU.
$ cargo test --release --package=scars --features="khal-e310x" --target=riscv32imac-unknown-none-elf
khal-sim
$ cargo test --release --package=scars --features="khal-sim" --target=x86_64-unknown-linux-gnu
stm32f4
$ cargo test --release --package=scars --features=khal-stm32f4 --target=thumbv7em-none-eabihf