Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/vshymanskyy/wasm2mpy

Compile WebAssembly to native MicroPython modules
https://github.com/vshymanskyy/wasm2mpy

armv7 esp32 micropython nrf52 raspberry-pi rp2040 rp2350 wasm webassembly webassembly-runtime

Last synced: 5 days ago
JSON representation

Compile WebAssembly to native MicroPython modules

Awesome Lists containing this project

README

        

# wasm2mpy

[![StandWithUkraine](https://raw.githubusercontent.com/vshymanskyy/StandWithUkraine/main/badges/StandWithUkraine.svg)](https://github.com/vshymanskyy/StandWithUkraine/blob/main/docs/README.md)
[![Build status](https://img.shields.io/github/actions/workflow/status/vshymanskyy/wasm2mpy/build.yml?branch=main&style=flat-square&logo=github&label=build)](https://github.com/vshymanskyy/wasm2mpy/actions)
[![GitHub license](https://img.shields.io/badge/license-MIT-blue?style=flat-square)](https://github.com/vshymanskyy/wasm2mpy)
[![Support vshymanskyy](https://img.shields.io/static/v1?label=support&message=%E2%9D%A4&color=%23fe8e86)](https://quicknote.io/da0a7d50-bb49-11ec-936a-6d7fd5a2de08)

`wasm2mpy` enables developers to write code in statically compiled languages and run it on MicroPython-based embedded systems (such as ESP32, Raspberry Pi Pico, STM32, and nRF52) with near-native performance. Since MicroPython is relatively slow for computationally intensive applications, `wasm2mpy` provides the tools necessary to run demanding software, such as AI models and signal processing algorithms, more efficiently.

## Status

| App \ Target | x86/x64 | armv6m | armv7m/+s/+d | esp8266[^2] | esp32 | rv32imc |
|-------------------|-----------|----------|---------|----------|------------|----------|
| ๐Ÿš€ TypeScript[^1] | โœ…โœ… | โœ… | โœ…โœ…โœ… | โš ๏ธ[^4] | โœ… | โœ… |
| ๐Ÿคฉ C++ | โœ…โœ… | โœ… | โœ…โœ…โœ… | ๐ŸŸก | โœ… | โœ… |
| ๐Ÿฆ€ Rust | โœ…โœ… | โœ… | ๐ŸŸก๐ŸŸกโœ… | โš ๏ธ[^4] | โœ… | โœ… |
| ๐Ÿค– TinyGo | โœ…โœ… | โœ… | ๐ŸŸก๐ŸŸกโœ… | โš ๏ธ[^4] | โœ… | โœ… |
| โšก Zig | โœ…โœ… | โœ… | โœ…โœ…โœ… | โš ๏ธ[^4] | โœ… | โœ… |
| โœจ Virgil | โœ…โœ… | โœ… | โœ…โœ…โœ… | ๐ŸŸก | โœ… | โœ… |
| โš™ WAT | โœ…โœ… | โœ… | โœ…โœ…โœ… | ๐ŸŸก | โœ… | โœ… |
| ๐Ÿ‡จ Coremark | โœ…โœ… | ๐Ÿšง | โœ…โœ…โœ… | ๐ŸŸก | โœ… | โœ… |

โœ… builds and runs OK
๐ŸŸก builds OK, doesn't run
๐Ÿšง work in progress

[^1]: [AssemblyScript](https://www.assemblyscript.org)
[^2]: `esp8266` requires the use of [`esp.set_native_code_location`](https://github.com/micropython/micropython/issues/14430#issuecomment-2332648018), and setting `WASM_PAGE_SIZE` to `8192` (or need to wait for [`WASM Custom Page Sizes`][1])
[^4]: not enough memory to run, need to wait for [`WASM Custom Page Sizes`][1]

## CoreMark results

- **ESP32-C3** 160MHz: `179.791`
- **STM32F405** 168MHz: `233.918`
- **ESP32** 240MHz: `228.363`
- **ESP32-S3** 240MHz: `271.573`
- **BL616** 320MHz: `344.293`
- **iMXRT1062** 600MHz: `1911.437`
- **i5-8250U** 1.6GHz: `18696.248`

## Upload and Run

Follow the [build instructions](BUILDING.md)

```sh
mpremote cp zig.mpy :lib/
mpremote exec "import zig; zig.setup()"
โšก Zig is running!
```

## Run any exported function

> [!NOTE]
> This requires adding some glue code to the runtime.
> Glue code can be auto-generated, but for now it's a manual process.

For example, `test/simple.wasm` just adds 2 numbers:

```wat
(module
(func (export "add") (param i32 i32) (result i32)
(i32.add (local.get 0) (local.get 1))
)
)
```

```log
MicroPython v1.24.0-preview.224.g6c3dc0c0b on 2024-08-22; Raspberry Pi Pico W with RP2040
Type "help()" for more information.
>>> import simple
>>> simple.add(3, 4)
7
>>> simple.add(10, 6)
16
```

## Access WASM module memory

```py
>>> import cpp
>>> cpp.setup()
๐Ÿคฉ C++ is running!
>>> cpp._memory[4096:4096+32]
bytearray(b' Blink\x00\xf0\x9f\xa4\xa9 C++ is running!\x00\n\x00\x00\x00')
>>> new_data = b"Hello C++ world"
>>> cpp._memory[4096+12:4096+12+len(new_data)] = new_data
>>> cpp.setup()
๐Ÿคฉ Hello C++ world
```

## How It Works?

The idea is very similar to [embedded-wasm-apps](https://github.com/wasm3/embedded-wasm-apps):

![image](https://github.com/wasm3/embedded-wasm-apps/blob/main/docs/how-it-works.png?raw=1)

## TODO

- [x] Support exports
- [ ] Auto-generate exports bindings
- [x] Support imports
- [x] Support memory
- [x] [Support `.a` inputs for `mpy_ld`](https://github.com/micropython/micropython/pull/15838)
- [ ] [XIP for native modules](https://github.com/micropython/micropython/pull/8381#issuecomment-2363022985)
- [ ] TBD: Support globals
- [x] Add RISC-V support: https://github.com/micropython/micropython/pull/15603
- [ ] Optimize codegen
- [ ] Use `u32` instead of `u64` for mem addresses
- [ ] Use a directly addressable `.bss` section as memory (skip indirection)
- [ ] Implement [WASM Custom Page Sizes][1]
- [ ] Implement WASM Exceptions
- [ ] Implement [WASM Stack Switching](https://github.com/WebAssembly/stack-switching/blob/main/proposals/stack-switching/Explainer.md)

## Further reading

- Discussions: [GitHub](https://github.com/orgs/micropython/discussions/15702), [Hacker News](https://news.ycombinator.com/item?id=41599579)
- [wasm2c](https://github.com/WebAssembly/wabt/blob/main/wasm2c/README.md), [w2c2](https://github.com/turbolent/w2c2)
- MicroPython [Native Modules](https://docs.micropython.org/en/latest/develop/natmod.html)
- [Feasibility of WASM for MicroPython](https://github.com/micropython/micropython/issues/15270#issuecomment-2280942885)

[1]: https://github.com/WebAssembly/custom-page-sizes/blob/main/proposals/custom-page-sizes/Overview.md