It's quite elaborate to take in new sensors to a project. It's about:
- selecting the right part(s)
- considering availability, price, features
- reading the documentation
- ensuring drivers fit
- learning the quirks that are not necessarily documented, anywhere
This repo covers various sensors (things that measure stuff) and actuators (things that move stuff), interesting to its author, and provides reliable, maintained Rust bindings to them.
Why Embassy?
Rust provides
async/await
support, similar to what you might know from the .NET or JavaScript ecosystems. Embassy is an executor for suchasync
functions; and an ecosystem of other, related stuff.In short,
async
is GREAT!!! Making async code allows concurrency to be coded and understood as if it was linear. This is a huge boost to speed of coding and maintainability of any embedded, non-trivial project. Thus, Embassy has its place as a fundamental component of this repo.
folder | what is it? | stability | comments | |
---|---|---|---|---|
Comms | ||||
comms/ble |
Working as custom Bluetooth (BLE) service | alpha | ||
comms/extras/ble-web-app |
Web app for interfacing with the sample BLE service | -- | ||
Development kits | ||||
devkit/rgb |
RGB LED | WIP | ||
devkit/button |
Button | -- | ||
DC motor controllers (brushed) | ||||
dc-motors/drv8871 |
Controller for brushed (simple) DC motors - 6.5..45V, 3.6A max | WIP | ||
Time of flight = distance sensors | ||||
tof/vl53l5cx_uld |
Time-of-flight distance sensor; 4x4 (60Hz) or 8x8 (15Hz), up to 400cm | beta | ||
tbd. Brushless motor controllers (VESC) |
Each subfolder contains a structure similar to this:
built-in/
├── Cargo.toml # Cargo build file
├── Makefile.dev # dev helper (optional)
├── README.md # you-know
├── [WIRING.md] # wiring of peripherals to MCU
├── [pins.toml] # configuration of MCU pins
├── build.rs # additional build details
├── examples
│ └── button.rs # examples you can run (if HW is properly set up)
├── [src] # library code (not every folder has it)
└── set-target.sh # (link to a) tool to switch between MCU's
By keeping the folder structure similar, getting to speed with a new kind of sensor/actuator should be as swift as possible. The steps should be familiar.
The repo focuses on ESP32 series MCU's, but this is mainly so that the stated support remains maintained and tested. If you are ready to take on maintenance for other MCU's, let the author know. The Rust / Embassy ecosystem, as such, provides the possibility to keep the repo very MCU independent. That alone is great. 🎉🎉🎈🎈
MCU | dev board | support level | notes |
---|---|---|---|
ESP32‑C3 | ESP32‑C3-DevKitC-02 | used regularly | See below |
ESP32‑C6 | ESP32-C6-DevKitM-01 | used regularly | No problems! |
Note: The repo does debug logs using defmt which means JTAG support is essential, for any development work.
-
The chip does provide a built in USB/JTAG circuitry, but the suggested devkit doesn't have a connector for it. You can solder one, or (perhaps a better approach?), attach a cable to breakout pins on a breadboard.
tbd. image pending
-
I2C functionality - and possibly other time sensitive functionality - is known to suffer from a JTAG specification issue that affects
probe-rs
's ability to interface with the chip. If you need long I2C communications, ESP32-C6 is the better chip to target.Note: The problem only involves debugging/logging over JTAG. You can quite well do long I2C transactions in production.
Projects such as esp-hal
use Cargo features mechanism for selecting the target MCU (e.g. esp32c3
vs. esp32c6
). While this seems to be the norm, this repo deviates from it.
The main reason is philosophical. Features should be used for - well - features of the code base, and they should be cumulative (by the Rust recommendations). MCU features, on the other hand, are exclusive.
So what do we do?
There are .config/cargo.toml
files under certain folders in the repo. These decide the MCU chip for the particular folder and its subfolders.
This means that you can, for example:
- use
dc-motors
code, targeting ESP32-C3, while.. - using
tof
code, targeting ESP32-C6
This at least matches the way the author works; there's always a particular breadboard connected with such software.
To change the folder's MCU target,
$ cd dc-motors
$ $ ./set-target.sh
1) esp32c3
2) esp32c6
Pick your target: 1
Going to edit:
- .cargo/config.toml
- ./drv8871/Cargo.toml
The edit is safe. You can see the changes made by 'git diff' (and revert by 'git restore --staged .cargo/config.toml ./drv8871/Cargo.toml').
Proceed? (Y/n) Y
Files '.cargo/config.toml' and './drv8871/Cargo.toml' now using:
MCU: esp32c3
TARGET: riscv32imc-unknown-none-elf
Please 'cargo build' or 'cargo run', as usual.
You can see the change by git diff
, i.e. the selected (default) MCU type gets stored in the version control.
This approach is a bit intrusive, but having used it for a while the author does prefer it over the features
approach, at least for the case of the Zoo (which is full of examples). For publishing a library, using the feature mechanism is likely preferrable (or only working solution).
- One of the dev kits mentioned above
- The necessary sensors (see particular subfolder)
- Breadboard
- Wires
Each sensor's subsection has a WIRING.md
that shows suggested wiring. You can change the pin layout by editing pins.toml
files found in each subfolder.
The repo can be used in two ways, depending on where probe-rs
is located:
-
probe-rs
remotelyThis is the recommended approach, pictured above. You have a separate computer for interfacing with the MCU, and connect to its
probe-rs
tool overssh
. Instructions for installation are provided below. -
probe-rs
over USB/IP (for WSL2)This is useful if you have a single (Windows) laptop. Here, USB/IP is used to bring the host's USB port to the VM (WSL2), and
probe-rs
runs within your development VM.
Since the project files simply refer to probe-rs
, either of these approaches works. With the WSL2 approach, you sacrifice galvanic isolation, i.e. your electronics are directly connected to the laptop. Consider using a USB Isolator. |2|
There are no instructions on setting up the USB/IP. Ask the author...
Let's get back to the default setup.
- Code editing happens on a host (Mac), using an IDE (Rust Rover)
- Compilation happens in a virtual machine (using Multipass for this); the whole Rust and embedded toolchain only needs to be installed within here.
- Hardware devices (MCU + sensors) are connected to a Raspberry Pi that runs
ssh
and hasprobe-rs
installed.
Note: Due to using WLAN, the software development and hardware setups are fully air-gapped from each other.
Note: Originally, the author used USB/IP for the setup. This, however, leads to very slow flash times.
It's maintained by Canonical, but on somewhat limited resources. Some other virtualization product might suit you better, if you are a company. Also, it doesn't have USB pass-through but since we anyways prefer physical isolation from the MCU, that's not really an issue.
Note! Since 1.14.0, the author has found Multipass to be somewhat more fragile than before! This is, however, under control, and should not keep you from using the solution. More info is available in the `mp´ repo.
For running Multipass on Windows, please note that only Windows Pro has Hyper-V hypervisor support. The author hasn't run Multipass on Windows (since WSL2 is there) but if he would, he'd pick a Pro license. You can run Multipass with Windows Home, but that involves VirtualBox 👎. Unless.. hint, hint#2
As you can see, there are some different ways to set up the toolchain.
What's important to take home from this is that
probe-rs
is the tool of choice for interacting with your development kit. The build scripts don't care whether such command leads to USB/IP or a ssh
-bridged installation.
Good luck! ☀️☀️☀️
These two repos help you to set up the environment discussed above:
-
Helps you create a Multipass VM that has the tools (Rust, Cargo,
probe-rs-remote
) used in the ZOO.This doesn't cover exactly all the tools, but gives a solid foundation. If additional tools are needed, they are mentioned in the subfolder
README
s (for example, thetof/vl53l5cx_uld
needsclang
andbindgen
CLI to be separately installed). -
The
probe-rs
overssh
bridging.Follow these instructions to set up your Raspberry Pi (or other such secondary computer) that runs
probe-rs
.Note: The front side setup is not needed if you use
mp
rust+emb/prep.sh
- it's already covered there.
Visit the subfolders, pick one you'd like to try. The instructions are within their particular README.md
.
Please please PLEASE give feedback on the GitHub!
Developed on:
macOS 15.2
Multipass 1.15.0
ESP32-C3-Devkit-C02 (revision 0.4)
ESP32-C6-DevKitM-1