master
1///! The quoted behavior definitions are from
2///! https://gcc.gnu.org/onlinedocs/gcc-12.1.0/gccint/Soft-float-library-routines.html#Soft-float-library-routines
3const common = @import("./common.zig");
4const comparef = @import("./comparef.zig");
5
6pub const panic = common.panic;
7
8comptime {
9 if (common.want_ppc_abi) {
10 @export(&__eqtf2, .{ .name = "__eqkf2", .linkage = common.linkage, .visibility = common.visibility });
11 @export(&__netf2, .{ .name = "__nekf2", .linkage = common.linkage, .visibility = common.visibility });
12 @export(&__lttf2, .{ .name = "__ltkf2", .linkage = common.linkage, .visibility = common.visibility });
13 @export(&__letf2, .{ .name = "__lekf2", .linkage = common.linkage, .visibility = common.visibility });
14 } else if (common.want_sparc_abi) {
15 @export(&_Qp_cmp, .{ .name = "_Qp_cmp", .linkage = common.linkage, .visibility = common.visibility });
16 @export(&_Qp_feq, .{ .name = "_Qp_feq", .linkage = common.linkage, .visibility = common.visibility });
17 @export(&_Qp_fne, .{ .name = "_Qp_fne", .linkage = common.linkage, .visibility = common.visibility });
18 @export(&_Qp_flt, .{ .name = "_Qp_flt", .linkage = common.linkage, .visibility = common.visibility });
19 @export(&_Qp_fle, .{ .name = "_Qp_fle", .linkage = common.linkage, .visibility = common.visibility });
20 @export(&_Qp_fgt, .{ .name = "_Qp_fgt", .linkage = common.linkage, .visibility = common.visibility });
21 @export(&_Qp_fge, .{ .name = "_Qp_fge", .linkage = common.linkage, .visibility = common.visibility });
22 }
23 @export(&__eqtf2, .{ .name = "__eqtf2", .linkage = common.linkage, .visibility = common.visibility });
24 @export(&__netf2, .{ .name = "__netf2", .linkage = common.linkage, .visibility = common.visibility });
25 @export(&__letf2, .{ .name = "__letf2", .linkage = common.linkage, .visibility = common.visibility });
26 @export(&__cmptf2, .{ .name = "__cmptf2", .linkage = common.linkage, .visibility = common.visibility });
27 @export(&__lttf2, .{ .name = "__lttf2", .linkage = common.linkage, .visibility = common.visibility });
28}
29
30/// "These functions calculate a <=> b. That is, if a is less than b, they return -1;
31/// if a is greater than b, they return 1; and if a and b are equal they return 0.
32/// If either argument is NaN they return 1..."
33///
34/// Note that this matches the definition of `__letf2`, `__eqtf2`, `__netf2`, `__cmptf2`,
35/// and `__lttf2`.
36fn __cmptf2(a: f128, b: f128) callconv(.c) i32 {
37 return @intFromEnum(comparef.cmpf2(f128, comparef.LE, a, b));
38}
39
40/// "These functions return a value less than or equal to zero if neither argument is NaN,
41/// and a is less than or equal to b."
42fn __letf2(a: f128, b: f128) callconv(.c) i32 {
43 return __cmptf2(a, b);
44}
45
46/// "These functions return zero if neither argument is NaN, and a and b are equal."
47/// Note that due to some kind of historical accident, __eqtf2 and __netf2 are defined
48/// to have the same return value.
49fn __eqtf2(a: f128, b: f128) callconv(.c) i32 {
50 return __cmptf2(a, b);
51}
52
53/// "These functions return a nonzero value if either argument is NaN, or if a and b are unequal."
54/// Note that due to some kind of historical accident, __eqtf2 and __netf2 are defined
55/// to have the same return value.
56fn __netf2(a: f128, b: f128) callconv(.c) i32 {
57 return __cmptf2(a, b);
58}
59
60/// "These functions return a value less than zero if neither argument is NaN, and a
61/// is strictly less than b."
62fn __lttf2(a: f128, b: f128) callconv(.c) i32 {
63 return __cmptf2(a, b);
64}
65
66const SparcFCMP = enum(i32) {
67 Equal = 0,
68 Less = 1,
69 Greater = 2,
70 Unordered = 3,
71};
72
73fn _Qp_cmp(a: *const f128, b: *const f128) callconv(.c) i32 {
74 return @intFromEnum(comparef.cmpf2(f128, SparcFCMP, a.*, b.*));
75}
76
77fn _Qp_feq(a: *const f128, b: *const f128) callconv(.c) bool {
78 return @as(SparcFCMP, @enumFromInt(_Qp_cmp(a, b))) == .Equal;
79}
80
81fn _Qp_fne(a: *const f128, b: *const f128) callconv(.c) bool {
82 return @as(SparcFCMP, @enumFromInt(_Qp_cmp(a, b))) != .Equal;
83}
84
85fn _Qp_flt(a: *const f128, b: *const f128) callconv(.c) bool {
86 return @as(SparcFCMP, @enumFromInt(_Qp_cmp(a, b))) == .Less;
87}
88
89fn _Qp_fgt(a: *const f128, b: *const f128) callconv(.c) bool {
90 return @as(SparcFCMP, @enumFromInt(_Qp_cmp(a, b))) == .Greater;
91}
92
93fn _Qp_fge(a: *const f128, b: *const f128) callconv(.c) bool {
94 return switch (@as(SparcFCMP, @enumFromInt(_Qp_cmp(a, b)))) {
95 .Equal, .Greater => true,
96 .Less, .Unordered => false,
97 };
98}
99
100fn _Qp_fle(a: *const f128, b: *const f128) callconv(.c) bool {
101 return switch (@as(SparcFCMP, @enumFromInt(_Qp_cmp(a, b)))) {
102 .Equal, .Less => true,
103 .Greater, .Unordered => false,
104 };
105}