use crate::{FloatChecker, NoisyFloat};
use core::{
cmp::Ordering,
convert::{From, TryFrom},
hash::{Hash, Hasher},
iter,
mem::transmute,
num::FpCategory,
ops::{Add, AddAssign, Div, DivAssign, Mul, MulAssign, Neg, Rem, RemAssign, Sub, SubAssign},
};
use num_traits::{
cast::{FromPrimitive, NumCast, ToPrimitive},
identities::{One, Zero},
Bounded, Float, FloatConst, Num, Signed,
};
impl<F: Float, C: FloatChecker<F>> Clone for NoisyFloat<F, C> {
#[inline]
fn clone(&self) -> Self {
Self::unchecked_new_generic(self.value)
}
}
impl<F: Float, C: FloatChecker<F>> Copy for NoisyFloat<F, C> {}
impl<F: Float, C: FloatChecker<F>> AsRef<F> for NoisyFloat<F, C> {
fn as_ref(&self) -> &F {
&self.value
}
}
impl<F: Float, C: FloatChecker<F>> PartialEq<F> for NoisyFloat<F, C> {
#[inline]
fn eq(&self, other: &F) -> bool {
self.value.eq(&other)
}
}
impl<F: Float, C: FloatChecker<F>> PartialEq for NoisyFloat<F, C> {
#[inline]
fn eq(&self, other: &Self) -> bool {
self.eq(&other.value)
}
}
impl<F: Float, C: FloatChecker<F>> Eq for NoisyFloat<F, C> {}
impl<F: Float, C: FloatChecker<F>> PartialOrd<F> for NoisyFloat<F, C> {
#[inline]
fn partial_cmp(&self, other: &F) -> Option<Ordering> {
self.value.partial_cmp(&other)
}
#[inline]
fn lt(&self, other: &F) -> bool {
self.value.lt(&other)
}
#[inline]
fn le(&self, other: &F) -> bool {
self.value.le(&other)
}
#[inline]
fn gt(&self, other: &F) -> bool {
self.value.gt(&other)
}
#[inline]
fn ge(&self, other: &F) -> bool {
self.value.ge(&other)
}
}
impl<F: Float, C: FloatChecker<F>> PartialOrd for NoisyFloat<F, C> {
#[inline]
fn partial_cmp(&self, other: &Self) -> Option<Ordering> {
self.value.partial_cmp(&other.value)
}
#[inline]
fn lt(&self, other: &Self) -> bool {
self.lt(&other.value)
}
#[inline]
fn le(&self, other: &Self) -> bool {
self.le(&other.value)
}
#[inline]
fn gt(&self, other: &Self) -> bool {
self.gt(&other.value)
}
#[inline]
fn ge(&self, other: &Self) -> bool {
self.ge(&other.value)
}
}
impl<F: Float, C: FloatChecker<F>> Ord for NoisyFloat<F, C> {
#[inline]
fn cmp(&self, other: &Self) -> Ordering {
if self.value < other.value {
Ordering::Less
} else if self.value == other.value {
Ordering::Equal
} else {
Ordering::Greater
}
}
}
impl<C: FloatChecker<f32>> Hash for NoisyFloat<f32, C> {
#[inline]
fn hash<H: Hasher>(&self, state: &mut H) {
let bits = if self.value == 0.0 {
0
} else {
unsafe { transmute::<f32, u32>(self.value) }
};
bits.hash(state);
}
}
impl<C: FloatChecker<f64>> Hash for NoisyFloat<f64, C> {
#[inline]
fn hash<H: Hasher>(&self, state: &mut H) {
let bits = if self.value == 0.0 {
0
} else {
unsafe { transmute::<f64, u64>(self.value) }
};
bits.hash(state);
}
}
impl<F: Float, C: FloatChecker<F>> Add<F> for NoisyFloat<F, C> {
type Output = Self;
#[inline]
fn add(self, rhs: F) -> Self {
Self::new(self.value.add(rhs))
}
}
impl<'a, F: Float, C: FloatChecker<F>> Add<&'a F> for NoisyFloat<F, C> {
type Output = Self;
#[inline]
fn add(self, rhs: &'a F) -> Self {
Self::new(self.value.add(*rhs))
}
}
impl<F: Float, C: FloatChecker<F>> Add for NoisyFloat<F, C> {
type Output = Self;
#[inline]
fn add(self, rhs: Self) -> Self {
self.add(rhs.value)
}
}
impl<'a, F: Float, C: FloatChecker<F>> Add<&'a Self> for NoisyFloat<F, C> {
type Output = Self;
#[inline]
fn add(self, rhs: &'a Self) -> Self {
self.add(rhs.value)
}
}
impl<F: Float, C: FloatChecker<F>> Sub<F> for NoisyFloat<F, C> {
type Output = Self;
#[inline]
fn sub(self, rhs: F) -> Self {
Self::new(self.value.sub(rhs))
}
}
impl<'a, F: Float, C: FloatChecker<F>> Sub<&'a F> for NoisyFloat<F, C> {
type Output = Self;
#[inline]
fn sub(self, rhs: &'a F) -> Self {
Self::new(self.value.sub(*rhs))
}
}
impl<F: Float, C: FloatChecker<F>> Sub for NoisyFloat<F, C> {
type Output = Self;
#[inline]
fn sub(self, rhs: Self) -> Self {
self.sub(rhs.value)
}
}
impl<'a, F: Float, C: FloatChecker<F>> Sub<&'a Self> for NoisyFloat<F, C> {
type Output = Self;
#[inline]
fn sub(self, rhs: &'a Self) -> Self {
self.sub(rhs.value)
}
}
impl<F: Float, C: FloatChecker<F>> Mul<F> for NoisyFloat<F, C> {
type Output = Self;
#[inline]
fn mul(self, rhs: F) -> Self {
Self::new(self.value.mul(rhs))
}
}
impl<'a, F: Float, C: FloatChecker<F>> Mul<&'a F> for NoisyFloat<F, C> {
type Output = Self;
#[inline]
fn mul(self, rhs: &'a F) -> Self {
Self::new(self.value.mul(*rhs))
}
}
impl<F: Float, C: FloatChecker<F>> Mul for NoisyFloat<F, C> {
type Output = Self;
#[inline]
fn mul(self, rhs: Self) -> Self {
self.mul(rhs.value)
}
}
impl<'a, F: Float, C: FloatChecker<F>> Mul<&'a Self> for NoisyFloat<F, C> {
type Output = Self;
#[inline]
fn mul(self, rhs: &'a Self) -> Self {
self.mul(rhs.value)
}
}
impl<F: Float, C: FloatChecker<F>> Div<F> for NoisyFloat<F, C> {
type Output = Self;
#[inline]
fn div(self, rhs: F) -> Self {
Self::new(self.value.div(rhs))
}
}
impl<'a, F: Float, C: FloatChecker<F>> Div<&'a F> for NoisyFloat<F, C> {
type Output = Self;
#[inline]
fn div(self, rhs: &'a F) -> Self {
Self::new(self.value.div(*rhs))
}
}
impl<F: Float, C: FloatChecker<F>> Div for NoisyFloat<F, C> {
type Output = Self;
#[inline]
fn div(self, rhs: Self) -> Self {
self.div(rhs.value)
}
}
impl<'a, F: Float, C: FloatChecker<F>> Div<&'a Self> for NoisyFloat<F, C> {
type Output = Self;
#[inline]
fn div(self, rhs: &'a Self) -> Self {
self.div(rhs.value)
}
}
impl<F: Float, C: FloatChecker<F>> Rem<F> for NoisyFloat<F, C> {
type Output = Self;
#[inline]
fn rem(self, rhs: F) -> Self {
Self::new(self.value.rem(rhs))
}
}
impl<'a, F: Float, C: FloatChecker<F>> Rem<&'a F> for NoisyFloat<F, C> {
type Output = Self;
#[inline]
fn rem(self, rhs: &'a F) -> Self {
Self::new(self.value.rem(*rhs))
}
}
impl<F: Float, C: FloatChecker<F>> Rem for NoisyFloat<F, C> {
type Output = Self;
#[inline]
fn rem(self, rhs: Self) -> Self {
self.rem(rhs.value)
}
}
impl<'a, F: Float, C: FloatChecker<F>> Rem<&'a Self> for NoisyFloat<F, C> {
type Output = Self;
#[inline]
fn rem(self, rhs: &'a Self) -> Self {
self.rem(rhs.value)
}
}
impl<F: Float + AddAssign, C: FloatChecker<F>> AddAssign<F> for NoisyFloat<F, C> {
#[inline]
fn add_assign(&mut self, rhs: F) {
self.value.add_assign(rhs);
C::assert(self.value);
}
}
impl<'a, F: Float + AddAssign, C: FloatChecker<F>> AddAssign<&'a F> for NoisyFloat<F, C> {
#[inline]
fn add_assign(&mut self, rhs: &'a F) {
self.value.add_assign(*rhs);
C::assert(self.value);
}
}
impl<F: Float + AddAssign, C: FloatChecker<F>> AddAssign for NoisyFloat<F, C> {
#[inline]
fn add_assign(&mut self, rhs: Self) {
self.add_assign(rhs.value);
}
}
impl<'a, F: Float + AddAssign, C: FloatChecker<F>> AddAssign<&'a Self> for NoisyFloat<F, C> {
#[inline]
fn add_assign(&mut self, rhs: &'a Self) {
self.add_assign(rhs.value);
}
}
impl<F: Float + SubAssign, C: FloatChecker<F>> SubAssign<F> for NoisyFloat<F, C> {
#[inline]
fn sub_assign(&mut self, rhs: F) {
self.value.sub_assign(rhs);
C::assert(self.value);
}
}
impl<'a, F: Float + SubAssign, C: FloatChecker<F>> SubAssign<&'a F> for NoisyFloat<F, C> {
#[inline]
fn sub_assign(&mut self, rhs: &'a F) {
self.value.sub_assign(*rhs);
C::assert(self.value);
}
}
impl<F: Float + SubAssign, C: FloatChecker<F>> SubAssign for NoisyFloat<F, C> {
#[inline]
fn sub_assign(&mut self, rhs: Self) {
self.sub_assign(rhs.value);
}
}
impl<'a, F: Float + SubAssign, C: FloatChecker<F>> SubAssign<&'a Self> for NoisyFloat<F, C> {
#[inline]
fn sub_assign(&mut self, rhs: &'a Self) {
self.sub_assign(rhs.value);
}
}
impl<F: Float + MulAssign, C: FloatChecker<F>> MulAssign<F> for NoisyFloat<F, C> {
#[inline]
fn mul_assign(&mut self, rhs: F) {
self.value.mul_assign(rhs);
C::assert(self.value);
}
}
impl<'a, F: Float + MulAssign, C: FloatChecker<F>> MulAssign<&'a F> for NoisyFloat<F, C> {
#[inline]
fn mul_assign(&mut self, rhs: &'a F) {
self.value.mul_assign(*rhs);
C::assert(self.value);
}
}
impl<F: Float + MulAssign, C: FloatChecker<F>> MulAssign for NoisyFloat<F, C> {
#[inline]
fn mul_assign(&mut self, rhs: Self) {
self.mul_assign(rhs.value);
}
}
impl<'a, F: Float + MulAssign, C: FloatChecker<F>> MulAssign<&'a Self> for NoisyFloat<F, C> {
#[inline]
fn mul_assign(&mut self, rhs: &'a Self) {
self.mul_assign(rhs.value);
}
}
impl<F: Float + DivAssign, C: FloatChecker<F>> DivAssign<F> for NoisyFloat<F, C> {
#[inline]
fn div_assign(&mut self, rhs: F) {
self.value.div_assign(rhs);
C::assert(self.value);
}
}
impl<'a, F: Float + DivAssign, C: FloatChecker<F>> DivAssign<&'a F> for NoisyFloat<F, C> {
#[inline]
fn div_assign(&mut self, rhs: &'a F) {
self.value.div_assign(*rhs);
C::assert(self.value);
}
}
impl<F: Float + DivAssign, C: FloatChecker<F>> DivAssign for NoisyFloat<F, C> {
#[inline]
fn div_assign(&mut self, rhs: Self) {
self.div_assign(rhs.value);
}
}
impl<'a, F: Float + DivAssign, C: FloatChecker<F>> DivAssign<&'a Self> for NoisyFloat<F, C> {
#[inline]
fn div_assign(&mut self, rhs: &'a Self) {
self.div_assign(rhs.value);
}
}
impl<F: Float + RemAssign, C: FloatChecker<F>> RemAssign<F> for NoisyFloat<F, C> {
#[inline]
fn rem_assign(&mut self, rhs: F) {
self.value.rem_assign(rhs);
C::assert(self.value);
}
}
impl<'a, F: Float + RemAssign, C: FloatChecker<F>> RemAssign<&'a F> for NoisyFloat<F, C> {
#[inline]
fn rem_assign(&mut self, rhs: &'a F) {
self.value.rem_assign(*rhs);
C::assert(self.value);
}
}
impl<F: Float + RemAssign, C: FloatChecker<F>> RemAssign for NoisyFloat<F, C> {
#[inline]
fn rem_assign(&mut self, rhs: Self) {
self.rem_assign(rhs.value);
}
}
impl<'a, F: Float + RemAssign, C: FloatChecker<F>> RemAssign<&'a Self> for NoisyFloat<F, C> {
#[inline]
fn rem_assign(&mut self, rhs: &'a Self) {
self.rem_assign(rhs.value);
}
}
impl<F: Float, C: FloatChecker<F>> Neg for NoisyFloat<F, C> {
type Output = Self;
#[inline]
fn neg(self) -> Self {
Self::new(self.value.neg())
}
}
impl<'a, F: Float, C: FloatChecker<F>> Neg for &'a NoisyFloat<F, C> {
type Output = NoisyFloat<F, C>;
#[inline]
fn neg(self) -> Self::Output {
Self::Output::neg(*self)
}
}
impl<F: Float, C: FloatChecker<F>> Zero for NoisyFloat<F, C> {
#[inline]
fn zero() -> Self {
Self::new(F::zero())
}
#[inline]
fn is_zero(&self) -> bool {
self.value.is_zero()
}
}
impl<F: Float, C: FloatChecker<F>> One for NoisyFloat<F, C> {
#[inline]
fn one() -> Self {
Self::new(F::one())
}
}
impl<F: Float, C: FloatChecker<F>> Num for NoisyFloat<F, C> {
type FromStrRadixErr = F::FromStrRadixErr;
#[inline]
fn from_str_radix(str: &str, radix: u32) -> Result<Self, Self::FromStrRadixErr> {
F::from_str_radix(str, radix).map(|v| Self::new(v))
}
}
impl<F: Float, C: FloatChecker<F>> ToPrimitive for NoisyFloat<F, C> {
#[inline]
fn to_i64(&self) -> Option<i64> {
self.value.to_i64()
}
#[inline]
fn to_u64(&self) -> Option<u64> {
self.value.to_u64()
}
#[inline]
fn to_isize(&self) -> Option<isize> {
self.value.to_isize()
}
#[inline]
fn to_i8(&self) -> Option<i8> {
self.value.to_i8()
}
#[inline]
fn to_i16(&self) -> Option<i16> {
self.value.to_i16()
}
#[inline]
fn to_i32(&self) -> Option<i32> {
self.value.to_i32()
}
#[inline]
fn to_usize(&self) -> Option<usize> {
self.value.to_usize()
}
#[inline]
fn to_u8(&self) -> Option<u8> {
self.value.to_u8()
}
#[inline]
fn to_u16(&self) -> Option<u16> {
self.value.to_u16()
}
#[inline]
fn to_u32(&self) -> Option<u32> {
self.value.to_u32()
}
#[inline]
fn to_f32(&self) -> Option<f32> {
self.value.to_f32()
}
#[inline]
fn to_f64(&self) -> Option<f64> {
self.value.to_f64()
}
}
impl<F: Float + FromPrimitive, C: FloatChecker<F>> FromPrimitive for NoisyFloat<F, C> {
#[inline]
fn from_isize(n: isize) -> Option<Self> {
Self::try_new(F::from_isize(n)?)
}
#[inline]
fn from_i8(n: i8) -> Option<Self> {
Self::try_new(F::from_i8(n)?)
}
#[inline]
fn from_i16(n: i16) -> Option<Self> {
Self::try_new(F::from_i16(n)?)
}
#[inline]
fn from_i32(n: i32) -> Option<Self> {
Self::try_new(F::from_i32(n)?)
}
#[inline]
fn from_i64(n: i64) -> Option<Self> {
Self::try_new(F::from_i64(n)?)
}
#[inline]
fn from_i128(n: i128) -> Option<Self> {
Self::try_new(F::from_i128(n)?)
}
#[inline]
fn from_usize(n: usize) -> Option<Self> {
Self::try_new(F::from_usize(n)?)
}
#[inline]
fn from_u8(n: u8) -> Option<Self> {
Self::try_new(F::from_u8(n)?)
}
#[inline]
fn from_u16(n: u16) -> Option<Self> {
Self::try_new(F::from_u16(n)?)
}
#[inline]
fn from_u32(n: u32) -> Option<Self> {
Self::try_new(F::from_u32(n)?)
}
#[inline]
fn from_u64(n: u64) -> Option<Self> {
Self::try_new(F::from_u64(n)?)
}
#[inline]
fn from_u128(n: u128) -> Option<Self> {
Self::try_new(F::from_u128(n)?)
}
#[inline]
fn from_f32(n: f32) -> Option<Self> {
Self::try_new(F::from_f32(n)?)
}
#[inline]
fn from_f64(n: f64) -> Option<Self> {
Self::try_new(F::from_f64(n)?)
}
}
impl<F: Float, C: FloatChecker<F>> NumCast for NoisyFloat<F, C> {
#[inline]
fn from<T: ToPrimitive>(n: T) -> Option<Self> {
F::from(n).and_then(|v| Self::try_new(v))
}
}
impl<C: FloatChecker<f32>> From<NoisyFloat<f32, C>> for f32 {
#[inline]
fn from(n: NoisyFloat<f32, C>) -> Self {
n.value
}
}
impl<C: FloatChecker<f64>> From<NoisyFloat<f64, C>> for f64 {
#[inline]
fn from(n: NoisyFloat<f64, C>) -> Self {
n.value
}
}
impl<C: FloatChecker<f32>> From<NoisyFloat<f32, C>> for f64 {
#[inline]
fn from(n: NoisyFloat<f32, C>) -> Self {
n.value as f64
}
}
impl<C: FloatChecker<f64>> TryFrom<f64> for NoisyFloat<f64, C> {
type Error = &'static str;
#[inline]
fn try_from(f: f64) -> Result<Self, Self::Error> {
Self::try_new(f).ok_or("illegal value")
}
}
impl<C: FloatChecker<f32>> TryFrom<f32> for NoisyFloat<f32, C> {
type Error = &'static str;
#[inline]
fn try_from(f: f32) -> Result<Self, Self::Error> {
Self::try_new(f).ok_or("illegal value")
}
}
impl<F: Float, C: FloatChecker<F>> Float for NoisyFloat<F, C> {
#[inline]
fn nan() -> Self {
panic!("unexpected NaN")
}
#[inline]
fn infinity() -> Self {
Self::new(F::infinity())
}
#[inline]
fn neg_infinity() -> Self {
Self::new(F::neg_infinity())
}
#[inline]
fn neg_zero() -> Self {
Self::new(F::neg_zero())
}
#[inline]
fn min_value() -> Self {
Self::new(F::min_value())
}
#[inline]
fn min_positive_value() -> Self {
Self::new(F::min_positive_value())
}
#[inline]
fn max_value() -> Self {
Self::new(F::max_value())
}
#[inline]
fn is_nan(self) -> bool {
self.value.is_nan()
}
#[inline]
fn is_infinite(self) -> bool {
self.value.is_infinite()
}
#[inline]
fn is_finite(self) -> bool {
self.value.is_finite()
}
#[inline]
fn is_normal(self) -> bool {
self.value.is_normal()
}
#[inline]
fn classify(self) -> FpCategory {
self.value.classify()
}
#[inline]
fn floor(self) -> Self {
Self::new(self.value.floor())
}
#[inline]
fn ceil(self) -> Self {
Self::new(self.value.ceil())
}
#[inline]
fn round(self) -> Self {
Self::new(self.value.round())
}
#[inline]
fn trunc(self) -> Self {
Self::new(self.value.trunc())
}
#[inline]
fn fract(self) -> Self {
Self::new(self.value.fract())
}
#[inline]
fn abs(self) -> Self {
Self::new(self.value.abs())
}
#[inline]
fn signum(self) -> Self {
Self::new(self.value.signum())
}
#[inline]
fn is_sign_positive(self) -> bool {
self.value.is_sign_positive()
}
#[inline]
fn is_sign_negative(self) -> bool {
self.value.is_sign_negative()
}
#[inline]
fn mul_add(self, a: Self, b: Self) -> Self {
Self::new(self.value.mul_add(a.value, b.value))
}
#[inline]
fn recip(self) -> Self {
Self::new(self.value.recip())
}
#[inline]
fn powi(self, n: i32) -> Self {
Self::new(self.value.powi(n))
}
#[inline]
fn powf(self, n: Self) -> Self {
Self::new(self.value.powf(n.value))
}
#[inline]
fn sqrt(self) -> Self {
Self::new(self.value.sqrt())
}
#[inline]
fn exp(self) -> Self {
Self::new(self.value.exp())
}
#[inline]
fn exp2(self) -> Self {
Self::new(self.value.exp2())
}
#[inline]
fn ln(self) -> Self {
Self::new(self.value.ln())
}
#[inline]
fn log(self, base: Self) -> Self {
Self::new(self.value.log(base.value))
}
#[inline]
fn log2(self) -> Self {
Self::new(self.value.log2())
}
#[inline]
fn log10(self) -> Self {
Self::new(self.value.log10())
}
#[inline]
fn max(self, other: Self) -> Self {
Self::new(self.value.max(other.value))
}
#[inline]
fn min(self, other: Self) -> Self {
Self::new(self.value.min(other.value))
}
#[inline]
fn abs_sub(self, other: Self) -> Self {
Self::new(self.value.abs_sub(other.value))
}
#[inline]
fn cbrt(self) -> Self {
Self::new(self.value.cbrt())
}
#[inline]
fn hypot(self, other: Self) -> Self {
Self::new(self.value.hypot(other.value))
}
#[inline]
fn sin(self) -> Self {
Self::new(self.value.sin())
}
#[inline]
fn cos(self) -> Self {
Self::new(self.value.cos())
}
#[inline]
fn tan(self) -> Self {
Self::new(self.value.tan())
}
#[inline]
fn asin(self) -> Self {
Self::new(self.value.asin())
}
#[inline]
fn acos(self) -> Self {
Self::new(self.value.acos())
}
#[inline]
fn atan(self) -> Self {
Self::new(self.value.atan())
}
#[inline]
fn atan2(self, other: Self) -> Self {
Self::new(self.value.atan2(other.value))
}
#[inline]
fn sin_cos(self) -> (Self, Self) {
let (a, b) = self.value.sin_cos();
(Self::new(a), Self::new(b))
}
#[inline]
fn exp_m1(self) -> Self {
Self::new(self.value.exp_m1())
}
#[inline]
fn ln_1p(self) -> Self {
Self::new(self.value.ln_1p())
}
#[inline]
fn sinh(self) -> Self {
Self::new(self.value.sinh())
}
#[inline]
fn cosh(self) -> Self {
Self::new(self.value.cosh())
}
#[inline]
fn tanh(self) -> Self {
Self::new(self.value.tanh())
}
#[inline]
fn asinh(self) -> Self {
Self::new(self.value.asinh())
}
#[inline]
fn acosh(self) -> Self {
Self::new(self.value.acosh())
}
#[inline]
fn atanh(self) -> Self {
Self::new(self.value.atanh())
}
#[inline]
fn integer_decode(self) -> (u64, i16, i8) {
self.value.integer_decode()
}
#[inline]
fn epsilon() -> Self {
Self::new(F::epsilon())
}
#[inline]
fn to_degrees(self) -> Self {
Self::new(self.value.to_degrees())
}
#[inline]
fn to_radians(self) -> Self {
Self::new(self.value.to_radians())
}
}
impl<F: Float + FloatConst, C: FloatChecker<F>> FloatConst for NoisyFloat<F, C> {
#[inline]
fn E() -> Self {
Self::new(F::E())
}
#[inline]
fn FRAC_1_PI() -> Self {
Self::new(F::FRAC_1_PI())
}
#[inline]
fn FRAC_1_SQRT_2() -> Self {
Self::new(F::FRAC_1_SQRT_2())
}
#[inline]
fn FRAC_2_PI() -> Self {
Self::new(F::FRAC_2_PI())
}
#[inline]
fn FRAC_2_SQRT_PI() -> Self {
Self::new(F::FRAC_2_SQRT_PI())
}
#[inline]
fn FRAC_PI_2() -> Self {
Self::new(F::FRAC_PI_2())
}
#[inline]
fn FRAC_PI_3() -> Self {
Self::new(F::FRAC_PI_3())
}
#[inline]
fn FRAC_PI_4() -> Self {
Self::new(F::FRAC_PI_4())
}
#[inline]
fn FRAC_PI_6() -> Self {
Self::new(F::FRAC_PI_6())
}
#[inline]
fn FRAC_PI_8() -> Self {
Self::new(F::FRAC_PI_8())
}
#[inline]
fn LN_10() -> Self {
Self::new(F::LN_10())
}
#[inline]
fn LN_2() -> Self {
Self::new(F::LN_2())
}
#[inline]
fn LOG10_E() -> Self {
Self::new(F::LOG10_E())
}
#[inline]
fn LOG2_E() -> Self {
Self::new(F::LOG2_E())
}
#[inline]
fn PI() -> Self {
Self::new(F::PI())
}
#[inline]
fn SQRT_2() -> Self {
Self::new(F::SQRT_2())
}
}
impl<F: Float + Signed, C: FloatChecker<F>> Signed for NoisyFloat<F, C> {
#[inline]
fn abs(&self) -> Self {
Self::new(self.value.abs())
}
#[inline]
fn abs_sub(&self, other: &Self) -> Self {
Self::new(self.value.abs_sub(other.value))
}
#[inline]
fn signum(&self) -> Self {
Self::new(self.value.signum())
}
#[inline]
fn is_positive(&self) -> bool {
self.value.is_positive()
}
#[inline]
fn is_negative(&self) -> bool {
self.value.is_negative()
}
}
impl<F: Float + Bounded, C: FloatChecker<F>> Bounded for NoisyFloat<F, C> {
#[inline]
fn min_value() -> Self {
Self::new(Float::min_value())
}
#[inline]
fn max_value() -> Self {
Self::new(Float::max_value())
}
}
impl<F: Float, C: FloatChecker<F>> iter::Sum for NoisyFloat<F, C> {
fn sum<I>(iter: I) -> Self
where
I: Iterator<Item = Self>,
{
Self::new(iter.map(|i| i.raw()).fold(F::zero(), |acc, i| acc + i))
}
}
impl<'a, F: Float, C: FloatChecker<F>> iter::Sum<&'a Self> for NoisyFloat<F, C> {
fn sum<I>(iter: I) -> Self
where
I: Iterator<Item = &'a Self>,
{
Self::new(iter.map(|i| i.raw()).fold(F::zero(), |acc, i| acc + i))
}
}
impl<F: Float, C: FloatChecker<F>> iter::Product for NoisyFloat<F, C> {
fn product<I>(iter: I) -> Self
where
I: Iterator<Item = Self>,
{
Self::new(iter.map(|i| i.raw()).fold(F::one(), |acc, i| acc * i))
}
}
impl<'a, F: Float, C: FloatChecker<F>> iter::Product<&'a Self> for NoisyFloat<F, C> {
fn product<I>(iter: I) -> Self
where
I: Iterator<Item = &'a Self>,
{
Self::new(iter.map(|i| i.raw()).fold(F::one(), |acc, i| acc * i))
}
}
#[cfg(feature = "approx")]
mod approx_impl {
use super::*;
use approx::{AbsDiffEq, RelativeEq, UlpsEq};
impl<F, C> AbsDiffEq<Self> for NoisyFloat<F, C>
where
F: Float + AbsDiffEq<Epsilon = F>,
C: FloatChecker<F>,
{
type Epsilon = NoisyFloat<F, C>;
fn default_epsilon() -> Self::Epsilon {
Self::Epsilon::new(F::default_epsilon())
}
fn abs_diff_eq(&self, other: &Self, epsilon: Self::Epsilon) -> bool {
self.raw().abs_diff_eq(&other.raw(), epsilon.raw())
}
}
impl<F, C> RelativeEq<Self> for NoisyFloat<F, C>
where
F: Float + RelativeEq<Epsilon = F>,
C: FloatChecker<F>,
{
fn default_max_relative() -> Self::Epsilon {
Self::new(F::default_max_relative())
}
fn relative_eq(
&self,
other: &Self,
epsilon: Self::Epsilon,
max_relative: Self::Epsilon,
) -> bool {
self.raw()
.relative_eq(&other.raw(), epsilon.raw(), max_relative.raw())
}
}
impl<F, C> UlpsEq<Self> for NoisyFloat<F, C>
where
F: Float + UlpsEq<Epsilon = F>,
C: FloatChecker<F>,
{
fn default_max_ulps() -> u32 {
F::default_max_ulps()
}
fn ulps_eq(&self, other: &Self, epsilon: Self::Epsilon, max_ulps: u32) -> bool {
self.raw().ulps_eq(&other.raw(), epsilon.raw(), max_ulps)
}
}
}