Is there currently a way to call a closure from cranelift? The ones I'm thinking of trying wouldn't have closed over any variables, but otherwise would be a first class function.
Rust closures have an unstable ABI. What you can do however is do Box<Box<dyn FnOnce()>>
, turns this into a raw pointer to pass to the cranelift compiled function and then provide an extern "C"
function which turns the raw pointer back into Box<Box<dyn FnOnce()>>
(or &Box<dyn Fn()>
if you have a dyn Fn()
closure) and does the call. Note that you need double allocation as the layout of Box<dyn FnOnce()>
is unstable.
Correct me if I'm wrong @bjorn3 but I think in this case a simpler answer to @JT's question is that if the closure doesn't capture any variables, then you can use it at a Rust type of fn(...)
instead of Fn*(...)
and then its representation is just a thin function pointer. You still have to deal with the Rust ABI for function calls being unstable but you shouldn't need to box anything or use trait objects, I think.
Missed the "wouldn't have closed over any variables" part.
Jamey Sharp said:
Correct me if I'm wrong bjorn3 but I think in this case a simpler answer to JT's question is that if the closure doesn't capture any variables, then you can use it at a Rust type of
fn(...)
instead ofFn*(...)
and then its representation is just a thin function pointer. You still have to deal with the Rust ABI for function calls being unstable but you shouldn't need to box anything or use trait objects, I think.
Can you fill out what you mean a bit? I can't quite picture how to tie that back to cranelift. Are you saying pass it in as a fn(...)
argument and then pass that value to cranelift?
You still need an extern "C"
wrapper function for calling the fn(...)
I think.
You did cast the function pointer to a regular pointer or usize and pass it to the compiled function. The compiled function can then call a rust extern "C"
function and pass this value. The rust function can transmute it back to a function pointer and call it.
you can do this without unsafe or transmutes I think: https://play.rust-lang.org/?version=stable&mode=debug&edition=2021&gist=b9871f0855b226fafe1cdc22cf9bd9b8
the Rust syntax for a closure (|...| { ... }
) will coerce to a fn(...) -> ...
type if it has no captures
That's right, Chris, except that calling the fn(...)
pointer from Cranelift is more complicated than calling it from Rust, because you can't put an extern "C"
on the closure.
In Wasmtime we make assumptions about the unstable Rust function-call ABI so that we can call such functions directly, but we also have a large testing infrastructure and some other advantages to give us confidence that new Rust versions haven't broken our usage. (edit: I might be wrong about this? maybe we do have extern "C"
annotations everywhere we need them)
ah, darn, I hadn't realized this can't take an ABI annotation
Thanks for the help :) this gives me a few ways to try
Last updated: Jan 24 2025 at 00:11 UTC