use indirect return for i128
and f128
on wasm32
This commit is contained in:
parent
93ba568ab9
commit
702134a930
3 changed files with 110 additions and 0 deletions
|
@ -1,3 +1,5 @@
|
|||
use rustc_abi::{BackendRepr, Float, Integer, Primitive};
|
||||
|
||||
use crate::abi::call::{ArgAbi, FnAbi};
|
||||
use crate::abi::{HasDataLayout, TyAbiInterface};
|
||||
|
||||
|
@ -27,6 +29,16 @@ where
|
|||
if ret.layout.is_aggregate() && !unwrap_trivial_aggregate(cx, ret) {
|
||||
ret.make_indirect();
|
||||
}
|
||||
|
||||
// `long double`, `__int128_t` and `__uint128_t` use an indirect return
|
||||
if let BackendRepr::Scalar(scalar) = ret.layout.backend_repr {
|
||||
match scalar.primitive() {
|
||||
Primitive::Int(Integer::I128, _) | Primitive::Float(Float::F128) => {
|
||||
ret.make_indirect();
|
||||
}
|
||||
_ => {}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
fn classify_arg<'a, Ty, C>(cx: &C, arg: &mut ArgAbi<'a, Ty>)
|
||||
|
|
49
tests/codegen/f128-wasm32-callconv.rs
Normal file
49
tests/codegen/f128-wasm32-callconv.rs
Normal file
|
@ -0,0 +1,49 @@
|
|||
//! Verify that Rust implements the expected calling convention for `f128`
|
||||
|
||||
//@ add-core-stubs
|
||||
//@ compile-flags: -O --target wasm32-wasip1
|
||||
//@ needs-llvm-components: webassembly
|
||||
|
||||
#![crate_type = "lib"]
|
||||
#![no_std]
|
||||
#![no_core]
|
||||
#![feature(no_core, lang_items, f128)]
|
||||
|
||||
extern crate minicore;
|
||||
|
||||
extern "C" {
|
||||
fn extern_call(arg0: f128);
|
||||
fn extern_ret() -> f128;
|
||||
}
|
||||
|
||||
#[no_mangle]
|
||||
pub extern "C" fn pass(_arg0: u32, arg1: f128) {
|
||||
// CHECK-LABEL: @pass(
|
||||
// an f128 is passed via registers
|
||||
// CHECK-SAME: fp128 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: f128) -> f128 {
|
||||
// CHECK-LABEL: @ret(
|
||||
// but an f128 is returned via the stack
|
||||
// CHECK-SAME: sret
|
||||
// CHECK: store fp128 %arg1
|
||||
// CHECK-NEXT: ret void
|
||||
arg1
|
||||
}
|
||||
|
||||
// Check that we consume the correct return ABI
|
||||
#[no_mangle]
|
||||
pub extern "C" fn forward(dst: *mut f128) {
|
||||
// CHECK-LABEL: @forward
|
||||
// CHECK-SAME: ptr{{.*}} %dst)
|
||||
// without optimizatons, an intermediate alloca is used
|
||||
// CHECK: call void @extern_ret
|
||||
// CHECK: store fp128
|
||||
// CHECK: ret void
|
||||
unsafe { *dst = extern_ret() };
|
||||
}
|
49
tests/codegen/i128-wasm32-callconv.rs
Normal file
49
tests/codegen/i128-wasm32-callconv.rs
Normal file
|
@ -0,0 +1,49 @@
|
|||
//! Verify that Rust implements the expected calling convention for `i128`/`u128`.
|
||||
|
||||
//@ add-core-stubs
|
||||
//@ compile-flags: -O --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() };
|
||||
}
|
Loading…
Add table
Add a link
Reference in a new issue