Stream: wasmtime

Topic: Async ABI implementation overview


view this post on Zulip Joel Dice (Dec 09 2024 at 23:51):

As I mentioned at last week's meeting, I'm planning to do an overview of the implementation work described in the Async RFC I posted a couple of months ago. It's currently scheduled for .

I've sent an invite to those who have already expressed interest. Please DM me your email address if you want me to add it to the invite.

Add support for the Component Model Async ABI to wasm-tools, wit-bindgen, and wasmtime. Rendered RFC

view this post on Zulip Joel Dice (Dec 12 2024 at 19:52):

Here's the recording: YouTube - - YouTube
and document: https://hackmd.io/LQhcwqb9QiyH-YNMyikMHA?view

or

view this post on Zulip Pat Hickey (Dec 12 2024 at 19:58):

Thanks I had a conflict with that meeting time so I really appreciate the recording!

view this post on Zulip raskyld (Dec 12 2024 at 20:45):

I just wanted to say thanks again! That was very informative!
The only things I haven't fully understood, I expect to understand them by reading the RFCs and checking the PRs :praise: !

view this post on Zulip Dave Bakker (badeend) (Dec 12 2024 at 21:52):

Regarding:

This reminds me of an old PR of mine where I ran into nearly the same problem; namely that WASI pollables need temporary mutable access to the ResourceTable, only for the duration of the poll method, but not in between.
The referenced PR bounced back-and-forth between different solutions, and ultimately didn't land anything. Anyhow, one of the solutions was to have a customWasiFuture trait which has an additional &mut Store parameter on its poll method. Also, the WasiFuture trait was auto-implemented for all regular Futures. Of course this bifurcates the Futures ecosystem. But just throwing it out there

Prior discussion: https://bytecodealliance.zulipchat.com/#narrow/stream/217126-wasmtime/topic/Change.20Subscribe.20trait Renamed the existing Pollable struct to PollableResource Reimplemented wasi...

view this post on Zulip Joel Dice (Dec 12 2024 at 21:53):

Yeah, I was thinking of something similar. What I couldn't figure out was how to interoperate ergonomically with regular Futures and async/await.

view this post on Zulip Joel Dice (Dec 12 2024 at 21:54):

The blanket WasiFuture implementation sounds interesting; hadn't thought of that.

view this post on Zulip Joel Dice (Dec 12 2024 at 22:00):

Here's an example of using Promise and PromisesUnordered (akin to futures::FuturesUnordered, used to multiplex multiple Promises concurrently) to juggle multiple concurrent streams, futures, and export calls in a somewhat realistic wasi-http scenario: https://github.com/bytecodealliance/wasmtime/pull/9582/files#diff-3c649a56157956a049a3450ccba943197b5b50dde10bcd1917a6c236bd7329ecR929-R1121
It's not super idiomatic, but it's not horrible either, IMHO.

This adds support for loading, compiling, linking, and running components which use the Async ABI along with the stream, future, and error-context types. It also adds support for generating host bi...

view this post on Zulip Joel Dice (Dec 12 2024 at 22:02):

And here's a simpler example where we just start three concurrent calls to the same exported function and wait for them all to complete: https://github.com/bytecodealliance/wasmtime/pull/9582/files#diff-3c649a56157956a049a3450ccba943197b5b50dde10bcd1917a6c236bd7329ecR481-R492

This adds support for loading, compiling, linking, and running components which use the Async ABI along with the stream, future, and error-context types. It also adds support for generating host bi...

view this post on Zulip Dave Bakker (badeend) (Dec 12 2024 at 22:04):

BTW, Those links don't link to anything specific for me. They both load the entire diff, from the top

view this post on Zulip Joel Dice (Dec 12 2024 at 22:05):

Yeah, maybe the diff is too huge for GH to allow deep linking; just fixed the first one; will fix the second

view this post on Zulip Dave Bakker (badeend) (Dec 12 2024 at 22:20):

Regarding the async/await ergonomics;
If I summarize correctly; for the regular non-async case, passing the store as a mutable reference is fine:

linker.func_wrap(|store: &mut Store| {
    blabla1(store);
    blabla2(store);
});

(heavily pseudo coding here :P )

view this post on Zulip Dave Bakker (badeend) (Dec 12 2024 at 22:21):

The problem in the async variant is that the async method should get access to the Store somehow, but isn't allowed to hold it across await boundaries. Ie. this is not ok:

linker.func_wrap_concurrent(|store: &mut Store| async {
    blabla1(store);
    something.await;
    blabla2(store); // `store` shouldn't be used across await-points!
});

view this post on Zulip Joel Dice (Dec 12 2024 at 22:23):

Right; non-awaiting host functions are easy: they can have exclusive access to the store since they will return immediately (unless it goes off and does a long computation, I guess; in that case it should spawn a task on Tokio's blocking thread pool and await that). And if a host function needs to await, we can't let it hold exclusive access to the store that whole time since it prevents anything else from happening for that component instance.

view this post on Zulip Dave Bakker (badeend) (Dec 12 2024 at 22:24):

So, instead of passing the store directly, maybe we can pass kind of "accessor" (which _can_ be used across awaits) which yields temporary accesses to the store (which can _not_ be used across await points).

linker.func_wrap_concurrent(|store_acc: &mut StoreAccessor| async {
    let mut guard: StoreGuard = store_acc.get();
    blabla1(&mut guard);
    something.await;
    blabla2(&mut guard); // `guard` can not be used across await points
})

similar-ish to Mutex & MutexGuard? With the exception that we don't actually need any locking here

view this post on Zulip Joel Dice (Dec 12 2024 at 22:27):

How would StoreAccessor be implemented? (and keep in mind we have to include a <T> where T might not be 'static, so that makes things more interesting)

view this post on Zulip Joel Dice (Dec 12 2024 at 22:28):

I recall exploring something like that myself, but I think it was a dead-end; could be that I missed something, though.

view this post on Zulip Joel Dice (Dec 12 2024 at 22:29):

I also thought about using thread- or task-locals, but that was also a dead-end due to lifetime-related issues.

view this post on Zulip Joel Dice (Dec 12 2024 at 22:30):

Under the hood, we can pass a *mut dyn VMStore around with impunity, but wrapping a safe, correct API around that seems impossible given the lifetime issues

view this post on Zulip Joel Dice (Dec 12 2024 at 22:31):

I even considered experimenting with this unstable feature, but again I ran into lifetime issues (can't use dyn Any for non-'static types)

Feature gate: #![feature(context_ext)] This is a tracking issue for allowing std::task::Context to carry arbitrary extension data. Public API impl Context { fn ext(&mut self) -> &mut dyn Any; } imp...

view this post on Zulip Joel Dice (Dec 12 2024 at 22:33):

If we were to start adding T: 'static bounds everywhere, we'd have a lot more options, but that's a tough sell.

view this post on Zulip Joel Dice (Dec 12 2024 at 22:35):

And maybe it wouldn't help much anyway given that we're still talking about raw pointers and all the attendant dangers.

view this post on Zulip Dave Bakker (badeend) (Dec 12 2024 at 22:35):

I was thinking in terms of some kind of task-local, and indeed hadn't thought about non-static T's :)


Last updated: Jan 24 2025 at 00:11 UTC