Stream: wasi

Topic: Slices and strs in Wiggle bindings


view this post on Zulip acfoltzer (May 07 2020 at 21:12):

tl;dr We're hoping to find safer ways to access slices and strs in Wiggle trait implementations, particularly in the presence of invalidating operations like yielding. But we don't want to mess up what's already working well for wasi-common.

Hey folks, this week I've been looking at how to start rolling out Wiggle-generated code to more hostcall implementations in our Wasm execution daemon. I've run into a couple points of additional friction vs writing hostcall implementations manually. I'm hoping y'all might have some ideas for how we might be able to reduce this friction?

The first problem is one that I've actually managed to sidestep for now: Lucet supports a family of yield functions on its VM context object which are quite useful for integrating Lucet execution with async Rust runtimes. For safety, these require an exclusive &mut reference to the VM context, but Wiggle only provides a shared & reference to the user-provided context type. I've sidestepped this issue for the moment by converting async calls within hostcalls to blocking calls using futures::executor::block_on. This works for now, but is not ideal because it prevents those blocked threads from handling other async work.

The bigger problem at the moment is that it's currently quite painstaking to get access to slices and strs in Wiggle implementations. It requires using the honor system with GuestBorrows, plus explicit unsafe to dereference a raw pointer. My understanding is that this is pretty uncommon in wasi-common compared to dealing with [pointers to] scalars, enums, and structs, but in our Wasm daemon the situation is practically the reverse: we're dealing with slices and strings in almost all of our hostcalls. If it was just a few functions in special situations, I would feel better about rolling this out with appropriate signposting and warnings, but right now we're holding off until the common case no longer requires this manually-checked reasoning.

In Lucet, we address this with dynamic borrow checking on heap accessors. This works out nicely, because any operations on the VM context that might invalidate the view of the heap such as yielding or growing the heap (a safe callback interface is still a TODO for us; we don't use callbacks in our daemon) require exclusive &mut access, and thus the borrow checker prevents the Refs and RefMuts from staying live during these operations. The hostcall implementor is then free to reacquire updated views of these resources on the other side of the invalidating operation.

I tried noodling around a bit adding RefCells to my Wiggle context, and to my Wiggle GuestMemory implementation, in order to achieve a similar effect, but I'm running into a couple problems:

  1. GuestPtrs contain shared references to the GuestMemory, and are provided to the implementation as function arguments. This means that even if we change the shared & reference to a dynamically-checked Ref, we have no way of re-acquiring that borrow after relinquishing it for an invalidating operation that requires an exclusive &mut or RefMut reference.

  2. I haven't found a workable solution for having the GuestMemory and the user-provided Wiggle context share borrow state. Having a single object produce both our views into memory, and mediate access to invalidating operations is how we get machine-checked assurance that we're not changing the world out from under ourselves. So naturally, I tried to make these values share a borrow state through RefCell in order to recover these semantics.

    What I found was that it's very difficult to nest RefCells in a way that allows functions to return a Ref or RefMut from the inner cell. While you can map over a Ref, the closure you map with must return a &U, it can't return Ref<'_, U>. This makes sense because of the ownership logic of the dynamic borrows involved here, and I'm guessing could be worked around with a custom variant of RefCell, but :grimacing:

So I'm stuck wishing for three things, and realizing they're probably incompatible with the current Wiggle API:

  1. I want to have my GuestMemory and user-provided context be the same object, with no indirection through RefCell, RwLock, etc.
  2. I want to have exclusive &mut access to that object in the body of my Wiggle trait implementations.
  3. I want to be able to either get fresh GuestPtrs on the other side of an invalidating operation, or have them be safe to hold across such an operation.

Do y'all have ideas for how we might be able to accommodate these wishes without having to tear down what we've already built and what works well for wasi-common?

view this post on Zulip Alex Crichton (May 07 2020 at 21:19):

@acfoltzer FWIW I believe I'm "at fault" for most of the current restrictions in wiggle, but that was mostly just me rationalizing Rust's memory model with executing arbitrary wasm code. Given that there's not a huge amount of wiggle (hah) room without larger changes.

view this post on Zulip Alex Crichton (May 07 2020 at 21:20):

For example if you've got a &mut T that points into wasm memory, then it's memory unsafe to just call back into the wasm. Nothing about GuestBorrows will help fix that unsoundness. This could be fixable, however, if you've got a larger system which dynamically prevents reentering the wasm once you have a &mut T

view this post on Zulip Alex Crichton (May 07 2020 at 21:20):

(and then you'd also dynamic-borrow-check to make sure the &mut T didn't overlap with anything)

view this post on Zulip acfoltzer (May 07 2020 at 21:21):

Yep, and that's exactly what we have in our daemon

view this post on Zulip acfoltzer (May 07 2020 at 21:21):

it seems like a fundamentally different approach to what Wiggle is currently doing, though, which led to this wall of text :upside_down:

view this post on Zulip Alex Crichton (May 07 2020 at 21:22):

It sounds like some of these may be solvable with codegen tweaks? My intention was that wiggle would generate 100% safe things, but if y'all have extra checking in place that wiggle isn't aware of then it would be pushing the unsafe boundary outwards

view this post on Zulip Alex Crichton (May 07 2020 at 21:22):

but yeah if y'all are statically preventing reentering wasm then that changes a whole lot about wiggle

view this post on Zulip acfoltzer (May 07 2020 at 21:22):

yeah, it's a bit of a hack but it works pretty nicely in practice

view this post on Zulip Alex Crichton (May 07 2020 at 21:23):

from a bindings-generator point of view which is purely consuming wasmtime APIs, it's impossible to provide that guarantee statically though

view this post on Zulip Alex Crichton (May 07 2020 at 21:23):

e.g. if other users of wiggle come along

view this post on Zulip acfoltzer (May 07 2020 at 21:23):

it does require some pretty gross stuff under the hood though, most notoriously https://github.com/bytecodealliance/lucet/blob/master/lucet-runtime/lucet-runtime-internals/src/vmctx.rs#L177

Lucet, the Sandboxing WebAssembly Compiler. Contribute to bytecodealliance/lucet development by creating an account on GitHub.

view this post on Zulip acfoltzer (May 07 2020 at 21:25):

we might have to wait for Pat to come back from vacation to really get to the point of making any decisions, because he's much more familiar with the API constraints you have on the wasmtime side

view this post on Zulip Alex Crichton (May 07 2020 at 21:26):

I may have also just mistaken the purpose of wiggle somewhat

view this post on Zulip acfoltzer (May 07 2020 at 21:26):

one of my fallback ideas though is to see about factoring the witx consumption and reasoning about Rust host-side types into a common module, and just doing different codegen for Lucet

view this post on Zulip Alex Crichton (May 07 2020 at 21:26):

I was viewing it from a lens of "anyone who's binding *.witx will want to use this to bind host functions"

view this post on Zulip Alex Crichton (May 07 2020 at 21:26):

but if it's instead "fastly is binding *.witx with these extra assumptions" then wiggle should look very different

view this post on Zulip Alex Crichton (May 07 2020 at 21:27):

it sounds like y'all are making assumptions about the wasm execution which other projects may not always be able to make

view this post on Zulip acfoltzer (May 07 2020 at 21:27):

hmm, do you have an example of that?

view this post on Zulip Alex Crichton (May 07 2020 at 21:27):

which changes quite a bit how nice a bindings library can be

view this post on Zulip Alex Crichton (May 07 2020 at 21:27):

oh so I don't think we have any other examples of wiggle in use, but I'm imagining a use case which allows reentrancy

view this post on Zulip Alex Crichton (May 07 2020 at 21:27):

e.g. you call into the host and then it calls you back during that host call

view this post on Zulip Alex Crichton (May 07 2020 at 21:28):

the possibility of that happening is a fundamental assumption of wiggle right now, but it sounds like y'all prevent that via other means

view this post on Zulip Alex Crichton (May 07 2020 at 21:28):

but no one's getting too too much use out of wiggle right now, so we can also just document how it's safely used

view this post on Zulip Alex Crichton (May 07 2020 at 21:29):

and change it to assume no reentrancy

view this post on Zulip acfoltzer (May 07 2020 at 21:29):

oh, you mean that some other thread in the host calls into the same Wasm module?

view this post on Zulip acfoltzer (May 07 2020 at 21:29):

rather than the hostcall implementation doing the callback on the same thread?

view this post on Zulip Alex Crichton (May 07 2020 at 21:29):

nah not another thread, but the same thread itself

view this post on Zulip Alex Crichton (May 07 2020 at 21:29):

like some funky syscall where you call host.call_me_back(my_fn_pointer) and then the host, during that function call, calls my_fn_pointer

view this post on Zulip Alex Crichton (May 07 2020 at 21:30):

so the native stack looks like [top] -> wasm code -> [hostcall] -> wasm code -> ...

view this post on Zulip Alex Crichton (May 07 2020 at 21:30):

where any borrows that [hostcall] has active are immediately invalidated by executing wasm code

view this post on Zulip acfoltzer (May 07 2020 at 21:30):

right. our assumption is that you can do that, but only through a function that takes &mut Vmctx so that outstanding heap references must be dropped

view this post on Zulip acfoltzer (May 07 2020 at 21:31):

(that function doesn't exist yet in our Vmctx interface though; yield is a better example right now)

view this post on Zulip Alex Crichton (May 07 2020 at 21:31):

ah ok, that's sort of the same thing though, the risk of allowing reentering wasm during a hostcall is active borrows

view this post on Zulip acfoltzer (May 07 2020 at 21:31):

exactly

view this post on Zulip Alex Crichton (May 07 2020 at 21:31):

but y'all are dynamically guaranteeing there are no active borrows with Rust's lifetime system (I think?)

view this post on Zulip Alex Crichton (May 07 2020 at 21:32):

that assumption wasn't baked into wiggle b/c that's an application-specific restriction you have

view this post on Zulip Alex Crichton (May 07 2020 at 21:32):

but wiggle could of course be rewritten or updated with that restriction

view this post on Zulip Alex Crichton (May 07 2020 at 21:32):

and simply say "applications gotta have something like this"

view this post on Zulip acfoltzer (May 07 2020 at 21:32):

that's right. Rust will complain if the heap view Ref<'_, [u8]> is live when you try to use &mut Vmctx

view this post on Zulip Jakub Konka (May 07 2020 at 21:32):

@acfoltzer Would it be possible to build around wiggle, here wiggle staying largely the same and you providing something more specific like @Alex Crichton is pointing out?

view this post on Zulip acfoltzer (May 07 2020 at 21:33):

that is what I was trying to do with the RefCell experiments I describe above

view this post on Zulip acfoltzer (May 07 2020 at 21:33):

maybe I just got myself backed into a corner with it, but I couldn't figure out how to make it work out

view this post on Zulip Jakub Konka (May 07 2020 at 21:33):

Oh and for the record, although perhaps I misunderstood you, we do quite a few calls to get a string out of guest in wasi-common

view this post on Zulip acfoltzer (May 07 2020 at 21:34):

ah, Pat was describing it as a less common case in wasi-common than it is in our daemon

view this post on Zulip Jakub Konka (May 07 2020 at 21:34):

oh it may be so

view this post on Zulip Alex Crichton (May 07 2020 at 21:34):

(also to clarify I didn't really understand what you were mentioning about RefCell and things above)

view this post on Zulip Jakub Konka (May 07 2020 at 21:34):

in wasi-common any syscall that includes "path" requires fetching a string

view this post on Zulip acfoltzer (May 07 2020 at 21:36):

Alex Crichton said:

(also to clarify I didn't really understand what you were mentioning about RefCell and things above)

hah sorry. that was a bit more of a brain dump than I had intended. I was basically trying to recover the connection between the lifetimes of the active borrows with the lifetime of the user-provided context type

view this post on Zulip Jakub Konka (May 07 2020 at 21:36):

and as far & vs &mut I thought that was the desired requirement so that we can easily stick any context object in an Rc/Arc

view this post on Zulip acfoltzer (May 07 2020 at 21:38):

ahh, that's good to know. we do not have that as a restriction in Lucet, so we've always had the hostcall context as a &mut

view this post on Zulip Alex Crichton (May 07 2020 at 21:39):

But also to be clear, this is again sort of just shaping wiggle as a library that makes minimal assumptions

view this post on Zulip acfoltzer (May 07 2020 at 21:39):

yeah, that makes sense

view this post on Zulip Alex Crichton (May 07 2020 at 21:40):

Wiggle right now is designed to assume a maximally malicious host which wants to stress everything

view this post on Zulip Alex Crichton (May 07 2020 at 21:40):

but it still wants to provide a 100% safe interface

view this post on Zulip Alex Crichton (May 07 2020 at 21:40):

so the trivial reason for requiring & for user contexts is that wasm can call the host twice on the stack

view this post on Zulip Alex Crichton (May 07 2020 at 21:40):

which makes it sort of like TLS, so you can only get a shared reference

view this post on Zulip Alex Crichton (May 07 2020 at 21:40):

once you start making other assumptions though these requirements go away

view this post on Zulip Alex Crichton (May 07 2020 at 21:41):

it sounds like y'all are making enough assumptions that wiggle is just getting in the way

view this post on Zulip Alex Crichton (May 07 2020 at 21:41):

I think the assumptions you're making though might be quite difficult for other applications to make as well, so I'm not sure if there's a nice reusable binding layer to share unfortunately

view this post on Zulip Jakub Konka (May 07 2020 at 21:42):

I think I agree with @Alex Crichton on this

view this post on Zulip Jakub Konka (May 07 2020 at 21:42):

Perhaps we could figure out if we could make wiggle-like interface pluggable into wasi-common rather than specifically requiring it?

view this post on Zulip acfoltzer (May 07 2020 at 21:42):

yeah, and it's possible this may just come down to us having faulty assumptions about the safety of our system. for example if you have a stack with [Rust] -> [Wasm] -> [Hostcall] -> [Wasm] -> [Hostcall] then you have two &mut Vmctx on the stack pointing to the same place. our argument has been this is okay because the second is entirely contained within the first, but maybe that's naïve

view this post on Zulip Jakub Konka (May 07 2020 at 21:42):

Not sure if that makes sense but then you guys could have a more specialised version of it and everything would be fine

view this post on Zulip acfoltzer (May 07 2020 at 21:43):

yeah, I'm leaning toward something like that

view this post on Zulip Alex Crichton (May 07 2020 at 21:44):

@acfoltzer I think that's a really fine line to walk there

view this post on Zulip Alex Crichton (May 07 2020 at 21:45):

I wouldn't say with certainty that it's safe or unsafe

view this post on Zulip acfoltzer (May 07 2020 at 21:45):

I'm glad we agree :laughing:

view this post on Zulip Alex Crichton (May 07 2020 at 21:45):

You can have two &mut T pointers on the stack that point to the same thing, but in safe rust rustc/llvm can see the pointers getting passed around

view this post on Zulip Alex Crichton (May 07 2020 at 21:45):

the unsafety would happen I believe if they can't see the pointers passed around

view this post on Zulip Alex Crichton (May 07 2020 at 21:46):

like:

let x: &mut Context = ...;
let y = x.field1;
foobar(); // may read/mutate `x`
assert_eq!(x.field1, y);

view this post on Zulip Alex Crichton (May 07 2020 at 21:46):

if rustc sees you pass x to foobar, then it knows it has to reload x.field1 after the call

view this post on Zulip Alex Crichton (May 07 2020 at 21:46):

if it doesn't see you pass x to foobar then it could assume it's impossible to change, so there's no need to reload x.field1

view this post on Zulip Alex Crichton (May 07 2020 at 21:46):

but if x.field1 was mutated in foobar then that's an issue

view this post on Zulip Alex Crichton (May 07 2020 at 21:47):

clearly if you do foobar(x) rustc/llvm will see through all that, but if it's actually some_wasm_call() it's not clear if LLVM/rustc can see enough to assume that pointers might change

view this post on Zulip acfoltzer (May 07 2020 at 21:47):

that makes sense, and certainly makes our current interface unsafe. but a callback interface that requires a &mut Vmctx would let Rust see the pointer being used

view this post on Zulip Alex Crichton (May 07 2020 at 21:48):

these are also sort of just vague thoughts though, I'm definitely not like on the unsafe guidelines wg or anything like that

view this post on Zulip acfoltzer (May 07 2020 at 21:49):

yeah, I'm glad to see that your thinking resembles mine quite a bit, though

view this post on Zulip acfoltzer (May 07 2020 at 21:50):

fine line indeed

view this post on Zulip acfoltzer (May 07 2020 at 21:51):

okay, well thank you for the extra context about the whys of the current interface. I think I'll have to wait for Pat to get back and chat with him about what our strategy should be. right now I'm leaning toward doing additional Lucet-specific codegen (we already do a bit in lucet-wiggle), but this is probably worth talking about more at the meeting next week

view this post on Zulip acfoltzer (May 07 2020 at 21:51):

given that wasmtime/lucet integration is on our roadmap, it'd be good to understand exactly what the assumptions are in each system, and know how we can reconcile them

view this post on Zulip Jakub Konka (May 07 2020 at 21:52):

acfoltzer said:

okay, well thank you for the extra context about the whys of the current interface. I think I'll have to wait for Pat to get back and chat with him about what our strategy should be. right now I'm leaning toward doing additional Lucet-specific codegen (we already do a bit in lucet-wiggle), but this is probably worth talking about more at the meeting next week

Sounds good! Oh and would you mind pinging a couple of links in my direction where this problem space appears in lucet? I used to keep track of you changes guys but sadly not for a while now...

view this post on Zulip acfoltzer (May 07 2020 at 21:52):

or if they can't be reconciled, how to turn the difference into one of the "knobs" we've discussed before for modularizing the systems

view this post on Zulip acfoltzer (May 07 2020 at 21:53):

@Jakub Konka I unfortunately can't share the daemon code where we're doing this hostcall implementation, if that's what you mean

view this post on Zulip acfoltzer (May 07 2020 at 21:54):

the lucet-wiggle stuff though is just a thin wrapper around wiggle crates. you can see the core idea here: https://github.com/bytecodealliance/lucet/blob/master/lucet-wiggle/generate/src/lib.rs

Lucet, the Sandboxing WebAssembly Compiler. Contribute to bytecodealliance/lucet development by creating an account on GitHub.

view this post on Zulip acfoltzer (May 07 2020 at 21:55):

sorry, meant to link to this line: https://github.com/bytecodealliance/lucet/blob/master/lucet-wiggle/generate/src/lib.rs#L64

Lucet, the Sandboxing WebAssembly Compiler. Contribute to bytecodealliance/lucet development by creating an account on GitHub.

view this post on Zulip Jakub Konka (May 07 2020 at 21:59):

acfoltzer said:

Jakub Konka I unfortunately can't share the daemon code where we're doing this hostcall implementation, if that's what you mean

Oh, sure thing, I actually meant the lucet stuff mainly and lucet-wiggle in particular :-)

view this post on Zulip Jakub Konka (May 07 2020 at 21:59):

acfoltzer said:

sorry, meant to link to this line: https://github.com/bytecodealliance/lucet/blob/master/lucet-wiggle/generate/src/lib.rs#L64

Awesome, thanks!

view this post on Zulip acfoltzer (May 07 2020 at 22:03):

thank you all for taking a look :heart:


Last updated: Jan 24 2025 at 00:11 UTC