Skip to content

Commit 08711d5

Browse files
authored
Unrolled build for rust-lang#121438
Rollup merge of rust-lang#121438 - coolreader18:wasm32-panic-unwind, r=cuviper std support for wasm32 panic=unwind Tracking issue: rust-lang#118168 This adds std support for `-Cpanic=unwind` on wasm, and with it slightly more fleshed out rustc support. Now, the stable default is still panic=abort without exception-handling, but if you `-Zbuild-std` with `RUSTFLAGS=-Cpanic=unwind`, you get wasm exception-handling try/catch blocks in the binary: ```rust #[no_mangle] pub fn foo_bar(x: bool) -> *mut u8 { let s = Box::<str>::from("hello"); maybe_panic(x); Box::into_raw(s).cast() } #[inline(never)] #[no_mangle] fn maybe_panic(x: bool) { if x { panic!("AAAAA"); } } ``` ```wat ;; snip... (try $label$5 (do (call $maybe_panic (local.get $0) ) (br $label$1) ) (catch_all (global.set $__stack_pointer (local.get $1) ) (call $__rust_dealloc (local.get $2) (i32.const 5) (i32.const 1) ) (rethrow $label$5) ) ) ;; snip... ```
2 parents 65cd843 + c7fcf43 commit 08711d5

File tree

9 files changed

+90
-10
lines changed

9 files changed

+90
-10
lines changed

compiler/rustc_codegen_llvm/src/llvm_util.rs

+9
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,7 @@ use crate::errors::{
55
};
66
use crate::llvm;
77
use libc::c_int;
8+
use rustc_codegen_ssa::base::wants_wasm_eh;
89
use rustc_codegen_ssa::traits::PrintBackendInfo;
910
use rustc_data_structures::fx::{FxHashMap, FxHashSet};
1011
use rustc_data_structures::small_c_str::SmallCStr;
@@ -98,6 +99,10 @@ unsafe fn configure_llvm(sess: &Session) {
9899
}
99100
}
100101

102+
if wants_wasm_eh(sess) {
103+
add("-wasm-enable-eh", false);
104+
}
105+
101106
if sess.target.os == "emscripten" && sess.panic_strategy() == PanicStrategy::Unwind {
102107
add("-enable-emscripten-cxx-exceptions", false);
103108
}
@@ -523,6 +528,10 @@ pub(crate) fn global_llvm_features(sess: &Session, diagnostics: bool) -> Vec<Str
523528
.map(String::from),
524529
);
525530

531+
if wants_wasm_eh(sess) && sess.panic_strategy() == PanicStrategy::Unwind {
532+
features.push("+exception-handling".into());
533+
}
534+
526535
// -Ctarget-features
527536
let supported_features = sess.target.supported_target_features();
528537
let mut featsmap = FxHashMap::default();

compiler/rustc_codegen_ssa/src/mir/block.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -1597,7 +1597,7 @@ impl<'a, 'tcx, Bx: BuilderMethods<'a, 'tcx>> FunctionCx<'a, 'tcx, Bx> {
15971597
let funclet;
15981598
let llbb;
15991599
let mut bx;
1600-
if base::wants_msvc_seh(self.cx.sess()) {
1600+
if base::wants_new_eh_instructions(self.cx.sess()) {
16011601
// This is a basic block that we're aborting the program for,
16021602
// notably in an `extern` function. These basic blocks are inserted
16031603
// so that we assert that `extern` functions do indeed not panic,

library/panic_unwind/src/gcc.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -62,7 +62,7 @@ pub unsafe fn panic(data: Box<dyn Any + Send>) -> u32 {
6262
let exception = Box::new(Exception {
6363
_uwe: uw::_Unwind_Exception {
6464
exception_class: rust_exception_class(),
65-
exception_cleanup,
65+
exception_cleanup: Some(exception_cleanup),
6666
private: [core::ptr::null(); uw::unwinder_private_data_size],
6767
},
6868
canary: &CANARY,

library/panic_unwind/src/lib.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -53,12 +53,12 @@ cfg_if::cfg_if! {
5353
target_os = "solid_asp3",
5454
all(target_family = "unix", not(target_os = "espidf")),
5555
all(target_vendor = "fortanix", target_env = "sgx"),
56+
target_family = "wasm",
5657
))] {
5758
#[path = "gcc.rs"]
5859
mod real_imp;
5960
} else {
6061
// Targets that don't support unwinding.
61-
// - family=wasm
6262
// - os=none ("bare metal" targets)
6363
// - os=uefi
6464
// - os=espidf

library/std/src/sys/personality/mod.rs

+4-4
Original file line numberDiff line numberDiff line change
@@ -16,11 +16,12 @@ mod dwarf;
1616
cfg_if::cfg_if! {
1717
if #[cfg(target_os = "emscripten")] {
1818
mod emcc;
19-
} else if #[cfg(target_env = "msvc")] {
19+
} else if #[cfg(any(target_env = "msvc", target_family = "wasm"))] {
2020
// This is required by the compiler to exist (e.g., it's a lang item),
2121
// but it's never actually called by the compiler because
22-
// _CxxFrameHandler3 is the personality function that is always used.
23-
// Hence this is just an aborting stub.
22+
// __CxxFrameHandler3 (msvc) / __gxx_wasm_personality_v0 (wasm) is the
23+
// personality function that is always used. Hence this is just an
24+
// aborting stub.
2425
#[lang = "eh_personality"]
2526
fn rust_eh_personality() {
2627
core::intrinsics::abort()
@@ -36,7 +37,6 @@ cfg_if::cfg_if! {
3637
mod gcc;
3738
} else {
3839
// Targets that don't support unwinding.
39-
// - family=wasm
4040
// - os=none ("bare metal" targets)
4141
// - os=uefi
4242
// - os=espidf

library/unwind/src/lib.rs

+7-1
Original file line numberDiff line numberDiff line change
@@ -6,6 +6,10 @@
66
#![cfg_attr(bootstrap, feature(cfg_target_abi))]
77
#![feature(strict_provenance)]
88
#![cfg_attr(not(target_env = "msvc"), feature(libc))]
9+
#![cfg_attr(
10+
all(target_family = "wasm", not(target_os = "emscripten")),
11+
feature(link_llvm_intrinsics)
12+
)]
913
#![allow(internal_features)]
1014

1115
cfg_if::cfg_if! {
@@ -29,9 +33,11 @@ cfg_if::cfg_if! {
2933
} else if #[cfg(target_os = "xous")] {
3034
mod unwinding;
3135
pub use unwinding::*;
36+
} else if #[cfg(target_family = "wasm")] {
37+
mod wasm;
38+
pub use wasm::*;
3239
} else {
3340
// no unwinder on the system!
34-
// - wasm32 (not emscripten, which is "unix" family)
3541
// - os=none ("bare metal" targets)
3642
// - os=hermit
3743
// - os=uefi

library/unwind/src/libunwind.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -91,7 +91,7 @@ pub struct _Unwind_Exception {
9191
pub enum _Unwind_Context {}
9292

9393
pub type _Unwind_Exception_Cleanup_Fn =
94-
extern "C" fn(unwind_code: _Unwind_Reason_Code, exception: *mut _Unwind_Exception);
94+
Option<extern "C" fn(unwind_code: _Unwind_Reason_Code, exception: *mut _Unwind_Exception)>;
9595

9696
// FIXME: The `#[link]` attributes on `extern "C"` block marks those symbols declared in
9797
// the block are reexported in dylib build of std. This is needed when build rustc with

library/unwind/src/unwinding.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -46,7 +46,7 @@ pub const unwinder_private_data_size: usize = core::mem::size_of::<UnwindExcepti
4646
- core::mem::size_of::<_Unwind_Exception_Cleanup_Fn>();
4747

4848
pub type _Unwind_Exception_Cleanup_Fn =
49-
extern "C" fn(unwind_code: _Unwind_Reason_Code, exception: *mut _Unwind_Exception);
49+
Option<extern "C" fn(unwind_code: _Unwind_Reason_Code, exception: *mut _Unwind_Exception)>;
5050

5151
#[repr(C)]
5252
pub struct _Unwind_Exception {

library/unwind/src/wasm.rs

+65
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,65 @@
1+
//! A shim for libunwind implemented in terms of the native wasm `throw` instruction.
2+
3+
#![allow(nonstandard_style)]
4+
5+
#[repr(C)]
6+
#[derive(Debug, Copy, Clone, PartialEq)]
7+
pub enum _Unwind_Reason_Code {
8+
_URC_NO_REASON = 0,
9+
_URC_FOREIGN_EXCEPTION_CAUGHT = 1,
10+
_URC_FATAL_PHASE2_ERROR = 2,
11+
_URC_FATAL_PHASE1_ERROR = 3,
12+
_URC_NORMAL_STOP = 4,
13+
_URC_END_OF_STACK = 5,
14+
_URC_HANDLER_FOUND = 6,
15+
_URC_INSTALL_CONTEXT = 7,
16+
_URC_CONTINUE_UNWIND = 8,
17+
_URC_FAILURE = 9, // used only by ARM EHABI
18+
}
19+
pub use _Unwind_Reason_Code::*;
20+
21+
pub type _Unwind_Exception_Class = u64;
22+
pub type _Unwind_Word = *const u8;
23+
24+
pub const unwinder_private_data_size: usize = 2;
25+
26+
#[repr(C)]
27+
pub struct _Unwind_Exception {
28+
pub exception_class: _Unwind_Exception_Class,
29+
pub exception_cleanup: _Unwind_Exception_Cleanup_Fn,
30+
pub private: [_Unwind_Word; unwinder_private_data_size],
31+
}
32+
33+
pub type _Unwind_Exception_Cleanup_Fn =
34+
Option<extern "C" fn(unwind_code: _Unwind_Reason_Code, exception: *mut _Unwind_Exception)>;
35+
36+
pub unsafe fn _Unwind_DeleteException(exception: *mut _Unwind_Exception) {
37+
if let Some(exception_cleanup) = unsafe { (*exception).exception_cleanup } {
38+
exception_cleanup(_URC_FOREIGN_EXCEPTION_CAUGHT, exception);
39+
}
40+
}
41+
42+
pub unsafe fn _Unwind_RaiseException(exception: *mut _Unwind_Exception) -> _Unwind_Reason_Code {
43+
#[cfg(panic = "unwind")]
44+
extern "C" {
45+
/// LLVM lowers this intrinsic to the `throw` instruction.
46+
// FIXME(coolreader18): move to stdarch
47+
#[link_name = "llvm.wasm.throw"]
48+
fn wasm_throw(tag: i32, ptr: *mut u8) -> !;
49+
}
50+
51+
// The wasm `throw` instruction takes a "tag", which differentiates certain
52+
// types of exceptions from others. LLVM currently just identifies these
53+
// via integers, with 0 corresponding to C++ exceptions and 1 to C setjmp()/longjmp().
54+
// Ideally, we'd be able to choose something unique for Rust, but for now,
55+
// we pretend to be C++ and implement the Itanium exception-handling ABI.
56+
cfg_if::cfg_if! {
57+
// for now, unless we're -Zbuild-std with panic=unwind, never codegen a throw.
58+
if #[cfg(panic = "unwind")] {
59+
wasm_throw(0, exception.cast())
60+
} else {
61+
let _ = exception;
62+
core::arch::wasm32::unreachable()
63+
}
64+
}
65+
}

0 commit comments

Comments
 (0)