//! Verify that Rust implements the expected calling convention for `i128`/`u128`. //@ add-core-stubs //@ compile-flags: -Copt-level=3 --target wasm32-wasip1 //@ needs-llvm-components: webassembly #![crate_type = "lib"] #![no_std] #![no_core] #![feature(no_core, lang_items)] extern crate minicore; extern "C" { fn extern_call(arg0: i128); fn extern_ret() -> i128; } #[no_mangle] pub extern "C" fn pass(_arg0: u32, arg1: i128) { // CHECK-LABEL: @pass( // an i128 is passed via registers // CHECK-SAME: i128 noundef %arg1 // CHECK: call void @extern_call unsafe { extern_call(arg1) }; } // Check that we produce the correct return ABI #[no_mangle] pub extern "C" fn ret(_arg0: u32, arg1: i128) -> i128 { // CHECK-LABEL: @ret( // but an i128 is returned via the stack // CHECK-SAME: sret // CHECK: store i128 %arg1 // CHECK-NEXT: ret void arg1 } // Check that we consume the correct return ABI #[no_mangle] pub extern "C" fn forward(dst: *mut i128) { // CHECK-LABEL: @forward // CHECK-SAME: ptr{{.*}} %dst) // without optimizatons, an intermediate alloca is used // CHECK: call void @extern_ret // CHECK: store i128 // CHECK: ret void unsafe { *dst = extern_ret() }; }