Ecosyste.ms: Awesome

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

Awesome Lists | Featured Topics | Projects

https://github.com/dtolnay/watt

Runtime for executing procedural macros as WebAssembly
https://github.com/dtolnay/watt

Last synced: 6 days ago
JSON representation

Runtime for executing procedural macros as WebAssembly

Awesome Lists containing this project

README

        

Watt
====

[github](https://github.com/dtolnay/watt)
[crates.io](https://crates.io/crates/watt)
[docs.rs](https://docs.rs/watt)

Watt is a runtime for executing Rust procedural macros compiled as WebAssembly.

```toml
[dependencies]
watt = "0.5"
```

*Compiler support: requires rustc 1.42+*


## Rationale

- **Faster compilation.** By compiling macros ahead-of-time to Wasm, we
save all downstream users of the macro from having to compile the macro logic
or its dependencies themselves.

Instead, what they compile is a small self-contained Wasm runtime (~3 seconds,
shared by all macros) and a tiny proc macro shim for each macro crate to hand
off Wasm bytecode into the Watt runtime (~0.3 seconds per proc-macro crate you
depend on). This is much less than the 20+ seconds it can take to compile
complex procedural macros and their dependencies.

- **Isolation.** The Watt runtime is 100% safe code with zero dependencies.
While running in this environment, a macro's *only possible interaction with
the world* is limited to consuming tokens and producing tokens. This is true
regardless of how much unsafe code the macro itself might contain! Modulo bugs
in the Rust compiler or standard library, it is impossible for a macro to do
anything other than shuffle tokens around.

- **Determinism.** From a build system point of view, a macro backed by
Wasm has the advantage that it can be treated as a purely deterministic
function from input to output. There is no possibility of implicit
dependencies, such as via the filesystem, which aren't visible to or taken
into account by the build system.


## Getting started

Start by implementing and testing your proc macro as you normally would, using
whatever dependencies you want (syn, quote, etc). You will end up with something
that looks like:

```rust
use proc_macro::TokenStream;

#[proc_macro]
pub fn the_macro(input: TokenStream) -> TokenStream {
/* ... */
}
```

`#[proc_macro_derive]` and `#[proc_macro_attribute]` are supported as well;
everything is analogous to what will be shown here for `#[proc_macro]`.

When your macro is ready, there are just a few changes we need to make to the
signature and the Cargo.toml. In your lib.rs, change each of your macro entry
points to a no\_mangle extern "C" function, and change the TokenStream in the
signature from proc\_macro to proc\_macro2.

It will look like:

```rust
use proc_macro2::TokenStream;

#[no_mangle]
pub extern "C" fn the_macro(input: TokenStream) -> TokenStream {
/* same as before */
}
```

Now in your macro's Cargo.toml which used to contain this:

```toml
# my_macros/Cargo.toml

[lib]
proc-macro = true
```

change it instead to say:

```toml
[lib]
crate-type = ["cdylib"]

[patch.crates-io]
proc-macro2 = { git = "https://github.com/dtolnay/watt" }
```

This crate will be the binary that we compile to Wasm. Compile it by running:

```console
$ cargo build --release --target wasm32-unknown-unknown
```

Next we need to make a small proc-macro shim crate to hand off the compiled Wasm
bytes into the Watt runtime. It's fine to give this the same crate name as the
previous crate, since the other one won't be getting published to crates.io. In
a new Cargo.toml, put:

```toml
[lib]
proc-macro = true

[dependencies]
watt = "0.5"
```

And in its src/lib.rs, define real proc macros corresponding to each of the ones
previously defined as no\_mangle extern "C" functions in the other crate:

```rust
use proc_macro::TokenStream;
use watt::WasmMacro;

static MACRO: WasmMacro = WasmMacro::new(WASM);
static WASM: &[u8] = include_bytes!("my_macros.wasm");

#[proc_macro]
pub fn the_macro(input: TokenStream) -> TokenStream {
MACRO.proc_macro("the_macro", input)
}
```

Finally, copy the compiled Wasm binary from
target/wasm32-unknown-unknown/release/my_macros.wasm under your implementation
crate, to the src directory of your shim crate, and it's ready to publish!


## Remaining work

- **Performance.** Watt compiles pretty fast, but so far I have not put any
effort toward optimizing the runtime. That means macro expansion can
potentially take longer than with a natively compiled proc macro.

Note that the performance overhead of the Wasm environment is partially offset
by the fact that our proc macros are compiled to Wasm in release mode, so
downstream `cargo build` will be running a release-mode macro when it would
have been running debug-mode for a traditional proc macro.

A neat approach would be to provide some kind of `cargo install watt-runtime`
which installs an optimized Wasm runtime locally, which the Watt crate can
detect and hand off code to if available. That way we avoid running things in
a debug-mode runtime altogether. The experimental beginnings of this can be
found under the [jit/](jit) directory.

- **Tooling.** The getting started section shows there are a lot of steps
to building a macro for Watt, and a pretty hacky patching in of proc-macro2.
Ideally this would all be more straightforward, including easy tooling for
doing reproducible builds of the Wasm artifact for confirming that it was
indeed compiled from the publicly available sources.

- **RFCs.** The advantages of fast compile time, isolation, and determinism
may make it worthwhile to build first-class support for Wasm proc macros into
rustc and Cargo. The toolchain could ship its own high performance Wasm
runtime, which is an even better outcome than Watt because that runtime can be
heavily optimized and consumers of macros don't need to compile it.


## This can't be real

To assist in convincing you that this is real, [here is serde\_derive compiled
to Wasm][wa-serde-derive]. It was compiled from the commit
[serde-rs/serde@1afae183][commit]. Feel free to try it out as:

[wa-serde-derive]: https://crates.io/crates/wa-serde-derive
[commit]: https://github.com/serde-rs/serde/commit/1afae183b06ffe47d054c796e68630752414cf9e

```rust
// [dependencies]
// serde = "1.0"
// serde_json = "1.0"
// wa-serde-derive = "0.1"

use wa_serde_derive::Serialize;

#[derive(Serialize)]
struct Watt {
msg: &'static str,
}

fn main() {
let w = Watt { msg: "hello from wasm!" };
println!("{}", serde_json::to_string(&w).unwrap());
}
```


## Acknowledgements

The current underlying Wasm runtime is a fork of the [Rust-WASM] project by
Yoann Blein and Hugo Guiroux, a simple and spec-compliant WebAssembly
interpreter.

[Rust-WASM]: https://github.com/yblein/rust-wasm


#### License


Everything outside of the `runtime` directory is licensed under either of Apache License, Version 2.0 or MIT license at your option. The `runtime` directory is
licensed under the ISC license.



Unless you explicitly state otherwise, any contribution intentionally submitted
for inclusion in this crate by you, as defined in the Apache-2.0 license, shall
be dual licensed as above, without any additional terms or conditions.