2024-02-23 23:48:20 -06:00
|
|
|
//@ compile-flags: -O --target wasm32-unknown-emscripten
|
|
|
|
//@ needs-llvm-components: webassembly
|
|
|
|
|
|
|
|
// Emscripten has its own unique implementation of catch_unwind (in `codegen_emcc_try`),
|
|
|
|
// make sure it generates something reasonable.
|
|
|
|
|
|
|
|
#![feature(no_core, lang_items, intrinsics, rustc_attrs)]
|
|
|
|
#![crate_type = "lib"]
|
|
|
|
#![no_std]
|
|
|
|
#![no_core]
|
|
|
|
|
2024-05-28 23:11:20 -05:00
|
|
|
#[lang = "sized"]
|
|
|
|
trait Sized {}
|
|
|
|
#[lang = "freeze"]
|
|
|
|
trait Freeze {}
|
|
|
|
#[lang = "copy"]
|
|
|
|
trait Copy {}
|
2024-02-23 23:48:20 -06:00
|
|
|
|
|
|
|
#[rustc_intrinsic]
|
2024-05-28 23:11:20 -05:00
|
|
|
fn size_of<T>() -> usize {
|
|
|
|
loop {}
|
|
|
|
}
|
2024-02-23 23:48:20 -06:00
|
|
|
|
|
|
|
extern "rust-intrinsic" {
|
|
|
|
fn catch_unwind(
|
|
|
|
try_fn: fn(_: *mut u8),
|
|
|
|
data: *mut u8,
|
2024-05-28 23:11:20 -05:00
|
|
|
catch_fn: fn(_: *mut u8, _: *mut u8),
|
2024-02-23 23:48:20 -06:00
|
|
|
) -> i32;
|
|
|
|
}
|
|
|
|
|
|
|
|
// CHECK-LABEL: @ptr_size
|
|
|
|
#[no_mangle]
|
|
|
|
pub fn ptr_size() -> usize {
|
|
|
|
// CHECK: ret [[PTR_SIZE:.*]]
|
|
|
|
size_of::<*mut u8>()
|
|
|
|
}
|
|
|
|
|
|
|
|
// CHECK-LABEL: @test_catch_unwind
|
|
|
|
#[no_mangle]
|
|
|
|
pub unsafe fn test_catch_unwind(
|
|
|
|
try_fn: fn(_: *mut u8),
|
|
|
|
data: *mut u8,
|
2024-05-28 23:11:20 -05:00
|
|
|
catch_fn: fn(_: *mut u8, _: *mut u8),
|
2024-02-23 23:48:20 -06:00
|
|
|
) -> i32 {
|
|
|
|
// CHECK: start:
|
|
|
|
// CHECK: [[ALLOCA:%.*]] = alloca
|
|
|
|
|
|
|
|
// CHECK: catch.i:
|
|
|
|
// CHECK: [[LANDINGPAD:%.*]] = landingpad
|
|
|
|
// CHECK: [[EXCEPTION:%.*]] = extractvalue {{.*}} [[LANDINGPAD]], 0
|
|
|
|
// CHECK: [[SELECTOR:%.*]] = extractvalue {{.*}} [[LANDINGPAD]], 1
|
|
|
|
|
|
|
|
// CHECK: [[IS_RUST_EXN:%.*]] = icmp eq {{.*}}[[SELECTOR]]
|
|
|
|
// CHECK: [[IS_RUST_EXN_I8:%.*]] = zext i1 [[IS_RUST_EXN]] to i8
|
|
|
|
|
|
|
|
// CHECK: store ptr [[EXCEPTION]], ptr [[ALLOCA]]
|
|
|
|
// CHECK: [[IS_RUST_SLOT:%.*]] = getelementptr inbounds i8, ptr [[ALLOCA]], [[PTR_SIZE]]
|
|
|
|
// CHECK: store i8 [[IS_RUST_EXN_I8]], ptr [[IS_RUST_SLOT]]
|
|
|
|
|
|
|
|
// CHECK: call void %catch_fn(ptr %data, ptr nonnull [[ALLOCA]])
|
|
|
|
|
|
|
|
catch_unwind(try_fn, data, catch_fn)
|
|
|
|
}
|