Well, it's WebAssembly running in wasm3 on an esp32 with an API for live application loading and execution over the network and an embedded-hal compatible implementation inside the runtime.
You may also be interested in:
- esp32-wasm-cli - a CLI tool for loading and running WASM appletts using this project
- rust-wasm-hal - an embedded-hal implementation with examples for use with this project
Extremely good question. The majority of ESP32 devices in my life run almost the same firmeware, with a bunch of code to manage the wifi / mqtt / file and key storage etc., then a small amount of code to interact with an actual sensor or actuator and publish that information. Also, I want to be able to deploy changes over the network without really thinking about it, and, I really like embedded rust...
The goal is to provide a common base image that provides all common functionality and re-exposes anything useful to the wasm runtime for application code, remote reloading of applications, and a Hardware Abstraction Layer (HAL) that makes it super easy to write and deploy applications. At the moment this is focussed on Rust, however, the WASM API should be broadly useful to other languages. Feel free to implement a HAL for your favourite language!
Absolutely cursed and a mostly untested work in progress. WASM and HTTP APIs need to be defined / documented, specification of applets and arguments should be expanded, and plent of components are yet to be implemented.
- WASM Drivers
- delay
- spi
- i2c (untested)
- uart
- Remote APIs
- Read/write files
- Read/write keys
- Load/unload wasm
- Start/stop wasm
- Fetch/stream application logs
- CLI
- Read/write files
- Read/write keys
- Load/unload wasm
- Start/stop wasm
- Fetch/stream application logs
- Misc
- Device individualisation (serial, p/n, etc.)
- Base image OTA
- mDNS device discovery
You'll need the ESP32 toolchains and IDF to develop the C project. If you're on linux, you can use a preconfigued docker image with docker run --rm -it -v
pwd:/work --workdir=/work --device=/dev/ttyUSB0 espressif/idf
. Non-linux users will have to follow the ESP IDF Getting Started Guide.
The wasm API is defined in wasm.c.
idf.py build
to buildidf.py flash
to flashidf.py monitor
to connect to the serial point
Running and managing applets is supported via the serial terminal (try help
for a command list) or via simple HTTP api:
- Load the binary to the device with
curl "http://ESP_IP/fs?file=/spiffs/test.wasm" -X POST --data-binary @test.wasm
- Load the task to memory with
curl "http://ESP_IP/app/cmd?cmd=load&name=test&file=/spiffs/test.wasm"
- Execute the task with
curl "http://ESP_IP/app/cmd?cmd=start"
- Optional stop the task with
curl "http://ESP_IP/app/cmd?cmd=stop"
- Unload the task from memory with
curl "http://ESP_IP/app/cmd?cmd=unload"
It is intended that this API be a) documented and b) replaced by esp32-wasm-cli
- When writing C binding functions, buffers must be resolved from offsets to addresses using
m3ApiOffsetToPtr
- You need to minimize the rustc stack size
"-C", "link-arg=-zstack-size=32768"
otherwise rustc defaults to using 1MB of stack and this won't run on devices without SPIRAM. The tradeoff here is that you may run out of stack space, so, ymmv.
Sure! Open an issue or a PR ^_^