Tock 2.2 represents two years of Tock development since v2.1.1. This release contains almost 3900 commits made across 840 PRs by 90 contributors (of which 48 are new contributors!). It is the first Tock release that can compile on a stable Rust toolchain, and contains many other important fixes, new subsystems, new platforms, new drivers, and major refactors.
-
Backwards Compatibility
Tock 2.2 extends its system call interface through one new system call (
Yield-WaitFor
), but retains backwards compatbility with Tock 2.1.1 for its core system call interface and all stabilized drivers.In this release, we revised Tock's alarm system call driver implementation to predictably wrap its
ticks
values at(2**32 - 1)
ticks, across all platforms. Before this change, hardware alarm implementations that were less than 32 bit wide would wrap before reaching(2**32 - 1)
ticks, which complicated correct alarm handling in userspace. In Tock 2.2, these alarm implementations are scaled to 32 bit, while also scaling their advertised frequency appropriately. While this change is non-breaking and compatible with the previous alarm implementation, it can expose such scaled alarms to userspace at significantly higher advertisedfrequency
values. Userspace alarm implementations that did not correctly handle such high frequencies may need to be fixed to support this new driver implementation. -
Security and
arch
-crate FixesTock 2.2 includes important and security-relevant fixes for its Cortex-M and RISC-V architecture support.
-
When switching between applications, the RISC-V PMP implementation did not correctly invalidate any additional memory protection regions that are not overwritten by the target app's PMP configuration. Under certain conditions this can allow an application to access private memory regions belonging to a different applications (such as when using IPC).
-
The Cortex-M (Armv7-M) and Cortex-M0/M0+ (Armv6-M) hard fault, interrupt and
svc
handlers contained a bug that could allow an application to execute inprivileged
mode after returning from the handler. This allows an application to execute code at kernel privileges and read / write arbitrary memory.
-
-
Stable Rust Support
This release removes all nightly Rust features from all of Tock's core kernel crates (such as
kernel
,arch/*
, andchips/*
). This allows Tock to be built on the Rust stable toolchain for the first time!We demonstrate this by switching the
hail
board to a stable toolchain in this release. We continue to compile other boards on the Rust nightly toolchain, as this enables some important code-size optimizations (such as by compiling our own, size-optimized core library). -
AppID
, Credentials and Storage PermissionsThis Tock release revisits how applications are identified in the kernel, and introduces a set of mechanisms that allow developers to identify, verify, and restrict applications that are running on a Tock kernel. AppIDs are the core mechanism to enable this and identify an application contained in a userspace binary. AppIDs allow the kernel to apply security policies to applications as their code evolves and their binaries change. We specify AppIDs, Credentials and their interactions with process loading in a draft TRD.
Additionally, we introduce a mechanism to assign applications permissions to access some persistent storage (e.g., keys in a key value store). This mechanism interacts with AppIDs (ShortIDs) and is also specified in a a draft TRD.
-
Major Refactors and Interface Changes
We implement a number of kernel-internal refactors and interface changes:
-
System call drivers are now mandated to either return
Success
orFailure(ErrorCode::NODEVICE)
for acommand
system call with command number0
. Previously, some drivers used this command number to also convey additional information to userspace. This release does not change the interface of any stabilized drivers, which will be updated as part of Tock 3.0. -
Tock 2.2 introduces [a new policy to support external dependencies][external-deps] in the upstream Tock codebase. As part of this effort, we split up the existing, single
capsules
crate into multipe crates (such ascapsules-core
,capsules-extra
, andcapsules-system
) with different guarantees concerning stability and use of external dependencies. Thecore
capsules crate contains capsules deemed essential to most Tock systems, as well as virtualizers which enable a given single peripheral to be used by multiple clients. Other capsules have been moved to theextra
capsules crate. Thesystem
capsules crate contains components that extend the functionality of the Tock core kernel, while not requiringunsafe
. -
Furthermore, the
DeferredCall
andDynamicDeferredCall
subsystems have been replaced with a more lightweight and unified deferred call infrastructure. This new approach has a smaller code size overhead and requires less setup boilerplate code thanDynamicDeferredCall
. -
LeasableBuffer
has been renamed toSubSlice
and features a significantly improved API. Multiple subsystems have been ported to this new type. -
Tock 2.2 introduces "configuration boards": variants of in-tree board definition showcasing certain subsystems or peripherals. These boards (under
boards/configurations
) are implemented by converting some Tock boards into combined "lib + bin" crates and extending these boards. -
Tock can now be built entirely using
cargo
and without its Makefiles. This change also simplifies downstream board definitions: -
A new
StreamingProcessSlice
helper provides a reusable data structure to convey a "stream" of data from capsures to userspace. This is used in Tock's new CAN driver, and is useful for ADC, networking, etc. -
Tock introduces a new interface for custom implementations of the userspace-syscall boundary to hook into the RISC-V trap handler, by specifying which registers are clobbered and providing a generic trampoline to jump to custom code on a trap.
-
-
New Boards
This release features support for 7 new boards in the upstream Tock codebase:
- sma_q3 by @dcz-self in tock#3182
- particle_boron by @twilfredo in tock#3196
- BBC HiFive Inventor by @mateibarbu19 in tock#3225
- SparkFun LoRa Thing Plus by @alistair23 in tock#3273
- makepython-nrf52840 by @bradjc in tock#3817
- Nano33BLE Sense Rev2 by @TheButterMineCutter in tock#3717
- VeeR EL2 simulation target by @wsipak in tock#4118
-
New HILs, Drivers and Features
Tock 2.2 features 6 new HILs:
- CAN bus by @teonaseverin in tock#3301
Buzzer
by @TeodoraMiu in tock#3084DateTime
by @Remus7 in tock#3559CycleCounter
by @codingHahn and @hudson-ayers in tock#3934public_key_crypto/SignatureVerify
by @bradjc in tock#3878Servo
by @inesmaria08 in tock#4126
An additional 40 PRs added support for various hardware peripherals, subsystems and other features.
-
IEEE 802.15.4 and 6LoWPAN Stack
We can now join a Thread network by running OpenThread as a libtock-c userspace implementation, thanks to a major refactor and redesign of Tock's IEEE 802.15.4 and 6LoWPAN stack.
Known issue: UDP transmit functionality is currently broken with a bug / inconsistency between the kernel and libtock-c implementation. When executing the transmit syscall, the libtock-c application fails to provide the src address and fails the error check that occurs for the transmit syscall. For more information, see the Tock 2.2 release testing issue: tock#4272 (comment)
In addition to the above, this release includes a plethora of other fixes, improvements and refactors. You can see the full list of changes at https://github.com/tock/tock/compare/release-2.1...release-2.2
Tock 2.1 has seen numerous changes from Tock 2.0. In particular, the new system call interface introduced with Tock 2.0 has been refined to provide more guarantees to processes with respect to sharing and unsharing buffers and upcalls. Other changes include the introduction of a userspace-readable allow system call, support for new HILs and boards, and various other bug-fixes and improvements to code size and documentation.
-
Breaking Changes
-
The implemented encoding of the system call return variant "Success with u32 and u64" has been changed to match the specification of TRD 104. Accordingly, the name of the
SyscallReturnVariant
enum variant has been changed fromSuccessU64U32
toSuccessU32U64
(tock#3175). -
VirtualMuxAlarm
s now require thesetup()
function to be called in board set up code after they are created (tock#2866).
-
-
Noteworthy Changes
-
Subscribe and allow operations are no longer handled by capsules themselves, but through the kernel's
Grant
logic itself (tock#2906). This change has multiple implications for users of Tock:-
The
Grant
type accepts the number of read-only and read-write allow buffers, as well as the number of subscribe upcalls. It will reserve a fixed amount of space perGrant
to store the respective allow and subscribe state. Thus, to make efficient use ofGrant
space, allow buffer and subscribe upcall numbers should be assigned in a non-sparse fashion. -
Legal allow and subscribe calls can no longer be refused by a capsule. This implies that it is always possible for an application to cause the kernel to relinquish a previously shared buffer through an
allow
operation. Similarly,subscribe
can now be used to infallibly ensure that a given upcall will not be scheduled by the kernel any longer, although already enqueued calls to a given upcall function can still be delivered even after asubscribe
operation. The precise semantics around these system calls are described in TRD 104.
-
-
Introduction of a new userspace-readable allow system call, where apps are explicitly allowed to read buffers shared with the kernel (defined in a draft TRD).
-
Introduction of a read-only state mechanism to convey information to processes without explicit system calls (tock#2381).
-
Improvements to kernel code size (e.g., tock#2836, tock#2849, tock#2759, tock#2823).
-
-
New HILs
hasher
public_key_crypto
-
New Platforms
- OpenTitan EarlGrey CW310
- Redboard Red-V B
- STM32F429I Discovery development board
- QEMU RISC-V 32-bit "virt" Platform
-
Deprecated Platforms
- OpenTitan EarlGrey NexysVideo
-
Known Issues
-
This release was tagged despite several known bugs in non-tier-1 boards, so as to avoid delaying the release. These include:
-
Raspberry Pi Pico: process faults when running IPC examples: tock#3183
-
The cortex-m exception handler does not correctly handle all possible exception entry cases. This is not known to currently manifest on any examples, but could with unlucky timing: tock#3109
-
STM32F303 Discovery:
adc
app runs, but eventually hangs in the app (seems to be caught in the exit loop, but not sure why it gets there) -
STM32F303 Discovery: kernel panics lead to only a partial printout of the panic message before the board enters a reboot loop
-
weact_f401ccu6:
gpio
example fails to generate interrupts on the input pin. This board is likely to be deprecated soon anyway, as it is no longer available for sale.
-
-
-
Many core kernel APIs have been redesigned and rewritten.
-
There are two new userspace system calls:
AllowReadOnly
andExit
. The oldAllow
system call has been renamed toAllowReadWrite
.AllowReadOnly
provides a mechanism for userspace to share a read-only buffer (e.g., constant data stored in flash) to the kernel.Exit
allows a process to complete execution and request that the kernel either terminate or restart it. -
The system call ABI has been rewritten. System calls can now return up to 4 registers of values to userspace on return. The ABI defines the format and structure of the allowed return types. TRD104 documents the new ABI.
-
The calling semantics and requirements for system calls is more clearly defined, especially with respect to calls to the Allow system calls and how buffers are managed. Furthermore, the lifetime of upcalls passed to the kernel with the
subscribe
system call has been defined. To enforce that the kernel doesn't maintain references to upcalls that it shouldn't (so userspace can reclaim any resources they require), upcalls are now managed by the core kernel. The changes to these calling semantics are documented in TRD104. -
Several types in the kernel have changed names, to better reflect their actual use and behavior.
-
AppSlice
is nowReadOnlyProcessBuffer
andReadWriteProcessBuffer
. -
Callback
is nowUpcall
(to distinguish upcalls from the kernel to userspace from general softare callbacks).Upcall
s are now stored in a special block of memory in grant regions and are managed by the kernel rather than drivers. This allows the kernel to enforce their swapping semantics. #2639 -
Platform
is nowSyscallDriverLookup
andChip
is now split intoChip
for chip-specific operations andKernelResources
for kernel operations. -
Driver
is nowSyscallDriver
.
-
-
-
The kernel namespace has been reorganized.
-
tock#2659 reorganizes the kernel namespace. The actual abstractions and types exported were not changed, but their places in the namespace were.
-
Almost everything is now exported as
kernel::module::Type
rather thankernel::Type
. -
/common
is split up into/utilities
and/collections
-
-
There is increased chip and board support.
-
RISC-V support has been extended to support progress and revisions to support microcontroller-type systems, including support for EPMP memory protection.
-
There is support for ARM CortexM0+ and CortexM7.
-
Board support adds:
- Nano RP2040 Connect
- Clue nRF52840
- BBC Micro:bit v2
- WeAct F401CCU6 Core Board
- i.MX RT 1052 Evaluation Kit
- Teensy 4.0
- Pico Explorer Base
- Rapsberry Pi Pico
- LiteX on Digilent Arty A-7
- Verilated LiteX simulation
- ESP32-C3-DevKitM-1
-
-
Major HIL changes
-
All HILs have changed significantly, to be in line with the new types within the kernel.
-
ReturnCode
has been removed from the kernel. HILs that used to returnReturnCode
now returnResult<(), ErrorCode>
, so thatOk
indicates a success result. #2508 -
There is a draft of a TRD describing guidelines for HIL design, which enumerates 13 principles HIL traits should follow.
-
The SPI, I2C, and CRC HILs have changed in how they handle buffers. SPI and I2C now correctly return buffers on error cases, and CRC now relies on
LeasableBuffer
to compute a CRC over a large block of memory. -
Digest has been extended to support multiple digest algorithms: in addition to HMAC SHA256 it now supports SHA224, SHA256, SHA384, SHA512, HMAC SHA384 and HMAC SHA512.
-
The Time HIL has been updated to better support
dyn
references when needed, by adding aConvertTicks
trait. This change is documented in TRD 105 (which, when finalized, obsoletes 101). -
Blanket implementations for UART trait groups have been added. Now, if a structure implements both
uart::Transmit
anduart::Receive
, it will automatically implementuart::UartData
. -
New HILs added:
- key/value store
- 8080 bus (for LCDs)
- text screen
- screen
- touch
-
-
In-kernel virtualizers for the following HILs have been added: AES, RNG, SHA
-
The kernel now checks whether loaded processes are compiled for the running kernel version. Because 2.0 changes the user/kernel ABI, processes compiled for Tock 1.x will not run correctly on a Tock 2.x kernel and vice versa. If the kernel detects that a process is compiled for the wrong kernel version it stops loading processes.
-
There have been changes to kernel internals and the build system to reduce code size. For example, kernel code that was highly replicated in monomorphized functions has been factored out (#2648).
-
All system call driver capsules that do not support use by multiple processes now use grant regions to store state and explicitly forbid access from multiple processes (e.g., #2518).
-
The process console has been improved and can now display memory maps for the kernel and processes.
-
Added
tools/stack_analysis.sh
andmake stack-analysis
for analyzing stack size. -
Improvements to
tools/print_tock_memory_usage.sh
for displaying code size. -
Transitioned uses of deprecated
llvm_asm!()
toasm!()
macro for better compile-time checking (#2449, #2363). -
Make it possible for boards to avoid using code space for peripherals they do not use (e.g., #2069).
-
Bug fixes.
-
Major HIL Changes
None
-
Loading and Restarting Processes Improvements
Processes can now fault and be restarted by the kernel, and #1565 allows a board configuration file to specify the restart policy that the kernel should use.
Process discovery, parsing, and creation was also overhauled in #1480 to remove
unsafe
from the TBF header parsing code. This allowsprocess::load_processes()
to return errors if process loading fails. Boards now need to handle theResult
return type.
-
Major HIL Changes
Three HILs have been revised to better support embedded devices and clean up the interface for users of the HILs.
-
Start on RISC-V Support
#1323, #1323, and #1345 add architecture support and boards to Tock for the RISC-V architecture.
-
Update Userland-Kernel Boundary Interface
#1318 updates the interface for switching to and returning from userspace to be less Cortex-M specific. The functions are more general and do not assume values are passed on the stack.
-
Kernel debug module
-
#1036, #1029, and #997 change
debug::panic
's signature. First, instead of taking a single LED,panic
takes a slice of LEDs as its first argument. Second, the Rust now uses aPanicInfo
struct to pass along information about where a panic occured, anddebug::panic
adopts the same structure. Third, architecture specific assembly code was removed from the kernel crate (including the debug module), requiringdebug::panic
to take in a particlar implementation of thenop
instruction. Finally,debug::panic
takes a reference to the process array (it is permissible to pass an empty array instead, but you won't get any information about process state on panic).Boards most likely call
debug::panic
from theirpanic_fmt
function:#[lang = "panic_fmt"] pub unsafe extern "C" fn panic_fmt(args: Arguments, file: &'static str, line: u32) -> ! { let led = ...; let writer = ...; debug::panic(led, writer, args, file, line) }
should now be:
use core::panic::PanicInfo; ... #[panic_implementation] pub unsafe extern "C" fn panic_fmt(pi: &PanicInfo) -> ! {[lang = "panic_fmt"] let led = ...; let writer = ...; debug::panic(&mut [led], writer, pi, &cortexm4::support::nop, &PROCESSES)
-
#1046 changes how the debug module in the kernel crate is structured. Instead of being a pseudo-process, debug is now treated more like a capsule, and needs a UART object to be passed to it. This means that
main.rs
needs to be updated to correctly set this up.First, if the debug UART bus is shared with console (or anything else), and this is likely the case, then a UART mux needs to be created. This is going to look slightly different depending on the underlying MCU, but for the SAM4L this looks like:
let uart_mux = static_init!( MuxUart<'static>, MuxUart::new( &sam4l::usart::USART0, // Choose the correct UART HW bus &mut capsules::virtual_uart::RX_BUF, 115200 ) ); hil::uart::UART::set_client(&sam4l::usart::USART0, uart_mux);
With the mux created, a user of the UART bus must be defined. This will be passed to the debug module.
let debugger_uart = static_init!(UartDevice, UartDevice::new(uart_mux, false)); debugger_uart.setup();
The following is the actual debug module, and must be created to use the
debug!()
macro. If debug is sharing a UART bus then the above mux and device is necessary, but if it is on a dedicated UART bus then that UART module can be passed in here instead.let debugger = static_init!( kernel::debug::DebugWriter, kernel::debug::DebugWriter::new( debugger_uart, // Replace with just a HW UART if no sharing is needed. &mut kernel::debug::OUTPUT_BUF, &mut kernel::debug::INTERNAL_BUF, ) ); hil::uart::UART::set_client(debugger_uart, debugger);
Finally, to get around Rust sharing rules, we need to create this wrapper:
let debug_wrapper = static_init!( kernel::debug::DebugWriterWrapper, kernel::debug::DebugWriterWrapper::new(debugger) ); kernel::debug::set_debug_writer_wrapper(debug_wrapper);
-
-
Reorganization of the kernel crate: The kernel crate has been restructured to enable many improvements to Tock, and to move to a more consistent design between the kernel crate and other parts of Tock. This change has happened through several pull requests: #975, #1044, #1109, #1111, #1113, #1115, #1171, and #1191.
The primary motivation for this is making the kernel crate architecture agnostic, so that Tock can be ported non Cortex-M platforms (#985).
A part of this reorganization is the introduction of Capabilities, or a compile-time access control mechanism in Tock based on being able to forbid
unsafe
code. Capabilities restrict what code in Tock can call certain sensitive functions, likeload_processes()
.-
The
Chip
in main.rs has to be instantiated withstatic_init!
to ensure it has a long enough lifetime. Now:let chip = static_init!(sam4l::chip::Sam4l, sam4l::chip::Sam4l::new());
-
Capabilities need to be created. Creating a capability requires the ability to call
unsafe
, so capsules cannot create capabilities, and instead must be passed the capability if they need access to protected functions.let process_management_capability = create_capability!(capabilities::ProcessManagementCapability); let main_loop_capability = create_capability!(capabilities::MainLoopCapability);
-
There is now a
Kernel
struct that needs to be instantiated by the board.Kernel
has a method for the kernel's main loop, instead of a global function in the kernel's base module. Board configurations (i.e. each board'smain.rs
) as a result need to instantiate a statically allocate this new struct.let board_kernel = static_init!(kernel::Kernel, kernel::Kernel::new(&PROCESSES)); board_kernel.kernel_loop(&hail, chip, Some(&hail.ipc), &main_loop_capability);
-
load_processes
takes theKernel
struct as an additional first argument, thechip
as a new second argument, and the required capability as the last argument. Creating aProcess
(whichload_processes()
does) requires a reference to the chip because the process object needs to have access to architecture-specific context switching functions, as well as chip-specific MPU functions.kernel::procs::load_processes( board_kernel, chip, &_sapps as *const u8, &mut APP_MEMORY, &mut PROCESSES, FAULT_RESPONSE, &process_management_capability, );
-
Creating a grant requires a capability, as not just any code should be able to allocate memory in the grant regions.
let memory_allocation_cap = create_capability!(capabilities::MemoryAllocationCapability);
To use:
board_kernel.create_grant(&memory_allocation_cap)
Creating grants is now handled through the main
Kernel
struct so that it can check that no grants are created after processes are setup in memory, since grants require space allocated in process memory.
-
-
#1032 updates the ADC HIL to explicitly specify the resolution of the sample and to clarify that samples are always left-aligned in the
u16
buffer. Previously, the only ADC implementation happened to be 12 bits and left-aligned, which callers only assumed. It also added a method to (if possible) report the reference voltage, which can be used to convert raw ADC samples to absolute voltages. Implementers of the ADC HIL must implement two new methods:/// Function to ask the ADC how many bits of resolution are in the samples /// it is returning. fn get_resolution_bits(&self) -> usize; /// Function to ask the ADC what reference voltage it used when taking the /// samples. This allows the user of this interface to calculate an actual /// voltage from the ADC reading. /// /// The returned reference voltage is in millivolts, or `None` if unknown. fn get_voltage_reference_mv(&self) -> Option<usize>;
-
UART HIL Refinements: This release saw several updates to the UART HIL, summarized in the UART HIL tracking issue.
-
#1073 removes
initialize
from the UART HIL. Implementations will need to disentangle board-specific initialization code, such as enabling the peripheral or assigning pins, from UART configuration code, such as baud rate or parity. Initialization is no longer part of the UART HIL and should be performed by the top-level board before passing the UART object to any other code. UART configuration is now controlled by the newconfigure
HIL method:/// Configure UART /// /// Returns SUCCESS, or /// /// - EOFF: The underlying hardware is currently not available, perhaps /// because it has not been initialized or in the case of a shared /// hardware USART controller because it is set up for SPI. /// - EINVAL: Impossible parameters (e.g. a `baud_rate` of 0) /// - ENOSUPPORT: The underlying UART cannot satisfy this configuration. fn configure(&self, params: UARTParameters) -> ReturnCode;
-
-
#1145 rewrites the HILs for random number generation. There are now two HILs,
entropy
andrng
(random number generation). They differ in the guarantees they give about the bits they produce. Theentropy
traits guarantee high entropy bits: 1 bit of entropy per bit generated, such that every bit generated has an equal chance of being 0 or 1 and is independent of any other bit produced by the trait: that observing the stream of bits provides zero information on what the future bits will be. Entropy's guarantees make it suitable for use in security and cryptography. Therng
traits provide bits that are assured to satisfy all standard NIST randomness tests, but do not promise that future bits cannot be guessed from past ones. E.g., the bits are random but not robust against an adversary.It also adds library components for converting between different entropy sources as well as converting an entropy source into a random number generator (but not a random number generator into an entropy source!). Any software that needs entropy for security or cryptography should use an
entropy
trait and not anrng
trait. -
Updates to linker and toolchain: As of #993 and #1031, the Tock kernel no longer requires GCC for compilation, and entirely uses the LLVM toolchain.
-
Boards now need to explicitly define room for the kernel stack. Something like the following should be in the board's main.rs:
/// Dummy buffer that causes the linker to reserve enough space for the stack. #[no_mangle] #[link_section = ".stack_buffer"] pub static mut STACK_MEMORY: [u8; 0x1000] = [0; 0x1000];
-
There are numerous changes to the shared board linker file. Individual boards need to be updated to not use variables, and instead define the entire
MEMORY
section:/* Memory Spaces Definitions, 448K flash, 64K ram */ /* Bootloader is at address 0x00000000 */ MEMORY { rom (rx) : ORIGIN = 0x00010000, LENGTH = 0x00020000 prog (rx) : ORIGIN = 0x00030000, LENGTH = 0x00040000 ram (rwx) : ORIGIN = 0x20000000, LENGTH = 0x00020000 } MPU_MIN_ALIGN = 8K;
-