zig/lib/compiler_rt/fmax.zig
Andrew Kelley 74608293af compiler_rt: common -> compiler_rt
use the "symbol" helper function in all exports

move all declarations from common.zig to compiler_rt.zig

flatten the tree structure somewhat (move contents of tiny files into
parent files)

No functional changes.
2026-02-09 20:03:15 -08:00

76 lines
2.3 KiB
Zig

const std = @import("std");
const builtin = @import("builtin");
const math = std.math;
const arch = builtin.cpu.arch;
const compiler_rt = @import("../compiler_rt.zig");
const symbol = compiler_rt.symbol;
comptime {
symbol(&__fmaxh, "__fmaxh");
symbol(&fmaxf, "fmaxf");
symbol(&fmax, "fmax");
symbol(&__fmaxx, "__fmaxx");
if (compiler_rt.want_ppc_abi) {
symbol(&fmaxq, "fmaxf128");
}
symbol(&fmaxq, "fmaxq");
symbol(&fmaxl, "fmaxl");
}
pub fn __fmaxh(x: f16, y: f16) callconv(.c) f16 {
return generic_fmax(f16, x, y);
}
pub fn fmaxf(x: f32, y: f32) callconv(.c) f32 {
return generic_fmax(f32, x, y);
}
pub fn fmax(x: f64, y: f64) callconv(.c) f64 {
return generic_fmax(f64, x, y);
}
pub fn __fmaxx(x: f80, y: f80) callconv(.c) f80 {
return generic_fmax(f80, x, y);
}
pub fn fmaxq(x: f128, y: f128) callconv(.c) f128 {
return generic_fmax(f128, x, y);
}
pub fn fmaxl(x: c_longdouble, y: c_longdouble) callconv(.c) c_longdouble {
switch (@typeInfo(c_longdouble).float.bits) {
16 => return __fmaxh(x, y),
32 => return fmaxf(x, y),
64 => return fmax(x, y),
80 => return __fmaxx(x, y),
128 => return fmaxq(x, y),
else => @compileError("unreachable"),
}
}
inline fn generic_fmax(comptime T: type, x: T, y: T) T {
if (math.isNan(x))
return y;
if (math.isNan(y))
return x;
if (math.signbit(x) != math.signbit(y))
return if (math.signbit(x)) y else x;
return if (x < y) y else x;
}
test "generic_fmax" {
inline for ([_]type{ f32, f64, c_longdouble, f80, f128 }) |T| {
const nan_val = math.nan(T);
const Int = std.meta.Int(.unsigned, @bitSizeOf(T));
try std.testing.expect(math.isNan(generic_fmax(T, nan_val, nan_val)));
try std.testing.expectEqual(@as(T, 1.0), generic_fmax(T, nan_val, 1.0));
try std.testing.expectEqual(@as(T, 1.0), generic_fmax(T, 1.0, nan_val));
try std.testing.expectEqual(@as(T, 10.0), generic_fmax(T, 1.0, 10.0));
try std.testing.expectEqual(@as(T, 1.0), generic_fmax(T, 1.0, -1.0));
try std.testing.expectEqual(@as(Int, @bitCast(@as(T, 0.0))), @as(Int, @bitCast(generic_fmax(T, 0.0, -0.0))));
try std.testing.expectEqual(@as(Int, @bitCast(@as(T, 0.0))), @as(Int, @bitCast(generic_fmax(T, -0.0, 0.0))));
}
}