Trevor Gross ba5def8a85 Add __addhf3, __subhf3, __mulhf3, __{eq,ge,gt,le,lt,ne,unord}hf2
LLVM does not currently emit these, but it is being discussed as an
option on platforms where `f32` is not hardware supported. Glibc/libgcc
also has the comparison functions [1] already.

The generic implementations for addition, subtraction, and
multiplication work for f16 without any complications, as do
comparisons, so add them here.

[1]: https://sourceware.org/git/?p=glibc.git;a=commit;h=6ec6c77867af4ddfec7323e0ac6ede89effca852
2025-08-09 15:47:10 -05:00

157 lines
4.4 KiB
Rust

#![cfg_attr(f16_enabled, feature(f16))]
#![cfg_attr(f128_enabled, feature(f128))]
#![allow(unused_macros)]
use builtins_test::*;
mod int_mul {
use super::*;
macro_rules! mul {
($($i:ty, $fn:ident);*;) => {
$(
#[test]
fn $fn() {
use compiler_builtins::int::mul::$fn;
fuzz_2(N, |x: $i, y: $i| {
let mul0 = x.wrapping_mul(y);
let mul1: $i = $fn(x, y);
if mul0 != mul1 {
panic!(
"{func}({x}, {y}): std: {mul0}, builtins: {mul1}",
func = stringify!($fn),
);
}
});
}
)*
};
}
mul! {
u64, __muldi3;
i128, __multi3;
}
}
mod int_overflowing_mul {
use super::*;
macro_rules! overflowing_mul {
($($i:ty, $fn:ident);*;) => {
$(
#[test]
fn $fn() {
use compiler_builtins::int::mul::$fn;
fuzz_2(N, |x: $i, y: $i| {
let (mul0, o0) = x.overflowing_mul(y);
let mut o1 = 0i32;
let mul1: $i = $fn(x, y, &mut o1);
let o1 = o1 != 0;
if mul0 != mul1 || o0 != o1 {
panic!(
"{func}({x}, {y}): std: ({mul0}, {o0}), builtins: ({mul1}, {o1})",
func = stringify!($fn),
);
}
});
}
)*
};
}
overflowing_mul! {
i32, __mulosi4;
i64, __mulodi4;
i128, __muloti4;
}
#[test]
fn overflowing_mul_u128() {
use compiler_builtins::int::mul::{__rust_i128_mulo, __rust_u128_mulo};
fuzz_2(N, |x: u128, y: u128| {
let mut o1 = 0;
let (mul0, o0) = x.overflowing_mul(y);
let mul1 = __rust_u128_mulo(x, y, &mut o1);
if mul0 != mul1 || i32::from(o0) != o1 {
panic!("__rust_u128_mulo({x}, {y}): std: ({mul0}, {o0}), builtins: ({mul1}, {o1})",);
}
let x = x as i128;
let y = y as i128;
let (mul0, o0) = x.overflowing_mul(y);
let mul1 = __rust_i128_mulo(x, y, &mut o1);
if mul0 != mul1 || i32::from(o0) != o1 {
panic!("__rust_i128_mulo({x}, {y}): std: ({mul0}, {o0}), builtins: ({mul1}, {o1})",);
}
});
}
}
macro_rules! float_mul {
($($f:ty, $fn:ident, $apfloat_ty:ident, $sys_available:meta);*;) => {
$(
#[test]
fn $fn() {
use compiler_builtins::float::{mul::$fn, Float};
use core::ops::Mul;
fuzz_float_2(N, |x: $f, y: $f| {
let mul0 = apfloat_fallback!($f, $apfloat_ty, $sys_available, Mul::mul, x, y);
let mul1: $f = $fn(x, y);
if !Float::eq_repr(mul0, mul1) {
panic!(
"{func}({x:?}, {y:?}): std: {mul0:?}, builtins: {mul1:?}",
func = stringify!($fn),
);
}
});
}
)*
};
}
#[cfg(not(x86_no_sse))]
mod float_mul {
use super::*;
#[cfg(f16_enabled)]
float_mul! {
f16, __mulhf3, Half, all();
}
// FIXME(#616): Stop ignoring arches that don't have native support once fix for builtins is in
// nightly.
float_mul! {
f32, __mulsf3, Single, not(target_arch = "arm");
f64, __muldf3, Double, not(target_arch = "arm");
}
}
#[cfg(f128_enabled)]
#[cfg(not(x86_no_sse))]
#[cfg(not(any(target_arch = "powerpc", target_arch = "powerpc64")))]
mod float_mul_f128 {
use super::*;
float_mul! {
f128, __multf3, Quad,
// FIXME(llvm): there is a bug in LLVM rt.
// See <https://github.com/llvm/llvm-project/issues/91840>.
not(any(feature = "no-sys-f128", all(target_arch = "aarch64", target_os = "linux")));
}
}
#[cfg(f128_enabled)]
#[cfg(any(target_arch = "powerpc", target_arch = "powerpc64"))]
mod float_mul_f128_ppc {
use super::*;
float_mul! {
f128, __mulkf3, Quad, not(feature = "no-sys-f128");
}
}