-
Notifications
You must be signed in to change notification settings - Fork 13k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Support for WebAssembly externref in non-web environment #103516
Comments
I believe this is essentially blocked by LLVM support: https://reviews.llvm.org/D122215 |
Okay. Thank you very much! |
@CryZe now that the changes you linked are merged, is there any plans to add this to Rust? |
I am not familiar with WASM or externref. What does this feature do, and what support is needed from Rust to enable it? Will it need language changes or a new |
An externref is an opaque object that the WebAssembly engine can pass into wasm functions from the outside. So in the case of a browser those are actual JavaScript objects. WebAssembly can then pass them around and store them in a dedicated table for them and later retrieve them from the table to then call for example an external function that takes those as arguments (so that could be a DOM function that operates on JavaScript objects). Importantly externrefs must stay fully opaque, so it's impossible to write them into linear memory as otherwise you could observe their bytes and possibly modify those. Also you wouldn't know how big they are anyway. One would have to check how LLVM treats them and design a language concept around that. |
Yes, this will probably require changing the ABI, but I don't think it will require adding a new one. A bit of contextThe reference types proposal was accepted into the wasm spec in Februrary 2021. LLVM support was added later in 2021. Clang also seems to mostly have support at this point. The reference types proposal adds a few new things. The most relevant for this discussion is:
An A See this Clang RFC for a more in-depth description of how LLVM implements this proposal. Rust implementationBefore this proposal, there was only a single global table, which contained all of the function pointers. It was not possible to extract a function pointer from the table into a stack or global value. This had a direct effect on the ABI. Because you could not represent function pointers directly, and because there was only a single table, function pointers were represented as indices into this table. As an example this code fn add(left: usize, right: usize) -> usize {
left + right
}
#[no_mangle]
pub fn add_ptr() -> fn(usize, usize) -> usize {
add
} currently compiles to this wasm module: (module
(type (;0;) (func (param i32 i32) (result i32)))
(type (;1;) (func (result i32)))
(func $_ZN3foo3add17hc001cc2609bca236E (type 0) (param i32 i32) (result i32)
local.get 1
local.get 0
i32.add)
(func $add_ptr (type 1) (result i32)
i32.const 1)
(table (;0;) 2 2 funcref)
(memory (;0;) 16)
(global $__stack_pointer (mut i32) (i32.const 1048576))
(global (;1;) i32 (i32.const 1048576))
(global (;2;) i32 (i32.const 1048576))
(export "memory" (memory 0))
(export "add_ptr" (func $add_ptr))
(export "__data_end" (global 1))
(export "__heap_base" (global 2))
(elem (;0;) (i32.const 1) func $_ZN3foo3add17hc001cc2609bca236E)) Specifically, the pointer to the In my opinion, this should ideally be changed. With the reference types proposal, it would be more appropriate to instead return a |
Ok. It sounds like this is purely an implementation concern - we can change the IR we send to LLVM and it will result in a speedup, without changes necessary to user code? |
No, it definitely is a language concern as an externref is a totally new kind of type with all sorts of corner cases that need to be resolved: void foo(__externref_t x) {
&x;
struct { __externref_t y; } z = { .y = x };
}
So it's a |
This probably needs some sort of RFC in that case. |
I have written a pre-RFC for WebAssembly Heap Type Support in Rust: https://internals.rust-lang.org/t/pre-rfc-webassembly-heap-types/19774 |
Hi everyone, we just proposed an RFC for arbitrary self types. The primary motivation is because for C++, Python and other interop we need to have smart pointer types which do not obey normal Rust semantics. This doesn't help with the fundamental LLVM-based opaque thingy that you need for pub struct WasmExternRef<T> {
index_into_wasm_bindgens_big_table: usize,
_phantom: std::marker::PhantomData<T>
}
impl std::ops::Receiver for WasmExternRef<T> {. // the new bit as enabled by the RFC linked above
type Target = T;
}
// Generated by wasm-bindgen
struct JSApi;
impl JSApi {
fn js_method(self: WasmExternRef<T>) { // note the weird self type here
// wasm-bindgen calls to javascript by accessing the index from the big table
}
}
fn main() {
let my_js_ref: WasmExternRef<JSApi> = // obtain somehow
my_js_ref.js_method(); // new bit enabled by RFC
} I'm not sure how much of this is already possible with wasm-bindgen as I've never used it. And it's probably not as good as first-class Thanks to pachi for spotting the possible link. |
…=wesleywiser Stabilize Wasm target features that are in phase 4 and 5 This stabilizes the Wasm target features that are known to be working and in [phase 4 and 5](https://github.com/WebAssembly/proposals/tree/04fa8c810e1dc99ab399e41052a6e427ee988180). Feature stabilized: - [Non-trapping float-to-int conversions](https://github.com/WebAssembly/nontrapping-float-to-int-conversions) - [Import/Export of Mutable Globals](https://github.com/WebAssembly/mutable-global) - [Sign-extension operators](https://github.com/WebAssembly/sign-extension-ops) - [Bulk memory operations](https://github.com/WebAssembly/bulk-memory-operations) - [Extended Constant Expressions](https://github.com/WebAssembly/extended-const) Features not stabilized: - [Multi-value](https://github.com/WebAssembly/multi-value): requires rebuilding `std` rust-lang#73755. - [Reference Types](https://github.com/WebAssembly/reference-types): no point stabilizing without rust-lang#103516. - [Threads](https://github.com/webassembly/threads): requires rebuilding `std` rust-lang#77839. - [Relaxed SIMD](https://github.com/WebAssembly/relaxed-simd): separate PR rust-lang#117468. - [Multi Memory](https://github.com/WebAssembly/multi-memory): not implemented. See rust-lang#117457 (comment) for more context. Documentation: rust-lang/reference#1420 Tracking issue: rust-lang#44839
…=wesleywiser Stabilize Wasm target features that are in phase 4 and 5 This stabilizes the Wasm target features that are known to be working and in [phase 4 and 5](https://github.com/WebAssembly/proposals/tree/04fa8c810e1dc99ab399e41052a6e427ee988180). Feature stabilized: - [Non-trapping float-to-int conversions](https://github.com/WebAssembly/nontrapping-float-to-int-conversions) - [Import/Export of Mutable Globals](https://github.com/WebAssembly/mutable-global) - [Sign-extension operators](https://github.com/WebAssembly/sign-extension-ops) - [Bulk memory operations](https://github.com/WebAssembly/bulk-memory-operations) - [Extended Constant Expressions](https://github.com/WebAssembly/extended-const) Features not stabilized: - [Multi-value](https://github.com/WebAssembly/multi-value): requires rebuilding `std` rust-lang#73755. - [Reference Types](https://github.com/WebAssembly/reference-types): no point stabilizing without rust-lang#103516. - [Threads](https://github.com/webassembly/threads): requires rebuilding `std` rust-lang#77839. - [Relaxed SIMD](https://github.com/WebAssembly/relaxed-simd): separate PR rust-lang#117468. - [Multi Memory](https://github.com/WebAssembly/multi-memory): not implemented. See rust-lang#117457 (comment) for more context. Documentation: rust-lang/reference#1420 Tracking issue: rust-lang#44839
My goal is to compile a function written in Rust to WebAssembly which has as an input/output type
externref
.Then I want to call this function as an export in Wasmtime. So, this function will receive an
externref
as an input and also return one.The WebAssembly Code in the wat format should look similar to this:
I have seen that there exists
wasm-bindgen
for browser hosts.Is there a way to produce
externref
as an input and output for exports of functions written in Rust that can be called in non-web environments like Wasmtime?The text was updated successfully, but these errors were encountered: