|
| 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