2019-01-10 04:43:18 +00:00
|
|
|
use std::{f32, f64};
|
|
|
|
|
2019-01-08 21:04:03 +00:00
|
|
|
// F32
|
|
|
|
pub extern "C" fn ceilf32(x: f32) -> f32 {
|
|
|
|
x.ceil()
|
|
|
|
}
|
|
|
|
|
|
|
|
pub extern "C" fn floorf32(x: f32) -> f32 {
|
|
|
|
x.floor()
|
|
|
|
}
|
|
|
|
|
|
|
|
pub extern "C" fn truncf32(x: f32) -> f32 {
|
|
|
|
x.trunc()
|
|
|
|
}
|
|
|
|
|
2019-01-10 04:43:18 +00:00
|
|
|
/// `f32.round()` doesn't have the correct behavior. Ideally, we'd use
|
|
|
|
/// "https://doc.rust-lang.org/std/intrinsics/fn.nearbyintf32.html" for this,
|
|
|
|
/// but support for stable compilers is necessary, so we must implement
|
|
|
|
/// this ourselves.
|
|
|
|
/// This is ported from "https://github.com/rust-lang/rust/issues/55107#issuecomment-431247454"
|
2019-01-08 21:04:03 +00:00
|
|
|
pub extern "C" fn nearbyintf32(x: f32) -> f32 {
|
2019-01-10 04:43:18 +00:00
|
|
|
#[inline]
|
|
|
|
fn copysign(x: f32, y: f32) -> f32 {
|
|
|
|
let bitmask = y.to_bits() & (1 << 31);
|
|
|
|
f32::from_bits(x.to_bits() | bitmask)
|
|
|
|
}
|
|
|
|
|
|
|
|
if x.is_nan() {
|
2019-01-10 17:20:23 +00:00
|
|
|
f32::from_bits(x.to_bits() | (1 << 22))
|
2019-01-10 04:43:18 +00:00
|
|
|
} else {
|
|
|
|
let k = f32::EPSILON.recip();
|
|
|
|
let a = x.abs();
|
|
|
|
if a < k {
|
|
|
|
copysign((a + k) - k, x)
|
|
|
|
} else {
|
|
|
|
x
|
|
|
|
}
|
|
|
|
}
|
2019-01-08 21:04:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// F64
|
|
|
|
pub extern "C" fn ceilf64(x: f64) -> f64 {
|
|
|
|
x.ceil()
|
|
|
|
}
|
|
|
|
|
|
|
|
pub extern "C" fn floorf64(x: f64) -> f64 {
|
|
|
|
x.floor()
|
|
|
|
}
|
|
|
|
|
|
|
|
pub extern "C" fn truncf64(x: f64) -> f64 {
|
|
|
|
x.trunc()
|
|
|
|
}
|
|
|
|
|
2019-01-10 04:43:18 +00:00
|
|
|
/// `f64.round()` doesn't have the correct behavior. Ideally, we'd use
|
|
|
|
/// "https://doc.rust-lang.org/std/intrinsics/fn.nearbyintf64.html" for this,
|
|
|
|
/// but support for stable compilers is necessary, so we must implement
|
|
|
|
/// this ourselves.
|
|
|
|
/// This is ported from "https://github.com/rust-lang/rust/issues/55007#issuecomment-431247454"
|
2019-01-08 21:04:03 +00:00
|
|
|
pub extern "C" fn nearbyintf64(x: f64) -> f64 {
|
2019-01-10 04:43:18 +00:00
|
|
|
#[inline]
|
|
|
|
fn copysign(x: f64, y: f64) -> f64 {
|
|
|
|
let bitmask = y.to_bits() & (1 << 63);
|
|
|
|
f64::from_bits(x.to_bits() | bitmask)
|
|
|
|
}
|
|
|
|
|
|
|
|
if x.is_nan() {
|
2019-01-10 17:20:23 +00:00
|
|
|
f64::from_bits(x.to_bits() | (1 << 51))
|
2019-01-10 04:43:18 +00:00
|
|
|
} else {
|
|
|
|
let k = f64::EPSILON.recip();
|
|
|
|
let a = x.abs();
|
|
|
|
if a < k {
|
|
|
|
copysign((a + k) - k, x)
|
|
|
|
} else {
|
|
|
|
x
|
|
|
|
}
|
|
|
|
}
|
2019-01-08 21:04:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/// A declaration for the stack probe function in Rust's standard library, for
|
|
|
|
/// catching callstack overflow.
|
|
|
|
extern "C" {
|
|
|
|
pub fn __rust_probestack();
|
|
|
|
}
|