Skip to content

Commit 31dc001

Browse files
committed
Auto merge of rust-lang#134290 - tgross35:windows-i128-callconv, r=<try>
Windows x86: Change i128 to return via the vector ABI Clang and GCC both return `i128` in xmm0 on windows-msvc and windows-gnu. Currently, Rust returns the type on the stack. Add a calling convention adjustment so we also return scalar `i128`s using the vector ABI, which makes our `i128` compatible with C. In the future, Clang may change to return `i128` on the stack for its `-msvc` targets (more at [1]). If this happens, the change here will need to be adjusted to only affect MinGW. Link: rust-lang#134288 (does not fix) try-job: x86_64-msvc try-job: x86_64-msvc-ext1 try-job: x86_64-mingw-1 try-job: x86_64-mingw-2
2 parents 4790a43 + d5bc586 commit 31dc001

File tree

7 files changed

+123
-62
lines changed

7 files changed

+123
-62
lines changed

compiler/rustc_codegen_cranelift/src/abi/mod.rs

+8-9
Original file line numberDiff line numberDiff line change
@@ -122,7 +122,7 @@ impl<'tcx> FunctionCx<'_, '_, 'tcx> {
122122
&mut self,
123123
name: &str,
124124
params: Vec<AbiParam>,
125-
returns: Vec<AbiParam>,
125+
mut returns: Vec<AbiParam>,
126126
args: &[Value],
127127
) -> Cow<'_, [Value]> {
128128
// Pass i128 arguments by-ref on Windows.
@@ -146,15 +146,14 @@ impl<'tcx> FunctionCx<'_, '_, 'tcx> {
146146
(params, args.into())
147147
};
148148

149-
// Return i128 using a return area pointer on Windows and s390x.
150-
let adjust_ret_param =
151-
if self.tcx.sess.target.is_like_windows || self.tcx.sess.target.arch == "s390x" {
152-
returns.len() == 1 && returns[0].value_type == types::I128
153-
} else {
154-
false
155-
};
149+
// Return i128 using the vector ABI on Windows
150+
let ret_scalar_i128 = returns.len() == 1 && returns[0].value_type == types::I128;
151+
if ret_scalar_i128 && self.tcx.sess.target.is_like_windows {
152+
returns[0].value_type = types::I64X2;
153+
}
156154

157-
if adjust_ret_param {
155+
if ret_scalar_i128 && self.tcx.sess.target.arch == "s390x" {
156+
// Return i128 using a return area pointer on s390x.
158157
let mut params = params;
159158
let mut args = args.to_vec();
160159

compiler/rustc_codegen_cranelift/src/cast.rs

+3-19
Original file line numberDiff line numberDiff line change
@@ -96,25 +96,9 @@ pub(crate) fn clif_int_or_float_cast(
9696
},
9797
);
9898

99-
if fx.tcx.sess.target.is_like_windows {
100-
let ret = fx.lib_call(
101-
&name,
102-
vec![AbiParam::new(from_ty)],
103-
vec![AbiParam::new(types::I64X2)],
104-
&[from],
105-
)[0];
106-
// FIXME(bytecodealliance/wasmtime#6104) use bitcast instead of store to get from i64x2 to i128
107-
let ret_ptr = fx.create_stack_slot(16, 16);
108-
ret_ptr.store(fx, ret, MemFlags::trusted());
109-
ret_ptr.load(fx, types::I128, MemFlags::trusted())
110-
} else {
111-
fx.lib_call(
112-
&name,
113-
vec![AbiParam::new(from_ty)],
114-
vec![AbiParam::new(types::I128)],
115-
&[from],
116-
)[0]
117-
}
99+
fx.lib_call(&name, vec![AbiParam::new(from_ty)], vec![AbiParam::new(types::I128)], &[
100+
from,
101+
])[0]
118102
} else if to_ty == types::I8 || to_ty == types::I16 {
119103
// FIXME implement fcvt_to_*int_sat.i8/i16
120104
let val = if to_signed {

compiler/rustc_codegen_cranelift/src/codegen_i128.rs

+8-22
Original file line numberDiff line numberDiff line change
@@ -33,28 +33,14 @@ pub(crate) fn maybe_codegen<'tcx>(
3333
(BinOp::Rem, true) => "__modti3",
3434
_ => unreachable!(),
3535
};
36-
if fx.tcx.sess.target.is_like_windows {
37-
let args = [lhs.load_scalar(fx), rhs.load_scalar(fx)];
38-
let ret = fx.lib_call(
39-
name,
40-
vec![AbiParam::new(types::I128), AbiParam::new(types::I128)],
41-
vec![AbiParam::new(types::I64X2)],
42-
&args,
43-
)[0];
44-
// FIXME(bytecodealliance/wasmtime#6104) use bitcast instead of store to get from i64x2 to i128
45-
let ret_place = CPlace::new_stack_slot(fx, lhs.layout());
46-
ret_place.to_ptr().store(fx, ret, MemFlags::trusted());
47-
Some(ret_place.to_cvalue(fx))
48-
} else {
49-
let args = [lhs.load_scalar(fx), rhs.load_scalar(fx)];
50-
let ret_val = fx.lib_call(
51-
name,
52-
vec![AbiParam::new(types::I128), AbiParam::new(types::I128)],
53-
vec![AbiParam::new(types::I128)],
54-
&args,
55-
)[0];
56-
Some(CValue::by_val(ret_val, lhs.layout()))
57-
}
36+
let args = [lhs.load_scalar(fx), rhs.load_scalar(fx)];
37+
let ret_val = fx.lib_call(
38+
name,
39+
vec![AbiParam::new(types::I128), AbiParam::new(types::I128)],
40+
vec![AbiParam::new(types::I128)],
41+
&args,
42+
)[0];
43+
Some(CValue::by_val(ret_val, lhs.layout()))
5844
}
5945
BinOp::Lt | BinOp::Le | BinOp::Eq | BinOp::Ge | BinOp::Gt | BinOp::Ne | BinOp::Cmp => None,
6046
BinOp::Shl | BinOp::ShlUnchecked | BinOp::Shr | BinOp::ShrUnchecked => None,

compiler/rustc_target/src/callconv/x86_win64.rs

+13-7
Original file line numberDiff line numberDiff line change
@@ -1,12 +1,12 @@
1-
use rustc_abi::{BackendRepr, Float, Primitive};
1+
use rustc_abi::{BackendRepr, Float, Integer, Primitive, RegKind, Size};
22

33
use crate::abi::call::{ArgAbi, FnAbi, Reg};
44
use crate::spec::HasTargetSpec;
55

66
// Win64 ABI: https://docs.microsoft.com/en-us/cpp/build/parameter-passing
77

88
pub(crate) fn compute_abi_info<Ty>(cx: &impl HasTargetSpec, fn_abi: &mut FnAbi<'_, Ty>) {
9-
let fixup = |a: &mut ArgAbi<'_, Ty>| {
9+
let fixup = |a: &mut ArgAbi<'_, Ty>, is_ret: bool| {
1010
match a.layout.backend_repr {
1111
BackendRepr::Uninhabited | BackendRepr::Memory { sized: false } => {}
1212
BackendRepr::ScalarPair(..) | BackendRepr::Memory { sized: true } => {
@@ -23,11 +23,16 @@ pub(crate) fn compute_abi_info<Ty>(cx: &impl HasTargetSpec, fn_abi: &mut FnAbi<'
2323
// (probably what clang calls "illegal vectors").
2424
}
2525
BackendRepr::Scalar(scalar) => {
26-
// Match what LLVM does for `f128` so that `compiler-builtins` builtins match up
27-
// with what LLVM expects.
28-
if a.layout.size.bytes() > 8
26+
if is_ret && matches!(scalar.primitive(), Primitive::Int(Integer::I128, _)) {
27+
// `i128` is returned in xmm0 by Clang and GCC, no
28+
// FIXME(#134288): This may change for the `-msvc` targets in the future.
29+
let reg = Reg { kind: RegKind::Vector, size: Size::from_bits(128) };
30+
a.cast_to(reg);
31+
} else if a.layout.size.bytes() > 8
2932
&& !matches!(scalar.primitive(), Primitive::Float(Float::F128))
3033
{
34+
// Match what LLVM does for `f128` so that `compiler-builtins` builtins match up
35+
// with what LLVM expects.
3136
a.make_indirect();
3237
} else {
3338
a.extend_integer_width_to(32);
@@ -37,8 +42,9 @@ pub(crate) fn compute_abi_info<Ty>(cx: &impl HasTargetSpec, fn_abi: &mut FnAbi<'
3742
};
3843

3944
if !fn_abi.ret.is_ignore() {
40-
fixup(&mut fn_abi.ret);
45+
fixup(&mut fn_abi.ret, true);
4146
}
47+
4248
for arg in fn_abi.args.iter_mut() {
4349
if arg.is_ignore() {
4450
// x86_64-pc-windows-gnu doesn't ignore ZSTs.
@@ -50,6 +56,6 @@ pub(crate) fn compute_abi_info<Ty>(cx: &impl HasTargetSpec, fn_abi: &mut FnAbi<'
5056
}
5157
continue;
5258
}
53-
fixup(arg);
59+
fixup(arg, false);
5460
}
5561
}

src/tools/compiletest/src/runtest.rs

+11-4
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
use std::borrow::Cow;
2-
use std::collections::{HashMap, HashSet};
2+
use std::collections::{BTreeSet, HashMap, HashSet};
33
use std::ffi::OsString;
44
use std::fs::{self, File, create_dir_all};
55
use std::hash::{DefaultHasher, Hash, Hasher};
@@ -1939,16 +1939,18 @@ impl<'test> TestCx<'test> {
19391939
// via `filecheck-flags` or by adding new header directives.
19401940

19411941
// Because we use custom prefixes, we also have to register the default prefix.
1942-
filecheck.arg("--check-prefix=CHECK");
1942+
// Deduplicate these in a set so revisions like `CHECK`, `MSVC`, and `NONMSVC` don't
1943+
// cause errors.
1944+
let mut check_prefixes = BTreeSet::from(["CHECK"]);
19431945

19441946
// Some tests use the current revision name as a check prefix.
19451947
if let Some(rev) = self.revision {
1946-
filecheck.arg("--check-prefix").arg(rev);
1948+
check_prefixes.insert(rev);
19471949
}
19481950

19491951
// Some tests also expect either the MSVC or NONMSVC prefix to be defined.
19501952
let msvc_or_not = if self.config.target.contains("msvc") { "MSVC" } else { "NONMSVC" };
1951-
filecheck.arg("--check-prefix").arg(msvc_or_not);
1953+
check_prefixes.insert(msvc_or_not);
19521954

19531955
// The filecheck tool normally fails if a prefix is defined but not used.
19541956
// However, we define several prefixes globally for all tests.
@@ -1960,6 +1962,11 @@ impl<'test> TestCx<'test> {
19601962
// Add custom flags supplied by the `filecheck-flags:` test header.
19611963
filecheck.args(&self.props.filecheck_flags);
19621964

1965+
for prefix in check_prefixes {
1966+
// Set the expected prefixes
1967+
filecheck.arg("--check-prefix").arg(prefix);
1968+
}
1969+
19631970
self.compose_and_run(filecheck, "", None, None)
19641971
}
19651972

tests/auxiliary/minicore.rs

+4-1
Original file line numberDiff line numberDiff line change
@@ -40,8 +40,11 @@ impl<T: ?Sized> LegacyReceiver for &mut T {}
4040
pub trait Copy: Sized {}
4141

4242
impl_marker_trait!(
43-
Copy => [ bool, char, isize, usize, i8, i16, i32, i64, u8, u16, u32, u64, f32, f64 ]
43+
Copy => [
44+
bool, char, isize, usize, i8, i16, i32, i64, i128, u8, u16, u32, u64, u128, f32, f64
45+
]
4446
);
47+
4548
impl<'a, T: ?Sized> Copy for &'a T {}
4649
impl<T: ?Sized> Copy for *const T {}
4750
impl<T: ?Sized> Copy for *mut T {}

tests/codegen/i128-x86-callconv.rs

+76
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,76 @@
1+
//! Verify that Rust implements the expected calling convention for `i128`/`u128`.
2+
3+
//@ revisions: MSVC MINGW
4+
//@ add-core-stubs
5+
//@ [MSVC] needs-llvm-components: x86
6+
//@ [MINGW] needs-llvm-components: x86
7+
//@ [MSVC] compile-flags: --target x86_64-pc-windows-msvc
8+
//@ [MINGW] compile-flags: --target x86_64-pc-windows-gnu
9+
//@ [MSVC] filecheck-flags: --check-prefix=WIN
10+
//@ [MINGW] filecheck-flags: --check-prefix=WIN
11+
12+
#![crate_type = "lib"]
13+
#![no_std]
14+
#![no_core]
15+
#![feature(no_core, lang_items)]
16+
17+
extern crate minicore;
18+
19+
extern "C" {
20+
fn extern_call(arg0: i128);
21+
fn extern_ret() -> i128;
22+
}
23+
24+
#[no_mangle]
25+
pub extern "C" fn pass(_arg0: u32, arg1: i128) {
26+
// CHECK-LABEL: @pass(
27+
// i128 is passed indirectly on Windows. It should load the pointer to the stack and pass
28+
// a pointer to that allocation.
29+
// WIN-SAME: %_arg0, ptr{{.*}} %arg1)
30+
// WIN: [[PASS:%[_0-9]+]] = alloca [16 x i8], align 16
31+
// WIN: [[LOADED:%[_0-9]+]] = load i128, ptr %arg1
32+
// WIN: store i128 [[LOADED]], ptr [[PASS]]
33+
// WIN: call void @extern_call
34+
unsafe { extern_call(arg1) };
35+
}
36+
37+
// Check that we produce the correct return ABI
38+
#[no_mangle]
39+
pub extern "C" fn ret(_arg0: u32, arg1: i128) -> i128 {
40+
// CHECK-LABEL: @ret(
41+
// i128 is returned in xmm0 on Windows
42+
// FIXME(#134288): This may change for the `-msvc` targets in the future.
43+
// WIN-SAME: i32{{.*}} %_arg0, ptr{{.*}} %arg1)
44+
// WIN: [[LOADED:%[_0-9]+]] = load <16 x i8>, ptr %arg1
45+
// WIN-NEXT: ret <16 x i8> [[LOADED]]
46+
arg1
47+
}
48+
49+
// Check that we consume the correct return ABI
50+
#[no_mangle]
51+
pub extern "C" fn forward(dst: *mut i128) {
52+
// CHECK-LABEL: @forward
53+
// WIN-SAME: ptr{{.*}} %dst)
54+
// WIN: [[RETURNED:%[_0-9]+]] = tail call <16 x i8> @extern_ret()
55+
// WIN: store <16 x i8> [[RETURNED]], ptr %dst
56+
// WIN: ret void
57+
unsafe { *dst = extern_ret() };
58+
}
59+
60+
#[repr(C)]
61+
struct RetAggregate {
62+
a: i32,
63+
b: i128,
64+
}
65+
66+
#[no_mangle]
67+
pub extern "C" fn ret_aggregate(_arg0: u32, arg1: i128) -> RetAggregate {
68+
// CHECK-LABEL: @ret_aggregate(
69+
// Aggregates should also be returned indirectly
70+
// WIN-SAME: ptr{{.*}}sret([32 x i8]){{.*}}[[RET:%[_0-9]+]], i32{{.*}}%_arg0, ptr{{.*}}%arg1)
71+
// WIN: [[LOADED:%[_0-9]+]] = load i128, ptr %arg1
72+
// WIN: [[GEP:%[_0-9]+]] = getelementptr{{.*}}, ptr [[RET]]
73+
// WIN: store i128 [[LOADED]], ptr [[GEP]]
74+
// WIN: ret void
75+
RetAggregate { a: 1, b: arg1 }
76+
}

0 commit comments

Comments
 (0)