2015-02-14 23:14:52 +00:00
|
|
|
// Copyright 2012-2015 The Rust Project Developers. See the COPYRIGHT
|
2012-12-04 00:48:01 +00:00
|
|
|
// file at the top-level directory of this distribution and at
|
|
|
|
// http://rust-lang.org/COPYRIGHT.
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
|
|
|
|
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
|
|
|
|
// option. This file may not be copied, modified, or distributed
|
|
|
|
// except according to those terms.
|
|
|
|
|
2017-04-06 20:30:26 +00:00
|
|
|
//! This module provides constants which are specific to the implementation
|
2017-10-06 14:41:04 +00:00
|
|
|
//! of the `f32` floating point data type.
|
|
|
|
//!
|
2017-12-22 01:29:14 +00:00
|
|
|
//! *[See also the `f32` primitive type](../../std/primitive.f32.html).*
|
2018-05-09 22:03:56 +00:00
|
|
|
//!
|
|
|
|
//! Mathematically significant numbers are provided in the `consts` sub-module.
|
2014-03-05 00:54:35 +00:00
|
|
|
|
2015-01-24 05:48:20 +00:00
|
|
|
#![stable(feature = "rust1", since = "1.0.0")]
|
2014-10-27 22:37:07 +00:00
|
|
|
#![allow(missing_docs)]
|
2011-12-28 01:20:14 +00:00
|
|
|
|
2015-12-18 12:29:49 +00:00
|
|
|
#[cfg(not(test))]
|
2015-04-18 06:45:55 +00:00
|
|
|
use intrinsics;
|
2015-12-18 12:29:49 +00:00
|
|
|
#[cfg(not(test))]
|
2017-11-01 19:59:40 +00:00
|
|
|
use sys::cmath;
|
2015-01-05 10:14:50 +00:00
|
|
|
|
2015-11-16 16:54:28 +00:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2015-04-17 22:32:42 +00:00
|
|
|
pub use core::f32::{RADIX, MANTISSA_DIGITS, DIGITS, EPSILON};
|
2015-11-16 16:54:28 +00:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2015-04-17 22:32:42 +00:00
|
|
|
pub use core::f32::{MIN_EXP, MAX_EXP, MIN_10_EXP};
|
2015-11-16 16:54:28 +00:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2014-05-01 05:23:26 +00:00
|
|
|
pub use core::f32::{MAX_10_EXP, NAN, INFINITY, NEG_INFINITY};
|
2015-11-16 16:54:28 +00:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2015-02-13 03:40:57 +00:00
|
|
|
pub use core::f32::{MIN, MIN_POSITIVE, MAX};
|
2015-11-16 16:54:28 +00:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2014-05-01 05:23:26 +00:00
|
|
|
pub use core::f32::consts;
|
|
|
|
|
2015-03-14 00:13:35 +00:00
|
|
|
#[cfg(not(test))]
|
2018-05-10 18:02:19 +00:00
|
|
|
#[lang = "f32_runtime"]
|
2015-03-14 00:13:35 +00:00
|
|
|
impl f32 {
|
|
|
|
/// Returns the largest integer less than or equal to a number.
|
|
|
|
///
|
2018-05-13 19:54:40 +00:00
|
|
|
/// # Examples
|
|
|
|
///
|
2015-03-14 00:13:35 +00:00
|
|
|
/// ```
|
2015-03-20 06:42:18 +00:00
|
|
|
/// let f = 3.99_f32;
|
|
|
|
/// let g = 3.0_f32;
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
|
|
|
/// assert_eq!(f.floor(), 3.0);
|
|
|
|
/// assert_eq!(g.floor(), 3.0);
|
|
|
|
/// ```
|
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
|
|
|
#[inline]
|
2015-08-16 07:42:53 +00:00
|
|
|
pub fn floor(self) -> f32 {
|
|
|
|
// On MSVC LLVM will lower many math intrinsics to a call to the
|
|
|
|
// corresponding function. On MSVC, however, many of these functions
|
|
|
|
// aren't actually available as symbols to call, but rather they are all
|
|
|
|
// `static inline` functions in header files. This means that from a C
|
|
|
|
// perspective it's "compatible", but not so much from an ABI
|
|
|
|
// perspective (which we're worried about).
|
|
|
|
//
|
|
|
|
// The inline header functions always just cast to a f64 and do their
|
|
|
|
// operation, so we do that here as well, but only for MSVC targets.
|
|
|
|
//
|
|
|
|
// Note that there are many MSVC-specific float operations which
|
|
|
|
// redirect to this comment, so `floorf` is just one case of a missing
|
|
|
|
// function on MSVC, but there are many others elsewhere.
|
|
|
|
#[cfg(target_env = "msvc")]
|
2016-01-05 20:14:38 +00:00
|
|
|
return (self as f64).floor() as f32;
|
2015-08-16 07:42:53 +00:00
|
|
|
#[cfg(not(target_env = "msvc"))]
|
2016-01-07 20:07:24 +00:00
|
|
|
return unsafe { intrinsics::floorf32(self) };
|
2015-08-16 07:42:53 +00:00
|
|
|
}
|
2015-03-14 00:13:35 +00:00
|
|
|
|
|
|
|
/// Returns the smallest integer greater than or equal to a number.
|
|
|
|
///
|
2018-05-13 19:54:40 +00:00
|
|
|
/// # Examples
|
|
|
|
///
|
2015-03-14 00:13:35 +00:00
|
|
|
/// ```
|
2015-03-20 06:42:18 +00:00
|
|
|
/// let f = 3.01_f32;
|
|
|
|
/// let g = 4.0_f32;
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
|
|
|
/// assert_eq!(f.ceil(), 4.0);
|
|
|
|
/// assert_eq!(g.ceil(), 4.0);
|
|
|
|
/// ```
|
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
|
|
|
#[inline]
|
2015-08-16 07:42:53 +00:00
|
|
|
pub fn ceil(self) -> f32 {
|
|
|
|
// see notes above in `floor`
|
|
|
|
#[cfg(target_env = "msvc")]
|
2016-01-05 20:14:38 +00:00
|
|
|
return (self as f64).ceil() as f32;
|
2015-08-16 07:42:53 +00:00
|
|
|
#[cfg(not(target_env = "msvc"))]
|
2016-01-05 20:14:38 +00:00
|
|
|
return unsafe { intrinsics::ceilf32(self) };
|
2015-08-16 07:42:53 +00:00
|
|
|
}
|
2015-03-14 00:13:35 +00:00
|
|
|
|
|
|
|
/// Returns the nearest integer to a number. Round half-way cases away from
|
|
|
|
/// `0.0`.
|
|
|
|
///
|
2018-05-13 19:54:40 +00:00
|
|
|
/// # Examples
|
|
|
|
///
|
2015-03-14 00:13:35 +00:00
|
|
|
/// ```
|
2015-03-20 06:42:18 +00:00
|
|
|
/// let f = 3.3_f32;
|
|
|
|
/// let g = -3.3_f32;
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
|
|
|
/// assert_eq!(f.round(), 3.0);
|
|
|
|
/// assert_eq!(g.round(), -3.0);
|
|
|
|
/// ```
|
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
|
|
|
#[inline]
|
2015-08-16 07:42:53 +00:00
|
|
|
pub fn round(self) -> f32 {
|
|
|
|
unsafe { intrinsics::roundf32(self) }
|
|
|
|
}
|
2015-03-14 00:13:35 +00:00
|
|
|
|
2015-04-13 14:21:32 +00:00
|
|
|
/// Returns the integer part of a number.
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
2018-05-13 19:54:40 +00:00
|
|
|
/// # Examples
|
|
|
|
///
|
2015-03-14 00:13:35 +00:00
|
|
|
/// ```
|
2015-03-20 06:42:18 +00:00
|
|
|
/// let f = 3.3_f32;
|
|
|
|
/// let g = -3.7_f32;
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
|
|
|
/// assert_eq!(f.trunc(), 3.0);
|
|
|
|
/// assert_eq!(g.trunc(), -3.0);
|
|
|
|
/// ```
|
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
|
|
|
#[inline]
|
2015-08-16 07:42:53 +00:00
|
|
|
pub fn trunc(self) -> f32 {
|
|
|
|
unsafe { intrinsics::truncf32(self) }
|
|
|
|
}
|
2015-03-14 00:13:35 +00:00
|
|
|
|
|
|
|
/// Returns the fractional part of a number.
|
|
|
|
///
|
2018-05-13 19:54:40 +00:00
|
|
|
/// # Examples
|
|
|
|
///
|
2015-03-14 00:13:35 +00:00
|
|
|
/// ```
|
2015-03-20 06:42:18 +00:00
|
|
|
/// use std::f32;
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
2015-03-20 06:42:18 +00:00
|
|
|
/// let x = 3.5_f32;
|
|
|
|
/// let y = -3.5_f32;
|
2015-03-14 00:13:35 +00:00
|
|
|
/// let abs_difference_x = (x.fract() - 0.5).abs();
|
|
|
|
/// let abs_difference_y = (y.fract() - (-0.5)).abs();
|
|
|
|
///
|
2015-03-20 06:42:18 +00:00
|
|
|
/// assert!(abs_difference_x <= f32::EPSILON);
|
|
|
|
/// assert!(abs_difference_y <= f32::EPSILON);
|
2015-03-14 00:13:35 +00:00
|
|
|
/// ```
|
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
|
|
|
#[inline]
|
2015-08-16 07:42:53 +00:00
|
|
|
pub fn fract(self) -> f32 { self - self.trunc() }
|
2015-03-14 00:13:35 +00:00
|
|
|
|
2015-03-20 06:42:18 +00:00
|
|
|
/// Computes the absolute value of `self`. Returns `NAN` if the
|
|
|
|
/// number is `NAN`.
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
2018-05-13 19:54:40 +00:00
|
|
|
/// # Examples
|
|
|
|
///
|
2015-03-14 00:13:35 +00:00
|
|
|
/// ```
|
2015-03-20 06:42:18 +00:00
|
|
|
/// use std::f32;
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
2015-03-20 06:42:18 +00:00
|
|
|
/// let x = 3.5_f32;
|
|
|
|
/// let y = -3.5_f32;
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
|
|
|
/// let abs_difference_x = (x.abs() - x).abs();
|
|
|
|
/// let abs_difference_y = (y.abs() - (-y)).abs();
|
|
|
|
///
|
2015-03-20 06:42:18 +00:00
|
|
|
/// assert!(abs_difference_x <= f32::EPSILON);
|
|
|
|
/// assert!(abs_difference_y <= f32::EPSILON);
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
2015-03-20 06:42:18 +00:00
|
|
|
/// assert!(f32::NAN.abs().is_nan());
|
2015-03-14 00:13:35 +00:00
|
|
|
/// ```
|
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
|
|
|
#[inline]
|
2018-04-10 14:36:23 +00:00
|
|
|
pub fn abs(self) -> f32 {
|
|
|
|
unsafe { intrinsics::fabsf32(self) }
|
|
|
|
}
|
2015-03-14 00:13:35 +00:00
|
|
|
|
|
|
|
/// Returns a number that represents the sign of `self`.
|
|
|
|
///
|
2015-03-20 06:42:18 +00:00
|
|
|
/// - `1.0` if the number is positive, `+0.0` or `INFINITY`
|
|
|
|
/// - `-1.0` if the number is negative, `-0.0` or `NEG_INFINITY`
|
|
|
|
/// - `NAN` if the number is `NAN`
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
2018-05-13 19:54:40 +00:00
|
|
|
/// # Examples
|
|
|
|
///
|
2015-03-14 00:13:35 +00:00
|
|
|
/// ```
|
2015-03-20 06:42:18 +00:00
|
|
|
/// use std::f32;
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
2015-03-20 06:42:18 +00:00
|
|
|
/// let f = 3.5_f32;
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
|
|
|
/// assert_eq!(f.signum(), 1.0);
|
2015-03-20 06:42:18 +00:00
|
|
|
/// assert_eq!(f32::NEG_INFINITY.signum(), -1.0);
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
2015-03-20 06:42:18 +00:00
|
|
|
/// assert!(f32::NAN.signum().is_nan());
|
2015-03-14 00:13:35 +00:00
|
|
|
/// ```
|
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
|
|
|
#[inline]
|
2018-04-10 14:36:23 +00:00
|
|
|
pub fn signum(self) -> f32 {
|
|
|
|
if self.is_nan() {
|
|
|
|
NAN
|
|
|
|
} else {
|
|
|
|
unsafe { intrinsics::copysignf32(1.0, self) }
|
|
|
|
}
|
|
|
|
}
|
2015-03-20 06:42:18 +00:00
|
|
|
|
2015-03-14 00:13:35 +00:00
|
|
|
/// Fused multiply-add. Computes `(self * a) + b` with only one rounding
|
2018-05-09 13:47:37 +00:00
|
|
|
/// error, yielding a more accurate result than an unfused multiply-add.
|
|
|
|
///
|
|
|
|
/// Using `mul_add` can be more performant than an unfused multiply-add if
|
|
|
|
/// the target architecture has a dedicated `fma` CPU instruction.
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
2018-05-13 19:54:40 +00:00
|
|
|
/// # Examples
|
|
|
|
///
|
2015-03-14 00:13:35 +00:00
|
|
|
/// ```
|
2015-03-20 06:42:18 +00:00
|
|
|
/// use std::f32;
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
2015-03-20 06:42:18 +00:00
|
|
|
/// let m = 10.0_f32;
|
|
|
|
/// let x = 4.0_f32;
|
|
|
|
/// let b = 60.0_f32;
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
|
|
|
/// // 100.0
|
|
|
|
/// let abs_difference = (m.mul_add(x, b) - (m*x + b)).abs();
|
|
|
|
///
|
2015-03-20 06:42:18 +00:00
|
|
|
/// assert!(abs_difference <= f32::EPSILON);
|
2015-03-14 00:13:35 +00:00
|
|
|
/// ```
|
2015-03-20 06:42:18 +00:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2015-03-14 00:13:35 +00:00
|
|
|
#[inline]
|
2015-08-16 07:42:53 +00:00
|
|
|
pub fn mul_add(self, a: f32, b: f32) -> f32 {
|
|
|
|
unsafe { intrinsics::fmaf32(self, a, b) }
|
|
|
|
}
|
2015-03-14 00:13:35 +00:00
|
|
|
|
2018-03-26 18:15:19 +00:00
|
|
|
/// Calculates Euclidean division, the matching method for `mod_euc`.
|
|
|
|
///
|
2018-03-27 23:41:40 +00:00
|
|
|
/// This computes the integer `n` such that
|
2018-03-26 18:15:19 +00:00
|
|
|
/// `self = n * rhs + self.mod_euc(rhs)`.
|
2018-03-27 23:41:40 +00:00
|
|
|
/// In other words, the result is `self / rhs` rounded to the integer `n`
|
2018-03-26 18:15:19 +00:00
|
|
|
/// such that `self >= n * rhs`.
|
|
|
|
///
|
2018-05-13 19:54:40 +00:00
|
|
|
/// # Examples
|
|
|
|
///
|
2018-03-26 18:15:19 +00:00
|
|
|
/// ```
|
|
|
|
/// #![feature(euclidean_division)]
|
|
|
|
/// let a: f32 = 7.0;
|
|
|
|
/// let b = 4.0;
|
|
|
|
/// assert_eq!(a.div_euc(b), 1.0); // 7.0 > 4.0 * 1.0
|
|
|
|
/// assert_eq!((-a).div_euc(b), -2.0); // -7.0 >= 4.0 * -2.0
|
|
|
|
/// assert_eq!(a.div_euc(-b), -1.0); // 7.0 >= -4.0 * -1.0
|
|
|
|
/// assert_eq!((-a).div_euc(-b), 2.0); // -7.0 >= -4.0 * 2.0
|
|
|
|
/// ```
|
|
|
|
#[inline]
|
|
|
|
#[unstable(feature = "euclidean_division", issue = "49048")]
|
|
|
|
pub fn div_euc(self, rhs: f32) -> f32 {
|
|
|
|
let q = (self / rhs).trunc();
|
|
|
|
if self % rhs < 0.0 {
|
|
|
|
return if rhs > 0.0 { q - 1.0 } else { q + 1.0 }
|
|
|
|
}
|
|
|
|
q
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Calculates the Euclidean modulo (self mod rhs), which is never negative.
|
|
|
|
///
|
|
|
|
/// In particular, the result `n` satisfies `0 <= n < rhs.abs()`.
|
|
|
|
///
|
2018-05-13 19:54:40 +00:00
|
|
|
/// # Examples
|
|
|
|
///
|
2018-03-26 18:15:19 +00:00
|
|
|
/// ```
|
|
|
|
/// #![feature(euclidean_division)]
|
|
|
|
/// let a: f32 = 7.0;
|
|
|
|
/// let b = 4.0;
|
|
|
|
/// assert_eq!(a.mod_euc(b), 3.0);
|
|
|
|
/// assert_eq!((-a).mod_euc(b), 1.0);
|
|
|
|
/// assert_eq!(a.mod_euc(-b), 3.0);
|
|
|
|
/// assert_eq!((-a).mod_euc(-b), 1.0);
|
|
|
|
/// ```
|
|
|
|
#[inline]
|
|
|
|
#[unstable(feature = "euclidean_division", issue = "49048")]
|
|
|
|
pub fn mod_euc(self, rhs: f32) -> f32 {
|
|
|
|
let r = self % rhs;
|
|
|
|
if r < 0.0 {
|
|
|
|
r + rhs.abs()
|
|
|
|
} else {
|
|
|
|
r
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-04-13 14:21:32 +00:00
|
|
|
/// Raises a number to an integer power.
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
|
|
|
/// Using this function is generally faster than using `powf`
|
|
|
|
///
|
2018-05-13 19:54:40 +00:00
|
|
|
/// # Examples
|
|
|
|
///
|
2015-03-14 00:13:35 +00:00
|
|
|
/// ```
|
2015-03-20 06:42:18 +00:00
|
|
|
/// use std::f32;
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
2015-03-20 06:42:18 +00:00
|
|
|
/// let x = 2.0_f32;
|
2015-03-14 00:13:35 +00:00
|
|
|
/// let abs_difference = (x.powi(2) - x*x).abs();
|
|
|
|
///
|
2015-03-20 06:42:18 +00:00
|
|
|
/// assert!(abs_difference <= f32::EPSILON);
|
2015-03-14 00:13:35 +00:00
|
|
|
/// ```
|
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
|
|
|
#[inline]
|
2018-04-10 14:36:23 +00:00
|
|
|
pub fn powi(self, n: i32) -> f32 {
|
|
|
|
unsafe { intrinsics::powif32(self, n) }
|
|
|
|
}
|
2015-03-14 00:13:35 +00:00
|
|
|
|
2015-04-13 14:21:32 +00:00
|
|
|
/// Raises a number to a floating point power.
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
2018-05-13 19:54:40 +00:00
|
|
|
/// # Examples
|
|
|
|
///
|
2015-03-14 00:13:35 +00:00
|
|
|
/// ```
|
2015-03-20 06:42:18 +00:00
|
|
|
/// use std::f32;
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
2015-03-20 06:42:18 +00:00
|
|
|
/// let x = 2.0_f32;
|
2015-03-14 00:13:35 +00:00
|
|
|
/// let abs_difference = (x.powf(2.0) - x*x).abs();
|
|
|
|
///
|
2015-03-20 06:42:18 +00:00
|
|
|
/// assert!(abs_difference <= f32::EPSILON);
|
2015-03-14 00:13:35 +00:00
|
|
|
/// ```
|
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
|
|
|
#[inline]
|
2015-08-16 07:42:53 +00:00
|
|
|
pub fn powf(self, n: f32) -> f32 {
|
|
|
|
// see notes above in `floor`
|
|
|
|
#[cfg(target_env = "msvc")]
|
2016-01-05 20:14:38 +00:00
|
|
|
return (self as f64).powf(n as f64) as f32;
|
2015-08-16 07:42:53 +00:00
|
|
|
#[cfg(not(target_env = "msvc"))]
|
2016-01-05 20:14:38 +00:00
|
|
|
return unsafe { intrinsics::powf32(self, n) };
|
2015-08-16 07:42:53 +00:00
|
|
|
}
|
2015-03-14 00:13:35 +00:00
|
|
|
|
2015-04-13 14:21:32 +00:00
|
|
|
/// Takes the square root of a number.
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
|
|
|
/// Returns NaN if `self` is a negative number.
|
|
|
|
///
|
2018-05-13 19:54:40 +00:00
|
|
|
/// # Examples
|
|
|
|
///
|
2015-03-14 00:13:35 +00:00
|
|
|
/// ```
|
2015-03-20 06:42:18 +00:00
|
|
|
/// use std::f32;
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
2015-03-20 06:42:18 +00:00
|
|
|
/// let positive = 4.0_f32;
|
|
|
|
/// let negative = -4.0_f32;
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
|
|
|
/// let abs_difference = (positive.sqrt() - 2.0).abs();
|
|
|
|
///
|
2015-03-20 06:42:18 +00:00
|
|
|
/// assert!(abs_difference <= f32::EPSILON);
|
2015-03-14 00:13:35 +00:00
|
|
|
/// assert!(negative.sqrt().is_nan());
|
|
|
|
/// ```
|
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
|
|
|
#[inline]
|
2015-08-16 07:42:53 +00:00
|
|
|
pub fn sqrt(self) -> f32 {
|
|
|
|
if self < 0.0 {
|
|
|
|
NAN
|
|
|
|
} else {
|
|
|
|
unsafe { intrinsics::sqrtf32(self) }
|
|
|
|
}
|
|
|
|
}
|
2015-03-14 00:13:35 +00:00
|
|
|
|
|
|
|
/// Returns `e^(self)`, (the exponential function).
|
|
|
|
///
|
2018-05-13 19:54:40 +00:00
|
|
|
/// # Examples
|
|
|
|
///
|
2015-03-14 00:13:35 +00:00
|
|
|
/// ```
|
2015-03-20 06:42:18 +00:00
|
|
|
/// use std::f32;
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
2015-03-20 06:42:18 +00:00
|
|
|
/// let one = 1.0f32;
|
2015-03-14 00:13:35 +00:00
|
|
|
/// // e^1
|
|
|
|
/// let e = one.exp();
|
|
|
|
///
|
|
|
|
/// // ln(e) - 1 == 0
|
|
|
|
/// let abs_difference = (e.ln() - 1.0).abs();
|
|
|
|
///
|
2015-03-20 06:42:18 +00:00
|
|
|
/// assert!(abs_difference <= f32::EPSILON);
|
2015-03-14 00:13:35 +00:00
|
|
|
/// ```
|
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
|
|
|
#[inline]
|
2015-08-16 07:42:53 +00:00
|
|
|
pub fn exp(self) -> f32 {
|
|
|
|
// see notes above in `floor`
|
|
|
|
#[cfg(target_env = "msvc")]
|
2016-01-05 20:14:38 +00:00
|
|
|
return (self as f64).exp() as f32;
|
2015-08-16 07:42:53 +00:00
|
|
|
#[cfg(not(target_env = "msvc"))]
|
2016-01-05 20:14:38 +00:00
|
|
|
return unsafe { intrinsics::expf32(self) };
|
2015-08-16 07:42:53 +00:00
|
|
|
}
|
2015-03-14 00:13:35 +00:00
|
|
|
|
|
|
|
/// Returns `2^(self)`.
|
|
|
|
///
|
2018-05-13 19:54:40 +00:00
|
|
|
/// # Examples
|
|
|
|
///
|
2015-03-14 00:13:35 +00:00
|
|
|
/// ```
|
2015-03-20 06:42:18 +00:00
|
|
|
/// use std::f32;
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
2015-03-20 06:42:18 +00:00
|
|
|
/// let f = 2.0f32;
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
|
|
|
/// // 2^2 - 4 == 0
|
|
|
|
/// let abs_difference = (f.exp2() - 4.0).abs();
|
|
|
|
///
|
2015-03-20 06:42:18 +00:00
|
|
|
/// assert!(abs_difference <= f32::EPSILON);
|
2015-03-14 00:13:35 +00:00
|
|
|
/// ```
|
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
|
|
|
#[inline]
|
2015-08-16 07:42:53 +00:00
|
|
|
pub fn exp2(self) -> f32 {
|
|
|
|
unsafe { intrinsics::exp2f32(self) }
|
|
|
|
}
|
2015-03-14 00:13:35 +00:00
|
|
|
|
|
|
|
/// Returns the natural logarithm of the number.
|
|
|
|
///
|
2018-05-13 19:54:40 +00:00
|
|
|
/// # Examples
|
|
|
|
///
|
2015-03-14 00:13:35 +00:00
|
|
|
/// ```
|
2015-03-20 06:42:18 +00:00
|
|
|
/// use std::f32;
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
2015-03-20 06:42:18 +00:00
|
|
|
/// let one = 1.0f32;
|
2015-03-14 00:13:35 +00:00
|
|
|
/// // e^1
|
|
|
|
/// let e = one.exp();
|
|
|
|
///
|
|
|
|
/// // ln(e) - 1 == 0
|
|
|
|
/// let abs_difference = (e.ln() - 1.0).abs();
|
|
|
|
///
|
2015-03-20 06:42:18 +00:00
|
|
|
/// assert!(abs_difference <= f32::EPSILON);
|
2015-03-14 00:13:35 +00:00
|
|
|
/// ```
|
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
|
|
|
#[inline]
|
2015-08-16 07:42:53 +00:00
|
|
|
pub fn ln(self) -> f32 {
|
|
|
|
// see notes above in `floor`
|
|
|
|
#[cfg(target_env = "msvc")]
|
2016-01-05 20:14:38 +00:00
|
|
|
return (self as f64).ln() as f32;
|
2015-08-16 07:42:53 +00:00
|
|
|
#[cfg(not(target_env = "msvc"))]
|
2016-01-05 20:14:38 +00:00
|
|
|
return unsafe { intrinsics::logf32(self) };
|
2015-08-16 07:42:53 +00:00
|
|
|
}
|
2015-03-14 00:13:35 +00:00
|
|
|
|
2018-01-09 11:26:00 +00:00
|
|
|
/// Returns the logarithm of the number with respect to an arbitrary base.
|
2018-01-08 18:22:37 +00:00
|
|
|
///
|
2018-01-09 11:26:00 +00:00
|
|
|
/// The result may not be correctly rounded owing to implementation details;
|
2018-01-08 18:22:37 +00:00
|
|
|
/// `self.log2()` can produce more accurate results for base 2, and
|
|
|
|
/// `self.log10()` can produce more accurate results for base 10.
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
2018-05-13 19:54:40 +00:00
|
|
|
/// # Examples
|
|
|
|
///
|
2015-03-14 00:13:35 +00:00
|
|
|
/// ```
|
2015-03-20 06:42:18 +00:00
|
|
|
/// use std::f32;
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
2018-01-08 18:22:37 +00:00
|
|
|
/// let five = 5.0f32;
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
2018-01-08 18:22:37 +00:00
|
|
|
/// // log5(5) - 1 == 0
|
|
|
|
/// let abs_difference = (five.log(5.0) - 1.0).abs();
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
2018-01-08 18:22:37 +00:00
|
|
|
/// assert!(abs_difference <= f32::EPSILON);
|
2015-03-14 00:13:35 +00:00
|
|
|
/// ```
|
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
|
|
|
#[inline]
|
2015-08-16 07:42:53 +00:00
|
|
|
pub fn log(self, base: f32) -> f32 { self.ln() / base.ln() }
|
2015-03-14 00:13:35 +00:00
|
|
|
|
|
|
|
/// Returns the base 2 logarithm of the number.
|
|
|
|
///
|
2018-05-13 19:54:40 +00:00
|
|
|
/// # Examples
|
|
|
|
///
|
2015-03-14 00:13:35 +00:00
|
|
|
/// ```
|
2015-03-20 06:42:18 +00:00
|
|
|
/// use std::f32;
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
2015-03-20 06:42:18 +00:00
|
|
|
/// let two = 2.0f32;
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
|
|
|
/// // log2(2) - 1 == 0
|
|
|
|
/// let abs_difference = (two.log2() - 1.0).abs();
|
|
|
|
///
|
2015-03-20 06:42:18 +00:00
|
|
|
/// assert!(abs_difference <= f32::EPSILON);
|
2015-03-14 00:13:35 +00:00
|
|
|
/// ```
|
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
|
|
|
#[inline]
|
2015-08-16 07:42:53 +00:00
|
|
|
pub fn log2(self) -> f32 {
|
2016-04-25 20:39:05 +00:00
|
|
|
#[cfg(target_os = "android")]
|
|
|
|
return ::sys::android::log2f32(self);
|
|
|
|
#[cfg(not(target_os = "android"))]
|
|
|
|
return unsafe { intrinsics::log2f32(self) };
|
2015-08-16 07:42:53 +00:00
|
|
|
}
|
2015-03-14 00:13:35 +00:00
|
|
|
|
|
|
|
/// Returns the base 10 logarithm of the number.
|
|
|
|
///
|
2018-05-13 19:54:40 +00:00
|
|
|
/// # Examples
|
|
|
|
///
|
2015-03-14 00:13:35 +00:00
|
|
|
/// ```
|
2015-03-20 06:42:18 +00:00
|
|
|
/// use std::f32;
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
2015-03-20 06:42:18 +00:00
|
|
|
/// let ten = 10.0f32;
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
|
|
|
/// // log10(10) - 1 == 0
|
|
|
|
/// let abs_difference = (ten.log10() - 1.0).abs();
|
|
|
|
///
|
2015-03-20 06:42:18 +00:00
|
|
|
/// assert!(abs_difference <= f32::EPSILON);
|
2015-03-14 00:13:35 +00:00
|
|
|
/// ```
|
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
|
|
|
#[inline]
|
2015-08-16 07:42:53 +00:00
|
|
|
pub fn log10(self) -> f32 {
|
|
|
|
// see notes above in `floor`
|
|
|
|
#[cfg(target_env = "msvc")]
|
2016-01-07 20:07:24 +00:00
|
|
|
return (self as f64).log10() as f32;
|
2015-08-16 07:42:53 +00:00
|
|
|
#[cfg(not(target_env = "msvc"))]
|
2016-01-07 20:07:24 +00:00
|
|
|
return unsafe { intrinsics::log10f32(self) };
|
2015-08-16 07:42:53 +00:00
|
|
|
}
|
2015-03-14 00:13:35 +00:00
|
|
|
|
|
|
|
/// The positive difference of two numbers.
|
|
|
|
///
|
|
|
|
/// * If `self <= other`: `0:0`
|
|
|
|
/// * Else: `self - other`
|
|
|
|
///
|
2018-05-13 19:54:40 +00:00
|
|
|
/// # Examples
|
|
|
|
///
|
2015-03-14 00:13:35 +00:00
|
|
|
/// ```
|
2015-03-20 06:42:18 +00:00
|
|
|
/// use std::f32;
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
2015-03-20 06:42:18 +00:00
|
|
|
/// let x = 3.0f32;
|
|
|
|
/// let y = -3.0f32;
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
|
|
|
/// let abs_difference_x = (x.abs_sub(1.0) - 2.0).abs();
|
|
|
|
/// let abs_difference_y = (y.abs_sub(1.0) - 0.0).abs();
|
|
|
|
///
|
2015-03-20 06:42:18 +00:00
|
|
|
/// assert!(abs_difference_x <= f32::EPSILON);
|
|
|
|
/// assert!(abs_difference_y <= f32::EPSILON);
|
2015-03-14 00:13:35 +00:00
|
|
|
/// ```
|
2015-03-20 06:42:18 +00:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2015-03-14 00:13:35 +00:00
|
|
|
#[inline]
|
2016-05-16 04:26:49 +00:00
|
|
|
#[rustc_deprecated(since = "1.10.0",
|
|
|
|
reason = "you probably meant `(self - other).abs()`: \
|
|
|
|
this operation is `(self - other).max(0.0)` (also \
|
|
|
|
known as `fdimf` in C). If you truly need the positive \
|
|
|
|
difference, consider using that expression or the C function \
|
|
|
|
`fdimf`, depending on how you wish to handle NaN (please consider \
|
|
|
|
filing an issue describing your use-case too).")]
|
2015-03-14 00:13:35 +00:00
|
|
|
pub fn abs_sub(self, other: f32) -> f32 {
|
|
|
|
unsafe { cmath::fdimf(self, other) }
|
|
|
|
}
|
|
|
|
|
2015-04-13 14:21:32 +00:00
|
|
|
/// Takes the cubic root of a number.
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
2018-05-13 19:54:40 +00:00
|
|
|
/// # Examples
|
|
|
|
///
|
2015-03-14 00:13:35 +00:00
|
|
|
/// ```
|
2015-03-20 06:42:18 +00:00
|
|
|
/// use std::f32;
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
2015-03-20 06:42:18 +00:00
|
|
|
/// let x = 8.0f32;
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
|
|
|
/// // x^(1/3) - 2 == 0
|
|
|
|
/// let abs_difference = (x.cbrt() - 2.0).abs();
|
|
|
|
///
|
2015-03-20 06:42:18 +00:00
|
|
|
/// assert!(abs_difference <= f32::EPSILON);
|
2015-03-14 00:13:35 +00:00
|
|
|
/// ```
|
2015-03-20 06:42:18 +00:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2015-03-14 00:13:35 +00:00
|
|
|
#[inline]
|
|
|
|
pub fn cbrt(self) -> f32 {
|
|
|
|
unsafe { cmath::cbrtf(self) }
|
|
|
|
}
|
|
|
|
|
2015-04-13 14:21:32 +00:00
|
|
|
/// Calculates the length of the hypotenuse of a right-angle triangle given
|
2015-03-14 00:13:35 +00:00
|
|
|
/// legs of length `x` and `y`.
|
|
|
|
///
|
2018-05-13 19:54:40 +00:00
|
|
|
/// # Examples
|
|
|
|
///
|
2015-03-14 00:13:35 +00:00
|
|
|
/// ```
|
2015-03-20 06:42:18 +00:00
|
|
|
/// use std::f32;
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
2015-03-20 06:42:18 +00:00
|
|
|
/// let x = 2.0f32;
|
|
|
|
/// let y = 3.0f32;
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
|
|
|
/// // sqrt(x^2 + y^2)
|
|
|
|
/// let abs_difference = (x.hypot(y) - (x.powi(2) + y.powi(2)).sqrt()).abs();
|
|
|
|
///
|
2015-03-20 06:42:18 +00:00
|
|
|
/// assert!(abs_difference <= f32::EPSILON);
|
2015-03-14 00:13:35 +00:00
|
|
|
/// ```
|
2015-03-20 06:42:18 +00:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2015-03-14 00:13:35 +00:00
|
|
|
#[inline]
|
|
|
|
pub fn hypot(self, other: f32) -> f32 {
|
|
|
|
unsafe { cmath::hypotf(self, other) }
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Computes the sine of a number (in radians).
|
|
|
|
///
|
2018-05-13 19:54:40 +00:00
|
|
|
/// # Examples
|
|
|
|
///
|
2015-03-14 00:13:35 +00:00
|
|
|
/// ```
|
2015-03-20 06:42:18 +00:00
|
|
|
/// use std::f32;
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
2015-03-20 06:42:18 +00:00
|
|
|
/// let x = f32::consts::PI/2.0;
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
|
|
|
/// let abs_difference = (x.sin() - 1.0).abs();
|
|
|
|
///
|
2015-03-20 06:42:18 +00:00
|
|
|
/// assert!(abs_difference <= f32::EPSILON);
|
2015-03-14 00:13:35 +00:00
|
|
|
/// ```
|
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
|
|
|
#[inline]
|
|
|
|
pub fn sin(self) -> f32 {
|
2015-06-25 22:33:42 +00:00
|
|
|
// see notes in `core::f32::Float::floor`
|
|
|
|
#[cfg(target_env = "msvc")]
|
2016-01-05 20:14:38 +00:00
|
|
|
return (self as f64).sin() as f32;
|
2015-06-25 22:33:42 +00:00
|
|
|
#[cfg(not(target_env = "msvc"))]
|
2016-01-05 20:14:38 +00:00
|
|
|
return unsafe { intrinsics::sinf32(self) };
|
2015-03-14 00:13:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Computes the cosine of a number (in radians).
|
|
|
|
///
|
2018-05-13 19:54:40 +00:00
|
|
|
/// # Examples
|
|
|
|
///
|
2015-03-14 00:13:35 +00:00
|
|
|
/// ```
|
2015-03-20 06:42:18 +00:00
|
|
|
/// use std::f32;
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
2015-03-20 06:42:18 +00:00
|
|
|
/// let x = 2.0*f32::consts::PI;
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
|
|
|
/// let abs_difference = (x.cos() - 1.0).abs();
|
|
|
|
///
|
2015-03-20 06:42:18 +00:00
|
|
|
/// assert!(abs_difference <= f32::EPSILON);
|
2015-03-14 00:13:35 +00:00
|
|
|
/// ```
|
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
|
|
|
#[inline]
|
|
|
|
pub fn cos(self) -> f32 {
|
2015-06-25 22:33:42 +00:00
|
|
|
// see notes in `core::f32::Float::floor`
|
|
|
|
#[cfg(target_env = "msvc")]
|
2016-01-05 20:14:38 +00:00
|
|
|
return (self as f64).cos() as f32;
|
2015-06-25 22:33:42 +00:00
|
|
|
#[cfg(not(target_env = "msvc"))]
|
2016-01-05 20:14:38 +00:00
|
|
|
return unsafe { intrinsics::cosf32(self) };
|
2015-03-14 00:13:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Computes the tangent of a number (in radians).
|
|
|
|
///
|
2018-05-13 19:54:40 +00:00
|
|
|
/// # Examples
|
|
|
|
///
|
2015-03-14 00:13:35 +00:00
|
|
|
/// ```
|
2016-06-22 22:33:07 +00:00
|
|
|
/// use std::f32;
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
2016-06-22 22:33:07 +00:00
|
|
|
/// let x = f32::consts::PI / 4.0;
|
2015-03-14 00:13:35 +00:00
|
|
|
/// let abs_difference = (x.tan() - 1.0).abs();
|
|
|
|
///
|
2016-06-22 22:33:07 +00:00
|
|
|
/// assert!(abs_difference <= f32::EPSILON);
|
2015-03-14 00:13:35 +00:00
|
|
|
/// ```
|
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
|
|
|
#[inline]
|
|
|
|
pub fn tan(self) -> f32 {
|
|
|
|
unsafe { cmath::tanf(self) }
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Computes the arcsine of a number. Return value is in radians in
|
|
|
|
/// the range [-pi/2, pi/2] or NaN if the number is outside the range
|
|
|
|
/// [-1, 1].
|
|
|
|
///
|
2018-05-13 19:54:40 +00:00
|
|
|
/// # Examples
|
|
|
|
///
|
2015-03-14 00:13:35 +00:00
|
|
|
/// ```
|
2015-03-20 06:42:18 +00:00
|
|
|
/// use std::f32;
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
2015-03-20 06:42:18 +00:00
|
|
|
/// let f = f32::consts::PI / 2.0;
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
|
|
|
/// // asin(sin(pi/2))
|
2016-05-16 04:26:49 +00:00
|
|
|
/// let abs_difference = (f.sin().asin() - f32::consts::PI / 2.0).abs();
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
2015-03-20 06:42:18 +00:00
|
|
|
/// assert!(abs_difference <= f32::EPSILON);
|
2015-03-14 00:13:35 +00:00
|
|
|
/// ```
|
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
|
|
|
#[inline]
|
|
|
|
pub fn asin(self) -> f32 {
|
|
|
|
unsafe { cmath::asinf(self) }
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Computes the arccosine of a number. Return value is in radians in
|
|
|
|
/// the range [0, pi] or NaN if the number is outside the range
|
|
|
|
/// [-1, 1].
|
|
|
|
///
|
2018-05-13 19:54:40 +00:00
|
|
|
/// # Examples
|
|
|
|
///
|
2015-03-14 00:13:35 +00:00
|
|
|
/// ```
|
2015-03-20 06:42:18 +00:00
|
|
|
/// use std::f32;
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
2015-03-20 06:42:18 +00:00
|
|
|
/// let f = f32::consts::PI / 4.0;
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
|
|
|
/// // acos(cos(pi/4))
|
2016-05-16 04:26:49 +00:00
|
|
|
/// let abs_difference = (f.cos().acos() - f32::consts::PI / 4.0).abs();
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
2015-03-20 06:42:18 +00:00
|
|
|
/// assert!(abs_difference <= f32::EPSILON);
|
2015-03-14 00:13:35 +00:00
|
|
|
/// ```
|
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
|
|
|
#[inline]
|
|
|
|
pub fn acos(self) -> f32 {
|
|
|
|
unsafe { cmath::acosf(self) }
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Computes the arctangent of a number. Return value is in radians in the
|
|
|
|
/// range [-pi/2, pi/2];
|
|
|
|
///
|
2018-05-13 19:54:40 +00:00
|
|
|
/// # Examples
|
|
|
|
///
|
2015-03-14 00:13:35 +00:00
|
|
|
/// ```
|
2015-03-20 06:42:18 +00:00
|
|
|
/// use std::f32;
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
2015-03-20 06:42:18 +00:00
|
|
|
/// let f = 1.0f32;
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
|
|
|
/// // atan(tan(1))
|
2016-05-16 04:26:49 +00:00
|
|
|
/// let abs_difference = (f.tan().atan() - 1.0).abs();
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
2015-03-20 06:42:18 +00:00
|
|
|
/// assert!(abs_difference <= f32::EPSILON);
|
2015-03-14 00:13:35 +00:00
|
|
|
/// ```
|
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
|
|
|
#[inline]
|
|
|
|
pub fn atan(self) -> f32 {
|
|
|
|
unsafe { cmath::atanf(self) }
|
|
|
|
}
|
|
|
|
|
2018-03-22 01:11:57 +00:00
|
|
|
/// Computes the four quadrant arctangent of `self` (`y`) and `other` (`x`) in radians.
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
|
|
|
/// * `x = 0`, `y = 0`: `0`
|
|
|
|
/// * `x >= 0`: `arctan(y/x)` -> `[-pi/2, pi/2]`
|
|
|
|
/// * `y >= 0`: `arctan(y/x) + pi` -> `(pi/2, pi]`
|
|
|
|
/// * `y < 0`: `arctan(y/x) - pi` -> `(-pi, -pi/2)`
|
|
|
|
///
|
2018-05-13 19:54:40 +00:00
|
|
|
/// # Examples
|
|
|
|
///
|
2015-03-14 00:13:35 +00:00
|
|
|
/// ```
|
2015-03-20 06:42:18 +00:00
|
|
|
/// use std::f32;
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
2015-03-20 06:42:18 +00:00
|
|
|
/// let pi = f32::consts::PI;
|
2018-03-22 01:11:57 +00:00
|
|
|
/// // Positive angles measured counter-clockwise
|
|
|
|
/// // from positive x axis
|
|
|
|
/// // -pi/4 radians (45 deg clockwise)
|
2015-03-20 06:42:18 +00:00
|
|
|
/// let x1 = 3.0f32;
|
|
|
|
/// let y1 = -3.0f32;
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
2018-03-22 01:11:57 +00:00
|
|
|
/// // 3pi/4 radians (135 deg counter-clockwise)
|
2015-03-20 06:42:18 +00:00
|
|
|
/// let x2 = -3.0f32;
|
|
|
|
/// let y2 = 3.0f32;
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
|
|
|
/// let abs_difference_1 = (y1.atan2(x1) - (-pi/4.0)).abs();
|
|
|
|
/// let abs_difference_2 = (y2.atan2(x2) - 3.0*pi/4.0).abs();
|
|
|
|
///
|
2015-03-20 06:42:18 +00:00
|
|
|
/// assert!(abs_difference_1 <= f32::EPSILON);
|
|
|
|
/// assert!(abs_difference_2 <= f32::EPSILON);
|
2015-03-14 00:13:35 +00:00
|
|
|
/// ```
|
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
|
|
|
#[inline]
|
|
|
|
pub fn atan2(self, other: f32) -> f32 {
|
|
|
|
unsafe { cmath::atan2f(self, other) }
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Simultaneously computes the sine and cosine of the number, `x`. Returns
|
|
|
|
/// `(sin(x), cos(x))`.
|
|
|
|
///
|
2018-05-13 19:54:40 +00:00
|
|
|
/// # Examples
|
|
|
|
///
|
2015-03-14 00:13:35 +00:00
|
|
|
/// ```
|
2015-03-20 06:42:18 +00:00
|
|
|
/// use std::f32;
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
2015-03-20 06:42:18 +00:00
|
|
|
/// let x = f32::consts::PI/4.0;
|
2015-03-14 00:13:35 +00:00
|
|
|
/// let f = x.sin_cos();
|
|
|
|
///
|
|
|
|
/// let abs_difference_0 = (f.0 - x.sin()).abs();
|
|
|
|
/// let abs_difference_1 = (f.1 - x.cos()).abs();
|
|
|
|
///
|
2015-03-20 06:42:18 +00:00
|
|
|
/// assert!(abs_difference_0 <= f32::EPSILON);
|
2016-04-16 06:42:37 +00:00
|
|
|
/// assert!(abs_difference_1 <= f32::EPSILON);
|
2015-03-14 00:13:35 +00:00
|
|
|
/// ```
|
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
|
|
|
#[inline]
|
|
|
|
pub fn sin_cos(self) -> (f32, f32) {
|
|
|
|
(self.sin(), self.cos())
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns `e^(self) - 1` in a way that is accurate even if the
|
|
|
|
/// number is close to zero.
|
|
|
|
///
|
2018-05-13 19:54:40 +00:00
|
|
|
/// # Examples
|
|
|
|
///
|
2015-03-14 00:13:35 +00:00
|
|
|
/// ```
|
2016-06-22 22:33:07 +00:00
|
|
|
/// use std::f32;
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
2016-06-22 22:33:07 +00:00
|
|
|
/// let x = 6.0f32;
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
2016-06-22 22:33:07 +00:00
|
|
|
/// // e^(ln(6)) - 1
|
|
|
|
/// let abs_difference = (x.ln().exp_m1() - 5.0).abs();
|
|
|
|
///
|
|
|
|
/// assert!(abs_difference <= f32::EPSILON);
|
2015-03-14 00:13:35 +00:00
|
|
|
/// ```
|
2015-03-20 06:42:18 +00:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2015-03-14 00:13:35 +00:00
|
|
|
#[inline]
|
|
|
|
pub fn exp_m1(self) -> f32 {
|
|
|
|
unsafe { cmath::expm1f(self) }
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns `ln(1+n)` (natural logarithm) more accurately than if
|
|
|
|
/// the operations were performed separately.
|
|
|
|
///
|
2018-05-13 19:54:40 +00:00
|
|
|
/// # Examples
|
|
|
|
///
|
2015-03-14 00:13:35 +00:00
|
|
|
/// ```
|
2015-03-20 06:42:18 +00:00
|
|
|
/// use std::f32;
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
2015-03-20 06:42:18 +00:00
|
|
|
/// let x = f32::consts::E - 1.0;
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
|
|
|
/// // ln(1 + (e - 1)) == ln(e) == 1
|
|
|
|
/// let abs_difference = (x.ln_1p() - 1.0).abs();
|
|
|
|
///
|
2015-03-20 06:42:18 +00:00
|
|
|
/// assert!(abs_difference <= f32::EPSILON);
|
2015-03-14 00:13:35 +00:00
|
|
|
/// ```
|
2015-03-20 06:42:18 +00:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2015-03-14 00:13:35 +00:00
|
|
|
#[inline]
|
|
|
|
pub fn ln_1p(self) -> f32 {
|
|
|
|
unsafe { cmath::log1pf(self) }
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Hyperbolic sine function.
|
|
|
|
///
|
2018-05-13 19:54:40 +00:00
|
|
|
/// # Examples
|
|
|
|
///
|
2015-03-14 00:13:35 +00:00
|
|
|
/// ```
|
2015-03-20 06:42:18 +00:00
|
|
|
/// use std::f32;
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
2015-03-20 06:42:18 +00:00
|
|
|
/// let e = f32::consts::E;
|
|
|
|
/// let x = 1.0f32;
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
|
|
|
/// let f = x.sinh();
|
|
|
|
/// // Solving sinh() at 1 gives `(e^2-1)/(2e)`
|
|
|
|
/// let g = (e*e - 1.0)/(2.0*e);
|
|
|
|
/// let abs_difference = (f - g).abs();
|
|
|
|
///
|
2015-03-20 06:42:18 +00:00
|
|
|
/// assert!(abs_difference <= f32::EPSILON);
|
2015-03-14 00:13:35 +00:00
|
|
|
/// ```
|
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
|
|
|
#[inline]
|
|
|
|
pub fn sinh(self) -> f32 {
|
|
|
|
unsafe { cmath::sinhf(self) }
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Hyperbolic cosine function.
|
|
|
|
///
|
2018-05-13 19:54:40 +00:00
|
|
|
/// # Examples
|
|
|
|
///
|
2015-03-14 00:13:35 +00:00
|
|
|
/// ```
|
2015-03-20 06:42:18 +00:00
|
|
|
/// use std::f32;
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
2015-03-20 06:42:18 +00:00
|
|
|
/// let e = f32::consts::E;
|
|
|
|
/// let x = 1.0f32;
|
2015-03-14 00:13:35 +00:00
|
|
|
/// let f = x.cosh();
|
|
|
|
/// // Solving cosh() at 1 gives this result
|
|
|
|
/// let g = (e*e + 1.0)/(2.0*e);
|
2016-05-16 04:26:49 +00:00
|
|
|
/// let abs_difference = (f - g).abs();
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
|
|
|
/// // Same result
|
2015-03-20 06:42:18 +00:00
|
|
|
/// assert!(abs_difference <= f32::EPSILON);
|
2015-03-14 00:13:35 +00:00
|
|
|
/// ```
|
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
|
|
|
#[inline]
|
|
|
|
pub fn cosh(self) -> f32 {
|
|
|
|
unsafe { cmath::coshf(self) }
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Hyperbolic tangent function.
|
|
|
|
///
|
2018-05-13 19:54:40 +00:00
|
|
|
/// # Examples
|
|
|
|
///
|
2015-03-14 00:13:35 +00:00
|
|
|
/// ```
|
2015-03-20 06:42:18 +00:00
|
|
|
/// use std::f32;
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
2015-03-20 06:42:18 +00:00
|
|
|
/// let e = f32::consts::E;
|
|
|
|
/// let x = 1.0f32;
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
|
|
|
/// let f = x.tanh();
|
|
|
|
/// // Solving tanh() at 1 gives `(1 - e^(-2))/(1 + e^(-2))`
|
|
|
|
/// let g = (1.0 - e.powi(-2))/(1.0 + e.powi(-2));
|
|
|
|
/// let abs_difference = (f - g).abs();
|
|
|
|
///
|
2015-03-20 06:42:18 +00:00
|
|
|
/// assert!(abs_difference <= f32::EPSILON);
|
2015-03-14 00:13:35 +00:00
|
|
|
/// ```
|
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
|
|
|
#[inline]
|
|
|
|
pub fn tanh(self) -> f32 {
|
|
|
|
unsafe { cmath::tanhf(self) }
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Inverse hyperbolic sine function.
|
|
|
|
///
|
2018-05-13 19:54:40 +00:00
|
|
|
/// # Examples
|
|
|
|
///
|
2015-03-14 00:13:35 +00:00
|
|
|
/// ```
|
2015-03-20 06:42:18 +00:00
|
|
|
/// use std::f32;
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
2015-03-20 06:42:18 +00:00
|
|
|
/// let x = 1.0f32;
|
2015-03-14 00:13:35 +00:00
|
|
|
/// let f = x.sinh().asinh();
|
|
|
|
///
|
|
|
|
/// let abs_difference = (f - x).abs();
|
|
|
|
///
|
2015-03-20 06:42:18 +00:00
|
|
|
/// assert!(abs_difference <= f32::EPSILON);
|
2015-03-14 00:13:35 +00:00
|
|
|
/// ```
|
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
|
|
|
#[inline]
|
|
|
|
pub fn asinh(self) -> f32 {
|
2016-03-11 18:36:46 +00:00
|
|
|
if self == NEG_INFINITY {
|
|
|
|
NEG_INFINITY
|
|
|
|
} else {
|
|
|
|
(self + ((self * self) + 1.0).sqrt()).ln()
|
2015-03-14 00:13:35 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Inverse hyperbolic cosine function.
|
|
|
|
///
|
2018-05-13 19:54:40 +00:00
|
|
|
/// # Examples
|
|
|
|
///
|
2015-03-14 00:13:35 +00:00
|
|
|
/// ```
|
2015-03-20 06:42:18 +00:00
|
|
|
/// use std::f32;
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
2015-03-20 06:42:18 +00:00
|
|
|
/// let x = 1.0f32;
|
2015-03-14 00:13:35 +00:00
|
|
|
/// let f = x.cosh().acosh();
|
|
|
|
///
|
|
|
|
/// let abs_difference = (f - x).abs();
|
|
|
|
///
|
2015-03-20 06:42:18 +00:00
|
|
|
/// assert!(abs_difference <= f32::EPSILON);
|
2015-03-14 00:13:35 +00:00
|
|
|
/// ```
|
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
|
|
|
#[inline]
|
|
|
|
pub fn acosh(self) -> f32 {
|
|
|
|
match self {
|
2015-04-18 06:45:55 +00:00
|
|
|
x if x < 1.0 => ::f32::NAN,
|
2015-03-14 00:13:35 +00:00
|
|
|
x => (x + ((x * x) - 1.0).sqrt()).ln(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Inverse hyperbolic tangent function.
|
|
|
|
///
|
2018-05-13 19:54:40 +00:00
|
|
|
/// # Examples
|
|
|
|
///
|
2015-03-14 00:13:35 +00:00
|
|
|
/// ```
|
2015-03-20 06:42:18 +00:00
|
|
|
/// use std::f32;
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
2015-03-20 06:42:18 +00:00
|
|
|
/// let e = f32::consts::E;
|
2015-03-14 00:13:35 +00:00
|
|
|
/// let f = e.tanh().atanh();
|
|
|
|
///
|
2016-05-16 04:26:49 +00:00
|
|
|
/// let abs_difference = (f - e).abs();
|
2015-03-14 00:13:35 +00:00
|
|
|
///
|
2016-05-16 04:26:49 +00:00
|
|
|
/// assert!(abs_difference <= 1e-5);
|
2015-03-14 00:13:35 +00:00
|
|
|
/// ```
|
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
|
|
|
#[inline]
|
|
|
|
pub fn atanh(self) -> f32 {
|
|
|
|
0.5 * ((2.0 * self) / (1.0 - self)).ln_1p()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-04-23 07:59:49 +00:00
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
2015-04-18 06:45:55 +00:00
|
|
|
use f32;
|
2013-04-23 07:59:49 +00:00
|
|
|
use f32::*;
|
2013-05-25 02:35:29 +00:00
|
|
|
use num::*;
|
2014-12-22 21:50:57 +00:00
|
|
|
use num::FpCategory as Fp;
|
2013-05-25 02:35:29 +00:00
|
|
|
|
2015-02-14 23:14:52 +00:00
|
|
|
#[test]
|
|
|
|
fn test_num_f32() {
|
|
|
|
test_num(10f32, 2f32);
|
|
|
|
}
|
|
|
|
|
2014-03-05 22:18:56 +00:00
|
|
|
#[test]
|
|
|
|
fn test_min_nan() {
|
|
|
|
assert_eq!(NAN.min(2.0), 2.0);
|
|
|
|
assert_eq!(2.0f32.min(NAN), 2.0);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_max_nan() {
|
|
|
|
assert_eq!(NAN.max(2.0), 2.0);
|
|
|
|
assert_eq!(2.0f32.max(NAN), 2.0);
|
|
|
|
}
|
|
|
|
|
2015-02-14 23:18:41 +00:00
|
|
|
#[test]
|
|
|
|
fn test_nan() {
|
2015-04-18 06:45:55 +00:00
|
|
|
let nan: f32 = f32::NAN;
|
2015-02-14 23:18:41 +00:00
|
|
|
assert!(nan.is_nan());
|
|
|
|
assert!(!nan.is_infinite());
|
|
|
|
assert!(!nan.is_finite());
|
|
|
|
assert!(!nan.is_normal());
|
2017-06-04 18:44:57 +00:00
|
|
|
assert!(nan.is_sign_positive());
|
2015-03-20 06:42:18 +00:00
|
|
|
assert!(!nan.is_sign_negative());
|
2015-02-14 23:18:41 +00:00
|
|
|
assert_eq!(Fp::Nan, nan.classify());
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_infinity() {
|
2015-04-18 06:45:55 +00:00
|
|
|
let inf: f32 = f32::INFINITY;
|
2015-02-14 23:18:41 +00:00
|
|
|
assert!(inf.is_infinite());
|
|
|
|
assert!(!inf.is_finite());
|
2015-03-20 06:42:18 +00:00
|
|
|
assert!(inf.is_sign_positive());
|
|
|
|
assert!(!inf.is_sign_negative());
|
2015-02-14 23:18:41 +00:00
|
|
|
assert!(!inf.is_nan());
|
|
|
|
assert!(!inf.is_normal());
|
|
|
|
assert_eq!(Fp::Infinite, inf.classify());
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_neg_infinity() {
|
2015-04-18 06:45:55 +00:00
|
|
|
let neg_inf: f32 = f32::NEG_INFINITY;
|
2015-02-14 23:18:41 +00:00
|
|
|
assert!(neg_inf.is_infinite());
|
|
|
|
assert!(!neg_inf.is_finite());
|
2015-03-20 06:42:18 +00:00
|
|
|
assert!(!neg_inf.is_sign_positive());
|
|
|
|
assert!(neg_inf.is_sign_negative());
|
2015-02-14 23:18:41 +00:00
|
|
|
assert!(!neg_inf.is_nan());
|
|
|
|
assert!(!neg_inf.is_normal());
|
|
|
|
assert_eq!(Fp::Infinite, neg_inf.classify());
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_zero() {
|
2015-04-18 06:45:55 +00:00
|
|
|
let zero: f32 = 0.0f32;
|
2015-02-14 23:18:41 +00:00
|
|
|
assert_eq!(0.0, zero);
|
|
|
|
assert!(!zero.is_infinite());
|
|
|
|
assert!(zero.is_finite());
|
2015-03-20 06:42:18 +00:00
|
|
|
assert!(zero.is_sign_positive());
|
|
|
|
assert!(!zero.is_sign_negative());
|
2015-02-14 23:18:41 +00:00
|
|
|
assert!(!zero.is_nan());
|
|
|
|
assert!(!zero.is_normal());
|
|
|
|
assert_eq!(Fp::Zero, zero.classify());
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_neg_zero() {
|
2015-04-18 06:45:55 +00:00
|
|
|
let neg_zero: f32 = -0.0;
|
2015-02-14 23:18:41 +00:00
|
|
|
assert_eq!(0.0, neg_zero);
|
|
|
|
assert!(!neg_zero.is_infinite());
|
|
|
|
assert!(neg_zero.is_finite());
|
2015-03-20 06:42:18 +00:00
|
|
|
assert!(!neg_zero.is_sign_positive());
|
|
|
|
assert!(neg_zero.is_sign_negative());
|
2015-02-14 23:18:41 +00:00
|
|
|
assert!(!neg_zero.is_nan());
|
|
|
|
assert!(!neg_zero.is_normal());
|
|
|
|
assert_eq!(Fp::Zero, neg_zero.classify());
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_one() {
|
2015-04-18 06:45:55 +00:00
|
|
|
let one: f32 = 1.0f32;
|
2015-02-14 23:18:41 +00:00
|
|
|
assert_eq!(1.0, one);
|
|
|
|
assert!(!one.is_infinite());
|
|
|
|
assert!(one.is_finite());
|
2015-03-20 06:42:18 +00:00
|
|
|
assert!(one.is_sign_positive());
|
|
|
|
assert!(!one.is_sign_negative());
|
2015-02-14 23:18:41 +00:00
|
|
|
assert!(!one.is_nan());
|
|
|
|
assert!(one.is_normal());
|
|
|
|
assert_eq!(Fp::Normal, one.classify());
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_is_nan() {
|
2015-04-18 06:45:55 +00:00
|
|
|
let nan: f32 = f32::NAN;
|
|
|
|
let inf: f32 = f32::INFINITY;
|
|
|
|
let neg_inf: f32 = f32::NEG_INFINITY;
|
2015-02-14 23:18:41 +00:00
|
|
|
assert!(nan.is_nan());
|
|
|
|
assert!(!0.0f32.is_nan());
|
|
|
|
assert!(!5.3f32.is_nan());
|
|
|
|
assert!(!(-10.732f32).is_nan());
|
|
|
|
assert!(!inf.is_nan());
|
|
|
|
assert!(!neg_inf.is_nan());
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_is_infinite() {
|
2015-04-18 06:45:55 +00:00
|
|
|
let nan: f32 = f32::NAN;
|
|
|
|
let inf: f32 = f32::INFINITY;
|
|
|
|
let neg_inf: f32 = f32::NEG_INFINITY;
|
2015-02-14 23:18:41 +00:00
|
|
|
assert!(!nan.is_infinite());
|
|
|
|
assert!(inf.is_infinite());
|
|
|
|
assert!(neg_inf.is_infinite());
|
|
|
|
assert!(!0.0f32.is_infinite());
|
|
|
|
assert!(!42.8f32.is_infinite());
|
|
|
|
assert!(!(-109.2f32).is_infinite());
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_is_finite() {
|
2015-04-18 06:45:55 +00:00
|
|
|
let nan: f32 = f32::NAN;
|
|
|
|
let inf: f32 = f32::INFINITY;
|
|
|
|
let neg_inf: f32 = f32::NEG_INFINITY;
|
2015-02-14 23:18:41 +00:00
|
|
|
assert!(!nan.is_finite());
|
|
|
|
assert!(!inf.is_finite());
|
|
|
|
assert!(!neg_inf.is_finite());
|
|
|
|
assert!(0.0f32.is_finite());
|
|
|
|
assert!(42.8f32.is_finite());
|
|
|
|
assert!((-109.2f32).is_finite());
|
|
|
|
}
|
|
|
|
|
2013-04-24 10:08:08 +00:00
|
|
|
#[test]
|
2015-02-14 23:14:52 +00:00
|
|
|
fn test_is_normal() {
|
2015-04-18 06:45:55 +00:00
|
|
|
let nan: f32 = f32::NAN;
|
|
|
|
let inf: f32 = f32::INFINITY;
|
|
|
|
let neg_inf: f32 = f32::NEG_INFINITY;
|
|
|
|
let zero: f32 = 0.0f32;
|
|
|
|
let neg_zero: f32 = -0.0;
|
2015-02-14 23:14:52 +00:00
|
|
|
assert!(!nan.is_normal());
|
|
|
|
assert!(!inf.is_normal());
|
|
|
|
assert!(!neg_inf.is_normal());
|
|
|
|
assert!(!zero.is_normal());
|
|
|
|
assert!(!neg_zero.is_normal());
|
|
|
|
assert!(1f32.is_normal());
|
|
|
|
assert!(1e-37f32.is_normal());
|
|
|
|
assert!(!1e-38f32.is_normal());
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_classify() {
|
2015-04-18 06:45:55 +00:00
|
|
|
let nan: f32 = f32::NAN;
|
|
|
|
let inf: f32 = f32::INFINITY;
|
|
|
|
let neg_inf: f32 = f32::NEG_INFINITY;
|
|
|
|
let zero: f32 = 0.0f32;
|
|
|
|
let neg_zero: f32 = -0.0;
|
2015-02-14 23:14:52 +00:00
|
|
|
assert_eq!(nan.classify(), Fp::Nan);
|
|
|
|
assert_eq!(inf.classify(), Fp::Infinite);
|
|
|
|
assert_eq!(neg_inf.classify(), Fp::Infinite);
|
|
|
|
assert_eq!(zero.classify(), Fp::Zero);
|
|
|
|
assert_eq!(neg_zero.classify(), Fp::Zero);
|
|
|
|
assert_eq!(1f32.classify(), Fp::Normal);
|
|
|
|
assert_eq!(1e-37f32.classify(), Fp::Normal);
|
|
|
|
assert_eq!(1e-38f32.classify(), Fp::Subnormal);
|
|
|
|
}
|
|
|
|
|
2013-04-25 01:53:04 +00:00
|
|
|
#[test]
|
|
|
|
fn test_floor() {
|
2013-05-06 11:51:48 +00:00
|
|
|
assert_approx_eq!(1.0f32.floor(), 1.0f32);
|
|
|
|
assert_approx_eq!(1.3f32.floor(), 1.0f32);
|
|
|
|
assert_approx_eq!(1.5f32.floor(), 1.0f32);
|
|
|
|
assert_approx_eq!(1.7f32.floor(), 1.0f32);
|
|
|
|
assert_approx_eq!(0.0f32.floor(), 0.0f32);
|
|
|
|
assert_approx_eq!((-0.0f32).floor(), -0.0f32);
|
|
|
|
assert_approx_eq!((-1.0f32).floor(), -1.0f32);
|
|
|
|
assert_approx_eq!((-1.3f32).floor(), -2.0f32);
|
|
|
|
assert_approx_eq!((-1.5f32).floor(), -2.0f32);
|
|
|
|
assert_approx_eq!((-1.7f32).floor(), -2.0f32);
|
2013-04-25 01:53:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_ceil() {
|
2013-05-06 11:51:48 +00:00
|
|
|
assert_approx_eq!(1.0f32.ceil(), 1.0f32);
|
|
|
|
assert_approx_eq!(1.3f32.ceil(), 2.0f32);
|
|
|
|
assert_approx_eq!(1.5f32.ceil(), 2.0f32);
|
|
|
|
assert_approx_eq!(1.7f32.ceil(), 2.0f32);
|
|
|
|
assert_approx_eq!(0.0f32.ceil(), 0.0f32);
|
|
|
|
assert_approx_eq!((-0.0f32).ceil(), -0.0f32);
|
|
|
|
assert_approx_eq!((-1.0f32).ceil(), -1.0f32);
|
|
|
|
assert_approx_eq!((-1.3f32).ceil(), -1.0f32);
|
|
|
|
assert_approx_eq!((-1.5f32).ceil(), -1.0f32);
|
|
|
|
assert_approx_eq!((-1.7f32).ceil(), -1.0f32);
|
2013-04-25 01:53:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_round() {
|
2013-05-06 11:51:48 +00:00
|
|
|
assert_approx_eq!(1.0f32.round(), 1.0f32);
|
|
|
|
assert_approx_eq!(1.3f32.round(), 1.0f32);
|
|
|
|
assert_approx_eq!(1.5f32.round(), 2.0f32);
|
|
|
|
assert_approx_eq!(1.7f32.round(), 2.0f32);
|
|
|
|
assert_approx_eq!(0.0f32.round(), 0.0f32);
|
|
|
|
assert_approx_eq!((-0.0f32).round(), -0.0f32);
|
|
|
|
assert_approx_eq!((-1.0f32).round(), -1.0f32);
|
|
|
|
assert_approx_eq!((-1.3f32).round(), -1.0f32);
|
|
|
|
assert_approx_eq!((-1.5f32).round(), -2.0f32);
|
|
|
|
assert_approx_eq!((-1.7f32).round(), -2.0f32);
|
2013-04-25 01:53:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_trunc() {
|
2013-05-06 11:51:48 +00:00
|
|
|
assert_approx_eq!(1.0f32.trunc(), 1.0f32);
|
|
|
|
assert_approx_eq!(1.3f32.trunc(), 1.0f32);
|
|
|
|
assert_approx_eq!(1.5f32.trunc(), 1.0f32);
|
|
|
|
assert_approx_eq!(1.7f32.trunc(), 1.0f32);
|
|
|
|
assert_approx_eq!(0.0f32.trunc(), 0.0f32);
|
|
|
|
assert_approx_eq!((-0.0f32).trunc(), -0.0f32);
|
|
|
|
assert_approx_eq!((-1.0f32).trunc(), -1.0f32);
|
|
|
|
assert_approx_eq!((-1.3f32).trunc(), -1.0f32);
|
|
|
|
assert_approx_eq!((-1.5f32).trunc(), -1.0f32);
|
|
|
|
assert_approx_eq!((-1.7f32).trunc(), -1.0f32);
|
2013-04-25 01:53:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_fract() {
|
2013-05-06 11:51:48 +00:00
|
|
|
assert_approx_eq!(1.0f32.fract(), 0.0f32);
|
|
|
|
assert_approx_eq!(1.3f32.fract(), 0.3f32);
|
|
|
|
assert_approx_eq!(1.5f32.fract(), 0.5f32);
|
|
|
|
assert_approx_eq!(1.7f32.fract(), 0.7f32);
|
|
|
|
assert_approx_eq!(0.0f32.fract(), 0.0f32);
|
|
|
|
assert_approx_eq!((-0.0f32).fract(), -0.0f32);
|
|
|
|
assert_approx_eq!((-1.0f32).fract(), -0.0f32);
|
|
|
|
assert_approx_eq!((-1.3f32).fract(), -0.3f32);
|
|
|
|
assert_approx_eq!((-1.5f32).fract(), -0.5f32);
|
|
|
|
assert_approx_eq!((-1.7f32).fract(), -0.7f32);
|
2013-04-25 01:53:04 +00:00
|
|
|
}
|
|
|
|
|
2015-02-14 23:14:52 +00:00
|
|
|
#[test]
|
2015-03-02 23:43:29 +00:00
|
|
|
fn test_abs() {
|
2015-02-14 23:14:52 +00:00
|
|
|
assert_eq!(INFINITY.abs(), INFINITY);
|
|
|
|
assert_eq!(1f32.abs(), 1f32);
|
|
|
|
assert_eq!(0f32.abs(), 0f32);
|
|
|
|
assert_eq!((-0f32).abs(), 0f32);
|
|
|
|
assert_eq!((-1f32).abs(), 1f32);
|
|
|
|
assert_eq!(NEG_INFINITY.abs(), INFINITY);
|
|
|
|
assert_eq!((1f32/NEG_INFINITY).abs(), 0f32);
|
|
|
|
assert!(NAN.abs().is_nan());
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_signum() {
|
|
|
|
assert_eq!(INFINITY.signum(), 1f32);
|
|
|
|
assert_eq!(1f32.signum(), 1f32);
|
|
|
|
assert_eq!(0f32.signum(), 1f32);
|
|
|
|
assert_eq!((-0f32).signum(), -1f32);
|
|
|
|
assert_eq!((-1f32).signum(), -1f32);
|
|
|
|
assert_eq!(NEG_INFINITY.signum(), -1f32);
|
|
|
|
assert_eq!((1f32/NEG_INFINITY).signum(), -1f32);
|
|
|
|
assert!(NAN.signum().is_nan());
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2015-03-20 06:42:18 +00:00
|
|
|
fn test_is_sign_positive() {
|
|
|
|
assert!(INFINITY.is_sign_positive());
|
|
|
|
assert!(1f32.is_sign_positive());
|
|
|
|
assert!(0f32.is_sign_positive());
|
|
|
|
assert!(!(-0f32).is_sign_positive());
|
|
|
|
assert!(!(-1f32).is_sign_positive());
|
|
|
|
assert!(!NEG_INFINITY.is_sign_positive());
|
|
|
|
assert!(!(1f32/NEG_INFINITY).is_sign_positive());
|
2017-06-04 18:44:57 +00:00
|
|
|
assert!(NAN.is_sign_positive());
|
|
|
|
assert!(!(-NAN).is_sign_positive());
|
2015-02-14 23:14:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2015-03-20 06:42:18 +00:00
|
|
|
fn test_is_sign_negative() {
|
|
|
|
assert!(!INFINITY.is_sign_negative());
|
|
|
|
assert!(!1f32.is_sign_negative());
|
|
|
|
assert!(!0f32.is_sign_negative());
|
|
|
|
assert!((-0f32).is_sign_negative());
|
|
|
|
assert!((-1f32).is_sign_negative());
|
|
|
|
assert!(NEG_INFINITY.is_sign_negative());
|
|
|
|
assert!((1f32/NEG_INFINITY).is_sign_negative());
|
|
|
|
assert!(!NAN.is_sign_negative());
|
2017-06-04 18:44:57 +00:00
|
|
|
assert!((-NAN).is_sign_negative());
|
2015-02-14 23:14:52 +00:00
|
|
|
}
|
|
|
|
|
2015-02-14 23:18:41 +00:00
|
|
|
#[test]
|
|
|
|
fn test_mul_add() {
|
2015-04-18 06:45:55 +00:00
|
|
|
let nan: f32 = f32::NAN;
|
|
|
|
let inf: f32 = f32::INFINITY;
|
|
|
|
let neg_inf: f32 = f32::NEG_INFINITY;
|
2015-02-14 23:18:41 +00:00
|
|
|
assert_approx_eq!(12.3f32.mul_add(4.5, 6.7), 62.05);
|
|
|
|
assert_approx_eq!((-12.3f32).mul_add(-4.5, -6.7), 48.65);
|
|
|
|
assert_approx_eq!(0.0f32.mul_add(8.9, 1.2), 1.2);
|
|
|
|
assert_approx_eq!(3.4f32.mul_add(-0.0, 5.6), 5.6);
|
|
|
|
assert!(nan.mul_add(7.8, 9.0).is_nan());
|
|
|
|
assert_eq!(inf.mul_add(7.8, 9.0), inf);
|
|
|
|
assert_eq!(neg_inf.mul_add(7.8, 9.0), neg_inf);
|
|
|
|
assert_eq!(8.9f32.mul_add(inf, 3.2), inf);
|
|
|
|
assert_eq!((-3.2f32).mul_add(2.4, neg_inf), neg_inf);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_recip() {
|
2015-04-18 06:45:55 +00:00
|
|
|
let nan: f32 = f32::NAN;
|
|
|
|
let inf: f32 = f32::INFINITY;
|
|
|
|
let neg_inf: f32 = f32::NEG_INFINITY;
|
2015-02-14 23:18:41 +00:00
|
|
|
assert_eq!(1.0f32.recip(), 1.0);
|
|
|
|
assert_eq!(2.0f32.recip(), 0.5);
|
|
|
|
assert_eq!((-0.4f32).recip(), -2.5);
|
|
|
|
assert_eq!(0.0f32.recip(), inf);
|
|
|
|
assert!(nan.recip().is_nan());
|
|
|
|
assert_eq!(inf.recip(), 0.0);
|
|
|
|
assert_eq!(neg_inf.recip(), 0.0);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_powi() {
|
2015-04-18 06:45:55 +00:00
|
|
|
let nan: f32 = f32::NAN;
|
|
|
|
let inf: f32 = f32::INFINITY;
|
|
|
|
let neg_inf: f32 = f32::NEG_INFINITY;
|
2015-02-14 23:18:41 +00:00
|
|
|
assert_eq!(1.0f32.powi(1), 1.0);
|
|
|
|
assert_approx_eq!((-3.1f32).powi(2), 9.61);
|
|
|
|
assert_approx_eq!(5.9f32.powi(-2), 0.028727);
|
|
|
|
assert_eq!(8.3f32.powi(0), 1.0);
|
|
|
|
assert!(nan.powi(2).is_nan());
|
|
|
|
assert_eq!(inf.powi(3), inf);
|
|
|
|
assert_eq!(neg_inf.powi(2), inf);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_powf() {
|
2015-04-18 06:45:55 +00:00
|
|
|
let nan: f32 = f32::NAN;
|
|
|
|
let inf: f32 = f32::INFINITY;
|
|
|
|
let neg_inf: f32 = f32::NEG_INFINITY;
|
2015-02-14 23:18:41 +00:00
|
|
|
assert_eq!(1.0f32.powf(1.0), 1.0);
|
|
|
|
assert_approx_eq!(3.4f32.powf(4.5), 246.408218);
|
|
|
|
assert_approx_eq!(2.7f32.powf(-3.2), 0.041652);
|
|
|
|
assert_approx_eq!((-3.1f32).powf(2.0), 9.61);
|
|
|
|
assert_approx_eq!(5.9f32.powf(-2.0), 0.028727);
|
|
|
|
assert_eq!(8.3f32.powf(0.0), 1.0);
|
|
|
|
assert!(nan.powf(2.0).is_nan());
|
|
|
|
assert_eq!(inf.powf(2.0), inf);
|
|
|
|
assert_eq!(neg_inf.powf(3.0), neg_inf);
|
|
|
|
}
|
|
|
|
|
2015-02-14 23:14:52 +00:00
|
|
|
#[test]
|
|
|
|
fn test_sqrt_domain() {
|
|
|
|
assert!(NAN.sqrt().is_nan());
|
|
|
|
assert!(NEG_INFINITY.sqrt().is_nan());
|
|
|
|
assert!((-1.0f32).sqrt().is_nan());
|
|
|
|
assert_eq!((-0.0f32).sqrt(), -0.0);
|
|
|
|
assert_eq!(0.0f32.sqrt(), 0.0);
|
|
|
|
assert_eq!(1.0f32.sqrt(), 1.0);
|
|
|
|
assert_eq!(INFINITY.sqrt(), INFINITY);
|
|
|
|
}
|
|
|
|
|
2015-02-08 01:02:24 +00:00
|
|
|
#[test]
|
|
|
|
fn test_exp() {
|
|
|
|
assert_eq!(1.0, 0.0f32.exp());
|
|
|
|
assert_approx_eq!(2.718282, 1.0f32.exp());
|
|
|
|
assert_approx_eq!(148.413162, 5.0f32.exp());
|
|
|
|
|
2015-04-18 06:45:55 +00:00
|
|
|
let inf: f32 = f32::INFINITY;
|
|
|
|
let neg_inf: f32 = f32::NEG_INFINITY;
|
|
|
|
let nan: f32 = f32::NAN;
|
2015-02-08 01:02:24 +00:00
|
|
|
assert_eq!(inf, inf.exp());
|
|
|
|
assert_eq!(0.0, neg_inf.exp());
|
|
|
|
assert!(nan.exp().is_nan());
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_exp2() {
|
|
|
|
assert_eq!(32.0, 5.0f32.exp2());
|
|
|
|
assert_eq!(1.0, 0.0f32.exp2());
|
|
|
|
|
2015-04-18 06:45:55 +00:00
|
|
|
let inf: f32 = f32::INFINITY;
|
|
|
|
let neg_inf: f32 = f32::NEG_INFINITY;
|
|
|
|
let nan: f32 = f32::NAN;
|
2015-02-08 01:02:24 +00:00
|
|
|
assert_eq!(inf, inf.exp2());
|
|
|
|
assert_eq!(0.0, neg_inf.exp2());
|
|
|
|
assert!(nan.exp2().is_nan());
|
|
|
|
}
|
|
|
|
|
2015-02-14 23:18:41 +00:00
|
|
|
#[test]
|
|
|
|
fn test_ln() {
|
2015-04-18 06:45:55 +00:00
|
|
|
let nan: f32 = f32::NAN;
|
|
|
|
let inf: f32 = f32::INFINITY;
|
|
|
|
let neg_inf: f32 = f32::NEG_INFINITY;
|
2015-02-14 23:18:41 +00:00
|
|
|
assert_approx_eq!(1.0f32.exp().ln(), 1.0);
|
|
|
|
assert!(nan.ln().is_nan());
|
|
|
|
assert_eq!(inf.ln(), inf);
|
|
|
|
assert!(neg_inf.ln().is_nan());
|
|
|
|
assert!((-2.3f32).ln().is_nan());
|
|
|
|
assert_eq!((-0.0f32).ln(), neg_inf);
|
|
|
|
assert_eq!(0.0f32.ln(), neg_inf);
|
|
|
|
assert_approx_eq!(4.0f32.ln(), 1.386294);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_log() {
|
2015-04-18 06:45:55 +00:00
|
|
|
let nan: f32 = f32::NAN;
|
|
|
|
let inf: f32 = f32::INFINITY;
|
|
|
|
let neg_inf: f32 = f32::NEG_INFINITY;
|
2015-02-14 23:18:41 +00:00
|
|
|
assert_eq!(10.0f32.log(10.0), 1.0);
|
|
|
|
assert_approx_eq!(2.3f32.log(3.5), 0.664858);
|
2015-04-18 06:45:55 +00:00
|
|
|
assert_eq!(1.0f32.exp().log(1.0f32.exp()), 1.0);
|
2015-02-14 23:18:41 +00:00
|
|
|
assert!(1.0f32.log(1.0).is_nan());
|
|
|
|
assert!(1.0f32.log(-13.9).is_nan());
|
|
|
|
assert!(nan.log(2.3).is_nan());
|
|
|
|
assert_eq!(inf.log(10.0), inf);
|
|
|
|
assert!(neg_inf.log(8.8).is_nan());
|
|
|
|
assert!((-2.3f32).log(0.1).is_nan());
|
|
|
|
assert_eq!((-0.0f32).log(2.0), neg_inf);
|
|
|
|
assert_eq!(0.0f32.log(7.0), neg_inf);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_log2() {
|
2015-04-18 06:45:55 +00:00
|
|
|
let nan: f32 = f32::NAN;
|
|
|
|
let inf: f32 = f32::INFINITY;
|
|
|
|
let neg_inf: f32 = f32::NEG_INFINITY;
|
2015-02-14 23:18:41 +00:00
|
|
|
assert_approx_eq!(10.0f32.log2(), 3.321928);
|
|
|
|
assert_approx_eq!(2.3f32.log2(), 1.201634);
|
|
|
|
assert_approx_eq!(1.0f32.exp().log2(), 1.442695);
|
|
|
|
assert!(nan.log2().is_nan());
|
|
|
|
assert_eq!(inf.log2(), inf);
|
|
|
|
assert!(neg_inf.log2().is_nan());
|
|
|
|
assert!((-2.3f32).log2().is_nan());
|
|
|
|
assert_eq!((-0.0f32).log2(), neg_inf);
|
|
|
|
assert_eq!(0.0f32.log2(), neg_inf);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_log10() {
|
2015-04-18 06:45:55 +00:00
|
|
|
let nan: f32 = f32::NAN;
|
|
|
|
let inf: f32 = f32::INFINITY;
|
|
|
|
let neg_inf: f32 = f32::NEG_INFINITY;
|
2015-02-14 23:18:41 +00:00
|
|
|
assert_eq!(10.0f32.log10(), 1.0);
|
|
|
|
assert_approx_eq!(2.3f32.log10(), 0.361728);
|
|
|
|
assert_approx_eq!(1.0f32.exp().log10(), 0.434294);
|
|
|
|
assert_eq!(1.0f32.log10(), 0.0);
|
|
|
|
assert!(nan.log10().is_nan());
|
|
|
|
assert_eq!(inf.log10(), inf);
|
|
|
|
assert!(neg_inf.log10().is_nan());
|
|
|
|
assert!((-2.3f32).log10().is_nan());
|
|
|
|
assert_eq!((-0.0f32).log10(), neg_inf);
|
|
|
|
assert_eq!(0.0f32.log10(), neg_inf);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_to_degrees() {
|
|
|
|
let pi: f32 = consts::PI;
|
2015-04-18 06:45:55 +00:00
|
|
|
let nan: f32 = f32::NAN;
|
|
|
|
let inf: f32 = f32::INFINITY;
|
|
|
|
let neg_inf: f32 = f32::NEG_INFINITY;
|
2015-02-14 23:18:41 +00:00
|
|
|
assert_eq!(0.0f32.to_degrees(), 0.0);
|
|
|
|
assert_approx_eq!((-5.8f32).to_degrees(), -332.315521);
|
|
|
|
assert_eq!(pi.to_degrees(), 180.0);
|
|
|
|
assert!(nan.to_degrees().is_nan());
|
|
|
|
assert_eq!(inf.to_degrees(), inf);
|
|
|
|
assert_eq!(neg_inf.to_degrees(), neg_inf);
|
2018-02-01 18:35:51 +00:00
|
|
|
assert_eq!(1_f32.to_degrees(), 57.2957795130823208767981548141051703);
|
2015-02-14 23:18:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_to_radians() {
|
|
|
|
let pi: f32 = consts::PI;
|
2015-04-18 06:45:55 +00:00
|
|
|
let nan: f32 = f32::NAN;
|
|
|
|
let inf: f32 = f32::INFINITY;
|
|
|
|
let neg_inf: f32 = f32::NEG_INFINITY;
|
2015-02-14 23:18:41 +00:00
|
|
|
assert_eq!(0.0f32.to_radians(), 0.0);
|
|
|
|
assert_approx_eq!(154.6f32.to_radians(), 2.698279);
|
|
|
|
assert_approx_eq!((-332.31f32).to_radians(), -5.799903);
|
|
|
|
assert_eq!(180.0f32.to_radians(), pi);
|
|
|
|
assert!(nan.to_radians().is_nan());
|
|
|
|
assert_eq!(inf.to_radians(), inf);
|
|
|
|
assert_eq!(neg_inf.to_radians(), neg_inf);
|
|
|
|
}
|
|
|
|
|
2013-05-13 14:11:35 +00:00
|
|
|
#[test]
|
|
|
|
fn test_asinh() {
|
|
|
|
assert_eq!(0.0f32.asinh(), 0.0f32);
|
|
|
|
assert_eq!((-0.0f32).asinh(), -0.0f32);
|
2013-08-08 18:38:10 +00:00
|
|
|
|
2015-04-18 06:45:55 +00:00
|
|
|
let inf: f32 = f32::INFINITY;
|
|
|
|
let neg_inf: f32 = f32::NEG_INFINITY;
|
|
|
|
let nan: f32 = f32::NAN;
|
2013-08-08 18:38:10 +00:00
|
|
|
assert_eq!(inf.asinh(), inf);
|
|
|
|
assert_eq!(neg_inf.asinh(), neg_inf);
|
2013-09-19 05:37:34 +00:00
|
|
|
assert!(nan.asinh().is_nan());
|
2013-05-13 14:11:35 +00:00
|
|
|
assert_approx_eq!(2.0f32.asinh(), 1.443635475178810342493276740273105f32);
|
|
|
|
assert_approx_eq!((-2.0f32).asinh(), -1.443635475178810342493276740273105f32);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_acosh() {
|
|
|
|
assert_eq!(1.0f32.acosh(), 0.0f32);
|
2013-09-19 05:37:34 +00:00
|
|
|
assert!(0.999f32.acosh().is_nan());
|
2013-08-08 18:38:10 +00:00
|
|
|
|
2015-04-18 06:45:55 +00:00
|
|
|
let inf: f32 = f32::INFINITY;
|
|
|
|
let neg_inf: f32 = f32::NEG_INFINITY;
|
|
|
|
let nan: f32 = f32::NAN;
|
2013-08-08 18:38:10 +00:00
|
|
|
assert_eq!(inf.acosh(), inf);
|
2013-09-19 05:37:34 +00:00
|
|
|
assert!(neg_inf.acosh().is_nan());
|
|
|
|
assert!(nan.acosh().is_nan());
|
2013-05-13 14:11:35 +00:00
|
|
|
assert_approx_eq!(2.0f32.acosh(), 1.31695789692481670862504634730796844f32);
|
|
|
|
assert_approx_eq!(3.0f32.acosh(), 1.76274717403908605046521864995958461f32);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_atanh() {
|
|
|
|
assert_eq!(0.0f32.atanh(), 0.0f32);
|
|
|
|
assert_eq!((-0.0f32).atanh(), -0.0f32);
|
2013-08-08 18:38:10 +00:00
|
|
|
|
2015-04-18 06:45:55 +00:00
|
|
|
let inf32: f32 = f32::INFINITY;
|
|
|
|
let neg_inf32: f32 = f32::NEG_INFINITY;
|
2013-08-08 18:38:10 +00:00
|
|
|
assert_eq!(1.0f32.atanh(), inf32);
|
|
|
|
assert_eq!((-1.0f32).atanh(), neg_inf32);
|
|
|
|
|
2013-09-19 05:37:34 +00:00
|
|
|
assert!(2f64.atanh().atanh().is_nan());
|
|
|
|
assert!((-2f64).atanh().atanh().is_nan());
|
2013-08-08 18:38:10 +00:00
|
|
|
|
2015-04-18 06:45:55 +00:00
|
|
|
let inf64: f32 = f32::INFINITY;
|
|
|
|
let neg_inf64: f32 = f32::NEG_INFINITY;
|
|
|
|
let nan32: f32 = f32::NAN;
|
2013-09-19 05:37:34 +00:00
|
|
|
assert!(inf64.atanh().is_nan());
|
|
|
|
assert!(neg_inf64.atanh().is_nan());
|
|
|
|
assert!(nan32.atanh().is_nan());
|
2013-08-08 18:38:10 +00:00
|
|
|
|
2013-05-13 14:11:35 +00:00
|
|
|
assert_approx_eq!(0.5f32.atanh(), 0.54930614433405484569762261846126285f32);
|
|
|
|
assert_approx_eq!((-0.5f32).atanh(), -0.54930614433405484569762261846126285f32);
|
|
|
|
}
|
|
|
|
|
2013-04-24 22:12:26 +00:00
|
|
|
#[test]
|
|
|
|
fn test_real_consts() {
|
2014-12-23 19:53:35 +00:00
|
|
|
use super::consts;
|
|
|
|
|
|
|
|
let pi: f32 = consts::PI;
|
|
|
|
let frac_pi_2: f32 = consts::FRAC_PI_2;
|
|
|
|
let frac_pi_3: f32 = consts::FRAC_PI_3;
|
|
|
|
let frac_pi_4: f32 = consts::FRAC_PI_4;
|
|
|
|
let frac_pi_6: f32 = consts::FRAC_PI_6;
|
|
|
|
let frac_pi_8: f32 = consts::FRAC_PI_8;
|
|
|
|
let frac_1_pi: f32 = consts::FRAC_1_PI;
|
|
|
|
let frac_2_pi: f32 = consts::FRAC_2_PI;
|
2015-04-18 06:45:55 +00:00
|
|
|
let frac_2_sqrtpi: f32 = consts::FRAC_2_SQRT_PI;
|
|
|
|
let sqrt2: f32 = consts::SQRT_2;
|
|
|
|
let frac_1_sqrt2: f32 = consts::FRAC_1_SQRT_2;
|
2014-12-23 19:53:35 +00:00
|
|
|
let e: f32 = consts::E;
|
|
|
|
let log2_e: f32 = consts::LOG2_E;
|
|
|
|
let log10_e: f32 = consts::LOG10_E;
|
|
|
|
let ln_2: f32 = consts::LN_2;
|
|
|
|
let ln_10: f32 = consts::LN_10;
|
2013-08-08 18:38:10 +00:00
|
|
|
|
|
|
|
assert_approx_eq!(frac_pi_2, pi / 2f32);
|
|
|
|
assert_approx_eq!(frac_pi_3, pi / 3f32);
|
|
|
|
assert_approx_eq!(frac_pi_4, pi / 4f32);
|
|
|
|
assert_approx_eq!(frac_pi_6, pi / 6f32);
|
|
|
|
assert_approx_eq!(frac_pi_8, pi / 8f32);
|
|
|
|
assert_approx_eq!(frac_1_pi, 1f32 / pi);
|
|
|
|
assert_approx_eq!(frac_2_pi, 2f32 / pi);
|
|
|
|
assert_approx_eq!(frac_2_sqrtpi, 2f32 / pi.sqrt());
|
|
|
|
assert_approx_eq!(sqrt2, 2f32.sqrt());
|
|
|
|
assert_approx_eq!(frac_1_sqrt2, 1f32 / 2f32.sqrt());
|
|
|
|
assert_approx_eq!(log2_e, e.log2());
|
|
|
|
assert_approx_eq!(log10_e, e.log10());
|
|
|
|
assert_approx_eq!(ln_2, 2f32.ln());
|
|
|
|
assert_approx_eq!(ln_10, 10f32.ln());
|
2013-04-24 22:12:26 +00:00
|
|
|
}
|
2017-01-24 17:15:11 +00:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_float_bits_conv() {
|
|
|
|
assert_eq!((1f32).to_bits(), 0x3f800000);
|
|
|
|
assert_eq!((12.5f32).to_bits(), 0x41480000);
|
|
|
|
assert_eq!((1337f32).to_bits(), 0x44a72000);
|
|
|
|
assert_eq!((-14.25f32).to_bits(), 0xc1640000);
|
2017-03-29 00:39:59 +00:00
|
|
|
assert_approx_eq!(f32::from_bits(0x3f800000), 1.0);
|
|
|
|
assert_approx_eq!(f32::from_bits(0x41480000), 12.5);
|
|
|
|
assert_approx_eq!(f32::from_bits(0x44a72000), 1337.0);
|
|
|
|
assert_approx_eq!(f32::from_bits(0xc1640000), -14.25);
|
Make float::from_bits transmute (and update the documentation to reflect this).
The current implementation/documentation was made to avoid sNaN because of
potential safety issues implied by old/bad LLVM documentation. These issues
aren't real, so we can just make the implementation transmute (as permitted
by the existing documentation of this method).
Also the documentation didn't actually match the behaviour: it said we may
change sNaNs, but in fact we canonicalized *all* NaNs.
Also an example in the documentation was wrong: it said we *always* change
sNaNs, when the documentation was explicitly written to indicate it was
implementation-defined.
This makes to_bits and from_bits perfectly roundtrip cross-platform, except
for one caveat: although the 2008 edition of IEEE-754 specifies how to
interpet the signaling bit, earlier editions didn't. This lead to some platforms
picking the opposite interpretation, so all signaling NaNs on x86/ARM are quiet
on MIPS, and vice-versa.
NaN-boxing is a fairly important optimization, while we don't even guarantee
that float operations properly preserve signalingness. As such, this seems like
the more natural strategy to take (as opposed to trying to mangle the signaling
bit on a per-platform basis).
This implementation is also, of course, faster.
2017-11-15 20:05:47 +00:00
|
|
|
|
|
|
|
// Check that NaNs roundtrip their bits regardless of signalingness
|
|
|
|
// 0xA is 0b1010; 0x5 is 0b0101 -- so these two together clobbers all the mantissa bits
|
|
|
|
let masked_nan1 = f32::NAN.to_bits() ^ 0x002A_AAAA;
|
|
|
|
let masked_nan2 = f32::NAN.to_bits() ^ 0x0055_5555;
|
|
|
|
assert!(f32::from_bits(masked_nan1).is_nan());
|
|
|
|
assert!(f32::from_bits(masked_nan2).is_nan());
|
|
|
|
|
|
|
|
assert_eq!(f32::from_bits(masked_nan1).to_bits(), masked_nan1);
|
|
|
|
assert_eq!(f32::from_bits(masked_nan2).to_bits(), masked_nan2);
|
2017-03-29 23:49:06 +00:00
|
|
|
}
|
2013-04-23 07:59:49 +00:00
|
|
|
}
|