a0a5bd85c9
These annotations fall into a few categories * Some simply aren't needed since functions will always be in the same CGU anyway and are already candidates for inlining. * Many are on massive functions which shouldn't be inlined across crates due to code size concerns. * Others aren't necessary since calls to this crate are rarely inlined anyway (since it's lowered through LLVM). If this crate is called directly and inlining is needed then LTO can always be turned on, otherwise this will benefit downstream consumers by avoiding re-codegen'ing so many functions.
38 lines
920 B
Rust
38 lines
920 B
Rust
#[cfg_attr(all(test, assert_no_panic), no_panic::no_panic)]
|
|
pub fn nextafterf(x: f32, y: f32) -> f32 {
|
|
if x.is_nan() || y.is_nan() {
|
|
return x + y;
|
|
}
|
|
|
|
let mut ux_i = x.to_bits();
|
|
let uy_i = y.to_bits();
|
|
if ux_i == uy_i {
|
|
return y;
|
|
}
|
|
|
|
let ax = ux_i & 0x7fff_ffff_u32;
|
|
let ay = uy_i & 0x7fff_ffff_u32;
|
|
if ax == 0 {
|
|
if ay == 0 {
|
|
return y;
|
|
}
|
|
ux_i = (uy_i & 0x8000_0000_u32) | 1;
|
|
} else if ax > ay || ((ux_i ^ uy_i) & 0x8000_0000_u32) != 0 {
|
|
ux_i -= 1;
|
|
} else {
|
|
ux_i += 1;
|
|
}
|
|
|
|
let e = ux_i.wrapping_shr(0x7f80_0000_u32);
|
|
// raise overflow if ux_f is infinite and x is finite
|
|
if e == 0x7f80_0000_u32 {
|
|
force_eval!(x + x);
|
|
}
|
|
let ux_f = f32::from_bits(ux_i);
|
|
// raise underflow if ux_f is subnormal or zero
|
|
if e == 0 {
|
|
force_eval!(x * x + ux_f * ux_f);
|
|
}
|
|
ux_f
|
|
}
|