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::call::{ArgAbi, FnAbi};
|
||||||
use crate::abi::{HasDataLayout, TyAbiInterface};
|
use crate::abi::{HasDataLayout, TyAbiInterface};
|
||||||
|
|
||||||
|
@ -27,6 +29,16 @@ where
|
||||||
if ret.layout.is_aggregate() && !unwrap_trivial_aggregate(cx, ret) {
|
if ret.layout.is_aggregate() && !unwrap_trivial_aggregate(cx, ret) {
|
||||||
ret.make_indirect();
|
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>)
|
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