Files
libm/src/math/nextafterf.rs
T
Alex Crichton a0a5bd85c9 Remove most #[inline] annotations
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.
2019-07-10 08:42:28 -07:00

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
}