Keyboard shortcuts

Press or to navigate between chapters

Press S or / to search in the book

Press ? to show this help

Press Esc to hide this help

Introduction

nano-ros is a lightweight ROS 2 client library for embedded real-time systems. It runs on bare-metal microcontrollers, FreeRTOS, NuttX, ThreadX, and Zephyr, as well as Linux and *BSD. The entire core stack is no_std compatible.

flowchart TB
    App["<b>Application</b><br/>Rust / C / C++ node code"]
    Core["<b>nano-ros core</b><br/>Executor · Node · pub/sub · services · actions · CDR"]
    RMW["<b>RMW backend</b><br/>Zenoh · XRCE-DDS · Cyclone DDS · custom"]
    Plat["<b>Platform</b><br/>clock · heap · threads · sleep · sockets · libc"]
    Wire(["ROS 2 / DDS / Zenoh wire"])
    App --> Core --> RMW --> Plat
    RMW -. wire-compatible .-> Wire

Four layers, swappable independently: the same node code runs over any RMW backend on any platform. Here is a complete Linux publisher — register a backend, open the executor, publish std_msgs/Int32 on /chatter once a second:

use nros::prelude::*;
use std_msgs::msg::Int32;

fn main() {
    // Pick the backend at compile time; this one line registers it.
    nros_rmw_zenoh::register().unwrap();

    let config = ExecutorConfig::new("tcp/127.0.0.1:7447").node_name("talker");
    let mut executor: Executor = Executor::open(&config).unwrap();

    let mut node = executor.create_node("talker").unwrap();
    let publisher = node.create_publisher::<Int32>("/chatter").unwrap();

    let mut count = 0i32;
    executor
        .register_timer(nros::TimerDuration::from_millis(1000), move || {
            publisher.publish(&Int32 { data: count }).unwrap();
            count += 1;
        })
        .unwrap();

    executor.spin_blocking(SpinOptions::default()).unwrap();
}

The same program in C and C++ is in the First Node guides: Rust · C · C++. When a project grows beyond one node, continue with Multi-Node Project Layout.

Key Features

  • Minimal stack — three software layers (application, nano-ros, transport). Lean dependency tree, fast compile times.
  • Pluggable middleware — choose Zenoh (agent-less, direct peer communication), XRCE-DDS (agent-based), or Cyclone DDS (RTPS wire-compatible with stock ROS 2) at compile time. Same application code regardless of backend.
  • Rust-first with C API — the core is written in Rust for memory safety and ergonomics, with a thin C FFI (Foreign Function Interface) layer following rclc conventions.
  • True no_std — runs on bare-metal Cortex-M3 with no heap allocator. The alloc and std features are opt-in.
  • Standalone toolingnros generate-rust produces message bindings without a ROS 2 installation (bundled interface definitions).
  • Formally verified — 160 Kani bounded model checking harnesses and 102 Verus deductive proofs cover CDR serialization, scheduling, and protocol correctness.
  • ROS 2 compatible — interoperates with standard ROS 2 nodes via rmw_zenoh_cpp, or directly over RTPS with rmw_cyclonedds_cpp (same wire protocol, no key rewriting). Topics, services, and actions work across the boundary.

Quick board check — does it work on the board I have today?

Vendor / form factorChipRTOS / no-RTOSLanguagesExample in repoROS 2 interop
ARM MPS2-AN385 (QEMU)Cortex-M3FreeRTOS / bareRust C C++ ¹examples/qemu-arm-{freertos,baremetal}/Verified
ST STM32F4-DiscoveryCortex-M4FFreeRTOS / bareRust ²board crate nros-board-stm32f4Verified
Espressif ESP32-C3RISC-V (RV32)ESP-IDFRust C C++integrations/nano-ros/Verified
Espressif ESP32-C3 (QEMU)RISC-VbareRustexamples/qemu-esp32-baremetal/Verified
QEMU virt RISC-V64RV64GCThreadXRust Cexamples/threadx-riscv64/Verified
Linux hostx86-64 / aarch64ThreadX simRust Cexamples/threadx-linux/Verified
QEMU Cortex-A9 / virtCortex-A9NuttX / ZephyrRust C C++examples/nuttx/, Zephyr samples/Verified
Pixhawk 4 / 6XSTM32F7 / H7NuttX (PX4)C++integrations/px4/module-template/Ready ⁴
Generic Cortex-M0+/M4/M7≥ 64 KB SRAMRTOS of choiceRust C C++Use your board’s vendor BSP + integrations shellsPattern shown

Legend: Verified = booted + tested in CI. Ready = builds and runs but no in-CI gate yet — drop into the matching examples/<plat>/ to compile and try.

Footnotes — ¹ MPS2-AN385 bare-metal is Rust-only (nros-c / nros-cpp need an RTOS for libc / heap). ² STM32F4 Rust path is the canonical target for the bare-metal board crate; FreeRTOS variant uses the shared nros-board-freertos glue. ³ ESP32-S3 needs the xtensa-esp32s3-none-elf Rust target via the espup toolchain installer (not rustup — Xtensa targets aren’t in upstream rust). ⁴ PX4 path is via the external-module template in integrations/px4/ — C++ only because PX4’s uORB binding is C++-only.

Supported platforms (by RTOS)

PlatformRTOSNetwork StackTargets
POSIXLinux / *BSDOS socketsx86-64, aarch64
Bare-metalNonesmoltcpCortex-M3, ESP32-C3, STM32F4
FreeRTOSFreeRTOSlwIPCortex-M3 (QEMU)
NuttXNuttXBSD socketsCortex-A7 (QEMU)
ThreadXThreadXNetX DuoRISC-V 64 (QEMU), Linux sim
ZephyrZephyrZephyr socketsVarious boards

RMW Backends

nano-ros supports several middleware backends, selected at compile time by adding the backend crate as a dependency:

  • Zenoh (nros-rmw-zenoh) — peer-to-peer via zenoh-pico. No agent process. Compatible with ROS 2 rmw_zenoh_cpp.
  • XRCE-DDS (nros-rmw-xrce-cffi) — agent-based via Micro-XRCE-DDS. Compatible with micro-ROS agent.
  • Cyclone DDS (nros-rmw-cyclonedds) — C++ shim; full RTPS wire-compat with stock rmw_cyclonedds_cpp.

Application code is identical regardless of backend — switch with a single Cargo feature flag or Zephyr Kconfig option.

Project Status

nano-ros is under active development. Core capabilities are functional and exercised in CI; see the platform chapters for per-target detail.

CapabilityStatus
Pub/SubComplete
ServicesComplete
ActionsComplete
ParametersComplete
ROS 2 interopComplete
Zenoh backendComplete
XRCE-DDS backendComplete
Cyclone DDS backendComplete (native + embedded; some embedded action paths in progress)
Zephyr supportComplete
QEMU bare-metalComplete
C APIComplete
C++ APIComplete
Message codegenComplete

How This Book Is Organized

  • Getting Started — install toolchains, build your first app, connect to ROS 2.
  • Concepts — understand the architecture, feature system, and backend model.
  • Guides — step-by-step walkthroughs for message generation, QEMU testing, and ESP32 development.
  • Platforms — per-RTOS setup and configuration.
  • Reference — API details, environment variables, build commands, and wire protocol.
  • Advanced — formal verification, real-time analysis, safety features, and contributing.

Choose Your Entry

Different readers want different paths through this book. Pick the shoe that fits and jump straight to the right page.

🧪 I’m taking a glance

You heard about nano-ros and want to know in 5 minutes whether it’s worth playing with.

🔌 I have an ESP32 on my desk right now

Already have hardware? Two-step path:

  1. Linux firstFirst Node — Rust on your host to verify the stack in ~10 minutes (nros setup native --rmw zenoh then cargo run).
  2. Then ESP32 — once Linux works, follow ESP32 (esp-hal) for the Rust cross-compile path. You need a second machine (or the host itself) running zenohd on your Wi-Fi network — the board needs network reach to the router. For a C-only path use ESP32 (ESP-IDF component) if you already have ESP-IDF set up.

🚀 I want to get started shipping something

You’ve decided to use nano-ros and want a working talker on Linux first, then maybe move to an MCU.

  1. Install + first build — install the nros CLI, then nros setup native --rmw zenoh.
  2. First Node in your language: Rust · C · C++.
  3. Building two or more nodes? Move next to Multi-Node Project Layout.
  4. Troubleshooting — First 10 Minutes if anything goes sideways.
  5. Cross-compile for an RTOS via the Embedded Starters section.

🔬 I’m evaluating capabilities

You’re a senior engineer or tech lead assessing nano-ros for adoption. You want to see scope of coverage, performance bounds, verification status, and trade-offs before committing.

💼 I’m scoping nano-ros for a fleet / product line

You’re a PM, CTO, or technical buyer. You want license terms, supplier reach, deployment patterns, and risk signals before you write the memo.

Still not sure?

Read the Introduction for the one-page overview. Every section above branches from there.

Setup Compared to Standard ROS 2

This page is for ROS 2 users who already know the normal desktop flow: install a ROS distro, create a workspace, run rosdep, build with colcon, and select an RMW at runtime with RMW_IMPLEMENTATION.

nano-ros keeps the workspace and package vocabulary, but changes the setup boundary because it targets embedded and RTOS builds.

Standard ROS 2 Flow

A typical ROS 2 application starts from a distro install:

source /opt/ros/humble/setup.bash
mkdir -p ~/ros2_ws/src
cd ~/ros2_ws
rosdep install --from-paths src --ignore-src -y
colcon build
source install/setup.bash

The middleware implementation is selected at process startup:

export RMW_IMPLEMENTATION=rmw_cyclonedds_cpp
ros2 run my_pkg my_node

That model assumes shared libraries, a hosted OS, and runtime plugin loading.

nano-ros Flow

Where standard ROS 2 installs a distro and resolves system packages with rosdep, nano-ros provisions a per-board toolchain with one command. nros setup replaces the distro install + rosdep: it ships prebuilt toolchains per platform per RMW — the cross-compiler, emulator, RMW host daemon, and SDK sources for a board are fetched from a pinned index into a shared store (${NROS_HOME:-~/.nros}/sdk). You do not install cross-toolchains by hand, and you do not need a ROS distro on the machine.

# 1. Build the in-tree nros CLI (analogous to installing a ROS distro, Phase 218):
source ./activate.sh        # OR: direnv allow / source ./activate.fish
just setup-cli              # builds packages/cli/target/release/nros

# 2. Provision a board + RMW (analogous to `rosdep install`):
nros setup native --rmw zenoh

# 3. Build + run an example (the nano-ros source is vendored in your project):
cd examples/native/rust/talker
cargo run

For embedded targets, name the board instead of native; nros setup fetches the matching prebuilt cross-toolchain + emulator + SDK:

nros setup qemu-arm-freertos --rmw zenoh     # arm-none-eabi-gcc, qemu, FreeRTOS+lwIP
nros setup zephyr            --rmw zenoh     # Zephyr west workspace + SDK bits
nros setup qemu-arm-nuttx    --rmw zenoh     # arm-none-eabi-gcc, qemu, NuttX

Useful flags: nros setup --list (every package + version), nros setup <board> --dry-run (resolve + print the plan, fetch nothing), nros setup --licenses (license-gated packages). See Supported Boards for the board list and nros CLI for every subcommand.

Contributors working on nano-ros itself drive the same index through justjust <module> setup calls nros setup <board> under the hood, so the provisioned toolchains are identical.

Choosing platform + RMW

Unlike standard ROS 2, RMW and platform are compile-time choices — there is no runtime RMW_IMPLEMENTATION switch on embedded targets (no dlopen). The pair is selected via CMake cache vars + Cargo features:

# Each example is a standalone CMake project that pulls nano-ros in
# via add_subdirectory.
set(NANO_ROS_PLATFORM freertos)
set(NANO_ROS_RMW      zenoh)
set(NANO_ROS_BOARD    mps2-an385-freertos)
add_subdirectory(<repo-root>  nano_ros)

target_link_libraries(my_app PRIVATE NanoRos::NanoRos)
nros_platform_link_app(my_app)
nano_ros_link_rmw(my_app RMW zenoh)

Multi-RMW bridges (one binary, two or more backends) use Executor::open_with_rmw("<name>", ...) + node_builder.rmw("<name>") — see Cross-backend Bridges.

What Stays Familiar

  • Workspace layout: one source checkout next to your packages.
  • Package metadata: downstream packages still use package.xml.
  • ROS vocabulary: nodes, publishers, subscriptions, services, actions, QoS profiles, parameters, and message packages keep ROS-shaped names.
  • colcon build still works as a consumer-side build for POSIX workspaces that already use it; embedded targets use cmake, cargo, west, or idf.py directly.
  • Interop: POSIX nano-ros nodes can communicate with standard ROS 2 nodes through compatible RMW backends (Zenoh, Cyclone DDS, XRCE).

What Changes

  • Source-only. No binary SDK tarball, no crates.io umbrella crate, no Arduino zip / ESP-IDF binary component / GitHub Releases artifact. The locked policy is git clone --branch=v<X.Y.Z> + in-tree build.
  • Per-board provisioning, no rosdep. nros setup <board> --rmw <rmw> is the single setup command. It fetches prebuilt toolchains (cross-gcc, emulator), the RMW host daemon, and SDK sources for exactly that board+RMW into ~/.nros/sdk — no system-wide package install, no ROS distro. (The just <module> setup recipes call the same command for contributors.)
  • Compile-time RMW + platform. Embedded targets can’t dlopen, so the RMW and platform combination is locked in by CMake cache vars (NANO_ROS_PLATFORM, NANO_ROS_RMW) and Cargo features at build time.
  • No install prefix. removed just install-local and every install(...) rule; consumers pull nano-ros into their build via add_subdirectory(<repo-root>). The integration shells under integrations/<rtos>/ re-export the same root CMake under each RTOS’s native package manager.
  • Generated bindings in-tree. Message codegen lands under <your-package>/generated/ (or OUT_DIR for Cargo builds), not in an installed ROS message library.
  • Configuration is build-time on embedded. Runtime env vars (ROS_DOMAIN_ID, NROS_LOCATOR — legacy alias ZENOH_LOCATOR, …) work on POSIX; embedded targets resolve config via CMake cache, Kconfig (Zephyr), Cargo features, or a sidecar nros.toml ([node] + [[transport]]).

Next Step

Continue with Installation, then run the ROS 2 Interoperability example before moving to a platform-specific guide.

Installation

nano-ros is distributed as source, vendored into the consumer’s project tree (git submodule, west manifest, ESP-IDF component, etc.) and built in-tree via add_subdirectory(nano-ros). There is no binary tarball, no system-wide install step, no find_package(NanoRos).

activate.sh is the after-install step, NOT a prereq. Sourcing activate.sh (or activate.fish, or direnv allow) only wires the workspace env into a new shell — it presumes the nros binary already exists at packages/cli/target/release/nros. The three bootstrap paths below (A/B/C) are what produce that binary; the activate file is what makes it findable in every subsequent shell.

See build-as-subdirectory.md for the canonical user incantation (4-line CMakeLists.txt). This page walks the surrounding workspace + per-target setup choices.

Pattern A: nano-ros lives inside your ROS 2 workspace

The recommended layout is to clone nano-ros into your workspace’s src/ directory alongside your packages:

~/ros2_ws/
├── src/
│   ├── nano-ros/             # <-- this repo
│   ├── pkg_a/                # your package(s)
│   ├── pkg_b/
│   └── …
└── (build/, install/, log/ — generated by colcon)

colcon build discovers nano-ros + each user package via package.xml, builds them in dependency order, and shares one nano-ros build across every consuming package. Users never run cmake manually — the per-user package’s CMakeLists.txt does add_subdirectory(../nano-ros nano_ros) under the hood.

A complete working example lives at examples/templates/multi-package-workspace/.

Pattern B: nano-ros as a third-party subdirectory

For C/C++ projects that don’t use colcon at all:

~/my_project/
├── CMakeLists.txt
├── src/main.c
└── third_party/
    └── nano-ros/             # git submodule

The top-level CMakeLists.txt:

cmake_minimum_required(VERSION 3.22)
project(my_app LANGUAGES C)

# `NROS_RMW` is the user-facing cache var (overridable via
# `-DNROS_RMW=<rmw>`); forward it to `NANO_ROS_RMW`, the var the
# nano-ros add_subdirectory reads. Matches the canonical example
# shape in examples/native/c/talker/CMakeLists.txt.
set(NANO_ROS_PLATFORM posix)
set(NROS_RMW "zenoh" CACHE STRING
    "Active RMW (zenoh|xrce|cyclonedds) — selects the backend linked into my_app.")
set(NANO_ROS_RMW "${NROS_RMW}")
add_subdirectory(third_party/nano-ros nano_ros)

add_executable(my_app src/main.c)
target_link_libraries(my_app PRIVATE NanoRos::NanoRos)
nros_platform_link_app(my_app)

nros_platform_link_app transitively wires the selected RMW backend on POSIX — no explicit nano_ros_link_rmw() call is needed. That is the entire consumption shape. See build-as-subdirectory.md for the full walkthrough.

Provision your toolchain with nros setup

nros setup is the single command that prepares a machine to build nano-ros for a given board. It ships prebuilt toolchains per platform per RMW — the cross-compiler, emulator, RMW host daemon, and any SDK sources a board needs are fetched from a pinned index and placed in a shared store (~/.nros/sdk). You do not install cross-toolchains by hand, and you do not need ROS 2 on the machine.

1. Get the nros CLI onto PATH

Pick one path from a fresh checkout — just is NOT a prereq.

A. Bare machine (no Rust, no just, no cargo):

git clone https://github.com/NEWSLabNTU/nano-ros.git
cd nano-ros
./scripts/bootstrap.sh base

Installs rustup, just, builds the in-tree nros CLI at packages/cli/target/release/nros, leaves the binary on PATH for this shell.

B. Already have cargo (most contributors):

cargo build --release --manifest-path packages/cli/Cargo.toml --bin nros
export PATH="$PWD/packages/cli/target/release:$PATH"

C. Tagged release, no Rust at all:

./scripts/install-nros-prebuilt.sh

Downloads the matching nros-<triple>.tar.gz from the GitHub release, sha256-verifies, installs to packages/cli/target/release/nros.

All three paths produce the per-checkout binary at packages/cli/target/release/nros. One checkout = one CLI version = one runtime ABI — no global install, no ~/.nros/bin PATH skew across worktrees.

2. Activate the workspace (every subsequent shell)

The bootstrap path you ran in §1 left nros on PATH for that shell only. Every new shell needs the workspace env wired in. Pick whichever fits your shell — all three wire the same env exports + PATH entries from the Phase 218.C SSoT activate.sh:

direnv allow                  # auto-activates on `cd nano-ros` (recommended)
source ./activate.sh          # bash / zsh, one-shot per shell
source ./activate.fish        # fish, one-shot per shell

The activate file also sources /opt/ros/humble/setup.bash if present (required by nros generate-rust + cyclonedds codegen + rmw_zenoh interop tests) and exports NROS_REPO_DIR.

3. Provision a board (+ RMW)

nros setup <board> --rmw <zenoh|xrce|cyclonedds>

nros setup resolves the board’s package set union the RMW’s host packages and fetches them all — prebuilt cross-toolchain + emulator + RMW daemon + board SDK sources. --rmw defaults to zenoh.

CommandProvisions
nros setup nativehost build; the zenoh router (zenohd)
nros setup native --rmw xrcehost build; the Micro-XRCE-DDS agent
nros setup native --rmw cycloneddshost build; Cyclone DDS runtime + idlc
nros setup qemu-arm-freertosarm-none-eabi-gcc, patched qemu-system-arm, FreeRTOS + lwIP sources, zenohd
nros setup qemu-arm-nuttxarm-none-eabi-gcc, qemu, NuttX sources
nros setup qemu-riscv64-threadxriscv64-*-gcc, qemu, ThreadX/NetX sources
nros setup threadx-linuxThreadX POSIX-sim sources
nros setup esp32the Espressif toolchain bits the esp-hal path needs
nros setup zephyrthe Zephyr west workspace + Zephyr SDK bits
nros setup mps2-an385 / stm32f4 / qemu-arm-baremetalbare-metal arm-none-eabi-gcc + qemu

Useful flags:

nros setup --list            # every package in the index + its version
nros setup <board> --dry-run # resolve + print the plan, fetch nothing
nros setup --licenses        # license-gated packages + how to install them
nros setup --tool qemu       # one tool by name
nros setup --source freertos-kernel   # one source package by name

Each board’s exact package set lives in the index; run nros setup <board> --dry-run to see precisely what a board pulls. See Supported Boards for the full board list and nros CLI for every subcommand.

Heads-up before your first example. Every nano-ros example (Linux talker, FreeRTOS talker, …) connects to its RMW host daemon at startup — zenohd for zenoh, the Micro-XRCE-DDS agent for xrce. Cyclone DDS is in-process — no separate daemon — so the heads-up below doesn’t apply if you ran nros setup … --rmw cyclonedds. nros setup … --rmw <rmw> installs the daemon into the nros store (~/.nros/sdk/<tool>/<version>/bin/; the cache root is ~/.nros/sdk/ — toolchains, transports, and daemons all land under there); you must then run it in a dedicated terminal before launching any example. For zenoh, put the store binary on PATH once and run it:

export PATH="$(dirname "$(ls -d ~/.nros/sdk/zenohd/*/bin/zenohd | tail -1)")":$PATH
zenohd        # leave running for the whole session

Without it the talker blocks forever on Executor::open with no output. Default ports: tcp/127.0.0.1:7447 on POSIX, tcp/10.0.2.2:7451 on QEMU FreeRTOS (Slirp forwards to host). Mismatch = silent hang; see Troubleshooting — First 10 Minutes.

After provisioning, follow the per-platform starter page (FreeRTOS, Zephyr, NuttX, …) for the board’s build + run steps.

Rust-only consumers

nano-ros is source-only — nothing is published to crates.io (decision 2026-05-14). The full nros crate can’t be published because it depends transitively on C/C++ submodules (zenoh-pico, mbedtls); nros-core isn’t carved out for crates.io either, to avoid a hybrid distribution model with version drift between the crates.io snapshot and in-repo HEAD.

Rust packages consume nano-ros via path dependency on the in-workspace checkout:

[dependencies]
nros = { path = "src/nano-ros/packages/core/nros",
         default-features = false,
         features = ["std", "rmw-cffi", "platform-posix", "ros-humble"] }

Each Rust package carries its own .cargo/config.toml patch entries when needed — see examples/templates/multi-package-workspace/src/pkg_rust_publisher/ for the pattern.

Contributor setup (working on nano-ros itself)

Contributors clone the repo and drive everything through just. The just setup recipes are thin wrappers over the same nros setup index — just <module> setup calls nros setup <board> under the hood, so the toolchains a contributor gets are identical to a user’s.

git clone https://github.com/NEWSLabNTU/nano-ros.git
cd nano-ros
./scripts/bootstrap.sh base   # path A from §1 above; gets rustup + just + nros
source ./activate.sh          # OR: direnv allow / source ./activate.fish
just setup all                # provision every supported board's SDK/toolchain

Diagnose missing tools (read-only):

just doctor tier=all

Provision one module:

just freertos setup           # → nros setup qemu-arm-freertos
just nuttx setup              # → nros setup qemu-arm-nuttx
just threadx_linux setup      # → nros setup threadx-linux

Docker environment

For a containerized environment with QEMU 7.2+:

just docker build
just docker shell      # Interactive shell with all tools
just docker test-qemu  # Run QEMU tests in container

Migrating from a pre-140 checkout

If you were on a nano-ros version that still had just install-local, see migration-install-local-removal.md for the one-page rewrite.

Next Steps

First Node — Rust (Linux)

Build, run, and verify a single nano-ros publisher node on Linux in about ten minutes. Uses the canonical Zenoh backend; no router needs to be pre-installed (the repo ships zenohd).

Stuck? See Troubleshooting — First 10 Minutes for the common first-build errors.

Prereqs

Pick one path from a fresh checkout — just is NOT a prereq.

A. Bare machine (no Rust, no just, no cargo):

./scripts/bootstrap.sh base

Installs rustup, just, builds the in-tree nros CLI at packages/cli/target/release/nros, leaves the binary on PATH for this shell.

B. Already have cargo (most contributors):

cargo build --release --manifest-path packages/cli/Cargo.toml --bin nros
export PATH="$PWD/packages/cli/target/release:$PATH"

C. Tagged release, no Rust at all:

./scripts/install-nros-prebuilt.sh

Downloads the matching nros-<triple>.tar.gz from the GitHub release, sha256-verifies, installs to packages/cli/target/release/nros.

Every subsequent shell sources the workspace env via one of:

direnv allow                  # if you use direnv
source ./activate.sh          # bash / zsh
source ./activate.fish        # fish

Then provision the native host (installs the zenoh router zenohd into a shared store — no ROS 2 needed):

nros setup native --rmw zenoh

See Install + first build (Linux) for more.

Project layout

The talker is a standalone Cargo package that pulls nano-ros in via a path dependency. Three files matter:

examples/native/rust/talker/
├── Cargo.toml          # path dep on `nros` + `nros-rmw-zenoh`
├── package.xml         # ROS-style manifest (drives codegen tooling)
├── generated/          # auto-generated message bindings (gitignored)
└── src/
    └── main.rs         # 60-line talker

POSIX talkers read the locator / domain from environment variables (NROS_LOCATOR — legacy alias ZENOH_LOCATOR — and ROS_DOMAIN_ID) — no nros.toml is needed. The nros.toml shape used by embedded targets shows up under the Embedded Starters section.

The Cargo.toml is the contract that wires nano-ros into your package. The in-tree talker is a member of the nano-ros workspace, so it does NOT carry a [workspace] table — cargo walks up and picks up the root Cargo.toml. Verbatim from examples/native/rust/talker/Cargo.toml (trimmed to the docs-relevant fields — the in-tree file also exposes rmw-cyclonedds / rmw-xrce features for the multi-RMW build path):

[package]
name    = "native-rs-talker"
version = "0.1.0"
edition = "2024"
license = "MIT OR Apache-2.0"
publish = false

[[bin]]
name = "talker"
path = "src/main.rs"

[features]
default   = ["rmw-zenoh"]
rmw-zenoh = ["dep:nros-rmw-zenoh"]

[dependencies]
nros = { path = "../../../../packages/core/nros",
         default-features = false,
         features = ["std", "rmw-cffi", "platform-posix"] }
nros-platform-cffi = { path = "../../../../packages/core/nros-platform-cffi",
                       features = ["posix-c-port"] }
nros-rmw-zenoh = { path = "../../../../packages/zpico/nros-rmw-zenoh",
                   features = ["std", "platform-posix", "ros-humble"],
                   optional = true }
std_msgs = { version = "*", default-features = false }
log = "0.4"
env_logger = "0.11"

Copying this out of the workspace? Once you move the directory elsewhere on disk, the path deps no longer resolve and there is no parent workspace to inherit from. Two options:

  1. Replace each path = "../../../../packages/..." with an absolute path to your nano-ros checkout, AND add an empty [workspace] table to stop cargo walking further up the filesystem.
  2. Keep it inside examples/ in your own fork of nano-ros — the simpler path while you’re learning the API.

Configure

Three runtime knobs, each overridable at three layers (defaults → config.toml → env vars):

KnobDefaultEnv override
Zenoh locatortcp/127.0.0.1:7447NROS_LOCATOR (legacy alias: ZENOH_LOCATOR)
ROS domain ID0ROS_DOMAIN_ID
Zenoh modeclientNROS_SESSION_MODE (legacy alias: ZENOH_MODE)

config.toml (optional, alongside Cargo.toml):

[zenoh]
locator   = "tcp/127.0.0.1:7447"
domain_id = 0

Build

cd examples/native/rust/talker
cargo build           # or: cargo build --release

First build pulls dependencies (~3 minutes). Re-builds finish in seconds.

Run

Three terminals (each command below blocks; keep them open):

# Terminal 1 — zenoh router. Blocks the shell until Ctrl-C.
zenohd                               # installed by `nros setup native`

# Terminal 2 — the talker. The talker logs via `log::info!`, so set
# RUST_LOG=info — without it `env_logger` only shows errors and the
# `Published: N` lines stay hidden.
cd examples/native/rust/talker
RUST_LOG=info cargo run
# Expected output (on stderr):
#   [INFO  native_rs_talker] nros Native Talker (Zenoh Transport)
#   [INFO  native_rs_talker] =========================================
#   [INFO  native_rs_talker] Published: 0
#   [INFO  native_rs_talker] Published: 1
#   …

That’s the nano-ros side fully working. Optional step: verify interop with stock ROS 2.

# Terminal 3 — stock ROS 2 with rmw_zenoh_cpp. NOTE: rmw_zenoh_cpp
# uses its OWN router daemon (`ros2 run rmw_zenoh_cpp rmw_zenohd`),
# NOT the in-tree zenohd from terminal 1. They need to peer with
# each other, or both clients need to point at the same router.
# Simplest: stop terminal 1 and run only `rmw_zenohd` instead, then
# launch the talker pointing at rmw_zenohd's port (default 7447).
source /opt/ros/humble/setup.bash
export RMW_IMPLEMENTATION=rmw_zenoh_cpp
ros2 run rmw_zenoh_cpp rmw_zenohd &       # in its own subshell
# Talker publishes best-effort; stock `ros2 topic echo` defaults to
# RELIABLE, so the QoS-mismatched echo silently delivers nothing.
# Force best-effort to receive:
ros2 topic echo /chatter std_msgs/msg/Int32 --qos-reliability best_effort

If ros2 topic echo shows no output despite the talker printing Published:, the routers aren’t peering — confirm both processes point at the same port (default tcp/127.0.0.1:7447).

Readiness signal. Within 5 seconds of RUST_LOG=info cargo run, the talker should print Published: 0 (the Rust talker pre-publishes 0 before the counter advances). If no Published: line in 30 seconds:

  1. Confirm RUST_LOG is set. Without RUST_LOG=info (or debug), env_logger filters out the Published: lines and the run looks silent even when it’s working.
  2. Confirm zenohd is running (terminal 1). Without it, the talker blocks on Executor::open indefinitely.
  3. Re-run with RUST_LOG=debug cargo run and look for “Failed to open session” — usually a wrong locator or wrong port.
  4. See Troubleshooting — First 10 Minutes.

GitHub source

Canonical, copy-out: examples/native/rust/talker/

Copy the directory, rename the package, and your starter is ready to modify.

Next

  • Add a subscription: examples/native/rust/listener/
  • Generate bindings for custom .msg / .srv / .action files: Message Generation
  • Cross-compile for an RTOS: pick the right Embedded Starter from the next section (FreeRTOS / Zephyr / NuttX / ThreadX / ESP32 / Bare-metal Cortex-M3).

First Node — C (Linux)

Build, run, and verify a single nano-ros publisher node on Linux from C. Uses CMake, the Zenoh backend, and add_subdirectory consumption.

Stuck? See Troubleshooting — First 10 Minutes for the common first-build errors.

Prereqs

Pick one path from a fresh checkout — just is NOT a prereq.

A. Bare machine (no Rust, no just, no cargo):

./scripts/bootstrap.sh base

Installs rustup, just, builds the in-tree nros CLI at packages/cli/target/release/nros, leaves the binary on PATH for this shell.

B. Already have cargo (most contributors):

cargo build --release --manifest-path packages/cli/Cargo.toml --bin nros
export PATH="$PWD/packages/cli/target/release:$PATH"

C. Tagged release, no Rust at all:

./scripts/install-nros-prebuilt.sh

Downloads the matching nros-<triple>.tar.gz from the GitHub release, sha256-verifies, installs to packages/cli/target/release/nros.

Every subsequent shell sources the workspace env via one of:

direnv allow                  # if you use direnv
source ./activate.sh          # bash / zsh
source ./activate.fish        # fish

Then provision the native host (installs the zenoh router zenohd into a shared store — no ROS 2 needed):

nros setup native --rmw zenoh

See Install + first build (Linux) for more.

Project layout

The talker is a standalone CMake project that pulls nano-ros via add_subdirectory(<repo-root>). Four files matter:

examples/native/c/talker/
├── CMakeLists.txt      # add_subdirectory + targets
├── package.xml         # ROS-style manifest (drives codegen tooling)
└── src/
    └── main.c          # ~100-line talker

An optional nros.toml sidecar can override the runtime locator / domain id (canonical schema described in Configuration). The shipped native C talker doesn’t carry one — locator + domain default to env vars.

The CMake preamble matches the canonical example shape in examples/native/c/talker/CMakeLists.txt — five set(...) lines plus the per-target link:

cmake_minimum_required(VERSION 3.22)
project(my_talker LANGUAGES C)

set(CMAKE_C_STANDARD 11)
set(CMAKE_C_STANDARD_REQUIRED ON)

# Pull nano-ros in. Adjust the relative path to your repo root.
# `NROS_RMW` is the user-facing cache var (overridable via
# `-DNROS_RMW=<rmw>`); the example forwards it to `NANO_ROS_RMW`,
# the var the nano-ros add_subdirectory reads.
set(NANO_ROS_PLATFORM posix)
set(NROS_RMW "zenoh" CACHE STRING
    "Active RMW (zenoh|xrce|cyclonedds) — selects the backend linked into my_talker.")
set(NANO_ROS_RMW "${NROS_RMW}")
add_subdirectory(<rel-path-to-nano-ros> nano_ros)

# Generate C bindings for std_msgs (transitively depends on
# builtin_interfaces; the dependency is declared explicitly).
nros_generate_interfaces(builtin_interfaces SKIP_INSTALL)
nros_generate_interfaces(std_msgs DEPENDENCIES builtin_interfaces
                                  SKIP_INSTALL)

add_executable(my_talker src/main.c)
target_link_libraries(my_talker PRIVATE
    std_msgs__nano_ros_c
    NanoRos::NanoRos)
nros_platform_link_app(my_talker)

nros_platform_link_app(my_talker) transitively wires the active RMW’s strong nros_app_register_backends() stub (calling nros_rmw_zenoh_register() for the zenoh build above). On POSIX you do not call nano_ros_link_rmw() explicitly — the platform module handles it.

The C entry point is int nros_app_main(int argc, char **argv) (not main); <nros/app_main.h> provides the OS-side main stub that wires signal handling and forwards to your function.

#include <nros/app_main.h>
#include <nros/init.h>
#include <nros/executor.h>
#include <nros/node.h>
#include <nros/publisher.h>
#include <nros/timer.h>
#include "std_msgs.h"

int nros_app_main(int argc, char** argv) {
    // 1. nros_init() — opens the zenoh session
    // 2. nros_executor_init() / nros_node_init()
    // 3. std_msgs_msg_int32_publisher_init() — typed publisher
    // 4. nros_timer_init() with a 1 Hz period + publish callback
    // 5. nros_executor_spin() until SIGINT
}

Configure

Three runtime knobs:

KnobDefaultEnv override
Zenoh locatortcp/127.0.0.1:7447NROS_LOCATOR
ROS domain ID0ROS_DOMAIN_ID
Node nametalkerhard-coded in source

nros.toml (optional) accepts the same [node] + [[transport]] schema as every other nano-ros tutorial (see Configuration); the C runtime reads it only when wired explicitly via nros_config_load() (see the example source).

Build

cd examples/native/c/talker
cmake -B build
cmake --build build

The first configure pulls and builds nano-ros’s Rust staticlibs (~3 minutes). Re-builds finish in seconds.

Run

Three terminals.

# 1. Start the zenoh router:
zenohd                               # installed by `nros setup native`

# 2. Run the talker:
cd examples/native/c/talker
./build/c_talker
# Expected output:
#   nros C Talker
#   =================
#   Published: 0
#   Published: 1
#   Published: 2
#   …

# 3. Verify from stock ROS 2:
source /opt/ros/humble/setup.bash
export RMW_IMPLEMENTATION=rmw_zenoh_cpp
# Talker publishes best-effort; stock `ros2 topic echo` defaults to
# RELIABLE, so the QoS-mismatched echo silently delivers nothing.
# Force best-effort to receive:
ros2 topic echo /chatter std_msgs/msg/Int32 --qos-reliability best_effort

Readiness signal. Within 5 seconds of ./build/c_talker, the binary should print Published: 0 on stdout — Rust + C + C++ all start the counter at 0 (Phase 208.D.9). If no Published: line in 30 seconds:

  1. Confirm zenohd is running (terminal 1). Without it, nros_support_init returns immediately with -4 (NROS_RET_NOT_FOUND — connection refused).
  2. Wrong locator / unreachable host → same -4 signature in stderr. Reachable host but mismatched port → talker hangs on session-open handshake rather than returning a code.
  3. See Troubleshooting — First 10 Minutes.

GitHub source

Canonical, copy-out: examples/native/c/talker/

Next

First Node — C++ (Linux)

Build, run, and verify a single nano-ros publisher node on Linux from C++14. Uses CMake, the Zenoh backend, and add_subdirectory consumption.

Stuck? See Troubleshooting — First 10 Minutes for the common first-build errors.

Prereqs

Pick one path from a fresh checkout — just is NOT a prereq.

A. Bare machine (no Rust, no just, no cargo):

./scripts/bootstrap.sh base

Installs rustup, just, builds the in-tree nros CLI at packages/cli/target/release/nros, leaves the binary on PATH for this shell.

B. Already have cargo (most contributors):

cargo build --release --manifest-path packages/cli/Cargo.toml --bin nros
export PATH="$PWD/packages/cli/target/release:$PATH"

C. Tagged release, no Rust at all:

./scripts/install-nros-prebuilt.sh

Downloads the matching nros-<triple>.tar.gz from the GitHub release, sha256-verifies, installs to packages/cli/target/release/nros.

Every subsequent shell sources the workspace env via one of:

direnv allow                  # if you use direnv
source ./activate.sh          # bash / zsh
source ./activate.fish        # fish

Then provision the native host (installs the zenoh router zenohd into a shared store — no ROS 2 needed):

nros setup native --rmw zenoh

See Install + first build (Linux) for more.

Project layout

The talker is a standalone CMake project that pulls nano-ros via add_subdirectory(<repo-root>). Two files matter:

examples/native/cpp/talker/
├── CMakeLists.txt      # add_subdirectory + targets
└── src/
    └── main.cpp        # ~70-line talker

POSIX talkers read the locator + domain from arguments passed to nros::init(...); no config.toml is needed. Embedded variants under examples/<plat>/cpp/talker/ carry a config.toml that their board crate reads.

CMake preamble matches the canonical example at examples/native/cpp/talker/CMakeLists.txt — five set(...) lines + per-target link. LANGUAGES C CXX (not CXX alone): the per-target register stub the nano-ros add_subdirectory emits is a C translation unit, so C must be enabled in this directory scope or the link fails.

cmake_minimum_required(VERSION 3.22)
project(my_talker LANGUAGES C CXX)

set(CMAKE_CXX_STANDARD 14)
set(CMAKE_CXX_STANDARD_REQUIRED ON)

# `NROS_RMW` is the user-facing cache var (overridable via
# `-DNROS_RMW=<rmw>`); the example forwards it to `NANO_ROS_RMW`,
# the var the nano-ros add_subdirectory reads.
set(NANO_ROS_PLATFORM posix)
set(NROS_RMW "zenoh" CACHE STRING
    "Active RMW (zenoh|xrce|cyclonedds) — selects the backend linked into my_talker.")
set(NANO_ROS_RMW "${NROS_RMW}")
add_subdirectory(<rel-path-to-nano-ros> nano_ros)

# Generate C++ bindings (LANGUAGE CPP — separate from the C variant).
nros_generate_interfaces(builtin_interfaces LANGUAGE CPP SKIP_INSTALL)
nros_generate_interfaces(std_msgs DEPENDENCIES builtin_interfaces
                                  LANGUAGE CPP SKIP_INSTALL)

add_executable(my_talker src/main.cpp)
target_link_libraries(my_talker PRIVATE
    std_msgs__nano_ros_cpp
    NanoRos::NanoRosCpp)
nros_platform_link_app(my_talker)

nros_platform_link_app(my_talker) transitively registers the selected RMW backend — on POSIX you do not call nano_ros_link_rmw() explicitly.

The C++ entry point is int nros_app_main(int argc, char** argv) (same as C); <nros/app_main.h> provides the OS-side main stub.

The body uses typed nros::Publisher<M> / nros::Subscription<M> wrappers over the C ABI:

#include <nros/app_main.h>
#include <nros/nros.hpp>
#include "std_msgs.hpp"

#define NROS_TRY_LOG(file, line, expr, ret) \
    std::fprintf(stderr, "[nros] %s:%d %s -> %d\n", file, line, expr, (int)ret)

int nros_app_main(int argc, char** argv) {
    NROS_TRY_RET(nros::init("tcp/127.0.0.1:7447", 0), 1);

    nros::Node node;
    NROS_TRY_RET(nros::create_node(node, "talker"), 1);

    nros::Publisher<std_msgs::msg::Int32> pub;
    NROS_TRY_RET(node.create_publisher(pub, "/chatter"), 1);

    // ... register a timer + spin
}

NROS_TRY_RET short-circuits on any non-OK return code and logs the expression that failed. Define NROS_TRY_LOG once (any sink — here std::fprintf) and reuse it across every call site.

Configure

Three runtime knobs:

KnobDefaultOverride
Zenoh locatortcp/127.0.0.1:7447First arg to nros::init
ROS domain ID0Second arg to nros::init
Node nametalkerFirst arg to nros::create_node

Reading from env in C++ is std::getenv("NROS_LOCATOR") plus the same nros::init call — see the GitHub source for the full pattern.

Build

cd examples/native/cpp/talker
cmake -B build
cmake --build build

First configure builds nano-ros’s Rust staticlibs (~3 minutes). Re-builds finish in seconds.

Run

Three terminals.

# 1. zenoh router (installed by `nros setup native`):
zenohd

# 2. Run the talker:
cd examples/native/cpp/talker
./build/cpp_talker
# Expected:
#   Published: 0
#   Published: 1
#   …

# 3. Verify from stock ROS 2:
source /opt/ros/humble/setup.bash
export RMW_IMPLEMENTATION=rmw_zenoh_cpp
# Talker publishes best-effort; stock `ros2 topic echo` defaults to
# RELIABLE, so the QoS-mismatched echo silently delivers nothing.
# Force best-effort to receive:
ros2 topic echo /chatter std_msgs/msg/Int32 --qos-reliability best_effort

Readiness signal. Within 5 seconds of ./build/cpp_talker, the binary should print Published: 0 — Rust + C + C++ all start the counter at 0 (Phase 208.D.9). If no Published: line in 30 seconds:

  1. Confirm zenohd is running (terminal 1). Without it, nros::init returns -100 (TransportError) — the NROS_TRY_RET macro logs the failed call to stderr.
  2. Check stderr for [nros] …/main.cpp:LINE nros::init(...) -> -N diagnostics. -3 / -100 both indicate transport open failed.
  3. See Troubleshooting — First 10 Minutes.

GitHub source

Canonical, copy-out: examples/native/cpp/talker/

Next

Porting a ROS 2 C++ node to nano-ros

Goal: take a normal ROS 2 C++ node (one that compiles + runs under colcon build against ros-humble-*) and run it under nano-ros — without rewriting the source. The Phase 209 compat layer (nros/rclcpp_compat.hpp + cmake/compat/NrosRclcppCompat.cmake + nros-diagnostic-updater) is built for this; the only delta is build-script glue.

The canonical proof lives at examples/templates/cpp-port-minimal-publisher/ — the ROS 2 tutorial’s minimal_publisher.cpp vendored unmodified, building against nano-ros via the three glue lines below.

Two layers of glue

Per-package CMakeLists.txt — zero nano-ros lines (Phase 210)

The ported pkg’s CMakeLists.txt carries only stock ROS 2 syntax. Same file builds under both colcon build AND a nano-ros build:

cmake_minimum_required(VERSION 3.20)
project(my_node LANGUAGES CXX)

find_package(ament_cmake REQUIRED)
find_package(rclcpp REQUIRED)
find_package(std_msgs REQUIRED)
# … any other msg packages the source #include's.

add_executable(my_node src/my_node.cpp)
ament_target_dependencies(my_node rclcpp std_msgs)

ament_package()

find_package(rclcpp) resolves through the rclcpp Find-stub (which auto-applies the rclcpp_compat.hpp force-include); find_package(std_msgs) resolves through the smart Find-stub (Phase 210.A.2 → walks NROS_INTERFACE_SEARCH_PATH > AMENT_PREFIX_PATH > bundled); the ament_target_dependencies compat shim wires both link targets.

Workspace umbrella CMakeLists.txt — one nano-ros include (Phase 210)

The umbrella CMakeLists.txt (sits at the workspace root, next to src/) is the only nano-ros-aware file:

cmake_minimum_required(VERSION 3.22)
project(my_workspace LANGUAGES CXX)
set(CMAKE_CXX_STANDARD 14)

# 1) Pull nano-ros in.
set(NANO_ROS_PLATFORM posix)
set(NROS_RMW "zenoh" CACHE STRING "Active RMW.")
set(NANO_ROS_RMW "${NROS_RMW}")
add_subdirectory("/path/to/nano-ros" nano_ros)

# 2) Point the smart Find-stub at this workspace's src/ (must precede the
#    NrosRclcppCompat include so workspace-pkg Find<pkg>.cmake auto-emit
#    picks it up).
set(NROS_INTERFACE_SEARCH_PATH "${CMAKE_SOURCE_DIR}/src")

# 3) Drop-in source-compat surface.
include("/path/to/nano-ros/cmake/compat/NrosRclcppCompat.cmake")

# 4) Bulk-build every workspace msg pkg in topo order (one line instead of
#    N add_subdirectory(src/<pkg>) lines).
nros_workspace_interfaces()

# 5) Build consumer apps.
add_subdirectory(src/my_node)

No nros_generate_interfaces(<pkg>) calls per consumer — the smart Find-stub does the codegen at find_package(<pkg>) time.

Reference fixture

examples/templates/local-msg-package/ ships the pattern end-to-end: two workspace msg pkgs (local_msgs, extra_msgs) with intra-workspace dep + a C++ consumer pulling msgs from BOTH the workspace AND AMENT (std_msgs, geometry_msgs, sensor_msgs) via one find_package shape. Cross-build proof: the same src/ builds under colcon build (CI-gated by Phase 210.F.2).

Legacy nros_generate_interfaces(<pkg>) shape

Per-package nros_generate_interfaces(std_msgs LANGUAGE CPP SKIP_INSTALL) calls still work (back-compat preserved) but are deprecated for new code — they bypass the ROS-convention smart Find-stub + workspace discovery. Existing in-tree examples will migrate as part of Phase 210.E.3.

What “just works” without source edits

The compat surface covers the patterns a typical ROS 2 C++ node uses:

rclcpp surfacenano-ros mappingNotes
class MyNode : public rclcpp::Noderclcpp::Node shim → nros::Executor + nros::NodeCtor takes (name).
std::make_shared<MyNode>()inherits enable_shared_from_thisshared_from_this() works.
create_publisher<M>(topic, qos)shared_ptr-returning wrapperqos can be rclcpp::QoS(10) or an int.
create_subscription<M>(topic, qos, callback)polling pump dispatched from spin*Capturing lambdas + std::function all work (Phase 209.A.follow-up).
create_wall_timer(period, callback)wall-timer dispatched from spin*std::chrono::duration arg, capturing-lambda callback.
rclcpp::init(argc, argv) / shutdown() / ok() / spin(n) / spin_some(n)wraps nros::init/shutdown/ok/spin_onceargc/argv ignored.
RCLCPP_INFO / WARN / ERROR / DEBUG / FATALdispatched through NROS_* macros_THROTTLE variants degrade to plain log.
rclcpp::QoS / KeepLast(n) / SystemDefaultsQoS()subclass of nros::QoS with the (depth) ctorChainable setters inherited.
diagnostic_updater::Updater + DiagnosticStatusWrappernros-diagnostic-updater shim (Phase 209.D)Publishes /diagnostics.
rclcpp_action::Server<A> / Client<A>aliases for nros::ActionServer/Client<A>The action call shapes (send_goal_async etc.) match.
RCLCPP_COMPONENTS_REGISTER_NODE(class)no-op macro + cmake-side rclcpp_components_register_node() emits a thin int main() per registrationSingle-binary embedded.
find_package(ament_cmake_auto / rclcpp / rclcpp_components / diagnostic_updater / std_msgs / …)Find-stubs at cmake/compat/stubs/~24 of the most-cited ROS 2 packages stubbed; add your own under cmake/compat/stubs/Find<pkg>.cmake for more.

What’s documented as “needs adapt” (codegen-side, not surface-side)

These are cosmetic codegen differences nano-ros’s per-package codegen and the upstream rosidl_default_runtime codegen don’t share. Both are tracked under Phase 210 (ROS-convention codegen).

  • Message string fields. nano-ros codegen emits nros::FixedString<N>, upstream emits std::string. Assigning a std::string needs a one-token adapter: message.data = s.c_str(). The reverse (std::string{}.c_str()) is what RCLCPP_INFO already takes.
  • Generated message header path. CLOSED (Phase 123.B.8 alias headers): nano-ros codegen emits BOTH the per-message form <std_msgs/msg/string.hpp> (upstream-shape) AND the umbrella <std_msgs/std_msgs.hpp>. Use whichever the original source picks.

What’s out of scope (will need code adapt or a follow-up phase)

  • rclcpp_lifecycle::LifecycleNode — Phase 209.H (deferred). Until that lands, replace LifecycleNode with Node + manual configure/activate bookkeeping.
  • Yaml-loaded parameters. declare_parameter<T>("name", default) reads from a launch yaml in stock ROS 2. nano-ros embedded has no yaml loader; Phase 209.F bakes the original yaml + the source into a constexpr parameter table (nros bake-params). Until that ships, expose parameters as compile-time constants (or via nano_ros_read_config(... "nros.toml")
    • nano_ros_generate_config_header(...)).
  • tf2, image_transport, pluginlib — out of nano-ros scope. Project- specific helpers (autoware universe_utils, PX4 uORB shims) are not nano-ros’s to ship; the porting user vendors them or replaces the call sites with raw nros-cpp ones.

When the port hits a gap

Open follow-ups: 209.F (yaml params bake), 209.H (LifecycleNode), 210.E.3 (in-tree migration of legacy nros_generate_interfaces(<pkg>) call sites). If your port surfaces a new gap not covered by the compat header, file it under Phase 209 (Track-A = tree-side fix that lands in cmake/compat/ or packages/core/nros-cpp/; Track-B = a codegen change).

In-tree regression fixtures:

Drop your reduced-case node under examples/templates/<your-port>/ and add it to CI once the gap closes.

Your own message package

You write a ROS 2 msg package once. The same src/my_msgs/ directory builds under both:

  • colcon build — upstream rosidl_default_generators produces the upstream-ROS bindings.
  • a nano-ros build — rosidl_generate_interfaces(...) is intercepted by nano-ros’s wrapper and routed through nano-ros codegen.

Different build systems, identical source tree. No nano-ros-specific files in the msg package.

The msg package — stock ROS shape

Drop a verbatim ROS msg pkg under src/ of your workspace:

src/my_msgs/
├── package.xml
├── CMakeLists.txt
└── msg/
    └── MyMsg.msg

src/my_msgs/package.xml:

<?xml version="1.0"?>
<package format="3">
  <name>my_msgs</name>
  <version>0.1.0</version>
  <description>My ROS 2 msg package</description>
  <maintainer email="you@example.org">you</maintainer>
  <license>Apache-2.0</license>

  <buildtool_depend>ament_cmake</buildtool_depend>
  <depend>std_msgs</depend>
  <build_depend>rosidl_default_generators</build_depend>
  <exec_depend>rosidl_default_runtime</exec_depend>

  <member_of_group>rosidl_interface_packages</member_of_group>

  <export>
    <build_type>ament_cmake</build_type>
  </export>
</package>

src/my_msgs/CMakeLists.txt:

cmake_minimum_required(VERSION 3.20)
project(my_msgs)

find_package(ament_cmake REQUIRED)
find_package(rosidl_default_generators REQUIRED)
find_package(std_msgs REQUIRED)

rosidl_generate_interfaces(${PROJECT_NAME}
    msg/MyMsg.msg
    DEPENDENCIES std_msgs
)

ament_export_dependencies(rosidl_default_runtime)
ament_package()

src/my_msgs/msg/MyMsg.msg:

std_msgs/Header header
string payload
int32 sequence

Zero nano-ros-specific lines. Run colcon build from this dir — upstream ROS produces a working msg package. Drop the same dir into a nano-ros build (below) — nano-ros codegen produces the equivalent bindings.

The consumer — stock ROS shape

// src/my_app/src/my_app.cpp
#include <chrono>
#include <memory>

#include <rclcpp/rclcpp.hpp>
#include "my_msgs/msg/my_msg.hpp"

using namespace std::chrono_literals;

class MyNode : public rclcpp::Node {
public:
    MyNode() : rclcpp::Node("my_node") {
        publisher_ = this->create_publisher<my_msgs::msg::MyMsg>("topic", 10);
        timer_ = this->create_wall_timer(500ms, [this]() {
            my_msgs::msg::MyMsg m;
            m.payload = "hello";
            publisher_->publish(m);
        });
    }
private:
    std::shared_ptr<rclcpp::TimerBase> timer_;
    std::shared_ptr<rclcpp::Publisher<my_msgs::msg::MyMsg>> publisher_;
};

int main(int argc, char* argv[]) {
    rclcpp::init(argc, argv);
    rclcpp::spin(std::make_shared<MyNode>());
    rclcpp::shutdown();
    return 0;
}

src/my_app/CMakeLists.txt:

cmake_minimum_required(VERSION 3.20)
project(my_app LANGUAGES CXX)

find_package(ament_cmake REQUIRED)
find_package(rclcpp REQUIRED)
find_package(my_msgs REQUIRED)
find_package(std_msgs REQUIRED)

add_executable(my_app src/my_app.cpp)
target_link_libraries(my_app
    rclcpp::rclcpp
    my_msgs::my_msgs
    std_msgs::std_msgs
)

ament_package()

The nano-ros umbrella — the only nano-ros-specific file

The two src/ packages are stock ROS. One umbrella CMakeLists.txt at the workspace root pulls nano-ros in:

cmake_minimum_required(VERSION 3.22)
project(my_workspace LANGUAGES CXX)
set(CMAKE_CXX_STANDARD 14)

# Pull nano-ros.
set(NANO_ROS_PLATFORM posix)
set(NROS_RMW "zenoh" CACHE STRING "Active RMW.")
set(NANO_ROS_RMW "${NROS_RMW}")
add_subdirectory(/path/to/nano-ros nano_ros)

# Point the smart Find-stub at this workspace (must precede the compat
# include so the workspace Find<pkg>.cmake auto-emit picks it up).
set(NROS_INTERFACE_SEARCH_PATH "${CMAKE_SOURCE_DIR}/src")

# Pull the rclcpp source-compat layer (find_package(rclcpp) etc.).
include(/path/to/nano-ros/cmake/compat/NrosRclcppCompat.cmake)

# Bulk-build every workspace msg pkg in topo order. One line, no
# add_subdirectory(src/<pkg>) per pkg.
nros_workspace_interfaces()

# Build the consumer app.
add_subdirectory(src/my_app)

Build:

cmake -B build -S .
cmake --build build -j
./build/src/my_app/my_app

Cross-build proof — same source under colcon

cd src && colcon build

src/my_msgs/ produces the upstream my_msgs bindings; src/my_app/ links against the upstream rclcpp + my_msgs::my_msgs. The nano-ros build above produces the same source linked against NanoRos::NanoRosCpp through the nano-ros codegen, with the smart Find-stub forwarding find_package(my_msgs)my_msgs::my_msgs.

The interface-package search path

find_package(<pkg>) walks three layers, highest priority first:

LayerSourceNotes
1NROS_INTERFACE_SEARCH_PATHColon/semicolon-separated colcon-src/-style roots; immediate subdirs with package.xml are candidates.
2AMENT_PREFIX_PATHThe standard ROS install-prefix layout (<prefix>/share/<pkg>/{msg,srv,action}/).
3Bundled<nano-ros>/packages/interfaces/<pkg>/ + <nano-ros>/share/nano-ros/interfaces/<pkg>/.

Shadowing — a workspace my_msgs and an AMENT my_msgs resolve to the workspace one, with a message(STATUS ...) line noting the shadow.

Shadowing contract

When two layers carry the same package name (e.g. a workspace std_msgs and /opt/ros/<distro>/share/std_msgs/), the higher layer wins — silently and deterministically. Concretely:

  1. NROS_INTERFACE_SEARCH_PATH > AMENT_PREFIX_PATH > bundled. The smart Find-stub (cmake/compat/stubs/_NrosFindRosMsgPackage.cmake) walks the three layers in order; the first hit wins. Lower layers are skipped entirely.

  2. The configure pass emits a message(STATUS ...) line noting the resolved path, e.g.

    -- nros: find_package(std_msgs) -> /path/to/workspace/src/std_msgs
    

    Grep your configure log for nros: find_package(<pkg>) to confirm which layer supplied each pkg.

  3. Intra-workspace shadowing — when two roots under NROS_INTERFACE_SEARCH_PATH both ship the same pkg, the nros_workspace_interfaces() bulk orchestrator keeps the earlier-listed one and warns about the shadowed copy.

  4. No partial overrides. Shadowing is whole-package: a workspace pkg replaces ALL of an AMENT pkg’s interfaces, not a subset. If you only want to add MyExtraMsg.msg to std_msgs, ship a separate pkg (e.g. my_std_extra_msgs) — don’t shadow.

Compile-time fail-safe

Shadowing is observed at compile time: if your workspace std_msgs declares a Marker.msg and your consumer includes "std_msgs/msg/marker.hpp", the build only succeeds when the workspace copy is the one linked. AMENT’s std_msgs ships no Marker.msg, so a fall-through resolves to a missing header. The build outcome is the strongest evidence — nm on the linked binary is the symbol-level corroborator.

Reference fixture for shadowing

The canonical smoke proof for the Layer 1 > Layer 2 case ships at examples/templates/workspace-shadowing/ — a workspace std_msgs carrying a Marker.msg shadows the AMENT-installed std_msgs. The fixture’s README.md walks through the cmake + nm verification. A regression test (packages/testing/nros-tests/tests/phase210_f4_shadowing.rs) re-runs the same proof in CI when AMENT is sourced.

Reference fixture

examples/templates/local-msg-package/ ships the exact pattern above end-to-end (two workspace msg pkgs with a dep between them, plus a consumer node). Use it as a copy-out template.

Troubleshooting — First 10 Minutes

The Linux starter walkthroughs assume nros setup native --rmw zenoh has run, zenohd is reachable, and the right Rust target is installed. When something goes wrong in the first ten minutes, the error you see usually points at one of the predictable misses below.

Each branch quotes the real stderr you can grep against — not a paraphrase. If your error text matches, the fix on the right is the one to try.

A. Build failures (cargo / cmake)

A1. Missing path-deps onto in-tree crates

error[E0432]: unresolved import `nros`
error: failed to load source for dependency `nros`
error: could not find `nros-rmw-zenoh`

The example’s Cargo.toml carries a path-dep onto the in-tree packages/core/nros* crates (the canonical copy-out shape). When the example is cargo-built from a stripped-down checkout (e.g. you vendored only the example dir into your own workspace), those path = "../../../../packages/…" entries resolve to nothing. Fix one of:

  • Build inside a full nano-ros checkout, OR
  • Rewrite the path = … entries to git = … against github.com/NEWSLabNTU/nano-ros and pin a rev.

This is not an nros setup issue — nros setup only fetches the SDK / source-package payload (zenoh-pico, mbedtls, cyclonedds, …); it does not synthesise missing Cargo dependencies.

A2. nros codegen tool not found

nros (codegen tool) not found on PATH or in packages/cli/target/release/
or ${NROS_HOME:-~/.nros}/bin. nano-ros assumes `nros` is provided
(Phase 218 carries the CLI in-tree at packages/cli/). Build it with:
  just setup-cli                 # or: just setup

Missing the nros binary on PATH and in the per-checkout location. Phase 218 builds it from the in-tree sub-workspace; first activate the workspace, then build:

source ./activate.sh        # OR: direnv allow / source ./activate.fish
just setup-cli              # builds packages/cli/target/release/nros

If packages/cli/target/release/nros exists but PATH doesn’t see it — that’s the [PATH] doctor status (D2 below), not this branch (the activate file is what puts it on PATH).

The nros binary ships the codegen — there is no separate nros-codegen build step. CMake examples auto-resolve nros from PATH / packages/cli/target/release/ / ${NROS_HOME:-~/.nros}/bin/; -D_NANO_ROS_CODEGEN_TOOL=<path> is an override, not a requirement.

A3. Rust target not installed

error: could not compile … due to previous error
the target `thumbv7m-none-eabi` is not installed

Add it:

rustup target add thumbv7m-none-eabi
# or whichever target the example's `.cargo/config.toml` names

A4. Cross linker not found

error: linker `arm-none-eabi-gcc` not found

The cross toolchain wasn’t provisioned. Run nros setup for your board (it ships a prebuilt arm-none-eabi-gcc):

nros setup qemu-arm-freertos       # or qemu-arm-nuttx / mps2-an385 / …

A5. Cyclone DDS runtime missing

ld: cannot find -lddsc
ld: cannot find -lcyclonedds-ddsc

The Cyclone DDS runtime wasn’t provisioned:

nros setup native --rmw cyclonedds
undefined reference to `nros_rmw_cyclonedds_register`
undefined reference to `dds_create_participant`

rmw-cyclonedds cannot link from cargo alone — the Cyclone backend is C++ + CMake, registered via nros_rmw_cffi_register from a CMake-built target. Phase 175 wired this through CMakeLists.txt + Corrosion. Use the cmake build path instead:

cd examples/native/c/talker        # (or cpp / rust)
cmake -B build-cyclone -DNROS_RMW=cyclonedds
cmake --build build-cyclone

The pure cargo build --features rmw-cyclonedds only succeeds for the zenoh-pico + xrce backends today.

A7. “current package believes it’s in a workspace”

error: current package believes it's in a workspace when it's not:
current:   …/Cargo.toml
workspace: /…/nano-ros/Cargo.toml

cargo walks up the directory tree looking for a workspace root and adopts the example into the outer nano-ros workspace. Per-example Cargo.tomls don’t ship an empty [workspace] table yet (tracked as F1 in phase-208-followups.md).

Hits on:

  • nested clones / worktrees of nano-ros that share an ancestor path with the outer nano-ros/Cargo.toml;
  • a user vendoring an example into their own workspace.

Workaround on a regular clone: build from the nano-ros root, e.g. cargo build -p qemu-bsp-talker, instead of cd’ing into the example dir.

A8. direnv allow reminder

NROS_PLATFORM_CFFI_INCLUDE not set (direnv allow, or build via just)
FREERTOS_PORT not set

Phase 208.D.1 made the common build sites autoresolve these from the in-tree checkout, so a fresh cargo build no longer panics on them in canonical examples. If your custom build site still does, run direnv allow once after clone, or set the env explicitly / build via the just <plat> recipe.

B. Binary runs but no output

B1. Rust: Failed to open session: Transport(ConnectionFailed)

thread 'main' panicked at examples/native/rust/talker/src/lib.rs:96:58:
Failed to open session: Transport(ConnectionFailed)

zenohd isn’t running, or isn’t reachable on the locator the talker is pointed at. Start it in another terminal (nros setup native --rmw zenoh lands zenohd under ${NROS_HOME:-~/.nros}/sdk/zenohd/; the activate file puts it on PATH):

zenohd --listen tcp/127.0.0.1:7447

Default ports: tcp/127.0.0.1:7447 on POSIX, tcp/10.0.2.2:7451 on QEMU FreeRTOS (Slirp forwards to host), 7452 NuttX, 7453 ThreadX-RV, 7454 ESP32, 7455 ThreadX-Linux, 7456 Zephyr.

B2. C: NROS_CHECK failed: nros_support_init(...) -> -4

NROS_CHECK failed at src/main.c:152: nros_support_init(&app.support, locator, domain_id) -> -4

Process exits 1 (the retval passed to NROS_CHECK_RET). -4 = NROS_RET_NOT_FOUND — the locator was unreachable (zenohd not running, or wrong port). Same fix as B1 above.

The C API entry point is nros_support_init, not nros_init or nros::init — those don’t exist in the C API.

B3. C++: process exits 156 after a nros::init failure

nros::init returned NROS_CPP_RET_TRANSPORT_ERROR (-100)

Same root cause as B1/B2 — zenohd not reachable. NROS_CPP_RET_TRANSPORT_ERROR = -100 is the C++ result code that NROS_TRY_RET propagates from main(); on POSIX this becomes (unsigned char)-100 = 156 as the process exit code. Treat “exited 156 after starting” as the C++ equivalent of B1.

B4. Override the locator at runtime

When the talker can’t reach the daemon and you don’t want to edit nros.toml, override the locator with the canonical env var:

NROS_LOCATOR=tcp/192.168.1.50:7447 ./build/c_talker
# Legacy alias (still accepted): ZENOH_LOCATOR=… ./build/c_talker
ROS_DOMAIN_ID=7 ./build/c_talker         # also overridable

The Rust / C / C++ talkers all read NROS_LOCATOR first, fall back to ZENOH_LOCATOR, then to nros.toml, then to the build-time default.

B5. Binary exits immediately, no error printed

Buffering: setvbuf(stdout, NULL, _IOLBF, 0) if you piped the run. POSIX terminals flush on newline; piped stdout full-buffers and may eat short outputs. Add a RUST_LOG=info (Rust) or unbuffer the C / C++ output (stdbuf -oL).

C. ROS 2 side sees nothing

C1. RMW mismatch

# On the ROS 2 side, default rmw_fastrtps_cpp will NOT see nano-ros:
export RMW_IMPLEMENTATION=rmw_zenoh_cpp     # for Zenoh
export RMW_IMPLEMENTATION=rmw_cyclonedds_cpp # for Cyclone

C2. QoS mismatch — echo silent, list sees the topic

ros2 topic list           # /chatter shown
ros2 topic echo /chatter  # … nothing

nano-ros publishers default to BEST_EFFORT; stock ros2 topic echo defaults to RELIABLE. The QoS-mismatched subscriber is created but receives no data. Force best-effort on the echo:

ros2 topic echo /chatter std_msgs/msg/Int32 --qos-reliability best_effort

D. Doctor + last-resort

D1. Use the per-platform doctor first

just freertos doctor       # FreeRTOS / QEMU / arm-none-eabi
just nuttx doctor          # NuttX
just zephyr doctor         # Zephyr
just threadx_linux doctor  # ThreadX-Linux
# etc.

Each scoped doctor is fast and prints the same fixit hints for the toolchain you actually need.

D2. [PATH] nros built but not on PATH

The doctor now reports this distinct from [MISSING] when the binary is built at packages/cli/target/release/nros (or the transitional ${NROS_HOME:-~/.nros}/bin/nros) but PATH doesn’t see it. Activate the workspace — it wires PATH:

source ./activate.sh        # bash / zsh
# OR
source ./activate.fish      # fish
# OR
direnv allow                # auto-activates on `cd nano-ros`

Don’t loop on just workspace cargo-tools — that re-runs the build which short-circuits on the same PATH miss.

D3. Full sweep (slow)

just doctor tier=default

Only run this when you’re standing up every supported platform in one go. It walks every per-platform doctor and can take a few minutes.

D4. File an issue

When all else fails, include:

  • the exact command you ran,
  • the full stderr,
  • rustc --version, cmake --version, qemu-system-arm --version,
  • nros --version.

What success looks like

A correctly-running Rust Linux talker (examples/native/rust/talker) prints something like this on stderr (with RUST_LOG=info):

[INFO  native_rs_talker] nros Native Talker (Zenoh Transport)
[INFO  native_rs_talker] =========================================
[INFO  native_rs_talker] Node created: talker
[INFO  native_rs_talker] Publisher created for topic: /chatter
[INFO  native_rs_talker] Published: 0
[INFO  native_rs_talker] Published: 1
[INFO  native_rs_talker] Published: 2

A correctly-running C talker (examples/native/c/talker) prints on stdout:

nros C Talker
=================
Published: 0
Published: 1
Published: 2

A correctly-running C++ talker prints the same Published: N line once per second.

The ROS 2 side (ros2 topic echo /chatter std_msgs/msg/Int32 --qos-reliability best_effort with RMW_IMPLEMENTATION=rmw_zenoh_cpp) should see:

data: 0
---
data: 1
---
data: 2
---

If you see all three of these — talker logging, ROS 2 echo output, and matching counter values — interop is verified end-to-end.

See also

Multi-Node Project Layout

You built the single-file talker in examples/native/rust/talker/: one main.rs, one Cargo.toml, one package — open a terminal, run cargo run, done. That shape covers a lot of ground: demos, driver tests, single-purpose microcontroller apps.

At some point it stops being enough.

When you outgrow one app

Three common triggers:

  1. Two or more nodes. You want a talker and a listener, or a sensor driver alongside a control loop. Splitting them into one entry binary per node file works until you need to tune their wiring or deploy them to different boards.

  2. A shared launch topology. You want to describe once how nodes are named, remapped, and parameterised — and reuse that description across a dev laptop, a hardware bring-up board, and a sim target.

  3. Multiple deploy targets. The same talker logic goes on a native Linux host for integration testing and on an STM32F4 for production. The node logic is identical; only the boot and board differ.

  4. Mixed implementation languages. You want to keep a C driver or legacy C node, but host the composed system in a C++ or Rust Entry pkg. The Node-pkg register ABI is language-neutral, so a C Node pkg can link into the same Entry binary as C++ or Rust Node pkgs.

That’s when you split your project into a multi-node workspace.

Canonical layout

Start with the whole project before diving into the parts:

my_robot_ws/
├── Cargo.toml                  # Rust workspace root, or CMakeLists.txt for C/C++
└── src/
    ├── talker_pkg/             # Node pkg: reusable node logic, no main()
    ├── listener_pkg/           # Node pkg: another reusable node
    ├── robot_bringup/          # Bringup pkg: launch XML + system.toml
    └── native_entry/           # Entry pkg: one runnable binary for one board

The roles are deliberately separated:

RoleOwnsDoes not own
Node pkgPublishers, subscriptions, timers, services, actions, callback bodiesBoard choice, launch topology, main()
Bringup pkgWhich nodes run, names, remaps, parameters, per-target topologyCompiled code
Entry pkgBoard/runtime selection and the runnable binaryNode behavior

A typical product has many Node pkgs, one Bringup pkg per logical system, and one Entry pkg per board or deploy target. The same talker_pkg and listener_pkg can be linked into a native host Entry pkg for integration testing and a Cortex-M Entry pkg for hardware.

Reading order

This group starts broad and then drills into each part:

  1. Project layout — this page: when to split and how the roles fit.
  2. Node packages — reusable node libraries with nros::node!.
  3. Bringup packages — launch XML, system.toml, remaps, parameters.
  4. Entry packages — the board-specific binary that boots the topology.
  5. C / C++ multi-node workspaces — the same structure through CMake.
  6. Mixed-language workspaces — C Node pkgs hosted by C/C++ Entry pkgs.
  7. Role reference — metadata fields and macro forms in reference style.

Prereqs

Pick one path from a fresh checkout — just is NOT a prereq.

A. Bare machine (no Rust, no just, no cargo):

./scripts/bootstrap.sh base

Installs rustup, just, builds the in-tree nros CLI at packages/cli/target/release/nros, leaves the binary on PATH for this shell.

B. Already have cargo (most contributors):

cargo build --release --manifest-path packages/cli/Cargo.toml --bin nros
export PATH="$PWD/packages/cli/target/release:$PATH"

C. Tagged release, no Rust at all:

./scripts/install-nros-prebuilt.sh

Downloads the matching nros-<triple>.tar.gz from the GitHub release, sha256-verifies, installs to packages/cli/target/release/nros.

Every subsequent shell sources the workspace env via one of:

direnv allow                  # if you use direnv
source ./activate.sh          # bash / zsh
source ./activate.fish        # fish

Then provision the native host:

nros setup native --rmw zenoh

The three roles in practice

Node pkg — a lib crate that contains one node’s logic. It declares nros::node!(T) and carries [package.metadata.nros.node] in its Cargo.toml. It has no fn main() — that lives in the Entry pkg. One Node pkg per node. Think of it as a composable building block: the same talker_pkg lib can be assembled into a native binary and an embedded binary without any source change.

Bringup pkg — a purely declarative directory that owns the launch topology. It contains a package.xml, a system.toml (listing which nodes run, how they’re wired, per-target deploy config), and a launch/ directory with ROS 2 launch XML. No Cargo.toml, no compiled code. Naming convention <system>_bringup, matching nav2 / Autoware / turtlebot3. This role is optional — a workspace with a single deploy target can fold the launch files and system.toml directly into the Entry pkg.

Entry pkg — a bin crate that boots a topology on one specific board. It carries [package.metadata.nros.entry] with deploy = "<board>" and a src/main.rs that is just nros::main!(...). One Entry pkg per deploy target. The Entry pkg links in all Node pkg libs, links in the board crate, and hands control to the nano-ros runtime.

The app-node shape you already know (examples/native/rust/talker/) is effectively a fused Entry + Node pkg: a single package that is both the logic and the boot point. That fusion is fine — and encouraged — for single-node work. Only split when you actually need the flexibility.

ROS 2 ↔ nano-ros command map

If you’re coming from ROS 2, here’s the mapping of the commands you already know:

ROS 2nano-rosNotes
ros2 pkg createnros new <name> --platform <plat> [--lang <lang>]scaffolds a Node pkg
colcon buildcargo build (Rust) / cmake --build build (C++)use the underlying tool directly
ros2 launch <pkg> <file>cargo run -p <entry_pkg>composed Entry pkg IS the launch product (Phase 212.N + 222.D); the old launch wrapper was removed in nros 0.5.0
(plan/validate)nros plannros checkresolve + statically check the topology
ros2 run <pkg> <exe>run the Entry pkg binary (cargo run)one Entry pkg per board

Build verbs (cargo for Rust, cmake for C/C++, west for Zephyr, idf.py for ESP-IDF) are used directly — there is no CLI build indirection. The composed Entry pkg binary IS the launch product: one Entry pkg = one binary = one process. Multi-process orchestration (equivalent to multiple ros2 launch nodes in separate processes) is a separate Entry pkg per deploy + a shell script / tmux session, not a CLI verb.

The app-node shape stays valid

There is no obligation to restructure. If your project is one node on one board, the app-node shape (src/main.rs + one package = both logic and boot) is perfectly idiomatic and has no runtime penalty. The three-role split is a tool for when you need it, not a gatekeeping requirement.

Where to go next

Walk through the multi-node project model step by step:

  1. Node packages — scaffold and implement Node pkgs with nros::node!.
  2. Bringup packages — declare your topology in a Bringup pkg.
  3. Entry packages — write the Entry pkg that boots everything together.
  4. C / C++ multi-node workspaces — use the same project shape with CMake.

For C Node pkgs hosted by a C++ Entry pkg, see Mixed-language workspace.

For the full API reference covering all three roles, see Role reference.

Node packages

This page is part of the Multi-Node Projects group. Previous: Project layout — Next: Bringup packages

A Node pkg is the unit of reusable behaviour in a multi-node workspace. It is a Rust library — a [lib] crate — that implements one node and registers it with nros::node!(T). The Entry pkg is what boots the binary; the Node pkg is what runs inside it.


Prereqs

Pick one path from a fresh checkout — just is NOT a prereq.

A. Bare machine (no Rust, no just, no cargo):

./scripts/bootstrap.sh base

Installs rustup, just, builds the in-tree nros CLI at packages/cli/target/release/nros, leaves the binary on PATH for this shell.

B. Already have cargo (most contributors):

cargo build --release --manifest-path packages/cli/Cargo.toml --bin nros
export PATH="$PWD/packages/cli/target/release:$PATH"

C. Tagged release, no Rust at all:

./scripts/install-nros-prebuilt.sh

Downloads the matching nros-<triple>.tar.gz from the GitHub release, sha256-verifies, installs to packages/cli/target/release/nros.

Every subsequent shell sources the workspace env via one of:

direnv allow                  # if you use direnv
source ./activate.sh          # bash / zsh
source ./activate.fish        # fish

Then provision the native host:

nros setup native --rmw zenoh

Scaffolding a Node pkg

Use nros new to create a skeleton:

nros new talker --platform native --lang rust

nros new creates a project skeleton. For a workspace, move (or create) the result under src/talker_pkg/ so the workspace root’s Cargo.toml can include it as a member:

# workspace Cargo.toml
[workspace]
resolver = "2"
members = ["src/talker_pkg", "src/listener_pkg", "src/native_entry"]

Node pkg anatomy

A Node pkg has three files:

src/talker_pkg/
├── package.xml          # ROS 2 manifest — <exec_depend> per message package
├── Cargo.toml           # [lib] + [package.metadata.nros.node] metadata
└── src/lib.rs           # impl Node + ExecutableNode; ends with nros::node!(Talker);

No fn main() here — a Node pkg is a library linked into an Entry pkg. The Entry pkg’s macro-generated runtime owns nros::init, executor open, RMW registration, and the spin/yield loop.


Cargo.toml — the [package.metadata.nros.node] block

The metadata block is what the nros CLI reads to discover, name, and wire this node into a topology. From examples/stm32f4/rust/talker_pkg/Cargo.toml:

[lib]
crate-type = ["rlib"]

[dependencies]
nros = { path = "../../../../packages/core/nros", default-features = false,
         features = ["alloc", "rmw-cffi", "platform-bare-metal", "ros-humble"] }

[package.metadata.nros.node]
class = "stm32f4_talker_pkg::Talker"
name = "talker"
default_namespace = "/"

The three fields in [package.metadata.nros.node]:

FieldPurpose
classFully-qualified Rust path to the type that impls Node + ExecutableNode
nameDefault ROS 2 node name (remappable at launch)
default_namespaceDefault namespace (remappable at launch)

For a native workspace the nros dep would use features = ["std", "rmw-cffi", "platform-posix", "ros-humble"] instead of platform-bare-metal. The RMW feature (rmw-zenoh, rmw-xrce, rmw-cyclonedds) is chosen at build time — it is not baked into the Node pkg itself.


src/lib.rs — the node implementation

A Node pkg implements two traits: Node (declarative registration) and ExecutableNode (per-callback body), then calls nros::node! to export the trampolines the Entry macro expects.

Here is the essential shape, drawn from examples/stm32f4/rust/talker_pkg/src/lib.rs (see that file for the full worked version):

#![allow(unused)]
fn main() {
use nros::{
    CallbackCtx, ExecutableNode, Node, NodeContext, NodeOptions, NodeResult,
    TimerDuration,
};

pub struct Talker;

impl Node for Talker {
    const NAME: &'static str = "talker";

    fn register(ctx: &mut NodeContext<'_>) -> NodeResult<()> {
        let mut node = ctx.create_node(NodeOptions::new("talker"))?;
        let chatter = node.create_publisher_for_topic::<MyMsg>("/chatter")?;
        let _timer =
            node.create_timer_for_callback_name("on_tick", TimerDuration::from_millis(1000))?;
        node.callback_for_name("on_tick")
            .publishes_entity(&chatter)?;
        Ok(())
    }
}

impl ExecutableNode for Talker {
    type State = i32;

    fn init() -> Self::State { 0 }

    // See the full example for the callback body.
}

nros::node!(Talker);   // <-- exports the trampolines; this is the last line
}

Key points:

  • Node::register is declarative — it runs once at startup to declare publishers, subscriptions, timers, and callback edges. No message bytes flow here.
  • ExecutableNode::on_callback is the body — called by the Entry pkg’s executor each time a callback fires. state is your per-node mutable storage.
  • nros::node!(Talker) must be the last public API call in the file. It generates the extern "C" trampolines the Entry macro imports.
  • There is no fn main() in a Node pkg.

package.xml — the ROS 2 manifest

A Node pkg that uses generated message types lists them as <exec_depend> entries. Minimal example:

<?xml version="1.0"?>
<package format="3">
  <name>talker_pkg</name>
  <version>0.1.0</version>
  <description>Talker node</description>
  <maintainer email="dev@example.com">Developer</maintainer>
  <license>MIT OR Apache-2.0</license>

  <depend>std_msgs</depend>

  <export>
    <build_type>ament_cargo</build_type>
  </export>
</package>

If your node uses no external message packages, the <depend> line can be omitted.


Building

From the workspace root, sync generated interfaces first, then let Cargo compile the Node pkgs and Entry pkg:

# From examples/workspaces/rust/ (or your workspace root):
nros ws sync
nros codegen-system --bringup demo_bringup
cargo build -p native_entry

No per-Node-pkg invocation is needed — the workspace resolver handles dependency ordering.

To cross-compile for an embedded target, pass --target and ensure .cargo/config.toml in the workspace root sets the right linker and target:

cargo build --target thumbv7em-none-eabihf

Next steps

  • Bringup packages — wire the Node pkgs together into a topology.
  • Entry packages — build the binary that boots the topology on real hardware or a host process.
  • Role reference — the full reference for all three roles, metadata fields, and the nros::main!() four forms.

Bringup packages

A Bringup pkg is the declarative glue that ties your Node packages together into a runnable topology. It owns the launch file, the wiring between nodes, and the per-target deploy config — all without any compiled code of its own.

Pre-requisite: You’ve scaffolded your Node packages following the Node packages guide. This page adds the demo_bringup layer between them and the Entry package that boots everything.


Prereqs

Pick one path from a fresh checkout — just is NOT a prereq.

A. Bare machine (no Rust, no just, no cargo):

./scripts/bootstrap.sh base

Installs rustup, just, builds the in-tree nros CLI at packages/cli/target/release/nros, leaves the binary on PATH for this shell.

B. Already have cargo (most contributors):

cargo build --release --manifest-path packages/cli/Cargo.toml --bin nros
export PATH="$PWD/packages/cli/target/release:$PATH"

C. Tagged release, no Rust at all:

./scripts/install-nros-prebuilt.sh

Downloads the matching nros-<triple>.tar.gz from the GitHub release, sha256-verifies, installs to packages/cli/target/release/nros.

Every subsequent shell sources the workspace env via one of:

direnv allow                  # if you use direnv
source ./activate.sh          # bash / zsh
source ./activate.fish        # fish

Then provision the native host:

nros setup native --rmw zenoh

What a Bringup pkg is

A Bringup pkg is pure declarative — no Cargo.toml, no CMakeLists.txt, no src/. Its job is to describe which nodes run, how they’re wired, and where they deploy. Naming convention: <system>_bringup (aliased <system>_launch), matching nav2 / Autoware / turtlebot3.

It is optional: required only when two or more Entry packages share one topology. A single-Entry workspace can fold launch/ + system.toml directly into the Entry pkg.


Anatomy

src/demo_bringup/
├── package.xml          # ROS 2 manifest; <exec_depend> per node pkg
├── system.toml          # [system] + [[component]] + [deploy.<target>]
├── launch/
│   └── system.launch.xml   # ROS 2 launch schema, verbatim
└── config/                 # optional — params.yaml, per-target overrides

system.toml — node wiring + deploy targets

The system.toml is the machine-readable topology. It lists every node the system runs, its class path, and one or more deploy targets (which board, which RMW, which domain).

Below is a minimal two-node example adapted from the real fixture at packages/testing/nros-tests/fixtures/orchestration_e2e/demo_pkg_bringup/system.toml:

[system]
name = "demo"
rmw = "zenoh"
domain_id = 0

[[component]]
pkg = "talker_pkg"
class = "talker_pkg::Talker"
name = "talker"

[[component]]
pkg = "listener_pkg"
class = "listener_pkg::Listener"
name = "listener"

[deploy.native]
kind = "self"
target = "x86_64-unknown-linux-gnu"

Key fields:

FieldMeaning
[system] nameLogical system name; used by nros plan/check
[system] rmwDefault RMW for all components (zenoh, xrce, cyclonedds)
[system] domain_idROS 2 domain (compile-time on embedded, runtime env on host)
[[component]] pkgThe ROS package name (matches <name> in package.xml)
[[component]] classFully-qualified Rust type (crate::TypeName)
[[component]] nameNode name at runtime
[deploy.<target>]Deploy target block; read by nros check and Entry codegen
[deploy.<t>] kind"self" = host native binary; "flash" = embedded target
[deploy.<t>] targetRust target triple

For multi-domain setups or cross-domain bridges add [[domain]] and [[bridge]] sections — see docs/design/0024-multi-node-workspace-layout.md §11 for the full schema.


launch/system.launch.xml — ROS 2 launch schema

The launch file uses the ROS 2 launch XML schema verbatim — nano-ros reads it with the same parser so existing nav2/Autoware/turtlebot3 XML pastes in and Just Works.

<launch>
  <node pkg="talker_pkg" exec="talker" name="talker"/>
  <node pkg="listener_pkg" exec="listener" name="listener"/>
</launch>

v1 tag set

TagPurpose
<launch>Root element
<arg name="…" default="…"/>Declare a launch argument
<node pkg="…" exec="…" name="…"/>Instantiate a node
<param name="…" value="…"/>Set a parameter (nested inside <node>)
<remap from="…" to="…"/>Topic/service remapping (nested inside <node>)
<group ns="…">Namespace a group of nodes
<include file="…"/>Nest another launch file

Substitutions

  • $(find <pkg>) — resolves to the package’s install/source path
  • $(var <arg>) — expands a launch argument
  • $(env <name>) — reads an environment variable

A richer example using args and remapping (taken from the real fixture):

<launch>
  <arg name="talker_name" default="talker" />

  <node pkg="talker_pkg" exec="talker" name="$(var talker_name)" output="screen">
    <param name="rate_hz" value="25" />
    <remap from="chatter" to="/chatter" />
  </node>

  <node pkg="listener_pkg" exec="listener" name="listener"/>
</launch>

Note: Python .launch.py files are not yet supported in v1 — use the XML schema above.


package.xml

A standard ROS 2 manifest. List each Node package as an <exec_depend>:

<?xml version="1.0"?>
<package format="3">
  <name>demo_bringup</name>
  <version>0.1.0</version>
  <description>Bringup package for the demo system</description>
  <maintainer email="you@example.com">Your Name</maintainer>
  <license>Apache-2.0</license>

  <exec_depend>talker_pkg</exec_depend>
  <exec_depend>listener_pkg</exec_depend>

  <export>
    <build_type>ament_cmake</build_type>
  </export>
</package>

No <build_depend> entries — there is nothing to compile.


Workflow: check → run

Once your Bringup pkg is written, use nros check to validate and cargo run to execute the topology:

# 1. Lint the bringup pkg (pure-declarative check — no Cargo.toml, stray files, etc.)
nros check --bringup src/demo_bringup

# 2. Lint the whole workspace (pkg/class rows, duplicate system.toml, etc.)
nros check --workspace .

# 3. Run the composed Entry binary (boots all nodes in a single process)
zenohd --listen tcp/127.0.0.1:7447 &   # router — in another shell
cargo run -p native_entry

Both nros check forms pass for the canonical template at examples/workspaces/rust/.

Caveat — nros plan with this template

  • nros plan demo_bringup resolves a topology into plan.json for static type/QoS checks, but it currently requires pre-collected source-metadata sidecars (record.json + per-pkg _metadata/*.json). The automatic metadata-build path (nros metadata --build) is not yet wired for lib-only Node pkgs, so nros plan does not produce a plan straight from this template. See packages/testing/nros-tests/fixtures/orchestration_e2e/ for the pre-collected-sidecar pipeline.

The canonical template README at examples/workspaces/rust/README.md is the source of truth for the current CLI state.


Runnable copy-out

examples/workspaces/rust/ is the canonical Rust 3-role workspace that pairs with this guide. Copy the whole directory out and rename the packages. nros ws sync materializes generated message crates, nros codegen-system bakes the Bringup package, and cargo build -p native_entry builds the Entry pkg.

The workspace README at examples/workspaces/rust/README.md documents the exact CLI commands that are verified green today.


When you don’t need a Bringup pkg

If you have a single Entry pkg and don’t plan to share the topology across multiple boards, fold launch/ and system.toml directly into the Entry pkg. The nros::main! macro accepts a launch = argument that names the bringup package:

#![allow(unused)]
fn main() {
// Multi-node: reads launch/system.launch.xml from demo_bringup
nros::main!(launch = "demo_bringup");

// Explicit file within a bringup pkg
nros::main!(launch = "demo_bringup:sim.launch.xml");
}

If the launch files live inside the Entry pkg itself, point at it by name. The Entry package page covers this in full.


Where to go next

Entry packages

An Entry pkg is the binary that boots a topology on a specific board. Where a Node pkg is a library (no fn main) and a Bringup pkg is purely declarative, the Entry pkg is the one thing that actually runs: it names a board, wires the runtime, and — for multi-node setups — points at a Bringup pkg that describes which nodes should be launched.

You have one Entry pkg per deploy target. A workspace targeting both a native workstation and an STM32F4 board has two Entry pkgs that reference the same Node pkgs; only the board and (optionally) the launch target differ.

Prereqs

Pick one path from a fresh checkout — just is NOT a prereq.

A. Bare machine (no Rust, no just, no cargo):

./scripts/bootstrap.sh base

Installs rustup, just, builds the in-tree nros CLI at packages/cli/target/release/nros, leaves the binary on PATH for this shell.

B. Already have cargo (most contributors):

cargo build --release --manifest-path packages/cli/Cargo.toml --bin nros
export PATH="$PWD/packages/cli/target/release:$PATH"

C. Tagged release, no Rust at all:

./scripts/install-nros-prebuilt.sh

Downloads the matching nros-<triple>.tar.gz from the GitHub release, sha256-verifies, installs to packages/cli/target/release/nros.

Every subsequent shell sources the workspace env via one of:

direnv allow                  # if you use direnv
source ./activate.sh          # bash / zsh
source ./activate.fish        # fish

Then provision the native host (the canonical first Entry pkg target; for STM32F4 / Zephyr / ESP32 swap in the matching nros setup board):

nros setup native --rmw zenoh

Package layout

src/native_entry/
├── package.xml
├── Cargo.toml           # [[bin]] + deps on node pkgs + board crate
│                        # + [package.metadata.nros.entry]
└── src/main.rs          # nros::main!(launch = "demo_bringup");

No library code lives here. The Entry pkg links the Node pkg rlibs and hands them to the runtime that nros::main!() generates.

Cargo.toml metadata

The [package.metadata.nros.entry] table tells the CLI which deploy target this binary is built for. The embedded example examples/stm32f4/rust/talker-embassy/ uses:

[package.metadata.nros.entry]
deploy = "embassy-stm32f4"

A native Entry pkg that references a Bringup pkg looks like:

[package.metadata.nros.entry]
deploy = "native"

[package.metadata.nros.deploy.native]
board     = "posix"
rmw       = "zenoh"
domain_id = 0

deploy is the key that nros check and the Entry macro use to find the board crate and verify the topology. Keep it short and descriptive — it becomes the identifier in nros plan output and in system.toml’s [deploy.<name>] table when you later add a Bringup pkg.

nros::main!() — four forms

#![allow(unused)]
fn main() {
// 1. Single-node self-bringup: reads [package.metadata.nros.entry] deploy
//    from Cargo.toml and boots the Node pkg that is the only member of
//    this workspace (or the one marked default).
nros::main!();

// 2. Single-node, explicit board type.
nros::main!(board = NativeBoard);

// 3. Multi-node: reference a Bringup pkg; boot its default launch file
//    (the one listed under [system] in system.toml).
nros::main!(launch = "demo_bringup");

// 4. Multi-node, explicit launch file.
nros::main!(launch = "demo_bringup:sim.launch.xml");

// 5. Full form: board + launch file + runtime arg overrides.
nros::main!(board = NativeBoard, launch = "demo_bringup:sim.launch.xml", args = [("use_sim","true")]);
}

The macro reads [package.metadata.nros.entry] at compile time to select the right board and executor backend. On Embassy / RTIC targets it emits the framework-specific #[embassy_executor::main] or #[rtic::app] body so your src/main.rs stays a single line.

The real examples/stm32f4/rust/talker-embassy/src/main.rs collapses to exactly this:

#![allow(unused)]
#![no_std]
#![no_main]

fn main() {
use defmt_rtt as _;
use panic_probe as _;

nros::main!();
}

Escape hatch

If you need more control than the macro provides — custom startup ordering, hardware init before the runtime, or a fully manual executor loop — you can bypass nros::main!():

#![allow(unused)]
fn main() {
// Option A: delegate init to the board crate, supply your own closure.
<NativeBoard as BoardEntry>::run(|runtime| {
    let node = runtime.create_node("talker", "/", &Default::default())?;
    // ...
    Ok(())
});

// Option B: fully manual — no board crate.
let executor = nros::Executor::open(&ExecutorConfig::default())?;
// wire nodes, spin manually ...
}

Option A is the right choice when you need to run something before the first spin (e.g. DMA setup, flash unlock). Option B is there for board-bringup authors adding a new platform.

Running a native Entry pkg

The verified path for the canonical Rust workspace is cargo run -p native_entry. Start a Zenoh router first, then boot the Entry binary from the workspace root:

# in another shell:
zenohd --listen tcp/127.0.0.1:7447 &

cargo run -p native_entry

native_entry opens the executor against the router, registers talker + listener (composed into a single process), and runs the topology.

The canonical Rust workspace is at examples/workspaces/rust/. For Zephyr, QEMU, ESP-IDF, and other non-native targets, use the platform’s native build/run tool or the focused just <plat> run recipe.

Running on Zephyr

On Zephyr the RTOS framework is the workflow: west build is the build verb, Kconfig selects the RMW, and the Entry is an ordinary Zephyr application. There is no nros build / nros launch build path here, and you do not type the RMW as a Cargo --features flag or bake the board into the package.

source ./activate.sh

# Provision message bindings once. This is platform-agnostic workspace
# provisioning (sibling to `west update` / `rosdep`), NOT a compile step —
# the same `nros ws sync` output feeds every board and every RMW.
nros ws sync

# west is the build verb. Choose the board with `-b`, and select the RMW with
# the matching Kconfig overlay via -DCONF_FILE. The Entry source never changes.
west build -b native_sim/native/64 src/zephyr_entry \
    -- -DCONF_FILE="prj.conf;prj-zenoh.conf"

west build -t run            # native_sim; `west flash` for hardware

Swap -b native_sim/native/64 for any other Zephyr board (-b nrf52840dk/nrf52840, -b stm32f4_disco, …) and prj-zenoh.conf for prj-xrce.conf / prj-cyclonedds.conf to pick a different RMW — nothing else changes. One Zephyr Entry pkg (src/zephyr_entry/) covers every Zephyr board: unlike the board-specific FreeRTOS / ThreadX Entries, Zephyr owns its board abstraction, so the board is chosen at west build -b time rather than baked into the package (see One Entry pkg per board).

The Entry source is identical to the native, FreeRTOS, and ThreadX Entries — the same one-line launch macro, with the launch file as the single source of truth for the node set:

#![allow(unused)]
fn main() {
// examples/workspaces/rust/src/zephyr_entry/src/lib.rs
nros::main!(launch = "demo_bringup:system.launch.xml");
}

One Entry pkg per board

Each deploy target gets its own Entry pkg. A workspace that runs on both native and embassy-stm32f4 would have two Entry pkgs that share the same Node pkg library:

Entry pkgdeploy keyBoard crate
native_entry"native"nros-board-posix
stm32f4_entry"embassy-stm32f4"nros-board-embassy-stm32f4

Both reference the same talker_pkg and listener_pkg Node pkg rlibs. The board crate provides the BoardEntry impl and any hardware-specific initialisation; the Node pkgs are board-agnostic.

Zephyr is the exception — one Entry per RTOS, not per board. Zephyr already owns its board abstraction, so a single zephyr_entry covers native_sim, nrf52, stm32, aemv8r, … with the board chosen at west build -b <board> time. Contrast FreeRTOS / ThreadX, whose board crates are board-specific, so each of those Entries bakes one board. See Running on Zephyr.

The examples/stm32f4/rust/talker-embassy/ example demonstrates the embedded shape: deploy = "embassy-stm32f4" + nros::main!(); on a no_std / no_main binary that delegates everything to the EmbassyStm32F4 board crate.

C / C++ Entry packages

C and C++ Entry packages use the same role split through CMake. See C / C++ multi-node workspaces.

Where to go next

C / C++ multi-node workspaces

The four previous chapters (project layout, node packages, bringup packages, entry packages) describe the canonical three-role node + bringup + entry shape against the Rust path (nros::node!(…) + nros::main!(launch = …)). This chapter shows the C and C++ path through the same shape, role-for-role.

Phase 219 closed the parity gap. Same launch.xml, same package.xml, same system.toml, same workspace pkg-index — the only thing that changes language-side is the cmake-fn / macro surface.

TL;DR — side-by-side

RoleRustC / C++
Node pkglib.rs with nros::node!(MyNode) + [package.metadata.nros.node] in Cargo.tomlTalker.{hpp,cpp} with a configure(::nros::Node&) component method (C++) / NROS_C_COMPONENT (C); CMakeLists.txt calling nano_ros_node_register(NAME … CLASS … SOURCES …)
Bringup pkgpackage.xml + system.toml + launch/*.launch.xml (no Cargo.toml)identical (language-agnostic)
Entry pkgsrc/main.rs with nros::main!(launch = "demo_bringup:system.launch.xml")src/main.cpp with NROS_MAIN(nros::board::NativeBoard, "demo_bringup:system.launch.xml"); CMakeLists.txt calling nano_ros_entry(NAME … LAUNCH "demo_bringup:system.launch.xml" DEPLOY native)
Workspace rootCargo.toml [workspace] members = […]CMakeLists.txt calling nano_ros_workspace(BACKEND zenoh PLATFORM posix SUBDIRS src/talker_pkg src/listener_pkg src/native_entry)
Buildnros ws sync + cargo build -p native_entrynros ws sync + cmake -S . -B build + cmake --build build
Bootcargo run -p native_entry./build/.../native_entry

The reference C++ workspace ships in-tree at examples/workspaces/cpp/. Copy the whole directory, rename the packages.

Workspace layout

Identical structure to the Rust template, swapping Cargo.tomlCMakeLists.txt:

my_ws/
├── CMakeLists.txt                # nano_ros_workspace(SUBDIRS …)
└── src/
    ├── talker_pkg/               # Node pkg (C++)
    │   ├── package.xml
    │   ├── CMakeLists.txt        # nano_ros_node_register(…)
    │   └── src/{Talker.hpp,Talker.cpp}
    ├── listener_pkg/             # Node pkg (C++)
    │   ├── package.xml
    │   ├── CMakeLists.txt
    │   └── src/{Listener.hpp,Listener.cpp}
    ├── demo_bringup/             # Bringup pkg (language-agnostic — copy/paste
    │   ├── package.xml           #          works between Rust and C++ workspaces)
    │   ├── system.toml
    │   └── launch/system.launch.xml
    └── native_entry/             # Entry pkg (C++)
        ├── package.xml
        ├── CMakeLists.txt        # nano_ros_entry(LAUNCH …)
        └── src/main.cpp          # NROS_MAIN(…) one-liner

Workspace root

Four declarations:

cmake_minimum_required(VERSION 3.22)
project(my_ws LANGUAGES C CXX)
include(<nano-ros>/cmake/NanoRosWorkspace.cmake)

nano_ros_workspace(
    BACKEND        zenoh                # zenoh | xrce | cyclonedds
    PLATFORM       posix                # posix | … (default posix)
    NANO_ROS_ROOT  <path-to-nano-ros>   # also: -D cache var, $NANO_ROS_ROOT,
                                        # or auto-walk for nros-sdk-index.toml
    SUBDIRS        src/talker_pkg
                   src/listener_pkg
                   src/native_entry
)

nano_ros_workspace() (Phase 219.I) does the heavy lifting in one call:

  1. Sets NANO_ROS_PLATFORM=posix + NANO_ROS_RMW=zenoh.
  2. add_subdirectory(<nano-ros>) once at root scope (so per-pkg subdirs don’t collide on re-include).
  3. include(NanoRosNodeRegister.cmake) + include(NanoRosEntry.cmake) once.
  4. add_subdirectory(<each member>) for each SUBDIRS entry.

Subdir CMakeLists begin with the dual call:

nano_ros_workspace_pkg_guard()  # no-op inside a workspace; bootstraps standalone solo

— the cmake equivalent of cargo [workspace] discipline. Every member compiles standalone (with -DNANO_ROS_ROOT=<path>) or as part of the workspace; the per-pkg CMakeLists doesn’t change between modes.

Node pkg

A typed component (RFC-0043) — no main(). The pkg ships a class with a configure(::nros::Node&) method that creates real entities (a Publisher, a Timer) and binds member callbacks by identity (member-fn-pointer template param, no string callback name, no interpreter). The Entry pkg constructs the object and calls configure(node); the executor dispatches the callbacks.

# src/talker_pkg/CMakeLists.txt
cmake_minimum_required(VERSION 3.22)
project(talker_pkg LANGUAGES C CXX)
nano_ros_workspace_pkg_guard()
nros_find_interfaces(LANGUAGE CPP SKIP_INSTALL)

nano_ros_node_register(
    NAME    talker
    CLASS   talker_pkg::Talker     # §212.L.4 — class prefix must equal PROJECT_NAME
    SOURCES src/Talker.cpp)

target_link_libraries(talker_pkg_talker_component PUBLIC std_msgs__nano_ros_cpp)
// src/talker_pkg/include/talker_pkg/Talker.hpp
#pragma once
#include <nros/component.hpp>
#include <nros/nros.hpp>
#include "std_msgs.hpp"

namespace talker_pkg {

class Talker {
    ::nros::Publisher<std_msgs::msg::Int32> pub_;
    ::nros::Timer timer_;
    int count_ = 0;

    void on_tick();  // real body; bound via &Talker::on_tick (no name)

  public:
    ::nros::Result configure(::nros::Node& node);
};

}  // namespace talker_pkg
// src/talker_pkg/src/Talker.cpp
#include "talker_pkg/Talker.hpp"

namespace talker_pkg {

void Talker::on_tick() {
    std_msgs::msg::Int32 m;
    m.data = count_++;
    (void)pub_.publish(m);
}

::nros::Result Talker::configure(::nros::Node& node) {
    ::nros::Result r = node.create_publisher(pub_, "/chatter");
    if (!r.ok()) return r;
    // Member-fn-pointer-as-template-param → no-alloc trampoline; `this` is ctx.
    return ::nros::bind_timer<Talker, &Talker::on_tick>(node, timer_, 1000, this);
}

}  // namespace talker_pkg

The Entry pkg constructs Talker in static storage and calls configure(node) on the real executor — the same component model the Rust nros::node!(Talker) + the C NROS_C_COMPONENT paths use, so C++, C, and Rust Node pkgs interoperate in one launch graph.

Scaffold a C++ Node pkg with:

$ nros new --component my-talker --lang cpp --use-case talker
✓ Created nano-ros C++ Node pkg 'my-talker'
  Class     : my_talker::Talker
  Node      : talker
  Kind      : typed component (RFC-0043)

Bringup pkg

Language-agnostic — copy verbatim from the bringup chapter. package.xml + system.toml + launch/system.launch.xml. No Cargo.toml, no CMakeLists.txt. Stock ROS 2 launch.xml from nav2 / Autoware / turtlebot3 pastes in modulo unsupported tags.

<!-- src/demo_bringup/launch/system.launch.xml -->
<launch>
  <node pkg="talker_pkg" exec="talker" name="talker"/>
  <node pkg="listener_pkg" exec="listener" name="listener"/>
</launch>

Entry pkg

The C++ Entry pkg’s CMakeLists.txt calls nano_ros_entry(LAUNCH …) — Phase 219.D added the LAUNCH keyword:

# src/native_entry/CMakeLists.txt
cmake_minimum_required(VERSION 3.22)
project(native_entry LANGUAGES C CXX)
nano_ros_workspace_pkg_guard()

nano_ros_entry(
    NAME    native_entry
    SOURCES src/main.cpp                              # user-authored
    LAUNCH  "demo_bringup:system.launch.xml"          # Phase 219.D
    DEPLOY  native)

At configure time the cmake fn shells nros codegen entry --lang cpp --typed, emits ${CMAKE_BINARY_DIR}/native_entry_nros_main_generated.cpp (the canonical int main() body that constructs each launch node’s component + calls configure(node) on the real executor via NativeBoard::run_components), appends it to the target’s sources, and auto-links every <pkg>_<exec>_component static lib the launch XML named (Phase 219.J). The user’s main.cpp is a single declarative line:

// src/native_entry/src/main.cpp
#include <nros/main.hpp>
NROS_MAIN(nros::board::NativeBoard, "demo_bringup:system.launch.xml")

NROS_MAIN(...) is a sentinel macro — the cmake fn owns the generated body, the user’s TU is documentation + IDE hint.

nros new scaffolds an Entry pkg:

$ nros new my-entry --lang cpp --platform native

Build + boot

nros ws sync
nros codegen-system --bringup demo_bringup
cmake -S . -B build -DNANO_ROS_ROOT=<path-to-nano-ros>
cmake --build build
./build/src/native_entry/native_entry

The build produces:

  • src/talker_pkg/libtalker_pkg_talker_component.a — the Node pkg static lib (_component is the compatibility target suffix).
  • src/listener_pkg/liblistener_pkg_listener_component.a — ditto.
  • src/native_entry/native_entry — the Entry exe, with the generated int main() + register-call sequence + Board boot stub linked in.

cmake configure is incremental — pinned CMAKE_CONFIGURE_DEPENDS on every file nros codegen entry reads (depfile from the CLI), so any launch.xml / package.xml / Node pkg edit re-runs codegen.

What runs where

ConcernLives in
Node entities + real callbacksNode pkg configure(::nros::Node&)
Topology + launch args + per-target deployBringup pkg system.toml + launch/*.launch.xml
int main() + executor init + spinGenerated TU emitted by the Entry pkg’s cmake fn
Board + RMW selectionEntry pkg’s nano_ros_entry(BOARD …) arg

Same partition as the Rust track — the only thing that changes is the syntax the user types into the three pkgs.

C / C++ scaffolding via nros new

CommandOutput
nros new <name> --lang cpp --platform nativeC++ Entry pkg (single-Node self-bringup; swap in a multi-Node LAUNCH arg post-219.D)
nros new <name> --lang c --platform nativeC Entry pkg (same shape)
nros new --component <name> --lang cpp --use-case talkerC++ Node pkg; --component is the compatibility scaffold flag
nros new system <name>_bringup --components a,bBringup pkg (language-agnostic — works for both Rust and C++ workspaces)

The C-side compatibility scaffold (nros new --component … --lang c) is available for pure-C Node pkgs. Pure-C and mixed C/C++ workspace examples live under examples/workspaces/.

See also

Mixed-language workspace

nano-ros Node pkgs are linked through a C ABI register trampoline, so one Entry pkg can host Node pkgs written in different languages. The native reference shape is:

  • C Node pkg for C code you want to keep in C.
  • C++ Entry pkg for the boot harness and generated launch wiring.
  • Optional C++ Node pkgs in the same workspace.
  • One Bringup pkg with the normal ROS 2 launch XML.

The mixed workspace is in examples/workspaces/mixed/. For a pure-C Entry host, use examples/workspaces/c/.

Layout

my_ws/
├── CMakeLists.txt
└── src/
    ├── c_talker_pkg/             # C Node pkg
    ├── cpp_listener_pkg/         # C++ Node pkg
    ├── demo_bringup/             # launch XML + system.toml
    └── native_entry/             # C++ Entry pkg

The root uses the same CMake workspace helper as the C++ track:

include(<nano-ros>/cmake/NanoRosWorkspace.cmake)

nano_ros_workspace(
    BACKEND  zenoh
    PLATFORM posix
    SUBDIRS  src/c_talker_pkg
             src/cpp_listener_pkg
             src/native_entry)

C Node pkg

A C Node pkg is a typed component (RFC-0043): no main(). It defines a state struct + a configure(node, executor, self) function and exports the C-ABI factory/configure seam via NROS_C_COMPONENT(StateT, configure_fn) — the typed Entry creates the node and runs configure on the real executor.

# Raw `/chatter` publisher carries the type name as a string → no generated C
# bindings needed (so no nros_find_interfaces for this pkg).
nano_ros_node_register(
    NAME     talker
    CLASS    c_talker_pkg::Talker
    LANGUAGE C
    TYPED
    SOURCES  src/Talker.c
    DEPLOY   native)
#include <stdint.h>
#include <nros/component.h>

typedef struct {
    _Alignas(8) uint8_t pub[NROS_C_PUBLISHER_STORAGE_SIZE];
    int32_t count;
} c_talker_pkg_t;

static void on_tick(void* ctx) {
    c_talker_pkg_t* self = (c_talker_pkg_t*)ctx;
    uint8_t buf[8] = {0x00, 0x01, 0x00, 0x00};  /* CDR_LE header + LE int32 */
    buf[4] = (uint8_t)self->count;
    (void)nros_cpp_publish_raw(self->pub, buf, sizeof(buf));
    self->count++;
}

static nros_ret_t talker_configure(const nros_cpp_node_t* node, void* executor,
                                   c_talker_pkg_t* self) {
    self->count = 0;
    int32_t rc = nros_cpp_publisher_create(node, "/chatter",
        "std_msgs::msg::dds_::Int32_", "", nros_c_qos_default(), self->pub);
    if (rc != 0) return rc;
    size_t timer;
    return nros_cpp_timer_create(executor, /*period_ms=*/1000, on_tick, self, &timer);
}

NROS_C_COMPONENT(c_talker_pkg_t, talker_configure)

nano_ros_node_register(... LANGUAGE C TYPED ...) injects NROS_PKG_NAME, so NROS_C_COMPONENT exports the __nros_c_component_<pkg>_{create,configure} seam the typed Entry calls — interoperable with C++ configure(Node&) and Rust nros::node! components in one launch graph.

Entry pkg

Use a C++ or Rust Entry pkg as the usual host for migration work. In the C++ template:

nano_ros_entry(
    NAME    native_entry
    SOURCES src/main.cpp
    BOARD   native
    LAUNCH  "demo_bringup:system.launch.xml"
    DEPLOY  native)

The launch file names both Node pkgs:

<launch>
  <node pkg="c_talker_pkg" exec="talker" name="talker"/>
  <node pkg="cpp_listener_pkg" exec="listener" name="listener"/>
</launch>

The generated Entry translation unit calls each package’s __nros_component_<pkg>_register symbol and the CMake sidecar links the matching static libraries. The symbol keeps the legacy component spelling for ABI compatibility; the user-facing package role is still Node pkg.

For a pure-C workspace, the Entry pkg uses the same launch-driven shape with LANG c:

nano_ros_entry(
    NAME    native_entry
    SOURCES src/main.c
    BOARD   native
    LAUNCH  "demo_bringup:system.launch.xml"
    LANG    c
    DEPLOY  native)

Scaffolding

# Current compatibility scaffold for Node pkgs:
nros new --component c_talker_pkg --lang c --use-case talker
nros new --component cpp_listener_pkg --lang cpp --use-case listener
nros new system demo_bringup --components c_talker_pkg,cpp_listener_pkg

For a complete working tree, copy the template instead of creating each package separately.

Role Reference: Node, Bringup, and Entry Packages

nano-ros multi-node workspaces split into three kinds of package:

  • Node pkg — a reusable, board-agnostic node library. Defines what a node does (publishers, subscribers, timers, services, actions) and registers itself with the nros::node!(T) macro. No main(), no board pick, no deploy config. (Previously called a component package; renamed to Node pkg to match ROS 2 composable-node naming.)
  • Bringup pkg — pure declarative: owns the launch topology and per-target deploy config. Contains a package.xml, system.toml, launch/*.launch.xml, and optional config/. No Cargo.toml, no compiled code. Named <system>_bringup. Optional — only required when ≥2 Entry pkgs share one topology; a single-Entry workspace folds launch/ + system.toml into the Entry pkg directly.
  • Entry pkg — a per-board binary that composes one or more Node pkgs into a runnable system. Owns the board choice (via the Board trait family), the launch file reference, and the deploy/domain/bridge config. Typically ~30 LoC of main.rs.

The split exists because a node’s logic is portable across boards, but boot + transport + deploy config is not. One Node pkg can be reused across native POSIX, FreeRTOS, and Zephyr targets by writing one Entry pkg per target.

single-Node convenience: for a single-Node workspace on native hosts, a Node pkg can declare [package.metadata.nros.entry] deploy = "<board>" in its own Cargo.toml and skip the Entry pkg directory entirely — see Single-Node-pkg convenience below. Embedded boards still need a hand-written Entry pkg.

Node pkg

A Node pkg is a normal Rust library (or C++ static library) with a few nano-ros-specific knobs:

src/talker_pkg/
├── Cargo.toml          # [lib] crate-type = ["rlib", "staticlib"]
│                       # [package.metadata.nros.node]
│                       #     class = "talker_pkg::Talker"
├── package.xml         # ROS 2 package manifest (<exec_depend> etc.)
├── src/
│   └── lib.rs          # impl Node for Talker { … }
│                       # nros::node!(Talker);
└── launch/             # OPTIONAL — per-node launch fragment
    └── talker.launch.xml

src/lib.rs declares the user class, implements Node + ExecutableNode (init / on_callback / optional tick), and ends with nros::node!(Talker); to emit the register trampoline. Codegen owns the spin loop — your code only describes what the node has and what its callbacks do.

Key rules:

  • No fn main(). A Node pkg builds as rlib + staticlib and is linked into an Entry pkg’s binary. Codegen synthesises the spin driver; you never hand-write one.
  • class field must start with the pkg dir name. nros check rejects class = "foo::Talker" inside src/talker_pkg/ — the pkg name and the class prefix are the same identity. (Phase 212.L.4.)
  • C++ / C analogue: nano_ros_node_register(NAME … CLASS … SOURCES …) cmake fn + a typed component in the source — C++ a configure(::nros::Node&) method, C a NROS_C_COMPONENT(StateT, configure_fn) seam (RFC-0043). Same conceptual shape, no Cargo.toml.
  • package.xml is mandatory. Even pure-Rust Node pkgs ship one — <exec_depend> lines drive ROS 2 launch discovery when the system runs through ros2 launch outside the nano-ros toolchain.

Bringup pkg (optional)

A Bringup pkg is pure declarative — it owns the launch topology and per-target deploy config, and contains no compiled code:

src/demo_bringup/
├── package.xml          # <name>demo_bringup</name>, <exec_depend> per node
├── system.toml          # [system] + [[component]] + [deploy.<target>] (+ [[domain]]/[[bridge]])
├── launch/
│   └── system.launch.xml   # ROS 2 launch schema, verbatim
└── config/                 # optional — params.yaml, etc.

No Cargo.toml, no CMakeLists.txt, no src/. Naming convention <system>_bringup (alias <system>_launch), matching nav2 / Autoware / turtlebot3. It is optional: required only when two or more Entry pkgs share one topology. A single-Entry workspace folds launch/ + system.toml into the Entry pkg directly.

launch/*.launch.xml is the ROS 2 launch schema verbatim — <launch>, <arg>, <node>, <param>, <remap>, <group>, <include>, with $(find <pkg>) / $(var) / $(env) substitutions. Stock nav2/Autoware XML pastes in and Just Works (Python .launch.py is not supported yet). See the workspace bringup tutorial.

Entry pkg

An Entry pkg is a binary crate that combines one or more Node pkgs with a board choice, a launch file, and per-board deploy config:

src/robot_entry/
├── Cargo.toml          # [[bin]] name = "robot_entry"
│                       # [dependencies]
│                       #     talker_pkg   = { path = "../talker_pkg" }
│                       #     listener_pkg = { path = "../listener_pkg" }
│                       #     nros-board-posix = { … }            # or another family
│                       # [package.metadata.nros.entry]
│                       #     deploy = "native"
│                       # [package.metadata.nros.deploy.native]
│                       #     board     = "posix"
│                       #     rmw       = "zenoh"
│                       #     domain_id = 0
│                       #     locator   = "tcp/127.0.0.1:7447"
├── package.xml         # <exec_depend>talker_pkg</exec_depend>, listener_pkg, …
└── src/
    └── main.rs         # one line: `nros::main!(launch = "demo_bringup");`

The nros::main!() proc-macro (Phase 212.N.9) reads the launch file at compile time, walks the workspace pkg-index for each <node pkg=…> entry, and expands to a fn main() that delegates to <Board as BoardEntry>::run(...), dispatching one <pkg>::register(runtime)? call per launch row. The macro has four forms; pick whichever matches your composition shape:

nros::main!();                                          // single-node self-bringup (reads [..nros.entry] deploy)
nros::main!(board = NativeBoard);                       // single-node, explicit board
nros::main!(launch = "demo_bringup");                   // multi-node, default launch from system.toml
nros::main!(launch = "demo_bringup:sim.launch.xml");    // multi-node, explicit file
nros::main!(                                            // all explicit
    board  = NativeBoard,
    launch = "demo_bringup:sim.launch.xml",
    args   = [("use_sim", "true")],
);

Form-1 (no args) reads [package.metadata.nros.entry] deploy = "<board>" from this Entry pkg’s own Cargo.toml and maps the board key ("native"/"freertos"/"zephyr"/…) to the right board crate via a small lookup table. Forms 2–4 use the user-supplied path verbatim. Forms 3/4 reference a Bringup pkg by <bringup>[:<file>] — the Bringup pkg’s dir hosts launch/<file>.launch.xml plus an optional system.toml naming the default file ([system] default_launch = "..."). The nros::main!() expansion replaces the older build.rs + include!(env!("OUT_DIR")/run_plan.rs) shape end-to-end; new Entry pkgs no longer need a build.rs or a nros-build build-dep — just nros + the target board crate.

Escape hatch: skip the macro entirely and call <NativeBoard as BoardEntry>::run(|runtime| { ... }), or go fully manual with nros::Executor::open(&ExecutorConfig::default()).

Key rules:

  • One Entry pkg per board target. Want to run the same nodes on native POSIX, on a QEMU-MPS2-AN385 FreeRTOS target, and on a real ThreadX board? Three Entry pkgs (robot_entry_native, robot_entry_qemu_freertos, robot_entry_acme_threadx) sharing the same Node pkgs and (usually) the same launch/system.launch.xml via symlink or <include>.
  • launch/system.launch.xml is the canonical name. nros plan resolution order: --file <path><dir>/launch/<pkg>.launch.xml<dir>/launch/system.launch.xml → the single <dir>/launch/*.launch.xml → synth (only for non-Entry, single-Node pkgs).
  • Deploy config lives in Cargo.toml. [package.metadata.nros.deploy.<target>] holds board / RMW / domain / locator per target; [[package.metadata.nros.domain]] and [[package.metadata.nros.bridge]] carry multi-domain topology.
  • C++ analogue: cmake fn nano_ros_entry(NAME <name> LANGUAGE CXX LAUNCH …) plus NROS_MAIN(...). Metadata flows through ${BUILD}/nros-metadata.json rather than a sidecar TOML.

Workspace shape

A typical multi-Node workspace, with one Entry pkg per supported board:

my_ws/
├── Cargo.toml          # [workspace] members = ["src/talker_pkg", "src/listener_pkg", "src/robot_entry"]
│                       # [workspace.metadata.nros] default_system = "demo_bringup"
└── src/
    ├── talker_pkg/         # Node pkg (lib, nros::node!)
    ├── listener_pkg/       # Node pkg
    ├── demo_bringup/       # Bringup pkg (declarative; no Cargo.toml)
    └── robot_entry/        # Entry pkg (bin, nros::main!(launch = "demo_bringup"))

cargo build at the workspace root builds everything via cargo’s native scheduler. nros plan reads [workspace.metadata.nros] default_system to pick the Entry pkg (or you pass nros plan robot_entry explicitly).

For C++-majority or mixed workspaces, CMake is the top-level driver instead — see the multi-node workspace design doc.

Single-Node-pkg convenience (cargo run Just Works)

For tiny fixtures and host-side dev loops, a Node pkg can declare itself as its own Entry pkg by adding [package.metadata.nros.entry] deploy = "<board>" to its Cargo.toml, alongside the usual [package.metadata.nros.node] and [package.metadata.nros.deploy.<target>] tables. src/main.rs collapses to one line:

// src/main.rs
nros::main!();

The macro reads deploy = "<board>" from this pkg’s own Cargo.toml, maps it to the right board crate, and emits fn main() + <this_pkg>::register(runtime)?; — the latter resolves through the companion src/lib.rs cargo auto-wires alongside the binary target. No build.rs, no launch file (one is synthesised in-memory), no hand-written boot glue. This is the L.7 self-entry planner path (Phase 212.L.7 + N.5 + N.9).

Limits of the convenience:

  • Native only. Embedded boards (FreeRTOS, ThreadX, Zephyr, bare-metal) still require a hand-written Entry pkg — board init is non-trivial enough that hiding it behind a one-liner does more harm than good.
  • One Node. Two or more Node pkgs in the same workspace = author an Entry pkg. The point of the convenience is to skip ceremony for tiny single-node fixtures, not to grow into a multi-node composition root.

Quick reference

You want…Use
Reusable node logic, board-independentNode pkg (nros::node!())
Per-board binary that runs N nodesEntry pkg (main.rs calls BoardEntry::run)
cargo run on host for a single-node fixtureNode pkg with [package.metadata.nros.entry] deploy = "native"
Same nodes on multiple boardsOne Node pkg set + one Entry pkg per board
Launch topology + per-target deploy configBringup pkg (declarative; optional, folds into Entry pkg when single-target)
Board hardware bringupBoard trait family (see porting chapter)

FreeRTOS (QEMU MPS2-AN385)

Single-node starter on FreeRTOS + lwIP, cross-compiled for Cortex-M3 and booted in QEMU MPS2-AN385. Slirp networking; no host TAP / bridge / sudo. Rust, C, and C++ talkers all live in-tree.

Prereqs. nros setup qemu-arm-freertos is the single command that prepares your machine for this board. It fetches a prebuilt toolchain set into the shared store at ~/.nros/sdk — the arm-none-eabi-gcc cross-compiler, the patched qemu-system-arm emulator, the FreeRTOS kernel + lwIP sources, and the RMW host daemon. You do not hand-install a cross-toolchain and you do not need a ROS 2 install.

Setup

Build the in-tree nros CLI (Phase 218):

source ./activate.sh        # OR: direnv allow / source ./activate.fish
just setup-cli              # builds packages/cli/target/release/nros

Then provision the board (--rmw defaults to zenoh; pick xrce or cyclonedds to match the example you intend to run):

nros setup qemu-arm-freertos --rmw zenoh

This fetches the cross-compiler, the patched qemu-system-arm, the FreeRTOS + lwIP sources, and the RMW host daemon (zenohd for zenoh, the Micro-XRCE-DDS agent for xrce) into ${NROS_HOME:-~/.nros}/sdk.

Project layout

Each language uses the standard nano-ros canonical example shape — standalone Cargo (Rust) or CMake (C / C++) project under examples/qemu-arm-freertos/<lang>/<example>/.

examples/qemu-arm-freertos/
├── rust/talker/             # Cargo package, cross-compile target = thumbv7m-none-eabi
│   ├── Cargo.toml
│   ├── .cargo/config.toml          # target + QEMU runner
│   ├── nros.toml                   # network + zenoh locator + scheduling
│   ├── package.xml
│   ├── generated/                  # codegen output — build.rs runs
│   │                               #   `nros generate-rust` on first
│   │                               #   `cargo build`; gitignored.
│   └── src/main.rs
├── c/talker/                 # CMake project, add_subdirectory consumption
│   ├── CMakeLists.txt
│   ├── nros.toml
│   ├── package.xml
│   └── src/main.c
└── cpp/talker/               # CMake C++14 project
    ├── CMakeLists.txt
    ├── nros.toml
    ├── package.xml
    └── src/main.cpp

The Rust Cargo.toml pulls the FreeRTOS board crate (nros-board-mps2-an385-freertos) which wraps the kernel + lwIP + LAN9118 driver build. The C / C++ CMakeLists.txt follows the canonical add_subdirectory(<repo-root>) + nano_ros_link_rmw(<target> RMW zenoh) pattern with NANO_ROS_BOARD = mps2-an385-freertos.

Configure

Network + Zenoh + scheduling live in nros.toml (parsed by the board crate at boot). Verbatim from the in-tree examples/qemu-arm-freertos/rust/talker/nros.toml:

# nano-ros config (direct mode). See
# docs/design/0004-configuration-and-transports.md.

[node]
domain_id = 0

[[transport]]
kind    = "ethernet"
ip      = "10.0.2.20/24"
mac     = "02:00:00:00:00:00"
gateway = "10.0.2.2"
locator = "tcp/10.0.2.2:7451"

[node.rt]
app_priority = 12
app_stack_bytes = 262144
zenoh_read_priority = 16
zenoh_read_stack_bytes = 5120
zenoh_lease_priority = 16
zenoh_lease_stack_bytes = 5120
poll_priority = 16
poll_interval_ms = 5

The 10.0.2.0/24 subnet is QEMU Slirp’s default; 10.0.2.2 is the Slirp gateway that forwards to host loopback. No TAP, no sudo.

Per-language test-fixture ports: Rust → 7451, C → 7551, C++ → 7651. The talker / listener under each <lang>/ uses the matching port; start zenohd on the one you intend to test against. The C / C++ example trees ship their own nros.toml with the matching port.

Build

# Rust:
cd examples/qemu-arm-freertos/rust/talker
cargo build --release

# C / C++ — use the cross-toolchain CMake invocation:
just freertos build-fixtures        # builds every in-tree zenoh +
                                    # DDS example across Rust / C / C++
# Or single-example (the `nros` CLI on PATH auto-resolves the codegen
# tool — no `-D_NANO_ROS_CODEGEN_TOOL=` needed):
toolchain="$(pwd)/cmake/toolchain/arm-freertos-armcm3.cmake"
cd examples/qemu-arm-freertos/c/talker
cmake -B build -DCMAKE_TOOLCHAIN_FILE="$toolchain" \
              -DCMAKE_BUILD_TYPE=Release
cmake --build build --parallel

First Rust build pulls + cross-compiles deps (~5 min). C / C++ build also compiles FreeRTOS kernel + lwIP — first run ~3 min.

Run

# 1. Start zenohd on the host (Slirp forwards 10.0.2.2:7451 → host:7451).
#    The just recipe wraps `zenohd --listen tcp/127.0.0.1:7451
#    --no-multicast-scouting` (the D.2 PATH shim resolves zenohd from
#    `~/.nros/sdk/zenohd/<v>/bin/zenohd`):
just freertos zenohd &
# Equivalent, if the recipe isn't available or you want the literal
# invocation (works as long as `zenohd` is on PATH):
zenohd --listen tcp/127.0.0.1:7451 --no-multicast-scouting &

# 2. Boot the talker in QEMU. The just recipe wraps qemu-system-arm
#    with the LAN9118 + Slirp wiring the example expects; works for
#    Rust as well (it builds + boots the in-tree binary):
just freertos talker
# Or, single-language, in the Rust example dir:
cd examples/qemu-arm-freertos/rust/talker
cargo run --release

# 3. Verify from stock ROS 2:
source /opt/ros/humble/setup.bash
export RMW_IMPLEMENTATION=rmw_zenoh_cpp
# Talker publishes best-effort; stock `ros2 topic echo` defaults to
# RELIABLE, so the QoS-mismatched echo silently delivers nothing.
# Force best-effort to receive:
ros2 topic echo /chatter std_msgs/msg/Int32 --qos-reliability best_effort

QEMU exits via Ctrl-A x.

For batch testing without manual QEMU launches: just freertos test runs every E2E (pub/sub, service, action) against a temporary in-test zenohd.

Readiness signal. Within ~20 seconds of QEMU boot, the talker should print Published: 0 on its semihosting stdout (the Rust talker pre-publishes 0 before the first counter bump). QEMU cold-boot through FreeRTOS init + lwIP DHCP + zenoh session open typically takes 10–15 s. If no Published: line in 30 seconds:

  1. Confirm zenohd is running on the host (Slirp forwards 10.0.2.2:7451 → host:7451). Without it the talker retries the zenoh handshake until QEMU is killed.
  2. Check the talker’s early log for lwIP DHCP timeout or Failed to open session.
  3. Bridge tip: ros2 topic echo /chatter from a stock ROS 2 install (with RMW_IMPLEMENTATION=rmw_zenoh_cpp) confirms end-to-end interop.
  4. See Troubleshooting — First 10 Minutes.

GitHub source

Canonical, copy-out:

Next

  • Subscriber: peer listener/ directory next to each talker.
  • Services + actions: peer service-*/ and action-*/ directories.
  • Real hardware: same code runs on STM32F4-Discovery / NXP-LPC55S69 / TI-MSP432 with a different board crate + linker script; see the Bare-metal Cortex-M3 page for the no-RTOS variant.
  • RTOS-specific debugging: FreeRTOS LAN9118 Debugging.

Zephyr (west module)

Single-node starter on Zephyr via the in-tree zephyr/ west module. nano-ros ships as a Zephyr module — west discovers it from your workspace’s west.yml, drops in a prj.conf Kconfig surface, and the standard west build / west flash flow takes care of the rest.

Contributor path? Building nano-ros’s own Zephyr examples straight from this repository (no west-managed workspace) is covered at Zephyr (contributor). The page below is the canonical user entry.

Just want a working starter? Clone the nano-ros-zephyr-example repo (west init -m …) — a manifest + zenoh talker app pinned to a tested Zephyr, with the same quickstart as below baked in. The steps here explain what it does so you can adapt it to your own workspace.

Prereqs. Run nros setup zephyr --rmw <rmw> once (see Prerequisites below) — it provisions the Zephyr west workspace + Zephyr SDK bits, the emulator, and your RMW’s host daemon into the shared store. No hand-installed Zephyr SDK, west, or cross-toolchain, and no ROS 2 install required. Python: 3.10+ on Zephyr 3.7 LTS, but ≥ 3.12 on Zephyr 4.x (4.x’s find_package(Python3) requires 3.12 — see the version matrix below). nano-ros’s imported west fragment zephyr/west.yml is a manifest-only file — it does NOT pull Zephyr itself; that has to be in your parent manifest (zephyrproject-rtos/zephyr).

Which Zephyr? — 3.7 LTS and 4.x both supported

nano-ros consumes as a module on both the 3.7 LTS line (supported to Jan 2027; the safety-island default) and the current 4.x rolling line. You build against whatever Zephyr your workspace already pins — nano-ros adapts. The two lines differ only in how you select an RMW and apply nano-ros’s Zephyr patches:

CapabilityZephyr 3.7 LTSZephyr 4.x
Min Python3.103.12 (find_package(Python3))
RMW selectionprj-<rmw>.conf overlay (-DCONF_FILE=...)-S nros-<rmw> snippet (or the overlay)
nano-ros patchesapplied during nros setup zephyr provisioningapplied during nros setup zephyr provisioning
Examples as samples / Twistersamples: + Twister (sample.nano-ros.*)
zenoh (native_sim)✅ build + e2e✅ build + e2e
cyclonedds (native_sim)✅ build + e2e✅ build · publish · receive · multicast-join (stable 2-node run pending a tracked k_mutex fix)
xrcebuild path WIP

native_sim networking uses NSOS (host loopback) on both lines — no TAP/bridge/root. The copy-out, snippet, patch-apply, and dual-line build flows are exercised in CI (just zephyr ci-both, just zephyr check-copy-out).

Project layout

A Zephyr workspace using nano-ros looks like any other Zephyr project — the nano-ros module sits beside Zephyr, your application sits beside both:

my_zephyr_ws/
├── .west/
├── zephyr/                            # cloned by `west init`
├── modules/
│   └── nano-ros/                      # imported via west.yml
└── apps/
    └── my_app/                        # your application
        ├── CMakeLists.txt
        ├── prj.conf                   # Kconfig — selects nros + RMW
        ├── west.yml                   # (optional) per-app manifest
        └── src/
            └── main.c                 # nros user code

The application CMakeLists.txt is a stock Zephyr app — find_package(Zephyr)

  • target_sources. No add_subdirectory(<nano-ros>) is needed; the module shell handles it once CONFIG_NROS=y flips on.

A minimal apps/my_app/ looks like this (mirrors examples/zephyr/c/talker/ with the names stripped to the essentials):

# apps/my_app/CMakeLists.txt
cmake_minimum_required(VERSION 3.20.0)
find_package(Zephyr REQUIRED HINTS $ENV{ZEPHYR_BASE})
project(my_app)
target_sources(app PRIVATE src/main.c)
// apps/my_app/src/main.c
#include <nros/nros.h>
#include <nros/publisher.h>
#include <std_msgs/msg/int32.h>

int main(void) {
    nros_init_args_t args = nros_init_args_default();
    nros_context_t ctx;
    nros_init(&args, &ctx);
    nros_node_t node;
    nros_create_node(&ctx, "my_app", &node);
    nros_publisher_t pub;
    nros_create_publisher(&node, std_msgs__msg__Int32_type_support(),
                          "/chatter", &pub);
    std_msgs__msg__Int32 msg = { .data = 0 };
    while (nros_ok(&ctx)) {
        nros_publish(&pub, &msg);
        msg.data++;
        k_sleep(K_SECONDS(1));
    }
    return 0;
}

prj.conf is the one shown in Configure below.

Prerequisites

nros setup provisions the parts nano-ros owns — the RMW host daemon (zenohd / Micro-XRCE-DDS agent) and the RMW transport submodules (zenoh-pico + mbedtls for zenoh, the cyclonedds fork) — from a pinned index into ${NROS_HOME:-~/.nros}/sdk / the nano-ros checkout. It does not replace Zephyr’s own SDK, and interface codegen still needs the ROS message definitions.

  1. Build the in-tree nros CLI (Phase 218, from the nano-ros checkout):
    source ./activate.sh        # OR: direnv allow / source ./activate.fish
    just setup-cli              # builds packages/cli/target/release/nros
    
  2. Provision the RMW (daemon + transports) from the nano-ros checkout:
    ( cd modules/nano-ros && nros setup zephyr --rmw zenoh )  # zenohd + zenoh-pico + mbedtls
    ( cd modules/nano-ros && nros setup --source px4-rs )     # workspace cargo-load dep
    
  3. Install the Zephyr SDK the standard Zephyr way (nros setup does not provide it) and expose it — export ZEPHYR_SDK_INSTALL_DIR=/path/to/zephyr-sdk-<ver> (or register it via the SDK’s setup.sh -c).
  4. Message definitions for codegen. The interface codegen resolves a package’s msg/*.msg from NROS_<PKG>_DIR (e.g. export NROS_STD_MSGS_DIR=/opt/ros/humble/share/std_msgs) — point it at a ROS install or any dir holding the .msg files.

The RMW host daemon must be running before an example connects (zenohd -l tcp/127.0.0.1:7456 for zenoh; the Micro-XRCE-DDS agent for xrce).

Configure

Add nano-ros (and a Zephyr pin — zephyr/west.yml is manifest-only; it does not pull Zephyr itself) to your workspace west.yml:

manifest:
  remotes:
    - name: nano-ros
      url-base: https://github.com/NEWSLabNTU
    - name: zephyr
      url-base: https://github.com/zephyrproject-rtos
  projects:
    - name: zephyr
      remote: zephyr
      revision: v3.7.0          # or your chosen 3.7 LTS / 4.x SHA
      path: zephyr
      import: true               # pulls Zephyr's own modules
    - name: nano-ros
      remote: nano-ros
      revision: main             # required — repo's default branch is
                                 # `main`; west defaults to `master`
                                 # otherwise and the fetch fails.
      path: modules/nano-ros
      import:
        file: zephyr/west.yml    # pulls nano-ros's transport deps

Then per-application prj.conf:

CONFIG_NROS=y
CONFIG_NROS_C_API=y
CONFIG_NROS_RMW_ZENOH=y                 # bool per RMW: NROS_RMW_{ZENOH,XRCE,CYCLONEDDS}
# (ROS edition is a build-time Cargo feature, NOT a Kconfig symbol — do not set
#  CONFIG_NROS_ROS_EDITION; Zephyr aborts on the undefined symbol.)

# Required for any networked RMW on QEMU / native_sim:
CONFIG_NETWORKING=y
CONFIG_NET_IPV4=y
CONFIG_NET_TCP=y

CONFIG_NROS=y activates the shell, which maps Kconfig values to NANO_ROS_* CMake cache vars and add_subdirectory()s the root nano-ros CMake. NanoRos::NanoRos is linked into your app library transparently.

Build

If your workspace is a fresh manifest-only dir (no .west/), initialise it first so west knows which west.yml is the manifest:

cd my_zephyr_ws
west init -l .                           # one-time; points west at the
                                         # local west.yml in cwd
west update                              # clones nano-ros + Zephyr into the workspace

(If you started from west init -m <remote>, both calls above are already done — go straight to west build below.)

The transports + px4-rs come from the prerequisites step (west update clones nano-ros but not its submodules). With the Zephyr SDK + NROS_STD_MSGS_DIR exported (also prerequisites), build your app — nros on PATH is auto-resolved as the codegen tool:

# native_sim (POSIX, no QEMU). The 3.7 line needs the NSOS line overlay; apply
# the NSOS patches first (see "apply nano-ros's patches" below).
overlay="$PWD/modules/nano-ros/cmake/zephyr/native-sim-line-3.7.conf"
west build -b native_sim/native/64 apps/my_app -- -DCONF_FILE="prj.conf;$overlay"

# A real board, e.g. Cortex-A9 (no native_sim overlay):
west build -b qemu_cortex_a9 apps/my_app

(Verified end-to-end on a fresh BYO west workspace: this builds to zephyr.exe and runs to Published: 0 against zenohd -l tcp/127.0.0.1:7456. On the 4.4 line, find_package(Python3) requires ≥ 3.12 and you select the RMW with -S nros-zenoh instead of the overlay.)

For a quick sanity check that the module is wired correctly:

west build -t menuconfig                 # confirm CONFIG_NROS=y is visible

Rust applications

Two things differ for a Rust app (C/C++ apps skip this section):

  1. The Rust crate’s [lib] must be named rustapp (crate-type = ["staticlib"]) — a zephyr-lang-rust contract: its rust_cargo_application() links librustapp.a. The Cargo package name is free.

  2. Generate the interface crates + the [patch.crates-io] wiring for YOUR layout — do not copy an in-repo example’s .cargo/config.toml: its ../../../../packages/core/... paths are repo-relative and break in a copied-out app. From your app dir, run (after the Prerequisites nros setup, which provides the codegen toolchain + message sources):

    nros generate-rust --generate-config \
        --nano-ros-path "$PWD/../../modules/nano-ros/packages/core"
    

    This writes generated/<pkg>/ (the message crates) and a .cargo/config.toml whose [patch.crates-io] points the nros-* crates at your modules/nano-ros/packages/core/* and the generated interfaces at generated/*. Adjust --nano-ros-path to your workspace’s modules/nano-ros/packages/core (the dir holding nros-core, nros-node, …). The example apps’ committed .cargo/config.toml is for the in-tree build only.

Run

# 1. Start zenohd on the host. The in-tree just recipe runs the
#    pinned in-tree zenohd on the zephyr fixture port (7456) — the
#    same port the example apps' `nros.toml` / Kconfig defaults pick
#    up:
just zephyr zenohd &
#    Or directly:
#    zenohd --listen tcp/0.0.0.0:7456 --no-multicast-scouting

# 2. Boot the app. nano-ros's own in-tree zephyr talker has a
#    matching just recipe for the canonical `native_sim` build path:
just zephyr talker
#    For a BYO west workspace + your own app:
# QEMU Cortex-A9:
west build -t run
# native_sim:
./build/zephyr/zephyr.exe

# 3. Verify from stock ROS 2 in another terminal:
source /opt/ros/humble/setup.bash
export RMW_IMPLEMENTATION=rmw_zenoh_cpp
# Talker publishes best-effort; stock `ros2 topic echo` defaults to
# RELIABLE, so the QoS-mismatched echo silently delivers nothing.
# Force best-effort to receive:
ros2 topic echo /chatter std_msgs/msg/Int32 --qos-reliability best_effort

The Zephyr boot banner runs first, then nano-ros prints Published: 0, Published: 1, … as the talker fires — Rust + C + C++ all start at 0 (Phase 208.D.9).

Readiness signal. On native_sim, expect Published: 0 within 5 seconds of just zephyr talker (or ./build/zephyr/zephyr.exe); on qemu_cortex_a9 expect it within ~15 seconds (QEMU cold boot + Zephyr init). If no Published: line in 30 seconds:

  1. Confirm CONFIG_NROS=y lit up via west build -t menuconfig; without it the module shell never add_subdirectory’s nano-ros.
  2. Check CONFIG_NETWORKING=y, CONFIG_NET_IPV4=y, CONFIG_NET_TCP=y in prj.conf — Zephyr networking is opt-in.
  3. Confirm zenohd reachable from the simulated network (Slirp needs 10.0.2.2:7456 on QEMU; native_sim uses host loopback).
  4. See Troubleshooting — First 10 Minutes.

Zephyr 4.x build gotchas.

  • Could NOT find Python3 ... required is at least "3.12" — 4.x needs Python ≥ 3.12. Provision one without sudo (e.g. uv venv --python 3.12 .venv312 && uv pip install west -r zephyr/scripts/requirements.txt) and run west through it (.venv312/bin/python -m west build ...), so the ROS descriptor-codegen subprocess still uses the system ROS Python.
  • attempt to assign the value ... to the undefined symbol ETH_NATIVE_POSIX — that symbol was renamed ETH_NATIVE_TAP in 4.x; the version-aware NSOS overlay handles it (just zephyr build-one does this automatically).

Zephyr 4.x: select the RMW with a snippet

On 4.x, nano-ros ships west snippets so you pick the RMW on the build line instead of hand-writing the overlay:

west build -b native_sim/native/64 -S nros-cyclonedds apps/my_app
#                                   ^^^^^^^^^^^^^^^^^^^  nros-zenoh | nros-cyclonedds | nros-xrce

The snippet (shipped via the module’s snippet_root) carries the RMW-common Kconfig — equivalent to merging prj-<rmw>.conf. The prj-<rmw>.conf / -DCONF_FILE path still works (and is the only option on 3.7).

nano-ros patches into your workspace

nano-ros needs a few patches to Zephyr’s native_sim NSOS driver (recvmsg, IPv4-multicast) so the host-loopback-based examples work. Both supported lines (3.7 LTS and 4.x) take the same pathnros setup zephyr reads zephyr/patches.yml and applies each patch against the workspace’s Zephyr tree, sha256-checked. No extra step on your side beyond the provisioning command (already run during Prerequisites).

To re-apply (e.g. after a west update reset the tree):

nros setup zephyr --rmw zenoh

(Earlier nano-ros revisions documented a west patch apply flow on 4.x — that required a workspace-side west extension that doesn’t ship with stock Zephyr. Phase 208.D.7 / E.9 unified both lines on the provisioner. Cyclone-DDS-on-Zephyr patches stay baked into the pinned cyclonedds submodule, not delivered through patches.yml.)

Copy out an example as your starting point

The examples/zephyr/<lang>/<role>/ dirs are copy-out clean — copy one into your own app tree and it builds against the nano-ros module with no reference back into the nano-ros repo:

cp -r modules/nano-ros/examples/zephyr/c/talker apps/my_app
# cyclonedds examples need the host idlc + the ROS message dirs:
export NROS_STD_MSGS_DIR=/opt/ros/humble/share/std_msgs   # PKG_DIR contract
west build -b native_sim/native/64 -S nros-zenoh apps/my_app

Cyclone idlc and the descriptor-gen scripts are located via the module’s exported cache vars (NROS_CYCLONE_IDLC, NROS_CYCLONE_SCRIPTS_DIR, NROS_CYCLONE_CMAKE_DIR); message-package dirs come from NROS_<PKG>_DIR env (defaulting to /opt/ros/humble/share/<pkg>). No /opt/ros or repo-relative paths are baked into the example.

See the zephyr/ module dir + its Kconfig for the canonical in-repo surface.

GitHub source

Next

  • Pick a real board (Nordic, NXP, STM32, …): swap -b <board> and add a board-specific overlay to your prj.conf.
  • Cyclone DDS on Cortex-A/R: see the DDS section of Choosing an RMW Backend for the required Kconfig deltas.
  • Build nano-ros’s own Zephyr examples without west: Zephyr (contributor).

NuttX (apps/external)

Single-node starter on NuttX. nano-ros plugs into the standard NuttX app discovery as an external app under apps/external/nano-ros/, exposing Kconfig knobs under Application Configuration → External Modules → nano-ros. Use this entry when your NuttX board ships its own kernel build and you want to add ROS 2 communication.

Contributor path? Building nano-ros’s own NuttX QEMU examples straight from this repository (no NuttX-managed workspace) is covered at NuttX (contributor). The page below is the canonical user entry.

Prereqs. Install the nros CLI once per machine, then run nros setup qemu-arm-nuttx --rmw <zenoh|xrce|cyclonedds> (--rmw defaults to zenoh). This fetches a prebuilt toolchain set into ${NROS_HOME:-~/.nros}/sdk — the NuttX cross-compiler, the emulator, the NuttX sources, and the RMW host daemon — so you do not hand-install a cross-toolchain and do not need a ROS 2 install:

source ./activate.sh        # OR: direnv allow / source ./activate.fish
just setup-cli              # builds packages/cli/target/release/nros (Phase 218)
nros setup qemu-arm-nuttx --rmw zenoh

You still need a NuttX ≥ nuttx-12 checkout with an apps/ sibling and Python 3.10+ for the NuttX configure scripts.

Project layout

NuttX’s “external apps” pattern places the app shim under $NUTTX_APPS_DIR/external/<name>/:

$NUTTX_DIR/                              # NuttX kernel checkout
$NUTTX_APPS_DIR/                             # sibling: apps tree
└── external/
    └── nano-ros/                        # symlink or submodule of
        ├── Make.defs                    #   integrations/nuttx/
        ├── Makefile
        ├── CMakeLists.txt               #   (cmake-driven NuttX builds)
        └── Kconfig
my_app/                                  # your application
├── package.xml
├── Cargo.toml | CMakeLists.txt
├── generated/                           # Rust codegen — build.rs runs
│                                        #   `nros generate-rust` on first
│                                        #   `cargo build`; gitignored.
└── src/main.{rs,c,cpp}

Wire the shell into your NuttX apps tree. Easiest path:

just nuttx setup        # contributor helper: stages the shell +
                        # example apps into $NUTTX_APPS_DIR/external/
                        # (delegates to `nros setup qemu-arm-nuttx`
                        # for the toolchain/SDK provisioning)

This runs scripts/nuttx/stage-external-apps.sh, which writes $NUTTX_APPS_DIR/external/Make.defs + Kconfig and symlinks the integration shell (external/nano-ros) plus every example app (external/nano-ros-<example>-<lang>). Menuconfig surfaces them under Application Configuration → External Modules.

If you’d rather wire it yourself (e.g. into a vendored apps tree):

ln -s /path/to/nano-ros/integrations/nuttx \
      $NUTTX_APPS_DIR/external/nano-ros
# then copy integrations/nuttx/external-Make.defs.in →
# $NUTTX_APPS_DIR/external/Make.defs and add a matching
# $NUTTX_APPS_DIR/external/Kconfig that `source`s the shell.

Configure

NuttX uses Kconfig as its single source of truth. After the symlink above:

cd $NUTTX_DIR
make menuconfig
# Navigate to:
#   Application Configuration → External Modules → nano-ros
#   [*] nano-ros ROS 2 client
#       RMW backend  →  zenoh           # zenoh | xrce | cyclonedds
#       ROS 2 edition →  humble

Networking Kconfig requirements live under Networking Support — enable CONFIG_NET, CONFIG_NET_TCP, CONFIG_NET_IPv4. For QEMU nsh_smp configurations the defaults already include these.

Runtime config (locator / domain id) is read from the companion nros.toml next to the example source. Verbatim from the in-tree examples/qemu-arm-nuttx/rust/talker/nros.toml (this is the file just nuttx talker consumes — port 7452 matches just nuttx zenohd):

# nano-ros config (direct mode). See
# docs/design/0004-configuration-and-transports.md.

[node]
domain_id = 0

[[transport]]
kind    = "ethernet"
ip      = "10.0.2.30/24"
gateway = "10.0.2.2"
locator = "tcp/10.0.2.2:7452"

The C + C++ variants ship analogous files with distinct ports (7552 and 7652) so parallel test runs don’t collide on one router; just nuttx zenohd binds 7452 (Rust). When you boot a C or C++ talker directly, either edit the locator line of its nros.toml

[[transport]]
kind    = "ethernet"
ip      = "10.0.2.30/24"
gateway = "10.0.2.2"
locator = "tcp/10.0.2.2:7452"   # was 7552 / 7652 — match `just nuttx zenohd`

— or start a sibling zenohd on the matching port (zenohd --listen tcp/127.0.0.1:7552 --no-multicast-scouting for C, …:7652 for C++).

Build

cd $NUTTX_DIR
make                                # full kernel + apps build

The Cargo build of nano-ros’s Rust staticlibs runs as a sub-step of the NuttX app build; libnros_c.a is linked at the final app link stage.

For CMake-driven NuttX builds:

cmake -B build -DBOARD=qemu-armv7a \
              -DCONFIG=nsh_smp
cmake --build build

Run

# 1. Start zenohd on the host (Slirp forwards 10.0.2.2:7452 → host).
#    The in-tree just recipe runs the daemon on the nuttx fixture
#    port (7452):
just nuttx zenohd &

# 2. QEMU NuttX (ARM). For nano-ros's own in-tree QEMU examples the
#    just recipe wraps qemu-system-arm with the right wiring. `talker`
#    here is the Rust variant; the C / C++ variants boot through the
#    `make`-driven path described under "Auto-configure glue" below:
just nuttx talker
#    For a NuttX-managed workspace where you've staged the
#    integration shell + your own app, mirror the recipe's actual
#    flags (see `just/nuttx.just::_run-qemu`):
qemu-system-arm -M virt -cpu cortex-a7 -nographic \
                -icount shift=auto \
                -kernel $NUTTX_DIR/nuttx \
                -netdev user,id=net0 \
                -device virtio-net-device,netdev=net0
# `$NUTTX_DIR/nuttx` is the linked NuttX ELF produced by `make`
# at the NuttX source root — adjust if your workspace puts it
# elsewhere (e.g. an out-of-tree build dir).
# At the NSH prompt, run the example's PROGNAME — the
# `make`-driven build registers every nano-ros example as a
# built-in command via Application.mk's `-Dmain=<PROGNAME>_main`
# rename. Real PROGNAMEs (from
# `packages/testing/nros-tests/tests/nuttx_make_e2e.rs::EXPECTED_PROGNAMES`):
nsh> nuttx_c_talker            # C talker
# nsh> nuttx_cpp_talker        # C++ talker
# nsh> nuttx_c_listener        # ...and listener / service / action variants

# Real hardware: standard NuttX flash flow (openocd / J-Link / etc.)

# 3. Verify from stock ROS 2 in another terminal:
source /opt/ros/humble/setup.bash
export RMW_IMPLEMENTATION=rmw_zenoh_cpp
# Talker publishes best-effort; stock `ros2 topic echo` defaults to
# RELIABLE, so the QoS-mismatched echo silently delivers nothing.
# Force best-effort to receive:
ros2 topic echo /chatter std_msgs/msg/Int32 --qos-reliability best_effort

Readiness signal. After typing the app’s NSH command (e.g. nuttx_c_talker), expect Published: 0 on the NSH console within 5 seconds — Rust + C + C++ all start the counter at 0 (Phase 208.D.9). If no Published: line:

  1. Confirm the app actually ran — ps should show your task.
  2. Confirm networking — ifconfig shows a configured interface. With the virtio-net + Slirp wiring above, eth0 comes up at 10.0.2.30 (matches examples/qemu-arm-nuttx/*/talker/nros.toml).
  3. Confirm zenohd reachable; the locator in nros.toml / nros_init arguments must match.
  4. See Troubleshooting — First 10 Minutes.

Auto-configure glue (NSH built-in registration)

The make-driven build above relies on a host-side glue layer that the in-tree just nuttx build-fixtures-make recipe owns end-to-end (see just/nuttx.just::build-fixtures-make). If you wire a NuttX workspace by hand, reproduce the same three steps:

  1. Swap in the nano-ros board defconfig. Stock NuttX qemu-armv7a/nsh ships without CONFIG_NET=y, virtio-net, or TLS_NELEM. The board defconfig packages/boards/nros-board-nuttx-qemu-arm/nuttx-config/defconfig already carries the full networking + TLS stack zenoh-pico needs; copy it to $NUTTX_DIR/.config and run make olddefconfig.
  2. Stage the integration shell + example apps. Run scripts/nuttx/stage-external-apps.sh "$NUTTX_APPS_DIR" to symlink integrations/nuttx/ and every example app into $NUTTX_APPS_DIR/external/. Remove $NUTTX_APPS_DIR/Kconfig so NuttX’s mkkconfig.sh rediscovers the new apps/external/Kconfig.
  3. Flip the nano-ros Kconfig knobs via kconfig-tweak. The recipe enables NROS, NROS_C_API, NROS_CPP_API, every NROS_EXAMPLE_<EX>_<LANG>, sets TLS_NELEM=8, disables LIBCXXNONE + enables LIBCXXTOOLCHAIN, and disables ALLSYMS for the bootstrap link. Re-run make olddefconfig so the newly-visible dependencies settle, then make.

kconfig-tweak ships in the kconfig-frontends package on most distros. Without it the recipe skips ("NuttX skip: kconfig-tweak not on PATH"); install it before retrying.

GitHub source

Next

  • Multiple apps: each app declares its own progname in Application Configuration → External Modules; they share the one libnros_c.a build via the external-app shell.
  • DDS on NuttX: bump the netbuffer Kconfig knobs (similar to the Zephyr DDS profile under Choosing an RMW Backend).
  • Build nano-ros’s own NuttX QEMU tests without a NuttX-managed workspace: NuttX (contributor).

ThreadX (Linux sim / RISC-V64 QEMU)

Single-node starter on Microsoft Azure RTOS ThreadX + NetX Duo (BSD socket layer). Two flavours ship in-tree:

  • threadx-linux — ThreadX user-space simulator on Linux. Fast build, host network stack, ideal for development.
  • threadx-riscv64 — QEMU virt machine with the RISC-V64 GCC toolchain. Full kernel + NetX Duo TCP/IP stack.

Rust, C, and C++ are supported on both flavours — just <flavour> build-fixtures produces threadx_cpp_* and riscv64_threadx_cpp_* binaries alongside the Rust + C ones. See the coverage matrix for the per-RMW cell status.

Prereqs. Install the nros CLI once, then run nros setup <board> --rmw <rmw> for the flavour you need (see Setup). It provisions the cross-compiler, emulator, RMW host daemon, and ThreadX/NetX sources — no hand-installed riscv64 cross toolchain, qemu-system-riscv64, or ROS 2 required.

Setup

nros setup is the single canonical command to prepare a machine to build nano-ros for a board. It ships prebuilt toolchains per platform per RMW — the cross-compiler, emulator, RMW host daemon, and SDK sources (the ThreadX/NetX sources, and for threadx-linux the POSIX-sim sources) are fetched from a pinned index into a shared store at ${NROS_HOME:-~/.nros}/sdk. You do not need ROS 2 installed.

Build the in-tree nros CLI (Phase 218):

source ./activate.sh        # OR: direnv allow / source ./activate.fish
just setup-cli              # builds packages/cli/target/release/nros

Provision the ThreadX flavour you need (+ the RMW):

nros setup threadx-linux --rmw zenoh          # POSIX-sim flavour; --rmw defaults to zenoh
nros setup qemu-riscv64-threadx --rmw zenoh   # only if you need the RISC-V64 QEMU flow
source ./setup.bash

The RMW host daemon must be running before any example: zenohd for zenoh, the Micro-XRCE-DDS agent for xrce. nros setup … --rmw <rmw> installs it.

Project layout

Each example is a standalone Cargo or CMake project under examples/threadx-{linux,riscv64}/<lang>/<example>/:

examples/threadx-linux/
├── rust/talker/                 # Cargo, target = x86_64-unknown-linux-gnu
│   ├── Cargo.toml
│   ├── package.xml
│   ├── generated/                # codegen output — build.rs runs
│   │                             #   `nros generate-rust` on first
│   │                             #   `cargo build`; gitignored.
│   └── src/main.rs
└── c/talker/                    # CMake, add_subdirectory
    ├── CMakeLists.txt
    ├── package.xml
    └── src/main.c

examples/qemu-riscv64-threadx/
├── rust/talker/                 # Cargo, target = riscv64gc-unknown-linux-gnu
│   └── ...
└── c/talker/
    └── ...

ThreadX-linux runs as a regular host process — no QEMU. NetX Duo uses the nx_bsd_* BSD socket shim layered on the host TCP stack (threadx-linux variant) or on its own NetX Duo TCP/IP stack (riscv64 variant).

Configure

Each talker carries a per-flavour nros.toml. Both files are reproduced verbatim below.

threadx-linux — examples/threadx-linux/rust/talker/nros.toml:

# nano-ros config (direct mode). See
# docs/design/0004-configuration-and-transports.md.

[node]
domain_id = 0

[[transport]]
kind    = "ethernet"
ip      = "192.0.3.10/24"
mac     = "02:00:00:00:00:00"
gateway = "192.0.3.1"
interface = "tap-tx0"
locator = "tcp/127.0.0.1:7455"

threadx-riscv64 — examples/qemu-riscv64-threadx/c/talker/nros.toml:

# nano-ros config (direct mode). See
# docs/design/0004-configuration-and-transports.md.

[node]
domain_id = 0

[[transport]]
kind    = "ethernet"
ip      = "10.0.2.40/24"
mac     = "52:54:00:12:34:56"
gateway = "10.0.2.2"
locator = "tcp/10.0.2.2:7553"

ThreadX-Linux normally uses a veth pair (tap-tx0) for an isolated host link, but nros setup threadx-linux does not create the interface — the test fixtures fall back to a loopback path when tap-tx0 is absent, which is fine for the happy-path tutorial. Bring up tap-tx0 by hand (ip link add … type veth …) only when you need real-network bridging. The QEMU-RISC-V64 fixture uses Slirp’s default 10.0.2.2 gateway just like the FreeRTOS QEMU flow.

Build

# threadx-linux:
just threadx_linux build-fixtures   # build all rust + c examples

# Single example:
cd examples/threadx-linux/rust/talker
cargo build --release

# threadx-riscv64:
just threadx_riscv64 build-fixtures

First setup builds ThreadX + NetX Duo (~3 min). Subsequent example builds finish in seconds.

Run

# threadx-linux (no QEMU). Step 1 brings up the in-tree zenohd on
# the threadx-linux port (7455). Step 2 runs the talker via the
# matching just recipe — same binary the example dir builds.
just threadx_linux zenohd &
just threadx_linux talker
# Expected (per src/main.rs structured logs):
#   Declaring publisher on /chatter (std_msgs/Int32)
#   Publisher declared
#   Published: 0
#   Published: 1
#   ...

# threadx-riscv64 (QEMU virt). Same shape — zenohd on 7453 first,
# then the talker recipe boots `qemu-system-riscv64` with the
# virtio-net + Slirp wiring baked in:
just threadx_riscv64 zenohd &
just threadx_riscv64 talker

# Verify from stock ROS 2:
source /opt/ros/humble/setup.bash
export RMW_IMPLEMENTATION=rmw_zenoh_cpp
# Talker publishes best-effort; stock `ros2 topic echo` defaults to
# RELIABLE, so the QoS-mismatched echo silently delivers nothing.
# Force best-effort to receive:
ros2 topic echo /chatter std_msgs/msg/Int32 --qos-reliability best_effort

For batch testing: just threadx_linux test runs every pubsub / service / action against an in-test zenohd.

Readiness signal. threadx-linux: Published: 0 within 3 seconds of just threadx_linux talker on a warm cache; a cold first run rebuilds the Rust example (~80 s on a fresh checkout) before the first publish lands. threadx-riscv64 (QEMU): within ~15 seconds of QEMU boot. If no Published: line:

  1. Confirm zenohd reachable on the locator from nros.toml (threadx-linux uses 127.0.0.1; riscv64 QEMU uses 10.0.2.2).
  2. threadx-linux: confirm the veth bridge came up via nros setup threadx-linux.
  3. See Troubleshooting — First 10 Minutes.

GitHub source

Next

  • Subscriber + service + action peers in the same example tree.
  • DDS on ThreadX: Cyclone DDS is the surviving DDS backend (nros-rmw-cyclonedds, selected via -DNANO_ROS_RMW=cyclonedds); see Choosing an RMW Backend.
  • Real hardware: same code runs against ThreadX vendor BSPs (Renesas Synergy, MIMXRT, etc.); replace the QEMU board crate with a vendor board crate.

ESP32 (esp-hal, bare-metal Rust)

Single-node starter on ESP32-C3 using the bare-metal esp-hal Rust path — no ESP-IDF — running under the Espressif QEMU fork (OpenETH ethernet). For the ESP-IDF component path (C / C++ apps), see ESP32 (ESP-IDF component).

Prereqs. nros setup esp32 is the single command that prepares your machine. It fetches a prebuilt esp-hal toolchain and the chosen RMW host daemon from a pinned index into the shared store at ~/.nros/sdk — you do not hand-install cross-compilers, and you do not need ROS 2 installed.

Setup

Build the in-tree nros CLI (Phase 218):

source ./activate.sh        # OR: direnv allow / source ./activate.fish
just setup-cli              # builds packages/cli/target/release/nros

Provision the board (and RMW):

nros setup esp32 --rmw zenoh     # --rmw defaults to zenoh; xrce | cyclonedds also valid

This pulls the SDK sources nano-ros owns (zenoh-pico + mbedtls submodules for zenoh; analogous for xrce / cyclonedds) and lands the RMW host daemon (zenohd for zenoh, the Micro-XRCE-DDS agent for xrce) under ${NROS_HOME:-~/.nros}/sdk (the activate file puts the in-tree CLI on PATH; legacy ${NROS_HOME:-~/.nros}/bin/ install remains supported transitionally). esp-hal itself is a Cargo dependency the example pulls in at build time, not a separately-installed toolchain; the only cross-toolchain you may need to add by hand is the rustup target — once per host:

rustup target add riscv32imc-unknown-none-elf      # ESP32-C3

Project layout

Each example is a standalone Cargo package targeting riscv32imc-unknown-none-elf (ESP32-C3). The board crate (nros-board-esp32-qemu) wraps the OpenETH / esp-hal init.

ESP32-S3 (Xtensa) is NOT supported today. The tutorial targets the RISC-V ESP32-C3 only. Xtensa targets do not ship via rustup (they require the espup toolchain installer) and the in-tree board crate is RISC-V only; this gap is tracked separately.

examples/qemu-esp32-baremetal/rust/talker/
├── Cargo.toml
├── .cargo/config.toml         # target = riscv32imc-unknown-none-elf
├── nros.toml                  # ethernet transport + zenoh locator
├── package.xml
├── generated/                 # codegen output — build.rs runs
│                              #   `nros generate-rust` on first
│                              #   `cargo build`; gitignored.
└── src/main.rs                # esp-hal init → nros_app_main

Configure

nros.toml carries the transport stack. The QEMU ESP32 board uses an ethernet transport via nros-board-esp32-qemu. Verbatim from examples/qemu-esp32-baremetal/rust/talker/nros.toml:

# nano-ros config (direct mode). See
# docs/design/0004-configuration-and-transports.md.

[node]
domain_id = 0

[[transport]]
kind    = "ethernet"
ip      = "10.0.2.50/24"
mac     = "02:00:00:00:00:01"
gateway = "10.0.2.2"
locator = "tcp/10.0.2.2:7454"

Build

# QEMU ESP32 (qemu-system-riscv32). `just esp32 build-qemu` (which
# `just esp32 talker` depends on) builds the QEMU-board variant; the
# example's build.rs invokes `nros generate-rust` automatically, so
# the `generated/` dir populates on first build (gitignored).
just esp32 build-qemu

Run

# QEMU ESP32. First bring up zenohd on the esp32 fixture port (7454):
just esp32 zenohd &
# Then boot the talker binary in qemu-system-riscv32 (esp32c3):
just esp32 talker
# Expected serial output (per src/main.rs):
#   Declaring publisher on /chatter (std_msgs/Int32)
#   Publisher declared
#   Published: 0
#   Published: 1
#   ...

# Verify from stock ROS 2 on the same network:
source /opt/ros/humble/setup.bash
export RMW_IMPLEMENTATION=rmw_zenoh_cpp
# Talker publishes best-effort; stock `ros2 topic echo` defaults to
# RELIABLE, so the QoS-mismatched echo silently delivers nothing.
# Force best-effort to receive:
ros2 topic echo /chatter std_msgs/msg/Int32 --qos-reliability best_effort

Readiness signal. QEMU ESP32: ~15 seconds after a warm cache — the just esp32 talker recipe re-runs build-qemu every invocation, so a first / cold run adds ~25 s of build time on top. If no Published: line:

  1. Wrong locator → talker logs zenoh open failed and retries. Confirm zenohd is reachable on the host IP (10.0.2.2:7454).
  2. Confirm .cargo/config.toml target is riscv32imc-unknown-none-elf (ESP32-C3). The tutorial does not support ESP32-S3 (Xtensa) yet.
  3. See Troubleshooting — First 10 Minutes.

GitHub source

Next

  • Subscriber + service + action peer directories under the same examples/qemu-esp32-baremetal/rust/.
  • ESP-IDF component path for C / C++ apps: ESP32 (ESP-IDF component).
  • ESP32-S3 (Xtensa) — not supported today. The Xtensa toolchain does not ship via rustup (it requires espup), and there is no in-tree Xtensa board crate. Stick with ESP32-C3 (RISC-V) for now.

ESP32 (ESP-IDF component)

Single-node starter on ESP32-family chips via the ESP-IDF component path — Espressif’s native C / C++ build system. For the bare-metal Rust (esp-hal) path, see ESP32 (esp-hal).

Prereqs. Two independent toolchains.

  1. ESP-IDF itself — ≥ 5.1, installed through Espressif’s own installer so idf.py is on PATH (source $IDF_PATH/export.sh). nros setup does not replace this; the IDF toolchain comes from idf.py install / Espressif’s tooling.

  2. The nano-ros side — the RMW host daemon (and any nano-ros host tools you use for testing) come from the nros CLI:

    source ./activate.sh        # OR: direnv allow / source ./activate.fish
    just setup-cli              # builds packages/cli/target/release/nros (Phase 218)
    nros setup esp32 --rmw zenoh     # lands the RMW host daemon
                                     # (zenohd for zenoh, the
                                     # Micro-XRCE-DDS agent for xrce)
                                     # in ${NROS_HOME:-~/.nros}/sdk, AND clones the
                                     # transport submodules
                                     # (zenoh-pico + mbedtls for zenoh)
                                     # into the nano-ros checkout
                                     # so the IDF build can compile
                                     # them in-tree.
    

Project layout

ESP-IDF apps are CMake projects with idf.py as the orchestrator. nano-ros plugs in as a component pulled by IDF’s component manager or by a local path during development.

my_idf_app/
├── CMakeLists.txt                 # top-level: `project(my_app)`
├── sdkconfig                      # IDF Kconfig (generated)
├── main/
│   ├── CMakeLists.txt             # `idf_component_register(REQUIRES nano-ros …)`
│   ├── idf_component.yml          # declares nano-ros as a managed dependency
│   ├── app_main.c | app_main.cpp
│   └── nros.toml                  # (optional) runtime locator + domain
└── components/                    # (optional) local components override

The idf_component.yml is the dependency manifest:

dependencies:
  nano-ros:
    # During development — local path to your nano-ros clone:
    path: ../../../nano-ros/integrations/nano-ros
    # Once published to the Espressif Component Registry:
    # version: "*"

The shell at integrations/nano-ros/ wraps the nano-ros root CMake into a standard IDF component, mapping IDF Kconfig knobs to NANO_ROS_* cache vars.

Configure

After idf.py menuconfig:

Component config → nano-ros
    RMW backend          (zenoh)        zenoh | xrce | cyclonedds
    ROS 2 edition        (humble)       humble | iron

The nano-ros component itself exposes only those two knobs. Wi-Fi credentials + zenoh locator are NOT in this Kconfig — provide them via your app’s own Kconfig.projbuild (Espressif’s standard pattern) or via environment variables, then pass them to nros::init(locator, domain_id) at startup.

Build

cd my_idf_app
idf.py set-target esp32c3        # or esp32s3, esp32, esp32c6
idf.py build

First build cross-compiles nano-ros’s Rust staticlibs + IDF components (~5 min). Re-builds finish in seconds.

Run

# Flash + monitor:
idf.py -p /dev/ttyUSB0 flash monitor
# Expected serial output:
#   I (1234) nano-ros: Wi-Fi connected
#   I (1456) nano-ros: zenoh session opened
#   I (1567) nano-ros: Published: 0

# Verify from stock ROS 2 on the same network:
source /opt/ros/humble/setup.bash
export RMW_IMPLEMENTATION=rmw_zenoh_cpp
# Talker publishes best-effort; stock `ros2 topic echo` defaults to
# RELIABLE, so the QoS-mismatched echo silently delivers nothing.
# Force best-effort to receive:
ros2 topic echo /chatter std_msgs/msg/Int32 --qos-reliability best_effort

QEMU ESP32 testing path: see the just esp_idf recipes — they boot the IDF binary in qemu-system-xtensa via Espressif’s patched QEMU.

Readiness signal. After idf.py flash monitor, expect I (XXXX) nano-ros: Wi-Fi connected followed by I (XXXX) nano-ros: Published: 0 within 10 seconds — Rust + C + C++ all start the counter at 0 (Phase 208.D.9). If no Published: line:

  1. Wi-Fi creds — IDF Kconfig under Component config → nano-ros must carry SSID + password OR your nros.toml must.
  2. Wrong locator — confirm host running zenohd is on the same Wi-Fi subnet (or routable to it). NAT will block discovery.
  3. idf.py menuconfig shows the Component config → nano-ros submenu (the component is wired) and CONFIG_NROS_RMW is set to a backend name (zenoh/xrce/cyclonedds). There is no separate CONFIG_NROS_ENABLED toggle on ESP-IDF; the component’s presence in main/idf_component.yml is the on-switch.
  4. See Troubleshooting — First 10 Minutes.

GitHub source

A complete reference app showing Wi-Fi + zenoh wiring on top of the component is not in-tree yet; the bare-metal examples/qemu-esp32-baremetal/rust/talker/ is the closest worked example.

Next

  • Bare-metal esp-hal Rust path: ESP32 (esp-hal).
  • Multi-component IDF apps: nano-ros sits next to other Espressif components (network, storage, sensors) — IDF’s component manager resolves them all.

Bare-metal Cortex-M3 (QEMU)

Single-node starter on bare-metal Cortex-M3 (QEMU MPS2-AN385) — no RTOS, no kernel scheduler. Pure cooperative spin via zpico_spin_once. Rust only. nros-c / nros-cpp are not supported on bare-metal targets (they assume a hosted RTOS for startup / heap / libc); see the examples coverage matrix for the policy.

When to use this path: ultra-constrained Cortex-M0+ / M3 / M4 targets with no OS scheduler, no pthread. If you have FreeRTOS or any RTOS, use the FreeRTOS starter instead — it’s more ergonomic and produces smaller code overall.

Prereqs. Install the nros CLI once per machine, then provision this board. nros setup fetches a prebuilt bare-metal toolchain (arm-none-eabi-gcc, qemu-system-arm, the zenoh router) plus the Rust thumbv7m-none-eabi target into a shared store — no manual cross-compiler install, no ROS 2 needed.

# Build the in-tree nros CLI (Phase 218):
source ./activate.sh        # OR: direnv allow / source ./activate.fish
just setup-cli              # builds packages/cli/target/release/nros

# Provision the bare-metal Cortex-M3 board (zenoh RMW is the default):
nros setup qemu-arm-baremetal --rmw zenoh

Real-board variants exist too: nros setup mps2-an385 and nros setup stm32f4 provision the same bare-metal toolchain for physical hardware.

Project layout

examples/qemu-arm-baremetal/rust/talker/
├── Cargo.toml
├── .cargo/config.toml         # target = thumbv7m-none-eabi
│                              # runner = qemu-system-arm ... -kernel
├── nros.toml                  # network + zenoh
├── package.xml
├── generated/                 # codegen output — build.rs runs
│                              #   `nros generate-rust` on first
│                              #   `cargo build`; gitignored.
└── src/main.rs                # #[entry] fn main() -> !

The board crate is nros-board-mps2-an385 (note: no -freertos suffix — this is the bare-metal variant) which provides:

  • Cortex-M3 startup + linker script
  • LAN9118 driver for smoltcp
  • BoardIdle::wfi() for cooperative wait

Configure

Verbatim from the in-tree examples/qemu-arm-baremetal/rust/talker/nros.toml:

# nano-ros config — QEMU ARM bare-metal talker (direct mode).
# Read by the nros-board-mps2-an385 board crate via Config::from_toml.

[node]
domain_id = 0

# Single ethernet transport running a zenoh session. `ip` is CIDR
# (address/prefix); the locator rides the transport.
[[transport]]
kind    = "ethernet"
ip      = "10.0.2.10/24"
mac     = "02:00:00:00:00:00"
gateway = "10.0.2.2"
rmw     = "zenoh"
locator = "tcp/10.0.2.2:7450"

QEMU Slirp networking — no host TAP / bridge / sudo. The zenohd default port is 7447; this example expects 7450 so start the router with zenohd --listen tcp/127.0.0.1:7450 (or edit nros.toml to match zenohd’s 7447 default).

Build

cd examples/qemu-arm-baremetal/rust/talker
cargo build --release

First build (~5 min) cross-compiles all of nano-ros’s Rust deps for thumbv7m-none-eabi. Re-builds finish in seconds.

Run

# 1. Bring up zenohd on the host (Slirp forwards 10.0.2.2:7450 → host
#    127.0.0.1:7450). The bare-metal test-fixture port is 7450, NOT
#    zenohd's default 7447 — edit `nros.toml` if you want 7447 instead.
#    The just recipe runs the in-tree zenohd installed by
#    `nros setup ... --rmw zenoh` on that port:
just qemu zenohd &
#    Or directly:
#    zenohd --listen tcp/127.0.0.1:7450 --no-multicast-scouting

# 2. Boot the talker in QEMU. The `just qemu talker` recipe wraps
#    qemu-system-arm with the LAN9118 networking wiring the example
#    expects — it's the only working invocation for this tutorial
#    (the example's `.cargo/config.toml` runner is bare `-kernel`,
#    no `-nic socket,model=lan9118,…`, so a plain `cargo run` boots
#    QEMU without networking):
just qemu talker
# Expected serial-over-semihosting output (per src/main.rs):
#   Declaring publisher on /chatter (std_msgs/Int32)
#   Publisher declared
#   Published: 0
#   Published: 1
#   ...

# 3. Verify from stock ROS 2:
source /opt/ros/humble/setup.bash
export RMW_IMPLEMENTATION=rmw_zenoh_cpp
# Talker publishes best-effort; stock `ros2 topic echo` defaults to
# RELIABLE, so the QoS-mismatched echo silently delivers nothing.
# Force best-effort to receive:
ros2 topic echo /chatter std_msgs/msg/Int32 --qos-reliability best_effort

QEMU exits via Ctrl-A x.

Readiness signal. Within ~15 seconds of QEMU boot (no RTOS init delay, but smoltcp + zenoh handshake still takes a few seconds), expect Published: 0 on semihosting stdout (the Rust talker pre-publishes 0 before the counter advances). If no Published: line:

  1. zenohd not running — talker spins on smoltcp poll until killed.
  2. Wrong LAN9118 emulation flag — qemu-system-arm needs -nic socket,model=lan9118,… or equivalent. The example’s .cargo/config.toml runner is bare -kernel (so a plain cargo run boots QEMU without networking); the LAN9118 wiring lives in the just qemu talker recipe (just/qemu-baremetal.just::talker), which is the working invocation for this tutorial. If you copy the runner out, mirror those flags.
  3. Cooperative spin starvation — if you added a long-running callback, the entire executor stalls; bare-metal has no preemption.
  4. See Troubleshooting — First 10 Minutes.

GitHub source

Constraints to be aware of

  • No alloc by default. Pure no_std + heapless for bounded collections. If you need alloc, opt in via the alloc feature on your board crate and supply a #[global_allocator].
  • No wake primitive. Cooperative single-thread spin only; the executor’s nros_platform_wake_* slots return Unsupported.
  • No preemption. A long-running user callback blocks every other dispatchable handle until it returns.
  • nros-c / nros-cpp NOT supported. These wrappers assume hosted-RTOS libc + heap. Pure-Rust API only on this target.

For Cortex-M3 with an RTOS, switch to the FreeRTOS starter.

Next

  • Subscriber / service / action peers under the same examples/qemu-arm-baremetal/rust/ tree.
  • Wake-callback opt-in: the wake-callback (latency-probe) bench under packages/testing/nros-bench/wake-latency-cortex-m3/ shows how to feed a backend’s transport-notify into the cooperative spin loop on bare-metal.
  • Real hardware: same code runs on STM32F4-Discovery with a different board crate (nros-board-stm32f4) and a different linker script.

PX4 Autopilot (external module)

Single-node starter on PX4 Autopilot via the external-module copy-out template. PX4’s EXTERNAL_MODULES_LOCATION pattern lets downstream firmware drop in nano-ros without forking PX4 itself. C++ only — PX4’s uORB binding is C++-only (Rust + C not in the coverage matrix).

Prereqs. A PX4-Autopilot ≥ v1.16 clone, the matching cross toolchain (e.g. gcc-arm-none-eabi for Pixhawk targets), and python3 with the PX4 development requirements installed (bash ./Tools/setup/ubuntu.sh once).

Project layout

PX4 external modules live outside the PX4 source tree and are hooked in at configure time. The template at integrations/px4/module-template/ has the PX4-required src/modules/<name>/ shape:

my_drone_firmware/
├── PX4-Autopilot/                          # PX4 source tree (submodule)
└── px4-modules/                            # passed via EXTERNAL_MODULES_LOCATION
    └── nano-ros/                           # copy-out from integrations/px4/module-template/
        └── src/
            ├── CMakeLists.txt              # populates config_module_list_external
            └── modules/
                └── nano_ros_app/
                    ├── CMakeLists.txt      # px4_add_module(... MAIN nano_ros_app)
                    └── nano_ros_app.cpp    # the user-editable app

Prereq. PX4 is a full-tier dependency. Run just setup px4 first to populate third-party/px4/PX4-Autopilot and third-party/px4/px4-rs. just px4 doctor reports the gap on a fresh clone.

just setup px4              # equivalent to: just px4 setup
just px4 doctor

Vendor the template into your firmware repo, then point PX4 at its parent directory + tell the template where nano-ros lives via NANO_ROS_DIR. The template accepts either form — a CMake cache variable (-DNANO_ROS_DIR=<path>) takes precedence over an environment variable (NANO_ROS_DIR=…), then the in-tree default:

cmake -B build -S PX4-Autopilot \
      -DCONFIG=px4_fmu-v5_default \
      -DEXTERNAL_MODULES_LOCATION=$PWD/px4-modules/nano-ros \
      -DNANO_ROS_DIR=$PWD/../nano-ros            # point at your nano-ros clone

# Or via environment, if you prefer:
NANO_ROS_DIR=$PWD/../nano-ros cmake -B build -S PX4-Autopilot \
      -DCONFIG=px4_fmu-v5_default \
      -DEXTERNAL_MODULES_LOCATION=$PWD/px4-modules/nano-ros

(EXTERNAL_MODULES_LOCATION must point at the dir containing the template’s src/ — that’s px4-modules/nano-ros, not its parent. PX4’s root CMakeLists.txt does add_subdirectory("${EXTERNAL_MODULES_LOCATION}/src" …), so the path resolves to px4-modules/nano-ros/src per the layout above.)

Inside the module, the canonical pattern bridges uORB → nano-ros. The module is a PX4Module subclass that runs in its own work queue, opens an nros executor, and forwards uORB messages onto a zenoh / DDS topic (or vice versa). Edit nano_ros_app.cpp to add your topic bindings.

Configure

The template does not ship a Kconfig overlay (no Kconfig.projbuild files). Module enablement is implicit once EXTERNAL_MODULES_LOCATION points at the template’s parent. Pass RMW + ROS-edition selection via CMake cache vars rather than menuconfig:

cmake -B build -S PX4-Autopilot \
      -DCONFIG=px4_fmu-v5_default \
      -DEXTERNAL_MODULES_LOCATION=$PWD/px4-modules/nano-ros \
      -DNANO_ROS_DIR=$PWD/../nano-ros \
      -DNANO_ROS_RMW=zenoh

(Adding a Kconfig overlay so the module appears under menuconfig → External modules is a follow-up task; for now the template is always enabled.)

Build

cd PX4-Autopilot
make px4_fmu-v5_default
# Or for the SITL simulator (POSIX target — easier to develop against):
make px4_sitl_default gazebo

The first build cross-compiles nano-ros’s Rust staticlibs alongside PX4’s NuttX kernel + apps (~10 min on a fresh checkout).

Run

# SITL: PX4 boots Gazebo + the autopilot binary
cd PX4-Autopilot
make px4_sitl_default gazebo
# In the PX4 console:
pxh> nano_ros_app start

# Real hardware (Pixhawk): flash via QGroundControl or
#     `make px4_fmu-v5_default upload` over the bootloader USB

# Verify from stock ROS 2 on the same network (only after you have
# added uORB → nano-ros forwarders to nano_ros_app.cpp — the shipped
# template forwards nothing on its own):
source /opt/ros/humble/setup.bash
export RMW_IMPLEMENTATION=rmw_zenoh_cpp
ros2 topic echo /vehicle_local_position px4_msgs/msg/VehicleLocalPosition

Readiness signal. After nano_ros_app start in the PX4 console, the shipped template logs nano-ros uORB backend registered and returns immediately — it doesn’t start a publisher loop or print a “bridge started” line on its own. You’re expected to edit nano_ros_app.cpp to wire your uORB → nano-ros forwards. If nano_ros_app reports register failed, check:

  1. Module didn’t register — the template logs nros_rmw_uorb_register() -> <rc> on startup (search the PX4 boot log for nros_rmw_uorb_register). A non-zero <rc> usually means a NuttX kernel-config / feature-gate mismatch.
  2. Once you’ve added forwarders, zenohd must be reachable from the autopilot’s network (Pixhawk: configured via QGroundControl or param set).
  3. uORB topic not advertised yet — start the upstream PX4 module that publishes it (commander start etc.) first.
  4. See Troubleshooting — First 10 Minutes.

GitHub source

Constraints to be aware of

  • uORB binding is C++-only. The PX4 example collapses uORB registration to a C++ port; Rust / C variants exist for non-PX4 RTOSes but not for PX4.
  • PX4’s NuttX kernel. Underneath, PX4 runs NuttX; if you need to debug at the kernel layer, the NuttX starter page applies too.
  • uORB throughput vs zenoh hops. uORB is in-process pub/sub at ~µs latency; zenoh adds network-RTT. Plan accordingly when bridging high-rate streams.

Next

  • Add your own uORB topics to the bridge: see the nano_ros_app.cpp template’s topic-table section.
  • Multi-vehicle: PX4-XRCE-Agent → nano-ros XRCE backend gives you the standard PX4-ROS bridge with nano-ros on the autopilot side.
  • For pure-NuttX (no PX4) firmware: see the NuttX starter.

ARM FVP (FVP_BaseR_AEMv8R)

Run nano-ros on Arm’s Base_RevC AEMv8-R Fast Models — the Cortex-A SMP profile under Zephyr 3.7 (board id fvp_baser_aemv8r/fvp_aemv8r_aarch64/smp). The FVP is the canonical local proxy for the safety-island reference platforms that follow the same hwv2 Zephyr shape; pair this chapter with the Zephyr (west module) starter for the build half.

Phase 217. The build half (just zephyr build-fvp-aemv8r{,-cyclonedds}) shipped in Phase 117.13–117.14; the run half — invoking FVP_BaseR_AEMv8R and piping UART 0–3 to stdout — landed as Phase 217.A on 2026-06-03. See docs/roadmap/phase-217-arm-fvp-local-runtime.md.

When to use

  • You need to exercise a Cortex-A SMP Zephyr image on a developer laptop without dedicated silicon.
  • You’re bringing up a new hwv2 safety-island target — the FVP is the closest in-tree reference for board.cmake + SMP boot + Cyclone DDS shape.
  • You’re validating Phase 117 stock-RMW interop locally before promoting to a hardware bench.

The FVP is not a replacement for QEMU on Cortex-M (mps2-an385, mps3-an547); those targets are covered by FreeRTOS (QEMU) and the Zephyr starter. The FVP is also not the same surface as Corellium’s AVH cloud FVP — AVH packages firmware as .coreimg and provisions via a remote API; local FVP loads a raw ELF via -a cluster0.cpu*=<elf>. AVH is out of scope for this chapter.

Prereqs

  • A working Zephyr 3.7 workspace — run nros setup zephyr once if you haven’t (see the Zephyr starter).
  • The Arm Base_RevC AEMv8R Fast Models binary — license-gated. Download from developer.arm.com — Arm Ecosystem FVPs after accepting the EULA. nano-ros does not download it (the [gated.arm-fvp] row in nros-sdk-index.toml declares the tool but the fetch is your responsibility — same policy as the NVIDIA Orin SPE FSP).

After installing, export one of the discovery env vars:

# Preferred — Zephyr's canonical name; takes highest priority.
export ARMFVP_BIN_PATH=/opt/Arm/FastModels/Base_RevC_AEMv8R/models/Linux64_GCC-9.3

# Alternative — directory layout from the gated installer; the
# resolver scans `models/Linux64_GCC-*/` underneath it.
export ARM_FVP_DIR=/opt/Arm/FastModels/Base_RevC_AEMv8R

If neither is set, FVP_BaseR_AEMv8R is discovered via PATH as a last-ditch fallback.

Installer surface (Phase 217.B.1)

After extracting the Arm FVP tarball, run the discovery script:

ARM_FVP_DIR=/path/to/extracted/fvp \
    scripts/installers/arm-fvp-installer.sh

The installer locates FVP_BaseR_AEMv8R under $ARM_FVP_DIR, symlinks the containing directory to ~/.nros/sdks/arm-fvp/current/ (atomic via ln -sfn), and prints the export ARMFVP_BIN_PATH=… line for your shell rc. Run scripts/installers/arm-fvp-installer.sh --print-env later to re-emit the export. It never downloads anything — gated-tool policy.

Doctor check (Phase 217.B.2)

nros doctor --board fvp-aemv8r-smp cross-checks the [gated.arm-fvp] entry in nros-sdk-index.toml and warns (never hard-fails — gated) when the FVP can’t be resolved via ARMFVP_BIN_PATH, ARM_FVP_DIR, PATH, or the canonical ~/.nros/sdks/arm-fvp/current/FVP_BaseR_AEMv8R landing path. The just zephyr run-fvp-aemv8r{,-cyclonedds} recipes do the equivalent inline via scripts/zephyr/resolve-fvp-bin.sh and skip with a clear hint when the binary can’t be found.

Build

The build half is unchanged from Phase 117.13 / 117.14:

# Phase 117.13 — Zephyr-only talker.
just zephyr build-fvp-aemv8r

# Phase 117.14 — C++ pub/sub over Cyclone DDS, the wire-compat
# reference for the safety-island slice.
just zephyr build-fvp-aemv8r-cyclonedds

Each recipe shells west build -b fvp_baser_aemv8r/fvp_aemv8r_aarch64/smp inside the zephyr-workspace/ directory and produces zephyr.elf at one of:

  • zephyr-workspace/build-fvp-aemv8r-talker/zephyr/zephyr.elf
  • zephyr-workspace/build-aemv8r-cyclonedds-talker/zephyr/zephyr.elf

Run

Once the build artifacts and ARM_FVP_DIR / ARMFVP_BIN_PATH are in place:

# Boot the Phase 117.13 talker.
just zephyr run-fvp-aemv8r

# Boot the Phase 117.14 cpp/cyclonedds talker.
just zephyr run-fvp-aemv8r-cyclonedds

Under the hood the recipe:

  1. Verifies west + the Zephyr workspace + zephyr.elf exist; skips with a hint otherwise.
  2. Resolves the FVP binary directory via scripts/zephyr/resolve-fvp-bin.sh (priority order: ARMFVP_BIN_PATHARM_FVP_DIR/models/Linux64_GCC-*/dirname $(command -v FVP_BaseR_AEMv8R)).
  3. Exports ARMFVP_BIN_PATH=<dir> and shells west build -d <build-dir> -t run, which drives Zephyr’s cmake/emu/armfvp.cmake target with the canonical boards/arm/fvp_baser_aemv8r/board.cmake -C flags — UART 0–3 piped to stdout, GICv3, cache state, NUM_CORES from CONFIG_MP_MAX_NUM_CPUS. No flags are duplicated in the just recipe.

Exit cleanly with Ctrl-C.

Expected output

The Zephyr 3.7 boot banner appears on UART0 first, followed by the talker. The exact line counts depend on CONFIG_BOOT_BANNER and your locator config, but the markers to look for are:

*** Booting Zephyr OS build v3.7.0 ***
[00:00:00.xxx,000] <inf> nros: session up (domain 0)
Published: 0/1
Published: 1/2
...

For the cpp/cyclonedds recipe, the same banner is followed by Cyclone DDS reader-match logs and std_msgs/Int32 publish lines. Verify ROS 2 interop by running a sibling listener in another terminal:

# stock ROS 2 — reads the FVP's Cyclone DDS publisher
ros2 topic echo /chatter std_msgs/msg/Int32

The same std_msgs/Int32 payload + byte-equal CDR framing must appear on both sides — that’s the Phase 117 stock-RMW interop contract.

Cross-references

Native POSIX

Use the native POSIX target for first experiments, CI smoke tests, and ROS 2 interoperability checks on Linux or *BSD. It uses OS sockets, the host process environment, and the same source checkout layout as the embedded targets.

When to Use It

  • You want the shortest path to a working nano-ros publisher or subscriber.
  • You are validating message generation, CMake integration, or ROS 2 interop before moving to an RTOS.
  • Your deployment target is Linux, *BSD, or an embedded Linux system.

Setup

Build the in-tree nros CLI (Phase 218), then provision the native host. For a host build there is no cross-toolchain to fetch — nros setup native installs only the RMW host daemon (zenohd for zenoh, the Micro-XRCE-DDS agent for xrce) into a shared store. ROS 2 is not required.

# Build the in-tree nros CLI:
source ./activate.sh        # OR: direnv allow / source ./activate.fish
just setup-cli              # builds packages/cli/target/release/nros

# Provision the native host (zenoh RMW is the default):
nros setup native --rmw zenoh        # or: --rmw xrce / --rmw cyclonedds

native and posix are accepted as the same board name.

For a colcon consumer workspace that already has nano-ros under src/:

colcon build && source install/setup.bash

Package Layout

For a POSIX application package:

my_posix_node/
├── package.xml
├── Cargo.toml          # Rust path
├── CMakeLists.txt      # C / C++ path
└── src/
    └── main.rs         # or main.c / main.cpp

Keep package beside nano-ros in workspace src/. Use path dependencies for Rust or add_subdirectory(<path-to-nano-ros>) for C/C++.

Code Example

Rust publisher skeleton:

use nros::prelude::*;

fn main() -> Result<(), Box<dyn std::error::Error>> {
    let config = ExecutorConfig::from_env().node_name("talker");
    let mut executor = Executor::open(&config)?;
    let mut node = executor.create_node("talker")?;
    let publisher = node.create_publisher::<std_msgs::msg::Int32>("/chatter")?;

    let mut msg = std_msgs::msg::Int32 { data: 0 };
    loop {
        publisher.publish(&msg)?;
        msg.data += 1;
        executor.spin_once(100);
    }
}

Use the full First Node — Rust walkthrough for generated messages and runnable commands.

Build and Run

Start the RMW host daemon (installed by nros setup native). For zenoh:

zenohd --listen tcp/127.0.0.1:7447

Run the node directly via cargo run (Rust) or cmake --build build && ./build/<binary> (C/C++), or via colcon build && source install/setup.bash && ros2 run … if the package lives in a colcon consumer workspace.

Configuration

Native examples can read runtime settings from the shell:

export ROS_DOMAIN_ID=0
export NROS_LOCATOR=tcp/127.0.0.1:7447

The same fields are compiled into embedded targets through their platform-specific configuration files. See Configuration for the full layering.

Deployment

POSIX deployment is normal process deployment: install the workspace, source install/setup.bash, set environment, and run the binary. For ROS 2 interop, run ROS 2 Interoperability and set the ROS side to rmw_zenoh_cpp.

Application Workflow

nano-ros users usually want one path: prepare package, write node, build it, then deploy to target. Use Concepts only when workflow raises a technical question.

1. Prepare Workspace and Package

nano-ros is shipped source-only — vendor it next to (or inside) your workspace, then provision the board’s toolchain with nros setup:

# Build the in-tree nros CLI (Phase 218), then provision your board (+ RMW):
./scripts/bootstrap.sh base
source ./activate.sh        # OR: direnv allow / source ./activate.fish
nros setup native --rmw zenoh        # or qemu-arm-freertos, zephyr, …

nros setup ships prebuilt toolchains per platform per RMW — see Installation.

For multi-package workspaces (Pattern A — recommended for POSIX + mixed C / C++ / Rust deployments), put nano-ros and your packages side-by-side under a shared src/:

~/ros2_ws/
├── src/
│   ├── nano-ros/                  # this repo
│   └── my_robot_node/
│       ├── package.xml
│       ├── Cargo.toml             # Rust, if using Rust API
│       ├── CMakeLists.txt         # C/C++, if using CMake
│       ├── config.toml            # embedded targets
│       └── src/
└── build/ install/ log/           # if you use colcon

For third-party C/C++ projects without colcon (Pattern B), pull nano-ros in as a third_party/nano-ros/ git submodule and consume it via add_subdirectory(third_party/nano-ros nano_ros).

See Installation for both patterns. The per-language starter pages document the canonical package shape in their “Project layout” sections: Rust, C, C++. For two or more nodes, use the Multi-Node Projects group: start from the full project layout, then drill into Node, Bringup, and Entry packages.

2. Write Node Code

Choose API language first:

  • Rust — use nros, generated message crates, and Executor.
  • C — include nros/nros.h and generate interfaces with nros_find_interfaces(LANGUAGE C) in CMake.
  • C++ — include nros/nros.hpp and use typed wrappers.

Start with one of the Linux starters above, then adapt to your target via the Embedded Starters section.

3. Generate Messages

If you use custom .msg, .srv, or .action files, generate bindings inside the workspace/build tree. See Message Binding Generation.

4. Configure Target

Pick a platform and an RMW backend at build time (compile-time choice — there is no RMW_IMPLEMENTATION runtime switch on embedded targets):

CMake sideCargo side
set(NANO_ROS_PLATFORM <plat>)feature platform-<plat>
set(NANO_ROS_RMW <rmw>)feature rmw-<rmw>-cffi (transports auto-pull)
set(NANO_ROS_BOARD <board>) (optional, embedded only)nros-board-<board> dep

Supported pairs: posix / freertos / nuttx / threadx / zephyr / esp32 / baremetal × zenoh / xrce / dds / cyclonedds. Not every cell is implemented — see the Coverage Matrix.

Runtime configuration (ROS_DOMAIN_ID, ZENOH_LOCATOR, …) works on POSIX. Embedded targets resolve config via CMake cache vars, Kconfig (Zephyr), Cargo features, or config.toml. See Configuration.

5. Build, Test, Deploy

For each canonical entry point:

# Single example (POSIX, Pattern A or B):
cd examples/native/rust/talker
cargo run

# Single C/C++ example (CMake + add_subdirectory):
cd examples/qemu-arm-freertos/cpp/talker
cmake -B build -DCMAKE_TOOLCHAIN_FILE=$PWD/../../../../../cmake/toolchain/arm-freertos-armcm3.cmake
cmake --build build

# Per-platform multi-example build:
just freertos build-fixtures
just zephyr  build-fixtures
just nuttx   build-fixtures

# Discover full-matrix commands for a platform:
just --group full-matrix --list zephyr

# Multi-component system (orchestration):
nros metadata my_system
nros plan my_system launch/my_system.launch.py
nros check
cargo build                 # or: cmake --build / west build / idf.py build

# POSIX-only colcon consumer-workspace build:
colcon build && source install/setup.bash

For target-specific deployment, go to the matching platform guide. Each guide covers toolchain setup, package layout, code example, build command, run/flash command, and deployment notes.

See Deployment Workflow.

Build as a CMake subdirectory

This is the way to integrate nano-ros into a C or C++ project. The nano-ros repo ships a top-level CMakeLists.txt that exposes everything via add_subdirectory(...). No install step, no find_package(NanoRos), no install prefix removed every last trace of that pipeline.

Layout

my_app/
├── CMakeLists.txt
├── main.c
└── third_party/
    └── nano-ros/        # git clone / git submodule of this repo

User project CMakeLists.txt

cmake_minimum_required(VERSION 3.22)
project(my_app C)

# Pick platform + RMW BEFORE add_subdirectory.
set(NANO_ROS_PLATFORM posix)   # posix | freertos | nuttx | threadx | zephyr | baremetal
set(NANO_ROS_RMW     zenoh)    # zenoh | dds | xrce | cyclonedds

add_subdirectory(third_party/nano-ros nano_ros)

add_executable(my_app main.c)
target_link_libraries(my_app PRIVATE NanoRos::NanoRos)
nros_platform_link_app(my_app)

# Optional — generate C bindings for ROS 2 .msg / .srv / .action files.
# nros_generate_interfaces() is reachable in-tree once nano-ros has been
# add_subdirectory'd; no install step required.
nros_generate_interfaces(std_msgs DEPENDENCIES builtin_interfaces SKIP_INSTALL)
target_link_libraries(my_app PRIVATE std_msgs__nano_ros_c)

That’s the whole story for the host-POSIX / zenoh case. CMake’s transitive target propagation pulls in libnros_c.a, libnros_rmw_zenoh_staticlib.a, the POSIX platform shim, system libraries (pthread, dl, m), and the per-build nros_config_generated.h header automatically.

Cache variables

VariableDefaultValues
NANO_ROS_PLATFORMposixposix, freertos (freertos_armcm3), nuttx (nuttx_armv7a), threadx (threadx_linux, threadx_riscv64), zephyr, baremetal
NANO_ROS_BOARD(unset)required for threadx (threadx-linux or riscv64-qemu) and baremetal (mps2-an385, stm32f4-nucleo, …)
NANO_ROS_RMWzenohzenoh, dds, xrce, cyclonedds
NANO_ROS_ROS_EDITIONhumblehumble, iron
NANO_ROS_BUILD_CODEGENONON / OFF

Variables MUST be set(...) BEFORE add_subdirectory(...) — the sub-project consumes them at include time.

What about installing?

deleted every install(...) rule. nano-ros is consumed in source form — never out of an installed prefix. If you need a shippable artefact, your user project owns the install layout; ship your binary, not nano-ros itself.

For RTOS users who want a more idiomatic surface than raw add_subdirectory, see the integration shells under integrations/<rtos>/ — they translate west / esp-idf / NuttX / PX4 manifests into the same root CMake. Each shell is a ~20-line wrapper around add_subdirectory(<repo>).

Worked example

The examples/native/c/talker/CMakeLists.txt is the canonical copy-out template — 20 lines including codegen + per-app fixup. All 83 in-tree C/C++ examples follow the same shape after.

Message Binding Generation

nros uses generated Rust bindings for ROS 2 message types. The nros generate-rust command generates no_std compatible bindings from package.xml dependencies.

Overview

The binding generator lives in the in-tree CLI sub-workspace at packages/cli/ and provides:

  • nros standalone binary
  • Pure Rust, no_std compatible output using heapless types
  • Automatic dependency resolution via ament index or bundled interfaces
  • .cargo/config.toml generation for crate patches

Prerequisites

  1. package.xml in project root - Declares ROS interface dependencies

    <?xml version="1.0"?>
    <package format="3">
      <name>my_package</name>
      <version>0.1.0</version>
      <description>My nros package</description>
      <maintainer email="dev@example.com">Developer</maintainer>
      <license>Apache-2.0</license>
      <depend>std_msgs</depend>
      <depend>geometry_msgs</depend>
      <export>
        <build_type>ament_cargo</build_type>
      </export>
    </package>
    
  2. nros tool built + on PATH

    The nros CLI is built from the in-tree sub-workspace at packages/cli/ (Phase 218) and put on PATH by the activate file:

    # From the nano-ros repository root
    source ./activate.sh        # OR: direnv allow / source ./activate.fish
    just setup-cli              # builds packages/cli/target/release/nros
    

    See Installation for the full walkthrough.

  3. ROS 2 environment (optional for standard types)

    Standard interfaces (std_msgs, builtin_interfaces) are bundled with nros and work without ROS 2. For additional packages (e.g., geometry_msgs, sensor_msgs), source a ROS 2 environment:

    source /opt/ros/humble/setup.bash
    

Workflow

Step 1: Create package.xml

Declare your ROS interface dependencies in <depend> tags:

<depend>std_msgs</depend>      <!-- For std_msgs::msg::Int32, String, etc. -->
<depend>example_interfaces</depend>  <!-- For service types -->

Step 2: Generate bindings

cd my_project
nros generate-rust

This will:

  1. Parse package.xml to find dependencies
  2. Resolve transitive dependencies (ament index + bundled interfaces)
  3. Filter to interface packages (those with msg/srv/action)
  4. Generate bindings to generated/ directory

Step 3: Add dependencies to Cargo.toml

Reference the generated crates using crates.io version specifiers:

[dependencies]
std_msgs = { version = "*", default-features = false }
example_interfaces = { version = "*", default-features = false }

The .cargo/config.toml patches redirect these to local paths.

Why msg crates are RMW-agnostic

Generated msg crates carry only the wire-format data type — no backend-specific code, no per-RMW Cargo feature. A user manifest is the plain pair:

[dependencies]
std_msgs = { version = "*", default-features = false }
nros     = { version = "*", features = ["rmw-cyclonedds"] }   # RMW choice lives here

Transport choice and message schema are orthogonal concerns and the manifest reflects that. This matches upstream rclcpp + rclrs, which both ship msg packages RMW-agnostic and let the RMW pick which descriptor representation it wants at runtime.

For DDS-based backends that need a per-type descriptor on the wire (Cyclone DDS today), the nros-rmw-cyclonedds shim builds those descriptors lazily on first pub/sub for a given message type, walks the static field schema exposed by nros-serdes (the Message trait with const TYPE_NAME + const FIELDS), and caches the result in a bounded no_std registry. No per-msg-pkg backend code is required.

Tracking + sizing knob (NROS_CYCLONEDDS_MAX_TYPES): see docs/roadmap/phase-212-ux-cargo-native-and-file-consolidation.md section 212.K.7.

Git Dependency Workflow

For projects that consume nros as a git dependency (not from within the nros repo), use --nano-ros-git instead of --nano-ros-path:

Step 1: Add git dependency to Cargo.toml:

[dependencies]
nros = { git = "https://github.com/jerry73204/nano-ros", default-features = false, features = ["std"] }
std_msgs = { version = "*", default-features = false }

Step 2: Create package.xml (same as above).

Step 3: Generate bindings with git patches:

source /opt/ros/humble/setup.bash
nros generate-rust --config --nano-ros-git

This generates .cargo/config.toml with git-based patches:

[patch.crates-io]
nros-core = { git = "https://github.com/jerry73204/nano-ros" }
nros-serdes = { git = "https://github.com/jerry73204/nano-ros" }
std_msgs = { path = "generated/std_msgs" }
builtin_interfaces = { path = "generated/builtin_interfaces" }

Step 4: Use in code

#![allow(unused)]
fn main() {
use std_msgs::msg::Int32;
use example_interfaces::srv::{AddTwoInts, AddTwoIntsRequest, AddTwoIntsResponse};

let msg = Int32 { data: 42 };
}

Command Options

nros generate-rust [OPTIONS]

Options:
      --manifest <PATH>       Path to package.xml [default: package.xml]
  -o, --output <DIR>          Output directory [default: generated]
      --config                Generate .cargo/config.toml with [patch.crates-io] entries
      --nano-ros-path <PATH>  Path to nros crates (for config patches, local dev)
      --nano-ros-git          Use nros git repo for config patches (external users)
      --force                 Overwrite existing bindings
  -v, --verbose               Enable verbose output

Generated Output Structure

my_project/
├── package.xml              # Your dependency declarations
├── Cargo.toml               # Your package manifest
├── src/
│   └── main.rs              # Your code using generated types
├── generated/               # Generated bindings (do not edit)
│   ├── std_msgs/
│   │   ├── Cargo.toml
│   │   └── src/
│   │       ├── lib.rs       # #![no_std]
│   │       └── msg/
│   │           ├── mod.rs
│   │           └── int32.rs
│   └── builtin_interfaces/  # Transitive dependency
│       └── ...
└── .cargo/
    └── config.toml          # [patch.crates-io] entries

Generated Code Features

no_std by default:

#![allow(unused)]
#![no_std]

fn main() {
pub mod msg;
}

std feature for optional std support:

[features]
default = []
std = ["nros-core/std", "nros-serdes/std"]

heapless types for embedded:

#![allow(unused)]
fn main() {
pub struct String {
    pub data: heapless::String<256>,
}

pub struct Arrays {
    pub data: heapless::Vec<i32, 64>,
}
}

Service types with Request/Response:

#![allow(unused)]
fn main() {
pub struct AddTwoInts;
pub struct AddTwoIntsRequest { pub a: i64, pub b: i64 }
pub struct AddTwoIntsResponse { pub sum: i64 }

impl RosService for AddTwoInts {
    type Request = AddTwoIntsRequest;
    type Reply = AddTwoIntsResponse;
}
}

Standalone Package Mode

Examples are configured as standalone packages (excluded from workspace) because each has its own .cargo/config.toml patches. Build each example from its own directory:

cd examples/native/rust/talker && cargo build --features zenoh
cd examples/native/rust/service-client && cargo build --features zenoh

Regenerating Bindings

To regenerate after ROS package updates or dependency changes:

nros generate-rust --force

Bundled Interfaces

nros ships standard .msg files for common packages so codegen works without a ROS 2 environment:

  • std_msgs (Bool, Int32, String, Header, etc.)
  • builtin_interfaces (Time, Duration)

These are located at packages/codegen/interfaces/. When a ROS 2 environment is sourced, the ament index takes precedence over bundled files.

Troubleshooting

“Package ‘X’ not found in ament index or bundled interfaces”

  • For standard types (std_msgs, builtin_interfaces): should work without ROS 2
  • For other packages: source ROS 2 environment: source /opt/ros/humble/setup.bash
  • Check package is installed: ros2 pkg list | grep X
  • Install if missing: sudo apt install ros-humble-X

Build errors with generated code

  • Regenerate with --force flag
  • Check nros crate compatibility

C Code Generation (CMake)

The nano_ros_generate_interfaces() CMake function generates C bindings for .msg, .srv, and .action files. It uses a bundled codegen library — no external nros binary needed.

Prerequisites

nano_ros_generate_interfaces() becomes available automatically once the consumer’s CMakeLists.txt invokes add_subdirectory(nano-ros). The codegen tool ships inside the in-tree nros CLI binary (packages/cli/target/release/nros, Phase 218; Phase 195.D had retired the nros-codegen submodule); cmake auto-resolves it from PATH / packages/cli/target/release/ / the transitional ${NROS_HOME:-~/.nros}/bin/. No separate build step.

Usage

See examples/native/c/custom-msg/CMakeLists.txt for a complete example.

Configuration Guide

nano-ros config is language-agnostic and has one home per concern — no setting lives in two places. Workspace/system config lives in Cargo.toml metadata (Rust) or CMake (C/C++) plus a universal system.toml; nros.toml is narrowed to the embedded direct-mode runtime file a single-node app’s board parses at boot. This guide covers what each file owns and the nros.toml format.

One home per concern

FileOwnsPer
Cargo.tomlRust build: crate, language deps, the RMW feature menu (rmw-zenoh/rmw-cyclonedds/rmw-xrce); workspace/system config via [workspace.metadata.nros] + [package.metadata.nros.{node,entry,application}]Rust project
CMakeLists.txtC/C++ build: targets, language deps, add_subdirectory(<repo>), the NROS_RMW option; node/entry registration via nano_ros_node_register / nano_ros_entryC/C++ project
.cargo/config.toml[patch.crates-io] dependency injection only (local crate + generated-msg paths), plus the cargo [build]/[target]/[env] knobs (target triple, runner, rustflags). No nano-ros runtime config.Rust project
package.xmlROS package identity + msg <depend>s (codegen input for nros generate)all
system.tomlSystem topology — components, deploy targets, domain, RMW. The language-agnostic universal descriptor (same schema for Rust/C/C++). Optional for single-node (the toolchain synthesises an implicit 1-component system when absent).bringup pkg
nros.tomlEmbedded direct-mode runtime only[node] / [[transport]] / [node.rt], parsed by the board at boot. NOT a workspace manifest (a workspace-root nros.toml is rejected by the CLI).embedded single-node app

Boundary rule. If a knob changes what is compiled/linked, it lives in the build file (Cargo.toml feature / CMakeLists.txt option). If it changes the system topology (components, deploy, domain, RMW), it lives in system.toml. If it is what an embedded single-node board does at boot (node identity, transports, RT), it lives in nros.toml.

Config home by language × scale

Mirrors RFC-0004 §3:

Single-nodeWorkspace
RustCargo.toml [package.metadata.nros.application] (+ nros::main!); optional system.toml to pin rmw/domainroot [workspace.metadata.nros] + node [package.metadata.nros.node] + entry [package.metadata.nros.entry] + bringup system.toml
C / C++CMakeLists.txt (NANO_ROS_PLATFORM/RMW) + package.xml; optional system.tomlnano_ros_node_register / nano_ros_entry per pkg + same system.toml + package.xml

The embedded single-node app that hand-writes main() (no codegen) is the case that keeps nros.toml — see below.

Full design rationale: RFC-0004 (configuration & transports); RMW backend selection & lowering is RFC-0031. For the multi-RMW runtime topic-forwarding bridge (a separate file/feature), see nros-bridge.toml — do not confuse it with the embedded-runtime nros.toml.

nros.toml — embedded direct-mode runtime

nros.toml is the embedded direct-mode runtime file for a hand-written single-node app. The board reads it via Config::from_toml (compile-baked with include_str! on embedded; filesystem/env on hosted) — no launch file, no planner, no generated main(). This is what the examples/** copy-out templates use (“boilerplate IS lesson”). It carries [node], [[transport]], and [node.rt] only.

It is not a workspace manifest and not read by nros plan/check/codegen-system — a workspace-root nros.toml is rejected by the CLI (nros migrate workspace). Workspace/system config lives in Cargo.toml metadata + system.toml (see the matrix above). The retired config.toml ([network]/[zenoh]/[scheduling]) was folded into the [node] / [[transport]] / [node.rt] sections here.

Shape

# nros.toml

[node]
domain_id = 0              # ROS 2 domain ID (0–232)
# namespace = "/"
# rmw = "zenoh"            # ACTIVE backend; must match a LINKED backend
                           # (the build file picks what is linked)

# One transport per session. A single ethernet/wifi/serial entry is the
# common case; multiple entries open via Executor::open_multi (multi-homed
# or cross-RMW). In-process topic forwarding is the separate [[bridge]] path.
[[transport]]
kind    = "ethernet"       # ethernet | wifi | serial | can
ip      = "10.0.2.10/24"   # CIDR — the prefix rides the address
mac     = "02:00:00:00:00:00"
gateway = "10.0.2.2"
locator = "tcp/10.0.2.2:7447"
# id      = "eth"          # bind key for multi-transport (defaults to rmw)
# interface = "tap-tx0"    # host interface (threadx-linux)

[node.rt]                  # scheduling / real-time (RTOS); omit for defaults
app_priority         = 12
app_stack_bytes      = 65536
zenoh_read_priority  = 16
zenoh_lease_priority = 16
poll_priority        = 16
poll_interval_ms     = 5

Per-kind transport fields:

kindfields
ethernetip (CIDR), mac, gateway, interface
wifissid, password, optional static ip/gateway
serial / candevice, baudrate
allid, rmw, locator

ip is CIDR (10.0.2.10/24) — the board derives the prefix or netmask from it. A serial locator carrying # (serial/UART_0#baudrate=115200) is fine — quote it.

How nros.toml is consumed

Rust (board Config::from_toml, compile-baked):

use nros_board_mps2_an385::{Config, run};

fn main() -> ! {
    run(Config::from_toml(include_str!("../nros.toml")), |config| {
        let exec = ExecutorConfig::new(config.zenoh_locator)
            .domain_id(config.domain_id);
        // ...
    })
}

C/C++ (CMake parses it into the NROS_APP_CONFIG struct):

nano_ros_read_config("${CMAKE_CURRENT_SOURCE_DIR}/nros.toml")
# Sets NROS_CONFIG_ZENOH_LOCATOR, NROS_CONFIG_DOMAIN_ID, NROS_CONFIG_IP, …
nros_support_init(&support, NROS_APP_CONFIG.zenoh.locator,
                  NROS_APP_CONFIG.zenoh.domain_id);

nros new scaffolds an nros.toml for embedded targets automatically.

Build-time environment variables

Read during cargo build (by build.rs) or by justfile recipes. Set in .env or export in your shell.

SDK paths

Auto-resolved by just setup <platform>; override if SDKs live elsewhere.

VariableDefaultDescription
FREERTOS_DIRthird-party/freertos/kernelFreeRTOS kernel source
FREERTOS_PORTGCC/ARM_CM3FreeRTOS portable layer
LWIP_DIRthird-party/freertos/lwiplwIP source
FREERTOS_CONFIG_DIRBoard crate’s config/FreeRTOSConfig.h
NUTTX_DIR / NUTTX_APPS_DIRthird-party/nuttx/…NuttX RTOS + apps
THREADX_DIR / NETX_DIRthird-party/threadx/…ThreadX + NetX Duo
THREADX_CONFIG_DIR / NETX_CONFIG_DIRBoard crate’s config/tx_user.h / nx_user.h

Buffer-tuning vars (ZPICO_*, XRCE_*, NROS_*) are optional — see the Environment Variables Reference.

Binary-size knobs (embedded)

On a constrained MCU, two build-time env vars (set in the example’s .cargo/config.toml [env], like the other NROS_* tuning) shed the parts a brokered client doesn’t need:

VariableDefaultEffect
NROS_LINK_IPonNROS_LINK_IP=0 on a serial-only node drops the IP link layer — zenoh-pico’s TCP/UDP link C (Z_FEATURE_LINK_TCP/UDP=0) and (with --gc-sections) the smoltcp platform impl. Serial link stays.
NROS_SMOLTCP_MAX_SOCKETS / NROS_SMOLTCP_MAX_UDP_SOCKETS4 / 2Sized for DDS RTPS (3 UDP/participant). A zenoh/XRCE client multiplexes everything over one session → set both to 1 to drop the spare socket buffers (≈8 KB each).
NROS_HEAP_SIZEper-board (64 KB mps2-an385, 32 KB stm32f4)Decimal bytes for the bare-metal static heap. The defaults are generous; size to the RMW’s working set (table below). E.g. NROS_HEAP_SIZE = "24576" on a zenoh-pico node cut the mps2-an385 .data 66 → 25 KB (−41 KB).

Static-heap sizing by backend (bare-metal FreeListHeap, set via NROS_HEAP_SIZE):

BackendPeak working setRecommended heapNotes
zenoh-pico (TCP)~16 KB24–32 KB (≈2× for fragmentation)peer middleware; alloc-based session/buffers
zenoh-pico (serial)lighter than TCP16–24 KBno TCP link buffers; verified running at 16 KB
XRCE (Micro-XRCE-DDS)~3 KB (micro-ROS figure)~8 KBstatic pools, discovery offloaded to the agent — the RAM-minimal backend; a measured bare-metal XRCE figure is pending an example (no bare-metal XRCE example ships yet — XRCE bare-metal needs a custom-transport injection)

Measured footprint

Honest, reproducible numbers per (platform, transport, backend, profile) — built with the in-tree examples, the release profile is cargo’s default (opt-3), the size profile is the scaffolded [profile.size] (opt-s + lto

  • strip, see Phase 204.3). RAM = data + bss. All cells are after --gc-sections + the size knobs above are applied where noted; the serial cell ships with the recipe below.
platformtransportbackendprofiletext (flash code)databssRAM total
qemu-arm-baremetal (mps2-an385, cortex-m3)ethernet (smoltcp)zenoh-picorelease177.4 KB67.0 KB91.7 KB158.7 KB
qemu-arm-baremetalethernetzenoh-picosize158.3 KB67.0 KB91.7 KB158.7 KB
qemu-arm-baremetalserial (no IP stack)zenoh-picorelease128.6 KB25.2 KB75.8 KB101.0 KB
qemu-arm-baremetalserialzenoh-picosize + recipe116.1 KB25.2 KB75.8 KB101.0 KB
stm32f4 (thumbv7em-eabihf, cortex-m4)ethernetzenoh-picorelease186.9 KB13.7 KB123.0 KB136.7 KB
stm32f4ethernetzenoh-picosize138.1 KB13.7 KB123.0 KB136.7 KB
qemu-arm-freertos (cortex-m3 + lwIP, RTOS-reused stack)ethernet (lwIP)zenoh-picorelease240.6 KB10.7 KB3.3 MB3.3 MB
qemu-arm-baremetal (Phase 207)serial (custom XRCE transport)XRCEsize, heap 24 KB, tight XRCE pools60.3 KB25.2 KB (heap 24 KB)8.8 KB~34 KB
micro-ROS reference (XRCE)serialXRCE-DDS Client-Os< 75 KB~3 KB~3 KB peak

The XRCE row uses the Phase 207.6 tight per-session pools — set in the example’s .cargo/config.toml [env] and read by nros-rmw-xrce-cffi’s build.rs: NROS_XRCE_STREAM_HISTORY=4, NROS_XRCE_CUSTOM_TRANSPORT_MTU=512, NROS_XRCE_MAX_SUBSCRIBERS=1, NROS_XRCE_MAX_SERVICE_SERVERS=1, NROS_XRCE_MAX_SERVICE_CLIENTS=1, NROS_XRCE_SUBSCRIBER_RING_DEPTH=1, NROS_XRCE_BUFFER_SIZE=256. Vendor defaults grow xrce_session_state_t to ~390 KB (which wouldn’t fit a 24 KB heap); these knobs drop it to ~12 KB. Defaults are unchanged for hosted / non-tight-RAM consumers — the env vars are pure opt-in.

How to read this:

  • The size profile (opt-s) shrinks .text by ~10–26 % with .bss/.data unchanged (opt-level doesn’t touch static buffers — those are the env knobs above). -Oz is not used — on smoltcp examples it grows .bss +24 KB by defeating opt-3’s per-socket dead-buffer DCE (see Phase 204.3).
  • Switching ethernet → serial sheds ~50 KB text + ~42 KB .data (no smoltcp stack, no IP link C, tuned heap) — the structural lever.
  • FreeRTOS + lwIP cells .bss is dominated by lwIP’s heap + FreeRTOS task stacks (3 MB is the configured headroom, not nano-ros overhead).
  • The micro-ROS / XRCE row is a reference, not a nano-ros measurement — no bare-metal XRCE example ships yet (needs a custom-transport injection); the path to parity is XRCE + serial + static pools.

Size-minimal recipe

Smallest measured nano-ros configuration today (qemu-arm-baremetal serial talker, 116 KB text / 101 KB RAM):

# Cargo.toml
[profile.size]
inherits = "release"
opt-level = "s"
lto = "fat"
codegen-units = 1
debug = false
strip = true
# .cargo/config.toml — gc + serial knobs
[target.thumbv7m-none-eabi]
rustflags = [
    "-C", "link-arg=--gc-sections",   # 204.8 — strip unreferenced fns/data
    "-C", "link-arg=-Tlink.x",
]

[env]
NROS_LINK_IP        = "0"      # 204.7 — drop zenoh-pico TCP/UDP link C
ZPICO_NO_SMOLTCP    = "1"      # skip smoltcp glue on bare-metal
NROS_HEAP_SIZE      = "24576"  # 204.5 — right-size for zenoh-pico working set
NROS_SMOLTCP_MAX_SOCKETS     = "1"   # 204.2 — brokered client multiplexes
NROS_SMOLTCP_MAX_UDP_SOCKETS = "1"

Build with cargo build --profile size, or fleet-wide via NROS_CARGO_PROFILE=size just <plat> build. nros new --platform baremetal already scaffolds the [profile.size] + the .cargo/config.toml shape (Phase 204.7/204.8); uncomment the serial block when you swap to a serial transport.

The deeper RAM win waits on XRCE on bare-metal (the ~3 KB-class client + static pools, with discovery offloaded to the agent) — tracked separately; zenoh-pico’s SUBSCRIBER_BUFFERS + alloc-based session are what keep this row’s .bss ~76 KB.

Cargo features (which RMW/platform is linked)

Features select the linked RMW backend, platform, and ROS edition. The nros.toml node.rmw picks which linked backend is active — the two are different layers (link vs run). Matrix + mutual-exclusion rules: Platform Model.

[dependencies]
nros = { path = "…/nros", default-features = false, features = [
    "rmw-cffi",            # generic C-vtable runtime registry
    "platform-bare-metal", # or platform-{freertos,nuttx,threadx,zephyr,posix}
    "ros-humble",          # or ros-iron
    "std", "alloc",        # optional, target-dependent
] }
# Exactly one RMW backend crate; its registration runs before main:
nros-rmw-zenoh = { path = "…/nros-rmw-zenoh", features = ["platform-bare-metal", "link-tcp", "ros-humble"] }
# …or nros-rmw-xrce-cffi / the cyclonedds CMake backend

Runtime environment (POSIX only)

On Linux/*BSD, ExecutorConfig::from_env() reads at process start (embedded targets bake nros.toml instead):

VariableDescriptionDefault
ROS_DOMAIN_IDROS 2 domain ID0
NROS_LOCATORRouter address (legacy alias ZENOH_LOCATOR)tcp/127.0.0.1:7447
NROS_SESSION_MODEclient / peer (legacy alias ZENOH_MODE)client
ZENOH_TLS_ROOT_CA_CERTIFICATE*TLS CA cert (path / base64)(none)

By deployment scenario

Scenarionros.tomlCargo featuresNotes
Desktop (POSIX)— (uses env)rmw-cffi, platform-posix, std + zenoh depExecutorConfig::from_env(); run zenohd locally
QEMU bare-metal[[transport]] ethernet ip/mac/gatewayrmw-cffi, platform-bare-metal, ros-humble + zenohTAP/slirp bridge
FreeRTOS hardware+ [node.rt]…, platform-freertos, …FREERTOS_DIR/LWIP_DIR
ESP32 WiFi[[transport]] kind="wifi" ssid/password…, platform-bare-metal, …SSID/PASSWORD build env
Zephyr module(Kconfig overlay, not nros.toml)(Kconfig → features)prj-<rmw>.conf
Minimal RAM (XRCE serial)[[transport]] kind="serial" baudrate + xrce depXRCE_* buffer tuning

.env

cp .env.example .env   # uncomment + adjust; gitignored; auto-loaded by just + direnv

Logging

nano-ros ships a ROS 2 style leveled logging facade (nros-log) with the same Logger + severity surface as rclcpp::Logger / rcutils_logging. Records flow through a single per-platform sink (PlatformSink) that delegates to whichever nros-platform-<rtos> is linked into the binary.

Severity ladder

REP-2012 style, matching rcutils_log_severity_t:

Severityu8When to use
Trace0Per-instruction granularity; off by default.
Debug1Diagnostic information useful while developing.
Info2Normal operation events worth surfacing once.
Warn3Unexpected but recoverable conditions.
Error4Errors the caller should surface; system continues.
Fatal5Unrecoverable; system is about to abort.

The numeric representation is stable and used by the nros_platform_log_write ABI.

Quick start

Rust

use nros_log::{Logger, Severity};
use nros_log::{nros_info, nros_warn};

static LOGGER: Logger = Logger::new("my_node");

fn main() {
    nros_log::register_logger(&LOGGER);
    nros_log::init(nros_log::sinks::default());

    nros_info!(&LOGGER, "started; domain = {}", 42);
    nros_warn!(&LOGGER, "queue depth {} exceeds soft limit", 5);
}

Inside a node, prefer the Node::logger() accessor — it resolves to the same intern’d entry when the name matches:

#![allow(unused)]
fn main() {
let mut node = executor.create_node("my_node")?;
nros_info!(node.logger(), "subscribed to {}", topic);
}

C

#include <nros/init.h>
#include <nros/node.h>
#include <nros/log.h>

int main(void) {
    nros_support_t support = nros_support_get_zero_initialized();
    nros_support_init(&support, "tcp/127.0.0.1:7447", 0);

    nros_node_t node = nros_node_get_zero_initialized();
    nros_node_init(&node, &support, "my_node", "/");

    nros_logger_t logger = nros_node_get_logger(&node);
    NROS_LOG_INFO(logger, "started; domain=%u", 42);
    NROS_LOG_WARN(logger, "queue depth %u exceeds soft limit", 5);

    nros_node_fini(&node);
    nros_support_fini(&support);
    return 0;
}

C++

#include <nros/nros.hpp>
#include <nros/log.hpp>

int main() {
    nros::init("tcp/127.0.0.1:7447", 0);

    nros::Node node;
    nros::create_node(node, "my_node");

    auto logger = node.get_logger();
    NROS_LOG_INFO(logger, "started; domain=%u", 42);
    NROS_LOG_WARN(logger, "queue depth %u exceeds soft limit", 5);

    nros::shutdown();
    return 0;
}

The first NROS_LOG_* emit auto-installs PlatformSink so C/C++ call sites work without an explicit init step.

Per-platform delivery

TargetBackend
POSIXfprintf(stderr, "[<LEVEL>] <name>: <msg>\n")
ZephyrLOG_INF / LOG_WRN / etc. (or printk if CONFIG_LOG=n); module nros
ESP-IDFesp_log_write with logger-name = ESP TAG
NuttXsyslog(priority, "%s", buf)
FreeRTOSboard-registered UART writer fn-ptr
ThreadXboard-registered UART writer fn-ptr
Bare-metal (mps2-an385)QEMU semihosting via cortex_m_semihosting::hio::hstderr
Bare-metal (stm32f4)defmt::info!("{=str}", msg)
Bare-metal (esp32 / esp32-qemu)board-registered fn-ptr (esp-println / RTT / serial-jtag)

Boards on platforms with no native logger (FreeRTOS / ThreadX / bare-metal Rust ESP32) supply their writer fn-ptr at run() time:

#![allow(unused)]
fn main() {
// from nros-board-mps2-an385-freertos/src/lib.rs
fn register_log_writer() {
    unsafe extern "C" fn writer(severity: u8, name_ptr: *const u8, name_len: usize,
                                 msg_ptr: *const u8, msg_len: usize) { … }
    unsafe { nros_platform_register_log_writer(Some(writer), None); }
}
}

Filtering

Compile-time ceiling

Cargo features on nros-log (pick at most one; default max-level-trace):

FeatureMacros above ceiling that emit
max-level-tracetrace / debug / info / warn / error / fatal
max-level-debugdebug / info / warn / error / fatal
max-level-infoinfo / warn / error / fatal
max-level-warnwarn / error / fatal
max-level-errorerror / fatal
max-level-off(none)

Below-ceiling macros expand to (); the format call is dead-code-eliminated.

Runtime per-logger threshold

#![allow(unused)]
fn main() {
let logger = nros_log::get_logger("my_node");
logger.set_level(nros_log::Severity::Warn);   // silences Trace/Debug/Info
}

Default = Severity::Info for any logger constructed via Logger::new(name).

Buffer size

nros-log formats each record into a stack-resident heapless::String<N>. N is picked at compile time by the buffer-size-<N> feature family (default 256). Overflow truncates and appends ; the macro never panics on a long format string.

FeaturePer-call-site stack frame
buffer-size-128128 B
buffer-size-256256 B (default)
buffer-size-512512 B
buffer-size-10241024 B

Interop with the log crate

Enable nros-log/log-compat:

#![allow(unused)]
fn main() {
nros_log::log_compat::install_log_crate_bridge()
    .expect("log crate already initialized");
// Now `log::info!(...)` calls from ecosystem crates flow through
// nros-log's dispatcher.

// Or fan out the other direction — re-emit nros records via `log`:
static SINKS: &[&dyn nros_log::LogSink] = &[
    &nros_log::sinks::PlatformSink,
    &nros_log::log_compat::LogCrateSink,
];
nros_log::init(SINKS);
}

Severity maps Trace↔Trace, Debug↔Debug, Info↔Info, Warn↔Warn, Error↔Error round-trip. Fatal folds into Error one-way (log has no Fatal).

Working examples

  • Rust: examples/native/rust/logging/
  • C: examples/native/c/logging/
  • C++: examples/native/cpp/logging/

Each demonstrates per-severity macros + the runtime threshold filter.

Reference

  • packages/core/nros-log/ — facade crate.
  • packages/core/nros-platform-cffi/include/nros/platform.hnros_platform_log_write / nros_platform_log_flush / nros_platform_register_log_writer ABI.
  • packages/core/nros-c/include/nros/log.h — C API surface (NROS_LOG_* macros + nros_log_emit_fmt).
  • packages/core/nros-cpp/include/nros/log.hpp — C++ macros (the legacy NROS_INFO / etc. file:line printf surface stays alongside the new NROS_LOG_* macros).

/rosout publication is explicitly out of scope today; the dispatcher’s &'static [&dyn LogSink] shape leaves room for an add-later RosoutSink that consumes records alongside PlatformSink.

Profiling Your Build

When a build feels slow, nros-build-profile tells you where the time went — codegen vs compile vs link vs flash, which unit dominated, and whether a shared crate was rebuilt redundantly.

It is a passive, read-only tool. Your build runs exactly as today on its native toolchain (west, cmake, idf.py, cargo); the profiler only reads the timing artifacts that build already produced. It never compiles or flashes anything, and it adds no nros build/test command — nros stays setup + codegen.

How it works

Every backend already emits timing data:

BackendArtifactOpt-in
west, cmake, idf.pybuild*/.ninja_lognone — ninja always writes it
cargo (native, esp32, cross)target*/cargo-timings/cargo build --timings

The profiler discovers and parses these, then prints a stage table (always) plus an optional per-unit drill-down and diagnostic hints.

Flow 1 — Zephyr / cmake / esp32-idf (no opt-in)

$ west build -b qemu_cortex_a53 examples/zephyr/rust/talker   # build/.ninja_log written
$ just profile examples/zephyr/rust/talker

Backend: ninja (west)        Total: 41.2s

Stage      Duration    %
codegen        1.1s   3%
compile       33.8s  82%
link           6.0s  15%

hints:
  - 1 unit = 61% of compile (libzenoh_pico.a, 20.6s)

Flow 2 — cargo (one flag for per-crate detail)

The coarse table works with any build; the per-crate drill-down needs --timings:

$ cd examples/native/rust/talker && cargo build --timings   # target/cargo-timings/ written
$ just profile examples/native/rust/talker --deep

Backend: cargo               Total: 30.3s

Stage      Duration    %
codegen        1.2s   4%
compile       24.8s  82%
link           3.9s  13%

slowest units:
  zenoh-pico-sys     18.1s ########
  nros-node           2.4s #
  std_msgs            1.1s #
  <80 more>           3.2s

hints:
  - nros-c compiled 3× — examples use isolated target/; pool target_dir to reuse the build

If you build cargo without --timings, you still get the coarse table plus a note telling you how to enable the drill-down — never an error.

Flow 3 — machine-readable (CI regression diffing)

$ just profile examples/zephyr/rust/talker --deep    # then add --json on the bin, or:
$ ./target/debug/nros-build-profile examples/zephyr/rust/talker --json
wrote examples/zephyr/rust/talker/nros-build-profile.json

A CI step can diff nros-build-profile.json across commits to catch build-time regressions (a stage’s seconds or a unit’s share jumping).

External (copy-out) projects

Copy-out example projects have no justfile. Run the bin directly — same output:

$ nros-build-profile <project-dir> --deep

Diagnostics

On top of the numbers, the profiler flags known slow patterns:

  • a large native (C/C++/-sys) unit with no incremental → suggest a compiler cache;
  • the same crate compiled more than once → suggest pooling target_dir;
  • the configured job count vs available RAM (the fixture-build OOM budget).

Silence them with --no-hints.

Deployment Workflow

Deployment means different things per target, but the order is stable: prepare toolchain, build package, move binary/firmware to target, then verify ROS 2 communication.

POSIX

Three equivalent entry points; pick by workspace shape:

# Per-example (Pattern B or any single binary):
cd examples/native/rust/talker
cargo run

# Multi-component system orchestration:
nros metadata my_system
nros plan my_system launch/my_system.launch.py
nros check
cargo run -p robot_entry

# Colcon consumer workspace (Pattern A):
colcon build && source install/setup.bash
ros2 run my_pkg my_node

For interop with stock ROS 2 over Zenoh, run the bundled router (built by just zenohd setup) and point ROS 2 at it:

zenohd --listen tcp/127.0.0.1:7447
export RMW_IMPLEMENTATION=rmw_zenoh_cpp

See Native POSIX.

RTOS and Bare-Metal

RTOS targets usually produce firmware images or simulator binaries:

just freertos build
just freertos test

For real hardware, deployment step becomes flash/load/monitor. For QEMU, deployment is launching simulator with correct network setup.

There is no nros deploy / nros build / nros run verb — Phase 222 removed those wrappers. nros is provisioner + codegen + metadata only; deployment runs on the vendor’s native tools. The embedded deploy contract is a documented three-step sequence (per RFC-0003 §4):

  1. Bakenros codegen-system --bringup <pkg> reads system.toml + [deploy.<board>] + launch/*.xml and emits the baked tree under build/<board>/.
  2. Build — the vendor tool builds it: cargo build / cmake --build / west build / idf.py build (the just <plat> build* recipes wrap these with the right -D args derived from [deploy.<board>]).
  3. Flash + monitor — the vendor tool again: probe-rs run / west flash / idf.py flash monitor, or the platform’s QEMU runner.

Platform guides should show:

  • package layout,
  • setup command,
  • toolchain requirements,
  • build command,
  • run/flash command,
  • ROS 2 interop or smoke-test command.

Zephyr

Zephyr deployment uses west:

just zephyr setup
source zephyr-workspace/env.sh
west build -b native_sim/native/64 nros/examples/zephyr/rust/talker
./build/zephyr/zephyr.exe

ESP32

ESP32 deployment uses the Espressif toolchain and flash tool:

just esp32 build
just esp32 talker

For physical boards, use the platform guide’s espflash path.

Verify

After deployment, verify from ROS 2 side:

ros2 topic list
ros2 topic echo /chatter std_msgs/msg/Int32 --qos-reliability best_effort

If discovery works but samples do not arrive, check domain ID, router mode, QoS reliability, and platform network setup.

ROS 2 Interop

nano-ros communicates with standard ROS 2 nodes using the rmw_zenoh_cpp middleware. Both sides connect to the same zenohd router and exchange CDR-encoded messages with compatible key expressions.

Prerequisites

  • ROS 2 Humble installed
  • rmw_zenoh_cpp package (sudo apt install ros-humble-rmw-zenoh-cpp on Ubuntu, or build from source)
  • zenohd router running

Quick Start

Open three terminals:

# Terminal 1: Start the zenoh router
zenohd --listen tcp/127.0.0.1:7447
# Terminal 2: Run the nano-ros talker
cd examples/native/rust/talker
RUST_LOG=info cargo run
# Terminal 3: Run a ROS 2 listener
source /opt/ros/humble/setup.bash
export RMW_IMPLEMENTATION=rmw_zenoh_cpp
export ZENOH_CONFIG_OVERRIDE='mode="client";connect/endpoints=["tcp/127.0.0.1:7447"]'
ros2 topic echo /chatter std_msgs/msg/Int32 --qos-reliability best_effort

You should see the ROS 2 listener printing messages published by nano-ros:

data: 1
---
data: 2
---

The Other Direction

ROS 2 publishers also work with nano-ros subscribers:

# Terminal 2: Run the nano-ros listener instead
cd examples/native/rust/listener
RUST_LOG=info cargo run

# Terminal 3: ROS 2 talker
source /opt/ros/humble/setup.bash
export RMW_IMPLEMENTATION=rmw_zenoh_cpp
export ZENOH_CONFIG_OVERRIDE='mode="client";connect/endpoints=["tcp/127.0.0.1:7447"]'
ros2 topic pub /chatter std_msgs/msg/Int32 "{data: 42}" --qos-reliability best_effort

Discovery

nano-ros publishes ROS 2-compatible liveliness tokens, so standard ROS 2 tools work:

source /opt/ros/humble/setup.bash
export RMW_IMPLEMENTATION=rmw_zenoh_cpp
export ZENOH_CONFIG_OVERRIDE='mode="client";connect/endpoints=["tcp/127.0.0.1:7447"]'

ros2 topic list        # Shows /chatter
ros2 topic info /chatter   # Shows publisher/subscriber count
ros2 node list         # Shows nano-ros nodes

Configuration

Domain ID

Both sides must use the same ROS domain ID. nano-ros reads ROS_DOMAIN_ID from the environment (default: 0):

ROS_DOMAIN_ID=42 cargo run    # nano-ros side
ROS_DOMAIN_ID=42 ros2 topic echo /chatter ...  # ROS 2 side

QoS

nano-ros defaults to BEST_EFFORT reliability and VOLATILE durability. When using ROS 2 subscribers, match the QoS:

ros2 topic echo /chatter std_msgs/msg/Int32 --qos-reliability best_effort

Without --qos-reliability best_effort, ROS 2 defaults to RELIABLE, which won’t match the nano-ros publisher’s BEST_EFFORT QoS.

rmw_zenoh Client Mode

By default, rmw_zenoh_cpp uses peer mode and won’t connect to a zenohd router. Force client mode with:

export ZENOH_CONFIG_OVERRIDE='mode="client";connect/endpoints=["tcp/127.0.0.1:7447"]'

Set this before any ros2 command.

Protocol Details

nano-ros uses the same wire format as rmw_zenoh_cpp:

  • Data key expression: <domain_id>/<topic>/<type>/TypeHashNotSupported (Humble). Example: 0/chatter/std_msgs::msg::dds_::Int32_/TypeHashNotSupported
  • Liveliness tokens: @ros2_lv/<domain>/<zid>/0/... for discovery
  • Message encoding: CDR little-endian with 4-byte encapsulation header [0x00, 0x01, 0x00, 0x00]
  • RMW attachment: 33-byte metadata (sequence number, timestamp, GID) appended via Zenoh attachment

ROS 2 Iron and Beyond

Iron uses actual type hashes (RIHS01_<sha256>) instead of TypeHashNotSupported. nano-ros currently supports Humble only. Iron support is planned.

Troubleshooting

Topic not visible in ros2 topic list: Ensure both nano-ros and ROS 2 use the same domain ID and that rmw_zenoh_cpp is in client mode pointing to the same router.

Discovery works but no messages received: Check that the QoS matches. Use --qos-reliability best_effort on the ROS 2 side.

rmw_zenoh not connecting: Verify ZENOH_CONFIG_OVERRIDE is set. Without it, rmw_zenoh uses peer mode and won’t find the router.

Next Steps

RMW Backends: Zenoh, XRCE-DDS, Cyclone DDS

nano-ros supports three RMW (ROS Middleware) backends for connecting embedded devices to a ROS 2 network. Each backend targets different deployment scenarios and resource constraints. Each Node picks its backend at build time; one binary can link multiple backends and bridge between them — see Cross-backend Bridges for the multi-RMW pattern.

Zenoh (rmw-zenoh)

The Zenoh backend uses zenoh-pico, a lightweight C client for the Zenoh protocol. The MCU participates directly in the Zenoh network – there is no protocol translation layer.

How it works:

  1. The MCU runs zenoh-pico in client mode, connecting to a zenohd router process over TCP, UDP, or TLS.
  2. zenoh-pico creates publishers and subscribers directly on the Zenoh network.
  3. ROS 2 nodes running rmw_zenoh_cpp connect to the same zenohd router, enabling transparent interop.
  4. In peer mode, two zenoh-pico devices can communicate directly without any router.

Key characteristics:

  • Peer-to-peer capable (no mandatory bridge process)
  • zenohd is a generic router, not a protocol translator – it forwards messages without interpreting them
  • If zenohd crashes, peers in client mode lose routing but the MCU continues running
  • Full ROS 2 graph discovery via liveliness tokens
  • Transport options: TCP, UDP, TLS (via zpico-smoltcp or platform sockets)

XRCE-DDS (rmw-xrce)

The XRCE-DDS backend uses Micro-XRCE-DDS-Client, the same client library used by micro-ROS. It follows an agent-based model where a lightweight client on the MCU delegates entity creation to an agent process.

How it works:

  1. The MCU runs the XRCE-DDS client, connecting to an Agent process over UDP or serial (UART).
  2. The client sends requests to the Agent: “create a publisher on topic X with type Y.”
  3. The Agent creates full DDS entities on behalf of the client and bridges data between the XRCE protocol and the DDS data space.
  4. ROS 2 nodes using any DDS-based RMW (FastDDS, Cyclone DDS) communicate through the Agent.

Key characteristics:

  • Agent is mandatory – the MCU cannot participate in the network without it
  • If the Agent crashes, the MCU loses all connectivity
  • Fully static memory allocation on the MCU (no heap required)
  • Client-side discovery is not supported; the Agent handles it
  • Transport options: UDP, serial (HDLC framing), CAN FD

Cyclone DDS (rmw-cyclonedds)

Maturity status. Cyclone DDS support is pub/sub-only today. Service create / recv / reply returns NROS_RMW_RET_UNSUPPORTED; status events (liveliness, deadline-miss, etc.) are not wired to Cyclone listeners yet. Wire-level interop with stock rmw_cyclonedds_cpp (Humble) works for topic publishing and subscribing. If your fleet needs RPC or lifecycle events over Cyclone, use Zenoh instead until the gaps close. Full known-limitations list: docs/reference/cyclonedds-known-limitations.md.

The Cyclone DDS backend uses Eclipse Cyclone DDS, the same DDS implementation that ROS 2 ships with via rmw_cyclonedds_cpp. Built as a standalone C++ library at packages/dds/nros-rmw-cyclonedds/ that registers itself with the runtime through the C ABI vtable in nros-rmw-cffi.

How it works:

  1. The application (or platform) calls nros_rmw_cyclonedds_register() once before nros::init() — this is automatic when the consumer’s CMake build sets -DNANO_ROS_RMW=cyclonedds.
  2. The runtime stores the vtable pointer; subsequent nros::init() calls dispatch through it for session, publisher, subscriber, and service operations.
  3. dds_create_domain / dds_create_participant / dds_create_topic / dds_create_writer / dds_create_reader are invoked under the hood; Cyclone owns its own RX threads.
  4. ROS 2 nodes using stock rmw_cyclonedds_cpp interoperate directly — same wire protocol, same discovery, no key rewriting (unlike rmw_zenoh’s <domain>/<topic>/<type>/... scheme).

Key characteristics:

  • Pure-C++ backend (not a Cargo crate) — Autoware contributors can read and extend the wrapper using the same patterns as actuation_module/include/common/dds/.
  • ROS 2 Tier-1 wire compat — pinned to Cyclone DDS tag 0.10.5 to match ros-humble-cyclonedds 0.10.5 + ros-humble-rmw-cyclonedds-cpp 1.3.4.
  • Static ddsi_config via dds_create_domain_with_rawconfig skips the XML parser; embedded-friendly.
  • Discovery via SPDP multicast or unicast peer list (mirrors Cyclone’s standard config knobs).
  • Heap required (Cyclone uses malloc); BUILD_SHARED_LIBS=ON produces libddsc.so for POSIX, static link for embedded.
  • No services / actions yet — service create/recv/reply currently returns NROS_RMW_RET_UNSUPPORTED.
  • No status events yetregister_subscriber_event / register_publisher_event / assert_publisher_liveliness slots are not wired to Cyclone listeners yet.

Build:

just cyclonedds setup       # build Cyclone DDS from third-party/dds/cyclonedds (tag 0.10.5)
just cyclonedds build-rmw   # build packages/dds/nros-rmw-cyclonedds
just cyclonedds test        # run the CTest harness

Each example picks its RMW via -DNANO_ROS_RMW=cyclonedds at configure time; the root CMakeLists.txt add_subdirectory’s packages/dds/nros-rmw-cyclonedds/ and links the resulting target into NanoRos::NanoRos. No build/install/ prefix, no find_package(NrosRmwCyclonedds) deleted both.

Availability follows from config — the build provisions Cyclone. Selecting cyclonedds (-DNANO_ROS_RMW=cyclonedds / --features rmw-cyclonedds) is your whole responsibility; nros_provide_cyclonedds() resolves the library: a prebuilt install you point at (-DCMAKE_PREFIX_PATH=<install> / -DCycloneDDS_DIR=), else self-provision from source — by default the pinned third-party/dds/cyclonedds submodule, or your own checkout via -DCYCLONEDDS_SOURCE_DIR=<path>. So a bare cmake/cargo build needs no just cyclonedds pre-step; freertos / threadx-rv64 / native examples build Cyclone on demand (sccache-accelerated), gated on the relevant cross toolchain. On a tier without the toolchain, the embedded-Cyclone tests are filtered out of test-all (skipped, not failed). idlc is a host tool, found on PATH (e.g. a ROS 2 install) or via -DIDLC_EXECUTABLE=. See docs/development/sdk-tiers.md § “CycloneDDS — self-provisioned in CMake”.

Known limitations. Cyclone DDS currently has a 2× CDR roundtrip per message, deferred status-event wiring, pending service request-id correlation, and incomplete Cortex-A/R Zephyr board support. See docs/reference/cyclonedds-known-limitations.md for the full list. ARMv8-R toolchain prep (Cortex-A 64-bit FVP, Cortex-R52 hardware) is in docs/reference/zephyr-armv8r-setup.md.

Comparison

AspectZenoh (rmw-zenoh)XRCE-DDS (rmw-xrce)Cyclone DDS (rmw-cyclonedds)
Client RAM~16 KB+ (heap required)~3 KB (fully static)~32 KB+ (heap required)
Client Flash~100 KB+~75 KB~150 KB+ (libddsc.so ~1.4 MB on POSIX, sized down on embedded link)
Bridge processzenohd (generic router)Agent (protocol translator)None — RTPS multicast directly
Peer-to-peerYes (no router needed)No (agent always required)Yes (RTPS native)
DiscoveryClient participatesAgent handles on behalfSPDP / SEDP on UDP multicast or static peer list
Entity creationClient creates directlyClient requests, agent createsClient creates directly
Transport optionsTCP, UDP, TLSUDP, serial, CAN FDUDP unicast + multicast (RTPS)
Heap allocationRequired (C-level)NoneRequired (Cyclone uses malloc)
ImplementationRust + zenoh-pico CRust + Micro-XRCE-DDS-Client CC++ wrapper over upstream Cyclone DDS C
ROS 2 interopVia rmw_zenoh_cpp + zenohdVia Agent + any DDS RMWDirect against rmw_cyclonedds_cpp (same upstream version)
Failure modeRouter crash = lose routingAgent crash = lose connectivityPeer goes offline = its samples stop arriving
C source files~100+28Upstream Cyclone (~600+ files, vendored unchanged via submodule)

Multi-backend binaries (bridges)

A single nano-ros binary can link more than one RMW backend and forward traffic between them. The bridge pattern is useful for:

  • Translating between protocols — a gateway node running zenoh ingress + DDS egress lets MCU fleets on zenoh-pico talk to an Autoware stack on Cyclone DDS without a separate translator.
  • Hard-real-time + best-effort split — a high-priority Node on Cyclone DDS for control loops, a low-priority Node on Zenoh for telemetry, both in one process.
  • Bringing up an XRCE Agent-free fleet — bridge XRCE devices to a Zenoh network so they look like first-class participants to stock ROS 2.

The pattern uses Executor::open_with_rmw("<name>", ...) to pin the primary session and node_builder("name").rmw("<other>").build() to open additional sessions on other backends. Both backends must be in the binary’s link line (Cargo manifest deps + register() call each).

A worked example lives at examples/bridges/native-rust-zenoh-to-dds/; the time-triggered variant under examples/bridges/tt-zenoh-to-xrce/ shows the same pattern under an ARINC-653-style cyclic schedule.

Full walkthrough: Cross-backend Bridges.

RMW Selection

RMW selection is a declared, language-agnostic, per-deploy value — you set it once in system.toml ([system].rmw, optionally overridden per target by [deploy.<t>].rmw) or via a CLI/build flag, and the toolchain lowers that declaration to each language’s native build mechanism: a Rust cargo rmw-<x> feature, or a CMake -DNANO_ROS_RMW cache var. The cargo feature and the CMake var below are the lowering targets the build uses — not the user-facing knob. A binary links the common rmw-cffi runtime plus exactly one backend; the linked backend self-registers through the nros_rmw_vtable_t C ABI, and the registry walker resolves it at Executor::open. See RFC-0031 for the full selection-and-lowering model and precedence rules.

Cargo.toml (Rust)

Lowered form of the declared RMW: add nros with the platform feature plus exactly one nros-rmw-<x> shim dep (cyclonedds is the exception — see below):

# Zenoh backend
[dependencies]
nros = { path = "<...>/packages/core/nros",
         default-features = false,
         features = ["std", "rmw-cffi", "platform-posix"] }
nros-rmw-zenoh = { path = "<...>/packages/zpico/nros-rmw-zenoh",
                   features = ["std", "platform-posix", "ros-humble"] }

# XRCE-DDS backend
[dependencies]
nros = { path = "<...>/packages/core/nros",
         default-features = false,
         features = ["std", "rmw-cffi", "platform-posix"] }
nros-rmw-xrce-cffi = { path = "<...>/packages/xrce/nros-rmw-xrce-cffi",
                       features = ["std"] }

# Cyclone DDS backend — Rust runtime sees the generic rmw-cffi C-ABI
# vtable; actual Cyclone wiring lives C++-side under
# packages/dds/nros-rmw-cyclonedds/ and is selected at CMake
# configure time via -DNANO_ROS_RMW=cyclonedds. The Rust
# manifest only carries the `rmw-cffi` feature; no Rust shim dep.
[dependencies]
nros = { path = "<...>/packages/core/nros",
         default-features = false,
         features = ["std", "rmw-cffi", "platform-posix"] }

Each example also calls <backend>::register() from main() before Executor::open — this drags the rlib’s CGU into the binary so the linkme distributed-slice walker finds the backend. C/C++ builds rely on the CMake-emitted strong stub from nano_ros_link_rmw(... RMW <x>) instead.

For C++ consumers, the declared RMW lowers to the CMake cache var:

cmake -S . -B build -DNANO_ROS_RMW=cyclonedds  # zenoh / xrce / cyclonedds

Kconfig (Zephyr)

# Zenoh backend
CONFIG_NROS_RMW_ZENOH=y

# XRCE-DDS backend
CONFIG_NROS_RMW_XRCE=y

# Cyclone DDS backend (Cortex-A/R Zephyr targets)
CONFIG_NROS_RMW_CYCLONEDDS=y

Enabling more than one simultaneously produces a compile_error!().

Cyclone DDS — per-platform configuration profile

The Cyclone DDS backend speaks raw RTPS over UDP multicast and unicast. The host networking stack on every RTOS needs IGMP enabled, adequate net- buffer pool sizing for the SPDP / SEDP discovery burst, and a way to join a multicast group. Concrete deltas per platform:

Zephyr (platform-zephyr)

# IGMP for SPDP multicast discovery (239.255.0.1:7400+).
CONFIG_NET_IPV4_IGMP=y

# Multicast group slots — RTPS uses up to 4 builtin groups per
# participant (SPDP + SEDP pubs / subs / topics).
CONFIG_NET_IF_MCAST_IPV4_ADDR_COUNT=4

# Net-buffer pools — defaults of 14 / 36 are too small for the SEDP
# burst when more than two service / action entities exist on each
# participant. Symptom of undersizing: runtime starvation as the
# discovery exchange backs up. Cortex_a9 has 512 MB SRAM; the cost
# is trivial.
CONFIG_NET_PKT_RX_COUNT=256
CONFIG_NET_PKT_TX_COUNT=128
CONFIG_NET_BUF_RX_COUNT=512
CONFIG_NET_BUF_TX_COUNT=256
CONFIG_NET_BUF_DATA_SIZE=512

# Heap — Cyclone DDS (malloc) + heapless mailboxes.
CONFIG_COMMON_LIBC_MALLOC_ARENA_SIZE=4194304
CONFIG_HEAP_MEM_POOL_SIZE=524288

On qemu_cortex_a9, additionally use a board overlay to bump the GEM driver’s RX / TX descriptor ring (default 32) so a brief drainer pause doesn’t immediately spill:

&gem0 {
    promiscuous-mode;
    rx-buffer-descriptors = <128>;
    tx-buffer-descriptors = <64>;
};

native_sim is not yet supported — Zephyr’s NSOS driver doesn’t forward IP_ADD_MEMBERSHIP to the host kernel (no IPPROTO_IP case in nsos_adapt_setsockopt). Use qemu_cortex_a9 for DDS-on-Zephyr testing instead.

FreeRTOS + lwIP (platform-freertos)

// FreeRTOSConfig.h / lwipopts.h
#define LWIP_IGMP            1   // SPDP multicast
#define LWIP_SO_RCVTIMEO     1   // Cyclone DDS recv timeouts
#define LWIP_BROADCAST       1
#define IP_REASSEMBLY        1   // RTPS DATA_FRAG fragments
#define MEMP_NUM_NETBUF      32  // discovery burst headroom

NuttX (platform-nuttx)

CONFIG_NET_IGMP=y
CONFIG_NET_BROADCAST=y
CONFIG_NET_UDP_NRECVS=4
CONFIG_NET_RECV_TIMEO=y

ThreadX + NetX Duo (platform-threadx)

tx_user.h / nx_user.h:

#define NX_ENABLE_IGMPV2          // IGMP v2 for RTPS multicast
// NetX BSD layer init must call `bsd_initialize` early — required
// for `setsockopt(SO_RCVTIMEO)` support.

Bare-metal smoltcp (platform-mps2-an385, platform-stm32f4,

platform-esp32-qemu)

#![allow(unused)]
fn main() {
// Bridge config in the board crate.
let mut config = smoltcp::iface::Config::new(...);
config.multicast_groups = vec![Ipv4Address::new(239, 255, 0, 1)];
// MulticastConfig::Strict in smoltcp 0.x; the bridge must expose
// a `join_multicast_group` API.
}

POSIX (platform-posix)

No configuration needed — the kernel does IGMP and setsockopt support natively. Just ensure the loopback interface is up (default).

When to Use Which

Choose Zenoh when:

  • You need ROS 2 interop with rmw_zenoh_cpp (the recommended ROS 2 middleware for Jazzy+)
  • You want peer-to-peer communication without any bridge process
  • Your MCU has at least ~16 KB of heap and ~100 KB of flash
  • You are using TCP or UDP networking
  • You want simpler deployment (zenohd is a single binary with no configuration required)

Choose XRCE-DDS when:

  • Your MCU has very limited RAM (under 8 KB available for middleware)
  • You need serial (UART) transport – useful for MCUs without networking hardware
  • You are integrating with an existing micro-ROS or DDS deployment
  • You want zero heap allocation on the MCU side
  • You need CAN FD transport

Choose Cyclone DDS when:

  • You want direct, brokerless ROS 2 interop with no zenohd and no Agent
  • Your MCU has at least ~32 KB RAM (heap for Cyclone + RTPS state) and a network stack with IGMP
  • You’re integrating with a DDS-based ROS 2 deployment running stock rmw_cyclonedds_cpp and want full RTPS wire-compat without protocol translation
  • Your tolerance for failure is “samples from peer X stop arriving” rather than “the whole router goes down”
  • Note: Cyclone DDS is pub/sub-only today (no services / actions yet)

Any of the three works well for:

  • Standard pub/sub and service patterns
  • Integration with ROS 2 desktop nodes
  • QEMU-based development and testing

Cross-backend bridges

Most nano-ros applications pick one RMW backend at compile time and never look back. A cross-backend bridge is the one case where a single binary needs two backends in parallel — one node receiving on backend A, another publishing on backend B, the executor draining both each tick.

Typical reasons:

  • Drone link. A flight controller speaks XRCE-DDS over a serial radio link; the ground-station LAN runs standard DDS. A bridge forwards telemetry verbatim between the two.
  • Field gateway. Sensor pods talk zenoh-pico over an unreliable Wi-Fi mesh; the ROS 2 datacenter side wants Cyclone DDS. A bridge republishes after a session-buffered drop check.
  • Safety carve-out. Mission-critical traffic stays on the deterministic backend; introspection / logging republishes to a best-effort backend so dashboards never starve the critical pipeline.

This chapter walks the model, the build knobs, and the three shipped examples.

The mental model — rclcpp::Node, twice

nano-ros mirrors rclcpp:

Executor exec;                   // owns the primary session
Node ingress = exec.node_builder("ingress").rmw("xrce").build();
Node egress  = exec.node_builder("egress").rmw("cyclonedds").build();
auto sub = ingress.create_subscription<Foo>(...);
auto pub = egress.create_publisher<Foo>(...);

The executor opens one primary session at open* time. Each extra node_builder(...).rmw(name) call:

  1. Looks up name in the registry of linked backends.
  2. If name == primary_rmw and the locator matches → the Node binds to the primary session (slot 0). No second session opened.
  3. Else → opens a fresh session through that backend’s open_with_rmw and stores it in extra_sessions[N-1]. The new Node’s session_idx = N.

spin_once() drains every session in turn. Handles created through a multi-Session Node route through the node’s resolved session record instead of the legacy support/session pointer kept for single-backend callers.

NodeRecord.session_idx is the dispatch key. Print it to verify which session each Node landed on:

#![allow(unused)]
fn main() {
exec.node(node_in).unwrap().session_idx   // 0 = primary
exec.node(node_out).unwrap().session_idx  // 1 = first extra
}

Build knobs — three audiences, three shapes

Rust binary

Add both backend crates to Cargo.toml and call each register() early in main:

[dependencies]
nros            = { ..., features = ["rmw-cffi"] }
nros-rmw-zenoh  = { ... }
nros-rmw-xrce-cffi = { ... }
fn main() {
    nros_rmw_zenoh::register().expect("register zenoh");
    nros_rmw_xrce_cffi::register().expect("register xrce");
    let mut exec = nros::Executor::open_with_rmw("zenoh",
        &nros::ExecutorConfig::from_env())?;
    let ingress = exec.node_builder("ingress").rmw("zenoh").build()?;
    let egress  = exec.node_builder("egress").rmw("xrce").build()?;
    // ...
}

Why explicit register()? Stable Rust drops un-referenced rlib CGUs from the final link even when the backend’s #[used] static RMW_INIT_ENTRIES exists. The register() call doubles as the symbol reference that drags the rlib in and the registration trigger. C / C++ builds dodge this because --whole-archive pulls every section unconditionally.

C binary

NANO_ROS_RMW=none switches off the root CMake’s RMW auto-pull. Pull each backend’s staticlib through corrosion_import_crate and wrap with --whole-archive so the registry walker finds both names:

set(NANO_ROS_RMW none)
add_subdirectory(${nano_ros_root} nano_ros)

corrosion_import_crate(
    MANIFEST_PATH ${nano_ros_root}/Cargo.toml
    CRATES nros-rmw-xrce-cffi
    NO_DEFAULT_FEATURES
    FEATURES std)
corrosion_import_crate(
    MANIFEST_PATH ${nano_ros_root}/Cargo.toml
    CRATES nros-rmw-zenoh-staticlib
    NO_DEFAULT_FEATURES
    FEATURES "platform-posix;ros-humble")

target_link_libraries(my_bridge PRIVATE
    NanoRos::NanoRos
    -Wl,--whole-archive
    nros_rmw_xrce_cffi-static
    nros_rmw_zenoh_staticlib-static
    -Wl,--no-whole-archive)

Then declare + call the register functions early in nros_app_main:

extern int8_t nros_rmw_xrce_register(void);
extern int8_t nros_rmw_zenoh_register(void);

int nros_app_main(int argc, char** argv) {
    if (nros_rmw_xrce_register() != 0)  return 1;
    if (nros_rmw_zenoh_register() != 0) return 1;
    // ... nros_support_init / nros_executor_init / nros_executor_node_init
}

nros_executor_node_init honours per-Node rmw_name + locator via the nros_node_options_t struct; the executor sets node.executor so subsequent nros_*_init calls (publisher, subscription, service, action) route to the right session.

C++ binary

Same CMake shape as C. The C++ surface (nros-cpp) is a thin header layer; the linker work is identical. Use the Executor::node_builder(name) chain:

auto exec = nros::Executor::open_with_rmw("zenoh", cfg);
auto ingress = exec.node_builder("ingress").rmw("zenoh").build();
auto egress  = exec.node_builder("egress").rmw("xrce").build();
auto pub = egress.create_publisher<std_msgs::String>("/chatter");
exec.register_subscription_on<std_msgs::String>(ingress, "/chatter",
    [&pub](const auto& msg) { pub->publish(msg); });

NROS_RMW environment variable

When a binary links multiple backends, set NROS_RMW to pin the primary before open():

NROS_RMW=zenoh ./my_bridge

The C-side nros_support_init reads it and routes Executor::open through open_with_rmw(name, ...). Without this, the linkme walker returns whichever backend’s ctor fired first, which is non-deterministic across link orderings. The bridge then opens another session against the same backend when the per-Node .rmw(name) matches the unintended primary — and most singleton-state backends (XRCE-DDS-Client’s uxrSession, zenoh-pico’s global g_session) refuse a second open.

The Rust path mirrors this: Executor::open consults $NROS_RMW first; the per-Node resolve_session_slot detects the primary-name match and returns slot 0 instead of opening a duplicate.

Memory + WCET budget

Each extra session adds:

  • One ConcreteSession (RMW-specific; see RMW Backends for sizing).
  • One register_wake_signal_on_extra wake-callback slot (std only; bare-metal targets share the primary wake).
  • One round-trip through spin_once per backend per tick.

The bridge’s per-tick WCET is the sum across linked backends:

bridge_wcet = Σ poll_wcet_i + Σ dispatch_wcet_j

Read the per-backend numbers in the Real-time budget per backend table and add them up — there is no parallelism between backends inside a single executor.

For deadline-critical bridges, partition by SchedContext instead of running everything on the default Fifo slot:

#![allow(unused)]
fn main() {
let ingress = exec.node_builder("ingress")
    .rmw("xrce")
    .sched(critical_sc)   // RT priority
    .build()?;
let egress = exec.node_builder("egress")
    .rmw("cyclonedds")
    .sched(best_effort_sc)
    .build()?;
}

The PiCAS-style per-callback OS-priority dispatcher (gated behind the scheduler-os-priority feature) routes each Node’s callbacks to its own OS priority slot so the slow backend cannot block the fast one.

Shipped examples

examples/bridges/tt-zenoh-to-xrce/

Pure-Rust bridge. Zenoh ingress → XRCE-DDS egress under an ARINC-653-style time-triggered cyclic schedule. Read this first — it shows the multi-RMW Executor::open_with_rmw("zenoh", ...) plus node_builder.rmw("xrce") per-session pin, with raw byte forwarding and no codegen.

zenohd --listen tcp/127.0.0.1:7447 &
build/xrce-agent/MicroXRCEAgent udp4 -p 8888 &
NROS_XRCE_LOCATOR=udp/127.0.0.1:8888 \
    cargo run -p native-rs-bridge-tt-zenoh-to-xrce

examples/bridges/tt-zenoh-to-cyclonedds/

The stock-Cyclone-DDS sibling: same TT schedule, but the egress is .rmw("cyclonedds"), forwarding onto the DDS databus where a stock rmw_cyclonedds_cpp (e.g. an Autoware listener) or another nano-ros cyclonedds node receives the samples. One structural difference from the XRCE variant: Cyclone rejects a raw publisher whose topic type has no registered dds_topic_descriptor_t, so the egress type’s schema is staged before the raw publisher is created —

#![allow(unused)]
fn main() {
// The Cyclone backend installs the registrar during its register():
nros_rmw_cyclonedds_sys::register()?;
// Stage std_msgs/msg/String's schema (NUL-terminated key — it is handed
// straight to Cyclone's C descriptor table):
nros_rmw::register_type_descriptor(
    "std_msgs/msg/String\0",
    &[nros_serdes::schema::Field { name: "data\0", ty: FieldType::String, offset: 0 }],
)?;
let pub_out = node_out.create_publisher_raw("/chatter", "std_msgs/msg/String", hash)?;
}

XRCE registers lazily from name+hash; Cyclone needs the descriptor up front. The backend links the vendored CycloneDDS (no -DNANO_ROS_RMW needed for the Rust binary — the nros-rmw-cyclonedds-sys dep is the selection).

zenohd --listen tcp/127.0.0.1:7447 &
ROS_DOMAIN_ID=0 cargo run -p native-rs-bridge-tt-zenoh-to-cyclonedds
# subscribe on Cyclone DDS /chatter (stock ROS 2 or a nano-ros cyclone node on
# the same ROS_DOMAIN_ID) and publish on zenoh /chatter to see bridged samples.

Coverage matrix

Bridge examples live under examples/bridges/<name>/ (cross-platform, transport- spanning) or under their canonical examples/<plat>/<lang>/bridge/<name>/ cell when the bridge is platform-specific. The examples README coverage matrix lists which <plat> × <lang> combinations ship a bridge today (a bridge spans RMW backends by nature, so RMW is not a directory axis).

Troubleshooting

SymptomLikely cause
Transport(ConnectionFailed) on open_with_rmw("X", ...)Backend X’s rlib not pulled into the link line. Rust: add a register() call. C / C++: confirm --whole-archive wraps the staticlib.
Second node’s .rmw("zenoh") returns Transport(...)Both nodes try to open against the same singleton-state backend. Set NROS_RMW=zenoh so the primary lands on zenoh + the second Node hits the session-cache (slot 0) instead of double-opening.
nros_publisher_init -> -7 after nros_executor_node_initStale build. The C-side multi-Session dispatch in entity-init paths landed in commit 42001c37; rebuild nros-c after pulling main.
Bridge spinning marker never reaches piped test harnessAdd setvbuf(stdout, NULL, _IOLBF, 0) at the top of nros_app_main. glibc full-buffers piped stdout; line-buffering surfaces readiness markers before the long-lived spin_period loop.

See also

QoS, Status Events, and Discovery

This page covers runtime event behavior that application authors need after choosing an RMW backend: liveliness, deadline misses, lost messages, and the places where nano-ros deliberately differs from standard ROS 2 waitset-style event handling.

Transport-level status events let an application observe conditions the underlying middleware detects: a remote node going silent, a publisher missing its rate, a subscriber dropping samples. nano-ros exposes a Tier-1 subset useful for embedded RTOS deployments — liveliness changes, deadline misses, and message loss.

This page describes the event surface, the dispatch model, and the patterns that motivate having events at all on RTOS targets.

The Tier-1 events

Five event kinds. Three on subscribers, two on publishers.

Subscriber events

KindFires whenUse
LivelinessChangedA tracked publisher’s liveliness state changes (publisher started asserting / stopped)Safety-island fail-over: trigger MRM when remote control node goes silent
RequestedDeadlineMissedAn expected sample didn’t arrive within the configured deadlinePeriodic-sensor pattern: 100 Hz topic with 15 ms deadline; fire alarm when late
MessageLostThe backend dropped one or more samples (typically: ring-buffer overflow, slow consumer)Diagnostic + adaptive: log, drop more aggressively, coalesce

Publisher events

KindFires whenUse
LivelinessLostThis publisher missed its own liveliness assertion deadlineSelf-monitoring: detect own task starvation
OfferedDeadlineMissedThis publisher promised X Hz, fell behindSelf-monitoring: detect overrunning compute

Registering callbacks

Same shape as message callbacks. Register at construction time; callback fires from inside spin_once when the backend detects the event.

Rust

#![allow(unused)]
fn main() {
use core::time::Duration;

let mut sub = node.create_subscription::<SensorReading>("/sensor")?;

// Liveliness — fires when the publisher comes / goes
sub.on_liveliness_changed(|status| {
    if status.alive_count == 0 {
        log::warn!("sensor publisher went silent");
        trigger_failover();
    }
})?;

// Deadline — fires when no sample within 15 ms
sub.on_requested_deadline_missed(Duration::from_millis(15), |status| {
    metric_inc(&LATE_SAMPLE_COUNT, status.total_count_change);
})?;

// Message lost — fires when backend drops a sample
sub.on_message_lost(|status| {
    log::warn!("dropped {} samples", status.total_count_change);
})?;
}

Async equivalents (spin_async / Embassy / tokio):

#![allow(unused)]
fn main() {
let status = sub.next_liveliness_change().await?;
let status = sub.next_deadline_miss().await?;
}

C

static void on_liveliness_changed(
        nros_subscription_t *sub,
        nros_liveliness_changed_status_t status,
        void *user_context) {
    if (status.alive_count == 0) {
        trigger_failover();
    }
}

nros_subscription_set_liveliness_changed_callback(
    sub, on_liveliness_changed, NULL);

Same shape on the four other event kinds. Each returns NROS_RMW_RET_UNSUPPORTED if the active backend doesn’t generate that event.

C++

sub.on_liveliness_changed([&](nros::LivelinessChangedStatus status) {
    if (status.alive_count == 0) trigger_failover();
});

sub.on_requested_deadline_missed(
    std::chrono::milliseconds(15),
    [&](nros::DeadlineMissedStatus status) {
        late_count_ += status.total_count_change;
    });

std::function overloads available with NROS_CPP_STD; freestanding mode uses C function pointers + user-context.

How dispatch works

Events fire from inside drive_io on the executor thread, the same way message callbacks do:

spin_once(timeout)
 └─ session.drive_io(...)
       ├─ backend RX worker detects message → fires message callback
       ├─ backend RX worker detects liveliness change → fires liveliness callback
       ├─ backend timer expires (deadline) → fires deadline callback
       └─ return

Same execution context, same priority, same constraints as message callbacks. The max_callbacks_per_spin knob (see RTOS Cooperation) covers events too — an event callback counts as one against the cap.

This means events get the same scheduling treatment as messages. max_callbacks_per_spin = 1 will fire either one message OR one event per spin_once, whichever the backend has ready first.

Why a callback model, not a waitset

Upstream rmw.h exposes events via a waitset — register an rmw_event_t handle in a waitset, call rmw_wait, then rmw_take_event to pull payload. nano-ros doesn’t have a waitset (see RMW vs upstream Section 4).

Reusing the existing message-callback path avoids introducing the waitset. Trade-off: applications can’t bulk-poll for events. For the Tier-1 set this isn’t load-bearing — events are rare, callbacks are cheap.

Backend support is uneven

Not every backend can generate every event. Apps must handle Unsupported errors when registering:

BackendLivelinessChanged / LostDeadlineMissedMessageLost
Cyclone DDS✗ Not wired through the nano-ros event API yet✗ Not wired yet✗ Not wired yet
XRCE-DDS✗ Not exposed (xrce-dds-client API limitation)✅ Shim-side clock check (sub: has_data / try_recv_raw; pub: publish_raw)✗ Not exposed
zenoh-pico✅ Poll-based via zenoh tokens (alive_count ∈ {0,1})✅ Shim-side clock check (<P as PlatformClock>::clock_ms)✅ Seq-gap detection from RMW attachment
uORB✗ No wire-level liveliness✗ No rate concept✅ Native (host mock + real PX4 via RustSubscriptionCallback publish-counter)

✓ = wired and tested. 🟡 = surface API works (returns Err while pending), wiring planned. ✗ = not feasible at this layer.

No current backend exposes the full Tier-1 event set through the nano-ros event API. Apps should call Subscriber::supports_event(kind) first or design for graceful fallback.

The Subscriber::supports_event(kind) query lets applications check support before registering:

#![allow(unused)]
fn main() {
if sub.supports_event(EventKind::RequestedDeadlineMissed) {
    sub.on_requested_deadline_missed(...)?;
} else {
    // fallback: app-side timeout monitoring
}
}

Apps that need cross-backend portability code defensively. Apps pinned to one backend can call register-and-unwrap.

What’s deliberately skipped

Three upstream event types are intentionally absent from the API:

*_MATCHED (publication / subscription matched)

Fires when a remote endpoint appears or disappears. Useful for discovery-tracking dashboards and dynamic-topology apps. Static- topology embedded apps don’t benefit; if a use case appears, the event kind is additive.

*_QOS_INCOMPATIBLE

Fires when publisher and subscriber QoS profiles can’t be reconciled. nano-ros surfaces this synchronously at create_publisher / create_subscriber time as NROS_RMW_RET_INCOMPATIBLE_QOS. No runtime event needed; the failure is visible at startup.

*_INCOMPATIBLE_TYPE

Type-hash mismatch. Same: surfaced synchronously as NROS_RMW_RET_TOPIC_NAME_INVALID (or a future NROS_RMW_RET_INCOMPATIBLE_TYPE) at creation. No runtime event.

Patterns

Drone-bridge fail-over on liveliness loss

#![allow(unused)]
fn main() {
let mut sub = node.create_subscription::<VehicleAttitude>("/vehicle_attitude")?;
sub.on_liveliness_changed(|status| {
    if status.alive_count == 0 {
        // PX4 commander.cpp went silent — trigger MRM
        request_minimum_risk_manoeuvre();
    } else if status.alive_count_change > 0 {
        // commander came back online; clear MRM if appropriate
        clear_failover();
    }
})?;
}

This pairs with the cross-backend bridge example pattern.

100 Hz sensor with deadline alarm

#![allow(unused)]
fn main() {
let mut sub = node.create_subscription::<SensorReading>("/imu")?;
sub.on_requested_deadline_missed(
    Duration::from_millis(15),    // expected 100 Hz, allow 15 ms deadline
    |status| {
        if status.total_count_change > 0 {
            log::error!("IMU late: {} missed deadlines", status.total_count_change);
            // Optional: enter degraded-mode controller
        }
    },
)?;
}

Slow-consumer logging

#![allow(unused)]
fn main() {
let mut sub = node.create_subscription::<Pointcloud>("/lidar")?;
sub.on_message_lost(|status| {
    log::warn!("dropped {} pointcloud frames (total: {})",
               status.total_count_change, status.total_count);
    // Tighten downstream filter, reduce work, coalesce, etc.
})?;
}

See also

Serial Transport

This guide covers using serial (UART) transport with nros on embedded targets. Serial transport connects an MCU directly to a zenoh router over UART, without needing an IP stack.

Overview

nros supports two transport mechanisms for connecting embedded devices to a zenoh network:

TransportCrateUse Case
TCP/UDP (Ethernet/WiFi)zpico-smoltcpMCUs with Ethernet MAC or WiFi radio
Serial (UART)zpico-serialMCUs with only UART, or point-to-point links

Serial transport uses zenoh-pico’s built-in COBS framing protocol over UART. No IP stack is required — the MCU sends and receives zenoh frames directly over a serial link to a host running zenohd.

When to Use Serial

  • Small MCUs — Cortex-M0/M0+ with no Ethernet MAC and insufficient RAM for smoltcp
  • Point-to-point — Direct UART connection to a host, no network infrastructure needed
  • Debugging — Serial output is visible in any terminal, easy to inspect
  • Mixed topology — Some nodes on Ethernet, others on serial, all bridged through zenohd

Architecture

┌──────────┐     UART      ┌──────────────┐     TCP      ┌──────────┐
│   MCU    │───────────────│   zenohd     │─────────────│  ROS 2   │
│  (nros)  │  COBS frames  │  (router)    │  zenoh net   │  node    │
└──────────┘               └──────────────┘              └──────────┘

The MCU connects to zenohd using a serial/... locator. zenohd bridges serial-connected nodes to the rest of the zenoh network (including ROS 2 nodes using rmw_zenoh).

Platform Support

Serial transport support varies by platform:

PlatformSerial ImplementationExtra Crate Needed
Bare-metal (MPS2-AN385, STM32F4)zpico-serial + UART driverYes
ESP32 / ESP32-QEMUzenoh-pico built-in (ESP-IDF serial)No
Zephyrzenoh-pico built-in (uart_poll_in/out)No
FreeRTOS / NuttXzenoh-pico built-in (POSIX /dev/ttyXXX)No
ThreadXzenoh-pico built-in (HAL DMA)No

On non-bare-metal platforms, serial just works by using a serial/... locator — no extra crates needed. zpico-serial only fills the gap for bare-metal targets where custom FFI symbols replace zenoh-pico’s system layer.

Board Crate Feature Selection

Each board crate uses Cargo features to select the transport:

# Use serial transport (disable default ethernet/wifi)
nros-board-mps2-an385 = { path = "...", default-features = false, features = ["serial"] }

# Use ethernet transport (default)
nros-board-mps2-an385 = { path = "..." }

# Both transports (runtime selection via locator string)
nros-board-mps2-an385 = { path = "...", features = ["serial"] }

Available Features by Board

Board CrateDefaultAlternativeBoth
nros-board-mps2-an385ethernetserialethernet,serial
nros-board-stm32f4ethernetserialethernet,serial
nros-board-esp32-qemuethernetserialethernet,serial

When both features are enabled, the transport is selected at runtime by the zenoh locator string in Config:

  • "tcp/192.0.3.1:7448" → Ethernet/WiFi
  • "serial/UART_0#baudrate=115200" → Serial

Quick Start: QEMU Serial Example

1. Build the Serial Talker

cd examples/qemu-arm-baremetal/rust/serial-talker
nros generate-rust
cargo build --release

2. Run in QEMU

cargo run --release

QEMU starts with -serial pty and prints the PTY path:

char device redirected to /dev/pts/3 (label serial0)

3. Connect zenohd

In another terminal, start zenohd with the serial link:

zenohd --connect serial//dev/pts/3#baudrate=115200

The MCU’s messages are now bridged to the zenoh network. Any zenoh subscriber (including ROS 2 nodes) can receive them.

4. Subscribe from Host

# Using zenoh CLI
z_sub -k "0/chatter/**"

# Or from a ROS 2 node
ros2 topic echo /chatter std_msgs/msg/Int32

Configuration

Serial Config

Board crates provide a serial_default() constructor:

#![allow(unused)]
fn main() {
use nros_board_mps2_an385::{Config, run};

let config = Config::serial_default();
// Defaults: baudrate=115200, locator="serial/UART_0#baudrate=115200"

run(config, |config| {
    let exec_config = ExecutorConfig::new(config.zenoh_locator)
        .domain_id(config.domain_id);
    let mut executor = Executor::open(&exec_config)?;
    // ...
    Ok(())
})
}

Custom Baud Rate

#![allow(unused)]
fn main() {
let config = Config::serial_default()
    .with_baudrate(921600)
    .with_zenoh_locator("serial/UART_0#baudrate=921600");
}

Locator Format

Serial locators follow zenoh-pico convention:

FormatExampleUse Case
serial/<dev>#baudrate=<baud>serial/UART_0#baudrate=115200Device name (Zephyr, ESP-IDF, bare-metal)
serial/<tx>.<rx>#baudrate=<baud>serial/0.1#baudrate=115200Pin numbers (Arduino)

QEMU PTY Testing

How It Works

QEMU’s -serial pty flag redirects the emulated UART to a host pseudo-terminal (PTY). This creates a virtual serial port that zenohd can connect to, enabling full end-to-end testing without physical hardware.

┌──────────────────────────────────────────────────────────┐
│                       Host                               │
│  ┌─────────┐    ┌───────────┐    ┌────────────────────┐  │
│  │ zenohd  │◄──►│ /dev/pts/N│◄──►│ QEMU MPS2-AN385   │  │
│  │         │    │  (PTY)    │    │ -serial pty         │  │
│  └────┬────┘    └───────────┘    │ UART0 ──► firmware  │  │
│       │                         └────────────────────┘  │
│       │ zenoh network                                    │
│  ┌────▼────┐                                             │
│  │ z_sub   │                                             │
│  │ or ROS 2│                                             │
│  └─────────┘                                             │
└──────────────────────────────────────────────────────────┘

QEMU Flags

The serial example’s .cargo/config.toml uses:

[target.thumbv7m-none-eabi]
runner = "qemu-system-arm -cpu cortex-m3 -machine mps2-an385 -nographic -semihosting-config enable=on,target=native -serial pty -kernel"

Key flags:

  • -serial pty — Expose UART0 as a host PTY
  • -nographic — No display window
  • -semihosting-config enable=on,target=native — Debug output via semihosting (separate from UART)
  • No -netdev / -net — Serial transport doesn’t need Ethernet

-icount shift=auto

For reliable serial communication, add -icount shift=auto to synchronize QEMU’s virtual clock with wall-clock time. Without this, QEMU runs the CPU at full speed, which can cause timing-sensitive serial handshakes to fail:

runner = "qemu-system-arm -cpu cortex-m3 -machine mps2-an385 -nographic -semihosting-config enable=on,target=native -icount shift=auto -serial pty -kernel"

Automated Testing

The integration test test_qemu_serial_pubsub_e2e in packages/testing/nros-tests/tests/emulator.rs automates the full workflow:

  1. Build the serial-talker example
  2. Launch QEMU with -serial pty
  3. Parse the PTY path from QEMU stderr
  4. Start zenohd with --connect serial//dev/pts/N#baudrate=115200
  5. Subscribe and verify message delivery

Run it with:

just test-qemu

Baud Rate Tuning

Baud RateUse Case
115200Default, safe for all hardware
460800Higher throughput, most USB-serial adapters
921600Maximum for many MCU UARTs

Higher baud rates increase throughput but may cause framing errors on noisy or long cables. QEMU ignores the baud rate (infinite speed), so rate tuning only matters on physical hardware.

Buffer Sizing

zenoh-pico serial uses a 1500-byte MTU with COBS framing. The maximum wire frame is 1516 bytes. zpico-serial uses a 2048-byte RX ring buffer per port, which accommodates one full frame plus overhead.

For high-throughput scenarios, ensure the MCU’s UART FIFO is drained frequently by calling executor.spin_once() in a tight loop.

Troubleshooting

“Session open failed” or Handshake Timeout

The zenoh serial handshake (Init → Ack) must complete within zenoh-pico’s timeout. Common causes:

  • Wrong PTY path — Check that zenohd connects to the correct /dev/pts/N
  • Baud rate mismatch — MCU and zenohd must use the same baud rate
  • QEMU timing — Add -icount shift=auto to slow down QEMU’s CPU clock

No Messages Received

  • Locator mismatch — Ensure the MCU’s zenoh_locator matches what zenohd expects
  • Domain ID — Both sides must use the same ROS 2 domain ID
  • zenohd not bridging — Verify zenohd is connected to the serial port and also listening on TCP for subscribers

UART Pin Conflicts

On physical hardware, ensure the UART TX/RX pins aren’t shared with the debug console. Some boards use UART0 for debug output — use a different UART for zenoh transport, or disable debug prints.

ESP32 Serial

ESP32 uses zenoh-pico’s built-in ESP-IDF serial implementation. No zpico-serial dependency is needed. Select serial transport in the board crate:

nros-board-esp32-qemu = { path = "...", default-features = false, features = ["serial"] }

The default locator is serial/UART_0#baudrate=115200. ESP32’s USB-JTAG-Serial peripheral or UART0/UART1 can be used.

RTIC Integration

RTIC (Real-Time Interrupt-driven Concurrency) is a concurrency framework for ARM Cortex-M that compiles tasks directly to hardware interrupt handlers — no RTOS kernel, no task control blocks, no software scheduler. nano-ros runs on top of RTIC by letting the framework own fn main, the scheduler, and the dispatchers, while nano-ros contributes one spin task and (optionally) one callback-dispatch task.

This chapter is the user-facing tutorial for that integration. For the underlying design — why per-Node dispatch strategies, why tags instead of closures, why no AsyncNode trait yet — see the sibling internals page Dispatch Strategy.

When to reach for nros::main!() instead of Pattern A

The current in-tree RTIC examples (e.g. examples/stm32f4/rust/talker-rtic/) use the Pattern A escape hatch: hand-written #[rtic::app], manual Executor::open inside #[init], manual spin_once(0) polling task. It looks like this:

#![allow(unused)]
fn main() {
// File: examples/stm32f4/rust/talker-rtic/src/main.rs (today, pre-216.B.5)
#[rtic::app(device = stm32f4xx_hal::pac, dispatchers = [USART1, USART2])]
mod app {
    #[init]
    fn init(cx: init::Context) -> (Shared, Local) {
        let syst = nros_board_stm32f4::init_hardware(&config, cx.device, cx.core);
        nros_rmw_zenoh::register().expect("Failed to register RMW backend");
        let mut executor = Executor::open(&exec_config).unwrap();
        let mut node = executor.create_node("talker").unwrap();
        let publisher = node.create_publisher::<Int32>("/chatter").unwrap();
        net_poll::spawn().unwrap();
        publish::spawn().unwrap();
        (Shared {}, Local { executor, publisher })
    }

    #[task(local = [executor], priority = 1)]
    async fn net_poll(cx: net_poll::Context) {
        loop {
            cx.local.executor.spin_once(core::time::Duration::from_millis(0));
            Mono::delay(10.millis()).await;
        }
    }
}
}

That pattern works, but it’s ~90 lines of glue per binary, and it puts the burden of getting RTIC + nros + the dispatcher list right on every example author. Phase 216.B.5 collapses it to one line:

#![allow(unused)]
fn main() {
// File: examples/stm32f4/rust/talker-rtic/src/main.rs (post-216.B.5)
#![no_std]
#![no_main]

use defmt_rtt as _;
use panic_probe as _;

nros::main!();
}

The nros::main!() proc-macro reads [package.metadata.nros.entry] deploy = "rtic-stm32f4" from the Entry pkg’s Cargo.toml, sees that the board’s metadata declares framework = "rtic", and emits a full #[rtic::app] module — including #[init], __nros_spin, and (if any deployed Node declares DispatchStrategy::Deferred) __nros_dispatch.

Pick nros::main!() whenever:

  • You want a one-line main.rs and don’t need custom RTIC tasks.
  • Your Node logic is portable across boards (the Node pkg is framework-agnostic; only the Entry pkg picks RTIC).
  • You’re happy with the default dispatcher list from the board crate.

Keep Pattern A when:

  • You need fine-grained control of dispatcher priorities, monotonic setup, or hand-tuned #[shared] state.
  • You’re shipping a one-off bring-up binary and don’t want the Node-pkg / Entry-pkg split overhead.

Both paths stay supported. The escape hatch is the “I want full control” path; nros::main!() is the ergonomic path on top.

The three pkg roles

A nano-ros RTIC workspace is three packages (the 3-pkg-role taxonomy, per docs/design/0024-multi-node-workspace-layout.md §11):

my_rtic_robot/
├── Cargo.toml                         # [workspace] members = [...]
└── src/
    ├── talker_pkg/                    # Node pkg — board-agnostic
    │   ├── package.xml
    │   ├── Cargo.toml
    │   └── src/lib.rs                 # impl Node for Talker + nros::node!(Talker)
    └── talker_entry/                  # Entry pkg — picks RTIC board
        ├── package.xml
        ├── Cargo.toml                 # [package.metadata.nros.entry] deploy = "rtic-stm32f4"
        └── src/main.rs                # nros::main!();
  • Node pkg — declares what the node does (publishers, subscriptions, services, actions). No main, no #[rtic::app], no board choice. Builds as rlib + staticlib and gets linked into one or more Entry pkgs.
  • Entry pkg — picks the board crate (nros-board-rtic-stm32f4), pins the deploy target, and runs nros::main!();.
  • Bringup pkg — optional, only when ≥2 Entry pkgs share the same launch/*.launch.xml topology. Skipped here because we have one binary.

The split exists so the same talker_pkg/ can be deployed under RTIC on STM32F4, under FreeRTOS on QEMU, and under POSIX on a Linux host without any per-target Node-pkg fork.

A minimal Node pkg — pub-only Talker

A pub-only Node declares DispatchStrategy::Inline (the default) — it publishes from its own RTIC task and never enters on_callback.

#![allow(unused)]
fn main() {
// File: src/talker_pkg/src/lib.rs
#![no_std]

use core::time::Duration;
use nros::prelude::*;
use std_msgs::msg::Int32;

pub struct Talker;

pub struct TalkerState {
    publisher: Publisher<Int32>,
    counter: i32,
}

impl Node for Talker {
    const NAME: &'static str = "talker";
    // Inline is the default; spelled out here for clarity.
    const DISPATCH: DispatchStrategy = DispatchStrategy::Inline;

    fn register(ctx: &mut NodeContext<'_>) -> NodeResult<()> {
        ctx.create_publisher::<Int32>("/chatter")?;
        Ok(())
    }
}

impl ExecutableNode for Talker {
    type State = TalkerState;

    fn init() -> Self::State {
        TalkerState { publisher: Publisher::placeholder(), counter: 0 }
    }

    fn tick(state: &mut Self::State, _ctx: &mut TickCtx<'_>) {
        state.counter = state.counter.wrapping_add(1);
        let _ = state.publisher.publish(&Int32 { data: state.counter });
    }

    fn tick_period(_state: &Self::State) -> Option<Duration> {
        Some(Duration::from_millis(1000))
    }
}

nros::node!(Talker);
}
# File: src/talker_pkg/Cargo.toml
[package]
name    = "talker_pkg"
version = "0.1.0"
edition = "2024"

[lib]
crate-type = ["rlib", "staticlib"]

[dependencies]
nros      = { workspace = true, default-features = false }
std_msgs  = { workspace = true }

[package.metadata.nros.node]
class = "talker_pkg::Talker"

The Entry pkg

# File: src/talker_entry/Cargo.toml
[package]
name    = "talker_entry"
version = "0.1.0"
edition = "2024"

[[bin]]
name = "talker_entry"
path = "src/main.rs"

[dependencies]
nros                       = { workspace = true, default-features = false }
nros-board-rtic-stm32f4    = { workspace = true }
talker_pkg                 = { path = "../talker_pkg" }

[package.metadata.nros.entry]
deploy = "rtic-stm32f4"

[package.metadata.nros.deploy.rtic-stm32f4]
board     = "rtic-stm32f4"
rmw       = "zenoh"
domain_id = 0
locator   = "tcp/192.168.1.10:7447"
#![allow(unused)]
fn main() {
// File: src/talker_entry/src/main.rs
#![no_std]
#![no_main]

use defmt_rtt as _;
use panic_probe as _;

nros::main!();
}

That’s the whole Entry pkg. The proc-macro reads deploy = "rtic-stm32f4", looks up the board crate’s framework = "rtic" metadata, and expands into a #[rtic::app(device = ::nros_board_rtic_stm32f4::pac, dispatchers = [USART1, USART2])] module with auto-generated #[init], __nros_spin, and per-Node state slots.

DispatchStrategy::Inline vs Deferred

A Node pkg declares DispatchStrategy via the Node::DISPATCH const. Two variants matter today; FromIsr is reserved as a design slot (see DispatchStrategy::FromIsr).

Inline — pub-only or tick-only Nodes

#![allow(unused)]
fn main() {
impl Node for Talker {
    const NAME: &'static str = "talker";
    const DISPATCH: DispatchStrategy = DispatchStrategy::Inline;
    // ...
}
}

Inline means: callbacks (if any) fire from the executor’s spin loop — the same RTIC task that polls the network transport. On RTIC that’s __nros_spin running at priority = 1.

Pick Inline when:

  • The Node has no subscriptions, no service handlers, no action handlers. Its only output is Publisher::publish calls from tick or from a custom RTIC task you spawn yourself.
  • The Node has subscriptions, but the per-message work is so cheap (microseconds, no locks, no shared-state touches) that running it inline with the spin loop is acceptable.

The Talker above is the canonical Inline shape: it publishes from tick at 1 Hz, never receives anything, and never blocks the spin loop.

Deferred — callback-driven Nodes

#![allow(unused)]
fn main() {
impl Node for Listener {
    const NAME: &'static str = "listener";
    const DISPATCH: DispatchStrategy = DispatchStrategy::Deferred;
    // ...
}
}

Deferred means: when a callback arrives, the spin loop enqueues a callback token plus context into a heapless::spsc::Queue and returns immediately. A separate __nros_dispatch RTIC task (typically priority = 2, one above spin) drains the queue and calls ExecutableNode::on_callback from its own task context.

Pick Deferred when:

  • The Node’s on_callback handler must hold an RTIC lock on a #[shared] resource — running it from the spin task would force every other dispatcher at the spin’s priority to wait on the lock.
  • The handler does non-trivial work (parses, integrates, posts to a hardware peripheral) and you want it scheduled independently from network polling.
  • The Node sits at a different priority tier than the spin task.

The tag-based registration API for Deferred Nodes

Deferred Nodes can’t use the closure form ctx.create_subscription("/chatter", |msg| { ... }) — the closure captures state by value, and a no-alloc framework runtime has no place to store an unknown closure type. Instead, you register with _static and dispatch via tag match in on_callback:

#![allow(unused)]
fn main() {
// File: src/listener_pkg/src/lib.rs
#![no_std]

use nros::prelude::*;
use std_msgs::msg::Int32;

pub struct Listener;

pub struct ListenerState {
    sub_chatter: SubscriptionTag,
}

impl Node for Listener {
    const NAME: &'static str = "listener";
    const DISPATCH: DispatchStrategy = DispatchStrategy::Deferred;

    fn register(ctx: &mut NodeContext<'_>) -> NodeResult<()> {
        let _tag = ctx.create_subscription_static::<Int32>("/chatter")?;
        Ok(())
    }
}

impl ExecutableNode for Listener {
    type State = ListenerState;

    fn init() -> Self::State {
        // The macro-emitted glue overwrites the placeholder with the
        // real tag at register time.
        ListenerState { sub_chatter: SubscriptionTag::placeholder() }
    }

    fn on_callback(
        state: &mut Self::State,
        cb: Callback<'_>,
        ctx: &mut CallbackCtx<'_>,
    ) {
        if state.sub_chatter == cb {
            let msg: Int32 = ctx.downcast().unwrap();
            defmt::info!("Received: {}", msg.data);
        }
    }
}

nros::node!(Listener);
}

The same shape applies to services (ServiceTag / create_service_static) and actions (ActionTag / create_action_static). The macro lint (216.A.6) rejects mixing closure registration into a Deferred Node — the error spans the offending call and suggests switching to the _static form.

The custom-task escape — nros::main!(custom_tasks = ...)

Coming in Phase 216.B.4. The syntax below is the locked design; it lands after 216.B.3 (the RTIC routing branch of the macro) is in tree.

Real RTIC applications don’t only have nano-ros tasks. You want a dedicated my_adc task to poll an ADC into a #[shared] buffer, or a my_ui task to drive an OLED. The custom_tasks = [...] form of nros::main!() folds your extra task fns into the generated #[rtic::app] module:

#![allow(unused)]
fn main() {
// File: src/talker_entry/src/main.rs (with custom tasks)
#![no_std]
#![no_main]

use defmt_rtt as _;
use panic_probe as _;

nros::main!(custom_tasks = [my_adc, my_ui]);

#[rtic_task(priority = 3, shared = [adc_data])]
async fn my_adc(mut ctx: my_adc::Context) {
    loop {
        let sample = read_adc();
        ctx.shared.adc_data.lock(|d| *d = sample);
        Mono::delay(10.millis()).await;
    }
}

#[rtic_task(priority = 2)]
async fn my_ui(_ctx: my_ui::Context) {
    loop {
        draw_screen();
        Mono::delay(33.millis()).await;
    }
}
}

The proc-macro extracts the user task tokens verbatim, splices them into the generated mod __nros_app, and adds their dispatchers to the RTIC dispatchers = [...] list. Signatures + attributes + priorities are preserved.

Custom tasks can interact with nano-ros via the #[shared] resources the macro exposes — typically executor (the spin executor) for raw publisher access, or a Node’s state slot for direct handler calls. Specifics will be locked when 216.B.4 lands.

DispatchStrategy::FromIsr — not yet

The third variant of DispatchStrategy is reserved for callbacks that fire directly from an ISR handler (e.g. a timer pulse triggering a publish without a scheduler hop). This is a design slot only; the implementation is deferred to Phase 216.E.1.

Landing it requires:

  • A reentrancy audit of the dispatch path.
  • A lock-free SPSC variant tolerant of ISR-priority producers.
  • A per-Node #[isr_safe] proof contract.

Until that work lands, nros check (Phase 216.D.1) rejects DispatchStrategy::FromIsr deployments with a clear diagnostic.

See also

Embassy Integration

Embassy is an async/await framework for embedded Rust, built around a cooperative executor that polls futures from a single context (per priority tier). nano-ros runs on top of Embassy by letting the framework own fn main, the spawner, and the async executor, while nano-ros contributes one __nros_spin_task async fn and (optionally) one __nros_dispatch_task async fn.

This chapter is the user-facing tutorial for that integration. For the underlying design — why per-Node dispatch strategies, why tags instead of closures, why on_callback stays sync even on async runtimes — see the sibling internals page Dispatch Strategy.

What Embassy buys you and where nano-ros fits

Embassy is an alternative to RTOS-style preemptive tasks: every “task” is an async fn driven by embassy_executor::Spawner::spawn. Tasks yield at .await points; there’s no scheduler tick, no context switch overhead beyond a future poll. For nano-ros that means:

  • A natural place for I/O-driven background work (SPI bus servicing, GPIO debounce, sensor frame parsing) that yields at every .await on a embassy_time::Timer or embassy_stm32::spi::Spi::transfer.
  • A natural place to spawn downstream work from inside a nano-ros callback — the “spawn-from-sync escape” below.
  • Cooperative scheduling means a Node on_callback runs to completion before the executor polls anything else at the same priority. Keep handlers short, hand off long work to a spawned task.

The current in-tree example (examples/stm32f4/rust/talker-embassy/src/main.rs) is Pattern A: hand-written #[embassy_executor::main], hand-written zenoh_poll_task, hand-written publisher_task. It’s a working template but it’s ~150 lines and each example author re-derives the spawn topology. Phase 216.C.4 collapses it to:

#![allow(unused)]
fn main() {
// File: examples/stm32f4/rust/talker-embassy/src/main.rs (post-216.C.4)
#![no_std]
#![no_main]

use defmt_rtt as _;
use panic_halt as _;

nros::main!();
}

The proc-macro reads [package.metadata.nros.entry] deploy = "embassy-stm32f4" from the Entry pkg’s Cargo.toml, sees that the board’s metadata declares framework = "embassy", and expands into a full #[embassy_executor::main] async fn main(spawner: Spawner) including the spin task spawn, the dispatch task spawn, and the run_plan registration call.

Why sync on_callback even on Embassy

A natural-feeling Embassy API would make ExecutableNode::on_callback an async fn. We don’t — for two reasons:

  1. The no-alloc contract. Async fns desugar to anonymous future types; storing them generically in the runtime requires either boxing (Box<dyn Future>alloc dependency) or const-generic GAT plumbing through every trait. Both add cost without buying anything Spawner::spawn doesn’t already give us.
  2. Framework-task routing. The runtime already dispatches callbacks from a framework-owned task (__nros_dispatch_task on Embassy, __nros_dispatch on RTIC). The Node author can spawn their own async task from inside the sync on_callback; that task runs under the same executor with no extra plumbing.

So on_callback keeps the same callback-token signature as RTIC, POSIX, and every other backend. The escape for “I need to await something” is the spawn-from-sync pattern below.

AsyncNode (an async-on-callback trait via RPITIT) is reserved as a design slot — see When to wait for AsyncNode at the end of this chapter.

The three pkg roles

The workspace shape is identical to RTIC (the 3-pkg-role taxonomy, per docs/design/0024-multi-node-workspace-layout.md §11):

my_embassy_robot/
├── Cargo.toml                           # [workspace] members = [...]
└── src/
    ├── listener_pkg/                    # Node pkg — board-agnostic
    │   ├── package.xml
    │   ├── Cargo.toml
    │   └── src/lib.rs                   # impl Node for Listener + nros::node!(Listener)
    └── listener_entry/                  # Entry pkg — picks Embassy board
        ├── package.xml
        ├── Cargo.toml                   # [package.metadata.nros.entry] deploy = "embassy-stm32f4"
        └── src/main.rs                  # nros::main!();

The Node pkg stays board-agnostic — listener_pkg/ from the RTIC chapter could be deployed under Embassy by swapping the Entry pkg. That’s the point of the split.

Entry pkg

# File: src/listener_entry/Cargo.toml
[package]
name    = "listener_entry"
version = "0.1.0"
edition = "2024"

[[bin]]
name = "listener_entry"
path = "src/main.rs"

[dependencies]
nros                       = { workspace = true, default-features = false }
nros-board-embassy-stm32f4 = { workspace = true }
listener_pkg               = { path = "../listener_pkg" }

[package.metadata.nros.entry]
deploy = "embassy-stm32f4"

[package.metadata.nros.deploy.embassy-stm32f4]
board     = "embassy-stm32f4"
rmw       = "zenoh"
domain_id = 0
locator   = "tcp/192.168.1.10:7447"
#![allow(unused)]
fn main() {
// File: src/listener_entry/src/main.rs
#![no_std]
#![no_main]

use defmt_rtt as _;
use panic_halt as _;

nros::main!();
}

DispatchStrategy::Deferred is the common case

Every callback-driven Embassy Node should declare DispatchStrategy::Deferred:

#![allow(unused)]
fn main() {
impl Node for Listener {
    const NAME: &'static str = "listener";
    const DISPATCH: DispatchStrategy = DispatchStrategy::Deferred;
    // ...
}
}

Deferred means: the spin task pushes signaled callbacks into an embassy_sync::channel::Channel<NoopRawMutex, _, CHANNEL_CAPACITY> (default capacity 32). A separate __nros_dispatch_task awaits the channel receive end and routes each callback to the right Node’s on_callback. Both tasks run cooperatively on the same Embassy executor; you can spawn your own tasks alongside them.

Inline is allowed but unusual on Embassy:

  • The nros check lint (Phase 216.D.1) emits a warning for framework = "embassy" with DispatchStrategy::Inline and suggests switching to Deferred — running callbacks inline on Embassy ties them to the spin task’s poll point, which makes the scheduling cost-vs-benefit confusing.
  • If your Node is genuinely pub-only (no subscriptions, no services, no actions), Inline is still fine — the Inline path simply never enters on_callback. The warning above does not apply for pure-publisher Nodes.

The spawn-from-sync escape

Real Embassy Nodes need to do async work downstream from a callback: write to an SPI bus, send a UART frame, poll a sensor with timeout. The pattern is:

  1. Hold an embassy_executor::Spawner on Self::State.
  2. From inside the sync on_callback, call state.spawner.spawn(handle_downstream(msg)).unwrap().
  3. The downstream #[embassy_executor::task] async fn handles the await-heavy work.

Worked example — Listener that writes received data to SPI

#![allow(unused)]
fn main() {
// File: src/listener_pkg/src/lib.rs
#![no_std]

use embassy_executor::Spawner;
use nros::prelude::*;
use std_msgs::msg::Int32;

pub struct Listener;

pub struct ListenerState {
    sub_chatter: SubscriptionTag,
    spawner: Spawner,
}

impl Node for Listener {
    const NAME: &'static str = "listener";
    const DISPATCH: DispatchStrategy = DispatchStrategy::Deferred;

    fn register(ctx: &mut NodeContext<'_>) -> NodeResult<()> {
        let _tag = ctx.create_subscription_static::<Int32>("/chatter")?;
        Ok(())
    }
}

impl ExecutableNode for Listener {
    type State = ListenerState;

    fn init() -> Self::State {
        ListenerState {
            sub_chatter: SubscriptionTag::placeholder(),
            // The macro-emitted glue populates the Spawner from the
            // EmbassyBoardEntry init hook; until then, hold a sentinel
            // that's overwritten before the first dispatch.
            spawner: Spawner::for_current_executor(),
        }
    }

    fn on_callback(
        state: &mut Self::State,
        cb: Callback<'_>,
        ctx: &mut CallbackCtx<'_>,
    ) {
        if state.sub_chatter == cb {
            let msg: Int32 = ctx.downcast().unwrap();
            // Spawn the async work and return immediately. The
            // dispatch task stays unblocked.
            state.spawner.spawn(handle_msg(msg)).unwrap();
        }
    }
}

#[embassy_executor::task(pool_size = 4)]
async fn handle_msg(msg: Int32) {
    // Pretend we have an SPI handle stashed somewhere accessible.
    // The real wiring depends on your peripheral-access pattern;
    // the point is `.await` is free here, even though on_callback
    // is sync.
    defmt::info!("handling /chatter sample: {}", msg.data);
    embassy_time::Timer::after_millis(5).await;
    // spi.write(&msg.data.to_le_bytes()).await.unwrap();
}

nros::node!(Listener);
}

Two things to note:

  • Spawner is Copy. Holding a Spawner on Self::State adds no runtime cost beyond an integer copy. You can clone it freely across multiple on_callback calls.
  • pool_size matters. #[embassy_executor::task(pool_size = N)] allocates N static slots. If you spawn faster than the spawned tasks complete, spawn(...) returns Err(SpawnError::Busy) — handle it (drop the message, log a warning, etc.). The default is pool_size = 1.

When the spawn pool fills up

Two patterns for backpressure:

  1. Drop on full. Treat the spawned task as best-effort. If the spawn returns an error, log it and continue.
  2. Channel-based queue. Pre-spawn one long-lived #[embassy_executor::task] async fn that holds an embassy_sync::channel::Channel receive end. The on_callback pushes into the channel’s send end (drop on full or block — your choice). Lets you control queue depth independently of pool_size.

The right pick depends on whether dropped messages are tolerable for your workload; nano-ros doesn’t impose either.

When to wait for AsyncNode

The spawn-from-sync escape covers every case we know of today. But if you find yourself writing a lot of one-off pool-of-one spawn dances — each callback spawning a task whose body is a single .await — that’s a smell, and we’d want to know.

The design slot for direct async callbacks is AsyncNode (Phase 216.E.2):

#![allow(unused)]
fn main() {
// Design sketch — NOT shipping today.
pub trait AsyncNode: 'static {
    const NAME: &'static str;
    const DISPATCH: DispatchStrategy = DispatchStrategy::Deferred;

    fn register(ctx: &mut NodeContext<'_>) -> NodeResult<()>;
    async fn on_callback(
        &mut self,
        callback: AsyncCallbackToken,
        ctx: CallbackCtx,
    );
}
}

It would compile only on Embassy targets (RTIC has no async runtime to drive RPITIT futures into; POSIX + RTOS don’t either), and the nros::node!() macro would emit a separate __nros_node_<pkg>_on_callback_async ABI symbol the Embassy dispatch task picks up. 216.E.2 lands only if real usage shows spawn-from-sync is consistently painful. If your application is hitting that case, file an issue with the call pattern that’s prompting it.

Until then: spawn from sync. It’s two lines per callback and stays fully no-alloc.

See also

Troubleshooting

Common issues and solutions when working with nros.

zenoh-pico Multiple Client Issues

Symptoms

When running multiple zenoh-pico clients (e.g., a talker and listener) simultaneously connecting to the same router, you may see errors like:

<err> nros_bsp: Failed to open zenoh session: -3
<err> rust: rustapp: BSP init failed: InitFailed(-5)

Or publisher/subscriber declarations fail with:

  • Error code -128 (_Z_ERR_GENERIC)
  • Error code -78 (_Z_ERR_SYSTEM_OUT_OF_MEMORY)

The first client typically succeeds while subsequent clients fail during z_declare_publisher() or z_declare_subscriber().

Root Cause

This is caused by the Z_FEATURE_INTEREST feature in zenoh-pico. The interest protocol implements write filtering to optimize network traffic by only sending data to interested subscribers. However, this feature has issues when multiple clients connect to the same router:

  1. The router tracks “interest” for each client
  2. When creating a publisher, zenoh-pico creates a write filter context
  3. The filter creation calls _z_session_rc_clone_as_weak() which can fail
  4. This manifests as _Z_ERR_SYSTEM_OUT_OF_MEMORY (-78) even when memory is available

The failure occurs in zenoh-pico/src/net/filtering.c:

ctx->zn = _z_session_rc_clone_as_weak(zn);
if (_Z_RC_IS_NULL(&ctx->zn)) {
    _z_write_filter_ctx_clear(ctx);
    z_free(ctx);
    _Z_ERROR_RETURN(_Z_ERR_SYSTEM_OUT_OF_MEMORY);
}

Solution

nros disables Z_FEATURE_INTEREST by default for all builds:

Native builds (build.rs):

#![allow(unused)]
fn main() {
let dst = cmake::Config::new(&zenoh_pico_build)
    // ...
    .define("Z_FEATURE_INTEREST", "0")
    .build();
}

Zephyr builds: The just zephyr setup recipe automatically patches zenoh-pico’s config.h to disable these features. If you set up the workspace manually or the patch wasn’t applied, run:

just zephyr setup  # Updates and patches existing workspace

Or manually edit modules/lib/zenoh-pico/include/zenoh-pico/config.h:

// Change from:
#define Z_FEATURE_INTEREST 1
#define Z_FEATURE_MATCHING 1

// To:
#define Z_FEATURE_INTEREST 0
#define Z_FEATURE_MATCHING 0

Then rebuild with --pristine:

west build -b native_sim/native/64 nros/examples/zephyr/rust/talker -d build-talker --pristine

Note: Z_FEATURE_MATCHING depends on Z_FEATURE_INTEREST, so both must be disabled.

References

  • rmw_zenoh_pico disables this feature by default
  • zenoh-pico filtering code: src/net/filtering.c
  • zenoh-pico interest protocol: src/session/interest.c

zenoh-pico Version Compatibility

Symptoms

Publisher works but z_publisher_put fails immediately:

zenoh_shim: z_publisher_put failed: -100
<err> rust: rustapp: Publish failed: PublishFailed(-1)
zenoh_shim: z_publisher_put failed: -73

Error codes:

  • -100: _Z_ERR_TRANSPORT_TX_FAILED - Transport transmission failed
  • -73: _Z_ERR_SESSION_CLOSED - Session closed after first failure

Root Cause

Version mismatch between zenoh-pico and zenohd:

  • zenoh-pico: 1.5.1 (in west.yml)
  • zenohd: 1.7.x (installed via cargo)

The zenoh protocol may have changed, causing transport-level incompatibilities.

Solution

Upgrade zenoh-pico to match zenohd version. All zenoh components in nros are pinned to the same version. Use just zenohd build to build the matching router from the submodule.

Temporary workaround: Install an older zenohd version:

cargo install zenoh --version 1.5.1 --features=zenohd

Multiple Zephyr Instance Issues

Ephemeral Port Conflict

Symptom: When running multiple Zephyr native_sim instances (e.g., talker and listener), both sessions fail to establish or one gets -100 (_Z_ERR_TRANSPORT_TX_FAILED) errors.

Root Cause: Zephyr native_sim uses a test entropy source that produces identical random number sequences unless seeded differently. This causes both instances to select the same ephemeral TCP port when connecting to the router, causing a port conflict.

Solution: Pass different --seed values to each native_sim instance:

# Start listener with one seed
./build-listener/zephyr/zephyr.exe --seed=12345

# Start talker with a different seed
./build-talker/zephyr/zephyr.exe --seed=67890

The --seed parameter initializes the test entropy source with a different value, producing different random numbers and thus different ephemeral ports.

For automated tests: The test framework (packages/testing/nros-tests) automatically assigns unique seeds to each process.

Cyclone DDS: Two native_sim Nodes Never Discover Each Other

Symptom: Two native_sim nodes with the Cyclone DDS RMW (e.g. talker + listener) each start, the publisher reports Published: N, but the subscriber stays at Received: 0 — no abort, no error.

Root Cause: Same fixed-entropy issue as above. Cyclone derives each participant’s DDSI GUID prefix from the entropy source; with identical entropy, both processes get the same GUID. DDSI requires per-participant unique GUIDs — a node treats SPDP from a peer that shares its own GUID as a self-announcement and drops it, so the two never form a proxy-participant pair and discovery never closes.

Solution: Pass different --seed values per instance (exactly as above):

./build-listener/zephyr/zephyr.exe --seed=11
./build-talker/zephyr/zephyr.exe   --seed=22

With distinct seeds the GUID prefixes differ and SPDP discovery completes. (native_sim Cyclone discovery is unicast-only — the embedded config points <Peers> at 127.0.0.1; native_sim’s NSOS multicast RX fd does not survive Cyclone’s select()-based socket waitset. Real hardware has real entropy and full multicast, so neither workaround is needed off native_sim.)


Network Configuration Issues

TAP Interface Setup (QEMU bare-metal only)

Symptom: QEMU cannot connect to the network.

Solution: Run the network setup script:

sudo ./scripts/qemu/setup-network.sh

This creates and configures the tap0 interface at 192.0.3.1. QEMU bare-metal guests live on 192.0.3.10+.

Zephyr native_sim networking

Zephyr native_sim uses NSOS (Native Sim Offloaded Sockets) — socket calls are forwarded to host syscalls, so there is no emulated L2 stack and no TAP bridge. Point each example at a host-loopback zenohd / XRCE Agent:

zenohd --listen tcp/127.0.0.1:7456  # or any host-accessible address

Multiple native_sim instances can coexist without bridge configuration.


Message Too Large / Truncated Messages

Symptoms

  • TransportError::MessageTooLarge when receiving service requests or subscription data
  • Messages arrive but with corrupted or incomplete payloads
  • Large messages (images, point clouds) silently disappear

Root Cause

nros uses static buffers at multiple layers, each with configurable size limits. A message must fit through every layer in the path to be delivered intact.

Zenoh backend layers:

LayerPosix DefaultEmbedded DefaultEnv Var
Defragmentation (Z_FRAG_MAX_SIZE)655362048ZPICO_FRAG_MAX_SIZE
Batch size (Z_BATCH_UNICAST_SIZE)655361024ZPICO_BATCH_UNICAST_SIZE
Per-entity shim buffer10241024– (named constant in code)
User receive buffer (RX_BUF)10241024– (const generic)

XRCE-DDS backend layers:

LayerPosix DefaultEmbedded DefaultEnv Var
Transport MTU4096512XRCE_TRANSPORT_MTU
Per-entity buffer10241024– (named constant in code)

Solutions

Increase transport-level limits (set before cargo build):

# Zenoh: allow 128 KB reassembled messages
ZPICO_FRAG_MAX_SIZE=131072 cargo build

# XRCE: increase MTU to 8 KB
XRCE_TRANSPORT_MTU=8192 cargo build

Increase per-entity buffer sizes (in code):

#![allow(unused)]
fn main() {
// Zenoh subscriber with 4 KB receive buffer
let sub = node.create_subscriber_sized::<MyMsg, 4096>(SubscriberOptions::new("/topic"))?;

// Zenoh publisher with 4 KB transmit buffer
let pub_ = node.create_publisher_sized::<MyMsg, 4096>(PublisherOptions::new("/topic"))?;
}

Clean rebuild after changing env vars (CMake caches old values):

cargo clean -p zpico-sys   # or: cargo clean -p xrce-sys
cargo build

Build Issues

zenoh-pico Submodule Not Found

Symptom:

zenoh-pico submodule not found at /path/to/zenoh-pico. Run: git submodule update --init

Solution:

git submodule update --init --recursive

CMake Cache Stale

Symptom: Changes to CMake defines (like Z_FEATURE_INTEREST) don’t take effect.

Solution: Clean the build cache:

# Clean the zenoh-pico sys crate build cache
cargo clean -p zpico-sys
touch packages/zpico/zpico-sys/build.rs
cargo build

For Zephyr builds, clean the west build directory:

rm -rf build/
west build -b native_sim/native/64 <app>

Test Failures

Tests Hang Forever

Symptom: Integration tests hang indefinitely.

Possible causes:

  1. No zenohd router running: Many tests require a zenohd router
  2. Port already in use: Another process is using port 7447
  3. Network misconfigured: TAP or bridge interfaces not set up

Solutions:

# Check if zenohd is running
pgrep zenohd

# Check if port 7447 is in use
ss -tlnp | grep 7447

# Verify QEMU bare-metal TAP (only needed for QEMU bare-metal platforms)
ip addr show tap0

Zephyr Tests Skip

Symptom: All Zephyr tests are skipped.

Cause: The test framework couldn’t find the Zephyr workspace or required components.

Solution: Ensure the Zephyr workspace is properly configured:

# Default location: $repo/zephyr-workspace/ (gitignored, in-tree).
# Legacy setups: $repo/../nano-ros-workspace/ (auto-detected).
ls -la zephyr-workspace
just zephyr doctor

# Or set explicitly
export NROS_ZEPHYR_WORKSPACE=/path/to/zephyr-workspace

# Verify west is available
west --version

Rust/C FFI Issues

Subscriber Callback Crashes

Symptom: When a Zephyr listener receives a message, the application crashes with a segfault or produces garbage values when accessing struct fields. Debug output may show valid pointers but invalid field values:

bsp_zephyr: sub->callback=0x60, sub->user_data=0x4  # These should be valid addresses!

Root Cause: The C BSP stores a pointer to the nros_subscriber_t struct when you call nros_bsp_create_subscriber(). If the Rust code moves the struct after this call (e.g., when returning it from a function), the C code’s pointer becomes dangling.

In Rust, values are moved by default:

#![allow(unused)]
fn main() {
// WRONG - struct will move when returned
pub fn create_subscriber(...) -> Result<BspSubscriber<M>, Error> {
    let mut sub = NanoRosSubscriber { ... };
    nros_bsp_create_subscriber(&mut sub, ...);  // C stores pointer to `sub`
    Ok(BspSubscriber { sub, ... })  // `sub` MOVES here - old address is invalid!
}
}

Solution: Use static storage or ensure the struct has a stable address before passing it to C:

use core::mem::MaybeUninit;
use core::ptr::addr_of_mut;

// Wrapper to make static storage Sync
struct StaticSubscriber(MaybeUninit<NanoRosSubscriber>);
unsafe impl Sync for StaticSubscriber {}

static mut SUBSCRIBER_STORAGE: StaticSubscriber =
    StaticSubscriber(MaybeUninit::uninit());

fn main() {
    let sub_ptr = unsafe {
        let storage_ptr = addr_of_mut!(SUBSCRIBER_STORAGE);
        let sub = (*storage_ptr).0.as_mut_ptr();
        // Initialize fields...
        sub
    };

    // Now the pointer is stable
    nros_bsp_create_subscriber(sub_ptr, ...);
}

Alternative approaches:

  • Use Box::new() (requires alloc) for heap allocation with stable address
  • Use Pin to prevent moves
  • Have the C code use indices instead of pointers

Why this happens: Unlike C where variables have fixed addresses, Rust moves values during assignment and return. When the C code stores a pointer during initialization, it doesn’t know the Rust struct will be moved later.

Function Pointer ABI Mismatch

Symptom: Callbacks passed between Rust and C crash or receive garbage arguments.

Solution: Ensure function pointers use extern "C":

#![allow(unused)]
fn main() {
// CORRECT - uses C calling convention
extern "C" fn my_callback(data: *const u8, len: usize, ctx: *mut c_void) {
    // ...
}

// WRONG - uses Rust calling convention (incompatible with C)
fn my_callback(data: *const u8, len: usize, ctx: *mut c_void) {
    // ...
}
}

zenoh-pico Error Codes

CodeNameDescription
-3_Z_ERR_TRANSPORT_OPEN_FAILEDCould not connect to router
-78_Z_ERR_SYSTEM_OUT_OF_MEMORYMemory allocation failed (or write filter issue)
-128_Z_ERR_GENERICGeneric error

For the complete list, see zenoh-pico/include/zenoh-pico/utils/result.h.

Differences from standard ROS 2

Coming from rclcpp, rclrs, or rclc? This page calls out where nano-ros looks the same, where it diverges, and the reason behind each choice. It is an orientation page — the per-language API references (Rust / C / C++) cover the surface itself.

The short version: nano-ros keeps ROS 2’s vocabulary (Node, Publisher, Subscription, Service, Client, ActionServer, ActionClient, Timer, Executor, QoS, message codegen) so existing nodes port cleanly. The trade-offs that come from running on a Cortex-M3 with 64 KB of heap shape every place where the surface diverges.

Same vocabulary, same wire format

  • ROS 2 entity model unchanged. A node owns publishers, subscriptions, services, clients, action servers, action clients, timers, and parameters.
  • Topic / service / action names follow ROS 2 conventions (/talker_node/chatter, /add_two_ints, /fibonacci).
  • Message types stay rosidl-shaped (std_msgs/msg/Int32, geometry_msgs/msg/Twist, …). CDR encoding on the wire.
  • Default backend (nros-rmw-zenoh) is bit-compatible with the upstream rmw_zenoh ROS 2 RMW. A nano-ros publisher and an rclcpp subscriber on the same zenohd router exchange messages without a bridge — see ROS 2 Interoperability.

Where it diverges, and why

1. The executor is the entry point

Standard ROS 2 starts with a global init (rclcpp::init / rclrs::Context::default_from_env / rclc_support_init) followed by node creation, then optionally an executor.

nano-ros inverts this: an Executor opens the RMW session and owns the runtime budget. Every node, publisher, subscription, service, client, action handle, and timer is allocated out of the executor’s arena.

Executor::open(&config) → Node → Publisher / Subscription / Service / Client / Timer

Why. The arena is fixed-size and known at compile time (NROS_EXECUTOR_ARENA_SIZE / NROS_EXECUTOR_MAX_CBS). On a 64 KB heap MCU we cannot afford the indirection of a global allocator behind every create_publisher call. The executor-as-arena pattern moves the size negotiation up to the application’s startup code, where it belongs.

2. Both manual-poll and callback paths are first-class

rclcpp is callback-only — every subscription needs a callback, the executor dispatches them. rclrs <0.7 was manual-poll only. rclc exposes both but treats manual-poll as the second-class path.

nano-ros treats them as equals. A subscription created via node.create_subscription(...) exposes try_recv(). A subscription registered via executor.add_subscription(callback) runs the callback during spin_once. Pick whichever fits the control loop.

Why. Embedded apps often want the predictability of an explicit poll inside a real-time loop. Callback dispatch is great for event-driven services but adds indirection that real-time engineers have to bound by hand. Offering both means the application picks.

3. Async pub/sub/service/action

rclcpp has no real async/await. rclrs 0.7+ added async but it sits next to the synchronous executor, not in it. rclc has none.

nano-ros has it as a first-class path: Executor::spin_async() wakes on RMW I/O, Subscription::recv().await, Client::call().await, ActionClient::send_goal().await, etc. Runs on tokio (POSIX), Embassy (FreeRTOS / RTIC), or any external Future-driver.

Why. Robotic control loops are stitched together from many concurrent waits — sensor data, service replies, action feedback, parameter updates. Async lets you write them as one straight-line function instead of hand-rolling state machines on top of a poll-based executor.

4. Heap is optional

ROS 2 RMW implementations all assume std + a heap. Even rclrs’s no_std story is “soon”.

nano-ros runs in three modes that map onto target capability:

ModeCargo featuresWhat works
stdstd (default on POSIX)Everything. POSIX threading, full async runtime.
no_std + allocalloc + a #[global_allocator]Everything except features that need std::sync::Mutex. Used by FreeRTOS / NuttX / ThreadX / Zephyr / ESP32.
no_std + nostd-runtime (cooperative)nostd-runtime, RTIC appsCooperative single-task — no threading at all. Used by bare-metal MPS2-AN385, single-core RTIC.

Why. Heap presence is not a binary “embedded yes/no” — it is a spectrum. Stm32-class boards have a heap; Cortex-M0+-class might not. The feature axis lets the same application code target both.

5. Backend selection at compile time, not runtime

Standard ROS 2 uses an RMW_IMPLEMENTATION env var read at process start. The plugin loader pulls a shared library, dispatches calls through C function pointers.

nano-ros bakes the backend in at compile time. The consuming Cargo.toml adds the backend crate directly (nros-rmw-zenoh / nros-rmw-cyclonedds / nros-rmw-xrce-cffi) alongside nros with the rmw-cffi feature; CMake options (-DNANO_ROS_RMW=zenoh) decide it for C/C++ builds. The backend’s #[ctor] registers its vtable with the nros-rmw-cffi runtime registry before main.

Why.

  • Dead-code elimination. A 32 KB Flash budget cannot afford to link every backend’s C client and pick at runtime. Linking only the selected backend cuts the binary by 60–80 %.
  • No plugin loader. Most embedded targets have no dlopen. The cost of the plugin abstraction is a permanent overhead with no payoff there.
  • Cross-compile sanity. RMW_IMPLEMENTATION baked into the binary means the build system already knows which backend’s C client to link — no separate “find shared library at runtime” step.

The trade-off is real: changing backends requires a rebuild. This is the right trade-off for the embedded use case; it would be the wrong trade-off for desktop ROS 2.

A binary can still link multiple backends and run them in parallel — one node on backend A, another on backend B, the executor draining both each tick. The Cross-backend Bridges chapter walks the build knobs (NROS_RMW=... primary selector, explicit per-backend register(), NANO_ROS_RMW= none cmake escape hatch) and the three shipped examples.

6. Message codegen lands inside your build, not a sibling library

Standard ROS 2 uses ament + rosidl to compile message packages (std_msgs, geometry_msgs, …) into separate shared libraries that your application links against.

nano-ros’s nros generate-rust (Rust) and nano_ros_generate_interfaces() (C / C++ via CMake) write message type definitions into your build tree. No _msgs library, no ament overlay, no colcon workspace required.

Why. Embedded cross-builds without a hosted ROS 2 install need to generate message types from package.xml + .msg files alone. The codegen tool ships its own bundled rosidl-flavoured .msg set inside the nros CLI (built in-tree from packages/cli/, Phase 218), so you don’t even need the upstream message packages on disk.

7. QoS profile is the full DDS field set; backends advertise per-policy support

Standard ROS 2 supports the full DDS QoS profile family (reliability, durability, history, depth, deadline, lifespan, liveliness, liveliness_lease_duration, avoid_ros_namespace_conventions) and performs profile matching between endpoints.

nano-ros’s nros_rmw_qos_t carries the same field set; standard profile constants (NROS_RMW_QOS_PROFILE_DEFAULT, _SENSOR_DATA, _SERVICES_DEFAULT, _PARAMETERS, _SYSTEM_DEFAULT) match upstream rmw_qos_profile_* field-for-field. ROS 2 apps porting across pull the equivalent constant unchanged.

Each backend advertises which policies it can enforce via Session::supported_qos_policies(). The runtime validates the requested QoS at entity-create time and returns IncompatibleQos synchronously when the backend can’t honour a requested policy:

#![allow(unused)]
fn main() {
if session.supported_qos_policies().contains(QosPolicyMask::DEADLINE) {
    // backend honours deadline; safe to set deadline_ms
} else {
    // app handles deadline monitoring itself
}
}

No silent downgrade. The runtime never quietly drops a requested policy. Apps either get the QoS they asked for or a hard error.

Why upstream-shaped struct, not a smaller subset. ROS 2 QoS is the established vocabulary; mismatched APIs make porting painful. The field set is small (24 bytes); apps that don’t request a policy leave its field at zero (“off”). Per-backend implementation is a separate question — which policies actually fire — answered by the support mask.

Why synchronous error instead of runtime event. Upstream’s RMW_EVENT_REQUESTED_INCOMPATIBLE_QOS event surfaces mismatches at run time. Most QoS mismatches are configuration errors visible at startup; the runtime path doesn’t need to handle them. The few that aren’t (cross-process QoS-mismatched discovery) the wire protocol handles itself — DDS endpoints negotiate via DDS Discovery, zenoh endpoints communicate intent through the topic-key encoding.

Manual liveliness assertion. Publishers configured with MANUAL_BY_TOPIC / MANUAL_BY_NODE liveliness call assert_liveliness() explicitly to refresh the lease. Available on every language surface (Rust Publisher<M>::assert_liveliness(), C nros_publisher_assert_liveliness(&pub), C++ pub.assert_liveliness()). Backends without manual-assertion wiring treat the call as a no-op — none of the surviving backends wire it natively yet. See Status events for the runtime-event side of liveliness, deadline, and message-lost.

Per-backend coverage is documented in RMW vs upstream § 7.

8. No runtime backend swap, no runtime introspection

Standard ROS 2 ships ros2 topic list, ros2 node info, dynamic endpoints discovery, rmw_get_* introspection.

nano-ros has none of that at runtime. The backend is fixed at compile time, the wire-protocol introspection is whatever the backend natively exposes (zenoh-pico’s z_query for SPDP, Cyclone DDS’s SPDP/SEDP discovery, …). Use the host-side ROS 2 tools for introspection and connect via the rmw_zenoh interop path.

Why. Every byte of “introspect what’s running” is overhead a microcontroller can’t justify when a host-side ROS 2 environment is one router-hop away.

9. Parameters: node-local server, no descriptors, no callbacks (yet)

Standard ROS 2 (rclcpp::Node) ships a rich parameter surface: declare_parameter<T> with ParameterDescriptor (description, ranges, read-only, dynamic typing), set_parameter returning a SetParametersResult, atomic multi-set, three callback hooks (pre_set / on_set / post_set), parameter overrides from the launch file or CLI, and a service-backed remote-introspection surface (/<node>/get_parameters, /<node>/set_parameters, …).

nano-ros’s nros::ParameterServer<Cap> (C++) and the equivalent C nros_param_server_t keep the vocabulary (declare_parameter<T>, get_parameter<T>, set_parameter<T>, has_parameter) but trim the surface aggressively for embedded use.

What we keep

  • Same five scalar types: bool, int64_t, double, string, plus the bool / int64_t / double / byte / string array variants on the C side (nros_param_*_array).
  • Same lifecycle: declare → get → set, with declare-once-then-typed-get semantics.
  • Optional service-backed exposure (~/get_parameters / ~/set_parameters / ~/list_parameters / …) when the param-services feature is enabled. This pulls in ROS 2 wire compat: declared parameters are visible to ros2 param list /<node> and ros2 param set.

What we drop, and why

Upstream featurenano-ros statusWhy dropped
ParameterDescriptor (description, ranges, read-only, dynamic_typing)not exposeddescriptor metadata is host-side concern; embedded server enforces type at declare-time, range checks belong in set callbacks (deferred — see below)
add_pre_set_parameters_callback / add_on_set_parameters_callback / add_post_set_parameters_callbackone combined nros_param_callback_t (server-wide, fires after set)three callbacks → three indirection slots × N subscribers; one callback covers the safety-island validation use case (reject if out of range)
set_parameter returning SetParametersResult (successful: bool, reason: string)returns nros_ret_tstring reason would force heap or fixed-buffer; ret code captures the binary outcome
set_parameters_atomicallynot exposedatomic multi-set requires transaction log; not justified by current embedded use
declare_parameters (multi-declare with namespace)not exposedone-by-one declare is fine for compile-time-known parameter sets
Parameter overrides from CLI / launch / yamlnot exposedembedded apps configure via Kconfig / Config struct; runtime overrides come over the wire via ~/set_parameters (when param-services is on)
Storage allocation policycompile-time <Capacity>, inline storageno heap; capacity sizing belongs in the application’s startup code, same as the executor arena

Storage shape difference

rclcppnano-ros
Containerstd::map<string, ParameterValue> (heap)nros_parameter_t storage[Capacity] (caller-owned, inline)
String valuestd::string (heap)fixed 128-byte slot, copy semantics
Array paramsstd::vector<T> (heap)caller-owned pointer + length (caller keeps storage alive)
Total fixed costunboundedCapacity × sizeof(nros_parameter_t) known at compile time

Class shape difference

rclcpp::Node owns the parameter store. nano-ros splits them:

// rclcpp
auto node = std::make_shared<rclcpp::Node>("ctrl");
node->declare_parameter<double>("ctrl_period", 0.15);
double v = node->get_parameter("ctrl_period").as_double();

// nano-ros
nros::Node node;
nros::ParameterServer<8> params;
NROS_TRY(nros::Node::create(node, "ctrl"));
NROS_TRY(params.declare_parameter<double>("ctrl_period", 0.15));
double v;
NROS_TRY(params.get_parameter<double>("ctrl_period", v));

Why split. Adding a parameter store to Node would require templating Node on capacity, which propagates through every create_publisher / create_subscription site. Composing ParameterServer<N> alongside the node keeps Node non-templated and matches the rest of the freestanding C++14 surface (callers own storage). params.raw() exposes the underlying nros_param_server_t* for future ROS 2 service-backed registration.

Why no Box<dyn FnMut> callback yet. The same constraint that shapes’s event callbacks applies here: nano-ros’s #[no_std] core forbids alloc-style indirection. A future descriptor-+-validation-callback path will use a function pointer + void* user_context pair, registered at declare-time. Tracked under the upstream parity backlog.

Going further. When upstream’s full parameter surface matters — describe_parameter, ranges, three-stage validation callbacks, override files — fall back to running a host-side ROS 2 node that exposes them and uses the embedded node only as the leaf publisher / subscriber.

What this means in practice

If you are coming from rclcpp:

  • Open an Executor, then create the node from it.
  • Decide poll vs. callback per subscription, not globally.
  • If the platform has std, nros::init() looks identical; if it is RTOS / bare-metal, plan the executor arena up front.
  • Pick nros-rmw-zenoh for ROS 2 interop; everything else is a different trade-off.

If you are coming from rclrs:

  • The umbrella crate is nros, not split into rclrs_*. nros::prelude gives you everything.
  • Executor::open(&config) is the equivalent of Context::default_from_env() + Executor::new(...).
  • The async surface is in nros::dds_async (re-exported at the crate root). Compatible with tokio out of the box.

If you are coming from rclc:

  • Same C names where they map (nros_node_init, nros_publisher_init, nros_subscription_init). Memory ownership rules are the same — the caller owns storage, the API initialises it.
  • See the C API reference for the full surface.

Going deeper

  • API surface, type by type → per-language references at Rust / C / C++.
  • Why the executor / RMW / platform layers split this way → Architecture Overview.
  • Cooperative no_std + nostd-runtime model → no_std Support.

nano-ros vs micro-ROS

The closest peer project to nano-ros is micro-ROS. Both ship a full embedded ROS 2 framework — client library + RMW + tooling targeted at MCU-class hardware. This page is an apples-to-apples comparison.

Why not compare to rmw_zenoh or ros2_rust? Scope mismatch. rmw_zenoh is a single RMW backend; nano-ros is a full client stack. ros2_rust (rclrs) targets hosted Linux only — it can’t run on the bare-metal or RTOS targets nano-ros and micro-ROS both address. Comparing nano-ros to either would be misleading.

Side-by-side

Axisnano-rosmicro-ROS
Project homeNEWSLab NTUROS 2 ecosystem (community + Bosch + eProsima)
First release20242019
Primary languageRust (with C + C++ bindings)C (rclc)
User-facing APIsRust + C + C++C only (rclc); experimental C++ in rclcpp_lite
RMW backend choiceZenoh, XRCE-DDS, Cyclone DDS — pick at compile timeXRCE-DDS only
Network modelPeer-to-peer (Zenoh / Cyclone DDS) or agent-based (XRCE)Agent-based only
Bridge process required?No for Zenoh / Cyclone DDS; yes for XRCEYes (Micro-XRCE-DDS Agent)
Supported RTOSesFreeRTOS, NuttX, ThreadX, Zephyr, ESP-IDF, PX4 (NuttX), POSIX, bare-metalFreeRTOS, NuttX, Zephyr, ESP-IDF, POSIX; PX4 is the canonical deployment
no_std coreYes — entire client stack compiles no_std + heaplessNo — rclc requires libc + a heap
Heap usageOptional on bare-metal (XRCE backend is fully static); required for Zenoh / Cyclone DDSRequired (malloc-based DDS-XRCE client)
RT scheduling storySchedContext API: FIFO / EDF / Sporadic / TimeTriggered classes; ARINC-653 cyclic-executive surface; per-callback runtime accounting + overrun detectionrclc executor with priority callbacks; no SchedContext / EDF / TT story
Multi-executor preemptionExecutor::open_threaded per-RTOS via PlatformScheduler traitSingle executor per process
Multi-backend bridge in one binaryYes — Executor::open_with_rmw + multi-NodeNo (single XRCE session per process)
DiscoveryZenoh liveliness, RTPS SPDP, XRCE-via-AgentXRCE-via-Agent
QoS supportBackend-dependent matrix (Zenoh 4/7, XRCE 4/7, Cyclone DDS 7/7)Subset of XRCE QoS
Formal verification160 Kani harnesses + 102 Verus proofs (CDR, scheduling, RMW glue)None published
E2E safety protocolCRC-32/ISO-HDLC + sequence tracking, EN 50159-mapped (safety-e2e feature)None
ROS 2 distro coverageHumble (Iron deferred — type-hash work pending)Humble, Iron, Jazzy
Build systemCargo + CMake + platform tools (west, idf.py, probe-rs) plus just recipes; C/C++ consume via add_subdirectory(<repo>)colcon + CMake; per-RTOS meta-build (create/configure/build/flash_firmware.sh)
Deploy/config modelEntry packages select board/RMW/deploy shape; Bringup packages own launch topology; platform tools build and flashcolcon.meta (hand-tuned static sizing) + configure_firmware.sh -t <transport> flags + hand-coded rclc app
Host-side brokernone (Zenoh P2P / Cyclone DDS brokerless); Agent only for XRCEMicro-XRCE-DDS Agent always required
Release modelSource-only (no crates.io, no precompiled binaries)Source-only + per-distro Debian packages
Code-size (Cortex-M XRCE talker)~75 KB flash (XRCE), ~100 KB+ (Zenoh)~30–50 KB (XRCE + rclc)
LicenseMIT OR Apache-2.0 (dual)Apache-2.0
GovernanceSingle-academic-lab maintainership todayROS 2 community + corporate stewards
Commercial supportNone as of writingBosch + eProsima offer services

Pick nano-ros when…

  • You want Rust as a first-class API, not bolted on. Memory safety + ownership semantics extend to your application code.
  • You want multi-backend flexibility — same binary can speak Zenoh + DDS, or you want to pick Zenoh’s peer-to-peer model over XRCE’s agent dependency.
  • You need scheduling primitives beyond priority callbacks — EDF / Sporadic / ARINC-653 TT classes, per-callback runtime accounting, formal SchedContext API.
  • You’re targeting safety-aware deployments that benefit from the E2E CRC + sequence tracking and the Kani / Verus harness coverage.
  • You want a lean source-only consumption model: git clone + add_subdirectory(<repo>). No crates.io drift, no pre-built per-distro binaries.

Pick micro-ROS when…

  • You’re already in the micro-ROS ecosystem (existing rclc code, established Agent deployment, ROS 2 Jazzy / Iron support).
  • Your toolchain is C-only and you don’t want to introduce Rust into the build pipeline.
  • Your target’s flash budget is tight (~30 KB ceiling) — the rclc + XRCE client is smaller than nano-ros + Zenoh.
  • You need commercial support contracts today (Bosch, eProsima, PIWeb).

Migration sketch

If you’re porting from micro-ROS to nano-ros:

  • rcl_node_initExecutor::create_node (Rust) / nros_executor_node_init (C).
  • rclc_executor_tnros::Executor.
  • rclc_publisher_init_defaultNode::create_publisher::<M> or nros_publisher_init.
  • micro-ROS’s rmw_uros_set_custom_transport → nano-ros’s custom transport pattern via nros_platform_* C ABI (see Custom Transport).
  • XRCE Agent deployment stays the same — point nano-ros’s XRCE backend at the same Agent.

See also

Migration Guide for ROS 2 Users

This scaffold maps standard ROS 2 concepts to nano-ros pages. It is not an API reference; use it as a checklist when moving an existing rclcpp, rclc, or rclrs node toward nano-ros.

Setup

Standard ROS 2 usually starts from a distro install and a runtime RMW choice. nano-ros starts from a source checkout + a compile-time target tuple:

git clone --branch=v<X.Y.Z> https://github.com/NEWSLabNTU/nano-ros.git
cd nano-ros
just setup       # print choices
just setup base  # native/ROS/zenoh quick start

Read Setup Compared to Standard ROS 2 before changing package code.

Node Lifecycle

ROS 2 applications typically call rclcpp::init(), create nodes, then spin. nano-ros opens an executor first; the executor owns the session, arena, and runtime budget. Nodes and entities are created from it.

See Differences from Standard ROS 2 and Execution Model and Two-Layer API.

Publishers and Subscriptions

Topic names, message names, and CDR wire encoding stay ROS-shaped. The main porting decision is whether to use:

  • polling handles from Node::create_*, or
  • callback registration through Executor::register_*.

Use polling for RTIC, Embassy, or tight RT loops. Use callbacks for desktop-style event-driven nodes.

Services and Actions

Service and action names map cleanly, but nano-ros exposes request / reply and goal / feedback / result paths through explicit handles and promises. Manual-poll paths may require explicit result handling in RTOS loops.

Start with the native examples, then check platform-specific examples for FreeRTOS, Zephyr, or bare-metal timing constraints.

QoS and Events

nano-ros keeps DDS-shaped QoS profile fields, but each backend advertises the policies it can enforce. Unsupported QoS is reported at entity creation instead of being silently downgraded.

See QoS, Status Events, and Discovery and Choosing an RMW Backend.

Message Generation

Standard ROS 2 builds generated message libraries as sibling packages. nano-ros generates Rust, C, or C++ bindings into the workspace or build tree and can use a shared generation cache.

See Message Binding Generation.

Backend Selection

Replace runtime RMW_IMPLEMENTATION=... with a compile-time selection:

  • posix-zenoh for early ROS 2 interop through rmw_zenoh_cpp.
  • *-xrce for agent-based micro-ROS-style deployments.
  • *-dds or *-cyclonedds for direct DDS/RTPS deployments where the platform supports the required networking and memory model.

Common Porting Traps

  • Assuming the backend can change without rebuilding.
  • Creating heap-heavy callbacks on no_std targets without alloc.
  • Expecting ROS 2 graph introspection APIs on constrained targets.
  • Forgetting to match ROS domain ID, QoS reliability, or zenoh router mode during interop tests.
  • Treating a platform guide as optional when cross-compiling for RTOS or bare-metal targets.

Architecture Overview

nano-ros is a ROS 2 client stack for embedded and RTOS targets. The important architectural idea is separation by responsibility: user code talks to a ROS-shaped API, the core runtime owns entities and serialization, the RMW layer moves bytes, and the platform layer supplies OS or hardware primitives.

Layers

Application
  Rust / C / C++ node code

Board package (optional)
  Hardware init, network drivers, config loading, entry point

Core runtime
  Executor, Node, pub/sub/service/action handles, parameters,
  message traits, CDR serialization

RMW backend
  Zenoh, XRCE-DDS, Cyclone DDS, or a custom backend

Platform
  Clock, allocation, threading, sleep, random, sockets, libc,
  and bare-metal network polling

POSIX applications usually depend directly on nros. Embedded applications often depend on a board package that initializes hardware, networking, and platform glue before running user code.

Core Runtime

The core runtime is middleware-agnostic. nros-node owns the Executor, node creation, entity handles, timers, and the two API styles:

  • Node::create_* returns handles that the caller polls or awaits.
  • Executor::register_* installs callbacks dispatched by spin_once.

Message and service types are ordinary generated Rust, C, or C++ types with CDR serialization. The RMW backend receives serialized bytes; it does not own rosidl typesupport.

For the API split, see Execution Model and Two-Layer API.

RMW Layer

The RMW layer is the transport boundary. It creates sessions, publishers, subscribers, services, clients, and action channels, then moves serialized samples over the selected wire protocol.

Each Node picks its backend at build time. That compile-time selection replaces standard ROS 2’s runtime RMW_IMPLEMENTATION plugin loader, which is not available on many embedded targets. A single binary can link multiple backends and bridge between them — see Cross-backend Bridges.

For user-facing backend selection, see Choosing an RMW Backend. For design rationale, see RMW API Design.

Platform Layer

The platform layer supplies the primitives that desktop ROS 2 normally gets from the operating system: monotonic time, memory allocation, threading, sleep, random IDs, TCP/UDP sockets, multicast, and libc helpers. Bare-metal ports may also expose a network_poll() hook so the runtime can advance smoltcp while waiting.

The platform is selected at compile time together with the RMW backend and ROS edition. See Platform Model for the feature axes and Custom Platform for porting.

Board Packages

A board package combines a platform implementation with hardware setup and drivers. It typically provides:

  • a Config type loaded from config.toml or target-specific build settings,
  • network and clock initialization,
  • driver setup for Ethernet, UART, WiFi, or simulator I/O,
  • a run() entry point that starts the scheduler or main loop.

Board packages are optional for POSIX but useful for RTOS and bare-metal targets. See Custom Board Package.

Data Flow

Publishing follows this path:

user message -> CDR serializer -> RMW publish bytes -> transport

Receiving reverses it:

transport -> RMW receive bytes -> CDR deserializer -> user handle/callback

This boundary keeps the transport layer small and lets the same message types work across Rust, C, and C++ APIs.

Where to Go Next

Two-Layer API

nano-ros exposes the same communication primitives — subscriptions, publishers, services, service clients, action servers, action clients, timers — through two layers with disjoint verbs. Picking between them is a per-entity deployment choice. The two layers share the same session and compose freely.

Layer 1 — caller pollsLayer 2 — executor dispatches
VerbNode::create_*Executor::register_*
ReturnsOwned handleHandle ID + dispatched closure
Receive shapetry_recv() / call()Promise<T> / try_accept_goal(...)Closure runs on rx / reply / timer fire
SchedulerCaller (RTIC, embassy, app loop)Executor::spin_once
Typical useRTIC tasks, single-task FreeRTOS, RT-bounded loopsCallback-shaped apps, multi-handler servers

Rationale

Embedded callers want both. RTIC owns scheduling and refuses to hand its loop to a library; an embassy task wants to .await a Promise; a FreeRTOS task-per-entity port wants tight try_recv polling. None of those fit the L2 callback model.

But a posix app that handles a dozen topics + a couple of services + an action server wants L2 — write one closure per entity and let spin_once() route events. Forcing it through L1 means writing a manual dispatcher.

The two layers are deliberately separate verb sets so the choice is explicit in source:

  • node.create_subscription::<M>("/topic") — caller polls. No executor magic.
  • executor.register_subscription::<M, _>("/topic", |m| { ... }) — executor dispatches.

The C / C++ FFI mirrors this split: nros_subscription_init (L1) vs nros_executor_register_subscription (L2); nros_subscription_init_polling (L1, inline-storage) vs register_subscription_* (L2, executor arena).

What goes where

graph LR
    EX["Executor::open(&config)"] --> NODE["executor.create_node(name)"]

    subgraph "L2 — register_*"
      direction LR
      EX --> R_T["register_timer"]
      EX --> R_S["register_subscription"]
      EX --> R_SVC["register_service"]
      EX --> R_AS["register_action_server"]
      EX --> R_AC["register_action_client"]
    end

    subgraph "L1 — create_*"
      direction LR
      NODE --> C_P["create_publisher"]
      NODE --> C_S["create_subscription"]
      NODE --> C_SS["create_service"]
      NODE --> C_SC["create_client"]
      NODE --> C_AS["create_action_server"]
      NODE --> C_AC["create_action_client"]
    end

Both layers are wired up by the same call to Executor::open — the register_* family attaches to the executor’s arena, create_* borrows the executor’s session through a short-lived Node. The handles returned by create_* outlive the Node (owned, not borrowed), so the canonical pattern is to scope the Node to creation time only:

let mut executor = Executor::open(&config)?;
let publisher = {
    let mut node = executor.create_node("talker")?;
    node.create_publisher::<Int32>("/chatter")?
};   // node drops here; publisher outlives it

Picking the right layer

Use L1 (Node::create_*) when:

  • The caller has its own scheduling primitive (RTIC dispatcher, embassy task, FreeRTOS task per entity).
  • The flow is fundamentally request-response — service-client Promise<Reply> and action-client send_goalget_result work this way.
  • The receive logic isn’t simple enough to express as a one-shot closure — e.g. a Fibonacci server that publishes feedback in a loop after accepting a goal.
  • You want zero allocations and no executor arena overhead (the L2 path stores the closure + its captures inline in the arena; the arena byte budget is per-build-tunable but non-zero).

Use L2 (Executor::register_*) when:

  • The flow is event-driven with a simple handler — log every received message, reply to a service synchronously, fire a 1 Hz publish.
  • You want spin_once to dispatch all handlers from one site (less state to thread through main).
  • The application is callback-shaped (matches rclcpp / rclpy mental models).

Service clients and action clients stay on L1 in the current code — they have no typed register_*_client<T> API (only register_*_raw byte-level). The typed user path is Node::create_client::<S> + client.call(&req)Promise<Reply>. The request-response + Promise::await model is the L1 shape.

C / C++ mirror

Both layers cross the FFI cleanly:

  • C, L1 pollingnros_subscription_init_polling writes a RawSubscription inline in nros_subscription_t._opaque. nros_subscription_try_recv_raw reads from the inline buffer. Same shape for service / service-client / action server / action client.
  • C, L2 callbacknros_subscription_init records the C callback pointer; nros_executor_register_subscription allocates the executor-arena entry.
  • C++ — typed templates wrap each FFI surface: nros::Subscription<M> + try_recv for L1, nros::PollingActionServer<A> for the L1 action path (122.3.d.b), the L2 executor-registered callback model via the existing nros::ActionServer<A> API.

For the per-FFI-function spec, see the Doxygen reference. For the example migration tally (32 examples on L2, 16 intentionally L1), see the unify-api-paths roadmap doc.

Event-driven path (122.3.c.6.e)

L1 polling assumes the caller has a spin loop. RTOS / embassy applications want the kernel to wake them when data lands. The Raw*::register_waker(&Waker) Rust API and the matching *_set_wake_callback(state, cb, ctx) C / C++ FFI plug a wake source into the same L1 handle. RMW backends that support waking (currently zenoh-pico for subscribers, service-clients, and service-servers) fire the callback from their rx path; other backends fall back to polling silently.

For the underlying primitive (ServiceServerTrait::register_waker in nros-rmw), see the trait doc. For the C wake-callback POD shape (nros_wake_state_t { fn_ptr, ctx }), see the Doxygen reference — same canonical-spec rule as the platform API.

Platform Model

nano-ros configures the library for a specific target through compile-time choices on three axes. nros itself carries one generic rmw-cffi runtime registry; the consuming Cargo.toml adds the chosen backend shim crate directly (one per RMW).

The Three Axes

RMW Backend

The RMW backend determines which middleware transport carries messages. nros exposes a single rmw-cffi feature — the generic C-vtable runtime registry. The consuming crate adds the concrete backend crate as a direct dependency; its #[ctor] registers a vtable with that registry before main.

Backend crateTransportDescription
nros-rmw-zenohzenoh-picoPeer-to-peer via Zenoh protocol; default, ROS-2-interop.
nros-rmw-xrce-cffiMicro-XRCE-DDS-ClientAgent-based via DDS-XRCE protocol.
nros-rmw-cycloneddsCyclone DDSC++ shim; standalone CMake project.

Unlike the platform axis, RMW is not mutually exclusive — Phase 104 lets one binary register multiple backends (bridge nodes). See Choosing an RMW Backend.

Platform (pick one)

The platform feature selects which OS primitives (threading, mutexes, clock, sleep, network) are linked. Each platform crate provides the symbols that the RMW backend’s C library requires at link time.

FeatureTargetThreadingNetwork
platform-posixLinux, *BSDpthreadsBSD sockets
platform-zephyrZephyr RTOSk_thread_createZephyr sockets
platform-bare-metalCortex-M, RISC-V, ESP32Single-threadedsmoltcp
platform-freertosFreeRTOSxTaskCreatelwIP
platform-nuttxNuttX RTOSpthreadsBSD sockets
platform-threadxAzure RTOS / ThreadXtx_thread_createNetX Duo

ROS Edition (pick one)

The ROS edition controls wire-format compatibility with specific ROS 2 releases.

FeatureDescription
ros-humbleHumble Hawksbill wire format (no type hash)
ros-ironIron Irwini wire format (type hash in key expression)

Cross-Cutting Features

These features are orthogonal to the three axes above and can be combined freely.

FeatureDescription
stdEnables std-dependent APIs: spin_blocking(), spin_period(), system clock
allocEnables heap-dependent APIs: boxed callbacks, param-services
safety-e2eCRC-32 integrity + sequence tracking (AUTOSAR E2E / EN 50159)
param-servicesROS 2 parameter service handlers (~/get_parameters, etc.). Implies alloc.
ffi-syncWraps transport FFI calls in critical_section::with() for RTOS reentrancy
sync-spinUse spin-lock mutex (default)
sync-critical-sectionUse critical-section mutex (for RTIC, Embassy)
unstable-zenoh-apiZero-copy receive path (Zenoh backend only)

Mutual Exclusivity Enforcement

The platform and ROS-edition axes are mutually exclusive — nros enforces this at compile time with compile_error!():

#[cfg(all(feature = "platform-posix", feature = "platform-zephyr"))]
compile_error!("Platform features are mutually exclusive — select at most one.");

#[cfg(all(feature = "ros-humble", feature = "ros-iron"))]
compile_error!("`ros-humble` and `ros-iron` are mutually exclusive.");

The RMW axis is not enforced this way — it is the consuming crate’s choice of which nros-rmw-* dependency to add (one, or several for bridge nodes).

Boards vs Platforms

Above the platform axis, board crates select a concrete chip + RTOS combination on top of one of the platform features. A board owns the linker layout, the vendor HAL wiring, the boot path, and any chip-specific peripherals (Ethernet PHY, IVC mailboxes, RTC). It does not carry its own platform axis — it pulls in one of the rows above.

Board crateUnderlying platformCPUNotes
nros-board-mps2-an385bare-metalCortex-M3QEMU MPS2-AN385 + LAN9118 + smoltcp
nros-board-stm32f4bare-metalCortex-M4STM32F4-Discovery
nros-board-esp32-qemubare-metalRISC-V (ESP32-C3)QEMU ESP32 + OpenETH + smoltcp
nros-board-orin-spefreertosCortex-R5FNVIDIA Jetson AGX Orin SPE + FreeRTOS-FSP

nros-board-orin-spe is the canonical example of the board-over-RTOS pattern: a Cortex-R5F SPE running NVIDIA’s FreeRTOS V10.4.3 FSP, with critical-section provided by the canonical nros_platform_critical_section_{acquire,release} C symbols (Cortex-R CPSR I-bit body inside the port) — not by a per-CPU Rust feature flag. See docs/roadmap/phase-100-orin-spe-infra.md for the wiring.

Example Configurations

Each config lists nros (with rmw-cffi + one platform-*) plus the chosen backend crate. POSIX additionally needs nros-platform-cffi with posix-c-port for a pure-cargo build.

QEMU bare-metal Cortex-M3 with Zenoh:

[dependencies]
nros = { path = "…/nros", default-features = false, features = [
    "rmw-cffi", "platform-bare-metal", "ros-humble",
] }
nros-rmw-zenoh = { path = "…/nros-rmw-zenoh", features = ["platform-bare-metal"] }

Zephyr with XRCE-DDS and safety features:

[dependencies]
nros = { path = "…/nros", default-features = false, features = [
    "rmw-cffi", "platform-zephyr", "ros-humble", "alloc", "safety-e2e", "ffi-sync",
] }
nros-rmw-xrce-cffi = { path = "…/nros-rmw-xrce-cffi", features = ["platform-zephyr"] }

Desktop Linux for development and testing:

[dependencies]
nros = { path = "…/nros", default-features = false, features = [
    "rmw-cffi", "platform-posix", "ros-humble", "std", "param-services",
] }
nros-rmw-zenoh = { path = "…/nros-rmw-zenoh", features = ["platform-posix", "link-tcp", "ros-humble"] }
nros-platform-cffi = { path = "…/nros-platform-cffi", features = ["posix-c-port"] }

FreeRTOS with lwIP networking:

[dependencies]
nros = { path = "…/nros", default-features = false, features = [
    "rmw-cffi", "platform-freertos", "ros-humble",
] }
nros-rmw-zenoh = { path = "…/nros-rmw-zenoh", features = ["platform-freertos"] }

NuttX with XRCE-DDS over serial:

[dependencies]
nros = { path = "…/nros", default-features = false, features = [
    "rmw-cffi", "platform-nuttx", "ros-humble",
] }
nros-rmw-xrce-cffi = { path = "…/nros-rmw-xrce-cffi", features = ["platform-nuttx", "xrce-serial"] }

no RMW crate dependency. Instead:

  • RMW. The consuming crate’s nros-rmw-* dep ships a #[ctor] that calls nros_rmw_<name>_register() at lib load (POSIX .init_array) or the caller invokes it from main (bare-metal). The backend installs a nros_rmw_vtable_t into nros-rmw-cffi’s named registry. Executor::open resolves the registered backend; the concrete session type is always CffiSession (vtable-backed).
  • Platform. nros’s platform-* feature resolves to nros-platform-cffi. The canonical nros_platform_* C symbols ship from packages/core/nros-platform-<plat>/src/*.c, linked at the consumer’s build site — posix-c-port for pure-cargo POSIX builds, the standalone lib<…>.a for CMake builds, or the kernel build system for NuttX / Zephyr.
  • Alias TUs. C translation units forward each transport C library’s z_* / uxr_* platform calls to the canonical nros_platform_* symbols: zpico-sys/c/zpico/platform_aliases.c (default-on platform-aliases feature) for zenoh-pico, and nros-rmw-xrce/src/platform_aliases.c (always compiled into nros-rmw-xrce-cffi) for XRCE-DDS. (These replaced the former zpico-platform-shim / xrce-platform-shim crates, deleted in Phase 129.)

The default feature set is std only. No RMW backend or platform is selected by default — the consuming crate chooses explicitly.

For implementation details on how to add a new platform, see the Porting Guide and Platform API Reference.

Board Integration

How you consume nano-ros depends on what your project’s build system looks like, not on what RTOS you’re targeting. This page maps user profile → recommended path. Pick your row, follow the linked guide.

The architecture behind the matrix lives in docs/design/0012-board-bsp-integration-architecture.md (layered model, vendor-BSP scaling, plan).

Consumption matrix

ProfileWorkflowRecommended path
Cargo-first Rust user (have RTOS sources)cargo build --target <triple> produces a single ELF. RTOS kernel built in-tree by Cargo.Generic board crate + env vars pointing at FREERTOS_DIR / THREADX_DIR / etc.
Vendor-IDE user (STM32CubeIDE, MCUXpresso IDE, etc.)Vendor’s existing FreeRTOS / ThreadX project; nano-ros as an add_subdirectory() library.add_subdirectory(third_party/nano-ros) from the vendor’s project.
Zephyr user (any board)west + DTS overlays. Vendor HALs come as Zephyr modules.Zephyr integration shellprojects: entry in your west.yml.
ESP-IDF user (any ESP32 chip)idf.py build.ESP-IDF integration shellidf.py add-dependency nano-ros.
NuttX user (any board)NuttX apps/external/ + Kconfig.NuttX integration shell — symlink under apps/external/.
PX4 userPX4 build pipeline.PX4 integration shellEXTERNAL_MODULES_LOCATION.
Niche RTOS / vendor forkStock RTOS kernel + vendor driver SDK. Cargo-driven build.Generic board crate + vendor overlay crate (~50 LOC). See the Vendor Overlay cookbook.

Generic board crate

For Cargo-first users targeting one of the four supported kernel families:

KernelCrateSDK env vars you set
FreeRTOS + lwIPnros-board-freertosFREERTOS_DIR, FREERTOS_PORT, LWIP_DIR, FREERTOS_CONFIG_DIR
ThreadX + NetX-Duonros-board-threadxTHREADX_DIR, THREADX_CONFIG_DIR, NETX_DIR, NETX_CONFIG_DIR
NuttXnros-board-nuttxNUTTX_DIR (kernel built by NuttX itself)
bare-metal Cortex-M + smoltcpnros-board-baremetal-cortex-mBOARD_LINKER_SCRIPT_DIR
# user_app/Cargo.toml
[dependencies]
nros-board-freertos = "0.1"
nros = { version = "0.1", default-features = false, features = ["rmw-cffi", "platform-freertos", "ros-humble"] }
nros-rmw-zenoh = { version = "0.1", features = ["platform-freertos"] }
std_msgs = { version = "*", default-features = false }
// user_app/src/main.rs
use nros::prelude::*;
use nros_board_freertos::{Config, run};

fn main() -> ! {
    run(Config::from_toml(include_str!("../config.toml")), |config| {
        let mut executor = Executor::open(
            &ExecutorConfig::new(config.zenoh_locator).node_name("my_node"),
        )?;
        // publishers, subscriptions, services, actions, timers...
        Ok::<(), NodeError>(())
    })
}
export FREERTOS_DIR=$HOME/sdk/freertos/kernel
export FREERTOS_PORT=GCC/ARM_CM3
export LWIP_DIR=$HOME/sdk/freertos/lwip
cargo build --release --target thumbv7m-none-eabi

The generic crate’s build.rs compiles the kernel + network stack

  • nano-ros platform glue into a single ELF. No vendor driver glue — that’s what overlay crates are for.

Vendor overlay crate

When your board needs vendor-specific drivers (STM HAL, NXP fsl_*, NVIDIA FSP, Renesas Synergy SSP, …) on top of one of the generic crates, write a small (~50 LOC) overlay crate that:

  1. Depends on the matching generic board crate.
  2. Re-exports Config + run.
  3. Implements #[no_mangle] board-init hooks (nros_board_init_clocks, nros_board_init_eth, nros_board_init_extra_drivers).
  4. Pulls vendor HAL .c sources via its own build.rs cc-rs invocation.

The full cookbook + working precedents (nros-board-orin-spe, nros-board-mps2-an385-freertos) live in Vendor Overlay Board Crate.

Why so many paths

Per the architecture doc, each RTOS already has its own package manager (Zephyr’s west + DTS, ESP-IDF’s component registry, NuttX’s apps/external/, PX4’s EXTERNAL_MODULES_LOCATION). nano-ros rides those rails instead of trying to re-invent a single “embedded library” mechanism that fits no vendor ecosystem cleanly.

The Cargo-first profile (with optional vendor overlay crates) covers the gap where a user is NOT inside an existing RTOS-IDE project and wants to drive their build from Cargo end-to-end. That’s the two-and-a-half rows at the top + bottom of the matrix.

Not on the matrix

  • No common driver HAL. STM HAL_*, NXP fsl_*, Espressif esp_*, Renesas R_* all stay vendor-owned. Overlay crates wrap them; nano-ros doesn’t abstract over them.
  • No DTS-equivalent for non-Zephyr platforms. Zephyr owns its board contract; everywhere else, board config is whatever your vendor IDE produces (CubeMX .ioc, NuttX defconfig, ESP-IDF sdkconfig).
  • No board crate per SKU. Generic + overlay covers the long tail. If you have an exotic board with custom HAL, write a ~50 LOC overlay; nano-ros doesn’t catalog every vendor SKU.

std and alloc Requirements

This document maps which parts of the nano-ros API require the std or alloc Cargo features. All core crates are #![no_std] by default and gate std/alloc-dependent code behind feature flags.

Feature Hierarchy

std  (default)
 └─ alloc
     └─ (base no_std)

Enabling std automatically enables alloc. Enabling alloc does not enable std. With --no-default-features, the entire library compiles without std or alloc.

Summary by Crate

Crateno_std basealloc additionsstd additions
nros-serdesCDR ser/de for primitives, heapless types, &str, [T; N]String and Vec<T> ser/de(none)
nros-coreTime, Duration, Clock (atomic fallback), lifecycle, logger, error types, action types(none)Clock::now() via SystemTime, std::error::Error impls
nros-rmwAll traits, QoS, sync primitives, safety/E2E protocolhandle_request_boxed() (Box<Reply>)(none)
nros-paramsParameterServer, ParameterValue, all parameter types (heapless)(none)ParameterVariant impls for std::string::String, std::vec::Vec
nros-nodeExecutor::open(), create_node(), spin_once(), spin_async(), Promise, pub/sub/service/action, timers (fn pointer callbacks)Boxed timer callbacks, handle_request_boxed(), parameter servicesspin_blocking(), spin_period(), ExecutorConfig::from_env(), halt flag
nrosRe-exports from above(same as above)SpinPeriodResult re-export

RMW Backend Crates

Crateno_std basealloc additionsstd additions
nros-rmw-zenohZenoh-pico RMW implementation, all pub/sub/service/action ops(none)(none)
zpico-sysFFI bindings to zenoh-pico C library(none)(none)
nros-rmw-xrceXRCE-DDS RMW implementation, all pub/sub/service/action ops(none)(none)
xrce-sysFFI bindings to Micro-XRCE-DDS-Client C library(none)(none)

All four backend crates are unconditionally #![no_std] and do not use alloc. The std feature gates only extern crate std (for macro availability in transport modules) and is propagated through the feature chain but does not add any API surface.

Detailed API Availability

Always Available (no_std, no alloc)

Executor and Node:

  • Executor::open(&config) – open RMW session
  • ExecutorConfig::new(locator) – manual configuration
  • executor.create_node(name) – create a node
  • executor.spin_once(timeout_ms) – single spin iteration
  • executor.spin_period_polling(period_ms) – periodic spin without std::thread::sleep

Two-layer API. unified the verb discipline:

  • Layer 1 (caller polls)Node::create_* returns an owned handle. Caller drives try_recv / call / try_accept_goal / try_recv_request_raw itself. Good for RTIC, Embassy, task-per-entity FreeRTOS.
  • Layer 2 (executor dispatches)Executor::register_* takes a closure; spin_once fires it on rx / reply / timer. Good for callback-shaped applications.

Both layers share the same session; mix per entity.

Publish/Subscribe:

  • L1 — node.create_publisher::<M>(topic), node.create_subscription::<M>(topic) (poll with try_recv())
  • L2 — executor.register_timer(period, || publisher.publish(...)), executor.register_subscription::<M, _>(topic, |msg| { ... })
  • publisher.publish(&msg) / publish_raw(&bytes) — publish messages

Services:

  • L1 — node.create_service::<S>(name) (poll with handle_request()), node.create_client::<S>(name) + client.call(&request)Promise<Reply> (poll with promise.try_recv() or .await).
  • L2 — executor.register_service::<S, _>(name, |req| reply). Service clients keep the L1 Promise shape; the typed callback API isn’t surfaced (only register_service_client_raw exists for byte-level use).

Actions:

  • L1 — node.create_action_server::<A>(name) + try_accept_goal / complete_goal, or node.create_action_client::<A>(name) + send_goalPromise<GoalId> / get_resultPromise<(GoalStatus, Result)>.
  • L2 — executor.register_action_server::<A, _, _>(name, goal_cb, cancel_cb) returns a handle for publishing feedback and completing goals. Action clients keep the L1 Promise shape for the same reason as service clients.

Async:

  • executor.spin_async() – async spin loop (drives I/O, dispatches callbacks, yields between iterations)
  • Promise<'a, T, Cli> – allocation-free promise, borrows client’s reply slot
  • Promise::try_recv() – non-blocking poll for reply
  • Promise: Future – implements core::future::Future for .await
  • Uses only core::future and core::task – no external async runtime dependency

Timers:

  • TimerHandle with function pointer callbacks (fn())
  • TimerDuration, TimerMode, TimerState

Serialization:

  • CdrWriter / CdrReader – CDR serialization to/from byte buffers
  • Serialize / Deserialize traits
  • Implementations for: bool, u8-u64, i8-i64, f32, f64, char, &str, [T; N], heapless::String<N>, heapless::Vec<T, N>

Time:

  • Time::new(), Time::from_nanos(), Time::to_nanos()
  • Time::from_secs_f64(), Time::to_secs_f64()
  • Duration::new(), Duration::from_nanos(), Duration::to_nanos()
  • Duration::from_secs_f64(), Duration::to_secs_f64()
  • Clock with atomic counter fallback (no wall-clock time)

Parameters (local only):

  • ParameterServer – store and retrieve parameters
  • ParameterValue enum with heapless collections
  • ParameterDescriptor, ParameterType, Parameter
  • ParameterBuilder for declaring parameters with constraints

Other:

  • LifecycleState, LifecycleTransition, LifecyclePollingNode
  • Logger (uses core::sync::atomic)
  • GoalId, GoalStatus, GoalResponse, CancelResponse
  • QosSettings, TopicInfo, ServiceInfo
  • SafetyValidator, IntegrityStatus (with safety-e2e feature)
  • Sync primitives: spin::Mutex or critical-section (feature-selected)

Requires alloc

APILocationWhy
Serialize/Deserialize for String, Vec<T>nros-serdesHeap-allocated containers
TimerCallback (Box<dyn FnMut() + Send>)nros-node/timer.rsBoxed closure for timer callbacks
Timer::new_with_box(), set_callback_box()nros-node/timer.rsConstruct/update boxed timer callbacks
ServiceServerHandle::handle_request_boxed()nros-node/handles.rsReturns Box<Reply> for large response types
param-services feature (all of it)nros-node/parameter_services.rsService response types (~1MB) require heap allocation

Parameter services detail: The param-services feature (which implies alloc) provides ROS 2 parameter service handlers for ~/get_parameters, ~/set_parameters, etc. Response types like GetParametersResponse contain heapless::Vec<ParameterValue, 64> – each ParameterValue is large, making the total response ~1MB. Box<Response> is required to avoid stack overflow. The core ParameterServer API works without alloc; only the ROS 2 service protocol layer requires it.

Requires std

APILocationWhy
Clock::now() (system/steady clock)nros-core/clock.rsUses std::time::SystemTime / UNIX_EPOCH
std::error::Error for NanoRosError, RclReturnCodenros-core/error.rsTrait requires std
ExecutorConfig::from_env()nros-node/types.rsUses std::env::var() + Box::leak()
Executor::spin_blocking(options)nros-node/spin.rsUses std::thread::sleep(), Arc<AtomicBool>
Executor::spin_period(duration)nros-node/spin.rsUses std::time::Instant, std::thread::sleep()
Executor::halt_flag()nros-node/spin.rsReturns Arc<AtomicBool> for cross-thread cancellation
SpinPeriodResultnros-node/types.rsContains std::time::Duration
ParameterVariant for std::string::String, std::vec::Vecnros-params/types.rsConvenience conversions for std types

Typical Configurations decoupled the nros umbrella from concrete RMW crates.

A consuming Cargo.toml lists three path deps: nros (with rmw-cffi + a platform-* feature), the chosen backend crate (nros-rmw-zenoh / nros-rmw-xrce-cffi), and — on POSIX — nros-platform-cffi with posix-c-port so the C nros_platform_* symbols link into a pure-cargo build. The backend crate’s #[ctor] registers its vtable before main.

Bare-metal / RTOS (no allocator):

nros = { path = "…/nros", default-features = false, features = ["rmw-cffi", "platform-bare-metal"] }
nros-rmw-zenoh = { path = "…/nros-rmw-zenoh", features = ["platform-bare-metal"] }

Full pub/sub, services, actions, timers (fn pointers), parameters (local). Async: spin_async(), Promise, try_recv(), .await – all available without std or alloc. Use spin_once() or spin_period_polling() in your main loop, or spin_async() with an async runtime (Embassy, RTIC v2).

Embedded with allocator (e.g., Zephyr with heap):

nros = { path = "…/nros", default-features = false, features = ["alloc", "rmw-cffi", "platform-zephyr"] }
nros-rmw-zenoh = { path = "…/nros-rmw-zenoh", features = ["platform-zephyr"] }

Adds boxed timer callbacks and handle_request_boxed() for large service replies.

Desktop / Linux:

nros = { path = "…/nros", default-features = false, features = ["std", "rmw-cffi", "platform-posix"] }
nros-rmw-zenoh = { path = "…/nros-rmw-zenoh", features = ["platform-posix", "link-tcp", "ros-humble"] }
nros-platform-cffi = { path = "…/nros-platform-cffi", features = ["posix-c-port"] }

Full API including spin_blocking(), spin_period(), from_env(), system clock. For async, use an external runtime (tokio current_thread + spawn_local for background spin).

Desktop with parameter services: add param-services to the nros feature list above. Adds ~/get_parameters, ~/set_parameters, etc. for ros2 param CLI interop.

C-Level Allocation

Both RMW backends compile and link C libraries that perform heap allocation independently of Rust’s alloc feature. Disabling the Rust alloc feature eliminates Rust-side heap usage (Box, Vec, String) but does not eliminate allocation in the C transport layer.

Backend crateC LibraryAllocator by Platform
nros-rmw-zenohzenoh-pico 1.7.2POSIX: malloc; Zephyr: k_malloc; FreeRTOS: pvPortMalloc; bare-metal: custom bump allocator
nros-rmw-xrce-cffiMicro-XRCE-DDS 3.0.1POSIX: malloc; Zephyr: k_malloc; FreeRTOS: pvPortMalloc; NuttX: malloc

This is by design: the C libraries manage their own session state, stream buffers, and protocol metadata using platform-provided allocators. The Rust alloc feature controls only the Rust API surface (boxed callbacks, heap containers, etc.) and is orthogonal to C-level memory management.

RTOS Cooperation

nano-ros runs on platforms that span pure cooperative bare-metal through multi-task preemptive RTOS to fully async / Future-driven runtimes. The executor’s spin loop has to cooperate with each of these without imposing a single execution model on all of them.

This page maps the common RTOS / runtime execution profiles to the configuration knobs the executor exposes. New apps pick a profile that matches their target’s scheduling discipline; the knobs translate that choice into bounded behaviour from drive_io.

The execution model spectrum

ModelDescriptionExample targets
Cooperative single-taskOne task / thread does all ROS work. No preemption from other tasks (there are none, or they’re all lower priority). Yielding happens at task boundaries.Bare-metal MPS2-AN385, single FreeRTOS task, single Zephyr thread
Preemptive priorityROS task runs at a fixed priority. Higher-priority tasks preempt mid-call by the kernel. ROS-internal entities (subs, timers, services, GCs) all share that priority — they don’t preempt each other.Typical FreeRTOS / ThreadX / Zephyr deployment with worker tasks
WCET-bounded real-timeEach “task” has a provable worst-case execution time. Tasks are dispatched directly from interrupts; no spin loops in the hot path.RTIC, Embassy, avionics WCET-validated code
Time-triggered cyclicFixed schedule. Each cycle does a fixed amount of work in a fixed time slot; ROS gets a fraction of the cycle and must yield.DO-178C / IEC 61508 controller frames
Async runtimeFutures registered with wakers; reactor drives. No spin loop visible to user code.tokio, Embassy futures, custom async runtimes

How drive_io behaves by default

The executor’s spin_once(timeout_ms) calls session.drive_io(...) and lets it drain all ready I/O before returning. After drive_io returns, the executor processes any expired timers and triggered guard conditions. If 10 messages arrived during the wait, all 10 callbacks fire in a single spin_once, then timer / GC dispatch happens once afterwards.

This is the right default for cooperative single-task apps and for async-runtime apps using spin_async. Both want throughput; neither benefits from per-callback scheduling opportunities.

For the other three models, the default has trade-offs the configuration knobs address.

Configuration knobs

KnobDefaultWhen to change
ExecutorConfig::max_callbacks_per_spinusize::MAX (drain all)Set to 1 for upstream-rclcpp-style “one callback per spin_once” — gives the executor a chance to re-check timers / GCs / yield between callbacks
ExecutorConfig::time_budget_per_spin_msNone (no budget)Set to fixed wall-clock budget for time-triggered apps — drive_io returns when the budget expires regardless of pending work
ExecutorConfig::spin_period_msplatform-dependentTighten for lower worst-case latency; loosen for less CPU spent in the spin loop

Backends opt into one additional behaviour automatically: Session::next_deadline_ms() tells the executor about the backend’s next internal event (lease keepalive, heartbeat). The executor caps drive_io’s timeout against it. No app configuration; transparent optimization.

Cooperative single-task

#![allow(unused)]
fn main() {
ExecutorConfig {
    max_callbacks_per_spin: usize::MAX,    // default — drain all
    time_budget_per_spin_ms: None,         // default — no budget
    spin_period_ms: 1,                     // tight loop on the dedicated task
    ..Default::default()
}
}

Drain everything; one task, no fairness concern. Spin tightly to keep latency low.

#![allow(unused)]
fn main() {
ExecutorConfig {
    max_callbacks_per_spin: 1,             // one callback per spin_once
    time_budget_per_spin_ms: None,
    spin_period_ms: 1,
    ..Default::default()
}
}

max_callbacks_per_spin = 1 matches upstream’s rclcpp single-threaded executor pattern. Each spin_once fires one callback and then re-checks timers + GCs. ROS-internal entities share the task priority, but the spin-loop iteration is the scheduling unit; timer expiries are bounded by one callback’s WCET, not the sum across all ready callbacks.

If max-callback dispatch latency is still too high in this profile (e.g., a single callback is slow), the next refinement is moving timer and guard-condition dispatch into drive_io’s loop so the max_callbacks = 1 cap covers them too. This is the path where one slow sub callback no longer delays a timer that should have fired mid-callback.

WCET-bounded real-time (RTIC / Embassy)

Don’t use the spin loop. Use the async path:

#![allow(unused)]
fn main() {
let executor = Executor::open_async(&config)?;
let sub = node.create_subscription_async::<MyMsg>("/topic")?;
loop {
    let msg = sub.recv().await?;        // suspends; waker integration
    handle(msg);
}
}

The async path doesn’t go through drive_io at all. Subscriptions register a Waker; the backend’s RX path wakes the waker; the async runtime schedules the receiving task. Per-task WCET analysis applies to each recv().await continuation, not to a spin loop.

Time-triggered cyclic

#![allow(unused)]
fn main() {
ExecutorConfig {
    max_callbacks_per_spin: usize::MAX,    // not the bottleneck here
    time_budget_per_spin_ms: Some(5),      // 5 ms ROS budget per cycle
    spin_period_ms: 5,                     // matches the cycle's ROS slot
    ..Default::default()
}
}

The cycle gives ROS a fixed wall-clock slot. time_budget_per_spin_ms bounds time spent in drive_io regardless of pending work. The backend respects the budget by checking elapsed wall-clock between callbacks and returning when exceeded. Pending work resumes next cycle.

Async runtime

drive_io not used in the hot path. The executor’s spin_async drives futures via wakers; drive_io becomes a polling tick internally with negligible overhead.

#![allow(unused)]
fn main() {
executor.spin_async().await
}

No knobs apply.

Trade-offs at a glance

ConfigurationThroughputPer-callback latencyTimer-callback fairnessCode-size cost
max_callbacks = MAX (default)HighBounded by ALL ready callbacks’ total WCETPoor under loadSmallest
max_callbacks = 1Slightly lower (more spin loop iterations)Bounded by ONE callback’s WCETGoodSame — the cap is just an integer
time_budget = Some(N)Lower (clock reads per callback)Bounded by N ms wall clockGood if N tight, fair if N looseOne clock read per callback (~10–50 ns)
async / spin_asyncPer-futurePer-future Future pollCooperative — futures yield voluntarilyAsync runtime cost

Backends and their wait primitives

drive_io’s sleep behaviour is backend- and platform-dependent. The spin loop’s “where does the thread sleep” question maps as:

PlatformSleep primitive in drive_ioWhen CPU is sleeping
POSIXselect / epoll_wait with deadlineInside drive_io
Zephyrk_poll / condvar with deadlineInside drive_io
FreeRTOSxSemaphoreTake(g_spin_sem, ticks)Inside drive_io
NuttXsem_timedwait with absolute deadlineInside drive_io
ThreadXtx_event_flags_get(..., TX_OR, ..., ticks)Inside drive_io
Bare-metal smoltcp + BoardIdlesmoltcp poll + wfi() between iterationsOutside drive_io (in the spin loop’s idle hook)
Bare-metal smoltcp without BoardIdlesmoltcp poll, busy loopNowhere — CPU spins

In all cases the user-visible API is Executor::spin_once(timeout); the platform-correct sleep happens transparently underneath.

See also

Porting Overview

nano-ros is designed for customization at three independent levels: RMW (transport protocol), Platform (OS or RTOS), and Board (hardware). The core crates define stable interfaces through Rust traits. You extend nano-ros by implementing those traits for your target – you never modify the core.

Which chapter do I need?

I want to…Chapter
Add a new transport protocol (MQTT, DDS, custom)Custom RMW Backend
Port to a new RTOS or bare-metal targetCustom Platform
Bring up nano-ros on a new MCU boardCustom Board Package

Most porting work falls into the second or third category. Adding a new RMW backend is rare and substantially more involved.

What stays untouched

The following core packages define the interfaces you implement. They compile and work without modification for any new target.

PackageRole
nrosFacade crate: re-exports and feature-axis enforcement
nros-coreMessage, service, and action type traits
nros-serdesCDR serialization
nros-nodeExecutor, Node, pub/sub/service/action handles
nros-rmwRMW trait definitions (Session, Publisher, Subscriber, etc.)
nros-platformPlatform trait definitions and ConcretePlatform type alias
zpico-sys alias TUMaps zenoh-pico z_* C symbols to nros_platform_* (default-on platform-aliases)
nros-rmw-xrce alias TUMaps XRCE-DDS uxr_* C symbols to nros_platform_*
nros-c, nros-cppC and C++ API wrappers

These define the interfaces. You implement them; you do not modify them.

The customization boundary

Everything in nano-ros sits on one side of a trait boundary defined in nros-platform/src/traits.rs.

Above the boundary (yours to write): board crates, platform crates, peripheral drivers, and application code.

Below the boundary (fixed): RMW backends, shim crates, core library, executor, and serialization.

Your platform crate implements traits as inherent methods on a zero-sized type. The shim crates automatically forward RMW-layer C symbols to your implementation through the ConcretePlatform type alias – no dynamic dispatch, no generics propagation.

Platform trait requirements by RMW backend

Not every trait is required. The set depends on which RMW backend the application uses.

Traitzenoh-picoXRCE-DDS
PlatformClockRequiredRequired
PlatformAllocRequired (~64 KB heap)Not needed
PlatformSleepRequiredNot needed
PlatformRandomRequiredNot needed
PlatformTimeRequiredNot needed
PlatformThreadingRequired (multi-threaded platforms)Not needed
PlatformTcpRequiredNot needed
PlatformUdpRequiredNot needed
PlatformSocketHelpersRequiredNot needed
PlatformNetworkPollBare-metal onlyNot needed
PlatformUdpMulticastDesktop platforms onlyNot needed
PlatformLibcBare-metal onlyNot needed

XRCE-DDS is significantly simpler to port: it is single-threaded, heap-free, and uses user-provided transport callbacks rather than a full socket API. A minimal XRCE-DDS port requires only PlatformClock and four C function pointers (open, close, read, write).

zenoh-pico requires a complete platform implementation but provides richer functionality (peer-to-peer, scouting, zero-copy receive, actions).

Registration

After implementing the required traits, you register your platform with two changes:

  1. Add a platform-<name> feature to nros-platform/Cargo.toml that pulls in your crate as an optional dependency.
  2. Add a ConcretePlatform type alias in nros-platform/src/resolve.rs gated by that feature.

The shim crates pick up the new platform automatically. No changes to RMW backends or core crates are needed.

Existing platform implementations

These serve as reference when writing a new port.

Platform crateTargetThreadingNetworking
nros-platform-posixLinux, *BSDpthreadslibc BSD sockets
nros-platform-freertosFreeRTOSFreeRTOS taskslwIP
nros-platform-nuttxNuttXpthreadsPOSIX sockets
nros-platform-threadxThreadXThreadX threadsNetX Duo
nros-platform-zephyrZephyrZephyr POSIXZephyr sockets
nros-platform-mps2-an385Cortex-M3 bare-metalSingle-threadedsmoltcp
nros-platform-stm32f4STM32F4 bare-metalSingle-threadedsmoltcp
nros-platform-esp32-qemuESP32-C3 (QEMU) bare-metalSingle-threadedsmoltcp + OpenETH

Further reading

The Board Trait Family

The Board trait family is the porting surface for a new MCU or host target. It lives in packages/core/nros-platform/src/board/ and pins the contract every board crate (nros-board-<board> or a user-authored crate in a downstream Entry pkg) implements. Phase 212.N introduces this surface; earlier prototypes used nros-board-common::board_init::*, and those legacy traits stay as a transition shim until Phase 212.N.7 lands.

A board impl tells nano-ros four things: how to initialize hardware, how to print a line of text, how to terminate, and (optionally) how to bring a transport up and gate on the network. With those four pieces the BoardEntry::run driver owns the boot lifecycle, and a user Entry pkg main.rs is a ~30 LoC shim.

Where the trait family sits

nros-platform::board
│
├── Board: BoardInit + BoardPrint + BoardExit     // super-trait (blanket impl)
│
├── TransportBringup: Board                       // mixin — Ethernet / WiFi / serial / CAN / USB CDC / IVC
├── NetworkWait:    Board                         // mixin — carrier / DHCP / link-up gate
│
└── BoardEntry: Board
        fn run<F, E>(setup: F) -> Result<(), E>
        where F: FnOnce(&mut RuntimeCtx) -> Result<(), E>;
  • BoardInit::init_hardware() — clock tree, pin mux, peripheral wakes. Runs once on boot before allocation. Panicking here is the same as panicking from fn main() — no recovery.
  • BoardPrint::println(args: core::fmt::Arguments<'_>) — emit a line. Boards wrap whatever stdout makes sense: cortex_m_semihosting::hprintln!, a vendor printf bridge, a UART writer, libc::write(STDOUT_FILENO, …), or printk.
  • BoardExit::{exit_success, exit_failure}() -> ! — terminate cleanly (or with failure). QEMU boards call cortex_m_semihosting::debug::exit; real hardware resets or halts; POSIX shells std::process::exit.
  • TransportBringup::init_transport() — bring the link layer up to L2 (Ethernet frames flow / WiFi associated / UART open at baud). Returns before any L3/IP state — that’s NetworkWait’s job.
  • NetworkWait::wait_link_up() — block until carrier + DHCP/static IP + default route. Only IP-aware transports implement it; CAN-only, serial-only, IVC-only boards skip it.
  • BoardEntry::run(setup) — the boot driver. Implementations live in the family driver crates (nros-board-{posix,freertos,threadx,…}); user Entry pkg main.rs calls it.

Board is itself a blanket-implemented super-trait: any type that carries BoardInit + BoardPrint + BoardExit automatically satisfies Board. Concrete board crates do not impl Board directly — they impl the three sub-traits (plus whichever mixins they need).

The BoardEntry::run lifecycle

BoardEntry::run owns the full boot → user-closure → exit flow. The exact body lives in the family driver crate (e.g. nros-board-posix, nros-board-freertos); each family folds its RTOS specifics in, but the order is fixed:

  1. BoardInit::init_hardware() — clocks, pinmux, MMIO setup.
  2. TransportBringup::init_transport() — driver up at L2 (skipped if the board doesn’t impl the mixin).
  3. NetworkWait::wait_link_up() — DHCP / carrier (skipped if the board doesn’t impl the mixin).
  4. Open the executor, build a RuntimeCtx with overlay knobs from the launch file / CLI, and invoke setup(&mut runtime). The codegen-emitted run_plan(runtime) body is what setup ultimately calls.
  5. Spin the executor to completion (or termination signal).
  6. BoardExit::exit_success() on Ok, BoardExit::exit_failure() on Err or any failed init step.

run returns Result<(), E> rather than ! so unit tests can drive it in a hosted process without exit() killing the test harness — but production boards still call exit_* from inside run’s body after spin returns, so in practice the caller’s Ok(()) arm is unreachable on a real target.

The setup callback is the only place user code runs inside run. Everything else is family-crate boilerplate.

RuntimeCtx

RuntimeCtx<'a> is the per-invocation overlay context the setup callback receives:

#![allow(unused)]
fn main() {
pub struct RuntimeCtx<'a> {
    pub params:  &'a [(&'a str, &'a str)],   // <param name=… value=…/> + -p name:=value
    pub remaps:  &'a [(&'a str, &'a str)],   // topic/service/action renames
    pub env:     &'a [(&'a str, &'a str)],   // env-style key/value (rarely set on embedded)
}
}

Slice-of-tuples, no_std-safe, no allocation. Codegen owns the storage and passes a &mut RuntimeCtx<'_> whose backing slices live in statics — RuntimeCtx::EMPTY is a const placeholder for launch-less single-node examples or unit tests.

Picking your transport mixins

What you implement on the transport axis depends on what link layers your board exposes:

Board transport classImplementNotes
Ethernet (smoltcp / lwIP / NetX BSD)TransportBringup + NetworkWaitBoth — driver up, then DHCP/link gate
WiFi (ESP32)TransportBringup + NetworkWaitSame shape — association is L2, DHCP is L3
Serial UART onlyTransportBringupNo IP, so no NetworkWait
CAN / USB CDC / IVCTransportBringupLink-layer only
Bridged-net (threadx-linux veth)TransportBringup + NetworkWaitHost kernel owns IP — wait_link_up just probes the bridge
Native POSIXNoneHost OS owns everything; the family crate’s run skips both mixins

Boards with multiple transports compose via an internal helper (e.g. a MultiTransport newtype) rather than blanket impls — each transport’s bringup is sequential and order-sensitive (init_link before link_up, sockets only after link).

Worked example — porting a new board

Suppose you’re adding nros-board-acme-cortex-m4-eth, a Cortex-M4 with a UART for println and an MII-attached PHY routed through smoltcp. The crate sits at packages/boards/nros-board-acme-cortex-m4-eth/ and depends on nros-platform, the family crate (nros-board-freertos if FreeRTOS is the RTOS), the matching packages/drivers/<phy>-smoltcp MAC driver, and a vendor HAL crate.

// packages/boards/nros-board-acme-cortex-m4-eth/src/lib.rs
#![no_std]

use nros_platform::board::{
    BoardEntry, BoardExit, BoardInit, BoardPrint,
    NetworkWait, TransportBringup,
    NetworkError, TransportError, RuntimeCtx,
};

pub struct AcmeCortexM4Eth;

impl BoardInit for AcmeCortexM4Eth {
    fn init_hardware() {
        acme_hal::clocks::init_hse_192mhz();
        acme_hal::pinmux::route_uart2();
        acme_hal::pinmux::route_eth_mii();
        acme_hal::eth::release_phy_reset();
    }
}

impl BoardPrint for AcmeCortexM4Eth {
    fn println(args: core::fmt::Arguments<'_>) {
        // 256-byte stack staging buffer is enough for our log lines;
        // pick whatever your UART driver wants.
        let mut buf = heapless::String::<256>::new();
        let _ = core::fmt::write(&mut buf, args);
        let _ = buf.push('\n');
        acme_hal::uart2::write_bytes(buf.as_bytes());
    }
}

impl BoardExit for AcmeCortexM4Eth {
    fn exit_success() -> ! { acme_hal::system::reset() }
    fn exit_failure() -> ! { acme_hal::system::halt_with_blinkenlight() }
}

impl TransportBringup for AcmeCortexM4Eth {
    fn init_transport() -> Result<(), TransportError> {
        // Brings the MAC up to L2; smoltcp owns the IP stack and joins
        // it in NetworkWait.
        acme_phy_smoltcp::init().map_err(|_| TransportError::DriverInit)?;
        acme_phy_smoltcp::wait_link(core::time::Duration::from_secs(5))
            .map_err(|_| TransportError::LinkDown)
    }
}

impl NetworkWait for AcmeCortexM4Eth {
    fn wait_link_up() -> Result<(), NetworkError> {
        acme_phy_smoltcp::dhcp_acquire(core::time::Duration::from_secs(10))
            .map_err(|_| NetworkError::DhcpTimeout)
    }
}

// BoardEntry comes from the family crate's blanket impl:
//   impl<B: Board + TransportBringup + NetworkWait> BoardEntry for B { fn run … }
// The family crate provides the FreeRTOS-shaped run body; you do not
// hand-write a BoardEntry impl unless your target is exotic enough to
// step outside the family.

That’s the whole board crate. A downstream Entry pkg consumes it as:

// pkgs/robot_acme_entry/src/main.rs
use nros_board_acme_cortex_m4_eth::AcmeCortexM4Eth;
use nros_platform::board::BoardEntry;

include!(concat!(env!("OUT_DIR"), "/run_plan.rs"));   // codegen-emitted

fn main() {
    let _ = <AcmeCortexM4Eth as BoardEntry>::run(|runtime| {
        run_plan(runtime)
    });
}

See the Role reference for the Entry-pkg surface.

Family driver crates

The family crate is where the BoardEntry::run body actually lives. Tier-1 families targeted by Phase 212.N.2:

  • nros-board-posix — native host (Linux / *BSD); init_transport/wait_link_up no-ops.
  • nros-board-freertos — FreeRTOS-Kernel + lwIP; run spawns the executor task, hands DHCP to lwIP’s hook.
  • nros-board-threadx — ThreadX + NetX BSD; same shape over NetX.
  • nros-board-nuttx — NuttX POSIX layer; init_transport shells ifup-style logic.
  • nros-board-zephyr — carve-out: Kconfig + DTS own BSP, family crate impls only NetworkWait over <zephyr/net/net_if.h>. The Rust staticlib cannot take over main on Zephyr.
  • nros-board-esp-idf — ESP-IDF component shape; WiFi association lives in init_transport, IP lease in wait_link_up.
  • nros-board-bare-metal — Cortex-M / RV32, no RTOS; minimal run body with a single-thread zp_read loop.

Current state: as of Phase 212.N.1 the trait surface lives in nros-platform; the family driver crates land in N.2 and the per-board shims in N.3. Until then, see packages/boards/nros-board-* for the in-tree boards that still ride the legacy nros-board-common::board_init::* traits — same conceptual shape, different module path.

Cross-references

  • Workspace shape + how an Entry pkg consumes a boardRole reference.
  • Multi-node composition rootdocs/design/0024-multi-node-workspace-layout.md.
  • Why the C ABI looks the way it doesCanonical Platform C ABI.
  • Platform trait set vs Board trait set — these are different traits with different roles. Platform* (clock / alloc / sockets / threading) sits below the RMW; Board* sits above the platform and owns the boot lifecycle. A bare-metal board crate typically depends on both: a nros-platform-* impl for the platform traits and a nros-board-* impl for the board traits.

Custom Board Package

A board crate is the top-level, batteries-included package that application code depends on. It provides three things:

  1. Config struct – network settings (IP, MAC, gateway), transport selection (Ethernet, serial, or WiFi via Cargo features), and a zenoh locator string.
  2. run() entry point – initializes hardware, starts the network stack, and calls a user-provided closure with &Config.
  3. Hardware initialization – PHY/MAC driver setup, interrupt configuration, clock init, and force-linking of platform shim crates.

Users never interact with platform crates or driver crates directly. They depend on a single board crate, call run(), and get a working environment for creating an Executor.

use nros_my_board::{Config, run};
use nros::prelude::*;

run(Config::default(), |config| {
    let exec_config = ExecutorConfig::new(config.zenoh_locator)
        .domain_id(config.domain_id);
    let mut executor = Executor::open(&exec_config)?;
    let mut node = executor.create_node("my_node")?;
    // publishers, subscriptions, services, actions, timers...
    Ok(())
})

Board = platform + hardware

A board crate combines two layers:

  • Platform crate (nros-platform-<os>) – generic RTOS or bare-metal primitives: clock, sleep, threading, memory, random number generation. Implements the traits defined in nros-platform. See Custom Platform for how to write one.
  • Hardware drivers – board-specific peripherals: Ethernet controller, WiFi radio, serial UART. Driver crates in packages/drivers/ (e.g., lan9118-smoltcp, openeth-smoltcp) implement the smoltcp Device trait for a specific Ethernet MAC/PHY.

The board crate glues these together: it depends on the platform crate for OS primitives, on driver crates for peripheral access, and on transport crates (nros-smoltcp, zpico-serial) for bridging the network stack to zenoh-pico. The result is a single dependency that gives application code everything it needs.

Application
    |
    v
Board crate (nros-my-board)
    |
    +-- Platform crate (nros-platform-<name>)
    +-- Driver crate (lan9118-smoltcp)
    +-- Transport bridge (nros-smoltcp or zpico-serial)
    +-- zpico-sys (its default-on alias TU supplies zenoh-pico z_* symbols)

Crate structure

A minimal board crate looks like this:

nros-my-board/
  Cargo.toml
  .gitignore        # /target/
  src/
    lib.rs          # extern crate force-links, pub use Config, pub fn run()
    config.rs       # Config struct with network fields
    node.rs         # Hardware init sequence + run() implementation

Cargo.toml

[package]
name = "nros-my-board"
version = "0.1.0"
edition = "2024"

[lib]
name = "nros_my_board"

[dependencies]
# Platform primitives (pick one)
nros-platform = { path = "../../core/nros-platform", features = ["platform-freertos"] }
nros-platform-freertos = { path = "../../core/nros-platform-freertos" }

# zenoh-pico FFI. Its default-on `platform-aliases` C TU emits every
# `z_*` / `_z_*` symbol zenoh-pico needs, forwarding to `nros_platform_*`.
# (Phase 129 retired the separate `zpico-platform-shim` crate.)
zpico-sys = { path = "../../zpico/zpico-sys", default-features = false }

# Ethernet transport (optional, gated by feature)
nros-smoltcp = { path = "../../drivers/nros-smoltcp", optional = true }
lan9118-smoltcp = { path = "../../drivers/lan9118-smoltcp", optional = true }
smoltcp = { version = "0.12", default-features = false, optional = true, features = [
    "medium-ethernet", "proto-ipv4", "socket-tcp", "socket-udp",
] }

# Serial transport (optional, gated by feature)
zpico-serial = { path = "../../zpico/zpico-serial", optional = true }
my-uart-driver = { path = "../../drivers/my-uart", optional = true }

# Board-specific dependencies
cortex-m = "0.7"
cortex-m-rt = "0.7"
cortex-m-semihosting = "0.5"
panic-semihosting = "0.6"

[features]
default = ["ethernet"]
ethernet = ["dep:nros-smoltcp", "dep:lan9118-smoltcp", "dep:smoltcp"]
serial = ["dep:zpico-serial", "dep:my-uart-driver"]

lib.rs

The lib module force-links shim crates and re-exports the public API.

#![no_std]

// Force-link: ensures `zpico-sys` (and its default-on `platform-aliases`
// C alias TU, which emits zenoh-pico's `z_*` FFI symbols) is pulled into
// the final binary even though Rust code never calls it directly. Without
// this, the linker drops the rlib and zenoh-pico fails to resolve. Phase
// 129 retired the separate `zpico-platform-shim` keep-alive.
extern crate zpico_sys;

mod config;
mod node;

pub use config::Config;
pub use node::{init_hardware, run};

// Re-export entry macro so examples can use #[entry]
pub use cortex_m_rt::entry;

// Convenience println! that routes to semihosting (QEMU) or UART
pub use cortex_m_semihosting;
#[macro_export]
macro_rules! println {
    ($($arg:tt)*) => { $crate::cortex_m_semihosting::hprintln!($($arg)*) };
}

pub fn exit_success() -> ! {
    cortex_m_semihosting::debug::exit(cortex_m_semihosting::debug::EXIT_SUCCESS);
    loop {}
}

pub fn exit_failure() -> ! {
    cortex_m_semihosting::debug::exit(cortex_m_semihosting::debug::EXIT_FAILURE);
    loop {}
}

config.rs

The Config struct holds all network and transport settings. Fields are gated by Cargo features so only relevant fields exist for a given build.

#[derive(Clone)]
pub struct Config {
    #[cfg(feature = "ethernet")]
    pub mac: [u8; 6],
    #[cfg(feature = "ethernet")]
    pub ip: [u8; 4],
    #[cfg(feature = "ethernet")]
    pub prefix: u8,
    #[cfg(feature = "ethernet")]
    pub gateway: [u8; 4],

    #[cfg(feature = "serial")]
    pub uart_base: usize,
    #[cfg(feature = "serial")]
    pub baudrate: u32,

    pub zenoh_locator: &'static str,
    pub domain_id: u32,
}

#[cfg(feature = "ethernet")]
impl Default for Config {
    fn default() -> Self {
        Self {
            mac: [0x02, 0x00, 0x00, 0x00, 0x00, 0x00],
            ip: [192, 0, 3, 10],
            prefix: 24,
            gateway: [192, 0, 3, 1],
            #[cfg(feature = "serial")]
            uart_base: 0x4000_4000,
            #[cfg(feature = "serial")]
            baudrate: 115200,
            zenoh_locator: "tcp/192.0.3.1:7447",
            domain_id: 0,
        }
    }
}

Provide builder methods (with_ip, with_mac, with_gateway, etc.) and factory presets (listener(), talker()) for common test topologies. See nros-board-mps2-an385/src/config.rs for a complete example including from_toml() parsing.

node.rs

The run() function is the entry point. For bare-metal targets it calls the user closure directly. For RTOS targets it creates a task, starts the scheduler, and runs the closure inside that task.

Bare-metal pattern:

pub fn run<F, E: core::fmt::Debug>(config: Config, f: F) -> !
where
    F: FnOnce(&Config) -> Result<(), E>,
{
    init_hardware(&config);
    match f(&config) {
        Ok(()) => exit_success(),
        Err(e) => {
            hprintln!("Error: {:?}", e);
            exit_failure()
        }
    }
}

RTOS pattern (FreeRTOS example):

pub fn run<F, E: core::fmt::Debug>(config: Config, f: F) -> !
where
    F: FnOnce(&Config) -> Result<(), E>,
{
    init_hardware(&config);
    // Create FreeRTOS task with the user closure
    create_app_task(f, config);
    // Start the scheduler -- never returns
    start_scheduler()
}

The init_hardware() function must follow this order:

  1. Clock – initialize the hardware timer (must happen before any clock reads)
  2. Cycle counter – enable DWT or equivalent for timing measurements
  3. RNG seed – seed PRNG with entropy (IP-based hash, semihosting time, etc.)
  4. Transport – Ethernet driver + smoltcp, or UART + zpico-serial
  5. Application – call the user closure

Ethernet peripherals and smoltcp state must live in static mut storage (they are referenced by FFI poll callbacks):

static mut ETH_DEVICE: MaybeUninit<Lan9118> = MaybeUninit::uninit();
static mut NET_IFACE: MaybeUninit<Interface> = MaybeUninit::uninit();
static mut NET_SOCKETS: MaybeUninit<SocketSet<'static>> = MaybeUninit::uninit();

Transport features

Board crates use Cargo features to select the communication transport:

  • ethernet (typically the default) – TCP/UDP via smoltcp or lwIP. Requires an Ethernet driver crate and nros-smoltcp.
  • serial – UART link via zpico-serial. Requires a UART driver crate.
  • wifi – WiFi via the platform’s native stack (ESP32). The zenoh-pico layer uses OS sockets directly.

At least one transport must be enabled. Enforce this at compile time:

#[cfg(not(any(feature = "ethernet", feature = "serial")))]
compile_error!("Enable at least one transport: `ethernet` or `serial`");

Config fields are #[cfg(feature = "...")]-gated per transport, so the struct only contains fields relevant to the enabled transport. Both ethernet and serial can be enabled simultaneously – the zenoh locator string determines which transport is used at runtime (tcp/... for Ethernet, serial/... for UART).

Reference implementations

Start from the crate closest to your target:

Board crateTransportPlatformLocation
nros-board-mps2-an385Ethernet + SerialBare-metalpackages/boards/nros-board-mps2-an385/
nros-board-mps2-an385-freertosEthernet (lwIP)FreeRTOSpackages/boards/nros-board-mps2-an385-freertos/
nros-board-esp32-qemuEthernet (OpenETH)Bare-metal (esp-hal)packages/boards/nros-board-esp32-qemu/
nros-board-stm32f4Ethernet + SerialBare-metalpackages/boards/nros-board-stm32f4/
nros-board-nuttx-qemu-armBSD socketsNuttXpackages/boards/nros-board-nuttx-qemu-arm/
nros-board-threadx-qemu-riscv64NetX DuoThreadXpackages/boards/nros-board-threadx-qemu-riscv64/

The bare-metal nros-board-mps2-an385 is the simplest starting point. The FreeRTOS variant shows how to add RTOS task creation and lwIP networking.

See also

  • Custom Platform – writing the platform crate that sits underneath a board crate
  • Porting Overview – the three customization axes (RMW, platform, board)
  • Creating Examples – detailed example layout patterns across native, BSP, and Zephyr targets

Vendor Overlay Board Crate

An overlay crate is a small (~50–150 LOC) Cargo crate that depends on a generic per-kernel board crate (nros-board-freertos, nros-board-threadx, nros-board-nuttx, nros-board-baremetal-cortex-{m,a}) and patches the deltas a specific vendor board / fork needs:

  • Vendor HAL source files (NXP fsl_*, STM HAL_*, NVIDIA FSP, …).
  • Per-chip linker script + startup assembly.
  • Custom kernel-config header (FreeRTOSConfig.h, tx_user.h).
  • Custom network-stack glue (vendor Ethernet driver wired to lwIP / NetX-Duo).
  • Custom clock-tree / pin-mux init.

This page documents the contract: what the generic crate exposes, what the overlay overrides, and how to publish a community / vendor overlay to crates.io.

Why overlays

nano-ros’s generic board crates cover the “stock RTOS source + your own drivers” workflow. Vendor SDKs (NXP MCUXpresso, STM32Cube, Espressif ESP-IDF, Renesas Synergy, NVIDIA FSP) ship forked kernels + custom drivers; bolting those into a generic crate would force a build-script branch per vendor. The overlay pattern keeps the generic crate clean: nano-ros ships the kernel-family scaffolding, vendors / community ship the per-fork glue.

See docs/roadmap/phase-152-board-bsp-abstraction-layer.md for the phase that landed the architecture.

Contract

A generic per-kernel board crate exposes:

ItemTypePurpose
ConfigstructTOML-loaded network + zenoh config; overlay can extend.
run(Config, FnOnce(&Config) -> Result<()>)functionEntry point. Initialises kernel + network, calls closure inside the app thread.
BoardInittraitHooks the overlay implements: init_clocks, init_eth, init_extra_drivers.
init_hardware()functionDefault no-op; overlay re-exports a board-specific version.

The overlay’s build.rs:

  1. Inherits the generic crate’s FREERTOS_DIR / THREADX_DIR / etc. env-var contract (overlay doesn’t override unless needed).
  2. Adds vendor HAL .c sources via its own cc::Build.
  3. Optionally regenerates the linker script (e.g. STM32F4 vs STM32F7 sector layout).

Minimal overlay shape

#![allow(unused)]
fn main() {
// nros-board-stm32f4-freertos/src/lib.rs
#![no_std]

// Re-export the generic Config + run from the upstream kernel crate.
pub use nros_board_freertos::{Config, run};

/// Board-specific clock-tree configuration.
/// Called from `run()` before lwIP init.
#[no_mangle]
pub extern "C" fn nros_board_init_clocks() {
    // HAL_RCC_OscConfig + HAL_RCC_ClockConfig + ...
}

/// Wire the STM32 ETH peripheral into lwIP.
/// Called from `run()` after kernel start, before app callback.
#[no_mangle]
pub extern "C" fn nros_board_init_eth() {
    // HAL_ETH_Init + lwIP netif_add binding
}
}
// nros-board-stm32f4-freertos/build.rs
use std::{env, path::PathBuf};

fn main() {
    let stm_hal_dir = env::var("STM32_HAL_DIR")
        .expect("set STM32_HAL_DIR to your STMicroelectronics HAL source dir");

    let mut hal = cc::Build::new();
    hal.flag("-mcpu=cortex-m4")
       .flag("-mthumb")
       .flag("-mfpu=fpv4-sp-d16")
       .flag("-mfloat-abi=hard")
       .include(format!("{stm_hal_dir}/Inc"));

    for f in &[
        "Src/stm32f4xx_hal_eth.c",
        "Src/stm32f4xx_hal_uart.c",
        "Src/stm32f4xx_hal_rcc.c",
        // ...
    ] {
        hal.file(format!("{stm_hal_dir}/{f}"));
    }
    hal.compile("stm32f4_hal");

    // Board-specific linker script wired via the generic crate's
    // BOARD_LINKER_SCRIPT_DIR env var.
    println!(
        "cargo:rustc-env=BOARD_LINKER_SCRIPT_DIR={}",
        PathBuf::from(env::var("CARGO_MANIFEST_DIR").unwrap())
            .join("config")
            .display()
    );
    println!("cargo:rerun-if-env-changed=STM32_HAL_DIR");
}
# nros-board-stm32f4-freertos/Cargo.toml
[package]
name = "nros-board-stm32f4-freertos"
version = "0.1.0"
edition = "2024"
license = "MIT OR Apache-2.0"
authors = ["Your Name <you@example.com>"]
description = "STM32F4 + FreeRTOS overlay on nros-board-freertos"
repository = "https://github.com/<you>/nros-board-stm32f4-freertos"

[dependencies]
nros-board-freertos = "0.1"

[build-dependencies]
cc = "1.0"

User application code stays identical to the generic-crate case except for the [dependencies] line:

#![allow(unused)]
fn main() {
use nros_board_stm32f4_freertos::{Config, run};
use nros::prelude::*;

run(Config::from_toml(include_str!("../config.toml")), |config| {
    let exec_config = ExecutorConfig::new(config.zenoh_locator);
    let mut executor = Executor::open(&exec_config)?;
    // ...
    Ok::<(), NodeError>(())
})
}

Canonical in-tree precedent

packages/boards/nros-board-orin-spe/ is the canonical FSP-FreeRTOS overlay (refactors it explicitly into this shape):

  • Re-exports Config + run from nros-board-freertos.
  • build.rs reads NV_SPE_FSP_DIR, pulls FreeRTOS V10.4.3 headers from NVIDIA’s FSP install.
  • Replaces lwIP with IVC link via zpico-link-ivc.
  • Provides nros_board_init_ivc() instead of init_eth().

packages/boards/nros-board-mps2-an385-freertos/ is the canonical “stock kernel + custom Ethernet driver” overlay:

  • Re-exports Config + run from nros-board-freertos.
  • build.rs adds the LAN9118 driver C sources + per-board linker script.
  • Provides LAN9118 IRQ-binding code in init_eth().

Read both for working code.

Naming convention

Publish to crates.io as nros-board-<vendor>-<chip-or-board>-<rtos>. Examples:

  • nros-board-stm32f4-freertos
  • nros-board-stm32h7-threadx
  • nros-board-nxp-mimxrt1064-freertos
  • nros-board-renesas-synergy-s7g2-threadx
  • nros-board-nordic-nrf5340-zephyr (rare — Zephyr generally owns board contract via DTS; only needed when a non-Zephyr nano-ros consumer wants to target an nRF board outside the Zephyr build)

Crates.io has no namespacing; the nros-board- prefix is the informal namespace. The nros-board- names listed audit are all unclaimed today.

What overlays DO

  • ✅ Re-export Config + run (or extend Config with vendor- specific fields and re-implement run if needed).
  • ✅ Add vendor HAL C sources via cc::Build.
  • ✅ Provide #[no_mangle] hooks the generic crate’s C glue calls (nros_board_init_clocks, nros_board_init_eth, etc.).
  • ✅ Ship board-specific config files (linker script, FreeRTOSConfig.h, tx_user.h).
  • ✅ Read vendor-SDK env vars (STM32_HAL_DIR, NXP_SDK_DIR, NV_SPE_FSP_DIR) and inject paths into cc-rs.

What overlays DON’T

  • ❌ Re-implement kernel build glue (that’s the generic crate’s job).
  • ❌ Add features that should live in the generic crate (push them upstream instead).
  • ❌ Duplicate nros-platform-<rtos> registration (the generic crate handles it).
  • ❌ Override nros-rmw-* selection (user picks RMW via Cargo features on nros, same as any nano-ros consumer).
  • ❌ Ship a fork of zenoh-pico / Cyclone DDS / mbedTLS (use the upstream’s manifest).

Testing an overlay locally

# 1. Clone or scaffold the overlay crate next to your application.
git clone https://github.com/<you>/nros-board-<your-vendor>-<rtos>

# 2. Point your application's Cargo.toml at it (path dep for dev).
[dependencies]
nros-board-<your-vendor>-<rtos> = { path = "../nros-board-<your-vendor>-<rtos>" }

# 3. Build with the vendor SDK env vars set.
export FREERTOS_DIR=$HOME/sdk/freertos/kernel
export FREERTOS_PORT=GCC/ARM_CM4F
export LWIP_DIR=$HOME/sdk/freertos/lwip
export STM32_HAL_DIR=$HOME/sdk/stm32cube/STM32F4xx_HAL_Driver
cargo build --release --target thumbv7em-none-eabihf

Skeleton template

templates/overlay-board/ ships a minimal skeleton:

  • Cargo.toml.template — deps on the generic kernel crate.
  • src/lib.rs.templatepub use re-exports + #[no_mangle] hook stubs.
  • build.rs.template — cc-rs HAL-source injection scaffold.
  • README.md.template — env vars + setup recipe.

Copy the directory, rename the placeholder, and fill in the vendor-specific bits. See templates/overlay-board/README.md for the per-file walkthrough.

Publishing to crates.io

Same flow as any Rust crate. Recommend:

  1. cargo publish --dry-run to sanity-check.
  2. Pin the generic crate dep to a minor-version range (nros-board-freertos = "0.1"); avoid ^0.0 — that won’t lock on patch bumps.
  3. Tag a release on your repo for traceability.
  4. Open a PR against book/src/getting-started/community-board-crates.md (TODO — landed by) to add a link to your crate.
  • docs/design/0012-board-bsp-integration-architecture.md — the layered model + the consumption matrix.
  • docs/roadmap/phase-152-board-bsp-abstraction-layer.md — the phase doc.
  • Custom Board Package — older guide; covers monolithic board crates before the overlay split.
  • Custom Platformnros-platform-<rtos> guide (the Layer 1 contract overlays rely on).

Importing a Board Crate

This chapter is for consumers of a nano-ros board crate – vendors and downstream applications (the Autoware Safety Island archetype) who want to bring a board into their own Zephyr app with as little glue as possible. If you are writing a new board crate, see The Board Trait Family and Custom Board Package instead.

Goal

Consume a nano-ros board crate from a downstream Zephyr application with a single CMake call. The consumer’s build script SHOULD NOT carry any of:

  • a hand-curated EXTRA_CONF_FILE listing the board crate’s prj.conf and per-board Kconfig snippets
  • a hand-curated DTC_OVERLAY_FILE
  • a hardcoded BOARD=<id> string
  • a hardcoded -DNANO_ROS_RMW=<rmw> define
  • a hand-rolled FVP / qemu launch shell

All of those are owned by the board crate. The consumer’s CMakeLists.txt declares which board it wants; nano-ros layers the rest in.

Prereqs

Before importing a board crate, the consumer needs:

  1. A Zephyr 3.7+ workspace managed by west. Zephyr 3.7 is the floor (the official in-tree zephyr-lang-rust module did not exist below it, so nothing earlier can link the nano-ros Rust staticlib); newer LTS lines work too.
  2. All gated SDK packages for the target board. Run nros doctor --board <name> to check; missing items can be installed with nros setup <board> (or nros setup --tool <t> for a single dependency). See Phase 191 / nros setup in the build commands reference.
  3. nano-ros listed as a project in your west.yml, and nano-ros’s Zephyr module exported on ZEPHYR_EXTRA_MODULES so Zephyr picks up its Kconfig + DTS roots.

A minimal west.yml fragment:

manifest:
  remotes:
    - name: newslab
      url-base: https://github.com/NEWSLabNTU
  projects:
    - name: nano-ros
      remote: newslab
      revision: main
      path: deps/nano-ros
      import: false
  self:
    west-commands: deps/nano-ros/scripts/west-commands.yml

And the matching environment:

export ZEPHYR_EXTRA_MODULES="$PWD/deps/nano-ros/zephyr"

The one-call pattern

A consumer CMakeLists.txt should look like this and nothing more:

cmake_minimum_required(VERSION 3.20)

# Single import call. Layers in BOARD, EXTRA_CONF_FILE, DTC_OVERLAY_FILE,
# NANO_ROS_RMW default, and the runner hint.
nano_ros_use_board(fvp-aemv8r-smp)

find_package(Zephyr REQUIRED HINTS $ENV{ZEPHYR_BASE})
project(my_app LANGUAGES CXX)

target_sources(app PRIVATE src/main.cpp)

Call order matters. nano_ros_use_board() MUST precede find_package(Zephyr ...). Zephyr reads BOARD, EXTRA_CONF_FILE, and DTC_OVERLAY_FILE during its find_package call; setting them after that point has no effect. The pattern above is the canonical shape – copy it verbatim and only fill in the board name.

nano_ros_use_board() is shipped by the nano-ros Zephyr module (Phase 215.B). It is available the moment ZEPHYR_EXTRA_MODULES includes deps/nano-ros/zephyr; no extra include() is required.

What the call layers in

SourceEffect
BOARDSet to NROS_BOARD_ZEPHYR_ID (from the board crate’s board.cmake) if the user did not pass -DBOARD=... on the command line.
EXTRA_CONF_FILEThe board crate’s prj.conf and any per-board Kconfig fragments (e.g. an HWv2 snippet) are appended. Any consumer-supplied EXTRA_CONF_FILE is preserved and layered AFTER the board’s.
DTC_OVERLAY_FILEThe board crate’s per-board DTS overlay is appended. Consumer overlays are preserved and layered after.
NANO_ROS_RMWDefaulted to NROS_BOARD_DEFAULT_RMW (from board.cmake) when the consumer did not pass -DNANO_ROS_RMW=....
NROS_BOARD_RUNNERCached for west fvp run (or another runner extension command) to pick up the right simulator binary / target-launcher.

The values themselves come from a single source of truth – the board crate’s board.cmake – which Phase 215.F’s drift audit keeps in sync with the crate’s Cargo.toml metadata.

Per-app overrides

The one-call pattern still gives the consumer escape hatches:

  • Pin a different Zephyr board id. Pass -DBOARD=<other> on the CMake / west build command line; nano_ros_use_board() notices the user value and emits a warning rather than clobbering it.
  • Pick a different RMW. Pass -DNANO_ROS_RMW=<rmw> (zenoh, xrce, or cyclonedds – see RMW backends). The board crate’s default is used only when nothing is set.
  • Layer extra Kconfig / DTS. Set EXTRA_CONF_FILE and DTC_OVERLAY_FILE after the call to nano_ros_use_board(). They will be applied on top of the board crate’s contributions, not in place of them.
nano_ros_use_board(fvp-aemv8r-smp)

# Extra app-specific Kconfig layered on top of the board's defaults.
list(APPEND EXTRA_CONF_FILE "${CMAKE_CURRENT_SOURCE_DIR}/boards/fvp-aemv8r-smp.conf")

find_package(Zephyr REQUIRED HINTS $ENV{ZEPHYR_BASE})
project(my_app LANGUAGES CXX)

Running

For boards whose runner is an FVP (the AEMv8-R archetype, Phase 214):

west build -d build
west fvp run -d build

west fvp run consults NROS_BOARD_RUNNER and the Phase 214.A resolver to locate the simulator binary, applies the board’s launch arguments, wires UART to stdout, and exits cleanly on Ctrl-C.

For any other runner the Zephyr-native command works unchanged:

west build -d build -t run

There is no need for a hand-written build.sh or run_fvp.sh shell wrapper.

Inspecting the manifest

To see exactly what nano_ros_use_board() will layer in for a given board:

nros board info fvp-aemv8r-smp

This prints the resolved Zephyr board id, the prj.conf + Kconfig fragment list, the DTS overlay, the default RMW, and the runner hint – everything the call would set.

To audit that the board.cmake mirror matches the canonical Cargo.toml metadata:

nros board info fvp-aemv8r-smp --check-drift

This exits 0 when the two agree and emits a field-by-field diff (with a non-zero exit code) when they don’t. The drift audit is the same one CI runs for every packages/boards/nros-board-* shipping a board.cmake.

Anti-patterns

Things that LOOK reasonable but the one-call pattern obviates:

  • Don’t hand-list the board’s prj.conf in EXTRA_CONF_FILE. It is already there; doing it again either duplicates or fights the layering order.
  • Don’t hardcode BOARD=<id> in a build.sh. The call sets it; hardcoding it short-circuits the nros board info inspection and the drift audit.
  • Don’t carry your own copy of boards/<id>.conf or boards/<id>.overlay mirroring the board crate’s. Vendor a delta only – the base ships in the crate.
  • Don’t reimplement the FVP runner as a shell script. west fvp run (Phase 214) covers FVP_BaseR_AEMv8R and the other supported simulators with the right CLI flags, the same way west build covers compilation.
  • Don’t include() files from deps/nano-ros/cmake/ directly. The public surface is nano_ros_use_board(); anything else is internal and may move.

Migrating an existing hand-glued consumer

Most downstream consumers (the ASI archetype is the canonical example) carry years of accumulated glue. The migration is mechanical:

  1. Find the per-board Kconfig and overlay entries in the current CMakeLists.txt / build.sh. Anything listing the board crate’s prj.conf, board overlay, or HWv2 snippet – delete it. Keep only entries that point at the consumer’s own deltas.
  2. Find any hardcoded BOARD=<id> string (in CMake set(BOARD ...), in west build -b <id>, or in build.sh). Delete it. nano_ros_use_board() will set it from board.cmake.
  3. Replace find_package(Zephyr) + ... + manual FVP launch with the canonical pattern above – one nano_ros_use_board() call, then find_package(Zephyr), then project(), then target_sources(app ...). Replace the FVP shell script call with west fvp run -d build.
  4. Keep only app-specific deltas in boards/<id>.conf / boards/<id>.overlay. For the ASI archetype this typically means Autoware-msg sizing (large topic / participant memory) and the application’s own GPIO map – everything generic moves into the board crate.

After migration, the consumer’s CMake should be roughly twenty lines, with nano_ros_use_board() as the only nano-ros-specific call.

Cross-references

Custom Platform

This guide walks through porting nano-ros to a new RTOS or bare-metal environment. A “platform” provides the OS-level primitives that nano-ros needs at runtime: clock, memory, sleep, threading, and networking. The core library is #![no_std] and makes zero platform calls directly – everything flows through your platform crate.

Canonical interface spec. The function-pointer signatures, parameter docs, ownership rules, blocking allowance, and failure modes for every method live in the platform-cffi Doxygen reference. Read it side-by-side with this guide — the tables below summarise the set of traits you need; the Doxygen documents the contract each call must obey.

Quick differences table. For a per-platform comparison (clock source, allocator, threading, networking, multicast support) across the platforms nano-ros already supports, see Platform Differences.

What you implement

All platform traits are defined in nros-platform/src/traits.rs. Your platform crate implements some or all of them as inherent methods on a zero-sized type (ZST). The set you need depends on your RMW backend.

Required for all backends

TraitMethodsPurpose
PlatformClockclock_ms(), clock_us()Monotonic time. Must use a hardware timer or OS tick – never a software counter that only advances when polled.

Required for zenoh-pico (rmw-zenoh)

TraitMethodsPurpose
PlatformAllocalloc(), realloc(), dealloc()Heap allocation. zenoh-pico needs ~64 KB.
PlatformSleepsleep_us(), sleep_ms(), sleep_s()Delay. On bare-metal with smoltcp, poll the network during busy-wait.
PlatformRandomrandom_u8() through random_u64(), random_fill()PRNG for session IDs and protocol nonces.
PlatformTimetime_now_ms(), time_since_epoch()Wall-clock time for logging. Return monotonic time if no RTC.
PlatformThreadingTasks, mutexes, recursive mutexes, condvars (19 methods)OS threading primitives. Single-threaded platforms provide no-op stubs.

Networking

TraitMethodsPurpose
PlatformTcpopen(), read(), send(), close(), …TCP client and server sockets.
PlatformUdpopen(), read(), send(), close(), …UDP unicast sockets.
PlatformSocketHelpersset_non_blocking(), accept(), close(), wait_event()Socket utility operations.

Optional

TraitWhen needed
PlatformUdpMulticastDesktop platforms using zenoh scouting. Not needed for embedded client mode.
PlatformNetworkPollBare-metal platforms using smoltcp. Called during sleep to process packets.
PlatformLibcBare-metal targets without a C runtime. Provides strlen, memcpy, etc.

For full method signatures, see the Platform API Reference.

Wiring into nros

Five files need changes to register a new platform. This example adds a fictional “MyOS” platform.

1. Create the platform crate

packages/core/nros-platform-myos/
  Cargo.toml
  src/
    lib.rs

The crate must have zero nros-* dependencies. It may depend on your RTOS bindings, HAL crates, or embedded-alloc.

2. Add the feature to nros-platform

In packages/core/nros-platform/Cargo.toml:

[features]
platform-myos = ["dep:nros-platform-myos"]

[dependencies]
nros-platform-myos = { version = "0.1.0", path = "../nros-platform-myos", optional = true }

3. Add the ConcretePlatform alias

In packages/core/nros-platform/src/resolve.rs:

#![allow(unused)]
fn main() {
#[cfg(feature = "platform-myos")]
pub type ConcretePlatform = nros_platform_myos::MyOsPlatform;
}

4. Propagate through the nros facade

In packages/core/nros/Cargo.toml, add platform-myos to the feature list so users can write nros = { features = ["rmw-zenoh", "platform-myos"] }.

5. Register your platform as an ABI marker

The platform shim crates are gone (Phase 129). Each transport C library’s platform symbols now come from a default-on C alias translation unit that forwards to the canonical nros_platform_* ABI — zpico-sys’s platform-aliases feature for zenoh-pico (z_* / _z_*), and nros-rmw-xrce’s always-compiled src/platform_aliases.c for XRCE-DDS (uxr_*). You do not activate or configure a shim.

What you add is a pure ABI marker feature so each transport crate’s build.rs can do per-platform source selection (e.g. strip the vendor system/<rtos>/system.c, gate the alias TU’s network section to bare-metal):

# packages/zpico/zpico-sys/Cargo.toml  (rmw-zenoh)
[features]
myos = []   # ABI marker only — no shim dependency

# packages/xrce/xrce-sys/Cargo.toml  (rmw-xrce)
[features]
myos = []   # ABI marker only

The alias TU (default-on platform-aliases) covers the full z_* surface — memory, sleep, random, time, yield, threading, condvar, clock, and network. zpico-sys’s build.rs defines NROS_PLATFORM_ALIASES and emits the network wrappers only where the vendor stack doesn’t already provide them (bare-metal). If your RTOS’s vendor system.c already supplies these symbols natively (as on Orin SPE’s FSP FreeRTOS), turn platform-aliases off for that platform to avoid duplicate-symbol link errors.

Rust path

This is the recommended approach. Create a ZST and implement each capability as inherent methods (not trait impls). nros-platform-cffi exposes these methods as the canonical nros_platform_* C symbols, which the alias TUs call.

Skeleton

#![allow(unused)]
fn main() {
// packages/core/nros-platform-myos/src/lib.rs
#![no_std]
use core::ffi::c_void;

/// Zero-sized type implementing platform methods for MyOS.
pub struct MyOsPlatform;

// -- Clock --
impl MyOsPlatform {
    pub fn clock_ms() -> u64 {
        // Call your RTOS tick API, e.g.:
        // unsafe { myos_get_tick_count() as u64 }
        todo!()
    }

    pub fn clock_us() -> u64 { Self::clock_ms() * 1000 }
}

// -- Alloc --
impl MyOsPlatform {
    pub fn alloc(size: usize) -> *mut c_void {
        // unsafe { myos_malloc(size) }
        todo!()
    }

    pub fn realloc(ptr: *mut c_void, size: usize) -> *mut c_void {
        // If your RTOS lacks realloc: alloc new, copy, free old
        todo!()
    }

    pub fn dealloc(ptr: *mut c_void) {
        // unsafe { myos_free(ptr) }
        todo!()
    }
}

// -- Sleep --
impl MyOsPlatform {
    pub fn sleep_us(us: usize) { Self::sleep_ms(us.div_ceil(1000)); }
    pub fn sleep_ms(ms: usize) {
        // unsafe { myos_thread_sleep(ms as u32) }
        todo!()
    }
    pub fn sleep_s(s: usize) { Self::sleep_ms(s * 1000); }
}

// -- Threading (stubs for single-threaded, real impls for RTOS) --
impl MyOsPlatform {
    pub fn mutex_init(m: *mut c_void) -> i8 {
        // Create a mutex via your RTOS API. Store the handle in `m`.
        // Return 0 on success, -1 on failure.
        todo!()
    }
    pub fn mutex_lock(m: *mut c_void) -> i8 { todo!() }
    pub fn mutex_unlock(m: *mut c_void) -> i8 { todo!() }
    // ... remaining threading methods (see traits.rs for the full list)
}
}

Key points

  • Inherent methods, not trait impls. The shim calls ConcretePlatform::clock_ms() directly. The traits in traits.rs document the contract, but the ZST uses inherent impl blocks.
  • c_void pointers for handles. Mutex, condvar, and task handles are opaque #[repr(C)] structs sized to hold your RTOS handle. Cast the *mut c_void to your internal type.
  • Recursive mutexes are required. zenoh-pico locks the same mutex recursively. On FreeRTOS this maps to xSemaphoreCreateRecursiveMutex; on pthreads, PTHREAD_MUTEX_RECURSIVE.
  • Seed the PRNG. A deterministic seed (like FreeRTOS rand() starting from 1) causes duplicate zenoh session IDs across QEMU instances. Seed from hardware entropy, IP address, or semihosting wall-clock.

Reference implementation

packages/core/nros-platform-freertos/src/lib.rs is a complete real-world example covering all categories: clock via xTaskGetTickCount, heap via pvPortMalloc/vPortFree, sleep via vTaskDelay, xorshift32 PRNG, and full threading with tasks, recursive mutexes, and condvars built on counting semaphores.

C/C++ path

If your platform is easier to implement in C, use the nros-platform-cffi C ABI. It is the canonical layer between nros and any C-implemented platform port.

1. Implement the platform symbols

The canonical header lives at packages/core/nros-platform-cffi/include/nros/platform.h. It declares roughly 45 free extern "C" functions — one per platform capability. Your port supplies a definition for each; the linker resolves them into the nros binary directly. There is no runtime registration call. Browse the rendered reference at /api/platform-cffi/ for per-function return-value, threading, and blocking conventions.

Include the header in your port:

// my_platform.c
#include <nros/platform.h>

Then define each symbol:

uint64_t nros_platform_clock_ms(void) {
    return myos_get_ticks();  // your RTOS tick API
}

void *nros_platform_alloc(size_t size) {
    return myos_malloc(size);
}

void nros_platform_dealloc(void *ptr) {
    myos_free(ptr);
}

void *nros_platform_realloc(void *ptr, size_t size) {
    return myos_realloc(ptr, size);
}

int8_t nros_platform_mutex_init(void *m) {
    myos_mutex_t *mx = (myos_mutex_t *)m;
    *mx = myos_mutex_create();
    return (*mx != NULL) ? 0 : -1;
}

/* ... define every other symbol declared in <nros/platform.h> ... */

Compile your translation unit into a static or object library and link it into the nros binary alongside the nros static library. No registration step is required at boot:

int main(void) {
    myos_init();

    /* nros calls nros_platform_* symbols directly */
    nros_executor_t exec;
    nros_executor_open(&exec, &config);
    /* ... */
}

3. Build configuration

Enable the platform-cffi feature instead of a platform-specific feature:

nros = { features = ["rmw-zenoh", "platform-cffi"] }

The rmw-zenoh feature is the lowering of the declared RMW: the backend is declared once in system.toml ([system].rmw / [deploy.<t>].rmw) and the toolchain emits the cargo feature. The feature is the build mechanism, not the user-facing knob — see RFC-0031.

All symbols declared in <nros/platform.h> are required. For capabilities your platform does not support (e.g., threading on single-threaded bare-metal), supply stubs that return 0 for mutex/condvar operations and -1 for nros_platform_task_init.

Networking

There are two paths for providing TCP/UDP sockets to zenoh-pico.

Option A: Rust networking (preferred)

Implement PlatformTcp, PlatformUdp, and PlatformSocketHelpers on your ZST. These methods map to your OS socket API (BSD sockets, lwIP, NetX Duo, etc.).

#![allow(unused)]
fn main() {
impl MyOsPlatform {
    pub fn tcp_open(sock: *mut c_void, endpoint: *const c_void, timeout_ms: u32) -> i8 {
        // Parse endpoint, call connect(), store fd in sock
        todo!()
    }

    pub fn tcp_read(sock: *const c_void, buf: *mut u8, len: usize) -> usize {
        // Call recv() on the socket fd
        todo!()
    }

    pub fn tcp_send(sock: *const c_void, buf: *const u8, len: usize) -> usize {
        // Call send() on the socket fd
        todo!()
    }

    pub fn tcp_close(sock: *mut c_void) {
        // Close the socket fd
    }
}
}

Activate the network shim feature in zpico-sys so the shim provides the _z_open_tcp, _z_read_tcp, etc. C symbols by forwarding to your Rust methods.

For bare-metal with smoltcp, use nros-smoltcp (in packages/drivers/) as the networking driver. It provides PlatformTcp and PlatformUdp implementations using smoltcp’s TCP/UDP sockets. The MAC/PHY driver lives in a sibling driver crate (e.g. lan9118-smoltcp, openeth-smoltcp) and implements smoltcp’s Device trait; nros-smoltcp consumes that Device and exposes the transport to zenoh-pico. Your platform crate implements PlatformNetworkPoll so the sleep loop can process packets — the platform crate stays free of smoltcp itself.

Option B: Keep zenoh-pico’s C network.c

If your platform already has a working zenoh-pico network.c (e.g., freertos/lwip/network.c or unix/network.c), you can compile it directly instead of implementing the Rust networking traits.

In this case, do not activate the network shim feature in zpico-sys. Instead, link the appropriate network.c through your build system. The C file provides the _z_open_tcp, _z_read_tcp, etc. symbols directly, bypassing the Rust shim.

This is the approach used by platforms with mature C networking stacks (lwIP on FreeRTOS, BSD sockets on NuttX, NetX Duo on ThreadX).

Common pitfalls

  • Poll-driven clocks. If the clock only advances when you call a function, timeouts and keep-alives break silently. Use a free-running hardware timer.
  • Stack overflow on RTOS. The Executor has an inline arena on the task stack. Use at least 16384 words (64 KB) for the application task on action examples.
  • Deterministic PRNG seeds. Duplicate zenoh session IDs cause silent connection failures. Seed from a source that varies across instances.
  • Missing recursive mutexes. zenoh-pico re-enters the same mutex. Non-recursive mutexes deadlock.
  • QEMU clock drift. Use -icount shift=auto for QEMU targets so the virtual clock tracks wall time during WFI.

Next steps

Adding a Platform (CMake Layer)

Scope. This page covers the CMake-side glue: how the build system picks the right platform shim, links the right archives, and dispatches per-app fixups (linker scripts, startup files, ISR vectors). For the runtime traits a new platform must implement in Rust / platform-cffi, see Custom Platform.

Since, porting a new platform to nano-ros is one new file:

nano-ros/cmake/platform/nano-ros-<plat>.cmake

Adding a platform no longer touches the root CMakeLists.txt, never edits per-example trees, and never duplicates platform helpers across 20+ examples.

Module contract (§A)

Every cmake/platform/nano-ros-<plat>.cmake must expose:

SymbolKindPurpose
NanoRos::PlatformALIAS libraryAliased to the platform-specific INTERFACE target; linked into NanoRos::NanoRos umbrella by the root CMakeLists.
nros_platform_<plat>_ifaceINTERFACE libraryCarries the platform staticlib + host-system libs + transitive deps.
nros_platform_link_app(target)functionPer-app fixups: linker script, startup objects, ISR vectors, RTOS-specific final-link tweaks. Empty for POSIX.
NROS_PLATFORM_LINK_FEATUREScache variableDefault link-feature set for this platform (e.g. tcp udp_unicast).

The root CMakeLists.txt dispatches via:

include("${CMAKE_CURRENT_SOURCE_DIR}/cmake/platform/nano-ros-${NANO_ROS_PLATFORM}.cmake")

When the user’s project sets NANO_ROS_PLATFORM=foo before add_subdirectory(nano-ros), your module at cmake/platform/nano-ros-foo.cmake runs and supplies the four contract elements above.

Minimal skeleton

# cmake/platform/nano-ros-foo.cmake

if(DEFINED _NROS_PLATFORM_FOO_INCLUDED)
    return()
endif()
set(_NROS_PLATFORM_FOO_INCLUDED TRUE)

set(NROS_PLATFORM_LINK_FEATURES tcp udp_unicast
    CACHE STRING "Default link features for the Foo platform")

# Build / pull in the platform staticlib. May be add_subdirectory(...)
# into packages/core/nros-platform-foo/ for a Cargo + cmake hybrid, or
# may declare an IMPORTED target pointing at a prebuilt RTOS archive.
add_subdirectory(
    "${CMAKE_CURRENT_LIST_DIR}/../../packages/core/nros-platform-foo"
    nros_platform_foo_build)

add_library(nros_platform_foo_iface INTERFACE)
if(TARGET nros_platform_foo)
    target_link_libraries(nros_platform_foo_iface INTERFACE nros_platform_foo)
endif()

if(NOT TARGET NanoRos::Platform)
    add_library(NanoRos::Platform ALIAS nros_platform_foo_iface)
endif()

function(nros_platform_link_app target)
    # Per-app fixups go here. Linker scripts, startup objects, ISR
    # vectors. Delegate to cmake/board/nano-ros-board-${NANO_ROS_BOARD}.cmake
    # when the platform supports multiple boards.
    if(DEFINED NANO_ROS_BOARD)
        include("${CMAKE_CURRENT_LIST_DIR}/../board/nano-ros-board-${NANO_ROS_BOARD}.cmake")
        if(COMMAND nros_board_link_app)
            nros_board_link_app(${target})
        endif()
    endif()
endfunction()

That’s it. The root CMakeLists picks up the new platform automatically — the validation block resolves cmake/platform/nano-ros-foo.cmake and fatals out only when no such file exists.

Board overlays

When your platform spans multiple boards (different MCUs, link scripts, peripheral inits), carve a board overlay:

nano-ros/cmake/board/nano-ros-board-<board>.cmake

Board overlays provide nros_board_link_app(target) and run inside nros_platform_link_app when the user sets NANO_ROS_BOARD. The board layer owns:

  • Linker script selection (target_link_options(${target} PRIVATE -T<script>))
  • Startup object emission (crt0.o, vector table, etc.)
  • MCU-specific final-link flags

See cmake/board/nano-ros-board-mps2-an385.cmake for a working example.

Where the existing layer-2 helpers live

For RTOS ports that compose the kernel + netstack + glue inside CMake (FreeRTOS, ThreadX, NuttX), the per-RTOS helper functions (nros_freertos_build_kernel, nros_threadx_compose_platform, …) stay at packages/core/nros-c/cmake/nros-<rtos>.cmake. The modules include(...) those helpers — the per-platform file is the dispatch entry, not the implementation.

See also

Custom Transport

nano-ros lets you plug a custom transport (USB-CDC, BLE GATT, RS-485 with framing, ring-buffer loopback, semihosting bridge, …) at runtime, without changing the board crate, Cargo features, or rebuilding. This is the runtime equivalent of micro-ROS’s rmw_uros_set_custom_transport(framing, params, open, close, write, read).

When to use

  • You have a serial-over-USB device that doesn’t fit serial, ethernet, or wifi.
  • You’re bridging through a host-side proxy (semihosting, RTT, OpenOCD).
  • You’re prototyping a transport on a target where the static path isn’t built yet.
  • You want a single firmware image that picks transport at boot from a config block.

If your transport fits one of the prebuilt static variants (platform-posix, platform-freertos, platform-nuttx, platform-threadx, platform-zephyr, platform-bare-metal), prefer that — the runtime hook trades binary-size optimisation for flexibility.

API layering (L0 / L1 / L2)

The custom transport surface is the project’s first canonical-C-ABI interface, designed per docs/design/0006-portable-rmw-platform-interface.md:

LayerOwnsCrates / files
L0 — canonical C ABI#[repr(C)] struct + abi_version: u32 field + four unsafe extern "C" fn pointers + user_data: *mut c_voidnros-rmw::custom_transport (Rust source); <nros/transport.h> is the cbindgen-emitted C header
L1 — language wrappersmechanical glue, no new design decisionsnros-rmw::set_custom_transport (Rust); nros_set_custom_transport (C); nros::set_custom_transport (C++)
L2 — typed app APIn/a — transport is platform-side

All design decisions live at L0. A new feature — say, a flush callback — lands in the L0 struct first; L1 wrappers follow mechanically.

The same canonical-C-ABI pattern was generalised to the full RMW backend surface (nros_rmw_vtable_t). For the host-language policy that decides whether a given backend lives in Rust, C, or C++, see RMW Backends — Host-Language Policy.

ABI versioning

The abi_version: u32 field at the head of the struct is mandatory. Consumers must fill in NROS_TRANSPORT_OPS_ABI_VERSION_V1 (Rust: nros_rmw::NROS_TRANSPORT_OPS_ABI_VERSION_V1; C: NROS_TRANSPORT_OPS_ABI_VERSION_V1; C++: filled in automatically by nros::set_custom_transport). Mismatched versions are rejected at registration time with NROS_RMW_RET_INCOMPATIBLE_ABI (-14); the slot stays whatever it was before the bad call.

The version bumps under two rules:

  • Major (e.g. V1V2): existing fields are removed or reordered. Old consumers fail cleanly via the version check.
  • Minor (struct gains an appended fn pointer / data field): version stays the same. New consumers detect the new fn via the size of the trailing _reserved region. Today there’s no such appendage — V1 is the inaugural version.

Implementing in another language

The L0 struct is plain C ABI (#[repr(C)] Rust ↔ struct C). Any language with C-FFI support can author both sides of the boundary — Zig, Python (ctypes / cffi), Lua-FFI, Go (cgo), Swift (@_cdecl), etc.

The reference implementation of “custom transport written in pure C” lives at packages/core/nros-rmw-cffi/tests/c_stubs/c_stub_transport.c. ~80 LOC; no Rust headers / cbindgen output / Rust types involved on the C side. Use it as a template for ports to other languages.

The corresponding Rust integration test (tests/c_stub_transport.rs) exercises the round-trip: register the C-built struct → drive each fn pointer from Rust → confirm the C-side counters bumped → confirm abi_version mismatch is rejected. Run via:

cargo test -p nros-rmw-cffi --features c-stub-test --test c_stub_transport

API

Rust

#![allow(unused)]
fn main() {
use core::ffi::c_void;
use nros_rmw::{NrosTransportOps, set_custom_transport};

unsafe extern "C" fn my_open(_ud: *mut c_void, _params: *const c_void) -> i32 { 0 }
unsafe extern "C" fn my_close(_ud: *mut c_void) {}
unsafe extern "C" fn my_write(_ud: *mut c_void, buf: *const u8, len: usize) -> i32 {
    // hand `buf[..len]` to the underlying medium, return 0 on success
    0
}
unsafe extern "C" fn my_read(_ud: *mut c_void, buf: *mut u8, len: usize, timeout_ms: u32) -> i32 {
    // read up to `len` bytes within `timeout_ms`, return non-negative count
    0
}

unsafe {
    set_custom_transport(Some(NrosTransportOps {
        abi_version: nros_rmw::NROS_TRANSPORT_OPS_ABI_VERSION_V1,
        _reserved: 0,
        user_data: my_uart_handle as *mut c_void,
        open: my_open,
        close: my_close,
        write: my_write,
        read: my_read,
    })).expect("abi_version must match runtime");
}
}

C

#include <nros/transport.h>

static nros_ret_t my_open(void *ud, const void *params) {
    (void)params;
    return my_uart_open((my_uart_t *)ud);
}
static void my_close(void *ud) { my_uart_close((my_uart_t *)ud); }
static nros_ret_t my_write(void *ud, const uint8_t *buf, size_t len) {
    return my_uart_write((my_uart_t *)ud, buf, len);
}
static int32_t my_read(void *ud, uint8_t *buf, size_t len, uint32_t timeout_ms) {
    return my_uart_read((my_uart_t *)ud, buf, len, timeout_ms);
}

int main(void) {
    nros_transport_ops_t ops = {
        .abi_version = NROS_TRANSPORT_OPS_ABI_VERSION_V1,
        ._reserved = 0,
        .user_data = &g_uart,
        .open = my_open,
        .close = my_close,
        .write = my_write,
        .read = my_read,
    };
    nros_set_custom_transport(&ops);

    // ... continue with nros_support_init, nros_node_init, etc.
}

C++

#include <nros/transport.hpp>

nros::TransportOps ops;
ops.user_data = &g_uart;
ops.open  = [](void *ud, const void*) -> int { return my_uart_open((MyUart*)ud); };
ops.close = [](void *ud)               { my_uart_close((MyUart*)ud); };
ops.write = [](void *ud, const uint8_t *buf, std::size_t len) -> int {
    return my_uart_write((MyUart*)ud, buf, len);
};
ops.read  = [](void *ud, uint8_t *buf, std::size_t len, std::uint32_t to) -> std::int32_t {
    return my_uart_read((MyUart*)ud, buf, len, to);
};
auto r = nros::set_custom_transport(ops);
NROS_TRY(r);

Captureless lambdas only. All four fields are raw C function pointers, not std::function. Pass per-instance state through user_data.

Threading contract

ConstraintRationale
read and write MAY be invoked concurrently from different threads on threaded backends.The zenoh-pico backend runs a dedicated read-task thread on multi-threaded platforms (POSIX, Zephyr, FreeRTOS, NuttX), so read runs there while the application/tx thread drives write. Single-threaded platforms (bare-metal, ThreadX) instead serialise both through the spin-once / drive_io path. A custom transport must therefore be full-duplex safe: do not serialise read and write behind one lock held across a blocking read, or the read thread will starve the writer and deadlock session declaration. Most media (TCP, full-duplex UART/USB-CDC) already allow concurrent read and write; mirror that. With std::net::TcpStream, store the stream directly and use &TcpStream (which implements Read/Write) from both callbacks rather than a Mutex<TcpStream>.
Callbacks must NOT be invoked from interrupt context.The runtime path may take internal locks; ISR context could deadlock. Wrap ISR-driven hardware in a queue + read poller.
user_data must outlive the transport’s active period.The runtime never copies it. Lifetime is from the first callback invocation through close returning.
set_custom_transport must be called BEFORE nros_support_init.The active backend reads the slot during Rmw::open. Calling after init is implementation-defined — backends may reject with NROS_RET_ALREADY_INIT.

Return-code conventions

  • open / write return 0 (NROS_RMW_RET_OK) on success, a negative nros_ret_t (e.g. NROS_RMW_RET_TIMEOUT, NROS_RMW_RET_ERROR) on failure.
  • read returns the non-negative byte count on success (may be less than len); a negative nros_ret_t on error / timeout.
  • close returns nothing — failures during teardown are best-effort.

Framing

Some transports need wire-level framing (HDLC for serial, length-prefix for stream sockets). The active backend decides whether framing is applied; the user vtable just sees raw bytes.

  • XRCE-DDS: pass framing=true to the backend’s init_transport_from_custom_ops(framing) for byte-stream transports (UART, USB-CDC). Pass framing=false for packet-oriented transports (UDP, BLE GATT).
  • zenoh-pico: framing is built into the wire protocol — always framing=false regardless of the underlying medium.

Backend coverage

BackendStatus
XRCE-DDS✅ Wired. nros_rmw_xrce::init_transport_from_custom_ops(framing) pulls the registered vtable into uxr_set_custom_transport_callbacks via four C trampolines.
zenoh-pico🟡 Deferred (zenoh). zenoh-pico’s custom-link API needs a per-platform _z_link_t shim; tracked separately. Zenoh users with custom transports today fork a zpico-platform-* crate.
Cyclone DDS➖ N/A. Cyclone DDS binds the platform’s network stack directly (POSIX / NSOS / lwIP / NetX-Duo BSD sockets) rather than going through the nano-ros custom-transport vtable; “porting” Cyclone to a new link means providing those sockets, not a transport plug-in.

Loopback test

packages/xrce/nros-rmw-xrce/tests/custom_transport.rs exercises the slot lifecycle + the XRCE bridge round-trip with stub callbacks (no real session). Run via:

cargo test -p nros-rmw-xrce --features platform-posix \
    --test custom_transport

See also

Porting a Custom RMW Backend

nano-ros ships with three RMW backends – zenoh-pico, Micro-XRCE-DDS, and Cyclone DDS. To add your own transport (MQTT, a proprietary bus, etc.), implement a small set of traits or fill in a C function table.

Two paths are available:

  • Rust path – implement the nros-rmw traits directly.
  • C/C++ path – fill in nros_rmw_vtable_t and register it at startup via nros-rmw-cffi.

What you implement

Your backend provides concrete types for six traits:

Rmw                    -- factory: opens a Session from RmwConfig
  Session              -- connection lifecycle, creates handles
    Publisher          -- send CDR-encoded messages
    Subscriber         -- non-blocking receive (poll-based)
    ServiceServerTrait -- receive requests, send replies
    ServiceClientTrait -- send requests, poll for replies

Most methods have default implementations. The required methods per trait are:

TraitRequired methods
Rmwopen()
Sessioncreate_publisher(), create_subscriber(), create_service_server(), create_service_client(), close()
Publisherpublish_raw(), buffer_error(), serialization_error()
Subscribertry_recv_raw(), deserialization_error()
ServiceServerTraittry_recv_request(), send_reply()
ServiceClientTraitsend_request_raw(), try_recv_reply_raw()

For full trait signatures and associated types, see RMW API Reference.

Use nros-platform for networking

Call ConcretePlatform::tcp_open() / udp_bind() from nros-platform rather than OS sockets directly. This makes your backend portable across every platform (POSIX, Zephyr, FreeRTOS, NuttX, ThreadX, bare-metal). If your transport library already abstracts networking (like zenoh-pico does), you can use its own I/O layer instead.


Rust path

1. Create the crate

Create packages/myproto/nros-rmw-myproto/ with nros-rmw and nros-core as dependencies (both default-features = false for no_std support). Follow the std/alloc feature forwarding pattern used by the existing backends.

2. Implement the traits

#![allow(unused)]
#![no_std]
fn main() {
use nros_rmw::*;

#[derive(Default)]
pub struct MyProtoRmw {
    // Optional pre-open configuration (agent address, TLS CA, serial
    // device handle, …) can live here and move into the Session at
    // `open` time. Every backend in-repo implements `Default` so the
    // caller can spell the common case as
    // `MyProtoRmw::default().open(&config)`.
}
impl Rmw for MyProtoRmw {
    type Session = MyProtoSession;
    type Error = TransportError;
    fn open(self, config: &RmwConfig) -> Result<MyProtoSession, TransportError> {
        todo!() // Parse config.locator, connect, map config.domain_id
    }
}

pub struct MyProtoSession { /* connection state */ }
impl Session for MyProtoSession {
    type Error = TransportError;
    type PublisherHandle = MyProtoPub;
    type SubscriberHandle = MyProtoSub;
    type ServiceServerHandle = MyProtoServer;
    type ServiceClientHandle = MyProtoClient;

    fn create_publisher(&mut self, t: &TopicInfo, q: QosSettings)
        -> Result<MyProtoPub, TransportError> { todo!() }
    fn create_subscriber(&mut self, t: &TopicInfo, q: QosSettings)
        -> Result<MyProtoSub, TransportError> { todo!() }
    fn create_service_server(&mut self, s: &ServiceInfo)
        -> Result<MyProtoServer, TransportError> { todo!() }
    fn create_service_client(&mut self, s: &ServiceInfo)
        -> Result<MyProtoClient, TransportError> { todo!() }
    fn close(&mut self) -> Result<(), TransportError> { todo!() }
    fn drive_io(&mut self, timeout_ms: i32) -> Result<(), TransportError> {
        let _ = timeout_ms; Ok(()) // poll network, dispatch to buffers
    }
}

pub struct MyProtoPub;
impl Publisher for MyProtoPub {
    type Error = TransportError;
    fn publish_raw(&self, data: &[u8]) -> Result<(), TransportError> { todo!() }
    fn buffer_error(&self) -> TransportError { TransportError::BufferTooSmall }
    fn serialization_error(&self) -> TransportError { TransportError::SerializationError }
}

pub struct MyProtoSub;
impl Subscriber for MyProtoSub {
    type Error = TransportError;
    fn try_recv_raw(&mut self, buf: &mut [u8])
        -> Result<Option<usize>, TransportError> { todo!() }
    fn deserialization_error(&self) -> TransportError { TransportError::DeserializationError }
}

pub struct MyProtoServer;
impl ServiceServerTrait for MyProtoServer {
    type Error = TransportError;
    fn try_recv_request<'a>(&mut self, buf: &'a mut [u8])
        -> Result<Option<ServiceRequest<'a>>, TransportError> { todo!() }
    fn send_reply(&mut self, seq: i64, data: &[u8])
        -> Result<(), TransportError> { todo!() }
}

pub struct MyProtoClient;
impl ServiceClientTrait for MyProtoClient {
    type Error = TransportError;
    fn send_request_raw(&mut self, req: &[u8])
        -> Result<(), TransportError> { todo!() }
    fn try_recv_reply_raw(&mut self, buf: &mut [u8])
        -> Result<Option<usize>, TransportError> { todo!() }
}
}

Pick the right TransportError variant

The runtime maps every TransportError variant to a named nros_rmw_ret_t constant at the C boundary. Picking a specific variant gives the caller actionable information instead of an opaque “failed somehow”:

VariantWhen to use
TimeoutBounded wait elapsed (drive_io, blocking call_raw).
WouldBlockResource momentarily unavailable; caller should retry. Distinct from NoDataNoData means the queue is empty, WouldBlock means it’s contended.
NoDataNon-blocking receive found the queue empty.
BufferTooSmallCaller’s &mut [u8] smaller than the next message.
MessageTooLargeIncoming message exceeds the backend’s static capacity.
InvalidArgumentNULL pointer, out-of-range value, missing required config.
UnsupportedBackend genuinely cannot perform this operation (e.g., uORB has no service surface).
IncompatibleQosEndpoints’ QoS profiles differ in a way the backend cannot reconcile.
TopicNameInvalidTopic / service name failed validation (empty, too long, illegal characters).
BadAllocAllocation failed on a heap-equipped backend.
LoanNotSupportedLending requested on an entity that doesn’t support it.
ConnectionFailed / DisconnectedTransport-level failures.
PublisherCreationFailed / SubscriberCreationFailed / ServiceServerCreationFailed / ServiceClientCreationFailedCatch-all for entity creation when no more specific code applies.
Backend(&str) / BackendDynamic(String)Backend-specific diagnostic that doesn’t map to any of the above.

The Rust trait surface is the source of truth; the C header <nros/rmw_ret.h> exposes the matching NROS_RMW_RET_* constants for C porters.

Factory shape

Rmw::open consumes self, not a &self. That shape asks every backend to treat its factory type as a value that carries any pre-open configuration (agent address, serial device, TLS CA, …) and moves that state into the returned Session:

// Default constructor (picks config from `&RmwConfig`):
let session = MyProtoRmw::default().open(&config)?;

// Explicit constructor when the backend has pre-open state
// that isn't in the middleware-agnostic `RmwConfig`:
let session = MyProtoRmw::with_endpoint("10.0.0.1", 7447).open(&config)?;

Conventions:

  • Every backend implements Default. Keeps the common call site short and lets generic code build a factory without knowing the backend type.
  • Provide new(...) / with_*(...) helpers for backend-specific pre-open state. Don’t bake it into RmwConfig — that type is the middleware-agnostic contract. If your backend needs an agent IP, a serial device path, or a certificate slot, take it on the factory constructor.
  • Read your own environment variables in <Backend>::from_env() if you want zero-boilerplate POSIX configuration. The shipped ExecutorConfig::from_env() only reads the middleware-agnostic NROS_LOCATOR / NROS_SESSION_MODE / ROS_DOMAIN_ID; anything backend-specific (e.g. NROS_XRCE_AGENT) stays on the backend side.
  • Post-open state lives in Session, never in static mut. The open(self, …) signature makes it natural to move the configured transport into the Session return value, which then owns the connection for the rest of its lifetime. A backend that still uses static mut session-global state will fail any multi-session test (backend.open(...) twice in one process should succeed).

3. Wire into nros

Three changes are needed to integrate the new backend:

a) In nros/Cargo.toml, add a feature and optional dependency:

rmw-myproto = ["dep:nros-rmw-myproto", "nros-node/rmw-myproto"]

b) In nros-node, add the concrete session type alias:

#![allow(unused)]
fn main() {
#[cfg(feature = "rmw-myproto")]
pub type ConcreteSession = nros_rmw_myproto::MyProtoSession;
}

c) Add compile_error! guards to enforce mutual exclusivity with the other backends (see existing guards in nros-node/src/session.rs).

Applications then select your backend with nros = { features = ["rmw-myproto", "platform-posix"] }.


C/C++ path

If your transport library is C or C++, use nros-rmw-cffi — a vtable of C function pointers that map one-to-one onto the Rust trait methods.

The hand-written header lives at packages/core/nros-rmw-cffi/include/nros/rmw_vtable.h. Browse the rendered reference at /api/rmw-cffi/ for per-field return-value, threading, and blocking conventions.

1. Fill in the vtable

#include <nros/rmw_vtable.h>
#include <nros/rmw_ret.h>
#include <nros/rmw_entity.h>

// -- Session lifecycle --
static nros_rmw_ret_t my_open(const char *locator, uint8_t mode,
                              uint32_t domain_id, const char *node_name,
                              nros_rmw_session_t *out) {
    /* Connect. On success: write out->backend_data with your session
     * pointer, return NROS_RMW_RET_OK. On failure: return one of the
     * named codes (NROS_RMW_RET_INVALID_ARGUMENT, NROS_RMW_RET_TIMEOUT,
     * NROS_RMW_RET_BAD_ALLOC, NROS_RMW_RET_ERROR). out->node_name is
     * already set by the runtime — read it for diagnostics. */
}
static nros_rmw_ret_t my_close(nros_rmw_session_t *session) { /* ... */ }
static nros_rmw_ret_t my_drive_io(nros_rmw_session_t *session, int32_t timeout_ms) {
    /* Dispatch network I/O for up to timeout_ms; return NROS_RMW_RET_OK
     * on success, NROS_RMW_RET_TIMEOUT / NROS_RMW_RET_ERROR otherwise. */
}

// -- Publisher --
static nros_rmw_ret_t my_create_publisher(
        nros_rmw_session_t *session,
        const char *topic, const char *type_name, const char *type_hash,
        uint32_t domain_id, const nros_rmw_qos_t *qos,
        nros_rmw_publisher_t *out) {
    /* Runtime has already filled out->topic_name / type_name / qos.
     * Backend writes out->backend_data with its publisher handle and
     * may set out->can_loan_messages = true to advertise the
     * loan_publish / commit_publish primitive. */
}
static void           my_destroy_publisher(nros_rmw_publisher_t *publisher) { /* ... */ }
static nros_rmw_ret_t my_publish_raw(nros_rmw_publisher_t *publisher,
        const uint8_t *data, size_t len) { /* ... */ }

// -- Subscriber --
static nros_rmw_ret_t my_create_subscriber(
        nros_rmw_session_t *session,
        const char *topic, const char *type_name, const char *type_hash,
        uint32_t domain_id, const nros_rmw_qos_t *qos,
        nros_rmw_subscriber_t *out) { /* ... */ }
static void    my_destroy_subscriber(nros_rmw_subscriber_t *subscriber) { /* ... */ }
static int32_t my_try_recv_raw(nros_rmw_subscriber_t *subscriber,
        uint8_t *buf, size_t buf_len) {
    /* >= 0 = bytes received (0 = no data),
     * negative nros_rmw_ret_t (e.g. NROS_RMW_RET_NO_DATA, _BUFFER_TOO_SMALL). */
}
static int32_t my_has_data(nros_rmw_subscriber_t *subscriber) { /* 1 = yes, 0 = no */ }

// -- Service Server --
static nros_rmw_ret_t my_create_service_server(
        nros_rmw_session_t *session,
        const char *service, const char *type_name, const char *type_hash,
        uint32_t domain_id,
        nros_rmw_service_server_t *out) { /* ... */ }
static void    my_destroy_service_server(nros_rmw_service_server_t *server) { /* ... */ }
static int32_t my_try_recv_request(nros_rmw_service_server_t *server,
        uint8_t *buf, size_t buf_len, int64_t *seq_out) { /* ... */ }
static int32_t my_has_request(nros_rmw_service_server_t *server) { /* 1 = yes, 0 = no */ }
static nros_rmw_ret_t my_send_reply(nros_rmw_service_server_t *server,
        int64_t seq, const uint8_t *data, size_t len) { /* ... */ }

// -- Service Client --
static nros_rmw_ret_t my_create_service_client(
        nros_rmw_session_t *session,
        const char *service, const char *type_name, const char *type_hash,
        uint32_t domain_id,
        nros_rmw_service_client_t *out) { /* ... */ }
static void    my_destroy_service_client(nros_rmw_service_client_t *client) { /* ... */ }
static int32_t my_call_raw(nros_rmw_service_client_t *client,
        const uint8_t *request, size_t req_len,
        uint8_t *reply_buf, size_t reply_buf_len) { /* ... */ }

static const nros_rmw_vtable_t MY_RMW = {
    .open                   = my_open,
    .close                  = my_close,
    .drive_io               = my_drive_io,
    .create_publisher       = my_create_publisher,
    .destroy_publisher      = my_destroy_publisher,
    .publish_raw            = my_publish_raw,
    .create_subscriber      = my_create_subscriber,
    .destroy_subscriber     = my_destroy_subscriber,
    .try_recv_raw           = my_try_recv_raw,
    .has_data               = my_has_data,
    .create_service_server  = my_create_service_server,
    .destroy_service_server = my_destroy_service_server,
    .try_recv_request       = my_try_recv_request,
    .has_request            = my_has_request,
    .send_reply             = my_send_reply,
    .create_service_client  = my_create_service_client,
    .destroy_service_client = my_destroy_service_client,
    .call_raw               = my_call_raw,
};

2. Register before opening a session

int main(void) {
    nros_rmw_cffi_register(&MY_RMW);    // before any nros call
    /* now use the nano-ros C or C++ API normally */
}

Build the static library with the matching feature combo:

cargo build -p nros-c --features rmw-cffi,platform-posix,ros-humble

3. Lifecycle and threading contract

The Rust traits behind this vtable (nros_rmw::Session, Publisher, …) document the per-method contract: thread safety, buffer ownership, blocking allowance. The C vtable inherits the same rules:

  • The vtable itself is registered once and read concurrently. Function pointers must be safe to invoke from any executor thread.
  • drive_io may block up to timeout_ms; it must not hold application-visible locks across the wait.
  • publish_raw, try_recv_raw, and send_reply may run concurrently from different executor threads — your backend handles serialisation.
  • try_recv_raw and try_recv_request are non-blocking: return 0 if no data is ready. The executor will retry after drive_io.
  • call_raw is the deprecated blocking client path. In-tree backends route blocking waits through the executor instead. Implement it as a polling loop only if you need to support legacy callers.

All strings are null-terminated and borrowed (caller owns the storage). Entities are typed structs with an opaque backend_data slot — the runtime fills metadata fields (topic_name, qos, …) before calling create_*; the backend writes backend_data. Return convention: NROS_RMW_RET_OK = success, negative = named nros_rmw_ret_t constant, positive = byte count (only on try_recv_* / call_raw).


Example: local echo RMW

Loops published messages back to subscribers – no real transport. Only pub/sub shown; service types are no-op stubs.

#![allow(unused)]
fn main() {
static mut ECHO_BUF: [u8; 1024] = [0; 1024];
static mut ECHO_LEN: usize = 0;

pub struct EchoPub;
impl Publisher for EchoPub {
    type Error = TransportError;
    fn publish_raw(&self, data: &[u8]) -> Result<(), TransportError> {
        unsafe {
            let len = data.len().min(ECHO_BUF.len());
            ECHO_BUF[..len].copy_from_slice(&data[..len]);
            ECHO_LEN = len;
        }
        Ok(())
    }
    fn buffer_error(&self) -> TransportError { TransportError::BufferTooSmall }
    fn serialization_error(&self) -> TransportError { TransportError::SerializationError }
}

pub struct EchoSub;
impl Subscriber for EchoSub {
    type Error = TransportError;
    fn try_recv_raw(&mut self, buf: &mut [u8]) -> Result<Option<usize>, TransportError> {
        unsafe {
            if ECHO_LEN == 0 { return Ok(None); }
            let len = ECHO_LEN;
            buf[..len].copy_from_slice(&ECHO_BUF[..len]);
            ECHO_LEN = 0;
            Ok(Some(len))
        }
    }
    fn deserialization_error(&self) -> TransportError { TransportError::DeserializationError }
}
}

Wire EchoPub/EchoSub into an EchoSession the same way as the skeleton above – create_publisher returns Ok(EchoPub), etc. The Rmw::open() impl just returns Ok(EchoSession) unconditionally.


What the ROS 2 ecosystem expects

Implementing the six traits compiles and runs, but a backend that stops there will not interoperate cleanly with ros2 CLI, RQt, or rmw_zenoh_cpp nodes. Real ROS 2 interop requires four extra invariants the traits do not express:

1. Discovery / liveliness tokens

ros2 node list, ros2 topic list, ros2 service list rely on discovery traffic. How you emit it depends on the backend protocol:

  • Zenoh-flavoured backends: publish a liveliness token per endpoint under @ros2_lv/<domain>/<zid>/<entity_kind>/<id>/…. See rmw-zenoh-protocol.md for the exact key grammar.
  • DDS-flavoured backends: use the SPDP/SEDP discovery traffic that your DDS stack provides, plus the ROS 2–specific USER_DATA payload (node name, namespace, enclave).

If your backend is brand new (not wire-compatible with zenoh or DDS), you still need some discovery channel for ros2 CLI tools to find your endpoints. The traits currently do not cover this — discovery happens inside create_publisher / create_subscriber / create_service_* as a side effect.

2. RMW attachments (per-message metadata)

Every published message carries ROS 2 metadata that consumers read through MessageInfo:

FieldSizeMeaning
sequence_number8 bytesint64 LE — monotonic per publisher
timestamp8 bytesint64 LE — source nanoseconds
gid16 bytesrandom per publisher, constant over its lifetime
  • Zenoh: the attachment rides alongside the payload as a zenoh Attachment. Humble uses a simple concatenation; Jazzy onward uses a VLE-encoded gid_length prefix.
  • DDS: sample identity and source timestamp fall out of the DDS sample info — your backend only has to forward them.

If you skip this, add_subscription_with_info() on consumers always reports MessageInfo::default(), and downstream features (safety-e2e checks, source-timestamp ordering) silently degrade.

3. Actions decompose into five underlying channels

ROS 2 actions are not a transport primitive — they are a pattern built on services and topics. Each action server exposes:

Sub-entityKindName suffix
send_goalService_action/send_goal
cancel_goalService_action/cancel_goal
get_resultService_action/get_result
feedbackTopic (pub)_action/feedback
statusTopic (pub)_action/status

A backend that implements the six core traits automatically supports actions — nros-node composes the five channels itself. The only backend-specific piece is the key / topic construction, which the Session::create_service_server and create_publisher methods already handle.

4. Type hashes

ROS 2 distroType hash
Humbleliteral string "TypeHashNotSupported"
Iron / Jazzy / RollingRIHS01_<sha256_hex> computed from the IDL

nano-ros currently targets Humble (the Iron type-hash roadmap doc tracks Iron support). A new backend that aims at a newer distro must compute the right hash string — the TopicInfo::type_hash field is already plumbed through.

5. QoS mapping

ROS 2 QoS (reliability, durability, history, depth) maps differently onto each backend:

  • Zenoh has reliable/best-effort and a KEEP_LAST(N) / KEEP_ALL buffering policy — direct mapping.
  • DDS has native QoS — almost 1:1.
  • Custom backends must either honour the requested QoS or document which fields are ignored. A best_effort publisher matched with a reliable subscriber is a QoS mismatch in ROS 2 — the transport must refuse the subscription (or flag it at runtime) rather than silently lose messages.

Further reading

  • RMW API Reference – full trait signatures, QoS profiles, error types, configuration structs.
  • RMW API Design – architectural motivation and comparison with the ROS 2 rmw interface.
  • RMW API: Differences from upstream rmw.h – side-by-side C-API comparison for porters coming from upstream ROS 2 RMW backends; covers the renames, the collapses, and the reason for each.
  • Zenoh-pico Symbol Reference – FFI symbol mapping for the zenoh-pico backend (useful as a reference for how an existing backend is structured).

Platform Porting Pitfalls

Hard-won lessons from porting nros to bare-metal ARM (QEMU MPS2-AN385), ESP32-C3, STM32F4, FreeRTOS, NuttX, ThreadX, and Zephyr. Read this before adding a new platform or debugging mysterious failures on an existing one.

Memory Layout

DMA buffer placement

On MCUs with multiple SRAM regions, DMA descriptors and buffers must be in DMA-accessible memory. The STM32F4 has 64 KB of CCM RAM that is NOT DMA-accessible — placing Ethernet descriptors there causes silent data corruption with no error interrupt.

Fix: Use linker script sections to force placement:

/* stm32f4.x — place DMA descriptors in SRAM1, not CCM */
.eth_descriptors (NOLOAD) : ALIGN(4) {
    *(.eth_descriptors .eth_descriptors.*)
} > RAM

In Rust, annotate the buffer:

#![allow(unused)]
fn main() {
#[link_section = ".eth_descriptors"]
static mut ETH_DMA_DESC: MaybeUninit<[u32; 256]> = MaybeUninit::uninit();
}

Stack sizing

Bare-metal and RTIC applications share a single stack. FreeRTOS/Zephyr give each task its own stack. In both cases, zenoh-pico’s internal processing needs significant stack space:

ContextMinimum stack
Bare-metal / RTIC main stack8 KB
FreeRTOS zenoh read task2 KB
FreeRTOS zenoh lease task1 KB
Zephyr zpico work queue thread2 KB

Stack overflow on embedded targets causes silent corruption — there is no guard page. FreeRTOS can detect overflow with configCHECK_FOR_STACK_OVERFLOW=2 and the high-water-mark API (uxTaskGetStackHighWaterMark), but only after the fact.

Heap sizing

zenoh-pico allocates heap memory during session open, publisher/subscriber creation, and message processing. Typical heap consumption:

PlatformMinimum heap
MPS2-AN385 (bare-metal)64 KB
ESP32-C364 KB (~16 KB static + dynamic)
STM32F464 KB
FreeRTOS (lwIP + zenoh)256 KB (configTOTAL_HEAP_SIZE)

Undersized heaps cause _Z_ERR_SYSTEM_OUT_OF_MEMORY (-78) during z_open() or entity creation, not during message I/O.

Rust struct move invalidation

When Rust returns a struct from a function, it moves the value to a new address. If C code stored a pointer to the original location (e.g., during init()), that pointer is now dangling.

Symptom: Callback crashes or garbage field values after struct is returned from an init function.

Fix: Use static storage, Box::pin(), or index-based references instead of raw pointers. See Troubleshooting: Subscriber Callback Crashes.

Clock Sources

Use hardware timers, not poll-driven counters

Every platform must provide a monotonic millisecond clock via smoltcp_clock_now_ms() (for smoltcp timestamping) and z_clock_now() / z_clock_elapsed_*() (for zenoh-pico timeouts). This clock must be backed by a hardware timer or OS clock — never by a software counter that only advances when spin() or poll() is called.

A poll-driven clock causes:

  • Incorrect timeouts: zenoh-pico uses z_clock_elapsed_ms() for session keep-alive, query timeout, and transport timeout. If the clock only advances during poll, idle periods (waiting for network I/O, sleeping between spins) are invisible to the timeout logic.
  • Timeout storms: After a long idle period, the next poll advances the clock by a large jump, causing multiple timeouts to fire simultaneously.
  • Broken RTIC integration: RTIC tasks yield with Mono::delay() between spins. A poll-driven clock doesn’t advance during the delay, making all timeouts effectively infinite.

Platform clock status

PlatformClock sourceNotes
MPS2-AN385 (zpico)CMSDK APB Timer0Hardware 25 MHz free-running counter. Wraps at ~171s, extended via WRAP_COUNT.
ESP32-C3esp_hal::time::InstantHardware timer, no manual updates needed.
ESP32-C3 (QEMU)esp_hal::time::InstantWorks with QEMU -icount 3.
STM32F4ARM DWT cycle counterHardware-backed. update_from_dwt() reads elapsed DWT cycles and advances the software counter. Must be called at least once per ~25.6s (at 168 MHz) to avoid missing DWT wraps.
FreeRTOSFreeRTOS kernel tickOS-managed via xTaskGetTickCount().
NuttXPOSIX clock_gettime()OS-managed.
ZephyrZephyr kernel tickOS-managed via k_uptime_get().
POSIX (native)std::time::InstantOS-managed.
XRCE MPS2-AN385CMSDK APB Timer0Same hardware timer as zpico MPS2-AN385. Call init_hardware_timer() before use.

Porting checklist for new platforms

  1. Identify a hardware timer — SysTick, a general-purpose timer, or DWT cycle counter. On RTOS platforms, use the OS tick API.
  2. Export smoltcp_clock_now_ms() -> u64 — called by zpico-smoltcp for TCP/IP timestamping.
  3. Export zenoh-pico clock symbolsz_clock_now(), z_clock_elapsed_us/ms/s(), z_clock_advance_us/ms/s(). These read from the same underlying counter.
  4. Never advance the clock inside smoltcp_network_poll() — the poll callback runs during network I/O and is not a reliable time source. Read the hardware timer directly.
  5. Handle timer wraps — 32-bit timers wrap. Track wrap count in an atomic or use a 64-bit hardware counter if available.

QEMU clock synchronization

QEMU’s virtual clock races ahead of wall-clock time during WFI, causing hardware timer-backed timeouts to fire before TAP network I/O completes. This is solved with -icount shift=auto, which makes virtual time advance at wall-clock speed during CPU sleep states.

See QEMU icount reference for the full explanation, parameter reference, and tradeoffs.

Networking

Ephemeral port conflicts

smoltcp’s ephemeral port allocator starts at port 49152 with a static counter that resets to zero on each boot. When QEMU instances are killed and restarted, the new instance picks the same source port, creating a 4-tuple collision with stale host-side TCP sockets.

Symptom: ConnectionFailed or TransportOpenFailed on the second test run. ss -tnap shows FIN-WAIT-1 sockets from the previous QEMU instance.

Fix (test infrastructure): Kill stale host TCP sockets between test runs:

#![allow(unused)]
fn main() {
// Clean stale TCP connections to QEMU IPs
for ip in &["192.0.3.10", "192.0.3.11"] {
    let _ = Command::new("ss")
        .args(["-K", "dst", ip])
        .status();
}
}

Fix (Zephyr native_sim): Pass different --seed values to each instance so the entropy source produces different ephemeral ports:

./build-listener/zephyr/zephyr.exe --seed=12345
./build-talker/zephyr/zephyr.exe --seed=67890

QEMU single-threaded I/O starvation

QEMU emulates the CPU and processes TAP network I/O in one thread. If the guest never executes WFI (Wait For Interrupt), QEMU never yields to its I/O event loop, and the TAP file descriptor is never serviced.

Symptom: First few packets work (buffered), then all networking stops. Services and actions time out. Pub/sub may appear to work because subscriber callbacks fire inline during zp_read().

Fix: In RTIC, yield between spin_once() calls:

#![allow(unused)]
fn main() {
// CORRECT — yields to idle task → WFI → QEMU processes TAP
cx.local.executor.spin_once(0);
Mono::delay(10.millis()).await;

// WRONG — busy-loops, starving QEMU I/O
loop {
    cx.local.executor.spin_once(0);
}
}

In bare-metal (no RTIC), call cortex_m::asm::wfi() in your idle loop.

TAP device assignment

Each QEMU peer must use a different TAP device. Two QEMU instances on the same TAP cause packet collision and loss.

QEMU talker  → tap-qemu0 ─┐
                           ├── qemu-br (192.0.3.1/24)
QEMU listener → tap-qemu1 ─┘

Subscriber startup ordering

Zenoh does not buffer messages for unknown subscribers. If the publisher starts before the subscriber’s declaration propagates through the router, early messages are lost.

Rule: Start subscriber first, wait 5 seconds for stabilization, then start publisher.

TAP queue discipline

The default fq_codel qdisc drops packets when QEMU emulation is slow (CoDel’s default 5ms target interprets emulation pauses as congestion, and per-flow scheduling disrupts zenoh-pico’s service reply timing). This causes TCP data segments to be dropped, breaking zenoh session establishment.

Fix: Use pfifo instead:

sudo tc qdisc replace dev tap-qemu0 root pfifo limit 1000
sudo tc qdisc replace dev tap-qemu1 root pfifo limit 1000

pfifo queues packets without dropping, which QEMU needs to absorb bursts during emulation pauses. Stale packets from killed QEMU processes accumulate in the queue but are harmless — the firmware seeds smoltcp’s ephemeral port from the host’s wall clock via ARM semihosting SYS_TIME, so each QEMU run uses a different source port and stale packets are silently ignored.

See TAP qdisc analysis for why fq_codel and noqueue don’t work for QEMU TAP devices.

zenoh-pico

Z_FEATURE_INTEREST causes multi-client failures

When multiple zenoh-pico clients connect to the same router, the interest protocol’s write filter creation can fail with -78 (_Z_ERR_SYSTEM_OUT_OF_MEMORY) even when memory is available.

Fix: Disable in all builds:

#![allow(unused)]
fn main() {
// build.rs
cmake::Config::new(&zenoh_pico_path)
    .define("Z_FEATURE_INTEREST", "0")
    .define("Z_FEATURE_MATCHING", "0")
    .build();
}

Z_FEATURE_MATCHING depends on Z_FEATURE_INTEREST — both must be disabled.

See Troubleshooting: zenoh-pico Multiple Client Issues.

Blocking TCP reads in cooperative schedulers

_z_read_tcp() can block for up to SOCKET_TIMEOUT_MS (default 10 seconds). In cooperative schedulers (RTIC, bare-metal main loop), this blocks the entire system since C FFI cannot yield.

Impact: Pub/sub works (callbacks fire inline during zp_read()), but services and actions fail because z_get() query replies need multiple zp_read() cycles, and the 5-second query timeout expires while _z_read_tcp blocks.

Current status: zpico_spin_once(0) is non-blocking (single_read=true), but the underlying z_get() timeout is still 5 seconds. RTIC service/action E2E tests are #[ignore]d pending a fully non-blocking TCP read path.

Version pinning

All zenoh components must be the same version. A version mismatch between zenoh-pico and zenohd causes transport-level failures (-100, _Z_ERR_TRANSPORT_TX_FAILED) that look like network issues.

nros pins zenohd and zenoh-pico to the same version. zenohd is built from the scripts/zenohd/zenoh/ submodule; zenoh-pico from zpico-sys/zenoh-pico/.

ESP32-C3 (RISC-V)

picolibc errno TLS crash

picolibc’s errno.h uses __thread (thread-local storage), which crashes on bare-metal RISC-V because TLS is not initialized. The crash is a hard fault with no useful backtrace.

Fix: Shadow errno.h with a non-TLS version:

// errno_override.h — no __thread, just a global
extern int errno;
#define EPERM  1
#define ENOENT 2
// ... subset of errno codes

Include this with -include errno_override.h in CFLAGS before picolibc headers.

Flash image format

ESP32-C3 QEMU requires a merged flash image, not a raw binary:

espflash save-image --merge --chip esp32c3 target/riscv32imc-unknown-none-elf/release/app app.bin

The --merge flag creates a 4 MB image with bootloader, partition table, and application combined.

FreeRTOS

Task priority inversion

The LAN9118 RX FIFO poll task and zenoh-pico read task must run at the same priority (typically priority 4). If the read task has higher priority, it monopolizes the CPU, preventing RX FIFO drain. TCP keep-alives are missed, and zenoh sessions expire.

Similarly, lwIP’s tcpip_thread must run at the same priority. Lower priority stalls packet processing.

Recursive mutexes required

zenoh-pico’s FFI layer requires recursive mutexes. Enable in FreeRTOS config:

#define configUSE_RECURSIVE_MUTEXES 1

Interrupt priority constraints

On Cortex-M3 (3-bit priority, 8 levels): ISRs at priority >= 5 (configLIBRARY_MAX_SYSCALL_INTERRUPT_PRIORITY) cannot call FreeRTOS APIs. Ethernet IRQ handlers that signal tasks must use a lower (numerically higher) priority.

Build System

Parallel test build races

When nextest runs test files in parallel and multiple tests build the same example with different features, cargo fingerprinting creates race conditions — one test overwrites the binary another test is about to execute.

Fix: Use --target-dir for each feature variant:

#![allow(unused)]
fn main() {
Command::new("cargo")
    .args(["build", "--release", "--features", "safety-e2e"])
    .arg("--target-dir").arg("target-safety")
    .status()?;
}

Add each target dir to the example’s .gitignore:

/target/
/target-safety/
/target-zero-copy/

CMake cache invalidation

Changes to CMake defines (like Z_FEATURE_INTEREST=0) are cached. Changing the value without cleaning the build has no effect.

Fix:

# Cargo (native)
cargo clean -p zpico-sys && cargo build

# Zephyr (west)
west build --pristine

Feature flag mutual exclusivity

The three platform axes — RMW backend, platform, ROS edition — are mutually exclusive within each axis. Enabling two platforms (e.g., platform-posix,platform-zephyr) causes compile errors with confusing messages about duplicate symbol definitions, not a clear “pick one” error.

Test Infrastructure

Orphan process prevention

When nextest is killed (Ctrl-C, OOM, timeout), child processes (zenohd, QEMU) can become orphans that hold ports and TAP devices.

Fix: Use PR_SET_PDEATHSIG(SIGKILL) in the pre-exec hook:

#![allow(unused)]
fn main() {
use std::os::unix::process::CommandExt;

let mut cmd = Command::new("zenohd");
unsafe {
    cmd.pre_exec(|| {
        // Kill this child when parent dies
        libc::prctl(libc::PR_SET_PDEATHSIG, libc::SIGKILL);
        // New process group for tree cleanup
        libc::setpgid(0, 0);
        Ok(())
    });
}
}

Important: Do not use pkill to clean up zenohd — other agents or users may have their own zenohd instances. Use process groups and death signals instead.

Sequential test execution for shared resources

Tests that use QEMU networking (TAP devices, zenohd on fixed ports) must run sequentially. Use nextest test groups:

# .config/nextest.toml
[test-groups.qemu-network]
max-threads = 1

Stale TCP connection cleanup

Between sequential QEMU tests, host-side TCP sockets may linger in FIN-WAIT-1 (the QEMU peer was killed before completing the TCP close handshake). These cause 4-tuple collisions when the next QEMU instance reuses the same source port.

Fix: Call ss -K dst <ip> between tests to force-destroy stale sockets:

#![allow(unused)]
fn main() {
pub fn cleanup_tap_network() {
    for ip in &["192.0.3.10", "192.0.3.11"] {
        let _ = Command::new("ss")
            .args(["-K", "dst", ip])
            .stdout(Stdio::null())
            .stderr(Stdio::null())
            .status();
    }
    std::thread::sleep(Duration::from_secs(1));
}
}

Design Overview

This section explains why nano-ros has the shape it does. It is not a tutorial (see Getting Started) and not an API reference (see the Reference chapters). It is the rationale a contributor needs to evaluate proposed changes, and the context a porter needs to make judgment calls when the Porting guides don’t cover their case.

Three design choices shape almost everything else:

1. The RMW layer was rewritten, not adopted

ROS 2’s rmw.h assumes a libc heap, an OS scheduler with preemptable threads, dynamic loaders, and middleware-owned background dispatch. None of those hold on a Cortex-M3 with 64 KB of RAM. nano-ros defines its own RMW abstraction (nros-rmw) that pushes I/O buffers to the caller, replaces wait sets with explicit drive_io(), drops dynamic graph discovery, and selects the backend at compile time.

RMW API Design

2. Future/Promise unifies blocking and async, with no internal spin

rclcpp owns the spin loop (rclcpp::spin(node) blocks the thread it was called on); rclc has no future/promise type at all. Neither model works on a single-threaded MCU. nano-ros uses a single Promise<T> value that supports three patterns – callback-driven polling, blocking-with-timeout, and .await – and never owns the spin loop. Every blocking helper takes the executor as an argument and drives it internally.

Client Library Model

3. Platform traits are independent and contract-first

zenoh-pico needs a heap, threading, networking, and a clock. XRCE-DDS only needs a clock and four C function pointers. Bare-metal has no scheduler. Each capability is a separate trait the platform may stub if unsupported. Every method has an explicit behavior contract: blocking? may-fail? unsupported-fallback?

Platform API Design

How to read this section

If you are evaluating a new feature: start with the design page closest to the change (RMW for transport-level changes, Client Library for executor or callback-shape changes, Platform API for OS-primitive changes). Each page lists the constraints the current shape satisfies; a new design must satisfy them or argue for relaxing one.

If you are porting nano-ros to new hardware or a new transport: the Porting chapters tell you what to implement. Read the matching design page to understand why each trait exists, which is what you need when the porting guide leaves a judgment call to you.

Client Library Model: nros vs rclc / rclcpp / rclrs

The RMW layer decouples the transport from the client library. The client library is the user-facing API on top of RMW: nodes, executors, publishers, subscribers, services, clients, callbacks, futures. ROS 2 ships three: rclc (C, MCU-focused), rclcpp (C++, desktop), and rclrs (Rust, experimental). nano-ros adds a fourth – the nros-node crate plus its nros-c / nros-cpp wrappers.

This page explains the shape of nros-node and how it differs from the three official client libraries. For trait signatures see Rust API / C API / C++ API.

Side-by-side

Concernrclcrclcpprclrsnros
TargetMCU (Micro-XRCE-DDS)Desktop / robotDesktop (alpha)MCU + RTOS + desktop
LanguageC99C++17Rust (std)Rust (no_std) + C + C++14
Node ownershipApp owns node + executorshared_ptr<Node> + ExecutorArc<Node> + executorExecutor owns the session; node borrows from it
Executor modelSingle-threaded, polledSingle-/multi-threaded, owned threadSingle-threaded, polledSingle-threaded, polled
Spin entry pointrclc_executor_spin_some(&exec, timeout)rclcpp::spin(node) (blocks, owns thread)executor.spin() (blocks current thread)executor.spin_once(timeout_ms)
Blocking primitiveNone (callback-only)std::shared_future::wait_for(timeout)Promise::wait(timeout) (Tokio-style)Promise::wait(&mut executor, timeout_ms)
Async primitiveNonestd::shared_future + spin_until_future_completeimpl FuturePromise<T> (impls Future + manual poll + executor-driven blocking wait)
Heap requirementOptional (static allocator)RequiredRequiredOptional (caller buffers; only zenoh-pico transport heap)
Threading requirementNoneRequired (std::thread)Required (std::thread)None (single-threaded valid)

Future / Promise as the unifying primitive

The same Promise<T> value serves three usage patterns. None of the other client libraries achieves this with a single type.

Pattern 1 – callback-driven (no blocking call):

let promise = client.call(&request)?;
// ... continue running other work ...
executor.spin_once(10);
if let Some(reply) = promise.try_recv()? {
    handle(reply);
}

Pattern 2 – blocking with timeout:

let mut promise = client.call(&request)?;
let reply = promise.wait(&mut executor, 5000)?;  // blocks up to 5s, drives I/O

Pattern 3 – async runtime (Tokio, Embassy, smol):

let reply = client.call(&request)?.await;  // Promise: Future<Output = T>

Contrast this with the alternatives:

  • rclcpp uses std::shared_future<Response>, which is bound to std::thread and std::condition_variable. There is no way to drive it on a cooperative single-threaded MCU. The blocking helper rclcpp::spin_until_future_complete() requires an owned thread.
  • rclc has no future or promise concept. The only way to receive a service reply is to register a callback that fires from rclc_executor_spin_some(). Building request/response chains becomes an explicit state machine.
  • rclrs has impl Future but no callback-poll path; you must .await from a Tokio task. No blocking-with-timeout helper that drives I/O internally.

In nano-ros, the same Promise carries a borrow of the standalone service client and a sequence number; try_recv() polls it, wait() polls it in a loop while spinning the executor, and the Future impl integrates with any executor that calls register_waker(&Waker) underneath.

No internal spin

rclcpp::spin(node) blocks the calling thread inside the library and owns the dispatch loop. This works on Linux but cannot work on bare-metal: there is no thread to give up, and the application loop must remain in user code (for power management, interrupt servicing, smoltcp polling, RTIC integration).

nano-ros never owns the spin loop. The user always writes the loop:

loop {
    executor.spin_once(10);
    // user can also: poll smoltcp, service interrupts, run other tasks, sleep
}

Convenience wrappers (spin(count), spin_blocking(opts), spin_period(duration)) exist for desktop-style use cases, but each is implemented as a spin_once() loop that the user could write by hand. On no_std targets they aren’t available – the user writes the loop.

This is the reason every blocking API in nros takes &mut Executor. Promise::wait, Stream::wait_next, Client::call_blocking, the C++ Future::wait(executor.handle(), ...), and the C nros_call_service(..., timeout_ms) all internally call spin_once() to keep I/O moving while waiting. They cannot rely on a background thread doing it for them, because there is none.

Executor is the session owner, not a singleton

In rclcpp, the Executor is a separate object that you attach nodes to (exec.add_node(node)). The node owns its own RMW context. Multiple executor classes (SingleThreadedExecutor, MultiThreadedExecutor, StaticSingleThreadedExecutor) exist as parallel implementations.

In nano-ros, the Executor is the RMW session owner. Calling Executor::open(&config) opens the transport; nodes are derived from the executor:

let mut executor = Executor::open(&config)?;
let mut node = executor.create_node("my_node")?;
let pub_ = node.create_publisher::<Int32>("/topic")?;

There is exactly one executor type. There is no separate “context” object, no add/remove-node lifecycle, no executor selection at runtime. The single-threaded model is baked in – adding a multi-threaded variant would require redesigning the callback dispatch, which we deliberately avoid to keep the bare-metal path viable.

The trade-off: you cannot share one transport session between two executors. In practice no embedded application wants this, and on desktop you can run multiple processes if you need it.

Explicit spin in blocking ops – why pass the executor?

Every blocking operation takes &mut Executor (Rust) or an executor handle (C/C++). This looks redundant when the executor is also the entity that created the operation. The reason is borrow-checker hygiene plus single-threaded I/O.

The standalone communication handles (StandaloneClient, StandaloneSubscription, the Promise returned from call) borrow from the session. They cannot also borrow the executor at the same time, because the executor owns the session. Passing &mut executor at the wait call – after the promise has been created and the borrow released – is the only way to get a mutable executor reference while a promise is in flight.

The deeper reason: there is no other thread that can drive I/O. If Promise::wait() did not have the executor, it could not call spin_once() – the network would freeze and the wait would always time out. By forcing the executor parameter, the API makes the I/O dependency explicit and impossible to forget.

Language parity

The same Future/Promise + explicit-executor model is preserved in C and C++ wrappers.

Rust uses Promise<'_, T> directly. It implements core::future::Future and exposes try_recv() + wait(&mut executor, ms).

C++ wraps the promise as nros::Future<T>:

auto fut = client.send_request(req);
ResponseType resp;
NROS_TRY(fut.wait(executor.handle(), 5000, resp));

Future::wait() takes void* executor_handle (from executor.handle() or nros::global_handle()) for the same reason the Rust API takes &mut executor. There is no global executor singleton; the handle is explicit.

C uses paired _async and blocking entry points:

// Blocking: drives the executor internally, returns reply or timeout.
nros_call_service(client, &req, sizeof(req), &reply, sizeof(reply), 5000);

// Async: returns immediately, result via callback registered ahead of time.
nros_action_send_goal_async(client, &goal, sizeof(goal));
nros_action_client_set_result_callback(client, on_result);
// User keeps calling nros_spin_once() to drive callbacks.

The C API has no Future/Promise type because C lacks generics, but the pattern is the same: an _async send-without-wait paired with a callback (or polled status), and a blocking call that drives the executor for you.

Summary

The four design choices that shape nano-ros’s client library:

  1. Single executor type that owns the session – no separate context object, no executor variants.
  2. No internal spin – the user always owns the spin loop; blocking helpers exist but are wrappers.
  3. Explicit &mut Executor on every blocking op – the API enforces the I/O dependency at the call site.
  4. Future/Promise as the unifying primitive – one type for callback-drive, blocking-with-timeout, and .await.

These choices are what make the same client library viable on a Cortex-M3 and a Linux workstation without a separate “MCU client library” like rclc.

RMW API Design: nros-rmw vs ros2/rmw

nano-ros defines its own RMW (ROS Middleware) abstraction in the nros-rmw crate. While it serves the same purpose as the official ros2/rmw interface – decoupling the client library from the transport backend – it is designed for no_std embedded systems and uses a fundamentally different approach.

This page documents the architectural differences and trade-offs. For trait signatures and the available backends, see RMW API Reference. For implementing a new backend, see Custom RMW Backend.

Why We Revised rmw.h

rmw.h was designed for ROS 2 on Linux: a process with a libc heap, an OS scheduler, dynamic loaders, and middleware-owned background threads. None of those assumptions hold on a Cortex-M3 with 64 KB of RAM. Each constraint below drove a specific change.

Heap availability

rmw.h heap-allocates everywhere – handles, serialized message buffers, wait sets, type support tables. Bare-metal targets often have no allocator; RTOS targets have allocators with hard total budgets (~16-256 KB) that must cover the application as well.

nros-rmw moves all I/O buffers to the caller. publish_raw(&[u8]) and try_recv_raw(&mut [u8]) operate on slices that the caller stack- or statically-allocates. Type metadata is a string-only TopicInfo struct, not a pointer-laden rosidl_message_type_support_t table. The only heap users in core paths are zenoh-pico’s internal transport buffers (~64 KB), exposed through PlatformAlloc and replaceable with a bump allocator on bare-metal.

Threading model

rmw.h assumes the middleware owns threads. rmw_wait() blocks the calling thread on a wait set; some implementations also spawn internal dispatch threads that fire callbacks asynchronously. Bare-metal has no scheduler; cooperative RTOS configurations can’t tolerate hidden threads.

nros-rmw replaces rmw_wait with Session::drive_io(timeout_ms) – a single call the executor invokes from its own (and only) thread. There is no wait set object, and no entity is implicitly polled by the middleware. The application drives all I/O explicitly. For async runtimes, subscribers and service clients expose register_waker(&Waker) so the transport’s C receive callback can wake a Rust future without a wait set abstraction.

Single-threaded callback dispatch

rmw.h permits multi-threaded executors and reentrant callbacks. Cooperative single-threaded targets cannot guarantee atomicity around RMW state without locks they don’t have.

nros-rmw assumes a single-threaded executor that owns the session for its lifetime. Callbacks run sequentially on the executor thread; no callback can preempt another. This eliminates the need for internal locking around publisher state, subscriber buffers, or service queues – a measurable code-size and runtime win on MCUs.

No dynamic discovery tables

rmw.h provides rmw_get_topic_names_and_types(), rmw_count_publishers(), rmw_get_node_names(), and similar graph-introspection APIs. These require maintaining a dynamic discovery cache, which costs heap and CPU continuously even when nothing reads it.

nros-rmw drops these APIs entirely. Discovery still happens at the transport layer (zenoh liveliness, XRCE-DDS session establishment), but it is not surfaced as queryable graph state. Applications that need topic introspection can issue a zenoh query directly.

Compile-time backend selection

rmw.h selects backends at runtime via dlopen() of librmw_*.so. This requires a dynamic loader (no embedded MCU has one) and forces every call through a vtable.

nros-rmw selects the backend at compile time via Cargo features. The Session trait uses associated types, so the compiler monomorphizes all transport calls – no vtables, no dynamic dispatch, no relocation overhead at startup. The trade-off is exactly one RMW backend per binary, enforced by compile_error!().

Architectural Pattern

AspectROS 2 rmwnros-rmw
LanguageC API (rmw/rmw.h)Rust traits
DispatchRuntime plugin loading (shared library via rmw_implementation)Compile-time monomorphization (Rust generics)
no_stdNo (requires libc, heap, POSIX)Yes (zero heap in core path)
Error modelrmw_ret_t integer codesTransportError enum + associated type Error per trait

ROS 2 selects the RMW backend at runtime by loading a shared library (e.g., rmw_fastrtps_cpp.so). This enables switching backends without recompilation but requires dynamic linking and heap allocation.

nros-rmw selects the backend at compile time via Cargo feature flags. The Session trait uses associated types, so the compiler monomorphizes all transport calls – no vtables, no dynamic dispatch, no heap. This is critical for MCUs with 16–256 KB of RAM.

Object Model

ROS 2

ROS 2 rmw has a deep initialization hierarchy:

rmw_init() → rmw_context_t
  → rmw_create_node() → rmw_node_t
    → rmw_create_publisher() → rmw_publisher_t*
    → rmw_create_subscription() → rmw_subscription_t*
    → rmw_create_service() → rmw_service_t*
    → rmw_create_client() → rmw_client_t*

Nodes are first-class RMW objects. Each rmw_node_t carries its own context, name, namespace, and security credentials. The RMW layer is responsible for node lifecycle and graph participation.

nros-rmw

nros-rmw is flatter – there is no node at the RMW level:

Rmw::open(&RmwConfig) → Session
  → session.create_publisher(&TopicInfo, QosSettings) → Self::PublisherHandle
  → session.create_subscriber(&TopicInfo, QosSettings) → Self::SubscriberHandle
  → session.create_service_server(&ServiceInfo) → Self::ServiceServerHandle
  → session.create_service_client(&ServiceInfo) → Self::ServiceClientHandle

Node lives one layer up in nros-node. It is purely a namespace and liveliness concern – it borrows the session from the executor and creates typed communication handles. The RMW layer only knows about sessions and communication endpoints.

Serialization Boundary

This is the most significant design difference.

ROS 2: The rmw layer operates on pre-serialized data. rcl and rosidl handle CDR serialization before calling rmw_publish() with an rmw_serialized_message_t. The rmw layer never sees typed messages – it only moves byte buffers. Type metadata is passed separately via rosidl_message_type_support_t structs.

nros-rmw: The traits include both raw and typed methods:

pub trait Publisher {
    // Raw: caller handles serialization
    fn publish_raw(&self, data: &[u8]) -> Result<(), Self::Error>;

    // Typed: serialize + publish in one call
    fn publish<M: RosMessage>(&self, msg: &M, buf: &mut [u8]) -> Result<(), Self::Error>;
}

pub trait Subscriber {
    fn try_recv_raw(&mut self, buf: &mut [u8]) -> Result<Option<usize>, Self::Error>;
    fn try_recv<M: RosMessage>(&mut self, buf: &mut [u8]) -> Result<Option<M>, Self::Error>;
}

The typed methods have default implementations that call the raw methods with CDR serialization/deserialization from nros-serdes. This keeps the RMW layer self-contained – no separate serialization layer is needed.

Type metadata uses simple structs (TopicInfo { name, type_name, type_hash }) instead of C type support function tables.

I/O and Readiness Model

ROS 2: Uses rmw_wait() with a wait set (rmw_wait_set_t) containing subscriptions, services, clients, guard conditions, and events. The caller constructs a wait set, adds handles, and blocks until any handle is ready. This is similar to select()/epoll().

nros-rmw: Uses a single drive_io(timeout_ms) method on the Session trait:

pub trait Session {
    fn drive_io(&mut self, timeout_ms: i32) -> Result<(), Self::Error> {
        let _ = timeout_ms;
        Ok(())
    }
}

This is a pull-based model: the executor calls drive_io() to poll the network and dispatch incoming data to internal subscriber buffers, then checks each entity with has_data(). There is no wait set – the executor iterates its dispatch table directly.

For async integration, subscribers and service clients expose register_waker(&Waker) instead of guard conditions. The transport backend calls waker.wake() from its C receive callback, bridging to Rust Future waking without the wait set abstraction.

Memory Model

ROS 2: Heap-allocates handles, messages, and serialization buffers. rmw_serialized_message_t wraps a dynamically-sized rcutils_uint8_array_t. Loaned message APIs (rmw_borrow_loaned_message, rmw_take_loaned_message) provide optional zero-copy for transports that support shared memory.

nros-rmw: Uses caller-provided &mut [u8] buffers everywhere. All receive and serialize operations write into stack-allocated or statically-allocated buffers:

// Caller provides the buffer
let mut buf = [0u8; 512];
let msg: Option<MyMsg> = subscriber.try_recv(&mut buf)?;

Zero-copy receive is supported via process_raw_in_place(), which invokes a closure with a reference to the subscriber’s internal receive buffer, avoiding the copy into a caller-provided buffer. This is gated behind the unstable-zenoh-api feature.

QoS Settings

ROS 2 rmw_qos_profile_t includes:

FieldROS 2nros-rmw
History (keep last/all)YesYes
DepthYesYes
Reliability (reliable/best-effort)YesYes
Durability (volatile/transient local)YesYes
DeadlineYesNo
LifespanYesNo
Liveliness (automatic/manual)YesNo
avoid_ros_namespace_conventionsYesNo

nros-rmw provides the four QoS policies that zenoh-pico and XRCE-DDS can actually enforce. The time-based policies (deadline, lifespan, liveliness) are omitted because the supported transports do not implement them.

Standard QoS profiles (QOS_PROFILE_DEFAULT, QOS_PROFILE_SENSOR_DATA, QOS_PROFILE_SERVICES_DEFAULT, etc.) match their ROS 2 equivalents for interoperability.

Service Client Model

ROS 2: Service clients are always asynchronous at the rmw level. rmw_send_request() sends a request and returns a sequence number. The reply is retrieved later via rmw_take_response(), typically driven by rmw_wait().

nros-rmw: Provides both models:

pub trait ServiceClientTrait {
    // Blocking: send request and wait for reply
    fn call_raw(&mut self, request: &[u8], reply_buf: &mut [u8]) -> Result<usize, Self::Error>;

    // Async: send request, poll for reply separately
    fn send_request_raw(&mut self, request: &[u8]) -> Result<(), Self::Error>;
    fn try_recv_reply_raw(&mut self, reply_buf: &mut [u8]) -> Result<Option<usize>, Self::Error>;
}

The blocking call_raw() is convenient for simple embedded applications. The async split (send_request_raw + try_recv_reply_raw) is used by the executor for non-blocking dispatch.

APIs Present in ROS 2 rmw but Absent in nros-rmw

ROS 2 rmw APIPurposeWhy absent
rmw_node_t / rmw_create_node()Node lifecycle at RMW levelNode is above the RMW layer in nros-node
rmw_wait_set_t / rmw_wait()Multiplexed readiness waitingReplaced by drive_io() + per-entity has_data()
rmw_guard_condition_tWake wait set from application codeReplaced by register_waker(&Waker)
rmw_event_tQoS event callbacks (deadline missed, etc.)QoS events not supported
rmw_get_topic_names_and_types()Graph introspectionDiscovery via zenoh liveliness, not exposed at trait level
rmw_get_node_names()Node discoverySame as above
rmw_count_publishers() / rmw_count_subscribers()Graph statisticsNot exposed
rosidl_message_type_support_tC type support tables for serializationReplaced by TopicInfo string metadata
rmw_serialize() / rmw_deserialize()Standalone serializationCDR handled by nros-serdes
rmw_borrow_loaned_message()Zero-copy shared memory publishNot supported (smoltcp/zenoh-pico don’t use shared memory)
Content-filtered topicsServer-side topic filteringNot supported

APIs Present in nros-rmw but Absent in ROS 2 rmw

nros-rmw APIPurpose
Publisher::publish<M>(msg, buf)Typed publish with built-in CDR serialization
Subscriber::try_recv<M>(buf)Typed receive with built-in CDR deserialization
Subscriber::process_raw_in_place(f)Zero-copy in-place processing via closure
Subscriber::try_recv_validated()E2E safety validation (CRC-32 + sequence tracking)
ServiceClientTrait::call_raw()Blocking request/reply (ROS 2 rmw is async-only)
ServiceServerTrait::handle_request<S>()Typed request handling with automatic CDR roundtrip
Session::drive_io(timeout_ms)Explicit network polling (ROS 2 rmw relies on middleware threads)

Summary

The core difference is that ROS 2 rmw is a C plugin interface designed for desktop systems with dynamic linking, heap allocation, and OS threading. nros-rmw is a Rust trait hierarchy designed for MCUs with static dispatch, stack allocation, and cooperative scheduling. The trade-off is flexibility (ROS 2 can swap backends at runtime) vs efficiency (nros eliminates all abstraction overhead at compile time).

Despite these differences, the two are wire-compatible when using the same transport. An nros node using nros-rmw-zenoh communicates with a ROS 2 node using rmw_zenoh_cpp through the same zenohd router, with matching QoS profiles and CDR encoding.

RMW API: Differences from upstream rmw.h

This page is for developers who already know upstream ROS 2’s rmw/rmw.h and want to write — or just understand — a backend for nano-ros. Side-by-side: what rmw.h looks like, what nros-rmw-cffi looks like, and why nano-ros diverges.

The C signatures shown for nano-ros come from <nros/rmw_vtable.h>. The Rust trait counterparts are an alternative entry point for porters who already work in Rust; this page sticks to the C-vtable surface throughout.

TL;DR

Concernupstream rmw.hnros-rmw-cffi
Plugin loadingdlopen("librmw_*.so") at runtimeSingle vtable registered at init
Init sequencermw_init_options_trmw_context_t → entitiesOne open() call, returns the session
Entity typesrmw_publisher_t / rmw_subscription_t / rmw_service_t / rmw_client_tTyped-with-opaque-tail: nros_rmw_publisher_t / _subscriber_t / _service_server_t / _service_client_t (visible metadata + opaque backend_data)
Waitrmw_wait(waitset, timeout) blocks the callerdrive_io(session, timeout_ms) drives I/O once
Serializationtypesupport-driven (rosidl)Pre-serialized CDR bytes only
Graph queriesrmw_get_topic_names_and_types, …None
QoS profilesFull DDS profile match between endpointsSame field set; per-backend support advertised; synchronous IncompatibleQos on create instead of runtime mismatch event
DDS eventsrmw_event_t (rmw_take_event)None
Loaned messagesOptional rmw_borrow_loaned_messageFirst-class loan_publish / loan_recv
Error returnsrmw_ret_t (RMW_RET_OK, …)nros_rmw_ret_t (NROS_RMW_RET_OK, …) — same named-constant style

1. Plugin loading vs. compile-time backend

Upstream. rmw_implementation resolves the backend at process start by opening a shared library:

RCUTILS_DLLIMPORT
const char * rmw_get_implementation_identifier(void);

The runtime calls dlopen against the path encoded by RMW_IMPLEMENTATION and binds every rmw_* symbol through the loader.

nano-ros. The backend is linked into the binary. A C backend registers its vtable once before any nros call:

#include <nros/rmw_vtable.h>

static const nros_rmw_vtable_t MY_VTABLE = { ... };

int main(void) {
    nros_rmw_cffi_register(&MY_VTABLE);
    nros_init(...);
}

Why. Most embedded targets have no dynamic loader. Even where dlopen exists, a 32 KB Flash budget can’t afford to link every backend’s C client and pick at runtime. Compile-time selection cuts the binary by 60–80 % and lets the linker drop unused entity paths.

2. Init sequence

Upstream. Three-step init:

rmw_init_options_t options = rmw_get_zero_initialized_init_options();
rmw_init_options_init(&options, allocator);
rmw_context_t context = rmw_get_zero_initialized_context();
rmw_init(&options, &context);
/* ... use context to create rmw_node_t, then rmw_publisher_t, … */
rmw_shutdown(&context);

nano-ros. One step — a session covers what upstream splits across options + context:

nros_rmw_session_t session = {0};
nros_rmw_ret_t ret = vtable->open(locator, mode, domain_id, node_name, &session);
/* ... use &session to create_publisher / create_subscriber / … */
vtable->close(&session);

Why. Upstream’s split is useful when an application owns multiple RMW instances with different options. Nano-ros assumes one session per process — one transport, one wire-protocol, one set of QoS defaults — so the options/context separation buys nothing.

3. Entity handles

Upstream. Every entity is a typed C struct with backend-private state hidden behind void * data:

typedef struct RMW_PUBLIC_TYPE rmw_publisher_t {
  const char * implementation_identifier;
  void * data;
  const char * topic_name;
  rmw_publisher_options_t options;
  bool can_loan_messages;
} rmw_publisher_t;

nano-ros. Hybrid: typed-with-opaque-tail. Each entity is a typed C struct exposing the metadata the runtime actually reads (topic name, type name, QoS, lending capabilities) inline; backend-private state stays behind an opaque backend_data pointer.

typedef struct nros_rmw_publisher_t {
    const char *   topic_name;          /* borrowed; outlives the publisher */
    const char *   type_name;           /* borrowed */
    nros_rmw_qos_t qos;
    bool           can_loan_messages;   /* matches upstream's field of the same name */
    uint8_t        _reserved[7];        /* forward-compat; must be zero */
    void *         backend_data;        /* opaque */
} nros_rmw_publisher_t;

nros_rmw_ret_t (*create_publisher)(
    nros_rmw_session_t * session,
    const char * topic_name, const char * type_name, const char * type_hash,
    uint32_t domain_id, const nros_rmw_qos_t * qos,
    nros_rmw_publisher_t * out);   /* runtime-allocated; backend fills */

Same shape for nros_rmw_subscriber_t. Service entities (nros_rmw_service_server_t, nros_rmw_service_client_t) and nros_rmw_session_t have no qos and no can_loan_messages — service-level QoS doesn’t generalise across non-DDS backends (see QoS, Section 7) and service request/reply uses the byte-buffer API rather than the loan primitive.

Forward-compat reserved bytes. Each entity carries an explicit _reserved[N] byte block (sized to fill the natural alignment slot before backend_data). New fields up to N bytes can be added later without changing struct size or the offset of any field after backend_data. Backends and runtime keep these bytes zero.

Storage ownership. The runtime allocates the entity-struct shell; the backend writes its backend_data (plus can_loan_messages for publisher / subscriber entities) into the runtime-supplied out-parameter at create_* time. The backend never mallocs a struct shell — embedded targets cannot afford a per-entity heap allocation. destroy_* releases only the backend’s backend_data; the shell stays valid until the runtime drops its owner.

Differences from upstream’s rmw_publisher_t.

  • Borrowed strings, not backend-owned copies. Upstream’s topic_name points to a backend-allocated string copied at create_publisher time. Ours points to caller (runtime) storage that outlives the publisher — no allocation per entity.
  • No implementation_identifier field. Backend selection is compile-time (see Section 1); there’s no plugin loader to dispatch through, so no need to identify which backend owns a struct.
  • can_loan_messages matches upstream. Same bool, same name, same semantics — true if the backend exposes the loan primitive (the CDR-byte zero-copy loan path). The runtime reads it once at create time and dispatches the publish path with no per-call branch.
  • depth: uint16_t. Upstream uses 32-bit; embedded queue depths are 1–100, the 16-bit width saves 2 bytes × N entities.
  • Explicit _reserved[N] bytes. Upstream uses an embedded rmw_publisher_options_t struct as the extension point; we reserve raw bytes inline. Same forward-compat property — new fields up to N bytes don’t break ABI — without the indirection.

Why this shape. Fully-opaque void * (the previous nano-ros design) forced every introspection through a vtable callback. Upstream’s “expose every field, backend keeps them in sync” forces duplicated state. The typed-with-opaque-tail middle ground exposes exactly the fields the runtime reads — no callback indirection — and keeps backend implementation state private. The struct layout is ABI; adding or reordering fields is a major-version bump.

4. drive_io vs. rmw_wait

Upstream. Clients block on a waitset that aggregates entities:

rmw_ret_t rmw_wait(
  rmw_subscriptions_t * subscriptions,
  rmw_guard_conditions_t * guard_conditions,
  rmw_services_t * services,
  rmw_clients_t * clients,
  rmw_events_t * events,
  rmw_wait_set_t * wait_set,
  const rmw_time_t * wait_timeout);

The middleware can also spawn its own background threads that fire callbacks asynchronously.

nano-ros. The executor calls a single drive-I/O entry point:

nros_rmw_ret_t (*drive_io)(nros_rmw_session_t * session, int32_t timeout_ms);

The backend dispatches whatever receive / send / wakeup work is pending and returns within timeout_ms. There is no waitset, no guard-condition aggregation, no implicit middleware thread.

How the two models differ in practice

The two designs distribute work across different layers:

PhaseUpstream rmw_waitnano-ros drive_io
Build the wait setExecutor rebuilds a waitset every spin_once, adds every entityExecutor registers entities once at construction; no per-spin rebuild
Blockrmw_wait blocks the thread on a kernel waitable (DDS WaitSet, condvar, kqueue) until any entity is ready or timeoutdrive_io blocks (sleep-model backends) or polls (poll-model backends) for up to timeout_ms
Signal readinessWait primitive raises per-entity ready flags; backend writes flags into the waitset’s status arraysBackend’s RX worker pulls bytes; the data is “ready” by virtue of being received
Dispatch user callbackExecutor’s spin_once picks one ready entity, calls rmw_take, fires its user callbackBackend’s RX worker / drive_io loop fires user callbacks while it has work to do
Per spin_onceExactly one user callback runs (single-threaded executor)All ready callbacks run before drive_io returns

The upstream model separates wait (kernel-blockable) from dispatch (executor-controlled). nano-ros today fuses them — the backend handles both inside one call.

What this buys, what this costs

Fusing wins on:

  • No per-spin waitset rebuild. Upstream’s add_handles_to_wait_set allocates and walks every entity each iteration. On a 100-entity executor at 1 kHz that’s 100 000 per-second add/remove operations plus heap churn. nano-ros registers entities once; the backend tracks them statically.
  • No kernel-waitable per entity. Upstream’s per-entity ready flags need a per-entity wakeable resource (DDS Condition, eventfd, pipe). nano-ros’s backend uses one wait primitive per session; per-entity tracking is in user-space backend state.
  • Backend RX worker fires callbacks directly. zenoh-pico’s _z_session_read_task invokes user callbacks during its read. nano-ros’s drive_io drains it; upstream’s rmw_zenoh would still have to round-trip through rmw_take.

Fusing costs on scheduling control. Upstream’s “one callback per spin_once” rule gives the executor an opportunity between every two callbacks to:

  • Re-check timer expirations
  • Re-check guard conditions
  • Yield to higher-priority work (multi-threaded executor)
  • Apply per-callback priority ordering

nano-ros’s drive_io runs all ready callbacks back-to-back, then the spin loop processes timers + GCs between drive_io calls. For a 100 ms drive_io call that fires 10 sub callbacks in 80 ms, a timer that should have fired 5 ms in is delayed 75 ms.

Where this fits each RTOS execution model

Execution modelFits drive_io today?
Cooperative single-task (one task does ROS, no priority competition)Yes — no other task to preempt; entity scheduling fairness is moot
Async / tokio / Embassy (futures, wakers)Yes — spin_async drives futures; drive_io not used in the hot path
Preemptive priority RTOS, ROS at one priority (FreeRTOS / ThreadX / Zephyr typical)Partial — kernel preemption from higher-priority tasks works; ROS-internal entity scheduling is batch-FIFO, timer expiries can be delayed by long sub callbacks at the same priority
WCET-bounded real-time (RTIC, DO-178C)Nodrive_io has unbounded execution time; callers needing per-callback WCET use the async path with explicit Waker integration instead
Time-triggered cyclicNo — no way to bound drive_io to a fixed wall-clock budget

The “Yes” rows are where nano-ros ships today. The “Partial” / “No” rows are addressed by the work below.

How RTOS cooperation will improve

Three forward-looking knobs land incrementally as the drive_io interface is extended. None breaks the default behaviour; each is opt-in for apps that need it.

  1. Backend-internal-deadline visibility. Session::next_deadline_ms() tells the executor when the backend’s next internal event (lease keepalive, heartbeat, ACK retransmit) is due. The executor caps drive_io’s timeout against it so the call doesn’t return sooner than expected on otherwise-quiet links. Saves one round-trip per quiet period.

  2. Per-call user-callback cap. drive_io accepts max_callbacks: an upper bound on user callbacks fired per call. Setting it to 1 reproduces upstream’s “one callback per spin_once” pattern. The runtime spin loop calls drive_io again to drain pending work, with timer / GC checks between iterations. Closes the priority-inversion footgun for preemptive priority RTOS.

  3. Wall-clock budget per call. drive_io accepts time_budget_ms: a wall-clock cap that bounds total time spent firing callbacks. Time-triggered cyclic apps configure a fixed slot per cycle; drive_io yields when the slot expires even if max_callbacks isn’t reached.

Once the cap (knob 2) ships, an additional refinement moves timer and guard-condition dispatch into the backend’s drive_io loop so the cap applies uniformly across all callback sources, not just backend-RX-driven ones. This unifies the dispatch path and makes max_callbacks = 1 mean “exactly one callback per spin_once, regardless of whether it’s a sub, service, timer, or guard condition.”

For the per-RTOS-model recommendations (which knobs to set, which defaults to use), see the RTOS Cooperation concepts page.

Why drive_io and not rmw_wait. Cooperative single-task runtimes (bare-metal, single-threaded RTOS) have one execution context. A waitset abstraction with kernel-blockable per-entity resources doesn’t fit — there is no kernel to provide them. drive_io makes the cooperative model explicit and lets backends pick the most efficient wait primitive available on their target (kernel block on multi-threaded; cooperative yield + WFI on bare-metal). The scheduling-control trade-off the upstream waitset gives up is recovered through the optional knobs above for apps that need it.

5. Serialization: CDR bytes, not typesupport

Upstream. Publish/take APIs accept typed messages plus typesupport pointers, and each backend implements serialization-from-typesupport itself:

rmw_ret_t rmw_publish(
  const rmw_publisher_t * publisher,
  const void * ros_message,
  rmw_publisher_allocation_t * allocation);

The backend dereferences ros_message according to a rosidl_message_type_support_t table to compute the wire bytes.

nano-ros. The runtime serializes upstream of the RMW. Backends receive already-CDR-encoded bytes:

nros_rmw_ret_t (*publish_raw)(nros_rmw_publisher_t * publisher,
                              const uint8_t * data, size_t len);

int32_t (*try_recv_raw)(nros_rmw_subscriber_t * subscriber,
                        uint8_t * buf, size_t len);

Why. rosidl typesupport is heavy: dynamic dispatch through a function-pointer table per field type, dependence on dynamic allocators for nested sequences, megabytes of generated symbol tables linked even for one message. Splitting the layers means:

  • Codegen (nros_generate_interfaces() from CMake) writes a fixed-shape C struct per message and a single <MsgType>_serialize_cdr(...) function.
  • The runtime calls it before handing bytes to the RMW.
  • The RMW backend stays transport-only — no rosidl, no typesupport, no allocator coupling.

6. No graph cache

Upstream. Graph introspection sits in the RMW:

rmw_ret_t rmw_get_topic_names_and_types(...);
rmw_ret_t rmw_count_publishers(...);
rmw_ret_t rmw_get_node_names(...);

Implementations maintain a discovery cache that costs heap and CPU continuously even when nothing reads it.

nano-ros. None of the above. Backends do whatever discovery their transport mandates (zenoh liveliness, XRCE-DDS session establishment), but no get_topic_names / count_publishers / node_names exists in the vtable.

Why. Graph introspection is fundamentally a host-side debugging need. An MCU has no terminal, no ros2 topic list. Wire-protocol interop with rmw_zenoh_cpp means standard ROS 2 tools running on a laptop can introspect the same domain as the MCU — at zero cost on the MCU.

7. QoS: full DDS-shaped profile, per-backend support advertised

Upstream. Full DDS QoS profile family with profile matching between endpoints:

typedef struct RMW_PUBLIC_TYPE rmw_qos_profile_t {
  rmw_qos_history_policy_t history;
  size_t depth;
  rmw_qos_reliability_policy_t reliability;
  rmw_qos_durability_policy_t durability;
  rmw_time_t deadline;
  rmw_time_t lifespan;
  rmw_qos_liveliness_policy_t liveliness;
  rmw_time_t liveliness_lease_duration;
  bool avoid_ros_namespace_conventions;
} rmw_qos_profile_t;

The backend negotiates compatibility (rmw_qos_profile_check_compatible) and surfaces mismatches as runtime events.

nano-ros. Same field set, packed into 24 bytes:

typedef struct nros_rmw_qos_t {
    uint8_t  reliability;
    uint8_t  durability;
    uint8_t  history;
    uint8_t  liveliness_kind;
    uint16_t depth;
    uint16_t _reserved0;
    uint32_t deadline_ms;             /* 0 = infinite */
    uint32_t lifespan_ms;             /* 0 = infinite */
    uint32_t liveliness_lease_ms;     /* 0 = infinite */
    bool     avoid_ros_namespace_conventions;
    uint8_t  _reserved1[3];
} nros_rmw_qos_t;

Standard profile constants (NROS_RMW_QOS_PROFILE_DEFAULT, _SENSOR_DATA, _SERVICES_DEFAULT, _SYSTEM_DEFAULT, _PARAMETERS) match upstream rmw_qos_profile_* field-for-field, so applications porting from rclcpp / rclrs can pull the equivalent profile constant unchanged.

Per-backend support, no silent downgrade

Each backend advertises which policies it can honour via Session::supported_qos_policies(), returning a QosPolicyMask bitfield. Policies a backend can’t enforce are explicit.

#![allow(unused)]
fn main() {
pub trait Session {
    fn supported_qos_policies(&self) -> QosPolicyMask {
        QosPolicyMask::CORE     // reliability + durability VOLATILE + history + depth
    }
}
}

The runtime validates the requested QoS against the backend’s mask at entity-create time. Requesting a policy the backend doesn’t support returns TransportError::IncompatibleQos (NROS_RMW_RET_INCOMPATIBLE_QOS at the C boundary) synchronously. There is no silent degradation — applications either get the requested QoS or a hard error.

Apps that need cross-backend portability check the mask at startup:

#![allow(unused)]
fn main() {
if session.supported_qos_policies()
    .contains(QosPolicyMask::DEADLINE)
{
    pub.create_with_qos(...deadline_ms = 100, ...);
} else {
    // app-side fallback: timeout monitoring in user code
}
}

Manual liveliness assertion

For LIVELINESS_MANUAL_BY_TOPIC and LIVELINESS_MANUAL_BY_NODE, publishers call assert_liveliness() explicitly:

#![allow(unused)]
fn main() {
pub.assert_liveliness()?;   // refresh this publisher's lease
}

C side: nros_publisher_assert_liveliness(&pub). C++ side: pub.assert_liveliness(). No-op for AUTOMATIC and NONE kinds.

Differences from upstream’s matching

Upstream surfaces QoS mismatches via runtime events (RMW_EVENT_REQUESTED_INCOMPATIBLE_QOS). nano-ros surfaces them synchronously at create time as IncompatibleQos. The mismatch is a configuration error visible at startup; the runtime path doesn’t need to handle it.

Two related choices:

  • No profile matching between publisher and subscriber. Each endpoint requests the QoS it wants from its backend; the backend enforces locally. Cross-endpoint compatibility is the wire protocol’s concern — DDS endpoints negotiate via DDS Discovery, zenoh endpoints communicate intent via the topic-key encoding, uORB endpoints share an in-process queue. nano-ros’s executor doesn’t run a profile-matching pass.
  • Wire metadata per backend. Lifespan needs per-sample timestamps; liveliness needs a keepalive mechanism. Each backend uses its native attachment / sample-info mechanism (Zenoh attachments, DDS RTPS sample-info, XRCE session pings) — no cross-backend metadata header.

Per-backend QoS coverage

The mask actually advertised by each backend’s Session::supported_qos_policies():

BackendReliability + Durability + History/DepthDeadlineLifespanLiveliness AutomaticLiveliness ManualLiveliness Leaseavoid_ros_namespace_conventions
Cyclone DDS✅ Native✅ Native✅ Native✅ Native✅ Native✅ Native✅ honoured
XRCE-DDS✅ Native (binary uxrQoS_t)✅ Shim-side clock check (sub: RequestedDeadlineMissed; pub: OfferedDeadlineMissed) + agent-side via FastDDS XML profile✅ Agent-side via FastDDS XML profile✅ Native✅ Configured via XML✅ Configured via XML✅ honoured
zenoh-pico✅ Shim-emulated✅ Clock-based check (sub + pub)✅ Subscriber-side filter using attachment timestamp✅ Trivial via session keepalive✅ Shim-side keepalive timer at pub publish path✅ Honouredn/a (no /rt/ prefix)
uORB✅ CORE only (intra-process, no wire)❌ No rate concept❌ No expiry concept❌ No wire-level livelinessn/a

Key takeaways:

  • The default QoS profile (RELIABLE + VOLATILE + KEEP_LAST(10) + AUTOMATIC) works on every backend.
  • Apps that need extended QoS but want to stay backend-portable: check supported_qos_policies() at startup and degrade gracefully.
  • For full DDS QoS: Cyclone DDS is the native DDS path; XRCE-DDS auto-routes through FastDDS XML profile when extended policies are set. Zenoh-pico fills the gap with shim-side emulation. uORB is intra-process only.

See Status events for how the deadline / liveliness / message-lost policies translate into runtime events, and User guide → Configuration for code examples.

8. Status events: callback-on-entity, Tier-1 subset

Upstream. Event APIs surface DDS-shaped notifications via a waitset-take pattern:

rmw_ret_t rmw_subscription_event_init(
  rmw_event_t * event, const rmw_subscription_t * sub,
  rmw_event_type_t type);

/* Add event_handle to a waitset alongside subscriptions. */
rmw_ret_t rmw_wait(...);

/* Poll fired events. */
rmw_ret_t rmw_take_event(
  const rmw_event_t * event_handle,
  void * event_info, bool * taken);

Eleven event types covering liveliness, deadline, QoS-incompatibility, match, message-lost, type-incompatibility — all dispatched through the waitset.

nano-ros. Callback-on-entity for a Tier-1 subset (liveliness changes, deadline misses, message lost). Skips the waitset. Skips Tier-2 (MATCHED) and Tier-3 (QOS_INCOMPATIBLE, INCOMPATIBLE_TYPE) — see “What’s skipped” below.

#![allow(unused)]
fn main() {
sub.on_liveliness_changed(|status| {
    if status.alive_count == 0 { trigger_failover(); }
})?;

sub.on_requested_deadline_missed(
    Duration::from_millis(15),
    |status| metric_inc(&LATE_SAMPLE_COUNT, status.total_count_change),
)?;

sub.on_message_lost(|status| log::warn!("dropped {}", status.total_count_change))?;
}

C side mirrors with nros_subscription_set_*_callback functions.

What lands

EventProducerSubscriber callback / Publisher callback
Liveliness changedsubon_liveliness_changed(LivelinessChangedStatus)
Liveliness lostpubon_liveliness_lost(DeadlineMissedStatus)
Requested deadline missedsubon_requested_deadline_missed(deadline, DeadlineMissedStatus)
Offered deadline missedpubon_offered_deadline_missed(deadline, DeadlineMissedStatus)
Message lostsubon_message_lost(MessageLostStatus)

What’s skipped

  • MATCHED — embedded apps usually have static topology; rarely load-bearing. Add the kind if a discovery-tracking app shows up; additive.
  • QOS_INCOMPATIBLE / INCOMPATIBLE_TYPE — these surface at create time, not as runtime events. The existing nros_rmw_ret_t codes (NROS_RMW_RET_INCOMPATIBLE_QOS) carry the diagnostic synchronously from create_publisher / create_subscriber. No event needed.

Dispatch — callback-on-entity, not waitset-take

Events fire from inside the existing drive_io callback-dispatch path. The backend’s RX worker detects an event in the same place it detects messages; runs the registered callback; loops. No separate waitset, no per-call take.

This reuses the message-callback dispatch model; events count against the max_callbacks cap from Section 4 the same way message callbacks do.

Why callback-on-entity instead of waitset-take. The waitset-take pattern requires a waitset abstraction nano-ros deliberately doesn’t have (see Section 4). Replacing it with per-entity callbacks reuses existing machinery, matches the message-callback ergonomics users already know, and keeps the bounded-storage property — each registered event-callback is a fixed-size struct embedded in the entity, no per-call allocation.

The trade-off: users can’t bulk-poll all events at once. For the Tier-1 events this isn’t load-bearing — events are rare, callbacks are cheap.

Backend coverage

Coverage is uneven and surfaces through Subscriber::supports_event (Rust) / register_*_event returning NROS_RMW_RET_UNSUPPORTED (C). Apps must handle “not supported” — not every backend will generate every event:

BackendLivelinessDeadlineMessage lost
Cyclone DDS🟡 Not wired through nano-ros events yet🟡 Not wired yet🟡 Not wired yet
XRCE-DDS❌ XRCE protocol carries no session→client liveliness callback🟡 Sub: shim-side clock check on try_recv_raw; pub: shim-side check on publish_raw. LivelinessChanged / LivelinessLost not feasible.topic_callback carries no per-sample sequence
zenoh-pico✅ Sub: per-publisher count via zpico_liveliness_get_count + wildcard keyexpr query; pub: shim-side keepalive timer fires LivelinessLost from publish_raw when MANUAL_BY_* lease expires.✅ Clock-based check at sub + pub, rate-limited to ≤ 1 fire per deadline period✅ Sequence-gap detection from RMW attachment
uORB❌ No wire-level liveliness❌ No rate concept✅ Native: RustSubscriptionCallback publish-counter delta on host mock + real PX4

assert_liveliness() (manual): not all backends expose manual liveliness through nano-ros yet. Unsupported backends’ default is Ok(()) (no-op) since they don’t honour MANUAL_BY_TOPIC / MANUAL_BY_NODE liveliness kinds.

9. Loaned messages first-class

Upstream. Optional, often unimplemented:

rmw_ret_t rmw_borrow_loaned_message(
  const rmw_publisher_t * publisher,
  const rosidl_message_type_support_t * type_support,
  void ** ros_message);

rmw_ret_t rmw_return_loaned_message_from_publisher(
  const rmw_publisher_t * publisher,
  void * loaned_message);

A backend that doesn’t implement these returns RMW_RET_UNSUPPORTED and the client falls back to a copying path.

nano-ros. Lending is a separate vtable surface. When the backend supports zero-copy publish, it implements loan_publish_* / commit_publish_*; when it supports zero-copy receive, it implements loan_recv_* / release_recv_*. The runtime checks the function pointers for non-NULL once at session open and takes the lending path for the lifetime of the session.

Why. Zero-copy isn’t optional on embedded — every avoidable copy is a copy of a CDR-encoded sensor frame from a 64 KB heap. Promoting the lending surface to first-class makes it visible at compile time (missing pointer = compile error against the lending build) and lets the runtime pre-arrange arena slots once instead of probing on every publish.

10. Error returns

Upstream. Single error type for everything:

typedef int32_t rmw_ret_t;
#define RMW_RET_OK              0
#define RMW_RET_ERROR           1
#define RMW_RET_TIMEOUT         2
#define RMW_RET_UNSUPPORTED     3
/* ... */

Pointer-returning calls indicate failure with NULL and set a thread-local error string via rmw_set_error_string.

nano-ros. Same named-constant style (<nros/rmw_ret.h>), different sign convention, no thread-local error string:

typedef int32_t nros_rmw_ret_t;
#define NROS_RMW_RET_OK                       0
#define NROS_RMW_RET_ERROR                   -1
#define NROS_RMW_RET_TIMEOUT                 -2
#define NROS_RMW_RET_BAD_ALLOC               -3
#define NROS_RMW_RET_INVALID_ARGUMENT        -4
#define NROS_RMW_RET_UNSUPPORTED             -5
#define NROS_RMW_RET_INCOMPATIBLE_QOS        -6
#define NROS_RMW_RET_TOPIC_NAME_INVALID      -7
#define NROS_RMW_RET_NODE_NAME_NON_EXISTENT  -8
#define NROS_RMW_RET_LOAN_NOT_SUPPORTED      -9
#define NROS_RMW_RET_NO_DATA                -10
#define NROS_RMW_RET_WOULD_BLOCK            -11
#define NROS_RMW_RET_BUFFER_TOO_SMALL       -12
#define NROS_RMW_RET_MESSAGE_TOO_LARGE      -13

Two return-shape conventions, picked by call shape:

ReturnsSuccessFailure
nros_rmw_ret_t + entity-struct out-param (open, create_publisher, create_subscriber, …)NROS_RMW_RET_OK, out->backend_data non-NULLnegative named constant
nros_rmw_ret_t (close, drive_io, publish_raw, send_reply, …)NROS_RMW_RET_OKnegative named constant
int32_t byte count (try_recv_raw, try_recv_request, call_raw)>= 0 (bytes received)negative nros_rmw_ret_t

Differences from upstream.

  • Negative for error. Upstream uses positive integer codes (RMW_RET_ERROR = 1); nano-ros uses negative so the byte-count convention can be unified into the same int32_t return.
  • No thread-local error string. No rmw_set_error_string, no rmw_get_error_string. The thread-local heap allocation that pattern needs is unaffordable on embedded targets. Backends log verbose diagnostics at the failure site through the platform’s printk equivalent — never buffered, never thread-local.
  • Smaller code-set. 13 codes total (vs upstream’s ~25). Phase set started from upstream’s and dropped codes that don’t apply (e.g., DDS event codes, RMW_RET_NODE_INVALID). Adding a code is a <nros/rmw_ret.h> header change only.

Why same style. Named constants make switch statements possible at call sites; bare negative ints don’t.

See also

Platform API Design

The platform API (nros-platform) sits below the RMW backend. It exposes the OS or hardware primitives that zenoh-pico and XRCE-DDS need: a clock, optionally a heap, optionally threading, optionally networking. This page explains why the trait surface is grouped the way it is.

Canonical interface spec. The function-pointer signatures, parameter docs, ownership rules, blocking allowance, and failure modes for every platform entry live in the platform-cffi Doxygen reference. This doc is rationale, not spec — when the two disagree, the Doxygen wins. The Rust trait crate nros-platform-api mirrors the C signatures one-for-one.

For per-platform behaviour comparison, see Platform Differences. For writing a new port, see Custom Platform.

Trait groups and rationale

The traits cluster into seven concern groups. Each group is independent: a platform can provide some and stub others, and the RMW backend declares which it actually needs.

Time – PlatformClock

Every backend needs a monotonic clock. zenoh-pico uses milliseconds for socket timeouts and lease management; it uses microseconds for finer-grained protocol math (KeepAlive intervals, lease expiry calculations). We expose both clock_ms and clock_us rather than a single clock_ns because:

  • 32-bit MCUs without a hardware microsecond tick cannot serve clock_ns accurately. Synthesizing nanoseconds from a 1 ms tick is a lie that hides clock resolution from the caller.
  • u64 nanoseconds wraps after ~584 years, but on a Cortex-M0 every multiply-divide on a 64-bit value costs cycles that the lease task does not have to spare.
  • The two functions can share a single hardware source: clock_us returns the raw counter, clock_ms divides by 1000. Platforms with a slow tick (1 ms) implement clock_us as clock_ms * 1000 and accept the resolution loss.

The clock is monotonic and wraparound-free for system lifetime. There is no failure mode – if the platform cannot produce time, it cannot run nano-ros at all.

Memory – PlatformAlloc

Only zenoh-pico needs PlatformAlloc. XRCE-DDS does not allocate. The trait is a thin malloc/realloc/free shim because zenoh-pico’s internal buffer types expect that contract.

Bare-metal platforms back this with a bump allocator (linked-list-allocator or embedded-alloc). RTOS platforms back it with pvPortMalloc (FreeRTOS), tx_byte_allocate (ThreadX), k_malloc (Zephyr), or malloc (NuttX, POSIX). zenoh-pico’s working set is ~64 KB total; the allocator must have at least that much budget, ideally more.

Threading – PlatformThreading

Three sub-areas: tasks (spawn/join/exit), mutexes (regular + recursive), and condition variables. Single-threaded targets (bare-metal) provide stub implementations: task_init returns -1 (so zenoh-pico’s lease task spawn fails gracefully and the application drives lease-keepalive itself), and mutex_lock/condvar_wait are no-ops that always succeed.

The condvar API is the load-bearing one: zenoh-pico’s blocking z_get and the C++ Future::wait both block on a condvar that the receive callback signals. On single-threaded platforms there is no thread to block, so the blocking C++ wait paths are not used (the C++ action client status note covers the migration to non-blocking polling).

Sleep / Random / Wall-time

Three small zenoh-pico-only traits grouped for convenience:

  • PlatformSleep – delay APIs. On bare-metal with smoltcp, the implementation must call network_poll() while busy-waiting, otherwise packets queue and the lease times out.
  • PlatformRandom – a 32-bit xorshift PRNG seeded with hardware entropy (or a user-supplied seed). Used for session IDs and protocol nonces, not cryptography.
  • PlatformTime – wall-clock time for log timestamps. On bare-metal with no RTC, return monotonic time as a fallback.

Networking – PlatformTcp / PlatformUdp / PlatformSocketHelpers

zenoh-pico’s network layer is split into three traits because the original C interface (unix/network.c) has three concerns:

  • TCP and UDP each have their own open/read/send/close because the backend opens different socket types for each.
  • PlatformSocketHelpers carries the cross-cutting operations – set_non_blocking, accept, generic close, and wait_event – that apply to either socket family.

Sockets and endpoints are opaque *mut c_void pointers; their underlying types vary per platform (POSIX int, lwIP struct netconn*, Zephyr socket descriptor, smoltcp SocketHandle). The shim layer auto-detects the type sizes from C headers at build time so the FFI boundary stays type-erased.

PlatformUdpMulticast is split out as a fourth networking trait because embedded targets that connect to a fixed tcp/host:port locator never multicast and should not pay the code-size cost of multicast plumbing.

NetworkPoll – PlatformNetworkPoll

Bare-metal only. network_poll() advances the smoltcp state machine, processing pending RX/TX. Platforms with kernel-level networking (Linux, lwIP-on-FreeRTOS, NetX-on-ThreadX, Zephyr sockets) drive their own NIC and don’t need this.

PlatformSleep and the wait_event helper both call network_poll() while waiting, so packets keep flowing during otherwise-idle time. Without this hook, smoltcp would only receive when the application explicitly asked for it – a recipe for dropped TCP segments.

Libc – PlatformLibc

zenoh-pico uses strlen, memcpy, errno, etc. directly. Bare-metal targets that link picolibc or newlib-nano get these for free. Targets without a C runtime (some no_std builds) provide the trait, which forwards to Rust implementations of the same functions.

This trait exists because the alternative – patching zenoh-pico to call platform shims for every libc function – would require modifying upstream sources we don’t control.

Why clock_ms and clock_us, not clock_ns

Summarized from above:

APIProsConsVerdict
clock_ns onlySingle function, finest resolution64-bit math on every call; lies on 1 ms-tick MCUsRejected
clock_ms onlyCheap, fits zenoh-pico’s lease mathInsufficient resolution for sub-millisecond protocol timingInsufficient
Both clock_ms and clock_usEach call is cheap and honest about its resolutionTwo functions to implementChosen

Behavior contracts

Each trait below has a contract table. Columns:

  • Method – name (matches the trait definition).
  • Blocking? – whether the method may suspend the caller.
  • May fail? – whether the method has a meaningful failure mode.
  • Unsupported fallback – what to do when the platform cannot provide the capability.
  • Notes – extra constraints (monotonicity, reentrancy, side effects).

PlatformClock

MethodBlocking?May fail?Unsupported fallbackNotes
clock_msNoNoRequired for any backendMonotonic; wraparound-free for system lifetime
clock_usNoNoRequired for any backendSame monotonic base as clock_ms

PlatformAlloc

MethodBlocking?May fail?Unsupported fallbackNotes
allocNoYes (null)Required for zenoh-picoCaller checks for null and propagates as RMW error
reallocNoYes (null)Required for zenoh-picoExisting block must be preserved on failure
deallocNoNoRequired for zenoh-picodealloc(null) is a no-op

PlatformSleep

MethodBlocking?May fail?Unsupported fallbackNotes
sleep_usYesNoRequired for zenoh-picoBare-metal must call network_poll() during busy-wait
sleep_msYesNoRequired for zenoh-picoSame
sleep_sYesNoRequired for zenoh-picoSame; typically implemented as sleep_ms(s * 1000)

PlatformRandom

MethodBlocking?May fail?Unsupported fallbackNotes
random_u8 / _u16 / _u32 / _u64NoNoRequired for zenoh-picoxorshift32 is sufficient; not cryptographic
random_fillNoNoRequired for zenoh-picoFills len bytes; no upper bound check

PlatformTime (wall-clock)

MethodBlocking?May fail?Unsupported fallbackNotes
time_now_msNoNoRequired for zenoh-picoReturn clock_ms if no RTC
time_since_epochNoNoRequired for zenoh-picoReturn (monotonic_s, 0) if no RTC

PlatformThreading – tasks

MethodBlocking?May fail?Unsupported fallbackNotes
task_initNoYes (-1)Return -1 on single-threadedzenoh-pico’s lease task must degrade gracefully
task_joinYesYesReturn 0 (success)Single-threaded never spawned a task to join
task_detachNoYesReturn 0Same
task_cancelNoYesReturn 0Same
task_exitNoNoNo-opCaller is the only thread
task_freeNoNoNo-opNo allocation to free

PlatformThreading – mutexes (regular and recursive)

MethodBlocking?May fail?Unsupported fallbackNotes
mutex_init / mutex_rec_initNoYesReturn 0Single-threaded has no mutex state
mutex_drop / mutex_rec_dropNoYesReturn 0Same
mutex_lock / mutex_rec_lockYesYesReturn 0 (success)Single-threaded: no contention possible
mutex_try_lock / mutex_rec_try_lockNoYesReturn 0Always “succeeds” on single-threaded
mutex_unlock / mutex_rec_unlockNoYesReturn 0Same

PlatformThreading – condition variables

MethodBlocking?May fail?Unsupported fallbackNotes
condvar_initNoYesReturn 0No state on single-threaded
condvar_dropNoYesReturn 0Same
condvar_signalNoYesReturn 0No waiter to wake
condvar_signal_allNoYesReturn 0Same
condvar_waitYesYesReturn 0Single-threaded must use polling instead – avoid this path
condvar_wait_untilYesYes (timeout)Return 0 immediatelySame; blocking C++ Future::wait deadlocks on single-threaded (use non-blocking polling instead)

PlatformTcp

MethodBlocking?May fail?Unsupported fallbackNotes
create_endpointYes (DNS)YesRequired for zenoh-picoBacked by getaddrinfo or platform equivalent
free_endpointNoNoRequiredMirrors freeaddrinfo
openYesYesRequiredConnect with timeout in ms
listenNoYesOptional (server mode)Bare-metal client typically returns -1
closeNoNoRequiredShutdown + close
readNo (after set_non_blocking)Yes (usize::MAX)RequiredReturns 0 if no data; must be non-blocking for zenoh-pico’s poll loop
read_exactYesYesRequiredUsed for length-prefixed framing
sendYesYesRequiredMay block on socket buffer full

PlatformUdp

MethodBlocking?May fail?Unsupported fallbackNotes
create_endpointYes (DNS)YesRequired for zenoh-picoSame as TCP but SOCK_DGRAM
free_endpointNoNoRequired
openNoYesRequiredUDP socket open is non-blocking
closeNoNoRequired
readNoYesRequiredrecvfrom; returns 0 if no datagram
read_exactYesYesRequiredRarely used (UDP is message-oriented)
sendYesYesRequiredsendto

PlatformSocketHelpers

MethodBlocking?May fail?Unsupported fallbackNotes
set_non_blockingNoYesRequiredCritical: enables non-blocking read
acceptNo (after set_non_blocking)YesOptionalServer-mode only
closeNoNoRequiredGeneric socket close
wait_eventYesYesRequiredMulti-threaded: yields to scheduler. Single-threaded: spins + network_poll

PlatformUdpMulticast (optional)

MethodBlocking?May fail?Unsupported fallbackNotes
mcast_openYesYesStub returns -1Skipped on embedded targets without scouting
mcast_listenYesYesStub returns -1Same
mcast_closeNoNoNo-op
mcast_read / mcast_read_exactVariesYesStub returns usize::MAX
mcast_sendYesYesStub returns usize::MAX

PlatformNetworkPoll (bare-metal only)

MethodBlocking?May fail?Unsupported fallbackNotes
network_pollNoNoOS-driven platforms: no-opAdvances smoltcp once; called by sleep_* and wait_event

PlatformLibc (bare-metal without libc)

MethodBlocking?May fail?Unsupported fallbackNotes
strlen, strcmp, strncmp, strchr, strncpyNoNoLinker provides if libc presentSame semantics as the C standard
memcpy, memmove, memset, memcmp, memchrNoNoSameSame
strtoulNoYes (errno)SameUsed by zenoh-pico to parse locator strings
errno_ptrNoNoSameReturns pointer to thread-local (or static) errno

Cross-cutting rules

Two contract rules apply to every trait method:

  1. Reentrancy. Methods may be called from any context the executor enters: a publisher callback, a service handler, or directly from user code. Implementations must not assume a particular calling thread or critical-section state. Single-threaded platforms get this for free; RTOS platforms must use reentrant primitives.

  2. No panics across the FFI boundary. All trait methods are exposed to C through the shim crates. Panicking through C is undefined behavior. Implementations return error codes (or usize::MAX for byte counts) instead of panicking. The exception is PlatformClock – if the clock cannot be read, the system is fundamentally broken and there is nothing useful to return.

Canonical Platform C ABI

nano-ros separates the platform layer (clock, sleep, allocator, threading, critical section, network, timer, …) from the rest of the code via a single canonical C ABI of free extern "C" symbols. Every supported port — POSIX, FreeRTOS, NuttX, ThreadX, Zephyr, ESP-IDF, bare-metal Cortex-M — implements the same symbols against its host kernel. RMW backends, codegen output, and the nros-node runtime all link against the ABI; nobody links against a specific port’s Rust crate.

This page is the contract: what the surface looks like, how to add a port, and why the shape is what it is. It is the implementation companion to Platform Model (user- facing axis description) and docs/design/0006-portable-rmw-platform-interface.md (L0/L1/L2 design rationale across RMW + platform).

Surface

Three hand-written headers under packages/core/nros-platform-cffi/include/nros/:

HeaderPurposeSymbol count
platform.hCore kernel surface: clock, sleep, alloc + heap stats, threading, scheduler, time, yield, random, critical section, opaque wake primitive.59
platform_net.hNetwork surface: TCP/UDP/multicast socket helpers, endpoint resolution, IVC.29
platform_timer.hPeriodic timer surface (nros_platform_timer_*).8

Every symbol has the prefix nros_platform_. A drift gate (scripts/check-platform-abi-mirror.sh) walks each header and asserts that the unsafe extern "C" {} mirror block in packages/core/nros-platform-cffi/src/lib.rs declares the same set — no symbol can land in C without its Rust mirror, no Rust mirror can declare a symbol without a C header decl. just check runs the gate on every CI build.

Why free symbols (not a vtable struct)

The RMW layer uses a NrosRmwVtable fn-ptr struct because a single binary registers multiple backends at runtime (bridge nodes). Platforms are different: one platform per binary, resolved at link time. A vtable would add an indirection on every clock read, every mutex lock, every socket send — overhead with no upside, because there is nobody to swap. Free symbols let the linker resolve calls direct, let LTO inline across the boundary, and let static analysis treat them like any other extern.

The full rationale is in docs/design/0006-portable-rmw-platform-interface.md under “Platform ABI: free symbols (no vtable)”.

How a port is built

A port can be written in either of two ways. The choice is per-port and not visible to consumers — both shapes resolve to the same symbols.

Path A — Rust trait + macro export

The Rust crate implements PlatformClock, PlatformAlloc, PlatformThreading, … on a marker type, then invokes one of the export macros:

#![allow(unused)]
fn main() {
use nros_platform_api::{
    PlatformClock, PlatformSleep, PlatformAlloc, PlatformThreading,
    PlatformCriticalSection,
    nros_platform_export, nros_platform_export_net, nros_platform_export_timer,
};

pub struct PosixPlatform;
impl PlatformClock for PosixPlatform { /* … */ }
impl PlatformSleep for PosixPlatform { /* … */ }
// … etc.

nros_platform_export!(PosixPlatform);
nros_platform_export_net!(PosixPlatform);
nros_platform_export_timer!(PosixPlatform);
}

The macros expand to the full set of #[unsafe(no_mangle)] pub extern "C" fn nros_platform_* bodies forwarding to the trait calls. This is what nros-platform-posix does today. It is also what the bare-metal board crates (nros-platform-mps2-an385, nros-platform-stm32f4, nros-platform-esp32, nros-platform-esp32-qemu) use: there is no host kernel to write idiomatic C against, so the single-task stub impls are written in Rust and exported via the macro.

Path B — pure C port

The kernel-side ports (FreeRTOS, NuttX, ThreadX, Zephyr, ESP-IDF) write the bodies directly in src/platform.c, src/net.c, src/timer.c inside each packages/core/nros-platform-<rtos>/ directory. The Rust side has no implementation file — the build script (or parent build system: NuttX make, Zephyr west module, ESP-IDF cmake) compiles the C sources against the kernel’s headers.

Pure-C bodies are the more natural fit when the kernel already speaks C and ships C headers (xSemaphoreCreate*, k_sem_*, tx_thread_*): the impl is one-to-one with the kernel call, no FFI dance through Rust’s calling convention. The shared crate nros-platform-critical-section is the canonical example for a single-symbol shim: the Rust side just calls the externs and registers the result with critical_section::set_impl!.

Adding a new port — checklist

  1. Decide A vs B. Greenfield host build with a Rust crate available? Path A. Vendor RTOS with a C SDK and no Rust toolchain on the build machine? Path B.
  2. Mirror, don’t extend. Implement every symbol in the three headers. The drift gate fails the build otherwise. If the kernel genuinely cannot provide a primitive, return the documented sentinel value (-1 for task_init on single-task RTOS, 0 for mutex_* on single-core no-preempt hardware) rather than skipping the symbol.
  3. Write the smoke test. Land a tests/<port>-c-smoke/ mini-app that links the new C port against the canonical headers and calls a representative symbol from each capability group. Wire it into just <port> test-c-port. The smoke tests are the runtime parity layer that the drift gate cannot enforce.
  4. For platforms that emit critical_section::Impl (i.e. anything that consumes a critical-section-using crate), make sure the binary pulls nros-platform-critical-section once — it does the critical_section::set_impl!(PlatformCs) registration against the canonical externs. Binaries that don’t need the global registration don’t pay for it.
  5. Update the drift gate’s smoke list (the HEADERS_REQUIRE_MACRO array in scripts/check-platform-abi-mirror.sh) if you add a new header. Adding a symbol to an existing header needs no script change — the grep is generic.

Capability groups

Within platform.h, the symbols are grouped by trait. Each group is exported by one Rust trait + one C-port section + one drift-gate match. Adding a capability means adding a row to every column.

CapabilityRust traitC sectionSymbols
ClockPlatformClockclock_msnros_platform_clock_ms
SleepPlatformSleepsleep_msnros_platform_sleep_ms
AllocPlatformAllocmalloc/realloc/freenros_platform_alloc{,_realloc,_free}
ThreadingPlatformThreadingmutex/condvar/tasknros_platform_{mutex,condvar,task}_*
Critical sectionPlatformCriticalSectionper-CPU interrupt masknros_platform_critical_section_{acquire,release}
SchedulerPlatformSchedulertask hintsnros_platform_scheduler_*
TimePlatformTimewall-clocknros_platform_time_ns
YieldPlatformYieldcooperative yieldnros_platform_yield
RandomPlatformRandombest-effort RNGnros_platform_random_*
WakePlatformThreading (wake methods)opaque binary-semaphorenros_platform_wake_{init,drop,wait_ms,signal,signal_from_isr,storage_size,storage_align}

platform_net.h covers TCP/UDP/multicast/IVC, and platform_timer.h covers periodic timers. The shapes follow the same pattern: trait → macro → header → drift gate.

Status and architecture

The platform tier today is:

  • One canonical header per capability family (platform.h, platform_net.h, platform_timer.h). Every port mirrors them exactly; a drift gate fails the build on divergence.
  • Pure-C ports under nros-platform-{posix,freertos,nuttx,threadx,zephyr,esp-idf} — the per-RTOS Rust platform crates were retired in favour of one C body per RTOS.
  • critical_section promoted to a canonical platform capability owned by every port’s C body; nros-platform-critical-section is the global-registration shim.
  • An opaque-storage wake primitive (nros_platform_wake_*) — a binary semaphore that lets the executor block on RMW activity without burning a thread.

The current canonical surface is 59 + 29 + 8 = 96 symbols across three headers, mirrored exactly by the Rust extern block, exported by six ports plus four bare-metal board crates, and gated by one drift script. Adding a capability touches all four columns in one PR.

The scalar / opaque-struct boundary (RFC-0034 D2)

Not every kernel service unifies the same way. The ABI splits into two classes, and the split is a design boundary, not tech debt:

  • Scalar services — fully unified. Alloc, sleep, clock, time, yield, random. Their ABI is plain scalars (sizes, millisecond counts, byte buffers) with no host-type in the signature, so a single canonical nros_platform_* symbol works for every RTOS and every consumer (core, RMW, vendored zenoh-pico/XRCE) funnels through it. The D8 gate (check-no-direct-kernel-alloc.sh) enforces this for allocation; the same pattern extends to the other scalars (the vendored z_*/uxr_* funnel is the CI-relink-gated remainder of phase-230).

  • Opaque-struct services — stay per-RTOS-vendored. Threading (mutex/condvar/task), the wake primitive, and network sockets carry RTOS-defined structs (SemaphoreHandle_t, TX_MUTEX, struct k_mutex, socket/NX_* control blocks) whose layout the vendor owns. A single C ABI cannot name those types portably, so each port vendors its own body and nano-ros exposes only opaque storage (*_storage_size/*_storage_align + an init/drop pair). This is why the ThreadX board’s tx_byte_allocate thread-stack / NetX-pool sites are not D8 violations — they are the task and net opaque services, allowed on a documented symbol-scoped lint allowlist.

Escape hatch (if an opaque service must move later). The way out is NOT to widen the C ABI with host types; it is to pin a canonical fixed-layout struct in platform.h, give each port a compile-time size_probe / _Static_assert that the canonical layout is ≥ the vendor struct (so opaque storage stays sound), and let the port memcpy/place the vendor object inside the canonical slot. Net is the first candidateplatform_net.h already exposes 29 symbols over an opaque socket handle — but it is deferred (RFC-0034 “out of scope”) until there is a consumer that needs portable socket structs. Threads and sync are the least likely to move (deepest struct coupling).

RMW Backends — Host-Language Policy

This page documents which language each RMW backend is implemented in, and the rule that decides it. The matrix was originally frozen 2026-05-07; the L tier (+ L.8, landed 2026-05-12) collapsed the public RMW surface so every backend now reaches the runtime via the same nros_rmw_vtable_t bridge. The underlying implementations still ship in whichever language their upstream library prefers, but consumers never see that — they only see the C vtable.

The rule (post-115.L)

Every backend installs itself via the nros_rmw_vtable_t bridge. The underlying library’s host language is an implementation detail of the per-backend -cffi shim; it does not appear on the consumer surface.

The original rule (a backend’s host language matches its underlying library’s native language unless overridden) is still how we pick the inside of each -cffi shim — but the shim itself is uniform: a small Rust or C++ TU that fills in a vtable and calls nros_rmw_cffi_register(&vtable) once at startup.

Hierarchy

nros-core (Rust) ──→ Rmw trait (internal; bridged by RustBackendAdapter<R>)
                        └──→ nros-rmw-cffi   (C ABI bridge, registry)
                                ↓ nros_rmw_vtable_t  (~17 fn ptrs)
                                ├──→ nros-rmw-zenoh    (wraps Rust nros-rmw-zenoh)
                                ├──→ nros-rmw-xrce-cffi     (links C nros-rmw-xrce)
                                ├──→ nros-rmw-cyclonedds    (C++ direct, no Rust)
                                └──→ nros-rmw-uorb      (C++ direct, no Rust)

The shims are the canonical consumer surface. Public Cargo features on nros / nros-c / nros-cpp (rmw-{zenoh,xrce}-cffi, cffi-{zenoh-cffi,xrce-c}) all route through the same nros_rmw_vtable_t runtime. The pre-L.7 direct-Rust-trait features (rmw-zenoh, rmw-xrce, rmw-uorb) are gone.

Phase 169 (2026-05-19) — dust-dds retired. The nros-rmw-dds Rust shim and the dust-dds upstream Rust DDS implementation have been removed (Phase 169.4). Cyclone DDS is the sole DDS backend; the nros-rmw-cyclonedds shim registers under its canonical name "cyclonedds" only. The previous "dds" generic slot is not aliased — callers always select Cyclone by its specific name (NROS_RMW=cyclonedds, target_link_libraries(... NanoRos::Rmw::cyclonedds), etc.).

Any language with stable C-ABI interop (C, C++, Zig, Rust, Go-via-cgo, Python-via-ctypes…) can implement a backend by filling in the vtable and calling nros_rmw_cffi_register(&vtable) once at startup.

Decision matrix (post-115.L, updated by Phase 171)

BackendUnderlying libUnderlying langShim crateVerdict
Cyclone DDSCyclone DDSC / C++nros-rmw-cyclonedds (C++ direct vtable; canonical DDS backend)keep
XRCEmicro-XRCE-DDS-ClientCnros-rmw-xrce-cffi (Rust shim over the C nros-rmw-xrce static lib; 115.K.2 ported)keep
zenoh-picozenoh-picoCnros-rmw-zenoh (Rust → vtable via RustBackendAdapter<ZenohRmw>)keep
uORBPX4 module SDKC++nros-rmw-uorb (C++ direct vtable; 115.K.4 port replaces legacy nros-rmw-uorb Rust crate)keep

Dust-DDS was retired in Phase 169 (2026-05-19) after repeated bring-up failures on embedded targets. It is intentionally absent from the active backend table; Cyclone DDS now fills the DDS slot.

Rust-backend cffi shape

For backends whose upstream library is Rust (zenoh-pico) the cffi shim ships as a tiny crate that calls RustBackendAdapter::<UnderlyingRmw>::register(). The adapter monomorphizes a static nros_rmw_vtable_t over the Rust Rmw trait impl and installs it into the C registry. Consumer code never sees the trait surface; it only sees the vtable.

The legacy direct-Rust-trait crates (nros-rmw-zenoh, nros-rmw-xrce) stay in the workspace as internal-only implementation libs of these shims. They have no public Cargo feature reaching them after.

C-/C++-backend cffi shape

For backends whose upstream library is C/C++ (Cyclone DDS, uORB, XRCE) the cffi shim is a standalone CMake project that builds a static C/C++ library and registers a nros_rmw_vtable_t at startup via nros_rmw_cffi_register. No RustBackendAdapter is involved. The Rust runtime sees these via the same registry; the NANO_ROS_RMW=<name> CMake selector flips a build-time macro that ensures the register call is wired into nros::init.

Cyclone DDS: runtime type introspection (Phase 212.K.7)

The Cyclone DDS shim does not require per-type backend code on the generated msg crate. Phase 212.K.7 inverted the original design where each msg crate carried an optional cyclonedds Cargo feature plus a Cyclone-specific descriptor sidecar.

In the runtime-introspection design every generated msg crate is purely the wire-format data type (#[derive]d struct + a tiny impl nros_serdes::Message exposing const TYPE_NAME + const FIELDS). On the first typed create_publisher<M> / create_subscription<M> for a given M, the nros-rmw-cyclonedds shim walks the static field schema, builds a Cyclone ddsi_sertype via Cyclone’s dynamic-type C API, and caches the pointer in a bounded heapless::FnvIndexMap<u64, NonNull<ddsi_sertype>, MAX_TYPES> guarded by a platform-selected mutex. Subsequent uses of the same M are hash-map hits.

End state: no <msg-pkg>/cyclonedds Cargo feature anywhere, no per-msg-pkg backend code, no codegen branching on the active RMW. The shape matches upstream rclcpp’s introspection typesupport + rclrs’s plain <pkg> = "*" consumer manifest.

Sizing knob: NROS_CYCLONEDDS_MAX_TYPES (default 32), wired through the existing nros-sizes build probe (same pattern as EXECUTOR_OPAQUE_U64S). See section 212.K.7 of docs/roadmap/phase-212-ux-cargo-native-and-file-consolidation.md for the work-item ledger.

When to revisit

This matrix is a snapshot. Update it when any backend’s situation changes:

  • A new backend lands → add a row + shim-crate + verdict.
  • An existing backend’s underlying library changes language (e.g. zenoh-pico ships a Rust port upstream) → swap the shim shape (Rust adapter vs. C/C++ direct) but the vtable bridge stays.

The rule stays. Only per-backend verdicts and shim shapes move.

Registry + naming

nros-rmw-cffi holds a fixed-size named registry of backend vtables. Each backend registers under a canonical name at process startup:

BackendNameRegistered by
zenoh-pico"zenoh"nros_rmw_zenoh_register() (auto-ctor on POSIX)
micro-XRCE-DDS-Client"xrce"nros_rmw_xrce_register() (C ctor on POSIX)
Cyclone DDS"cyclonedds"nros_rmw_cyclonedds_register() (Phase 169.5 — canonical DDS backend; no generic "dds" alias)
uORB"uorb" (future)TBD

Naming policy

  • Lowercase ASCII identifying the protocol / wire format. Not the transport variant — "xrce" covers both XRCE-UDP and XRCE-serial; the transport is selected via the locator (udp/... vs serial:/dev/...).
  • Stable across releases. Renaming a registered name is a breaking change for bridge code that selects backends by string.
  • No "default" for new backends. The string "default" is reserved for the legacy single-arg nros_rmw_cffi_register shim — single-backend builds where the backend’s specific name doesn’t matter.

Capacity

Registry size: NROS_RMW_MAX_BACKENDS build-time env var consumed by nros-rmw-cffi/build.rs. Default 8. Range [1, 64]. Set lower on Cortex-M0+ (where each slot’s ~40 B costs); set higher for bridge nodes with 4+ backends. Hitting the cap = subsequent nros_rmw_cffi_register_named returns NROS_RMW_RET_ERROR.

Default-backend convention

Executor::open and any create_node call without an explicit .rmw(name) selector use the first-registered backend — the nano-ros equivalent of ROS 2’s RMW_IMPLEMENTATION. Single- backend binaries with one auto-registering backend Just Work without user code mentioning the backend’s name.

The user-facing knob is a declared, language-agnostic, per-deploy value (system.toml [system].rmw / [deploy.<t>].rmw, or a CLI/build flag) that the toolchain lowers to each language’s native link mechanism. The Cargo feature / shim dep and the CMake cache var below are those lowering targets — what the build uses, not how a user picks a backend (see RFC-0031):

  • Cargo (Rust): the declared RMW lowers to the nros rmw-<x> feature plus the matching nros-rmw-<x> shim dep in the consumer’s [dependencies]. Linking the shim crate is what registers the backend.
  • CMake: cmake -DNANO_ROS_RMW=zenoh ... (C/C++ users). The nano_ros_link_rmw(... RMW zenoh) helper auto-generates the per-target nros_app_register_backends() strong stub.

No runtime env-var override; selection is fixed at link time (RTOS-friendly, matches our static-link world).

Symbol-survival mechanism

Backend register symbols must survive linker dead-strip. Four mechanisms, layered:

  1. linkme distributed-slice (/ 128.H.2) — each backend contributes an RMW_INIT_ENTRIES entry through the nros_rmw_register_backend! macro. nros_support_init / Executor::open walks the slice and calls each entry. Canonical on Linux / *BSD / Windows / POSIX. Macro expands to a no-op on RTOS targets where linkme can’t recognise the section (NuttX, Zephyr, ESP-IDF, FreeRTOS bare-metal).
  2. Rust ctor (legacy fallback): #[unsafe(link_section = ".init_array")] #[used] static AUTO_REGISTER_CTOR. #[used] keeps rustc from dead-stripping.
  3. C ctor (legacy fallback): __attribute__((constructor)) static void nros_rmw_<name>_register_ctor. Same survival via .init_array walk by libc startup.
  4. CMake strong stub (landed): the nano_ros_link_rmw(<target> RMW <name>) helper at cmake/NanoRosLink.cmake:62-117 emits an auto-generated TU per target that defines a strong nros_app_register_backends() calling every linked RMW’s nros_rmw_<name>_register(). The weak default in libnros_c_weak_stubs.a is overridden. This is the canonical path on every RTOS where linkme can’t survive.
  5. Explicit user call (Rust no_std bridges): nros_rmw_<name>::register() from main() — drags the rlib’s CGU into the binary so the linkme entry is reachable. See examples/bridges/native-rust-zenoh-to-dds/.

Bare-metal + RTOS targets that don’t run .init_array rely on (4). Pure-Rust no_std binaries with multiple backends rely on (5). POSIX builds get (1) + (2) + (3) for free.

Ctor ordering

POSIX .init_array runs ctors in link order, not in any user-controlled sequence. When multiple backends auto-register in one binary, the first to fire owns the default slot — the one selected by Executor::open() / nros::init() with no .rmw(name) argument. The order is reproducible per link graph but not portable across linkers (lld vs. mold vs. gold) or build configs (LTO can reorder via --print-icf-sections collapse) and must not be relied on for correctness.

Disambiguation is the user’s job in multi-backend binaries. Use the named entry points:

  • Rust: Executor::open_with_rmw("zenoh", &cfg) for the primary session; node_builder("name").rmw("cyclonedds").build() for additional Nodes.
  • C: nros_node_init_ex with nros_node_options_t.rmw_name set.
  • C++: nros::Executor::open_with_rmw(...) and nros::NodeBuilder::rmw(...) mirror the Rust API (Phase 104.C.9).

The examples/bridges/native-rust-zenoh-to-cyclonedds/ demo shows the pattern end-to-end: both Zenoh and Cyclone DDS backend ctors fire at lib-load (so the registry has both "zenoh" and "cyclonedds" slots populated), and open_with_rmw("zenoh", ...) plus node_builder("egress").rmw("cyclonedds") pin each session to its intended backend without depending on link-order luck.

Single-backend builds keep the legacy ergonomics — only one ctor fires, the default-slot convention picks it up, and no user-visible name is ever required. The cost of naming is paid only when multiple backends coexist.

Real-time budget per backend

The poll loop’s worst-case execution time is dominated by the backend’s transport drain. Bridge users summing bridge_wcet = Σ poll_i + Σ dispatch_j need each backend’s contribution; this table captures the current best-effort estimates from per-backend microbenchmarks (packages/testing/nros-bench/wcet-cycles-qemu/, packages/testing/nros-bench/wake-latency-cortex-m3/) + heap-usage stats from cargo build --release symbol dumps.

Backendpoll_wcet_usBuffer-pool sizeNotes
zenoh-pico (nros-rmw-zenoh)~50–200 µs nominal on Cortex-M3 (FreeRTOS QEMU); P99 ≤ 1 ms under 100 Hz pub loadZ_BATCH_UNICAST_SIZE (default 6500 B/peer) + 4 KB per subscription buffered ringWake-cb collapses idle wait to kernel xSemaphore post — sub-poll-period latency when transport notifies. POSIX cv-wait path same shape, ~1 µs notify-to-dispatch.
XRCE-DDS (nros-rmw-xrce-cffi)~100–500 µs per uxr_run_session_time on POSIX; agent-round-trip dominates over local poll. Bare-metal targets pay the same poll cost.STREAM_HISTORY (4) × UXR_CONFIG_UDP_TRANSPORT_MTU (512 B default) ≈ 2 KB/stream; one input + one output stream per sessionPoll-only — set_wake_callback slot is NULL; spin_once cv-wait still wakes on its deadline. Agent does the reliable-retransmit accounting; client adds ~10 µs per stream per tick.
Cyclone DDS (nros-rmw-cyclonedds)~150–600 µs on POSIX; C++ listener callback latency depends on Cyclone’s reader-cache scan.Cyclone’s RTPS history per the DDS QoS History.depth (default 10) + Cyclone’s own DDSI buffer pool (~32 KB default)Listener-side set_wake_callback wiring is follow-up — today the C++ vtable sets the slot NULL. Memory footprint dominated by Cyclone itself, not the nano-ros shim.

Bridge users: sum the poll_wcet_us for every backend the bridge process opens, then add per-callback dispatch budget (typically <10 µs for the executor’s arena dispatch + the user callback’s own work). A bridge_picas_priority regression test (blocked on the PiCAS dispatcher) will eventually pin a bar to this table.

Per-backend README.md files live at packages/{zpico/nros-rmw-zenoh,dds/nros-rmw-cyclonedds,xrce/nros-rmw-xrce-cffi}/README.md (when present); reach out to the backend’s maintainer for fresh microbench numbers on a different target class than the ones above.

See also

  • roadmap doc — Appendix D carries LOC sizing, port shapes, and risk notes.
  • Custom Transport porting guide — how the transport vtable composes with the RMW vtable.
  • packages/dds/nros-rmw-cyclonedds/ — reference layout for the C++ vtable consumer.
  • packages/px4/nros-rmw-uorb/ — reference layout for the C++ vtable consumer with PX4 SDK integration.

ROS 2 rmw_zenoh Interoperability

This document describes how nros communicates with standard ROS 2 nodes using rmw_zenoh_cpp.

Status: WORKING

nros <-> ROS 2 rmw_zenoh communication is fully operational as of January 2025.

Architecture

┌─────────────────────────┐          ┌─────────────────────────┐
│   ROS 2 Node            │          │   nros Node             │
│   (rmw_zenoh_cpp)       │<-------->│   (zenoh-pico)          │
│                         │  Zenoh   │                         │
│   ros2 topic echo       │  Router  │   native-rs-talker      │
│   /chatter              │ (zenohd) │                         │
└─────────────────────────┘          └─────────────────────────┘

Both nodes connect to the same Zenoh router (zenohd) or communicate directly in peer mode.

Quick Start

# Terminal 1: Start zenoh router
zenohd --listen tcp/127.0.0.1:7447

# Terminal 2: Run nros talker
cargo run -p native-rs-talker --features zenoh -- --tcp 127.0.0.1:7447

# Terminal 3: Run ROS 2 listener
source /opt/ros/humble/setup.bash
export RMW_IMPLEMENTATION=rmw_zenoh_cpp
export ZENOH_CONFIG_OVERRIDE='mode="client";connect/endpoints=["tcp/127.0.0.1:7447"]'
ros2 topic echo /chatter std_msgs/msg/Int32 --qos-reliability best_effort

Protocol Requirements

1. Data Topic Key Expression Format

For ROS 2 Humble, the data key expression format is:

<domain_id>/<topic_name>/<type_name>/<type_hash>

Important: For Humble, use TypeHashNotSupported as the type hash (no RIHS01_ prefix).

Example:

0/chatter/std_msgs::msg::dds_::Int32_/TypeHashNotSupported

Note: Newer ROS 2 versions (Iron+) use RIHS01_<hash> format with actual type hashes.

Implementation: TopicInfo::to_key() in the transport traits module

2. Liveliness Token Format

rmw_zenoh uses Zenoh liveliness tokens for discovery. Without these, ros2 topic list won’t show nros topics.

Node Liveliness Token:

@ros2_lv/<domain>/<zid>/0/0/NN/%/%/<node_name>

Publisher Liveliness Token:

@ros2_lv/<domain>/<zid>/0/11/MP/%/%/<node_name>/<topic>/<type>/RIHS01_<hash>/<qos>

Subscriber Liveliness Token:

@ros2_lv/<domain>/<zid>/0/11/MS/%/%/<node_name>/<topic>/<type>/RIHS01_<hash>/<qos>

Key differences from data keyexprs:

  • Liveliness tokens do use RIHS01_ prefix
  • Topic names use % instead of / (e.g., %chatter not /chatter)
  • ZenohId must be in LSB-first hex format

Implementation: Ros2Liveliness in the transport/zenoh module

3. QoS String Format

The QoS portion of liveliness tokens uses this format:

reliability:durability:history,depth:deadline_sec,deadline_nsec:lifespan_sec,lifespan_nsec:liveliness,liveliness_sec,liveliness_nsec

Values:

  • Reliability: 1=RELIABLE, 2=BEST_EFFORT
  • Durability: 1=TRANSIENT_LOCAL, 2=VOLATILE
  • History: 1=KEEP_LAST

For BEST_EFFORT/VOLATILE with KEEP_LAST depth 1:

2:2:1,1:,:,:,,

Important: Empty values default to RELIABLE which causes QoS mismatch. Always specify explicit values.

4. CDR Message Format

Messages use CDR (Common Data Representation) little-endian encoding:

[0x00, 0x01, 0x00, 0x00]  // CDR encapsulation header (LE)
[... CDR payload ...]     // Message data

Implementation: the serdes crate

5. RMW Attachment Format

rmw_zenoh requires metadata attached to each published message. For Humble, the serialization format is:

FieldSizeFormat
sequence_number8 bytesint64, little-endian
timestamp8 bytesint64, little-endian (nanoseconds)
gid_length1 byteVLE encoded (value: 16)
gid16 bytesRandom bytes, constant per publisher

Total: 33 bytes

The serialization uses Zenoh’s ze_serializer to ensure compatibility with zenoh-cpp’s deserializer.

Implementation: serialize_rmw_attachment() in zenoh-pico/src/serializer.rs

6. ZenohId Format

The ZenohId in liveliness tokens must be formatted in LSB-first (little-endian) hex format.

#![allow(unused)]
fn main() {
// Correct: LSB-first
fn to_hex_string(&self) -> String {
    let mut hex = String::new();
    for byte in self.bytes.iter() {  // LSB first
        write!(&mut hex, "{:02x}", byte).ok();
    }
    hex
}
}

Common Issues and Solutions

Issue: Discovery works but no messages received

Symptom: ros2 topic list shows the topic, ros2 topic info shows correct QoS, but ros2 topic echo receives nothing.

Cause: Data keyexpr format mismatch.

Solution: For Humble, don’t use RIHS01_ prefix in data keyexprs:

  • Wrong: 0/chatter/std_msgs::msg::dds_::Int32_/RIHS01_TypeHashNotSupported
  • Correct: 0/chatter/std_msgs::msg::dds_::Int32_/TypeHashNotSupported

Issue: Topic not visible in ros2 topic list

Symptom: nros publishes but ROS 2 doesn’t see the topic.

Cause: Liveliness token format incorrect.

Solutions:

  1. Check ZenohId is LSB-first hex format
  2. Ensure topic name uses % prefix (e.g., %chatter)
  3. Verify QoS string has explicit values, not defaults

Issue: QoS incompatibility warnings

Symptom: ROS 2 logs QoS incompatibility or subscriber doesn’t receive.

Cause: Publisher and subscriber QoS don’t match.

Solution: Use BEST_EFFORT for both:

ros2 topic echo /chatter std_msgs/msg/Int32 --qos-reliability best_effort

Issue: rmw_zenoh not connecting to router

Symptom: rmw_zenoh uses peer mode by default, not connecting to zenohd.

Solution: Force client mode:

export ZENOH_CONFIG_OVERRIDE='mode="client";connect/endpoints=["tcp/127.0.0.1:7447"]'

Implementation Status

ComponentStatusLocation
Data keyexpr formatDoneTopicInfo::to_key()
Liveliness tokensDoneRos2Liveliness
QoS formatDone2:2:1,1:,:,:,,
CDR serializationDoneserdes crate
RMW attachmentDoneserialize_rmw_attachment()
ZenohId formatDoneZenohId::to_hex_string()
Node integrationDoneConnectedNode

Version Compatibility

ROS 2 VersionType Hash FormatStatus
HumbleTypeHashNotSupportedWorking
Iron+RIHS01_<sha256>Needs type hash computation

References

FreeRTOS LAN9118 Debugging

This page collects the low-level FreeRTOS + lwIP + QEMU LAN9118 notes that are useful when the platform guide is not enough.

Stack

  • Board: QEMU MPS2-AN385, Cortex-M3, 25 MHz.
  • Ethernet: LAN9118 MMIO at 0x4020_0000, IRQ 13.
  • Software: FreeRTOS, lwIP threaded mode (NO_SYS=0), zenoh-pico over BSD sockets.

Default task layout:

PriorityTaskRole
4tcpip_threadlwIP protocol processing
4poll taskdrains LAN9118 RX FIFO into lwIP
4zenoh read / leasezenoh-pico background I/O
3app tasknano-ros executor and user code
0idlemust execute WFI for QEMU networking

The poll task must run at least as high as the zenoh read task; if it cannot drain the RX FIFO, TCP keep-alives are missed and zenoh sessions expire.

LAN9118 Registers

Useful direct registers:

OffsetNamePurpose
0x00RX_DATA_PORTRX FIFO data
0x20TX_DATA_PORTTX FIFO data
0x40RX_STAT_PORTRX packet status
0x50ID_REVchip ID / revision
0x54IRQ_CFGinterrupt output configuration
0x58INT_STSinterrupt status
0x6CRX_CFGRX configuration
0x70TX_CFGTX configuration
0x7CRX_FIFO_INFRX FIFO status / bytes used
0x80TX_FIFO_INFTX FIFO free space
0xA4MAC_CSR_CMDindirect MAC CSR command
0xA8MAC_CSR_DATAindirect MAC CSR data

Indirect MAC CSR registers include MAC_CR (TXEN/RXEN), ADDRH, ADDRL, MII_ACC, and MII_DATA. Always wait for the CSR busy bit to clear before starting another access.

QEMU Delivery Model

QEMU routes LAN9118 traffic through a legacy hub:

guest LAN9118 NIC <-> QEMU hub <-> backend (Slirp / TAP / socket)

When the guest transmits, QEMU delivers the frame synchronously through the hub. Slirp or TAP replies, such as ARP responses, may arrive in the RX FIFO before the guest TX register write returns. This is useful for diagnostics: check RX_FIFO_INF immediately after TX, before yielding to FreeRTOS.

QEMU drops frames that arrive before MAC_CR_RXEN is enabled, and its LAN9118 model does not flush queued packets when RXEN changes. Enable RX before traffic starts.

Common Pitfalls

Diagnostic shows zero RX packets. The poll task may have consumed the packet before diagnostics read the register. Check RX_FIFO_INF immediately after TX or temporarily prevent the poll task from running.

Network init hangs. lwIP threaded mode requires the FreeRTOS scheduler to be running before tcpip_init(). Initialize networking from a task, not before vTaskStartScheduler().

HardFault on first context switch. The Cortex-M3 FreeRTOS port expects vPortSVCHandler, xPortPendSVHandler, and xPortSysTickHandler at the vector table positions. Avoid wrapper functions except for the guarded SysTick case.

lwIP core-lock assertion. Keep LWIP_TCPIP_CORE_LOCKING=0; the setup path calls netif functions from the app task after tcpip_init().

Intermittent stalls. Always write IRQ_CFG_DEFAULT (0x22000111) during LAN9118 init, even when polling.

Corrupted RX frames. RX_STAT_PORT packet length includes the 4-byte FCS. Read all aligned FIFO words, including CRC bytes, but pass only pkt_len - 4 bytes to lwIP.

Second QEMU node cannot connect. Seed rand() uniquely per node. zenoh-pico uses z_random_fill() for session IDs; identical QEMU boots otherwise produce duplicate IDs.

lwIP ASSERT: Invalid mbox. The application stack is too small. The executor arena lives on the FreeRTOS task stack; use 64 KB for service and action examples.

Action client times out on result. Manual-polling action servers created with create_action_server() must call try_handle_get_result() explicitly after completing a goal.

No inbound frames in QEMU. The idle hook must execute WFI so QEMU’s main event loop can service network file descriptors:

void vApplicationIdleHook(void) {
    __asm__ volatile("wfi");
}

Debugging Tips

  • Use ARM semihosting for early printf output.
  • Launch QEMU with -monitor telnet:127.0.0.1:4444,server,nowait and run info network or info qtree.
  • Send a diagnostic ARP request and check for immediate RX FIFO data.
  • Confirm ID_REV, MAC_CR TXEN/RXEN, TX_FIFO_INF, RX_FIFO_INF, and INT_STS during bring-up.

References

  • FreeRTOS-Plus-TCP MPS2_AN385 driver.
  • Zephyr drivers/ethernet/eth_smsc911x.c.
  • ARM CMSIS LAN9220 Ethernet driver templates.
  • QEMU hw/net/lan9118.c.
  • nano-ros packages/drivers/lan9118-smoltcp/.

Patched qemu-system-arm Binary

nano-ros ships a project-local QEMU build that the test harness uses in preference to whatever qemu-system-arm is on $PATH. This chapter explains why it exists, how it gets built, how a test picks it up, and how to add a new patch.

Why

Two production-blocking issues motivated the patched build:

  1. LAN9118 RX FIFO drain bug (mainline QEMU through at least 11.0). Under sustained burst RX, the LAN9118 emulator’s lan9118_can_receive() can stop returning true even after the guest drains the FIFO, so frames silently disappear. bisected this and ships the fix as third-party/qemu/patches/0001-hw-net-lan9118-add-can_receive-flush-on-FIFO-drain.patch. Bare-metal MPS2-AN385 RTPS / Zenoh tests on the system QEMU sporadically fail without it.

  2. -netdev dgram,local.type=unix,… requires QEMU 7.2+. Ubuntu jammy ships QEMU 6.2, and the dgram-tunnel pattern is how NuttX / ThreadX DDS multi-instance tests cross-deliver frames between two QEMU processes after retired the broken -netdev socket,mcast= cross-process path. With a too-old system QEMU, those tests fall back to [SKIPPED].

Both problems disappear once tests use the patched build at build/qemu/bin/qemu-system-arm.

Layout

third-party/qemu/qemu/              # submodule, pinned to stable-11.0
third-party/qemu/patches/           # patch series, applied on top
  0001-hw-net-lan9118-…patch
build/qemu/bin/qemu-system-arm      # final installed binary
build/qemu/share/qemu/…             # firmware, etc.

The submodule URL and pin live in .gitmodules:

[submodule "third-party/qemu/qemu"]
    path = third-party/qemu/qemu
    url = https://gitlab.com/qemu-project/qemu.git
    branch = stable-11.0

stable-11.0 is QEMU 11.0.x — already past the 7.2 cutoff for -netdev dgram unix and recent enough that the patch series is small.

Build

just qemu setup-qemu (pulled in by just setup qemu and just setup all) does the full build:

just qemu setup-qemu

The recipe:

  1. Inits the submodule shallowly if it is not already present.
  2. Short-circuits when build/qemu/bin/qemu-system-arm is newer than every file under third-party/qemu/patches/ (touch a patch to force a rebuild).
  3. Resets the submodule to its pinned tip, applies every .patch under third-party/qemu/patches/ in alphabetical order via git apply.
  4. Configures with --target-list=arm-softmmu (no other arches, no docs, no tools) and --enable-slirp.
  5. make -j$(nproc) && make install into build/qemu/.

End-to-end cost is roughly ten minutes the first time and ~150 MB of disk. Subsequent runs are no-ops.

just qemu doctor reports the build status and clearly distinguishes the patched build from the system fallback.

How tests pick it up

The single resolver lives in packages/testing/nros-tests/src/qemu.rs:

#![allow(unused)]
fn main() {
pub fn qemu_system_arm_path() -> std::ffi::OsString { … }
pub fn qemu_system_arm_cmd()  -> std::process::Command { … }
}

Selection order:

  1. QEMU_SYSTEM_ARM env var — developer override / CI pin.
  2. Project-local <workspace>/build/qemu/bin/qemu-system-arm when it exists (auto-detected via CARGO_MANIFEST_DIR walk to the workspace Cargo.toml).
  3. System qemu-system-arm on $PATH — kept as fallback so a minimal install still produces a clean [SKIPPED] rather than an exec error.

Every Command::new("qemu-system-arm") in the test crates goes through this helper. New tests must use it.

just/qemu-baremetal.just, just/nuttx.just and just/freertos.just do the same in shell, gating their qemu-system-arm invocations through:

QEMU_BIN := absolute_path("build/qemu") / "bin/qemu-system-arm"

# inside a recipe:
{{ if path_exists(QEMU_BIN) == "true" { QEMU_BIN } else { "qemu-system-arm" } }} -M virt …

Smoke test

packages/testing/nros-tests/tests/qemu_patched_binary.rs asserts:

  • qemu_system_arm_path() resolves to either QEMU_SYSTEM_ARM or <workspace>/build/qemu/bin/qemu-system-arm.
  • The patched binary reports version ≥ 7.2.
  • -netdev help advertises dgram (the multi-instance backend).

Tests use nros_tests::skip! when the patched build is absent — a fresh clone without just qemu setup-qemu surfaces a clear [SKIPPED] with the suggested remedy instead of silently passing.

Adding a new patch

  1. Land the upstream fix or write a downstream-only patch against the pinned submodule tip.
  2. Save it as a numbered file under third-party/qemu/patches/ (e.g. 0002-hw-net-…patch). Keep one logical change per patch.
  3. Bump any inline comment that names specific patches.
  4. Touch the patch file (or just commit it) — just qemu setup-qemu detects the patch is newer than the installed binary and rebuilds.
  5. Bump the relevant CI cache key (see) so other machines also rebuild.

Submodule pin bump

When upstream rolls a new stable branch and an existing patch either lands upstream or no longer applies cleanly:

  1. Edit .gitmodules branch = stable-… to the new branch name.
  2. cd third-party/qemu/qemu && git fetch && git checkout origin/stable-…
  3. Re-run just qemu setup-qemu. If a patch fails to apply, either drop it (landed upstream) or rebase it onto the new tip.
  4. Commit the submodule SHA bump together with any patch-series updates.

Scope

Only qemu-system-arm is unified. qemu-system-riscv32 is Espressif’s fork (different upstream, different patches); qemu-system-riscv64 and qemu-system-aarch64 ship no patches today. Other arches stay on the system binary until they accumulate their own patches; the helper pattern is easy to copy when that happens.

Opaque Storage Sizing

nano-ros exposes a C and C++ API on top of Rust types whose layout is chosen by the Rust compiler. The C/C++ side has to allocate storage for those types — by value, on the user’s stack or BSS — without knowing their exact byte size. The runtime makes Rust the single source of truth for those sizes: the values flow from core::mem::size_of of the real Rust types into auto-generated C/C++ headers, with no hand-tuned constants.

The pattern

Every size that crosses the Rust / C boundary follows the same path:

  1. Export from nros::sizes — the nros umbrella crate defines a pub const FOO_SIZE: usize = core::mem::size_of::<T>(); and emits a #[used] static __NROS_SIZE_FOO: [u8; FOO_SIZE] = [0u8; FOO_SIZE] with a #[no_mangle] symbol whose storage size in the rlib is FOO_SIZE. The two artefacts come from a single export_size! macro invocation:

    #![allow(unused)]
    fn main() {
    // packages/core/nros/src/sizes.rs
    export_size!(pub PUBLISHER_SIZE = RmwPublisher);
    export_size!(pub SUBSCRIBER_SIZE = RmwSubscriber);
    export_size!(pub EXECUTOR_SIZE = nros_node::Executor);
    // ...etc
    }
  2. Probe from consumer build scriptsnros-c/build.rs and nros-cpp/build.rs use the helper crate nros-sizes-build to find the compiled nros rlib (via cargo metadata + a glob over target/<triple>/<profile>/deps/) and read the __NROS_SIZE_* symbol storage sizes with the object crate. No subprocess, no llvm-nm; pure Rust.

  3. Emit #define NROS_FOO_SIZE into a generated headernros_config_generated.h (C) and nros_cpp_config_generated.h (C++) carry the probe values. types.h includes the generated config transitively, so every nros C header sees NROS_*_SIZE automatically.

  4. C/C++ structs use the macros

    typedef struct nros_publisher_t {
        /* ... */
        _Alignas(8) uint8_t _opaque[NROS_PUBLISHER_SIZE];
    } nros_publisher_t;
    
    class Publisher {
        alignas(8) uint8_t storage_[NROS_PUBLISHER_SIZE];
        /* ... */
    };
    

    The Rust side reads/writes the same bytes via &mut *(opaque as *mut RmwPublisher). C and Rust agree on the size by construction.

What the SSoT covers today

The nros::sizes module exports:

SymbolTypeUsed by
SESSION_SIZERmwSessionnros_support_t._opaque
PUBLISHER_SIZERmwPublishernros_publisher_t._opaque, nros::Publisher<M>::storage_
SUBSCRIBER_SIZERmwSubscribernros::Subscription<M>::storage_
SERVICE_CLIENT_SIZERmwServiceClientnros::Client<S>::storage_
SERVICE_SERVER_SIZERmwServiceServernros::Service<S>::storage_
EXECUTOR_SIZEnros_node::Executornros_executor_t._opaque, nros::Executor::storage_
GUARD_CONDITION_SIZEnros_node::GuardConditionHandlenros_guard_condition_t._guard_opaque, nros::GuardCondition::storage_
LIFECYCLE_CTX_SIZEnros_node::lifecycle::LifecyclePollingNodeCtxnros_lifecycle_state_machine_t._opaque_storage
ACTION_SERVER_INTERNAL_SIZEActionServerInternalLayoutnros_action_server_t._internal
CPP_ACTION_SERVER_SIZECppActionServerLayoutnros::ActionServer<A>::storage_
CPP_ACTION_CLIENT_SIZECppActionClientLayoutnros::ActionClient<A>::storage_

Plus three *Internal C-API shim structs (ServiceServerInternal, ServiceClientInternal, ActionClientInternal) that cbindgen now emits directly into nros_generated.h because they’re #[repr(C)] — the C side just embeds them as typed fields, no opaque storage at all.

Layout-mirror trick

Some downstream types — nros-c::ActionServerInternal, nros-cpp::CppActionServer, nros-cpp::CppActionClient — embed C-API pointer types (*mut nros_action_server_t, *const nros_goal_handle_t) that aren’t visible from the nros umbrella crate. They can’t be referenced from nros::sizes directly, but their byte size only depends on the field shape: pointers are pointers, Option<extern "C" fn> collapses to a fn-pointer-sized slot via niche optimization, etc.

The fix is a layout-mirror struct in nros::sizes:

#![allow(unused)]
fn main() {
// packages/core/nros/src/sizes.rs
#[repr(C)]
#[doc(hidden)]
pub struct ActionServerInternalLayout {
    pub handle: nros_node::ActionServerRawHandle,
    pub executor_ptr: *mut c_void,
    pub c_goal_callback: unsafe extern "C" fn(
        *mut c_void, *const c_void, *const u8, usize, *mut c_void,
    ) -> i32,
    // ...same field shape as the real `ActionServerInternal`,
    // with C-API pointers replaced by `*mut c_void`...
}
export_size!(pub ACTION_SERVER_INTERNAL_SIZE = ActionServerInternalLayout);
}

Downstream then asserts byte-equivalence at compile time:

#![allow(unused)]
fn main() {
// packages/core/nros-c/src/opaque_sizes.rs
const _: () = assert!(
    size_of::<crate::action::ActionServerInternal>()
        == size_of::<nros::sizes::ActionServerInternalLayout>(),
    "ActionServerInternal size diverges from nros::sizes::ActionServerInternalLayout — \
     update the layout mirror in `nros/src/sizes.rs`"
);
}

The tripwire: any field-shape change in the real wrapper (adding a field, changing a pointer to a value, etc.) must be paired with an update to the mirror. The build fails immediately if they diverge.

How sizing works today

  • All four *Internal shim types are #[repr(C)] and embedded as typed fields in their outer nros_*_t structs.
  • All seven C++ wrappers (Publisher, Subscription, Service, Client, ActionServer, ActionClient, GuardCondition) use the NROS_*_SIZE / NROS_CPP_*_STORAGE_SIZE probe macros.
  • types.h ships zero *_OPAQUE_U64S macros; the four consumer module headers route through the probe.
  • The remaining hand-coded “upper bound” assertions (e.g., the EXECUTOR_OPAQUE_U64S envelope check in nros-c/build.rs) are defence-in-depth: they fire if a Rust type accidentally exceeds a configured envelope (which would normally also be flagged by the config knobs).

Adding a new size export

When a new Rust handle type needs to cross the FFI boundary:

  1. Add the export_size!(pub FOO_SIZE = nros_node::Foo) line to nros/src/sizes.rs. (If Foo lives in a downstream crate that nros can’t import, define a FooLayout mirror struct first and add a byte-equivalence assert in the owning crate.)
  2. Add a let probe_foo = probed.get("FOO_SIZE").copied().unwrap_or(0) line to nros-c/build.rs (and/or nros-cpp/build.rs), and emit #define NROS_FOO_SIZE {probe_foo} into the generated config header.
  3. Use NROS_FOO_SIZE in the C/C++ struct that needs the storage.

Cross-target verification: the same probe runs for every target triple. There’s no per-platform branching — size_of::<T>() resolves to the right value at the target’s compile time.

See also

  • packages/core/nros-sizes-build/src/lib.rs for the rlib probe implementation.
  • packages/core/nros/src/sizes.rs for the canonical exports.

Scheduling Models

This chapter introduces the real-time scheduling models used by the platforms nano-ros supports. Understanding these models helps you make informed decisions about task priorities, deadline guarantees, and platform selection for your application.

Background: What Is Real-Time Scheduling?

A real-time scheduler decides which task (or thread) runs on the CPU at any given moment. The key property is predictability — not raw speed. A system that always finishes a 10 ms task in exactly 10 ms is more “real-time” than one that finishes it in 1 ms but occasionally takes 50 ms.

Two key concepts recur across all models:

  • Priority: a numeric value that determines which task runs when multiple tasks are ready. Higher-priority tasks preempt lower-priority ones.
  • Preemption: the ability to interrupt a running task to run a higher-priority one. Non-preemptive (cooperative) systems only switch tasks at explicit yield points.

Hard vs. Soft Real-Time

TypeGuaranteeConsequence of missed deadline
HardDeadline must never be missedSystem failure (safety hazard)
SoftDeadline should rarely be missedDegraded quality (dropped frame, late message)

nano-ros targets soft real-time by default (bounded message latency, bounded memory). Hard real-time is achievable on RTIC and with careful priority assignment on RTOS platforms.

Scheduling Algorithms

Fixed-Priority Preemptive (FPP)

The most common RTOS scheduling algorithm. Each task has a static priority assigned at creation time. The scheduler always runs the highest-priority ready task. When a higher-priority task becomes ready, it immediately preempts the running task.

Time ──────────────────────────────────────►
        ┌───────┐           ┌───────┐
High    │ Task A│           │ Task A│        (preempts B when ready)
        └───┬───┘           └───┬───┘
            │ ┌───┐    ┌───┐   │ ┌───┐
Low         └►│ B │    │ B │   └►│ B │       (runs when A is blocked)
              └───┘    └───┘     └───┘

Schedulability analysis uses Rate-Monotonic Analysis (RMA): assign higher priority to tasks with shorter periods. RMA is optimal for independent periodic tasks — if any fixed-priority assignment can meet all deadlines, RMA can too.

The response time of a task under FPP is:

R_i = C_i + Σ ⌈R_i / T_j⌉ × C_j    (for all higher-priority tasks j)

Where C_i is worst-case execution time, T_j is the period of higher-priority task j. The task meets its deadline if R_i ≤ D_i.

Used by: FreeRTOS, ThreadX, NuttX (SCHED_FIFO), Zephyr (preemptive threads), RTIC.

Round-Robin (Time-Sliced)

Tasks at the same priority level share CPU time in equal time slices (quanta). When a task’s quantum expires, the scheduler switches to the next same-priority task. Tasks at different priority levels still follow FPP rules.

Time ──────────────────────────────────────►
        ┌──┐┌──┐┌──┐┌──┐┌──┐┌──┐
Pri 3   │A ││B ││A ││B ││A ││B │            (equal time slices)
        └──┘└──┘└──┘└──┘└──┘└──┘

Round-robin prevents starvation among equal-priority tasks but adds scheduling jitter (a task may wait up to (N-1) × quantum before running, where N is the number of same-priority tasks).

Used by: NuttX (SCHED_RR), FreeRTOS (when configUSE_TIME_SLICING=1), ThreadX (when time_slice > 0).

Cooperative (Non-Preemptive)

Tasks run until they explicitly yield the CPU. No preemption occurs. This eliminates the need for locks (no race conditions) but requires every task to yield frequently. A single task that runs too long blocks all others.

Time ──────────────────────────────────────►
        ┌──────────┐     ┌────┐
Task A  │ runs     │     │    │              (runs until yield())
        └────┬─────┘     └──┬─┘
             │ ┌────────┐   │ ┌──────┐
Task B       └►│ runs   │   └►│ runs │       (gets CPU only when A yields)
               └────────┘     └──────┘

Used by: Zephyr (cooperative threads via K_PRIO_COOP), bare-metal super-loops, Embassy async executor.

Interrupt-Driven (Hardware Scheduling)

The CPU’s interrupt controller acts as a hardware scheduler. Each interrupt source has a hardware priority level. When an interrupt fires, the hardware saves context and jumps to the handler — no software scheduler overhead. Nested interrupts provide preemption between priority levels.

On ARM Cortex-M, the Nested Vectored Interrupt Controller (NVIC) provides:

  • Up to 256 priority levels (typically 8–16 usable)
  • Zero-cycle context switch for tail-chaining interrupts
  • Deterministic latency (12 cycles to handler entry on Cortex-M3)
                          NVIC
IRQ Priority    ┌──────────────────┐
   0 (highest)  │ SysTick          │──► timing / OS tick
   1            │ UART RX          │──► message receive
   2            │ Timer            │──► periodic publish
   3 (lowest)   │ PendSV           │──► background work
                └──────────────────┘

This is the most deterministic scheduling model — no jitter from software task switching, no priority inversion, and worst-case latency is bounded by the longest critical section that disables interrupts.

Used by: RTIC (exclusively), bare-metal interrupt handlers.

Earliest Deadline First (EDF)

A dynamic-priority algorithm: the task with the nearest deadline always runs next. EDF is optimal — it can schedule any task set that is schedulable by any algorithm, up to 100% CPU utilization (vs. ~69% for RMA with harmonic periods).

Time ──────────────────────────────────────►
        ┌───┐       ┌───┐
Task A  │ d=5│      │d=15│     (deadline 5, then 15)
        └───┘       └───┘
            ┌────┐      ┌────┐
Task B      │d=10│      │d=20│  (deadline 10, then 20)
            └────┘      └────┘

EDF is harder to analyze than FPP (no simple closed-form response time) and harder to implement (priority changes every scheduling decision). Overload behavior is also less predictable — under FPP, low-priority tasks miss deadlines first; under EDF, deadline misses can cascade unpredictably.

Used by: Zephyr (CONFIG_SCHED_DEADLINE + k_thread_deadline_set()). Not currently used by nano-ros.

Platform Scheduling Comparison

RTIC (ARM Cortex-M)

RTIC is not an RTOS — it is a concurrency framework that compiles directly to hardware interrupt handlers. There is no scheduler, no task control blocks, and no context switch overhead.

PropertyValue
ModelInterrupt-driven (NVIC hardware scheduling)
PreemptionYes — hardware interrupt nesting
Priority levels4–16 (depends on Cortex-M variant)
Priority directionLower number = higher priority (ARM convention)
Context switch12 cycles (Cortex-M3 tail-chain)
Mutual exclusionStack Resource Policy (SRP) — compile-time deadlock-free
Scheduling analysisFully static — all priorities and resources known at compile time

RTIC’s key innovation is the Stack Resource Policy (SRP): shared resources are protected by ceiling-based priority elevation, not locks. The compiler proves at build time that no deadlock or unbounded priority inversion can occur. This gives interrupt-driven scheduling the safety of a cooperative model.

#![allow(unused)]
fn main() {
#[rtic::app(device = stm32f4xx_hal::pac)]
mod app {
    #[task(priority = 2, shared = [sensor_data])]
    async fn read_sensor(mut ctx: read_sensor::Context) {
        ctx.shared.sensor_data.lock(|data| {
            *data = read_adc();  // runs at ceiling priority
        });
    }

    #[task(priority = 1, shared = [sensor_data])]
    async fn publish(mut ctx: publish::Context) {
        ctx.shared.sensor_data.lock(|data| {
            node.publish(*data);  // cannot deadlock — SRP guarantee
        });
    }
}
}

nano-ros integration: RTIC tasks call nano-ros directly — no executor needed. Each RTIC task can own a publisher, subscription, or service handle. See examples/stm32f4/rust/talker-rtic/.

FreeRTOS

The most widely deployed RTOS. Uses fixed-priority preemptive scheduling with optional round-robin time-slicing for same-priority tasks.

PropertyValue
ModelFixed-priority preemptive (FPP)
PreemptionYes — configUSE_PREEMPTION=1 (default)
Priority levelsconfigMAX_PRIORITIES (typically 8–32)
Priority directionHigher number = higher priority
Context switch~80 cycles on Cortex-M (PendSV handler)
Mutual exclusionMutexes with optional priority inheritance
Time slicingOptional — configUSE_TIME_SLICING

FreeRTOS tasks are created with xTaskCreate(), specifying a priority and stack size. The scheduler runs the highest-priority ready task. When multiple tasks share a priority and time-slicing is enabled, they round-robin at tick boundaries.

Priority inheritance: FreeRTOS mutexes optionally support priority inheritance to mitigate priority inversion. When a low-priority task holds a mutex needed by a high-priority task, the low-priority task temporarily inherits the high priority until it releases the mutex.

Without inheritance:        With inheritance:
  H ──blocks──►             H ──blocks──►
  M ──runs────►             L ──promoted──► (runs at H's priority)
  L ──holds mutex           L ──releases──► H runs immediately
     (M preempts L →        (no M preemption → bounded inversion)
      unbounded inversion)

nano-ros task layout on FreeRTOS:

TaskDefault PriorityStackRole
nros_app3 (Normal)64 KBExecutor, callbacks, spin
net_poll4 (AboveNormal)1 KBPoll LAN9118 RX FIFO
zenoh read4 (AboveNormal)5 KBSocket read, message decode
zenoh lease4 (AboveNormal)5 KBKeep-alive, lease monitor
tcpip_thread4 (AboveNormal)4 KBlwIP protocol processing

ThreadX (Azure RTOS)

ThreadX is designed for deeply embedded systems with a unique preemption-threshold feature not found in other RTOSes.

PropertyValue
ModelFixed-priority preemptive (FPP) with preemption-threshold
PreemptionYes — with configurable threshold per thread
Priority levels32 (0–31)
Priority directionLower number = higher priority
Context switch~60 cycles (optimized assembly for each architecture)
Mutual exclusionMutexes with priority inheritance
Time slicingPer-thread configurable (time_slice parameter)

The preemption-threshold is ThreadX’s distinguishing feature. Each thread has two priority values:

  1. Priority: determines scheduling order (which thread runs next)
  2. Preemption-threshold: the minimum priority that can preempt this thread
Thread A: priority=10, preempt_threshold=5
  → Scheduled based on priority 10
  → Only threads with priority 0–4 can preempt it
  → Threads with priority 5–9 must wait, even though they're higher priority

Thread B: priority=10, preempt_threshold=10  (threshold = priority)
  → Normal behavior — any higher-priority thread can preempt

This effectively creates non-preemptive regions without disabling interrupts. A thread performing a critical sequence of operations can set a low preemption-threshold to prevent most preemption while still allowing the highest-priority threads through.

The academic basis is the dual-priority model: preemption-threshold reduces context switches (and thus stack usage) while preserving schedulability. Research shows it can reduce RAM requirements by 30–50% compared to pure FPP, because fewer threads need independent stacks for preemption frames.

nano-ros on ThreadX: Currently sets preempt_threshold = priority (no benefit). Future work will expose this as a configurable option.

NuttX

NuttX is a POSIX-compliant RTOS, meaning it implements the full pthread and sched APIs from IEEE 1003.1. This makes it the most portable platform — standard POSIX real-time scheduling is well-understood and widely taught.

PropertyValue
ModelPOSIX SCHED_FIFO (FPP), SCHED_RR, or SCHED_SPORADIC
PreemptionYes (FIFO/RR), configurable
Priority levels1–255 (POSIX sched_param.sched_priority)
Priority directionHigher number = higher priority
Context switchKernel-managed, architecture-dependent
Mutual exclusionPOSIX mutexes with PTHREAD_PRIO_INHERIT or PTHREAD_PRIO_PROTECT
Scheduling policyPer-thread via sched_setscheduler()

NuttX supports three POSIX scheduling policies:

SCHED_FIFO (First-In-First-Out): Pure fixed-priority preemptive. A task runs until it blocks, yields, or is preempted by a higher-priority task. Tasks at the same priority run in FIFO order — no time-slicing.

SCHED_RR (Round-Robin): Same as SCHED_FIFO but with time-slicing for same-priority tasks. Each task gets a time quantum before the scheduler switches to the next task at that priority.

SCHED_SPORADIC (NuttX extension): Implements the sporadic server algorithm for aperiodic event handling. A task alternates between a high “normal” priority and a low “background” priority based on its execution budget:

Budget = 5ms, Replenish period = 20ms

Time ──────────────────────────────────────────────────►
        ┌─────┐                    ┌─────┐
High    │5ms  │                    │5ms  │     (budget)
        └──┬──┘                    └──┬──┘
           │ ┌──────────────┐        │
Low        └►│ background   │        └►...    (budget exhausted)
             └──────────────┘
                    ▲ replenish after 20ms

The sporadic server bounds the interference that an aperiodic task imposes on periodic tasks, making it analyzable with standard RMA techniques. This is valuable for event-driven ROS callbacks that fire at irregular intervals.

Priority inversion protocols: NuttX supports both POSIX priority inheritance (PTHREAD_PRIO_INHERIT) and priority ceiling (PTHREAD_PRIO_PROTECT):

ProtocolHow it worksTrade-off
Priority inheritanceHolder inherits waiter’s priorityDynamic — may chain through multiple locks
Priority ceilingMutex has fixed ceiling priority; holder runs at ceilingStatic — simpler analysis, avoids chained inversion

nano-ros on NuttX: Currently runs with kernel defaults (no explicit scheduling policy). Future work will add SCHED_FIFO and priority configuration.

Zephyr

Zephyr provides the most flexible scheduling model, supporting cooperative threads, preemptive threads, and EDF in a single system.

PropertyValue
ModelCooperative + preemptive + optional EDF
PreemptionConfigurable per-thread
Priority levelsCooperative: K_PRIO_COOP(0) to K_PRIO_COOP(N) (negative values); Preemptive: K_PRIO_PREEMPT(0) to K_PRIO_PREEMPT(N) (positive values)
Priority directionLower number = higher priority
Context switchArchitecture-dependent (~100 cycles on Cortex-M)
Mutual exclusionk_mutex with priority inheritance
Meta-IRQUltra-high-priority threads that preempt even cooperative threads

Zephyr’s scheduling model has three tiers:

Priority Number Line:
◄── higher priority                    lower priority ──►
    │                │                    │
    │  Meta-IRQ      │  Cooperative       │  Preemptive
    │  (negative,    │  (negative,        │  (0 or positive)
    │   special)     │   non-preemptible) │
    │                │                    │

Cooperative threads (K_PRIO_COOP): Cannot be preempted by other threads (only by interrupts). They run until they explicitly yield or block. This is useful for critical sections that must not be interrupted by other threads, while still allowing hardware interrupts.

Preemptive threads (K_PRIO_PREEMPT): Standard FPP behavior. Can be preempted by any higher-priority thread (cooperative or preemptive).

Meta-IRQ threads (CONFIG_NUM_METAIRQ_PRIORITIES): Special ultra-high-priority threads that can preempt even cooperative threads. Used for work that must complete with interrupt-like urgency but needs thread context (e.g., stack, blocking calls). This fills the gap between ISR context (limited API) and thread context (preemptible).

Deadline scheduling: Zephyr optionally supports EDF via CONFIG_SCHED_DEADLINE. Threads call k_thread_deadline_set() to declare their next deadline. Among threads at the same priority level, the scheduler picks the one with the earliest deadline. This allows EDF within a priority band while preserving FPP across bands — a hybrid approach that combines EDF’s optimality with FPP’s predictable overload behavior.

nano-ros on Zephyr: Currently uses a single main thread with default priority. The async service example uses Embassy’s executor with kernel-backed waking (zephyr::embassy::Executor).

Priority Inversion

Priority inversion is a well-studied problem in real-time systems. It occurs when a high-priority task is indirectly blocked by a low-priority task through a shared resource, while a medium-priority task runs unimpeded.

The classic example (from the Mars Pathfinder incident, 1997):

  High-priority task ──► blocks on mutex held by Low
  Medium-priority task ──► preempts Low (doesn't need mutex)
  Low-priority task ──► holds mutex, can't run (M preempts)

  Result: High is blocked by Medium indefinitely

Solutions

SolutionApproachPlatforms
Priority inheritanceMutex holder inherits highest waiter’s priorityFreeRTOS, ThreadX, NuttX, Zephyr
Priority ceilingMutex has fixed ceiling; holder runs at ceilingNuttX (PTHREAD_PRIO_PROTECT)
SRP (Stack Resource Policy)Compile-time ceiling, zero runtime overheadRTIC
Preemption-thresholdLimit which tasks can preemptThreadX
Lock-free designAvoid shared resources entirelynano-ros single-slot buffers

nano-ros mitigates priority inversion architecturally: subscriptions use single-slot buffers with atomic overwrites — no mutex needed between publisher and subscriber tasks. The executor processes callbacks in a single task, eliminating inter-task resource sharing for most use cases.

Choosing a Scheduling Model

CriterionRTICFreeRTOSThreadXNuttXZephyr
DeterminismBest (hardware)Good (FPP)Good (FPP+threshold)Good (POSIX FPP)Good (FPP+coop+EDF)
Worst-case latency12 cycles~80 cycles~60 cyclesKernel-dependent~100 cycles
Priority inversionImpossible (SRP)InheritanceInheritance + thresholdInheritance + ceilingInheritance
Analysis toolsCompile-time proofsRMA/RTARMA/RTA + thresholdPOSIX standard RMARMA + EDF analysis
FlexibilityLow (ARM only)MediumMedium-HighHigh (POSIX)Highest
RAM overheadLowest (no TCBs)LowLow (threshold reduces stacks)Medium (kernel)Medium (kernel)

Use RTIC when: you need hard real-time on ARM Cortex-M, want compile-time scheduling proofs, and can live without dynamic task creation.

Use FreeRTOS when: you need a widely supported RTOS with a small footprint and a large ecosystem. Good for projects where portability across MCU vendors matters more than advanced scheduling features.

Use ThreadX when: you need deterministic scheduling with reduced RAM (preemption-threshold), or your project targets Azure IoT infrastructure. ThreadX is also safety-certified (IEC 61508 SIL 4, ISO 26262 ASIL D).

Use NuttX when: you want POSIX compatibility (reuse Linux-targeted code on embedded), need SCHED_SPORADIC for aperiodic events, or want PTHREAD_PRIO_PROTECT for static priority ceiling analysis.

Use Zephyr when: you need maximum scheduling flexibility (cooperative + preemptive + EDF in one system), want a Linux Foundation backed project with broad hardware support, or need meta-IRQ for interrupt-like thread priorities.

Further Reading

  • Liu, C.L. and Layland, J.W. (1973). “Scheduling Algorithms for Multiprogramming in a Hard-Real-Time Environment.” Journal of the ACM, 20(1), 46–61. — The foundational paper for Rate-Monotonic Analysis.
  • Sha, L., Rajkumar, R., and Lehoczky, J.P. (1990). “Priority Inheritance Protocols: An Approach to Real-Time Synchronization.” IEEE Transactions on Computers, 39(9), 1175–1185. — Priority inheritance and priority ceiling protocols.
  • Baker, T.P. (1991). “Stack-Based Scheduling of Realtime Processes.” Real-Time Systems, 3(1), 67–99. — The Stack Resource Policy used by RTIC.
  • Wang, Y. and Saksena, M. (1999). “Scheduling Fixed-Priority Tasks with Preemption Threshold.” RTCSA. — The theoretical basis for ThreadX’s preemption-threshold.
  • Buttazzo, G.C. (2011). Hard Real-Time Computing Systems: Predictable Scheduling Algorithms and Applications. Springer. — Comprehensive textbook covering FPP, EDF, and hybrid approaches.

Dispatch Strategy

Phase 216 design rationale. This chapter explains why DispatchStrategy is shaped the way it is — the trichotomy, the per-Node granularity, the tag-based callback API, the __nros_node_<pkg>_dispatch_strategy() ABI symbol, and the backward-compat contract. For the user-facing tutorials see RTIC Integration and Embassy Integration.

The Inline / Deferred / FromIsr trichotomy

A nano-ros Node declares Node::DISPATCH: DispatchStrategy to tell the codegen + lint layers how its callbacks need to be delivered:

#![allow(unused)]
fn main() {
// File: packages/core/nros-platform/src/board/dispatch.rs
#[repr(u8)]
#[derive(Copy, Clone, Debug, PartialEq, Eq, Hash)]
pub enum DispatchStrategy {
    Inline = 0,
    Deferred = 1,
    FromIsr = 2,
}
}

The variants:

  • Inline — Callbacks fire from the executor’s spin loop, in the same task that drives transport I/O. Default for every Node (preserves every pre-Phase-216 Node pkg unchanged). Served by every runtime: POSIX, FreeRTOS, NuttX, Zephyr, ThreadX, bare-metal, RTIC (proxied via __nros_dispatch task when needed), Embassy (likewise — though Inline is unusual under cooperative async; see the matrix below).

  • Deferred — Callbacks land in a board-side queue (heapless::spsc::Queue on RTIC, embassy_sync::channel::Channel on Embassy). A framework-owned dispatch task drains the queue and drives ExecutableNode::on_callback from its own task context, decoupling callback latency from network polling. Required for callback-driven Nodes on RTIC + Embassy.

  • FromIsr — Callbacks fire directly from an ISR handler. Design slot only — implementation deferred to Phase 216.E.1. Reserved so the lint matrix has a stable discriminant to reject against; current builds error out at the nros check layer.

Framework × Strategy matrix

The matrix nros check (Phase 216.D.1) enforces:

              ┌──────────────────────────────────────────────────────┐
              │            DispatchStrategy                          │
              │  Inline      │  Deferred       │  FromIsr            │
┌─────────────┼──────────────┼─────────────────┼─────────────────────┤
│ posix       │  OK          │  OK             │  ERR: no ISRs       │
│ rtos        │  OK          │  OK             │  ERR: no ISRs       │
│ rtic        │  WARN: pref. │  OK (canonical) │  FUTURE (216.E.1)   │
│             │  Deferred    │                 │                     │
│ embassy     │  WARN: pref. │  OK (canonical) │  ERR: no ISR exec   │
│             │  Deferred    │                 │                     │
└─────────────┴──────────────┴─────────────────┴─────────────────────┘
  • POSIX + RTOS: both Inline and Deferred work. Inline is the default because the executor’s spin loop is the natural place to dispatch callbacks on hosted targets.
  • RTIC + Embassy: Deferred is canonical. Inline is permitted (the framework adapter still serves it) but nros check warns: callback-driven Nodes that run inline tie their handler latency to spin task scheduling, which is rarely what you want under hardware-interrupt or async-cooperative scheduling.
  • FromIsr on POSIX/RTOS: rejected — there’s no meaningful “ISR context” for nano-ros to dispatch from on a hosted OS.
  • FromIsr on RTIC: future — needs the reentrancy audit + SPSC rework + per-Node #[isr_safe] contract called out in Phase 216.E.1.
  • FromIsr on Embassy: rejected — Embassy has no concept of “callback fires from an ISR handler”; ISR-driven work hands off to an async task via embassy_sync::signal::Signal or similar.

Why per-Node, not per-callback

A Node that wants its /chatter subscription to run inline but its /heartbeat subscription to run deferred is conceptually expressible — we could put DispatchStrategy on each create_subscription call. We don’t, for two reasons:

  1. The lint matrix collapses. Per-callback strategies multiply the (framework, strategy) matrix by the number of callbacks per Node. The error surface grows quadratically and the messages get harder to phrase. Per-Node keeps each Node either fully Inline or fully Deferred — one strategy to reason about per pkg.
  2. Implementation simplicity. The dispatch task drains a single queue and routes each entry by CallbackId to a single Node’s on_callback. Per-callback would mean tagging each entry with its own strategy at enqueue time and forking the dispatch path. Adds weight for a feature no real user has asked for yet.

If a real user demonstrates a Node that genuinely wants mixed strategies — typically because one subscription handler must hold a high-priority lock while another doesn’t — Phase 216.E.3 is the slot to reconsider. Until then: YAGNI.

Why FromIsr is a design slot, not an impl

The FromIsr discriminant exists in the enum today so that:

  • The [repr(u8)] discriminants are stable. Inline = 0, Deferred = 1, FromIsr = 2 are wire-frozen — the __nros_node_<pkg>_dispatch_strategy() ABI symbol returns a u8 that nros check reads without linking the Node crate. Adding FromIsr later (when 216.E.1 lands) would either renumber existing discriminants (breaking already-compiled Node binaries) or require a new ABI symbol.
  • The lint matrix has somewhere to point. Without the variant in the enum, the nros check matrix would have to special-case “user wrote FromIsr but it doesn’t exist yet” via spelling-comparison rather than enum match. Worse error messages, worse evolvability.

The actual implementation needs three pieces that aren’t there yet:

  • Reentrancy audit. Every step of the dispatch path — signal_callback, queue push, RMW raw-CDR buffer ownership — must be re-entrant against ISR-priority producers. Today’s path assumes thread-context callers.
  • Lock-free SPSC variant. heapless::spsc::Queue is single- producer / single-consumer at thread priority. ISR-priority producer + thread-priority consumer needs a stronger ordering contract (memory barrier on the ISR-side push at minimum, possibly a different queue type).
  • Per-Node #[isr_safe] proof contract. on_callback must not call anything that can block, panic, or allocate. Statically proving that for arbitrary user code is a documentation + attribute exercise we haven’t undertaken.

The substrate Phase 214.J built (atomic_waker for cross-task notification) is the building block for the SPSC variant; landing 214.J was a precondition for being able to even prototype FromIsr. The full impl is deferred until a real ISR-driven driver demands it.

Tag-based callback API rationale

The closure-based registration API:

#![allow(unused)]
fn main() {
// Pre-216.A.4 — still valid for Inline Nodes.
ctx.create_subscription("/chatter", |msg: Int32| {
    handle(msg);
});
}

works fine when the callback runs synchronously on the spin task: the closure captures state by value (or by &mut) and gets called inline during spin_once. The captured environment lives on the spin task’s stack frame, no heap, no boxing.

Deferred dispatch breaks that assumption. The callback now fires from a different task than the one that called register, which means the closure environment must either:

  • Outlive both tasks (require 'static capture — move || everywhere, but state still needs somewhere to live), or
  • Be stored generically in the runtime, which means erasing the closure type to Box<dyn FnMut(...)> (alloc-dependent) or to extern "C" fn (no captures at all).

Both options conflict with the no-alloc + framework-task-routed contract. So Phase 216.A.4 introduces tags:

#![allow(unused)]
fn main() {
// File: packages/core/nros/src/dispatch_tag.rs
pub struct SubscriptionTag(&'static str);
pub struct ServiceTag(&'static str);
pub struct ActionTag(&'static str);

impl From<SubscriptionTag> for CallbackId<'static> { /* ... */ }
impl PartialEq<Callback<'_>> for SubscriptionTag { /* ... */ }
}

The tag carries only the &'static str callback identifier — zero runtime cost, no captures, FFI-safe. State lives on ExecutableNode::State; the macro-emitted init() body wires the tag fields by calling the _static registration variants (create_subscription_static / create_service_static / create_action_static) and storing the returned tag onto Self::State.

Dispatch then matches the tag against the Callback<'_> event:

#![allow(unused)]
fn main() {
fn on_callback(state: &mut Self::State, cb: Callback<'_>, ctx: &mut CallbackCtx<'_>) {
    if state.sub_chatter == cb {
        let msg: Int32 = ctx.downcast().unwrap();
        // ...
    } else if state.sub_heartbeat == cb {
        let beat: Heartbeat = ctx.downcast().unwrap();
        // ...
    }
}
}

The PartialEq<Callback<'_>> impl on each tag type does the comparison in &'static str terms — O(ptr_eq) in the common case when the runtime hands back the same &'static the user registered.

Inline keeps closures. The closure-vs-tag split is enforced by the macro lint (216.A.6): a Deferred Node using create_subscription (the closure form) fails to compile with a clear error pointing at the registration call and suggesting the _static form. An Inline Node using _static is allowed — but Deferred → closure is rejected. Zero migration cost for pre-216 Node pkgs (all defaulted to Inline) and a forced-correct API for the Deferred path.

The __nros_node_<pkg>_dispatch_strategy() ABI symbol

The nros::node!() macro emits, per Node pkg:

#![allow(unused)]
fn main() {
// Emitted by nros::node!(Talker) — Phase 216.A.5.
#[unsafe(no_mangle)]
extern "C" fn __nros_node_talker_pkg_dispatch_strategy() -> u8 {
    <Talker as ::nros::Node>::DISPATCH as u8
}
}

Three consumers care about this symbol:

  1. nros check (Phase 216.D.1). Statically inspects the Node crate’s .rmeta or links the staticlib + reads the symbol via dlsym/GetProcAddress (host-side check; embedded targets only read it from .rmeta). Compares against the Entry pkg’s board framework metadata using the matrix above; rejects mismatches at nros check time, before the user runs cargo build.

  2. The nros::main!() proc-macro (Phase 216.B.3 / C.3). When expanding the Entry pkg’s main.rs it walks the registered Node list and reads each pkg’s strategy. If any Node is Deferred, the generated #[rtic::app] / #[embassy_executor::main] body includes the __nros_dispatch task; if all are Inline, the dispatch task is omitted (zero overhead for Inline-only workspaces).

  3. Future runtime diagnostic tools. A nros doctor or nros topology style introspection tool can read the symbols from a linked binary and print the (pkg, strategy) table without having to re-parse the source. Useful for post-mortem on a binary you didn’t build yourself.

The extern "C" + [repr(u8)] ABI is the contract. It must not break across nano-ros versions — adding a new strategy variant means adding a new discriminant (FromIsr = 2 was reserved up front exactly to avoid this), never renumbering an existing one.

The trait surface split (post-214.K.1)

Phase 214.K.1 renamed the board-side dispatch sink from NodeRuntime to NodeDispatchRuntime (the user-facing sink kept the NodeRuntime name, which is now in packages/core/nros/src/node.rs). Phase 216 lands its new methods on NodeDispatchRuntime:

#![allow(unused)]
fn main() {
// File: packages/core/nros-platform/src/board/runtime.rs
pub trait NodeDispatchRuntime {
    // ... existing methods unchanged ...

    fn signal_callback(&mut self, _cb_id: CallbackId<'_>, _ctx: &mut CallbackCtx<'_>) {
        panic!("signal_callback not implemented for Inline runtime");
    }
    fn dispatch_strategy(&self) -> DispatchStrategy {
        DispatchStrategy::Inline
    }
}
}

Both methods are defaulted — zero-touch for the existing Inline impls (ExecutorNodeRuntime in nros, NullNodeRuntime in nros-platform). A Deferred runtime overrides both:

#![allow(unused)]
fn main() {
// File: packages/boards/nros-board-rtic-stm32f4/src/runtime.rs (sketch)
impl NodeDispatchRuntime for RticRuntime {
    fn dispatch_strategy(&self) -> DispatchStrategy {
        DispatchStrategy::Deferred
    }
    fn signal_callback(&mut self, cb_id: CallbackId<'_>, ctx: &mut CallbackCtx<'_>) {
        // SAFETY: SPSC producer is single-threaded by RTIC priority assignment.
        self.queue_producer.enqueue((cb_id, ctx.snapshot())).unwrap();
        // Wake the dispatch task; RTIC will schedule it at its declared priority.
        __nros_dispatch::spawn().ok();
    }
}
}

signal_callback’s default panic is the right behavior: the Inline path never calls it (callbacks flow through the existing inline trampoline). If a Deferred Node ends up on an Inline runtime — for example because the user manually picked the wrong board — the panic is louder than silently dropping the callback. The lint at Phase 216.D.1 prevents this combination from compiling in the first place; the panic is a belt-and-braces backstop.

Backward compatibility

Two contracts:

  1. Defaulted associated const. Node::DISPATCH is const DISPATCH: DispatchStrategy = DispatchStrategy::Inline; in the trait definition. Edition 2024 supports defaulted associated consts as stable, so every pre-216 impl Node for ... block that doesn’t mention DISPATCH continues to compile and is treated as Inline.
  2. Closure API preserved on the Inline path. The Inline runtime keeps the closure-based registration path (create_subscription, create_service, create_action). The macro lint only rejects closure use when DISPATCH = Deferred. Every Phase 212 Node pkg using closures stays valid without changes.

The migration shape for a Phase 212 Node that wants to move to Deferred is:

#![allow(unused)]
fn main() {
// Before — Phase 212 Inline-by-default.
impl Node for Listener {
    const NAME: &'static str = "listener";
    fn register(ctx: &mut NodeContext<'_>) -> NodeResult<()> {
        ctx.create_subscription::<Int32>("/chatter", |msg| {
            defmt::info!("Received: {}", msg.data);
        })?;
        Ok(())
    }
}

// After — Phase 216 Deferred.
impl Node for Listener {
    const NAME: &'static str = "listener";
    const DISPATCH: DispatchStrategy = DispatchStrategy::Deferred;
    fn register(ctx: &mut NodeContext<'_>) -> NodeResult<()> {
        let _tag = ctx.create_subscription_static::<Int32>("/chatter")?;
        Ok(())
    }
}

impl ExecutableNode for Listener {
    type State = ListenerState;
    fn init() -> Self::State {
        ListenerState { sub_chatter: SubscriptionTag::placeholder() }
    }
    fn on_callback(state: &mut Self::State, cb: Callback<'_>, ctx: &mut CallbackCtx<'_>) {
        if state.sub_chatter == cb {
            let msg: Int32 = ctx.downcast().unwrap();
            defmt::info!("Received: {}", msg.data);
        }
    }
}
}

The migration adds three things: the DISPATCH const, the tag-typed State field, and the on_callback body. The closure body in the “before” version becomes the if state.sub_chatter == cb { ... } branch in the “after” version — same code, hoisted to a method.

See also

  • RTIC Integration — user-facing tutorial for the RTIC side of dispatch.
  • Embassy Integration — user-facing tutorial for the Embassy side of dispatch + spawn-from-sync.
  • Scheduling Models — the real-time scheduling backdrop against which dispatch strategy choices are made.
  • docs/roadmap/phase-216-baremetal-framework-integration.md — the locked spec.
  • packages/core/nros-platform/src/board/dispatch.rs — the DispatchStrategy enum.
  • packages/core/nros/src/dispatch_tag.rs — the tag types.

Real-Time Analysis

This chapter describes static analysis methods and tools for detecting anti-patterns that violate real-time guarantees in Rust embedded applications.

Overview

Real-time systems require deterministic execution times. Common anti-patterns that break real-time guarantees include:

Anti-PatternProblemDetection Method
Unbounded loopsInfinite execution timeClippy + custom lints
RecursionStack overflow, unbounded depthcargo-call-stack
Heap allocationNon-deterministic timing, fragmentationno_std + forbid patterns
Blocking I/OUnbounded wait timesCustom lints
Missing timeoutsOperations can hang foreverCustom lints
Large stack framesStack overflowcargo-call-stack

Built-in Clippy Lints

Loop and Iteration Lints

# Enable all loop-related lints
cargo clippy -- \
    -D clippy::infinite_iter \
    -D clippy::while_immutable_condition \
    -D clippy::never_loop \
    -D clippy::empty_loop
LintDetects
infinite_iterIterator chains guaranteed to be infinite
while_immutable_conditionLoop conditions that can never change
never_loopLoops that exit on first iteration
empty_looploop { } without body (use loop { hint::spin_loop() })

Memory and Performance Lints

cargo clippy -- \
    -W clippy::large_stack_arrays \
    -W clippy::large_types_passed_by_value \
    -W clippy::box_collection \
    -W clippy::rc_buffer

Create clippy.toml in your project root:

# Maximum size for stack-allocated arrays (bytes)
array-size-threshold = 512

# Warn on types larger than this passed by value
trivial-copy-size-limit = 16

# Cognitive complexity threshold
cognitive-complexity-threshold = 15

Running Clippy for Real-Time Code

# Strict mode for real-time code
cargo clippy --all-targets -- \
    -D warnings \
    -D clippy::all \
    -W clippy::pedantic \
    -D clippy::infinite_iter \
    -D clippy::while_immutable_condition \
    -A clippy::module_name_repetitions

Stack Analysis with cargo-call-stack

Installation and Usage

# Install (requires nightly)
cargo +nightly install cargo-call-stack

# Build with stack size info
cd examples/stm32f4/rust/rtic
RUSTFLAGS="-Z emit-stack-sizes" cargo +nightly build --release

# Generate call graph with stack sizes
cargo +nightly call-stack --release > call_graph.dot

# Visualize (requires graphviz)
dot -Tsvg call_graph.dot -o call_graph.svg

Interpreting Results

The output shows:

  • Each function’s stack frame size
  • Call graph relationships
  • Maximum stack depth through any path
  • Cycles (recursion) in the call graph

Example output:

digraph {
    "main" [label="main\n256 bytes"]
    "zenoh_poll" [label="zenoh_poll\n128 bytes"]
    "publisher_task" [label="publisher_task\n512 bytes"]

    "main" -> "zenoh_poll"
    "main" -> "publisher_task"
}

Limitations

  • Requires fat LTO (lto = "fat" in Cargo.toml)
  • Limited support for programs linking std
  • Indirect calls (function pointers, trait objects) may not be analyzed
  • Best for embedded no_std programs

Preventing Heap Allocation

Method 1: no_std Without Allocator

The simplest approach – don’t provide a global allocator:

#![allow(unused)]
#![no_std]
#![no_main]

fn main() {
// No #[global_allocator] defined
// Attempting to use Box, Vec, String will fail to compile
}

Method 2: Compile-Time Enforcement

#![allow(unused)]
#![no_std]
#![forbid(unsafe_code)]  // Also prevents custom allocators

fn main() {
use heapless::{Vec, String};  // Static-sized alternatives

// This will NOT compile:
// let v = alloc::vec::Vec::new();  // Error: no allocator

// This works:
let v: heapless::Vec<u8, 256> = heapless::Vec::new();
}

Method 3: Custom Lint (Dylint)

For projects that need alloc but want to restrict usage in certain modules:

#![allow(unused)]
fn main() {
// In real-time critical code, add:
#![deny(clippy::disallowed_methods)]
}

With clippy.toml:

disallowed-methods = [
    { path = "alloc::vec::Vec::push", reason = "Use heapless::Vec in RT code" },
    { path = "alloc::boxed::Box::new", reason = "No heap in RT code" },
    { path = "alloc::string::String::new", reason = "Use heapless::String" },
]

Detecting Missing Timeouts

Pattern: I/O Operations Without Timeout

Anti-pattern:

#![allow(unused)]
fn main() {
// BAD: No timeout - can block forever
let data = socket.read(&mut buf)?;
}

Correct pattern:

#![allow(unused)]
fn main() {
// GOOD: Explicit timeout
socket.set_read_timeout(Some(Duration::from_millis(100)))?;
let data = socket.read(&mut buf)?;
}

Manual Audit Checklist

For code review, check that these operations have timeouts:

  • TcpStream::connect() – use connect_timeout()
  • socket.read() / socket.write() – set socket timeout
  • channel.recv() – use recv_timeout()
  • zenoh operations – configure session timeout
  • Any blocking syscall

Detecting Unbounded Loops

Clippy Detection

cargo clippy -- -D clippy::infinite_iter -D clippy::while_immutable_condition

Manual Patterns to Audit

Potentially unbounded:

#![allow(unused)]
fn main() {
// BAD: No termination guarantee
loop {
    if let Some(msg) = queue.pop() {
        process(msg);
    }
}

// BAD: Condition may never be false
while !flag.load(Ordering::Relaxed) {
    do_work();
}
}

Bounded alternatives:

#![allow(unused)]
fn main() {
// GOOD: Bounded iteration count
for _ in 0..MAX_ITERATIONS {
    if let Some(msg) = queue.pop() {
        process(msg);
    } else {
        break;
    }
}

// GOOD: Timeout-based termination
let deadline = Instant::now() + Duration::from_millis(10);
while Instant::now() < deadline {
    if let Some(msg) = queue.pop() {
        process(msg);
    } else {
        break;
    }
}
}

Detecting Recursion

cargo-call-stack Cycle Detection

# Will report cycles in call graph
cargo +nightly call-stack --release 2>&1 | grep -i "cycle"

Clippy Recursion Lint

# Warn on unconditional recursion
cargo clippy -- -D clippy::unconditional_recursion

Manual Pattern Detection

#![allow(unused)]
fn main() {
// BAD: Direct recursion
fn process(node: &Node) {
    process(&node.child);  // May overflow stack
}

// BAD: Mutual recursion
fn a() { b(); }
fn b() { a(); }

// GOOD: Iterative with explicit stack
fn process(root: &Node) {
    let mut stack: heapless::Vec<&Node, 32> = heapless::Vec::new();
    stack.push(root).ok();

    while let Some(node) = stack.pop() {
        // Process node
        if stack.push(&node.child).is_err() {
            // Handle stack full - bounded!
            break;
        }
    }
}
}

Advanced: Custom Dylint Lints

Setting Up Dylint

# Install dylint
cargo install cargo-dylint dylint-link

# Create a new lint library
cargo dylint new realtime_lints
cd realtime_lints
LintPurpose
blocking_in_asyncDetect std::thread::sleep, blocking I/O in async
unbounded_loop_in_taskFlag loops without bounds in RTIC tasks
heap_in_interruptDetect allocation in #[interrupt] handlers
mutex_in_interruptFlag std::sync::Mutex in interrupt context
float_in_isrWarn about FP operations in ISRs (soft-float targets)
missing_timeoutI/O operations without timeout

Tool Summary

ToolDetectsEffortCI-Ready
ClippyInfinite iters, recursion, large arraysLowYes
cargo-call-stackStack usage, recursion cyclesMediumYes
MiriUndefined behavior, memory errorsLowYes
DylintCustom patterns (blocking, heap, etc.)HighYes
MIRAIAbstract interpretation, all pathsHighExperimental
no_stdPrevents std heap allocationLowAutomatic
heaplessStatic collectionsLowN/A

Quick Reference: Lint Flags

# Copy-paste for strict real-time checking
cargo clippy -- \
    -D warnings \
    -D clippy::all \
    -D clippy::infinite_iter \
    -D clippy::while_immutable_condition \
    -D clippy::never_loop \
    -D clippy::empty_loop \
    -D clippy::unconditional_recursion \
    -W clippy::large_stack_arrays \
    -W clippy::large_types_passed_by_value \
    -W clippy::cognitive_complexity

WCET Baselines

Worst-Case Execution Time (WCET) measurements for nros core operations, collected via DWT cycle counting on Cortex-M3.

Measurement Platform

PropertyValue
TargetARM Cortex-M3 (thumbv7m-none-eabi)
MachineQEMU lm3s6965evb (for infrastructure validation)
ClockDWT CYCCNT (cycle-accurate on real hardware)
Iterations100 per benchmark (10 warmup)
Optimizationrelease profile (opt-level = "s")

DWT limitation on QEMU: QEMU’s Cortex-M3 emulation does not implement the DWT cycle counter – all reads return 0. The benchmark infrastructure is validated by the fact that it compiles, runs, and produces structured output. Actual cycle counts must be collected on real hardware (e.g., STM32F4 at 168 MHz, STM32F7 at 216 MHz).

Benchmark Categories

CDR Serialization:

FunctionNotes
serialize Int32Single i32 field
deserialize Int32Single i32 field
serialize TimeTwo fields (i32 + u32)
roundtrip Int32Serialize + deserialize
serialize w/ headerCDR encapsulation header + Int32

Node API:

FunctionNotes
Node::new()StandaloneNode creation
create_publisher()Register Int32 publisher
serialize_message()Node-level serialize to buffer

Safety E2E:

FunctionNotes
crc32 (64B)CRC-32/ISO-HDLC, 64-byte payload
crc32 (256B)CRC-32/ISO-HDLC, 256-byte payload
crc32 (1024B)CRC-32/ISO-HDLC, 1024-byte payload
validate()SafetyValidator sequence check
full pipeline (128B)Extract attachment + CRC + validate

Sanity Bound

All functions are expected to complete within 100,000 cycles for typical payloads on a Cortex-M3 at any clock rate. This is a conservative upper bound; actual WCET should be well below this:

  • CRC-32 is O(n) in payload size with a single table lookup per byte
  • SafetyValidator::validate() is O(1) – a few comparisons and an increment
  • The full pipeline combines CRC computation with attachment parsing and validation

Running the Benchmark

# Build all QEMU examples (includes rs-wcet-bench)
just qemu build

# Run the WCET benchmark
just qemu test-wcet

Static WCET Analysis

For certification (ISO 26262 ASIL C/D, DO-178C DAL A/B), dynamic measurement is insufficient – static WCET analysis is required. Candidate tools:

ToolTypeLicenseNotes
PlatinStatic (IPET)Open sourceAnalyzes LLVM IR + machine code
aiTStatic (abstract interpretation)Commercial (AbsInt)Industry standard for DO-178C / ISO 26262
SWEETStatic (flow analysis)AcademicResearch tool; good for Cortex-M
ChronosStatic (IPET)Open sourceAcademic; limited embedded target support

Recommended path: Use Platin for open-source WCET bounding on the LLVM IR produced by rustc. For certification evidence, aiT provides the strongest tool qualification story (pre-qualified per ISO 26262 Part 8).

References

Build System & Caching

Audience. Contributors working on the repo build/test matrix. End users never need this — they consume nano-ros with a single add_subdirectory(<repo-root>), no install step, no find_package. This handbook documents the other build world: the in-repo just orchestration that builds + tests every platform × RMW, and the caching that keeps it incremental and correct.

Canonical sources this page summarises: CLAUDE.md (“Build”, “Build tiers”, “Build parallelism”), Phase 176 (unified jobserver), Phase 181 (fixture SSOT

  • Ninja), Phase 177.9 (staleness probes).

TL;DR cheat-sheet

just setup                 # one-time SDK/toolchain install (tiered; see CLAUDE.md)
direnv allow               # once after clone — else zpico-sys/build.rs panics

just build                 # workspace + transports (fast inner loop)
just build-examples        # + every example
just build-all             # + test fixtures (everything test-all consumes)
just build-test-fixtures   # just the fixtures + writes the test-all stamp

just test-unit             # ~5s          ⊂
just test-integration      # ~30s         ⊂
just test                  #              ⊂
just test-all              # heavy QEMU/Zephyr/ROS-interop/miri/codegen

just <plat> build|build-all|test|ci   # narrow to one platform first
NROS_BUILD_JOBS=8 just build-all       # cap parallelism

Rule of thumb: a platform-specific failure → run the narrow just <plat> build-all (or closest build/build-examples/build-fixtures) before the root just build-all. Always just ci after a task. Never sudo — if a step needs it, tell the user.

Generators — Ninja by default

The repo’s CMake builds use Ninja (-G Ninja), falling back to Make only when ninja is absent. Ninja gives reliable incremental rebuilds and fits the Phase 176 fifo jobserver; Make was dropped from the staleness path because make -q mis-reported up-to-dateness.

  • Configure-once: scripts/build/cmake-incremental.sh::nros_cmake_configure_if_needed configures a build dir only when its CMakeCache.txt / generated build system is missing; cmake --build then handles reconfigure on CMakeLists changes.
  • Generator-mismatch wipe: a dir configured with the other generator is rm -rf’d and reconfigured (you can’t switch generators in place).
  • Pinned tools: the unified jobserver needs make ≥ 4.4 + ninja ≥ 1.13 (apt’s 4.3/1.10 lack the fifo jobserver). just workspace install-make / install-ninja build them into third-party/{make,ninja}; .envrc puts them on PATH (incl. a gmake → make-4.4 alias).

Per-RMW build dirs = cache isolation

Each example builds into a per-RMW directory with its own cargo target dir:

examples/<plat>/<lang>/<example>/
  build-zenoh/       cargo/   …   # -DNROS_RMW=zenoh
  build-xrce/        cargo/   …   # -DNROS_RMW=xrce
  build-cyclonedds/  cargo/   …   # -DNROS_RMW=cyclonedds

Because each RMW (and its Corrosion cargo target dir) is physically separate, selecting a different RMW cannot collide with another RMW’s cache — there is no shared target dir to invalidate. Platform is fixed per build dir (-DNANO_ROS_PLATFORM=<plat> at configure). This layout is why the old manual cache-invalidation idea (archived Phase 145) was retired: the directory shape makes it unnecessary. (Rust-only examples build via plain cargo into target-<rmw>/; the per-RMW principle is the same.)

Parallelism — NROS_BUILD_JOBS + the unified jobserver

One knob scales every parallel recipe:

NROS_BUILD_JOBS=8 just build-all     # default: nproc

Unified jobserver (Phase 176). just build-all auto-routes to build-all-jobserver when the pinned make 4.4 + ninja 1.13 are present: a single GNU-make fifo jobserver spans cargo + build-script cc + ninja-via-west

  • cmake, allocating tokens dynamically instead of a static per-tool split. Under the jobserver (NROS_JOBSERVER=1) recipes drop their explicit -j/--parallel/CMAKE_BUILD_PARALLEL_LEVEL so the tools inherit the pool.
  • Same artifacts either way; without the pinned tools it falls back to a static split.
  • NROS_NO_JOBSERVER=1 forces the static path.
  • Never re-introduce a hardcoded parallel --jobs <n> without threading ${NROS_BUILD_JOBS:-N} through.

See docs/roadmap/archived/phase-176-unified-jobserver-build-orchestration.md.

Build & test tiers

Each tier is a strict superset of the previous:

Buildcontains
buildworkspace + transports
build-examples⊃ build + every example
build-all⊃ build-examples + test fixtures
Testwall-clockcontains
test-unit~5 sunit
test-integration~30 s⊃ + integration
test
test-allminutes⊃ + heavy QEMU/Zephyr/ROS-interop + doc + miri + C codegen

Per-platform: just <plat> build ⊂ build-examples ⊂ build-fixtures ⊂ build-all and just <plat> test|test-all|ci. <plat> = target families (qemu, zephyr, board groups). Support services (zenohd, cyclonedds) are not platform scopes. Orchestration lives in justfile + just/*.just.

Fixture SSOT — examples/fixtures.toml

Per-fixture build options (features, --target-dir, env, per-RMW variants, cmake -D defs, cross target) live in one manifest, examples/fixtures.toml, consumed by both the build recipes and the test-all staleness probe — so the build and the probe use identical options (no feature-thrash).

  • Reader: scripts/build/fixtures-manifest.py list --platform <p> --lang <l> [--rmw <r>] [--for-probe]. Emits \x1f-separated records (unit-separator, not tab — tab is IFS-whitespace).
  • Builder: scripts/build/fixtures-build.sh <plat> [lang] [rmw] — one shared loop; rust cells go through cargo build, C/C++ cells through cmake --build (configure-once + per-RMW dir). Platform-wide -D defs (toolchain, codegen tool, SDK dirs) are injected by the recipe via NROS_CMAKE_EXTRA_DEFS.
  • Cross-toolchain platforms inject RUSTUP_TOOLCHAIN / SDK env via the recipe (e.g. ESP32-C3 = riscv32imc on the workspace nightly + build-std).

Staleness discipline

A prebuilt fixture that’s silently stale would let test-all run against the wrong binary. The discipline:

  • Presence gate. build-test-fixtures stamps target/nextest/.fixtures-built; test-all’s _require-fixtures fast-fails (~1 s) with a hint if it’s absent (bypass: NROS_SKIP_FIXTURE_CHECK=1).
  • Rust cells. scripts/test/rust-fixture-stale.sh reuses cargo’s own fingerprint — cargo build … --message-format=json reports "fresh":false for a stale unit, so the probe both detects and self-heals.
  • C/C++ cells. scripts/test/cmake-fixture-stale.sh runs the incremental cmake --build (near-no-op when fresh) and flags cells that actually rebuilt. (ninja -n is unusable here — Corrosion’s cargo step is an always-run custom command, so it always reports pending.)
  • Probe opt-out. Cells needing a recipe-injected toolchain the probe can’t supply (e.g. the ESP32 nightly + build-std) set skip_probe = true; the reader’s --for-probe omits them so they don’t toolchain-thrash.
  • Source-list / drift gates. zpico-sys (vendored zenoh-pico, Phase 136.6) and nros-rmw-xrce-cffi (vendored uxr/micro-cdr, Phase 145.4) verify each vendored source root resolves to a real dir with .c files and panic with a git submodule update --init hint on drift. cbindgen build scripts (nros-c, nros-cpp, zpico-sys) emit cargo:rerun-if-changed=cbindgen.toml
    • src/.
  • Probe runs in preflight. _check-fixtures-stale runs both probes (rust + cmake) over the manifest before the test-all nextest stage; it warns + self-heals rather than hard-failing.

Patched QEMU

QEMU networked tests use a patched qemu-system-arm (icount + MPS2 fixes) built by just qemu setup-qemu into build/qemu/bin/. The harness picks it up automatically (nros_tests::qemu::qemu_system_arm_cmd()); the system binary is the fallback. See Patched qemu-system-arm.

See also

  • Build Commands — the user-facing quick reference.
  • zpico-sys Build Architecture — the zenoh-pico cross-compile.
  • CLAUDE.md — the authoritative build/test-tier + parallelism policy.
  • Phase docs (docs/roadmap/archived/): 176 (jobserver), 181 (fixture SSOT + Ninja), 177.9 (staleness probes).

CLI lives in the monorepo (Phase 218)

The nros CLI lives in-tree at packages/cli/ as a sub-workspace — own Cargo.toml, own Cargo.lock, own member crates (nros-cli, nros-cli-core, nros-build, cargo-nano-ros, rosidl-*, colcon-cargo-ros2, nros-msg-to-idl). It is built per checkout via just setup-clipackages/cli/target/release/nros, then put on PATH by the activate file. See also:

Why sub-workspace, not root member

The runtime workspace’s no_std feature-unification view (Phase 214.F.3) is the project’s most fragile invariant — a single workspace member adding a std-activating dep without target-gating turns the embedded build red on a path the test matrix doesn’t always exercise. The CLI’s deps (clap, askama, syn, ureq, …) are aggressively std-only; making them workspace siblings of the runtime crates would either (a) require target-gating every single one or (b) effectively disable the F.3 guard.

The sub-workspace shape keeps the two surfaces categorically separate. The pattern mirrors packages/testing/nros-{tests,bench,smoke}/, which carved out the same way for the same reason.

Why per-checkout, not ~/.nros/bin

Confirmed during the Phase 218 UX walkthrough: contributors with multiple nano-ros worktrees (phase branches, ASI integration trees, downstream forks) need each tree to point at its own CLI. A global install would silently version-skew across trees the moment the user cds — and because the CLI’s codegen format is structurally bound to the nros-core / nros-c / nros-cpp crates in the same checkout, a global CLI is a known footgun.

The per-checkout shape makes the activate file the single switch — no which nros surprise across cwd boundaries. One checkout = one CLI version = one runtime ABI, by construction; no version-pin matrix to police.

Resolution chain

scripts/build/cargo.sh::nros_cli_bin resolves the active binary:

  1. $NROS_CLI (explicit override)
  2. nros on $PATH (activate file puts the repo-local one here)
  3. packages/cli/target/release/nros (per-checkout)
  4. ${NROS_HOME:-~/.nros}/bin/nros (transitional, removed once every active branch lands on 218)
  5. error

The Phase 218.E ABI guard checks the in-tree CLI build matches the runtime crates’ codegen ABI hash; opt out via NROS_SKIP_VERSION_CHECK=1 when intentionally bisecting a CLI mismatch.

See also

zpico-sys Build Architecture

zpico-sys is nano-ros’s Rust *-sys crate for zenoh-pico. After (2026-05-18) the build path is a single cc-rs invocation driven entirely by a TOML manifest. Earlier phases shipped a CMake/cc-rs hybrid with per-RTOS Rust functions; the unified path collapses that surface to one consumer + one data file.

The manifest

packages/zpico/zpico-sys/zenoh_platforms.toml declares every per-platform datum the build needs. Two top-level table groups:

  • [platform.<name>] — one block per supported platform (posix, zephyr, freertos-lwip, nuttx, threadx, bare-metal, generic, orin-spe).
  • [arch.<name>] — reusable target-arch compiler-flag profiles shared across platforms that target the same CPU family (cortex-m3, cortex-m4f, cortex-a7, cortex-r5-softfp, riscv32imc, riscv32gc, riscv64gc).

Per-platform fields

FieldTypeUse
inheritsstringMerge from another [platform.*] block before applying this one.
defineslist[str]Unconditional cc::Build::define(name, None).
defines_kvmapcc::Build::define(name, Some(value)).
defines_envmapValue from env, falls back to default.
includelist[str]Glob roots under zenoh-pico/src/. The drift gate validates these exist.
excludelist[str]Drop entries from include matches.
extra_sourceslist[ExtraSource]Additional .c files outside zenoh-pico/src/. See below.
required_envlist[RequiredEnv]SDK paths the build needs. See below.
include_pathslist[str]Header search paths; interpolated.
include_paths_conditionallist[ConditionalPath]Header paths gated by when.
archstring | list[str]Name(s) of [arch.*] block(s) to apply. Single name (TOML scalar) for single-arch platforms; list (TOML array) for multi-arch platforms like bare-metal (cortex-m3 + riscv32imc). build.rs walks the list in order and applies the first arch whose target_match hits the build target..
compiletableopt_level / warnings / cflags.
picboolcc::Build::pic override (NuttX flat builds use false).
link.*mapPer-link-feature policy. Values: true / false (force on/off) or "feature" (defer to CARGO_FEATURE_LINK_<X>).
mbedtlsstringpkg-config / vendored / none.
system_libslist[str]cargo:rustc-link-lib= entries.
rerun_if_env_changedlist[str]cargo:rerun-if-env-changed=… triggers.

Per-arch fields

FieldTypeUse
target_matchstringSubstring or <prefix>* glob the target triple must match.
target_excludestringVeto when the target triple contains this substring (e.g. cortex-m3 excludes thumbv7em so Cortex-M4 doesn’t pick it).
cflagslist[str]Compiler flags.
needs_picolibcboolAdd picolibc sysroot’s include/ to the search path.
needs_errno_overrideboolGenerate + prepend the errno-override shadow header (RISC-V picolibc TLS-errno workaround).
needs_riscv_compilerboolProbe for a RISC-V cross-cc.

Interpolation tokens

Every path / include_paths / extra_sources.path / include_paths_conditional.path string is run through a small interpolator before being passed to cc-rs:

TokenValue
{nros}CARGO_MANIFEST_DIR (zpico-sys/).
{out}OUT_DIR.
{src}zenoh-pico/src/ under {nros}.
{env:VAR}Value of env var VAR; build fails with a clear error if unset.

when matcher

Conditional includes carry a when table. Each field is optional; present fields ALL must hold for the matcher to fire.

FieldMeaning
target_matchSubstring (or <prefix>* glob) that must be in the target triple.
target_notSubstring that must NOT be in the target triple. Special value "embedded" matches any of the known embedded RTOSes.
if_envEnv var that must be set (any value).

ExtraSource shape

extra_sources = [
  { path = "{nros}/c/platform/threadx/task.c" },
  { path = "{nros}/c/platform/threadx/log_uart.c",
    if_env = "NROS_ZPICO_LOG_TO_UART",
    with_define = ["ZENOH_LOG_PRINT", "zpico_log_print"] },
]

if_env skips the file when the env var is absent. with_define adds the matching cc::Build::define(name, Some(value)) whenever the source is included (use name alone for unconditional None).

RequiredEnv shape

required_env = [
  { name = "THREADX_DIR",
    help = "ThreadX kernel source. Run just setup threadx_linux or export THREADX_DIR=$PWD/third-party/threadx/kernel",
    validate_subdir = "common/inc" },
]

Missing env vars panic at build time with the help string; present env vars whose value lacks the validate_subdir subdir also panic with the offending path printed. No silent fallthroughs.

Adding a new platform

Adding a new RTOS is a TOML edit:

  1. Pick an [arch.<name>] block for the target CPU. If the family isn’t already there (or compiler flags differ), add an [arch.*] block. Reusable across platforms.
  2. Write the [platform.<your_rtos>] block with defines + include + extra_sources + required_env + include_paths.
  3. The Cargo feature for the new platform must be added in Cargo.toml and wired into the existing match in build.rs that maps Cargo feature → manifest platform name.
  4. Verify with cargo build -p zpico-sys --features <your_rtos> --target <triple>.

No build.rs edits needed for the data layer.

Adding a new target arch

Adding a new CPU family is one [arch.*] block + any of the needs_picolibc / needs_errno_override / needs_riscv_compiler flags it requires. Existing platforms that need to target it set arch = "<your_arch>" (or extend their existing arch = [...] list with the new name).

For platforms that span multiple architectures (e.g. bare-metal covers cortex-m3 for qemu-arm-baremetal / stm32f4 AND riscv32imc for ESP32-C3), declare every arch the platform supports and let build.rs’s first-match dispatch pick the right one per target triple:

[platform.bare-metal]
arch = ["cortex-m3", "riscv32imc"]   # first arch_matches wins

This is what makes cargo check work on both qemu-arm-baremetal/rust/zenoh/talker and esp32/rust/zenoh/{listener,talker} from the same platform entry — the picolibc sysroot wired up by arch.riscv32imc.needs_picolibc = true is added to the cc-rs -I list only when the build target is riscv32imc-*.

mbedTLS source policy

Manifest’s mbedtls field per-platform:

  • pkg-config — discover via the pkg-config build crate. POSIX hosts only. The build script also synthesizes .pc files when the host doesn’t ship them (Ubuntu’s libmbedtls-dev is the motivating case).
  • vendored — pull from the in-tree mbedtls/ submodule sources. Bare-metal / RTOS targets without a system mbedTLS.
  • nonelink-tls users on this platform get a link error; the platform doesn’t support TLS.

The selection only matters when link-tls is on (controlled by the CARGO_FEATURE_LINK_TLS env var).

Source-drift gate

Every include root in zenoh_platforms.toml must (a) resolve to a real directory under zenoh-pico/src/, (b) contain ≥1 .c file or sub-directory. A failed check panics build-time with the offending key + expected path. owns this; full set-equality vs. the resolved cc-rs source list lands as a follow-up.

Manifest-driven consumer

build.rs’s build_zenoh_pico_unified consumes a ResolvedPlatform (the merged inherits chain). It:

  1. Validates required_env + validate_subdir.
  2. Generates the version header in {out}/zenoh-pico-version/.
  3. Applies the [arch.*] profile if its target_match / target_exclude predicates pass.
  4. Adds core sources (8 protocol subdirs + system/common) and per-platform extra_sources (with if_env / with_define).
  5. Sets include paths (unconditional + conditional after matches).
  6. Adds defines (defines / defines_kv / defines_env).
  7. Handles mbedTLS per the manifest’s mbedtls field.
  8. Applies shim slot counts.
  9. Applies compile settings + pic override.
  10. Compiles to libzenohpico.a.
  11. Registers rerun_if_env_changed.

The consumer is the only function that actually invokes cc-rs for zenoh-pico. Every per-platform delta lives in the TOML.

  • docs/roadmap/phase-136-zpico-sys-unified-build.md — the phase doc that drove this refactor.
  • book/src/concepts/platform-model.md — Boards vs Platforms; manifest is the platform-side knob.
  • book/src/internals/rmw-backends.md — RMW host-language policy.

Formal Verification

nano-ros uses three complementary verification tools to ensure correctness of its core libraries:

  • Kani – bounded model checking with 160 harnesses across 6 crates. Checks memory safety, integer overflow, and panic-freedom within bounded inputs.
  • Verus – deductive (unbounded) proofs with 102 verified properties. Proves functional correctness of scheduling, serialization, time arithmetic, and safety protocols.
  • Miri – runtime undefined behavior detection for unsafe code.

All three run via just recipes:

just verify          # Run both Kani + Verus
just verify-kani     # Kani only (~3 min)
just verify-verus    # Verus only (~1 sec)
just test-miri       # Miri UB detection

Kani

Kani is a bounded model checker that translates Rust code to a verification IR and exhaustively explores all possible executions up to a given bound.

just verify-kani

160 harnesses are spread across 6 crates:

CrateFocus
nros-serdesCDR serialization round-trip, buffer bounds
nros-coreTime arithmetic, type invariants
nros-paramsParameter value constraints, range checking
nros-cFFI boundary safety
nros-ghost-typesBuffer state machine transitions (overflow/lock)
nros-nodeExecutor scheduling, subscription handling

Kani checks three properties by default:

  1. Memory safety – no out-of-bounds access, no use-after-free
  2. Overflow freedom – no integer overflow on arithmetic operations
  3. Panic freedom – no reachable panic!, unwrap(), or assert! failure

A typical run takes approximately 3 minutes on a modern machine.

Miri

Miri is an interpreter for Rust’s Mid-level IR that detects undefined behavior at runtime:

just test-miri

Miri catches issues that neither the compiler nor Kani can detect:

  • Invalid pointer dereferences
  • Uninitialized memory reads
  • Violations of aliasing rules (Stacked Borrows / Tree Borrows)
  • Data races in concurrent code

If Miri fails with “contains outdated or invalid JSON”, clean the cache:

rm -rf target/miri

Verus

Verus is a deductive verification tool for Rust. Unlike Kani (which checks bounded executions), Verus proves properties hold for all possible inputs using SMT solving.

Quick Reference

just verification verus   # Download Verus binary to tools/ (or `just verification setup` for kani + verus)
just verify-verus    # Run all Verus proofs
just verify          # Run both Kani + Verus

Verification crate: packages/verification/nros-verification/

Type Specifications

Every type used inside verus! { } that is defined outside the macro needs an external_type_specification. How you write it determines whether Verus treats the type as transparent or opaque.

Transparent enums (variant matching allowed)

Use external_type_specification without external_body:

#![allow(unused)]
fn main() {
use nros_node::TriggerCondition;

verus! {

#[verifier::external_type_specification]
pub struct ExTriggerCondition(TriggerCondition);

// Now we can match on variants in spec functions:
pub open spec fn trigger_eval_spec(cond: TriggerCondition, ready: Seq<bool>) -> bool {
    match cond {
        TriggerCondition::Any => ...,
        TriggerCondition::All => ...,
        TriggerCondition::Always => true,
        TriggerCondition::One(index) => ...,
    }
}

} // verus!
}

This works for enums whose variants are visible (public). Verus sees the full variant structure and allows pattern matching in specs and proofs.

Opaque types (no internal access)

Use external_type_specification with external_body:

#![allow(unused)]
fn main() {
#[verifier::external_type_specification]
#[verifier::external_body]
pub struct ExSomeOpaqueType(SomeOpaqueType);
}

Use this for types with private fields, complex internals, or when you only need to pass them around without inspecting their structure. You cannot match on variants or access fields.

Structs with public fields

Use external_type_specification without external_body – fields become accessible in specs:

#![allow(unused)]
fn main() {
#[verifier::external_type_specification]
pub struct ExDuration(nros_core::time::Duration);

// Fields are accessible:
pub assume_specification[ Duration::to_nanos ](self_: &Duration) -> (n: i64)
    ensures n == (self_.sec as i64) * 1_000_000_000i64 + (self_.nanosec as i64);
}

Linking Specs to Production Code

assume_specification

Axiomatically declares a contract on a production function. The contract is trusted (not verified by Verus) – a human auditor must confirm the spec matches the implementation.

#![allow(unused)]
fn main() {
pub assume_specification[ TriggerCondition::evaluate ](
    self_: &TriggerCondition,  // &self becomes a named parameter
    ready: &[bool],
) -> (ret: bool)
    ensures
        ret == trigger_eval_spec(*self_, ready@);
}

Rules:

  • &self must be written as self_: &Type (named parameter, not method syntax)
  • ready@ converts &[bool] to Seq<bool> (Verus view conversion)
  • *self_ dereferences &TriggerCondition to TriggerCondition for spec matching
  • The return value must be named: -> (ret: bool)

Ghost models

For types that can’t be imported (private fields, feature-gated behind C FFI, etc.), create a ghost model – a manually maintained mirror:

#![allow(unused)]
fn main() {
/// Ghost representation of TimerState (mirrors nros_node::timer::TimerState).
pub struct TimerGhost {
    pub period_ms: u64,
    pub elapsed_ms: u64,
    pub mode: TimerModeGhost,
    pub canceled: bool,
}
}

Ghost models have weaker guarantees than assume_specification. Correctness relies on manual comparison with the production source code.

Trust Levels

Every proof falls into one of three trust levels:

LevelMechanismWhat’s trustedStrength
Formally linkedassume_specification + external_type_specificationThe spec matches the impl (human audit of ~4 lines)Strongest
Ghost modelManual struct/enum mirrorLine-by-line correspondence with production sourceMedium
Pure mathArithmetic identitiesOnly the math itselfWeakest (no code link)

Document the trust level of each proof in the module-level doc comment.

Pitfalls

verify = true on production crates

Never add [package.metadata.verus] verify = true to a production crate that contains function pointers, dyn Trait, or closures. Verus will attempt THIR erasure on all items in the crate and panic at erase.rs:237.

Only the verification crate (nros-verification) should have verify = true. Production crates are used as regular dependencies – Verus compiles them with standard rustc without attempting verification.

vstd dependency

vstd is published on crates.io (vstd = "0.0.0-2026-02-08-0120"). Use the registry version, not a path dependency. The pre-built Verus release does not include the full source tree needed for path dependencies (missing dependencies/prettyplease etc.).

Items outside verus! are external

Any type, function, or trait defined outside the verus! { } macro is treated as external by Verus. To use it in specs, you must register it with external_type_specification or reference it via assume_specification.

Closures and iterators

Verus cannot verify code containing closures (including .iter().any(|&r| r)) or function pointers. Mark such items with #[verifier::external] if they’re in a crate being verified, or keep them in production code that Verus never touches.

Edition 2024

Verus’s bundled rustc (1.93.0) supports edition 2024. The verification crate uses edition = "2024" to match the nros workspace. No special configuration is needed.

Adding a New Proof

  1. Identify the production function and its source location
  2. Determine the trust level:
    • Can you import the type? Use external_type_specification
    • Is the type behind a feature gate with C FFI? Use a ghost model
    • Is it pure arithmetic? Pure math proof
  3. Write the spec function inside verus! { }
  4. If formally linking: add assume_specification and document which source lines the auditor should compare
  5. Write the proof function with ensures clauses
  6. Run just verify-verus to check
  7. Update the module doc comment with the new proof’s trust level

File Organization

packages/verification/nros-verification/src/
├── lib.rs            # Module declarations
├── scheduling.rs     # Timer + trigger + executor proofs
├── communication.rs  # CDR serialization safety proofs
├── cdr.rs            # CDR round-trip integrity proofs
├── time.rs           # Duration/Time arithmetic proofs
├── action.rs         # GoalStatus state machine proofs
├── params.rs         # ParameterValue + range proofs
└── e2e.rs            # End-to-end data path proofs

Proofs are organized by what they guarantee to the application developer, not by source crate. A single proof module may reference types from multiple production crates.

E2E Safety Protocol

The safety-e2e feature described in this document has been implemented. It is available in nros-rmw, nros-node, and the top-level nros crate.

Problem Statement

nano-ros treats zenoh as a trusted transport. Messages are serialized (CDR), transmitted over zenoh, and deserialized without any integrity verification. In safety-critical deployments (Autoware safety island), the transport channel must be treated as untrusted – the EN 50159 “black channel” principle.

Without E2E protection:

  • No CRC or checksum on message payloads (corruption undetected)
  • No subscriber-side sequence tracking (message loss undetected)
  • No duplicate detection (message repetition undetected)
  • No freshness validation (stale messages accepted silently)
  • No source authentication (masquerade undetected)

Existing Infrastructure

The nano-ros transport layer carries metadata that partially addresses these concerns, but historically only on the publisher side.

RMW Attachment (33 bytes)

Every published message includes a zenoh attachment with:

OffsetSizeFieldPurpose
0-78sequence_number (i64 LE)Monotonically increasing per publisher
8-158timestamp (i64 LE, nanos)Publication time
161VLE length (always 16)GID size prefix
17-3216rmw_gidRandom per-publisher identifier

This exists for rmw_zenoh_cpp interoperability. Without safety-e2e, the subscriber parses it into MessageInfo but does not validate sequence continuity, freshness, or source identity.

Message Data Flow

Publisher side:
  User msg -> CdrWriter::serialize() -> CDR payload (with 4-byte header)
  CDR payload -> ShimPublisher::publish_raw()
    -> Compute seq++, timestamp
    -> Serialize 33-byte RMW attachment
    -> zenoh publish(payload, attachment)

Subscriber side:
  zenoh callback -> Copy payload + attachment to static SubscriberBuffer
  User calls try_recv_raw() -> Copy from static buffer to user buffer
  User buffer -> CdrReader::deserialize() -> User msg
  (Attachment parsed into MessageInfo but NOT validated without safety-e2e)

Key Observations

  1. Sequence numbers exist but are only checked by subscribers when safety-e2e is enabled.
  2. Timestamps exist but freshness validation requires a clock source (deferred).
  3. Publisher GID exists but source authentication requires a registration mechanism.
  4. CRC is added by the safety-e2e feature. It is computed over the CDR payload and appended to the attachment.
  5. The attachment travels out-of-band from the CDR payload in zenoh. This is beneficial – the CRC in the attachment provides a diverse check (different data paths).

EN 50159 Threat Model

EN 50159 defines 7 threat classes for communication over untrusted channels:

ThreatEN 50159 DefenseIntegration Approach
CorruptionCRCCRC-32 covering CDR payload, stored in extended attachment
RepetitionSequence numberSubscriber tracks expected sequence, flags duplicates
DeletionSeq + timeoutSubscriber detects sequence gaps
InsertionSeq + authSequence validation rejects unexpected messages
ResequencingSequence numberSubscriber validates monotonic sequence
DelayTimestamp/timeoutSubscriber compares message timestamp to current time
MasqueradeAuthenticationSubscriber validates expected source GID

5 of 7 defenses only require subscriber-side validation of data that already exists in the attachment. Only CRC requires new publisher-side computation.

CRC Architecture

The safety-e2e feature extends the zenoh attachment from 33 to 37 bytes:

Existing attachment (33 bytes):
  [seq:8][timestamp:8][vle:1][gid:16]

Extended attachment (37 bytes):
  [seq:8][timestamp:8][vle:1][gid:16][crc32:4]

The CRC covers the CDR payload bytes (not the attachment itself).

Algorithm: CRC-32/ISO-HDLC

Standard Ethernet CRC (polynomial 0xEDB88320 reflected), used by both AUTOSAR E2E Profile 1/2 and EN 50159. The 1KB lookup table is const-generated at compile time. Deterministic execution time for WCET analysis.

Interoperability

  • rmw_zenoh_cpp reads exactly 33 bytes using VLE parsing – extra bytes are ignored
  • nano-ros without safety-e2e reads 33 bytes – succeeds normally
  • nano-ros with safety-e2e reads 37 bytes – extracts CRC from bytes 33-36

Subscriber-Side Validation

Sequence Tracking

The subscriber maintains an expected_seq counter (initialized to -1). On each received message:

  • First message: initializes expected_seq to message_seq + 1
  • Contiguous: message_seq == expected_seq – normal delivery
  • Duplicate: message_seq < expected_seq – flagged in IntegrityStatus
  • Gap: message_seq > expected_seq – gap count reported in IntegrityStatus

CRC Validation

If the attachment is longer than 33 bytes, the subscriber extracts the CRC-32 from bytes 33-36 and compares it against a locally computed CRC of the received payload. If the attachment is 33 bytes (legacy or ROS 2 publisher), crc_valid is None.

Feature Integration

The safety-e2e feature flag propagates through the crate hierarchy:

nros (top-level) -> nros-node -> nros-rmw
  safety-e2e        safety-e2e   safety-e2e

Changes by layer

nros-rmw (core changes):

  • safety module: CRC-32 function, IntegrityStatus type, SafetyValidator state tracker
  • ShimPublisher::publish_raw(): computes CRC and extends attachment to 37 bytes
  • SubscriberBuffer: attachment buffer sized to 37 bytes
  • ShimSubscriber: SafetyValidator field, try_recv_validated() method

nros-node (API surface):

  • ShimNodeSubscription: try_recv_safe() returning (M, IntegrityStatus)

Unchanged:

  • CDR serialization (nros-serdes) – payload format unchanged
  • Core types (nros-core) – no new traits needed
  • Zenoh backend (nros-rmw-zenoh) – attachment handling supports variable sizes
  • Existing try_recv() API – unchanged behavior

Memory Impact

ComponentWithout safety-e2eWith safety-e2e
CRC-32 lookup table0+1024 bytes (.rodata)
Subscriber attachment buffers (8x)8 x 33 = 264 bytes8 x 37 = 296 bytes (+32)
SafetyValidator per subscriber (8x)08 x ~24 = 192 bytes
Total+1248 bytes

ROS 2 Interoperability

SenderReceiverCRCSequenceResult
nano-ros (safety)nano-ros (safety)ValidatedTrackedFull E2E protection
nano-ros (safety)nano-ros (no safety)IgnoredNot trackedWorks, no protection
nano-ros (safety)ROS 2 (rmw_zenoh)IgnoredNot trackedWorks, no protection
nano-ros (no safety)nano-ros (safety)NoneTrackedPartial (seq only)
ROS 2 (rmw_zenoh)nano-ros (safety)NoneTrackedPartial (seq only)

No interoperability is broken. Safety degrades gracefully when one side doesn’t support it.

AUTOSAR E2E Comparison

FeatureAUTOSAR E2E P01AUTOSAR E2E P02nano-ros E2E
CRCCRC-8 (8-bit)CRC-8 (8-bit)CRC-32 (32-bit, stronger)
Counter4-bit (0-14)4-bit (0-14)64-bit (i64, no rollover in practice)
Data ID16-bit16-bit128-bit GID (stronger)
Alive counterYesYesYes (via sequence)
TimeoutConfigurableConfigurableDeferred (needs clock abstraction)
State machineE2E_SMSameSimpler (valid/invalid per message)

nano-ros E2E is stronger than AUTOSAR P01/P02 in CRC and counter width, but currently lacks the state machine and timeout features.

Future Extensions

  • Freshness validation: Requires a MonotonicClock trait and platform implementations. The existing timestamp field already carries the publication time.
  • Source authentication: The rmw_gid (16-byte random publisher ID) can be used for masquerade detection if the subscriber registers expected sources.
  • Watchdog supervision: Separate from E2E protocol – monitors alive heartbeats, deadline completion, and logical execution flow.
  • Formal verification: CRC correctness, sequence tracking invariants, and IntegrityStatus::is_valid() are candidates for Verus proofs.

Production Readiness Checklist

A copy-out checklist for teams piloting nano-ros toward production deployment. Each box is a concrete validation step, not a marketing claim. The book documents the framework’s intent + plumbing; this page is what your team needs to confirm on your target before shipping.

Why a separate checklist? nano-ros is production-capable, but some acceptance items are hardware-gated (P99 latency on real Cortex-M3, multicast on real silicon, NuttX SCHED_SPORADIC under kernel config) and can’t be validated in CI. The checklist gives you the steps to close those gaps for your deployment.

1. Real-time metrics (hardware-validated)

The book’s quoted poll-WCET / P99-latency numbers come from QEMU. DWT cycle counters are best-effort under emulation. For production claims, re-measure on your actual silicon.

  • End-to-end P99 latency (publisher → executor callback) on your target MCU at its production clock + load. Target: ≤ design budget. Tooling: wake-latency-cortex-m3 bench at packages/testing/nros-bench/wake-latency-cortex-m3/.
  • Worst-case stack depth per task. Tool: cargo-call-stack, cargo-stack-sizes, or the ARM stack-analyzer for C/C++.
  • Heap fragmentation pattern over 24 h at nominal load. Spot-check with mallinfo or your RTOS’s heap-stats API.
  • Wake latency (transport-rx interrupt → first user callback dispatched). Required if you use the async / poll- blocked spin path.
  • Spin-loop budget overrun rate under sustained pub load. Executor::spin_once(timeout) returns the overrun count.

2. Platform-specific validation

Per-RTOS gaps that the book documents as “tested in CI” cover reference boards — your actual board / kernel-config combination may differ.

  • Multicast / IGMP if using DDS (RTPS). Confirm SPDP discovery actually fires on your RTOS + driver. Untested on FreeRTOS + ThreadX as of writing.
  • Clock wraparound + extension correctness on long-running deployments. The platform’s nros_platform_time_now_ms must handle u32 wrap (49.7 days) and u64 extend.
  • Allocator behavior under memory pressure. Boot-time alloc OK on most RTOSes; mid-run alloc only on std POSIX. Confirm your hot paths don’t allocate.
  • Network packet loss recovery. Drop 5% of packets in your lab and confirm the talker / listener recovers.
  • Critical-section regions are short. The platform’s nros_platform_critical_section_* ABI is the IRQ-disable surface; long critical sections starve other ISRs.

3. RMW backend certification

  • Backend version pinned to a tested tuple. Zenoh-pico 1.7.2 (matches rmw_zenoh_cpp). Cyclone DDS 0.10.5 (matches ros-humble-cyclonedds). XRCE-DDS Micro-Client at its workspace pin.
  • All required QoS policies supported by your backend. The Choosing an RMW Backend capability matrix lists per-backend coverage (Zenoh: 4/7; XRCE: 4/7; Cyclone DDS: 7/7).
  • Discovery stability over your network topology. Zenoh-pico in client mode needs zenohd reachable; loss of router = lost routing but local node lives. XRCE needs Agent uptime. Cyclone DDS discovers via multicast SPDP.
  • Bridge stability if multi-backend. Confirm no memory bloat over 72 h with two registered RMWs running.
  • Cyclone DDS limitations checked. Status events and some stock-ROS interop slices are still in progress; embedded RTOS ports remain gated on a hosted Cyclone runtime.

4. Safety + formal verification

  • just verify-kani clean against your build. 160 bounded harnesses; non-trivial coverage of CDR + scheduling + RMW glue.
  • just verify-verus clean. 102 deductive proofs.
  • CRC32 attached if using safety-e2e feature. The 37-byte attachment is transparent to stock ROS 2 (ignored gracefully) and detected by other nano-ros nodes.
  • Timeout bounds on every blocking call. spin_once timeout, Promise::wait_for(timeout), recv_timeout. No WAIT_FOREVER.
  • Parameter store capacity ≥ declared parameter count ([param-services] feature gate).
  • Stack overflow detection enabled by your platform (FreeRTOS configCHECK_FOR_STACK_OVERFLOW, Zephyr stack sentinels, NuttX CONFIG_DEBUG_STACK).

5. Interop testing

  • Publish from nano-ros, subscribe with stock ROS 2. RMW_IMPLEMENTATION=rmw_zenoh_cpp for Zenoh, rmw_cyclonedds_cpp for Cyclone, rmw_fastrtps_cpp for DDS (interop tier).
  • Message type compatibility for any custom .msg you’ve added. Round-trip a sample message through ROS 2’s rosbag2 to confirm wire-level parity.
  • QoS profile matching. Mismatched reliability / durability / history kill discovery silently on DDS / RTPS.
  • Lifecycle callbacks fire on node startup / shutdown if you’ve opted into lifecycle-services.
  • Cross-RTOS interop: if your fleet mixes RTOSes (e.g. Zephyr sensor + FreeRTOS actuator + POSIX coordinator), confirm all three sides see each other.

6. Failure recovery

  • Agent / router restart: kill zenohd (Zenoh) or MicroXRCEAgent (XRCE) mid-run. Confirm reconnection. For DDS / Cyclone this is N/A (no central process).
  • Network partition → reconnection. Block the talker’s egress with iptables for 30 s, then unblock. Verify the listener resumes within your design SLA.
  • Heap exhaustion path: graceful degradation OR clean crash + restart? If hosted-RTOS + watchdog, restart is usually correct. If bare-metal, you probably have no restart story — confirm your design assumes this.
  • Stack overflow detection triggers a panic / fault rather than silent corruption.
  • Power loss mid-write (if persisting state). Not nano-ros’s concern, but mention it in your design review.

7. Operational concerns

  • Bootloader + OTA strategy. Out of scope for nano-ros but mandatory for fleet deployments — name it explicitly in your project plan.
  • Log / diagnostics exfiltration. nros-log provides the logging surface; pick a sink (UART, RTT, semihosting, or ROS 2 /rosout over the wire).
  • Time synchronization (NTP, PTP, RTC). nano-ros doesn’t ship a time-sync layer; your fleet design must.
  • Watchdog coverage: the executor’s spin_period reports overruns, but it doesn’t pet a hardware watchdog. Wire one manually.

8. License + governance

  • License: MIT OR Apache 2.0 (dual). Both permissive, no GPL copyleft, OK for proprietary firmware. Confirm your legal team is comfortable.
  • Third-party dependencies: zenoh-pico (Eclipse), Cyclone DDS (Eclipse), Micro-XRCE-DDS-Client (Apache-2). Vendored dependencies carry license files in each third-party/*/LICENSE.
  • Patent grant: Apache 2.0 carries an explicit patent grant; MIT does not. Most adopters rely on the Apache half.
  • Support model: nano-ros has no commercial support entity as of writing. Plan accordingly — either staff in-house expertise or contract a consultancy.
  • Roadmap visibility: track the project’s roadmap directory in the upstream repo. Items are numbered and dated.

9. Maintainability pledge

The single-maintainer signal in §8 is a real adoption risk. Add explicit mitigations to your project plan rather than treating “open-source” as the answer:

  • Maintenance horizon: record the date you adopted and the upstream commit cadence (e.g. git log --since=… --oneline | wc -l over the last 90 days). Re-check quarterly.
  • Escalation path: identify the primary maintainer contact (from Cargo.toml authors + GitHub commit history). For CVEs, use a public GitHub security advisory.
  • Governance model: nano-ros is BDFL-style (benevolent-dictator-for-life) with no formal RFC / vote process. Major design decisions are documented in the roadmap directory; align your expectations accordingly.
  • Fork mitigation: the dual MIT / Apache-2.0 license lets your team fork and maintain independently if upstream goes dormant. Budget for that possibility, not the assumption that it will happen.
  • Distro tracking: nano-ros targets ROS 2 Humble today. Iron / Jazzy support depends on the type-hash work tracked in the upstream roadmap (no public ETA yet). If your product must ship on Iron / Jazzy at launch, plan to contribute the type-hash port or wait for upstream.

Scoring rubric

For each section above, count [x] boxes as your readiness score. Suggested gates:

Score per sectionStatus
8/8Production-ready for that axis
5–7/8Pilot deployment OK; close gaps before scale
3–4/8Lab / prototype only
< 3/8Block on these items first

Sum across all 9 sections (§1–9). Below ~50 / 70 you have foundational work to do; above ~62 / 70 you’re at production quality on every axis where nano-ros can be validated externally.

See also

Zenoh-pico Symbol Reference

This page documents the ~55 FFI symbols that zenoh-pico requires at link time. These symbols are provided by zpico-sys’s C alias translation unit (c/zpico/platform_aliases.c, built by the default-on platform-aliases feature), which forwards each z_* / _z_* call to the canonical nros_platform_* ABI. When porting to a new platform, you implement an nros-platform-<name> crate (see Custom Platform) that supplies the nros_platform_* symbols — you do not provide the z_* symbols directly.

Historical note: this mapping used to live in a separate zpico-platform-shim crate; Phase 129 deleted it and folded the forwarders into the zpico-sys alias TU.

This page serves as a reference for understanding what the alias TU maps and what capabilities your nros-platform-<name> crate must provide.

Platform crate structure

packages/core/nros-platform-<name>/
├── Cargo.toml
└── src/
    ├── lib.rs
    ├── clock.rs
    ├── memory.rs
    ├── sleep.rs
    ├── random.rs
    ├── time.rs
    ├── threading.rs          # no-op stubs or real RTOS impl
    ├── socket_stubs.rs       # if using smoltcp
    └── network.rs            # if using smoltcp

Cargo.toml must have zero nros-* dependencies. It may depend on:

  • Hardware HAL crate (e.g., stm32f4xx-hal, esp-hal)
  • zpico-smoltcp (if using smoltcp networking)
  • embedded-alloc (for heap on bare-metal)
  • RTOS bindings crate

Required FFI symbols

Clock (critical)

The clock is the most important primitive. zenoh-pico uses it for session keep-alive, query timeouts, and transport timeouts. It must be backed by a hardware timer or OS tick — never by a software counter that only advances when polled.

// Monotonic clock — returns an opaque timestamp (lower 32 bits of ms)
usize   z_clock_now(void);

// Elapsed time since a previous z_clock_now() value
c_ulong z_clock_elapsed_us(usize *time);
c_ulong z_clock_elapsed_ms(usize *time);
c_ulong z_clock_elapsed_s(usize *time);

// Advance a timestamp value by a duration
void    z_clock_advance_us(usize *clock, c_ulong duration);
void    z_clock_advance_ms(usize *clock, c_ulong duration);
void    z_clock_advance_s(usize *clock, c_ulong duration);

If using smoltcp (bare-metal networking), also provide:

// Called by zpico-smoltcp for TCP/IP timestamping
u64 smoltcp_clock_now_ms(void);

Implementation checklist:

  1. Identify a hardware timer (SysTick, GPT, DWT) or use the OS tick API (xTaskGetTickCount, k_uptime_get, clock_gettime)
  2. Handle 32-bit timer wraps — track a wrap count in an atomic or use a 64-bit counter
  3. Never advance the clock inside smoltcp_network_poll() — read the hardware timer directly
  4. Verify with QEMU: use -icount shift=auto to synchronize virtual time with wall-clock time

Reference implementations:

PlatformClock sourceFile
MPS2-AN385CMSDK APB Timer0 (25 MHz)nros-platform-mps2-an385/src/clock.rs
STM32F4ARM DWT cycle counternros-platform-stm32f4/src/clock.rs
ESP32-C3 (QEMU)esp_hal::time::Instantnros-platform-esp32-qemu/src/clock.rs
FreeRTOSxTaskGetTickCount()Use OS tick directly
NuttXclock_gettime(CLOCK_MONOTONIC)POSIX API

Memory

zenoh-pico allocates heap during session open and entity creation. Typical minimum: 64 KB heap.

void *z_malloc(usize size);
void *z_realloc(void *ptr, usize size);
void  z_free(void *ptr);

Options:

  • embedded-alloc FreeListHeap (bare-metal)
  • RTOS heap (pvPortMalloc / tx_byte_allocate)
  • System malloc (POSIX, NuttX)

Sleep

i8 z_sleep_us(usize time);
i8 z_sleep_ms(usize time);
i8 z_sleep_s(usize time);

All return 0 (_Z_RES_OK). On bare-metal, busy-wait using the clock. On RTOS, delegate to vTaskDelay / tx_thread_sleep / k_sleep.

If using smoltcp, poll the network stack during busy-wait sleep to avoid missing packets.

Random

zenoh-pico needs randomness for session IDs, SN initialization, and scouting nonces.

u8   z_random_u8(void);
u16  z_random_u16(void);
u32  z_random_u32(void);
u64  z_random_u64(void);
void z_random_fill(void *buf, usize len);

A simple xorshift32 PRNG is sufficient. Seed it with hardware entropy (RNG peripheral, ADC noise, semihosting wall-clock time) during init.

Time

System time (wall clock). Used for logging and z_time_now_as_str().

u64         z_time_now(void);
const char *z_time_now_as_str(char *buf, c_ulong buflen);
c_ulong     z_time_elapsed_us(u64 *time);
c_ulong     z_time_elapsed_ms(u64 *time);
c_ulong     z_time_elapsed_s(u64 *time);
i8          _z_get_time_since_epoch(ZTimeSinceEpoch *t);

Where ZTimeSinceEpoch is:

#[repr(C)]
struct ZTimeSinceEpoch {
    u32 secs,
    u32 nanos,
}

On bare-metal without an RTC, return monotonic time or zeros.

Threading

For single-threaded platforms (bare-metal, RTIC), provide no-op stubs. For RTOS platforms, implement real task/mutex/condvar operations.

Task operations:

i8   _z_task_init(ZTask *task, ZTaskAttr *attr, void*(*fun)(void*), void *arg);
i8   _z_task_join(ZTask *task);
i8   _z_task_detach(ZTask *task);
i8   _z_task_cancel(ZTask *task);
void _z_task_exit(void);
void _z_task_free(ZTask **task);

Mutex operations:

i8 _z_mutex_init(ZMutex *m);
i8 _z_mutex_drop(ZMutex *m);
i8 _z_mutex_lock(ZMutex *m);
i8 _z_mutex_try_lock(ZMutex *m);
i8 _z_mutex_unlock(ZMutex *m);

Recursive mutex operations:

i8 _z_mutex_rec_init(ZMutexRec *m);
i8 _z_mutex_rec_drop(ZMutexRec *m);
i8 _z_mutex_rec_lock(ZMutexRec *m);
i8 _z_mutex_rec_try_lock(ZMutexRec *m);
i8 _z_mutex_rec_unlock(ZMutexRec *m);

Condition variable operations:

i8 _z_condvar_init(ZCondvar *cv);
i8 _z_condvar_drop(ZCondvar *cv);
i8 _z_condvar_signal(ZCondvar *cv);
i8 _z_condvar_signal_all(ZCondvar *cv);
i8 _z_condvar_wait(ZCondvar *cv, ZMutex *m);
i8 _z_condvar_wait_until(ZCondvar *cv, ZMutex *m, u64 *abstime);

Single-threaded stubs: Return 0 for all mutex/condvar operations. Return -1 for _z_task_init (task creation is not supported).

RTOS implementations: Map to xTaskCreate/xSemaphoreCreateMutex (FreeRTOS), tx_thread_create/tx_mutex_create (ThreadX), or pthread_create/pthread_mutex_init (NuttX, POSIX). zenoh-pico requires recursive mutexes (configUSE_RECURSIVE_MUTEXES=1 on FreeRTOS).

Sockets

If using smoltcp (bare-metal), socket operations are handled by zpico-smoltcp. Your platform crate provides thin shims:

#[repr(C)]
struct ZSysNetSocket { i8 _handle, bool _connected }

i8   _z_socket_set_non_blocking(const ZSysNetSocket *sock);  // Return 0
i8   _z_socket_accept(const ZSysNetSocket *in, ZSysNetSocket *out);  // Return -1
void _z_socket_close(ZSysNetSocket *sock);
i8   _z_socket_wait_event(void *peers, ZMutexRecRef *mutex);  // Return 0

If using OS sockets (POSIX, NuttX, Zephyr), zenoh-pico’s built-in socket layer handles everything — no socket stubs needed.

libc stubs (bare-metal only)

Bare-metal targets without a C runtime need standard C library functions:

usize  strlen(const char *s);
int    strcmp(const char *s1, const char *s2);
int    strncmp(const char *s1, const char *s2, usize n);
char  *strchr(const char *s, int c);
char  *strncpy(char *dest, const char *src, usize n);
void  *memcpy(void *dest, const void *src, usize n);
void  *memmove(void *dest, const void *src, usize n);
void  *memset(void *dest, int c, usize n);
int    memcmp(const void *s1, const void *s2, usize n);
void  *memchr(const void *s, int c, usize n);
c_ulong strtoul(const char *nptr, char **endptr, int base);
int   *__errno(void);

RTOS platforms (FreeRTOS, NuttX, ThreadX, Zephyr) ship their own libc — you do not need these stubs.

Network poll callback (smoltcp only)

If using smoltcp for networking, provide a poll callback that zpico-smoltcp calls to process network events:

void smoltcp_network_poll(void);

And a Rust API for the board crate to register the network state:

#![allow(unused)]
fn main() {
pub unsafe fn set_network_state(
    iface: *mut Interface,
    sockets: *mut SocketSet<'static>,
    device: *mut (),
);

pub unsafe fn clear_network_state();
}

Step-by-step procedure

  1. Create the platform crate (nros-platform-<name>) – see Custom Platform for the full guide
  2. Implement and verify the clock — this is the #1 cause of porting failures. Print clock_ms() in a loop and verify monotonic advance
  3. Implement remaining primitives — memory, random, sleep, time, threading, sockets. Each module is independent
  4. Wire into nros-platform — add a feature and ConcretePlatform alias
  5. Create the board crate — see Custom Board Package
  6. Add the platform feature to nros with mutual exclusivity checks
  7. Write an example — see Creating Examples
  8. Add test infrastructurejust test-<name> recipe + nextest group

Platform capability summary

CapabilityBare-metalRTOS (FreeRTOS/ThreadX)POSIX-like (NuttX/Zephyr)
ClockHardware timerOS tick APIclock_gettime
Memoryembedded-allocRTOS heapSystem malloc
SleepBusy-wait + pollvTaskDelaynanosleep
ThreadingNo-op stubsReal tasks + mutexespthreads
Socketssmoltcp shimslwIP or NetX socketsBSD sockets
RandomSeeded xorshiftRTOS RNG or xorshift/dev/urandom
libcHand-written stubsRTOS libcSystem libc
Network pollsmoltcp_network_pollStack-specific pollNot needed

Common pitfalls

See Platform Porting Pitfalls for detailed failure modes including poll-driven clocks, DMA buffer placement, QEMU I/O starvation, recursive mutexes, stack sizing, and heap sizing.

XRCE-DDS Symbol Reference

XRCE-DDS has a minimal platform abstraction layer. It is single-threaded, heap-less, and delegates networking to user-provided transport callbacks. The 2-3 required FFI symbols (uxr_millis, uxr_nanos) are provided by nros-rmw-xrce’s C alias translation unit (src/platform_aliases.c, compiled into nros-rmw-xrce-cffi on every target), which forwards them to the canonical nros_platform_* ABI. When porting to a new platform, you implement an nros-platform-<name> crate (see Custom Platform) that supplies the nros_platform_* symbols — you do not provide the uxr_* symbols directly.

Historical note: this mapping used to live in a separate xrce-platform-shim crate; Phase 129 deleted it and moved the uxr_* forwarders into the nros-rmw-xrce alias TU.

Platform crate structure

The XRCE-DDS clock symbols are a subset of what the platform crate provides. Your nros-platform-<name> crate provides the canonical clock primitives (nros_platform_time_now_ms, nros_platform_clock_us) and the alias TU maps them to the uxr_* symbols XRCE-DDS expects.

packages/core/nros-platform-<name>/
├── Cargo.toml
└── src/
    └── lib.rs          # clock + other primitives

Cargo.toml must have zero nros-* dependencies. It may depend on:

  • Hardware HAL crate (e.g., stm32f4xx-hal, esp-hal)
  • xrce-smoltcp (if using smoltcp networking)

Required FFI symbols

Clock (2 symbols)

XRCE-DDS uses the clock for session timeouts and time synchronization.

// Monotonic millisecond clock
i64 uxr_millis(void);

// Monotonic nanosecond clock
i64 uxr_nanos(void);

These must be backed by a hardware timer or OS tick — same rules as the zenoh-pico clock.

Reference implementations:

PlatformClock sourceFile
MPS2-AN385CMSDK APB Timer0 (25 MHz)nros-platform-mps2-an385/src/clock.rs
Zephyrk_uptime_get()xrce-zephyr/src/xrce_zephyr.c
POSIXclock_gettime(CLOCK_MONOTONIC)Built-in time.c from XRCE library

smoltcp clock (1 symbol, if using smoltcp)

If using xrce-smoltcp for bare-metal networking:

u64 smoltcp_clock_now_ms(void);

This is the same symbol used by zpico-smoltcp — shared clock implementations work for both RMW backends.

Transport callbacks

Instead of requiring socket FFI symbols, XRCE-DDS uses a custom transport abstraction. You register four callbacks via uxr_set_custom_transport_callbacks():

bool   open(uxrCustomTransport *transport);
bool   close(uxrCustomTransport *transport);
size_t write(uxrCustomTransport *transport, const uint8_t *buf, size_t len, uint8_t *errcode);
size_t read(uxrCustomTransport *transport, uint8_t *buf, size_t len, int timeout_ms, uint8_t *errcode);

These are typically provided by a transport crate, not the platform crate:

TransportCrateDescription
smoltcp UDPxrce-smoltcpBare-metal networking via smoltcp
Zephyr UDPxrce-zephyrZephyr BSD sockets
POSIX UDPBuilt-inUses XRCE library’s POSIX transport

The transport crate registers its callbacks during initialization. Your platform crate does not need to implement sockets.

What you do NOT need

Unlike zenoh-pico, XRCE-DDS does not require:

  • Memory symbols — XRCE-DDS is heap-less; all buffers are statically sized
  • Sleep symbols — polling is driven by uxr_run_session_time()
  • Random symbols — no session ID randomization needed
  • Threading symbols — single-threaded by design
  • libc stubs — Micro-XRCE-DDS has minimal C dependencies

This makes XRCE-DDS the easiest RMW backend to port to new platforms.

Platform-conditional compilation in xrce-sys

The xrce-sys crate uses Cargo features to select platform behavior:

FeatureEffect
posixCompiles built-in time.c (uses clock_gettime, BSD sockets)
bare-metalSkips time.c; expects uxr_millis()/uxr_nanos() from platform crate
zephyrCompiles xrce_zephyr.c (Zephyr kernel APIs)
freertosRTOS-specific; skips time.c
nuttxRTOS-specific; skips time.c
threadxRTOS-specific; skips time.c

When adding a new platform, you may need to add a feature to xrce-sys that skips the default time.c compilation and lets your platform crate provide the clock symbols.

Step-by-step procedure

  1. Create the platform cratenros-platform-<name>/ (see Custom Platform)
  2. Implement the canonical clock primitivesnros_platform_time_now_ms() / nros_platform_clock_us(); the alias TU (nros-rmw-xrce/src/platform_aliases.c) maps these to uxr_millis() / uxr_nanos() for you
  3. Implement smoltcp_clock_now_ms() if using smoltcp transport
  4. Add a feature to xrce-sys for the new platform if needed
  5. Choose or implement a transport crate — reuse xrce-smoltcp for bare-metal, or implement a new transport if the platform has its own networking stack
  6. Create the board crate — see Custom Board Package
  7. Add the platform feature to nros with mutual exclusivity checks
  8. Write an example and tests

Example: bare-metal MPS2-AN385

The simplest reference is nros-platform-mps2-an385. It supplies the canonical nros_platform_* clock symbols; the XRCE alias TU (nros-rmw-xrce/src/platform_aliases.c) then derives uxr_millis / uxr_nanos from them — you never hand-write the uxr_* symbols:

/* nros-rmw-xrce/src/platform_aliases.c — provided for you, not the porter */
int64_t uxr_millis(void) {
    return (int64_t) nros_platform_time_now_ms();
}
int64_t uxr_nanos(void) {
    return (int64_t) nros_platform_clock_us() * 1000;
}

So a porter only implements the canonical primitives in their platform crate (nros_platform_time_now_ms, nros_platform_clock_us), plus smoltcp_clock_now_ms() if using the smoltcp transport. The MPS2-AN385 platform reads these from a hardware timer with wrap detection.

Where clock_ms() reads from a hardware timer with wrap detection.

Common pitfalls

  • Agent connectivity — XRCE-DDS requires a Micro-XRCE-DDS Agent running on the host. Ensure the agent is reachable from the target before debugging the platform layer.
  • Reliable stream historySTREAM_HISTORY must be >= 2 (recommend 4). History=1 fails to recycle slots between separate uxr_run_session_until_all_status calls.
  • Flush after request_datauxr_buffer_request_data must be flushed with uxr_run_session_time immediately after being called. Unflushed requests in the reliable output stream cause intermittent timeouts.

Creating Examples

This guide describes the canonical example shape adopted (layout) + (consumption). When adding a new example, copy the nearest existing peer; the per-platform working examples are the authoritative templates.

Canonical layout

Every example is a self-contained, copy-out project under one of:

PathUsed for
examples/<plat>/<lang>/<example>/The standard cell. RMW is selected at build time (Cargo features / -DNANO_ROS_RMW= / Kconfig overlay), not encoded in the path. A single-package “app” example here is the canonical starter shape; the multi-package workspace shape (Node + Bringup + Entry pkgs) kicks in at ≥2 nodes — see Multi-Node Projects.
examples/bridges/<name>/Cross-RMW gateway examples (one binary, multiple backends).
examples/templates/<name>/Multi-platform copy-out recipes (e.g. multi-package-workspace).

The <plat> × <lang> coverage matrix (RMW chosen at build time) is authoritative in examples/README.md. Intentionally empty cells (bare-metal C/C++, PX4 Rust, …) are listed in the same file; do not fill them without lifting the underlying constraint.

Variant naming uses suffix form so peers sort together: talker-rtic, service-client-async, talker-rtic-mixed. Avoid parallel parent directories like async-*/ or rtic-*/.

Non-example binaries

Tests / benches / smokes are not under examples/. They live under packages/testing/:

UseLocation
Performance, fairness, stress, large-msgpackages/testing/nros-bench/<name>/
Driver / board bringup smoke (no nros API)packages/testing/nros-smoke/<name>/
Fixture binaries built by integration testspackages/testing/nros-tests/bins/<name>/

Consumption shape

Rust (native or Cargo cross-target)

Each example is a standalone Cargo package with empty [workspace] table — it does not participate in any walking-up workspace:

[package]
name = "native-rs-zenoh-talker"
edition = "2024"
publish = false

[[bin]]
name = "talker"
path = "src/main.rs"

[dependencies]
nros = { path = "../../../../packages/core/nros",
         default-features = false,
         features = ["std", "rmw-cffi", "platform-posix"] }
nros-rmw-zenoh = { path = "../../../../packages/zpico/nros-rmw-zenoh",
                   features = ["std", "platform-posix", "ros-humble"] }

[workspace]

cargo build / cargo run from inside the example directory is the canonical invocation. There is no workspace-wide cargo build that picks up examples — they are explicitly out-of-workspace.

C / C++ (CMake)

Each example is a standalone CMake project that pulls nano-ros via add_subdirectory(<repo-root>). The canonical four-line preamble:

cmake_minimum_required(VERSION 3.22)
project(my_example LANGUAGES C CXX)

set(NANO_ROS_PLATFORM <plat>)
set(NANO_ROS_RMW      <rmw>)
set(NANO_ROS_BOARD    <board>)        # embedded only
add_subdirectory(<rel-path-to-repo-root> nano_ros)

add_executable(my_example src/main.c)
target_link_libraries(my_example PRIVATE NanoRos::NanoRos)
nros_platform_link_app(my_example)
nano_ros_link_rmw(my_example RMW <rmw>)

nano_ros_link_rmw emits the strong-stub nros_app_register_backends() that calls every linked RMW’s nros_rmw_<x>_register() symbol — the auto-registration path for targets where linkme’s distributed-slice contribution isn’t picked up by the linker (FreeRTOS, NuttX, Zephyr, ESP-IDF).

There is no find_package(NanoRos) path deleted it along with just install-local, every install(...) rule, and every Config.cmake.in template.

Per-example contents

examples/<plat>/<lang>/<example>/
├── package.xml                    # ROS-style manifest for the example
├── Cargo.toml | CMakeLists.txt    # Rust or C/C++ build entry
├── .cargo/config.toml             # Rust only — target + .cargo patches
├── config.toml                    # network + zenoh settings (embedded)
├── src/                           # main.rs / main.c / main.cpp
├── generated/                     # codegen output for any custom msgs
└── README.md                      # usage instructions

Each example’s Cargo.toml / CMakeLists.txt builds in isolation — no workspace reliance, no path heuristics walking up the source tree.

Message generation

Examples with custom .msg, .srv, or .action files generate bindings in-tree under generated/. The generated/ directory is gitignored per-example (only packages/interfaces/rcl-interfaces/ generated bindings live in git Arduino bundle exception aside).

source /opt/ros/humble/setup.sh        # for rosidl tooling
nros generate-rust            # or generate-c / generate-cpp / generate-all

For BSP / cross-target examples that maintain their own .cargo/config.toml, pass --config --nano-ros-path <relative>:

nros generate-rust --config --nano-ros-path ../../../packages

The --config flag uses ConfigPatcher to idempotently add [patch.crates-io] entries while preserving existing [build] / [target.*] sections.

For CMake consumers:

nros_find_interfaces(LANGUAGE C SKIP_INSTALL)
nano_ros_generate_interfaces(... LANGUAGE C)

See Message Generation for the full reference + package.xml schema.

Adding a new example — checklist

  1. Pick the canonical cell. Confirm <plat>/<lang>/<name> isn’t in the “intentionally empty” list in examples/README.md.
  2. Copy the nearest peer. Identical-RMW + adjacent-platform is the lowest-risk template (e.g. copy examples/qemu-arm-freertos/c/talker to make a new FreeRTOS C/zenoh example).
  3. Update names + package.xml. Rename Cargo.toml’s name and [[bin]] entries (Rust) or project(...) and add_executable(...) targets (CMake).
  4. Regenerate bindings. Run nros generate-rust-* against the new package.xml. Custom messages need their own package.xml in the consuming example.
  5. Build standalone. cargo build or cmake -B build && cmake --build build from the example directory. No walking-up workspace allowed.
  6. Wire the test fixture (optional). If the example needs an E2E gate, add a builder in packages/testing/nros-tests/src/fixtures/binaries/<plat>.rs that runs cargo build / cmake --build and points the test at the resulting binary.
  7. Update examples/README.md coverage matrix if you filled a previously-empty cell.

Per-platform notes

PlatformSource file shapeBuild commandNotes
nativesrc/main.rs, src/main.c, src/main.cppcargo run / cmake --buildFull std. Pattern A or B.
qemu-arm-baremetalsrc/main.rs with #[entry]cargo run (runner = qemu-system-arm …)No std. Pure Cortex-M3.
qemu-arm-freertossrc/main.rs / src/main.cpp / src/main.ccargo run (Rust) or cmake --build (C/C++)FreeRTOS kernel + lwIP.
nuttxsrc/main.rs / src/main.ccmake --build (NuttX export tarball)NuttX kernel.
threadx-linux / threadx-riscv64src/main.rscmake --buildThreadX + NetX Duo.
esp32src/main.rscargo run (esp-hal)bare-metal esp-hal.
zephyrsrc/lib.rs (staticlib) or src/main.cppwest buildKconfig + west module.

Per-platform deep-dives — toolchain setup, Kconfig variables, runner scripts — live in the Platform Guides.

See Also

Contributing

This chapter covers the development workflow for nano-ros.

Development Setup

Choose the setup tier. base is enough for most code review and native development; all prepares the full test-all matrix:

just setup               # print setup choices
just setup base          # workspace tools + zenohd
just setup all           # all supported SDKs/services
source ./setup.bash

Build the entire workspace including examples:

just build

Use groups to inspect the command surface:

just --group main --list
just --group full-matrix --list
just --group full-matrix --list zephyr

Quality Checks

Always run just ci after completing a task that needs the full maintainer gate. This runs the normal checks plus the full test matrix entry points:

just ci

For a narrower local iteration, run just check and just test.

For platform-specific CI, use just <platform> ci (e.g., just freertos ci).

Testing

nano-ros has several test tiers, each with its own just recipe:

RecipeWhat it testsExternal deps
just test-unitUnit tests (no external deps)None
just test-integrationRust integration tests, no heavy QEMU/Zephyr (builds zenohd automatically)None
just testtest-unit + test-integration (default dev tier)None
just test-docrustdoc doctests for the nros umbrella crateNone
just test-miriUndefined behavior detection (standalone)None
just qemu testQEMU bare-metal examplesqemu-system-arm
just freertos testFreeRTOS QEMU E2Eqemu-system-arm + arm-none-eabi-gcc
just nuttx testNuttX QEMU E2Enightly + qemu-system-arm
just threadx_linux testThreadX Linux user-space E2Egcc
just threadx_riscv64 testThreadX RISC-V QEMU E2Eqemu-system-riscv64
just zephyr testZephyr E2E (native_sim + NSOS)west
just esp32 testESP32-C3 QEMU E2Enightly + qemu-system-riscv32
just test-allEverything: test + heavy QEMU/Zephyr/ROS-interop + test-doc + test-miri + C codegenAll of the above

All test recipes accept a verbose argument for live output.

Test organization

  • Reusable Rust tests go in packages/testing/nros-tests/tests/
  • Shell-based test scripts go in tests/ with corresponding justfile entries
  • Temporary tests can be run directly in the shell, then converted to proper tests once validated

Build isolation

Nextest runs each test file as a separate process in parallel. When multiple tests build the same example with different features, use --target-dir to isolate output directories (e.g., target-safety/, target-zero-copy/). Add new target dirs to the example’s per-directory .gitignore.

Code Style

Rust Edition 2024

nano-ros uses Rust edition 2024, which requires:

  • unsafe extern "C" { ... } – extern blocks require the unsafe keyword
  • #[unsafe(no_mangle)]no_mangle requires the unsafe attribute
  • Unsafe operations inside unsafe fn need explicit unsafe { ... } blocks

The nros-c crate keeps #![allow(unsafe_op_in_unsafe_fn)] due to 420+ FFI operations.

Unsafe conventions

  • Minimize unsafe usage; prefer safe abstractions
  • Document safety invariants with // SAFETY: comments
  • Use wrapper types (like SubscriberBufferRef) to encapsulate unsafe access patterns

no_std patterns

All core crates support #![no_std] with optional std/alloc features:

  • #[cfg(feature = "alloc")] gates Vec/Box usage
  • #[cfg(feature = "std")] gates std-only features
  • heapless::Vec is used in no_std contexts

Unused variables

  • Rename to _name with a comment explaining why
  • Use #[allow(dead_code)] for test struct fields

Message Types

Message types are always generated, never hand-written. Use the codegen tool:

nros generate-rust

Bundled interface definitions live in packages/codegen/interfaces/. Example generated/ directories are gitignored and recreated by just generate-bindings. Only packages/interfaces/rcl-interfaces/generated/ is checked into git.

System Packages

Never install system packages or run sudo directly. If a system dependency is needed, document what the user should install.

Temporary Files

Create temporary files in the project’s tmp/ directory (git-ignored), not in /tmp.

PR Workflow

  1. Create a branch from main
  2. Make changes
  3. Run just ci – all checks must pass
  4. Submit for review

Verification

For changes to core crates, run the formal verification suite:

just verify          # Kani + Verus
just verify-kani     # Kani bounded model checking (~3 min)
just verify-verus    # Verus deductive proofs (~1 sec)

nros CLI reference

The nros binary is the user entry point for nano-ros: scaffolding, message codegen, SDK provisioning, topology planning, static checks, diagnostics, and board introspection.

The old cargo nano-ros … cargo subcommand has been removed. Use nros directly.

Install

The nros CLI ships from the in-tree sub-workspace at packages/cli/ (Phase 218). Build it per checkout, then activate the workspace to put it on PATH:

./scripts/bootstrap.sh base
source ./activate.sh        # OR: direnv allow / source ./activate.fish

The resulting binary lives at packages/cli/target/release/nros. One checkout = one CLI version = one runtime ABI; contributors with multiple nano-ros worktrees get per-tree CLIs with no global PATH skew. Once on PATH, nros --help lists every verb. The transitional ${NROS_HOME:-~/.nros}/bin/nros install location remains supported for users mid-migration from the pre-218 release-fetch shape.

Verbs

nros setup [<board>] [--rmw <rmw>] [--tool <name>] [--source <name>] [--prefix <dir>] [--list] [--licenses] [--dry-run]

Provision the toolchain/SDK for a board. nros setup is the single canonical provisioning command — it ships prebuilt toolchains per platform per RMW (cross-compiler, emulator, RMW host daemon, SDK sources) from a pinned index into a shared store (${NROS_HOME:-~/.nros}/sdk). No hand-installed cross-toolchains; no ROS distro required.

nros setup native --rmw zenoh            # host build + zenoh router
nros setup qemu-arm-freertos --rmw xrce  # arm-none-eabi-gcc, qemu, FreeRTOS, XRCE agent
nros setup zephyr                        # Zephyr west workspace + SDK bits
FlagEffect
<board>resolve + fetch this board’s package set (see Supported Boards)
--rmw <zenoh|xrce|cyclonedds>also provision the RMW’s host daemon/tool (default zenoh); resolves board ∪ rmw packages
--tool <name>install a single tool by name (e.g. --tool qemu)
--source <name>provision a single source package by name (repeatable)
--prefix <dir>install a --tool here instead of the shared store
--listlist every package in the index + its version
--licensesshow license-gated packages + how to install them
--dry-runresolve + print the plan without fetching anything

Board names: native, posix, qemu-arm-baremetal, mps2-an385, stm32f4, qemu-arm-freertos, qemu-arm-nuttx, qemu-riscv64-threadx, threadx-linux, esp32, qemu-esp32-baremetal, zephyr, and more — run nros setup --list or nros setup <board> --dry-run.

nros new <name> --platform <plat> [--rmw <rmw>] [--lang <lang>] [--use-case <case>] [--force]

Scaffold a new nano-ros project. Emits a colcon-compatible package.xml plus a hello-world Rust / C / C++ skeleton tuned for the chosen platform.

FlagValuesDefault
--platformnative, freertos, nuttx, threadx, zephyr, esp32, posix, baremetal(required)
--rmwzenoh, xrce, cycloneddszenoh
--langrust, c, cpprust
--use-casetalker, listener, service, actiontalker
--forceoverwrite an existing directoryoff

Deploy mode: nros new --deploy <name> [--kind <runner>] [--target <triple>] [--board <b>] [--bringup <pkg>] [--from-launch <path>] [--from-profile <name>] scaffolds a [deploy.<name>] target into the bringup package’s system.toml (RFC-0004 §4 — the deploy-target SSOT), instead of a project. --kind is a free-form runner key (self, qemu, flash, …). The bringup package is discovered automatically when the workspace exposes exactly one; pass --bringup <pkg> to pick one when there are several. --from-launch also seeds the bringup [system].default_launch; --from-profile forks an existing [deploy.<name>] in the same system.toml.

nros generate <lang> [--manifest <path>] [--output <dir>] [--ros-edition <edition>] [--force] [--verbose] [--generate-config]

Generate ROS 2 message bindings from a package.xml. Rust users should prefer the direct nros generate-rust command; C and C++ users normally use the CMake integration.

ArgumentValuesDefault
<lang>rust, c, cpp, all(required)
--manifestpath to package.xmlpackage.xml
--outputoutput directorygenerated
--ros-editionhumble, ironhumble
--generate-configemit .cargo/config.toml patches (Rust only)off

nros metadata <system_pkg> [--workspace <path>] [--out-dir <dir>] [--metadata <existing.json>] [--build [--nano-ros-workspace <path>]] — walk a colcon-style workspace under <workspace>/src/

collecting component source metadata into build/<system_pkg>/nros/source-metadata.json. The result feeds nros plan. With --build, any declared component (component_nros.toml) missing its source-metadata is produced by the metadata-mode build: nano-ros compiles + runs the component against an in-memory recorder to emit the JSON. --build needs the nano-ros workspace (--nano-ros-workspace or NROS_WORKSPACE).

nros plan <system_pkg> <launch_file> [LAUNCH_ARGS...] [options] — resolve a ROS 2 launch file (or a precomputed

play_launch_parser record.json) plus the source metadata into a typed build/<system_pkg>/nros/nros-plan.json IR. Picks per-component SchedContext bindings, node graph wiring, parameter remaps, and generated-package layout.

FlagUse
--record <file>Skip launch-file parsing; consume an existing record.json
--workspace <path>Override the workspace root (default $PWD)
--out-dir <dir>Override build/<system_pkg>/nros/
--metadata <file>Reuse a prior source-metadata.json
--manifest <file>ROS launch manifest YAML overlay
--nros-toml <file>nano-ros deployment overlay (nros.toml)

nros check [plan]

Validate an nros-plan.json (default build/nros/nros-plan.json): static checker — catches unconnected required topics, conflicting QoS, missing parameters, and SchedContext binding errors before the platform build runs. A .toml argument is instead validated as a root nros.toml (the workspace deployment SSOT) — [system]/[deploy.<name>] shape, default-deploy + system references, bridge endpoints, etc.

nros explain [<plan>]

Render a generated nros-plan.json in human-readable form.

nros explain                            # reads build/nros/nros-plan.json
nros explain path/to/nros-plan.json     # explicit path

Run after nros plan to inspect the resolved component graph, topic wiring, parameter bindings, and SchedContext assignments before committing to a platform build.

nros config show --system <pkg> [--workspace <dir>]

Prints the resolved effective config for a bringup system (rmw, domain_id, locator, and the safety / param_services / lifecycle capability axes) with a per-value provenance columnsystem.toml [section] vs the built-in default. If a legacy per-package nros.toml overlay still declares any of those blocks, it is flagged DEPRECATED by name (RFC-0004 §3.1). Omit the <pkg> value to default to the workspace’s default_system (or the sole bringup); --workspace defaults to the cwd. The nros check command surfaces the same overlay warnings when validating a system.toml.

The legacy nros config show/check --config <path> reader for config.toml was removed (phase-256): config.toml is retired (RFC-0004 §8) and no example ships one. Embedded runtime config lives in [package.metadata.nros.deploy.<t>].

nros ws <subcommand>

Workspace-level message-package utilities. Manages codegen for *.msg packages in a colcon-style src/ tree and keeps [patch.crates-io] blocks in sync across Rust consumers.

eval "$(nros ws env)"   # add src/ to NROS_INTERFACE_SEARCH_PATH
nros ws sync            # run codegen for all msg pkgs + write [patch.crates-io]
nros ws status          # freshness check (non-fatal): n up-to-date / n stale / n missing
nros ws list            # list discovered msg + Rust-consumer pkgs
nros ws clean           # remove generated/ + auto-managed patch blocks
nros ws doctor          # lint workspace pkgs (package.xml markers, stale patches, …)
SubcommandDescription
envPrint shell export adding <dir> (default ./src) to NROS_INTERFACE_SEARCH_PATH
syncCodegen workspace msg pkgs + write [patch.crates-io] into each Rust consumer’s patch authority Cargo.toml. Pre-cargo step; run once after editing *.msg files
statusNon-fatal freshness check — sibling of sync --check
listList discovered msg + Rust-consumer pkgs (kind, name, dir per row)
cleanRemove generated/ + auto-managed [patch.crates-io] blocks; leaves user-written sections alone
doctorLint: warn on missing <member_of_group> rosidl_interface_packages</member_of_group>, malformed package.xml, stale patch blocks

nros codegen-system [--bringup <pkg>] [--target <triple>] [--out <dir>] [--launch <file>] [--ahead-of-vendor <pio|px4>]

Host-time system bake: reads <bringup>/system.toml + <bringup>/launch/system.launch.xml and emits the baked compile-time C config + component registration glue consumed by every embedded RTOS adapter.

nros codegen-system --bringup demo_bringup
nros codegen-system --bringup demo_bringup --target thumbv7em-none-eabihf
nros codegen-system --bringup demo_bringup --ahead-of-vendor pio   # + PlatformIO library.json
nros codegen-system --bringup demo_bringup --ahead-of-vendor px4   # + PX4 module dirs
FlagDescription
--workspace <path>Workspace root (default: cwd)
--bringup <pkg>Bringup pkg name or path. Defaults to [workspace.metadata.nros].default_system
--target <triple>Target triple for cross-compile bake context
--out <dir>Output directory; nros-system/ subdir created inside. Default: <workspace>/build/<bringup>/
--launch <file>Multi-launch disambiguation: pick <bringup>/launch/<file> (--file is an alias)
--exec <exec><node exec="…"> override for synthesised launches
--ahead-of-vendorpio: emit PlatformIO library.json; px4: emit one PX4-native nros_<component>/ module dir per component

nros doctor [--platform <name>] [--workspace <path>]

Shells out to just doctor from the auto-detected workspace root. The justfile orchestrates every per-module doctor recipe (just nuttx doctor, just zephyr doctor, …) and is the source of truth for “healthy”. --platform <name> scopes the check to a single module.

nros board list [--workspace <path>]

Enumerate every nros-board-* crate under packages/boards/. Output columns are name | description; structured chip | flash | ram | supported_rmw fields are deferred to a future board-descriptor TOML.

nros version

Print toolchain + library versions.

nros completions <shell>

Emit shell completion scripts to stdout.

ShellInstall snippet
bashnros completions bash > ~/.local/share/bash-completion/completions/nros
zshnros completions zsh > "${fpath[1]}/_nros"
fishnros completions fish > ~/.config/fish/completions/nros.fish
powershellnros completions powershell > $PROFILE.parent\nros.ps1

Comparison: nros vs. just

Want to …Use
Scaffold a projectnros new …
Generate Rust bindingsnros generate-rust
Plan + check a multi-component system from a ROS 2 launch filenros metadatanros plannros check
Build / flash / runPlatform tools: cargo, cmake --build, west, idf.py, probe-rs, or focused just <platform> … recipes
Orchestrate the workspace (setup, doctor, CI, multi-platform sweeps)just …

nros routes through the nros-cli-core library. just recipes that wrap user-flow operations call into nros for consistency; internal recipes (build matrices, CI orchestration) keep their current shape.

Release pipeline status

There is no nros release verb and no crates.io / Arduino zip / ESP-IDF binary / GitHub Releases tarball channel. Per the archive decision (2026-05-19), nano-ros is consumed by git clone --branch=v<X.Y.Z> plus the in-tree build recipes documented at Installation. Downstream RTOS package managers (Zephyr west, ESP-IDF idf_component_yml, NuttX Kconfig) consume the same source tree via the integration shells under integrations/<rtos>/.

Rust API

The Rust user-facing surface lives in the nros umbrella crate. Generated by rustdoc --no-deps; always reflects the current main branch.

API reference

nros rustdoc — start here.

That single crate re-exports every type a user application needs: Executor, Node, Publisher<M>, Subscription<M>, Service<S>, Client<S>, ActionServer<A>, ActionClient<A>, Timer, GuardCondition, plus the nros::prelude glob-import.

Where to start

  • New to nano-ros? → nros::prelude
  • Two-layer API (L1 polling vs L2 callback)? → Two-Layer API — concept page with the verb discipline and per-layer use cases.
  • Executor-driven app? → nros::Executor
  • Async/spin_async? → nros::dds_async (re-exports of the async surface)

Generating locally

cargo doc --no-deps -p nros --open

Internals (porting only)

These crates are not part of the user API. They surface only when writing a custom RMW backend or porting to a new platform — see Custom RMW Backend / Custom Platform.

C API

The C user-facing surface lives in packages/core/nros-c/include/nros/*.h. Generated by Doxygen; always reflects the current main branch.

API reference

C API Doxygen — start here.

The umbrella header nros/nros.h pulls in every public surface a user application needs.

Where to start

Two-layer API. Every entity exposes two parallel C entry-point sets: nros_*_init (Layer 1, caller polls) and nros_executor_register_* (Layer 2, executor callback). Layer 1 grew an _init_polling + try_recv_*_raw / send_*_raw family for inline-storage callers (no executor arena). Layer 2 keeps the existing nros_*_init + executor-register pair. See Two-Layer API for the verb discipline and per-layer trade-offs.

Event-driven wake callbacks. Each L1 polling entity supports nros_*_set_wake_callback(entity, &state, cb, ctx) for event- driven RTOS / embassy callers. State is a caller-owned nros_wake_state_t POD; the backend fires cb(ctx) on rx / reply / request arrival. Per-channel for actions (set_{goal,cancel,get_result}_wake_callback on the server, set_{goal_response,cancel_response,result,feedback}_wake_callback on the client). See the Doxygen for signatures.

Async action client

<nros/action.h> exposes two parallel entry-point families for sending goals, requesting results, and cancelling:

  • Async (non-blocking): nros_action_send_goal_async(), nros_action_get_result_async(), nros_action_cancel_goal(), nros_action_try_recv_feedback(). Return immediately. Replies are delivered through the callbacks registered with nros_action_client_set_goal_response_callback(), nros_action_client_set_feedback_callback(), and nros_action_client_set_result_callback(). The callbacks fire from nros_executor_spin_some() — your spin loop is the only place callbacks run.
  • Blocking convenience: nros_action_send_goal() and nros_action_get_result(). These call the async variant and then drive nros_executor_spin_some() internally on a wall-clock budget until the reply lands (or 15 s / 30 s timeout). They take the nros_executor_t* explicitly because the action client stores its handle as an opaque slot inside the executor (registered via nros_executor_register_action_client()). The same applies to nros_action_client_wait_for_action_server() and nros_action_client_action_server_is_ready(). Calling any of them from inside a dispatch callback returns NROS_RET_REENTRANT.

The equivalent service client helper nros_client_call() does not take an explicit executor — the client stashes the executor pointer during nros_executor_register_client() and recovers it internally — but it follows the same async-then-spin contract.

Canonical pattern: declare the executor, register the client, then either call the blocking helper or drive nros_executor_spin_some() yourself between async calls until your callback flags the result. See the example layouts in examples/native/c/action-client/ and examples/qemu-arm-freertos/c/action-client/.

added a separate L1 polling family (nros_action_client_init_polling + the _raw send/recv siblings) that stores ActionClientCore inline in the nros_action_client_t._opaque slot and skips the executor entirely; see the Doxygen for the full L1 verb list.

CMake integration

Out-of-tree projects use the NanoRos::C CMake target and the nano_ros_generate_interfaces() function — covered in docs/reference/c-api-cmake.md.

Generating locally

doxygen packages/core/nros-c/Doxyfile
xdg-open target/doc/c-api/html/index.html

Internals (porting only)

The rmw-cffi and platform-cffi Doxygen sites are not part of the user API — they document the C vtables for porters writing a custom RMW backend or platform port. See RMW API and Platform API.

C++ API

The C++ user-facing surface lives in packages/core/nros-cpp/include/nros/*.hpp. Generated by Doxygen; always reflects the current main branch.

API reference

C++ API Doxygen — start here.

The umbrella header nros/nros.hpp pulls in every public surface a user application needs.

Where to start

The library is freestanding C++14 — no STL, no exceptions. Define NROS_CPP_STD if you want the optional std::string / std::function / std::chrono overloads.

Two-layer API. The nros::ActionServer<A> / ActionClient<A> templates are L2 callback handles (set callbacks via nros_cpp_action_server_set_callbacks, dispatched by the executor). The new nros::PollingActionServer<A> / PollingActionClient<A> are L1 polling templates with the same typed serdes glue but caller-driven control flow. Subscriptions / services / service clients are already L1-shaped in nros-cpp. See Two-Layer API for the per-layer rationale.

Event-driven wake callbacks. PollingActionServer<A> and PollingActionClient<A> expose a nested WakeState POD and typed set_{goal,cancel,get_result,goal_response, cancel_response,result,feedback}_wake_callback(state, cb, ctx) methods. Pair an nros_cpp_wake_state_t with each entity / channel to wake on rx instead of polling.

Generating locally

doxygen packages/core/nros-cpp/Doxyfile
xdg-open target/doc/cpp-api/html/index.html

RMW API

The RMW (ROS middleware) vtable is the porting boundary between nano-ros and a concrete pub/sub transport (zenoh-pico, XRCE-DDS, Cyclone DDS, uORB, …). RMW is internal — user applications use the Rust / C / C++ APIs, not the vtable directly.

Canonical reference

The C vtable nros_rmw_vtable_t is the source of truth. Every function pointer’s brief, parameter docs, ownership rules (buffer-borrowed vs caller-owned), blocking / non-blocking classification, return-code conventions, and lifetime contract for loaned slots live in the Doxygen output.

SurfaceLink
rmw-cffi Doxygen (canonical)HTML · header

To regenerate locally:

just doc-rmw-cffi   # produces target/doxygen/rmw-cffi/

This page does not duplicate the interface specification — read the Doxygen for that.

Reference implementations

Concrete backends. Each crate’s README.md walks the implementation; the source is the worked example to copy.

BackendSourceNotes
zenoh-picopackages/zpico/nros-rmw-zenohDefault. C transport via zenoh-pico. Native zero-copy publish via z_bytes_from_static_buf.
micro-XRCE-DDS-Clientpackages/xrce/nros-rmw-xrceC-only shim; agent-based.
Cyclone DDSpackages/dds/nros-rmw-cycloneddsC++ shim; standalone CMake project.
PX4 uORBpackages/px4/nros-rmw-uorbTyped-trampoline registry over PX4 uORB.

The zenoh-pico shim is the canonical reference port.

Writing a custom backend

Platform API

The platform API is the porting boundary between nano-ros and a concrete OS / RTOS / bare-metal target. Each platform provides a clock, optionally a heap, optionally threading, optionally networking. Platform is internal — user applications use the Rust / C / C++ APIs, not the platform vtable directly.

Canonical reference

The C vtable in packages/core/nros-platform-cffi/include/nros/platform_vtable.h is the source of truth. Every function pointer’s brief, parameter docs, ownership rules, blocking / non-blocking classification, and ISR-safe contract live in the Doxygen output.

SurfaceLink
platform-cffi Doxygen (canonical)HTML · header

To regenerate locally:

just doc-platform-cffi   # produces target/doxygen/platform-cffi/

This page does not duplicate the interface specification — read the Doxygen for that.

Reference implementations

Each row is a complete worked example. The crate’s README.md walks the implementation; the source is the worked solution to copy.

CrateTargetSource
nros-platform-posixLinux / *BSDpackages/core/nros-platform-posix
nros-platform-nuttxNuttX RTOSpackages/core/nros-platform-nuttx
nros-platform-freertosFreeRTOSpackages/core/nros-platform-freertos
nros-platform-threadxAzure RTOS / ThreadXpackages/core/nros-platform-threadx
nros-platform-zephyrZephyr RTOSpackages/core/nros-platform-zephyr
nros-platform-mps2-an385Cortex-M3 (QEMU)packages/platforms/nros-platform-mps2-an385
nros-platform-stm32f4STM32F4packages/platforms/nros-platform-stm32f4
nros-platform-esp32-qemuESP32-C3 (QEMU)packages/platforms/nros-platform-esp32-qemu

The POSIX implementation is the canonical reference port.

Writing a custom platform

Platform Differences

Per-platform behaviour comparison. The Doxygen reference defines what each vtable entry must do; this page covers how each shipped platform crate fulfils that contract and which optional capabilities each platform exposes.

At a glance

PlatformClock sourceAllocatorThreadingNetworkingUDP multicastWall-clockNotes
POSIXclock_gettime(CLOCK_MONOTONIC)libc mallocpthreadslibc BSD sockets (Rust)Yesclock_gettime(CLOCK_REALTIME)Canonical reference port.
NuttXPOSIX aliaslibc mallocpthreadszenoh-pico unix/network.c (C)YesPOSIXMost paths inherit POSIX behaviour.
FreeRTOSxTaskGetTickCountpvPortMallocFreeRTOS taskslwIP via freertos-lwip-sys (Rust)Tick-based, no RTCMulticast gated on lwIP LWIP_IGMP=1 (untested).
Zephyrk_uptime_getk_mallocZephyr POSIX pthreadsZephyr POSIX sockets (Rust)Yes (native_sim via NSOS)clock_gettime(CLOCK_REALTIME)native_sim multicast routes through the NSOS IPPROTO_IP patch.
ThreadXtx_time_gettx_byte_allocateThreadX threadsNetX Duo BSD network.c (C)tx_time_get fallbackMulticast gated on NetX Duo nx_igmp_* (untested).
Bare-metal (MPS2-AN385)CMSDK Timer0bump allocatorsingle-threadednros-smoltcp (Rust)Yes (smoltcp 0.12 IGMP)monotonic fallbackCortex-M3 QEMU.
Bare-metal (STM32F4)DWT cycle counterbump allocatorsingle-threadednros-smoltcp (Rust)Yesmonotonic fallbackCortex-M4F.
Bare-metal (ESP32)esp_timer_get_timebump allocatorsingle-threadednros-smoltcp (Rust)Yesmonotonic fallbackXtensa LX6.
Bare-metal (ESP32-C3 QEMU)esp_timer_get_timebump allocatorsingle-threadednros-smoltcp (Rust)Yesmonotonic fallbackRISC-V.

Canonical interface contract. What “Yes” / “—” / specific functions mean — buffer ownership, blocking allowance, valid value ranges — is in the platform-cffi Doxygen reference. This page is a comparison table, not a spec.

Behaviour notes by trait group

Time (PlatformClock)

All ports surface a monotonic clock. Resolution varies:

  • POSIX / NuttX: nanosecond-resolution clock_gettime, exposed at ms + µs.
  • FreeRTOS: xTaskGetTickCount ticks (typically 1 ms). clock_us = clock_ms × 1000 — the resolution lie is documented.
  • Zephyr: k_uptime_get ms + k_cyc_to_us_floor64(k_cycle_get_64()) µs.
  • ThreadX: tx_time_get ticks (default 100 Hz; bump via TX_TIMER_TICKS_PER_SECOND for finer resolution).
  • Bare-metal (Cortex-M): hardware timer counter (CMSDK Timer0, DWT cycle counter). µs is the native resolution; ms is µs / 1000.
  • Bare-metal (ESP32): esp_timer_get_time returns µs natively.

The clock is monotonic and wraparound-free for the duration of nros::initnros::shutdown. Platforms with 32-bit timers run a software extender on overflow.

Memory (PlatformAlloc)

Only zenoh-pico calls PlatformAlloc::{alloc, realloc, free}. XRCE-DDS does not allocate. Recommended heap budget: 64 KB minimum for zenoh-pico’s working set; bare-metal ports typically allocate 128 KB+ via a bump allocator (linked-list-allocator or embedded-alloc).

The RTOS allocators (pvPortMalloc, tx_byte_allocate, k_malloc, libc malloc) honour their respective heap-region configurations. On bare-metal the heap region is defined in the linker script and consumed at startup by the bump allocator.

Threading (PlatformThreading)

Multi-threaded ports (POSIX, NuttX, FreeRTOS, Zephyr, ThreadX) expose real tasks, mutexes, and condition variables. zenoh-pico’s lease task and read task spawn into these. Single-threaded bare-metal ports stub task_init to return -1 so the spawn fails gracefully; the application drives spin_once from a single context.

Recursive mutexes are required by zenoh-pico. FreeRTOS uses xSemaphoreCreateRecursiveMutex; ThreadX uses tx_mutex_create(..., TX_INHERIT, ...); POSIX uses PTHREAD_MUTEX_RECURSIVE.

Sleep / Random / Wall-time (PlatformSleep, PlatformRandom, PlatformTime)

PlatformSleep on RTOS ports uses the kernel’s tickless sleep (vTaskDelay, tx_thread_sleep, k_msleep, libc usleep). On bare-metal with smoltcp, the sleep helper also runs network_poll() in a busy loop — packets must keep flowing during otherwise-idle waits or the zenoh lease times out.

PlatformRandom is a 32-bit xorshift PRNG seeded with hardware entropy (getrandom, RNG peripheral) or a build-time constant on platforms without entropy. Used for session IDs and protocol nonces; not cryptographic.

PlatformTime returns wall-clock for log timestamps. Bare-metal platforms with no RTC fall back to the monotonic clock — log timestamps then count from boot, not Unix epoch.

Networking (PlatformTcp, PlatformUdp, PlatformSocketHelpers, PlatformUdpMulticast, PlatformNetworkPoll)

The split into four traits maps directly onto zenoh-pico’s unix/network.c interface. Each port wires the four traits independently:

  • C-backed networking (NuttX, ThreadX): the platform shim forwards into zenoh-pico’s bundled unix/network.c (NuttX) or a NetX-Duo equivalent (ThreadX) — the C code talks to BSD-shape sockets directly.
  • Rust-backed networking (POSIX, FreeRTOS, Zephyr, bare-metal): the platform crate implements TCP / UDP in Rust. POSIX wraps libc, FreeRTOS wraps lwIP via freertos-lwip-sys, Zephyr wraps the POSIX layer, bare-metal goes through nros-smoltcp.

PlatformUdpMulticast (RTPS SPDP, zenoh scouting) ships fully wired on POSIX, NuttX, Zephyr, and bare-metal (smoltcp 0.12 IGMP group join). FreeRTOS and ThreadX have no multicast yet — gated by lwIP’s IGMP=1 (untested) and NetX Duo’s nx_igmp_* (untested).

PlatformNetworkPoll is bare-metal only. The implementation advances the smoltcp state machine; without it, smoltcp would only receive when the application explicitly asked. PlatformSleep and the wait_event helper both run network_poll() while waiting so packets continue flowing.

Libc (PlatformLibc)

zenoh-pico uses strlen, memcpy, errno, etc. directly. Bare-metal targets that link picolibc or newlib-nano satisfy these for free; the trait exists for no_std targets without a C runtime, which forward to Rust implementations.

Common pitfalls

For platform-specific gotchas (DMA buffer placement, ephemeral port conflicts, picolibc errno TLS, QEMU clock synchronization, Z_FEATURE_INTEREST mutex exhaustion, etc.), see Platform Porting Pitfalls.

For the design rationale (why these trait groups, why both ms and µs, why split PlatformNetworkPoll), see Platform API Design.

For writing a new port, see Custom Platform. For the canonical interface spec, see the Doxygen reference.

Supported Boards

Procurement-grade compatibility matrix. Each row lists a real vendor + board model and reports nano-ros’s status on it. Rows marked Tested boot in CI; Ready rows compile and run but have no in-CI gate yet; Untested rows compile per the architecture support but no one has reported booting nano-ros on them.

VendorBoardMCU / SoCArchDefault RTOSStatusExample / board crate
ARMMPS2-AN385 (QEMU)Cortex-M3Armv7-MFreeRTOS / bareTestedexamples/qemu-arm-freertos/, examples/qemu-arm-baremetal/
STMicroSTM32F4-DiscoverySTM32F407Cortex-M4FFreeRTOS / bareTestedpackages/boards/nros-board-stm32f4/
STMicroSTM32H7-NucleoSTM32H743Cortex-M7FFreeRTOS / ZephyrReadyUse FreeRTOS / Zephyr starter with nros-board-freertos overlay
STMicroPixhawk 4 (FMUv5)STM32F765Cortex-M7FNuttX (PX4)Readyintegrations/px4/module-template/
STMicroPixhawk 6X / 6CSTM32H753Cortex-M7FNuttX (PX4)Readyintegrations/px4/module-template/
NordicnRF52840-DKCortex-M4FArmv7E-MZephyrUntestedZephyr starter — supply -b nrf52840dk_nrf52840
NordicnRF5340-DKCortex-M33 (dual)Armv8-MZephyrUntestedZephyr starter — supply -b nrf5340dk_nrf5340_cpuapp
EspressifESP32-C3-DevKitRISC-V (RV32IMC)RISC-Vbare / ESP-IDFTestedexamples/qemu-esp32-baremetal/rust/, integrations/nano-ros/
EspressifESP32-C6-DevKitRISC-VRISC-VESP-IDFUntestedSame ESP-IDF path as C3
NXPLPC55S69-EVKCortex-M33Armv8-MZephyrUntestedZephyr -b lpcxpresso55s69_cpu0
NXPMIMXRT1170-EVKCortex-M7 + M4Armv7-MFreeRTOS / ZephyrUntestedFreeRTOS starter + vendor BSP
TILP-CC1352P7Cortex-M4FArmv7E-MFreeRTOS / TI-RTOSUntestedFreeRTOS starter + TI driver overlay
RP2040Raspberry Pi PicoCortex-M0+Armv6-Mbare / FreeRTOSUntestedBare-metal Cortex-M3 path — Cortex-M0+ has only 4 NVIC priority levels (per-callback OS-priority dispatch is disqualified — pub/sub still works fine)
QEMUvirt RISC-V64rv64gcRISC-VThreadXTestedexamples/threadx-riscv64/
QEMUCortex-A9 (Versatile)Cortex-A9Armv7-AZephyr / NuttXTestedZephyr -b qemu_cortex_a9, NuttX qemu-armv7a
Arm FVPBase_RevC AEMv8R (SMP)Cortex-A SMPArmv8-RZephyr 3.7Tested (build); license-gated runtimeSee ARM FVP getting-started chapter; just zephyr build-fvp-aemv8r{,-cyclonedds} + run-fvp-aemv8r{,-cyclonedds}
Linux host(sim)x86-64 / aarch64x86 / ArmThreadX simTestedexamples/threadx-linux/
Linux host(native)x86-64 / aarch64x86 / ArmPOSIXTestedexamples/native/

How to add a new board

  1. Pick the matching RTOS path. Cortex-M3 / M4 / M7 + RTOS → use FreeRTOS or Zephyr starter. Cortex-M0+ → bare-metal starter (limited; no NVIC priority headroom). Cortex-A / RISC-V64 → NuttX or Zephyr. Xtensa / RISC-V32 + Wi-Fi → ESP-IDF or esp-hal.
  2. Find or write a board crate. Existing crates under packages/boards/nros-board-*/ cover most QEMU + reference dev kits. Real-hardware boards need a thin board crate that supplies startup, linker script, and BoardIdle::wfi() (bare-metal) or wraps the RTOS’s BSP (FreeRTOS / Zephyr).
  3. Run the existing example tree. Each row above points at the canonical example dir. Cross-compile the talker / listener and verify against stock ROS 2 + RMW_IMPLEMENTATION=rmw_zenoh_cpp.
  4. Report back. Open an issue with the working build + flash + run commands so the row moves from Untested / Ready to Tested.

Caveats by chip family

  • Cortex-M0+ (RP2040, STM32F0, nRF51): only 4 NVIC priority levels. Per-callback OS-priority dispatch (a research scheduler shape originally proposed by Choi et al. as PiCAS, RTAS ’21) is disqualified on this class; nano-ros’s user-space EDF / FIFO scheduler is the only option. Pub/sub works fine.
  • Xtensa ESP32 / ESP32-S2 / ESP32-S3: needs the esp-rs fork of rustc (rustup target add does not cover Xtensa; install via https://github.com/esp-rs/rust-build).
  • Cortex-A9 / A53: hosted-RTOS only (NuttX, Zephyr). Heap + libc required; bare-metal Cortex-A is not in the coverage matrix.
  • PX4 boards: NuttX is the underlying kernel; the PX4 module template is the canonical entry path — see PX4 Autopilot.

See also

Environment Variables Reference

Configuration File

All environment variables can be set in a .env file at the project root:

cp .env.example .env
# Edit .env — uncomment and adjust values as needed
  • justfile.env is auto-loaded. Missing file silently ignored.
  • direnv.envrc sources .env if present.
  • Manualset -a; source .env; set +a before cargo build.

Variables in .env take precedence over justfile defaults but are overridden by explicit shell exports.

Runtime Configuration

Examples use ExecutorConfig::from_env() for configuration:

VariableDescriptionDefault
ROS_DOMAIN_IDROS 2 domain ID0
NROS_LOCATORRMW locator (tcp/…, udp/…, serial/…, or tls/…)tcp/127.0.0.1:7447
NROS_SESSION_MODESession mode: client or peerclient
ZENOH_TLS_ROOT_CA_CERTIFICATEPath to CA certificate (PEM) for TLS(none)
ZENOH_TLS_ROOT_CA_CERTIFICATE_BASE64Base64-encoded CA certificate for TLS(none)
ZENOH_TLS_VERIFY_NAME_ON_CONNECTVerify server hostname in TLS (true/false)(none)

Deprecated legacy names: ZENOH_LOCATOR and ZENOH_MODE are still accepted (they fall back to NROS_LOCATOR / NROS_SESSION_MODE) but will print a one-line deprecation warning to stderr. Migrate to the NROS_* names. ZENOH_TLS_* names are kept because TLS is currently zenoh-specific.

TLS Notes

  • POSIX: requires libmbedtls-dev (just setup base checks it). File-path and base64 cert loading are both supported.
  • Bare-metal: only ZENOH_TLS_ROOT_CA_CERTIFICATE_BASE64 is supported (no filesystem). The certificate is embedded at build time.
  • The link-tls Cargo feature must be enabled on both the example and the nros crate.

Build-Time Configuration

VariableDescriptionRequired
ZENOH_PICO_DIRCMake install prefix for pre-built zenoh-pico (use with system-zenohpico feature on zpico-sys)Only with system-zenohpico
SSIDWiFi network name for ESP32 examplesRequired for build-examples-esp32
PASSWORDWiFi password for ESP32 examplesRequired for build-examples-esp32

ARM FVP (FVP_BaseR_AEMv8R)

License-gated — nano-ros does not download the binary. Set one of the discovery vars after accepting the Arm EULA and installing locally. See the ARM FVP getting-started chapter for the end-to-end build+run walk-through.

VariableDescriptionDefault
ARMFVP_BIN_PATHDirectory containing FVP_BaseR_AEMv8R (Zephyr-canonical, highest priority).(unset)
ARM_FVP_DIRInstall root; resolver scans models/Linux64_GCC-*/ underneath. Matches sdk-index.(unset)

If neither is set, scripts/zephyr/resolve-fvp-bin.sh falls back to dirname $(command -v FVP_BaseR_AEMv8R). Phase 217.A — just zephyr run-fvp-aemv8r{,-cyclonedds} skips gracefully when the binary can’t be resolved.

After extracting the Arm-provided tarball, run scripts/installers/arm-fvp-installer.sh with ARM_FVP_DIR set to the extraction root — it locates FVP_BaseR_AEMv8R, symlinks the directory to ~/.nros/sdks/arm-fvp/current/, and prints the export ARMFVP_BIN_PATH=… line for your shell rc. Verify with nros doctor --board fvp-aemv8r-smp, which cross-checks the [gated.arm-fvp] entry in nros-sdk-index.toml and warns (never hard-fails — license-gated) when the binary is missing.

FreeRTOS / NuttX / ThreadX SDK Paths

These are auto-resolved by justfile recipes (defaulting to external/ paths from just freertos setup / just nuttx setup / just threadx_linux setup). Override via env vars if sources are elsewhere.

VariableDefaultDescription
FREERTOS_DIRthird-party/freertos/kernelFreeRTOS kernel source
FREERTOS_PORTGCC/ARM_CM3FreeRTOS portable layer
LWIP_DIRthird-party/freertos/lwiplwIP source
FREERTOS_CONFIG_DIRBoard crate’s config/FreeRTOSConfig.h + lwipopts.h
NUTTX_DIRthird-party/nuttx/nuttxNuttX RTOS source
NUTTX_APPS_DIRthird-party/nuttx/nuttx-appsNuttX apps source
THREADX_DIRthird-party/threadx/kernelThreadX kernel source
THREADX_CONFIG_DIRBoard crate’s config/ThreadX config (tx_user.h)
NETX_DIRthird-party/threadx/netxduoNetX Duo source
NETX_CONFIG_DIRBoard crate’s config/NetX Duo config (nx_user.h)

Buffer Tuning

All optional – platform-appropriate defaults apply if unset. See Configuration for deployment-scenario guidance and platform guides for target-specific sizing.

Zenoh-pico (ZPICO_*)

VariableDescriptionDefaultCrate
ZPICO_FRAG_MAX_SIZEMax reassembled message size after defragmentation65536 / 2048zpico-sys
ZPICO_BATCH_UNICAST_SIZEMax unicast batch size before fragmentation65536 / 1024zpico-sys
ZPICO_BATCH_MULTICAST_SIZEMax multicast batch size8192 / 1024zpico-sys
ZPICO_MAX_PUBLISHERSMax concurrent publishers in zenoh shim8zpico-sys
ZPICO_MAX_SUBSCRIBERSMax concurrent subscribers in zenoh shim8zpico-sys
ZPICO_MAX_QUERYABLESMax concurrent queryables in zenoh shim8zpico-sys
ZPICO_MAX_LIVELINESSMax concurrent liveliness tokens in zenoh shim16zpico-sys
ZPICO_MAX_PENDING_GETSMax concurrent in-flight service calls4zpico-sys
ZPICO_SUBSCRIBER_BUFFER_SIZEPer-subscriber static buffer in zenoh shim1024nros-rmw-zenoh
ZPICO_SERVICE_BUFFER_SIZEPer-service-server static buffer in zenoh shim1024nros-rmw-zenoh
ZPICO_GET_REPLY_BUF_SIZEStack buffer for service client replies4096zpico-sys
ZPICO_GET_POLL_INTERVAL_MSSingle-threaded polling interval in zenoh_shim_get()10zpico-sys
NROS_SMOLTCP_MAX_SOCKETSMax concurrent TCP sockets (smoltcp); brokered default since Phase 204.2. Legacy alias: ZPICO_SMOLTCP_MAX_SOCKETS.1 (brokered)nros-smoltcp
NROS_SMOLTCP_MAX_UDP_SOCKETSMax concurrent UDP sockets (smoltcp); 1 by default, 4 with the nros-smoltcp/rtps feature (Phase 204.2). Legacy alias: ZPICO_SMOLTCP_MAX_UDP_SOCKETS.1 (brokered)nros-smoltcp
NROS_SMOLTCP_BUFFER_SIZEPer-socket staging buffer (smoltcp). Legacy alias: ZPICO_SMOLTCP_BUFFER_SIZE.2048nros-smoltcp
NROS_SMOLTCP_CONNECT_TIMEOUT_MSTCP connection timeout (smoltcp). Legacy alias: ZPICO_SMOLTCP_CONNECT_TIMEOUT_MS.30000nros-smoltcp
NROS_SMOLTCP_SOCKET_TIMEOUT_MSTCP read/write timeout (smoltcp). Legacy alias: ZPICO_SMOLTCP_SOCKET_TIMEOUT_MS.10000nros-smoltcp

XRCE-DDS (XRCE_*)

VariableDescriptionDefaultCrate
XRCE_TRANSPORT_MTUCustom transport MTU; also sizes stream buffers (4x MTU) and UDP staging4096 / 512xrce-sys
XRCE_MAX_SUBSCRIBERSMax concurrent subscribers8nros-rmw-xrce
XRCE_MAX_SERVICE_SERVERSMax concurrent service servers4nros-rmw-xrce
XRCE_MAX_SERVICE_CLIENTSMax concurrent service clients4nros-rmw-xrce
XRCE_BUFFER_SIZEPer-slot static buffer size1024nros-rmw-xrce
XRCE_STREAM_HISTORYReliable stream history depth (must be >= 2)4nros-rmw-xrce
XRCE_ENTITY_CREATION_TIMEOUT_MSTimeout for entity creation1000nros-rmw-xrce
XRCE_SERVICE_REPLY_TIMEOUT_MSTimeout for service replies1000nros-rmw-xrce
XRCE_SERVICE_REPLY_RETRIESNumber of service reply retries5nros-rmw-xrce
XRCE_MAX_SESSION_CONNECTION_ATTEMPTSMax session connection attempts10xrce-sys
XRCE_MIN_SESSION_CONNECTION_INTERVALMin interval between connection attempts (ms)25xrce-sys
XRCE_MIN_HEARTBEAT_TIME_INTERVALMin heartbeat interval (ms)100xrce-sys
XRCE_UDP_META_COUNTIn-flight UDP packets per direction (smoltcp)4xrce-smoltcp

Core (NROS_*)

VariableDescriptionDefaultCrate
NROS_EXECUTOR_MAX_CBSMax executor callback slots (compile-time fixed array size)4nros-node
NROS_EXECUTOR_ARENA_SIZEExecutor arena size in bytes (compile-time fixed array size)4096nros-node
NROS_SUBSCRIPTION_BUFFER_SIZEDefault subscription/service buffer size (bytes)1024nros-node
NROS_EXECUTOR_MAX_HANDLESMax handles in a C API executor16nros-c
NROS_MAX_SUBSCRIPTIONSMax subscriptions in a C API executor8nros-c
NROS_MAX_TIMERSMax timers in a C API executor8nros-c
NROS_MAX_SERVICESMax services in a C API executor4nros-c
NROS_LET_BUFFER_SIZEBuffer size for LET semantics per handle512nros-c
NROS_MESSAGE_BUFFER_SIZEMax buffer size for subscription/service data4096nros-c
NROS_MAX_CONCURRENT_GOALSMax concurrent goals per action server (compile-time constant, not env-var configurable)4nros-c
NROS_MAX_PARAMETERSMax parameters in parameter server32nros-params
NROS_MAX_PARAM_NAME_LENMax parameter name length64nros-params
NROS_MAX_STRING_VALUE_LENMax string parameter value length256nros-params
NROS_MAX_ARRAY_LENMax parameter array length32nros-params
NROS_MAX_BYTE_ARRAY_LENMax byte array parameter length256nros-params

Quick Reference

Finding Commands

Root commands are grouped by audience:

just --list                         # grouped root command overview
just --groups                       # group names
just --group main --list            # normal development workflows
just --group full-matrix --list     # build-all / fixtures / test-all / ci
just --group setup --list           # setup and doctor entry points
just --group maintenance --list     # clean and generated-binding commands
just --group docs --list            # book and API docs

Platform and backend commands stay namespaced:

just --group full-matrix --list zephyr
just zephyr build-fixtures
just zephyr build-all
just qemu setup-network             # QEMU TAP networking only
just zephyr help                    # Zephyr-specific help
just zenohd build                   # Build the pinned zenoh router

Install the local book tooling before previewing docs:

just setup-docs                     # mdbook + mdbook-mermaid
just book-serve                     # serve book/src with live reload
just book                           # full deployed preview with API docs

Test profiling & slow-test reporting

just test and just test-all always print the slowest tests after the run summary — the top 20 by duration (binary, test name, time, status), parsed from the active profile’s target/nextest/<profile>/junit.xml. No flag needed; it is part of the normal output.

Deeper profiling is opt-in and adds nextest’s experimental event/output recording plus artifact export. It preserves the normal nextest execution model (same filters, cargo profile, parallelism) — it only records and exports, so leave it off for routine runs:

NROS_NEXTEST_RECORD=1 just test
NROS_NEXTEST_RECORD=1 just test-all

Each profiled run writes a timestamped directory under tmp/ with a stable -latest symlink:

tmp/nextest-profile-test-YYYYMMDD-HHMMSS/      tmp/nextest-profile-test-latest -> …
tmp/nextest-profile-test-all-YYYYMMDD-HHMMSS/  tmp/nextest-profile-test-all-latest -> …

Artifacts in that directory:

  • nextest-run.zip — portable recording archive; replay with full captured output (incl. successful tests) via cargo nextest replay.
  • nextest-trace.json — Chrome/Perfetto timeline (slot/group occupancy, idle slots, retries, long-pole tests). Canonical concurrency artifact.
  • junit.xml — copy of the run’s JUnit report.
  • env.txt, command.txt — the knobs and exact command used.

Knobs:

VariableEffect
NROS_NEXTEST_RECORD=1Enable recording + artifact export.
NROS_NEXTEST_RECORD_DIR=<path>Override the output dir (and its -latest link).
NROS_NEXTEST_TRACE_GROUP_BY=slot|binaryPerfetto grouping; default slot (wall-clock/concurrency view).
NROS_NEXTEST_REPLAY_LOG=1Also write nextest-replay.log (full captured stdout/stderr — can be large on chatty tests). Off by default; rely on the portable archive otherwise.
NROS_NEXTEST_RECORD_KEEP_STATE=1Keep the temp NEXTEST_STATE_DIR (<dir>/state); removed after export otherwise.
NROS_NEXTEST_RUN_PROFILE=fail-fastStop at the first failure instead of the default --no-fail-fast full report (uses target/nextest/fail-fast/junit.xml).

Overhead and retention: recording adds event/output-store writes during the run, and nextest-run.zip (plus an enabled nextest-replay.log) can get sizable on output-heavy suites — keep it opt-in for local runs and prune old tmp/nextest-profile-* directories. Recording uses a profile-local NEXTEST_STATE_DIR, so it never pollutes the user’s global nextest record store. Do not reach for --no-capture to inspect output: it serializes execution and skews every timing. Use the replay archive instead.

Manual Testing

# Base setup builds/checks the in-tree zenoh router.
just setup base
source ./setup.bash

# Terminal 1: Router
zenohd --listen tcp/127.0.0.1:7447

# Terminal 2: Talker
cd examples/native/rust/talker && RUST_LOG=info cargo run --features zenoh

# Terminal 3: Listener
cd examples/native/rust/listener && RUST_LOG=info cargo run --features zenoh

UDP Transport

On native/POSIX, zenoh-pico has built-in UDP support via OS sockets:

# Use UDP instead of TCP for the zenoh locator
NROS_LOCATOR=udp/127.0.0.1:7447 cargo run --features zenoh

On bare-metal, enable the link-udp-unicast feature to use UDP over smoltcp:

nros = { features = ["rmw-zenoh", "platform-bare-metal", "link-tcp", "link-udp-unicast"] }

The rmw-zenoh feature here is the lowering of the declared RMW — you declare the backend once in system.toml ([system].rmw / [deploy.<t>].rmw) and the toolchain sets the cargo feature; the feature is what the build uses, not the user-facing selector. See RFC-0031.

TLS Transport

TLS layers on top of TCP using mbedTLS. Requires a self-signed certificate (or real CA cert) and the link-tls Cargo feature.

Generate a test certificate:

openssl req -x509 -newkey ec -pkeyopt ec_paramgen_curve:prime256v1 \
  -keyout key.pem -out cert.pem -days 365 -nodes -subj "/CN=localhost"

Native/POSIX (requires libmbedtls-dev – installed by just setup base):

# Terminal 1: Router with TLS
zenohd --no-multicast-scouting --listen tls/localhost:7447 \
  --cfg 'transport/link/tls/listen_certificate:"cert.pem"' \
  --cfg 'transport/link/tls/listen_private_key:"key.pem"'

# Terminal 2: Talker with TLS
NROS_LOCATOR=tls/localhost:7447 \
  ZENOH_TLS_ROOT_CA_CERTIFICATE=cert.pem \
  cargo run -p native-rs-talker --features link-tls

Bare-metal (QEMU ARM):

On bare-metal, only base64-encoded certificates are supported (no filesystem). Build with --features link-tls:

# Build TLS-enabled examples
cd examples/qemu-arm-baremetal/rust/talker
cargo build --release --features link-tls

The CA certificate must be passed via ZENOH_TLS_ROOT_CA_CERTIFICATE_BASE64 at runtime (through the zenoh config), or embedded in the binary at build time.

ROS 2 Interop

# Terminal 1: Router
zenohd --listen tcp/127.0.0.1:7447

# Terminal 2: nros talker
cd examples/native/rust/talker && RUST_LOG=info cargo run --features zenoh

# Terminal 3: ROS 2 listener
source /opt/ros/humble/setup.bash
export RMW_IMPLEMENTATION=rmw_zenoh_cpp
ros2 topic echo /chatter std_msgs/msg/Int32 --qos-reliability best_effort

Actions

# Terminal 1: Router
zenohd --listen tcp/127.0.0.1:7447

# Terminal 2: Action server (Fibonacci example)
cd examples/native/rust/action-server && cargo run

# Terminal 3: Action client
cd examples/native/rust/action-client && cargo run

Zephyr action tests:

just zephyr build-all          # Build all Zephyr examples (Rust + C + C++ + XRCE)
just zephyr test               # Run zenoh E2E tests (covers actions)

Docker Development Environment

Docker provides QEMU 7.2 (from Debian bookworm) which fixes TAP networking issues present in Ubuntu 22.04’s QEMU 6.2.

# One-time setup: add yourself to docker group
sudo usermod -aG docker $USER
# Log out and back in, or run: newgrp docker

# Build and use Docker environment
just docker build              # Build nano-ros-qemu image
just docker shell              # Interactive shell
just docker test-qemu          # Run QEMU tests in container
just docker help               # Show all Docker commands

QEMU Bare-Metal Testing

Run bare-metal Cortex-M3 examples on QEMU (MPS2-AN385 machine with LAN9118 Ethernet).

# Build prerequisites
just qemu build-zenoh-pico    # Build zenoh-pico for ARM Cortex-M3
just qemu build                # Build all QEMU examples

# Non-networked tests (no setup required)
just qemu test                # Bare-metal QEMU integration tests

# Networked talker/listener test (Docker Compose - recommended)
just docker test-qemu         # Runs zenohd, talker, listener in separate containers

Docker Compose Architecture:

┌─────────────────────────────────────────────────────────────┐
│              Docker Network: 172.20.0.0/24                  │
│                                                             │
│  ┌─────────────┐  ┌─────────────┐  ┌─────────────────────┐  │
│  │   zenohd    │  │   talker    │  │      listener       │  │
│  │ 172.20.0.2  │  │ 172.20.0.10 │  │    172.20.0.11      │  │
│  │             │  │  ┌───────┐  │  │  ┌───────────────┐  │  │
│  │             │  │  │ QEMU  │  │  │  │     QEMU      │  │  │
│  │             │  │  │ ARM   │──┼──┼──│     ARM       │  │  │
│  │             │  │  │ TAP   │  │  │  │     TAP       │  │  │
│  │             │  │  └───────┘  │  │  └───────────────┘  │  │
│  └──────▲──────┘  └──────┼──────┘  └─────────┼───────────┘  │
│         └────────────────┴───────────────────┘              │
│                    NAT to zenohd                            │
└─────────────────────────────────────────────────────────────┘

Manual networked test (3 terminals, requires host TAP setup):

# Terminal 1: Setup network + start router
just qemu setup-network                    # Requires sudo
zenohd --listen tcp/0.0.0.0:7447

# Terminal 2: Talker (192.0.2.10)
./scripts/qemu/launch-mps2-an385.sh --tap tap-qemu0 \
    --binary examples/qemu-arm-baremetal/rust/talker/target/thumbv7m-none-eabi/release/qemu-bsp-talker

# Terminal 3: Listener (192.0.2.11)
./scripts/qemu/launch-mps2-an385.sh --tap tap-qemu1 \
    --binary examples/qemu-arm-baremetal/rust/listener/target/thumbv7m-none-eabi/release/qemu-bsp-listener

Run just qemu help for more options.

Zephyr Setup

just zephyr setup       # Initialize workspace at $repo/zephyr-workspace/
just zephyr test        # Run zenoh tests (native_sim uses NSOS on host loopback)
just zephyr test-xrce   # Run XRCE tests

The workspace lives at $repo/zephyr-workspace/ by default (gitignored). Set $NROS_ZEPHYR_WORKSPACE to install elsewhere. Legacy sibling installs at ../nano-ros-workspace/ are auto-detected; run ./scripts/zephyr/migrate-workspace.sh to consolidate.

See Zephyr for full details.

nros-bridge.toml — bridge configuration schema

Consumed by nros::run_from_config (Rust) and the future C/C++ mirror (nros::run_from_config in nros-cpp).

Not the orchestration nros.toml. This is the bridge runtime config (multi-RMW byte forwarding). It is named nros-bridge.toml to avoid colliding with the orchestration nros.toml (component/system build config) — different lifecycle, different schema.

The file lives next to the binary (or anywhere the program can read it) and selects:

  1. Which RMW backends to open, under what locator / domain.
  2. How those backends are wired into bridge entries that forward traffic between them.

The binary’s Cargo.toml (or target_link_libraries) still selects which backends are linked — the TOML file selects which of the linked backends are used in this run. Backend names in the file that don’t match a linked backend surface as ConfigError::OpenSession.

Top-level structure

# nros-bridge.toml — sibling of the binary
[[node]]
name    = "field"
rmw     = "zenoh"
locator = "tcp/10.0.0.1:7447"

[[node]]
name    = "control"
rmw     = "cyclonedds"
locator = "domain=0"

[[bridge]]
type      = "std_msgs/Int32"
type_hash = "RIHS01_..."
from      = { node = "field",   topic = "/sensor/raw" }
to        = { node = "control", topic = "/sensor/raw" }

Run via:

fn main() -> Result<(), nros_bridge::ConfigError> {
    nros_bridge::run_from_config("nros-bridge.toml")
    // or, via the umbrella feature: nros::run_from_config("nros-bridge.toml")
}

[[node]]

One block per backend session. The first [[node]] becomes the primary session (extra_sessions[0] in Executor::open_multi); the rest open as extras keyed by rmw.

KeyTypeDefaultNotes
namestringrequiredLogical name. Bridge entries reference this.
rmwstringrequiredCanonical backend name: "zenoh", "xrce", "cyclonedds". Must match a backend the binary linked in.
locatorstring""Backend-specific locator. Zenoh uses tcp/..., udp4/..., serial/...; DDS uses domain=<n>; XRCE uses udp4://...:port.
domain_idu320ROS domain id passed to the backend.
namespacestring"/"Default namespace applied to handles created on this node.

Locator scheme grammar (zenoh)

SchemeExampleMeaning
tcp/<host>:<p>tcp/10.0.0.1:7447TCP unicast
udp/<host>:<p>udp/10.0.0.1:7447UDP unicast
serial/<dev>serial//dev/ttyUSB0UART (host-side) or board UART (bare-metal)
tls/<host>:<p>tls/router.example.org:7447TLS over TCP (requires link-tls on backend)

DDS / XRCE follow their native locator conventions; consult each backend’s docs for the exact form.

[[bridge]]

One block per topic forwarded between two [[node]]s. Each bridge creates a RawSubscription on the source side and an EmbeddedRawPublisher on the destination side and pumps bytes once per executor tick.

KeyTypeDefaultNotes
typestringrequiredROS type name ("std_msgs/Int32"). Backends use it for liveliness / discovery.
type_hashstring""ROS 2 RIHS type hash for the typed binding. May be left empty if both sides ignore it.
fromendpointrequired{ node = "<name>", topic = "<topic>" }. The source node + topic.
toendpointrequired{ node = "<name>", topic = "<topic>" }. The destination node + topic.

Bidirectional bridges are two [[bridge]] entries — one in each direction. The runtime stamps a bridge_origin attachment field on forwarded frames and drops on receive when the origin matches the local backend, so an echo pair does not loop.

Error semantics

run_from_config returns ConfigError:

VariantCause
IoFile read failed (missing / unreadable).
ParseTOML malformed or a required field missing.
UnknownNodeA [[bridge]] references a node name no [[node]] declared.
OpenSessionExecutor::open_multi rejected a spec — usually a backend name no RMW_INIT_ENTRIES entry registered under.
BuildNodecreate_node_on failed (registry exhausted, name too long, …).
BuildEntityCreating the source subscription or destination publisher failed (typically backend rejection of the topic name / type / QoS).

Any error short-circuits the runtime; bridges that opened cleanly before the failure are not pumped.

Linked-backend matrix

Backend nameCargo dep that contributes the RMW_INIT_ENTRIES entry
zenohnros-rmw-zenoh = { ... }
xrcenros-rmw-xrce-cffi = { ... }
cycloneddsnros-rmw-cyclonedds static lib (CMake side, --whole-archive)

Add a backend to your Cargo.toml (or target_link_libraries) to make its name available in the TOML; remove it to fence off use even when the TOML file lists it (yields OpenSession at startup).

Migrating off just install-local

TL;DR. deleted just install-local, the build/install/ prefix, every install(...) rule, every Config.cmake.in template, and the find_package(NanoRos) consumption path. nano-ros is now consumed exclusively via add_subdirectory(<repo-root>) from the user’s CMakeLists.txt.

If your project was on a pre-140 checkout, this page is the one-page rewrite that gets you onto the supported shape.

Before (pre-140)

cmake_minimum_required(VERSION 3.16)
project(my_app C)

find_package(NanoRos REQUIRED CONFIG)
nano_ros_generate_interfaces(std_msgs SKIP_INSTALL)
add_executable(my_app src/main.c)
target_link_libraries(my_app PRIVATE
    std_msgs__nano_ros_c
    NanoRos::NanoRos)

Build:

# In the nano-ros checkout:
just install-local                                # populates build/install/

# In the user project:
cmake -S . -B build -DCMAKE_PREFIX_PATH=<nano-ros>/build/install
cmake --build build

After (post-140)

cmake_minimum_required(VERSION 3.22)
project(my_app C)

set(NANO_ROS_PLATFORM posix)
set(NANO_ROS_RMW     zenoh)
add_subdirectory(<path-to-nano-ros> nano_ros)

nano_ros_generate_interfaces(std_msgs SKIP_INSTALL)
add_executable(my_app src/main.c)
target_link_libraries(my_app PRIVATE
    std_msgs__nano_ros_c
    NanoRos::NanoRos)
nros_platform_link_app(my_app)

Build:

# No nano-ros-side install step. Just configure your project.
cmake -S . -B build
cmake --build build

What changed

Pre-140Post-140
find_package(NanoRos REQUIRED CONFIG)add_subdirectory(<path-to-nano-ros>) after set(NANO_ROS_PLATFORM ...) + set(NANO_ROS_RMW ...)
just install-local(deleted) — nano-ros builds in-tree per-example via Corrosion
cmake -DCMAKE_PREFIX_PATH=<nano-ros>/build/install(not needed)
nano_ros_link_platform(target) / nano_ros_link_rmw(target)(folded into NanoRos::NanoRos’s INTERFACE link libraries)
nros_freertos_compose_platform(target) / nros_threadx_compose_platform(target)nros_platform_link_app(target) — single per-app fixup hook
find_package(NrosRmwCyclonedds CONFIG REQUIRED)(auto-wired when NANO_ROS_RMW=cyclonedds)
cmake_minimum_required(VERSION 3.16)cmake_minimum_required(VERSION 3.22) (matches root nano-ros)

RTOS-specific notes

  • Zephyr. Consume nano-ros via the zephyr/ module — drop nano-ros into your west.yml, set CONFIG_NROS=y + CONFIG_NROS_RMW="zenoh" in prj.conf.
  • ESP-IDF. integrations/nano-ros/ is a component manifest — add it to your idf_component.yml.
  • NuttX. integrations/nuttx/ is a apps/external/ shim — symlink (or copy) and enable via make menuconfig.
  • PX4. integrations/px4/module-template/ is a EXTERNAL_MODULES_LOCATION template — copy out, edit the user glue, point PX4 at it.

Each shell internally does add_subdirectory(<nano-ros>) with the right NANO_ROS_PLATFORM cache var set; user-side code is the same as the C/C++ snippet above.

What you lose

  • A pre-built <prefix>/lib/libnros_c.a you could link from arbitrary external projects. Post-140, each consumer builds the staticlib in its own build tree (Corrosion handles caching via the Cargo target dir).
  • The build/install/share/nano-ros/interfaces/ bundled-interface drop. Codegen now resolves interface files via the ament index (when colcon-discovered) or directly from packages/codegen/interfaces/ inside the nano-ros checkout.
  • The cmake --install build step. There is nothing nano-ros-side to install. Your own project ships its binary; nano-ros is a static dependency.

Why this change

find_package(NanoRos) was a Debian-style “library installed once, consumed by many projects” model. The RTOS workflows that nano-ros actually targets (Zephyr west, ESP-IDF, NuttX apps/external, PX4 EXTERNAL_MODULES_LOCATION) all consume dependencies as source trees inside the user’s workspace — not from an installed prefix. The install path drifted out of sync with the cargo path more than once (UDP multicast linker error was the proof) and added ~30 s warm / ~10 min cold to every test-all run. Removing it collapses two surfaces into one.

See docs/roadmap/phase-140-install-local-rip-off.md for the full design notes and the audit table that drove the deletion.