Skip to content
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

__wasm_call_ctors, __wasm_call_dtors #471

Closed
realuptime opened this issue Feb 19, 2022 · 9 comments
Closed

__wasm_call_ctors, __wasm_call_dtors #471

realuptime opened this issue Feb 19, 2022 · 9 comments

Comments

@realuptime
Copy link

Hey,

Why in the 14 release __wasm_call_ctors and __wasm_call_ctors is called after each export function call?
In the 12 release everything worked fine and all the variables and state was preserved, but 14 release broke this.

@realuptime
Copy link
Author

Hey, I found a workaround after reading this:
https://reviews.llvm.org/D81689

This new behavior is disabled when the input has an explicit call to
__wasm_call_ctors, indicating code not expecting new-style command
support.

So in case anyone has this issue that static constructors and destructors are being called for each export function, make a call to __wasm_call_ctors in _start/main function:

In my case, C++:
// declare external C function
extern "C" { // mandatory!
extern void __wasm_call_ctors();
}

WA_EXPORT("main")
int main(void)
{
__wasm_call_ctors(); // this call is needed!
}

And not the linker generates code where:
exported functions are no longer wrapped in __wasm_call_ctors / __wasm_call_dtors calls!!!!

@realuptime
Copy link
Author

realuptime commented Feb 20, 2022

I also suggest to avoid static destructors for unexpected llvm changes that will break correct functionality!

To detect them use the -Wexit-time-destructors flag.
I also enable -Werror to be sure.

It is also worth noting that clang has support for not generating static destructors: -fno-c++-static-destructors
https://clang.llvm.org/docs/AttributeReference.html#no-destroy

Calling by default C++ static constructors and functions (static int x = somefunc()) for each "export" entry point is a very very bad idea!!!
I had to compile with -fno-c++-static-destructors first and then refactor my code.
Or use no_destroy flag (https://clang.llvm.org/docs/AttributeReference.html#no-destroy).
For example, on the project that I am working on I am calling a function(export.NeedsDisplay()) for each rendered frame, which could be 1000 times per second without VSYNC!

The project:
http://myapp.eu/wasi/

I also wrote about this issue to llvm guys:
https://reviews.llvm.org/D81689#3333973

@sbc100
Copy link
Member

sbc100 commented Feb 22, 2022

The idea here is that there are two different types of wasm modules that one might build: -mexec-model=command and -mexec-model=reactor. In the default "command" model your application entry points act line "main" in that they run a single function that the lifetime of the module is bound to that single entry point. In this mode the idea is that state does not persist between calls the module (just like the main in /bin/ls.. each time you call it your can a fresh process).

What you are building sound more like a "reactor" which is more like shared library that maintains state between calls into it from the outside.

You can read more about this change here: #13

@realuptime
Copy link
Author

Thank you Sam!

PiotrSikora added a commit to PiotrSikora/proxy-wasm-rust-sdk that referenced this issue Apr 4, 2022
This is primarly a mechanism to deliver a workaround for a breaking
change in Rust v1.56.0 (which updated LLVM to v13.0), which changed
existing reactors into multi-entry commands, leading to performance
degradation and/or unusable Proxy-Wasm plugins.

See: WebAssembly/WASI#471

This is delivered as a macro to make it somehow compatible with the
unstable "-Z wasi-exec-model=reactor" feature.

Signed-off-by: Piotr Sikora <piotrsikora@google.com>
@PiotrSikora
Copy link
Contributor

The issue is that this change broke previously working WASI reactors.

Note that Rust supports building proper WASI reactors only in nightly using -Z wasi-exec-model=reactor, and the only(?) way to create WASI reactors using stable is to build cdylib library (.wasm module without a start function) with exported entrypoints.

This worked fine until Rust v1.56.0 (which updated LLVM to v13.0), which started adding command_export wrappers to the existing entrypoints. This unexpected behavior leads to either degraded performance (WASI constructors and destructors called on each entrypoint, which is a source of significant overhead for shallow entrypoints) or pretty unusable modules (in restricted environments that don't always allow hostcalls).

Thanks for the workaround, but I don't understand why this was added as an opt-out instead of an opt-in feature.

@sunfishcode
Copy link
Member

The C execution model is that there's a main function, when it's called, the program runs, and when it exits, the program completes. wasi-libc, though its use of musl, inherits this assumption. Calling functions before libc's startup code has run, or after its shutdown code has run, isn't how it was anticipated to work. In practice, it appears that doing so kind of worked, but only if you didn't call into certain parts of libc.

"-mexec-model=reactor" is the LLVM wasm backend's way to support this use case, with a design and implementation that actually anticipates being used this way. It supports calling exports without rerunning the constructor on each entrypoint.

@PiotrSikora
Copy link
Contributor

Yes, I understand that.

What I'm saying is that the LLVM change (and the lack of support for -mexec-model=reactor in Rust stable) changed previously working "handmade" WASI reactors (running constructors only once) into new-style multi-entry commands (running constructors on each entry).

@sunfishcode
Copy link
Member

Ultimately, I made a judgement call. It appears I misjudged how many people were using "handmade" WASI reactors here.

@PiotrSikora
Copy link
Contributor

Yeah, no worries. The landscape was very different 2 years ago when you originally authored that change... and it wouldn't even be an issue if wasi-exec-model=reactor stabilized in Rust in the meantime.

Anyway, I mostly wanted to bring more attention to this issue and provide more context for other people to find, since I would definitely spend way more time tracking this down if @realuptime didn't open this issue.

PiotrSikora added a commit to proxy-wasm/proxy-wasm-rust-sdk that referenced this issue Apr 7, 2022
This is primarly a mechanism to deliver a workaround for a breaking
change in Rust v1.56.0 (which updated LLVM to v13.0), which changed
existing reactors into multi-entry commands, leading to performance
degradation and/or unusable Proxy-Wasm plugins.

See: WebAssembly/WASI#471

This is delivered as a macro to make it somehow compatible with the
unstable "-Z wasi-exec-model=reactor" feature.

Signed-off-by: Piotr Sikora <piotrsikora@google.com>
tosky19941209 added a commit to tosky19941209/wasm_proxy_module that referenced this issue Oct 2, 2024
This is primarly a mechanism to deliver a workaround for a breaking
change in Rust v1.56.0 (which updated LLVM to v13.0), which changed
existing reactors into multi-entry commands, leading to performance
degradation and/or unusable Proxy-Wasm plugins.

See: WebAssembly/WASI#471

This is delivered as a macro to make it somehow compatible with the
unstable "-Z wasi-exec-model=reactor" feature.

Signed-off-by: Piotr Sikora <piotrsikora@google.com>
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

No branches or pull requests

4 participants