Skip to content

Commit c4e3300

Browse files
committed
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
1 parent 847247b commit c4e3300

File tree

2 files changed

+20
-17
lines changed

2 files changed

+20
-17
lines changed

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
}

tests/codegen/i128-x86-callconv.rs

+7-10
Original file line numberDiff line numberDiff line change
@@ -38,12 +38,11 @@ pub extern "C" fn pass(_arg0: u32, arg1: i128) {
3838
#[no_mangle]
3939
pub extern "C" fn ret(_arg0: u32, arg1: i128) -> i128 {
4040
// CHECK-LABEL: @ret(
41-
// i128 is returned on the stack on Windows.
42-
// FIXME: this ABI does not agree with Clang or MinGW GCC
43-
// WIN-SAME: ptr{{.*}} sret([16 x i8]){{.*}} [[RET:%_[0-9]+]], i32{{.*}} %_arg0, ptr{{.*}} %arg1)
44-
// WIN: [[LOADED:%[0-9]+]] = load i128, ptr %arg1
45-
// WIN: store i128 [[LOADED]], ptr [[RET]]
46-
// WIN: ret void
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]]
4746
arg1
4847
}
4948

@@ -52,10 +51,8 @@ pub extern "C" fn ret(_arg0: u32, arg1: i128) -> i128 {
5251
pub extern "C" fn forward(dst: *mut i128) {
5352
// CHECK-LABEL: @forward
5453
// WIN-SAME: ptr{{.*}} %dst)
55-
// WIN: [[RETURNED:%[_0-9]+]] = alloca [16 x i8], align 16
56-
// WIN: call void @extern_ret({{.*}} [[RETURNED]])
57-
// WIN: [[TMP:%[_0-9]+]] = load i128, ptr [[RETURNED]]
58-
// WIN: store i128 [[TMP]], ptr %dst
54+
// WIN: [[RETURNED:%[_0-9]+]] = tail call <16 x i8> @extern_ret()
55+
// WIN: store <16 x i8> [[RETURNED]], ptr %dst
5956
// WIN: ret void
6057
unsafe { *dst = extern_ret() };
6158
}

0 commit comments

Comments
 (0)