#![allow(dead_code)]
pub(crate) use lib::{f32, f64, mem};
use lib::{fmt, iter, ops};
use super::cast::{AsCast, TryCast};
use super::primitive::Primitive;
pub trait Number:
Primitive +
iter::Product + iter::Sum +
ops::Add<Output=Self> +
ops::AddAssign +
ops::Div<Output=Self> +
ops::DivAssign +
ops::Mul<Output=Self> +
ops::MulAssign +
ops::Rem<Output=Self> +
ops::RemAssign +
ops::Sub<Output=Self> +
ops::SubAssign
{}
macro_rules! number_impl {
($($t:ty)*) => ($(
impl Number for $t {}
)*)
}
number_impl! { u8 u16 u32 u64 usize i8 i16 i32 i64 isize f32 f64 }
#[cfg(has_i128)] number_impl! { u128 i128 }
pub trait Integer:
Number + Eq + Ord +
fmt::Octal + fmt::LowerHex + fmt::UpperHex +
ops::BitAnd<Output=Self> +
ops::BitAndAssign +
ops::BitOr<Output=Self> +
ops::BitOrAssign +
ops::BitXor<Output=Self> +
ops::BitXorAssign +
ops::Not +
ops::Shl<Self, Output=Self> +
ops::Shl<u8, Output=Self> +
ops::Shl<u16, Output=Self> +
ops::Shl<u32, Output=Self> +
ops::Shl<u64, Output=Self> +
ops::Shl<usize, Output=Self> +
ops::Shl<i8, Output=Self> +
ops::Shl<i16, Output=Self> +
ops::Shl<i64, Output=Self> +
ops::Shl<isize, Output=Self> +
ops::Shl<i32, Output=Self> +
ops::ShlAssign<Self> +
ops::ShlAssign<u8> +
ops::ShlAssign<u16> +
ops::ShlAssign<u32> +
ops::ShlAssign<u64> +
ops::ShlAssign<usize> +
ops::ShlAssign<i8> +
ops::ShlAssign<i16> +
ops::ShlAssign<i32> +
ops::ShlAssign<i64> +
ops::ShlAssign<isize> +
ops::Shr<Self, Output=Self> +
ops::Shr<u8, Output=Self> +
ops::Shr<u16, Output=Self> +
ops::Shr<u32, Output=Self> +
ops::Shr<u64, Output=Self> +
ops::Shr<usize, Output=Self> +
ops::Shr<i8, Output=Self> +
ops::Shr<i16, Output=Self> +
ops::Shr<i64, Output=Self> +
ops::Shr<isize, Output=Self> +
ops::Shr<i32, Output=Self> +
ops::ShrAssign<Self> +
ops::ShrAssign<u8> +
ops::ShrAssign<u16> +
ops::ShrAssign<u32> +
ops::ShrAssign<u64> +
ops::ShrAssign<usize> +
ops::ShrAssign<i8> +
ops::ShrAssign<i16> +
ops::ShrAssign<i32> +
ops::ShrAssign<i64> +
ops::ShrAssign<isize>
{
const ZERO: Self;
const ONE: Self;
const TWO: Self;
const MAX: Self;
const MIN: Self;
const BITS: usize;
fn max_value() -> Self;
fn min_value() -> Self;
fn count_ones(self) -> u32;
fn count_zeros(self) -> u32;
fn leading_zeros(self) -> u32;
fn trailing_zeros(self) -> u32;
fn pow(self, i: u32) -> Self;
fn checked_add(self, i: Self) -> Option<Self>;
fn checked_sub(self, i: Self) -> Option<Self>;
fn checked_mul(self, i: Self) -> Option<Self>;
fn checked_div(self, i: Self) -> Option<Self>;
fn checked_rem(self, i: Self) -> Option<Self>;
fn checked_neg(self) -> Option<Self>;
fn checked_shl(self, i: u32) -> Option<Self>;
fn checked_shr(self, i: u32) -> Option<Self>;
fn wrapping_add(self, i: Self) -> Self;
fn wrapping_sub(self, i: Self) -> Self;
fn wrapping_mul(self, i: Self) -> Self;
fn wrapping_div(self, i: Self) -> Self;
fn wrapping_rem(self, i: Self) -> Self;
fn wrapping_neg(self) -> Self;
fn wrapping_shl(self, i: u32) -> Self;
fn wrapping_shr(self, i: u32) -> Self;
fn overflowing_add(self, i: Self) -> (Self, bool);
fn overflowing_sub(self, i: Self) -> (Self, bool);
fn overflowing_mul(self, i: Self) -> (Self, bool);
fn overflowing_div(self, i: Self) -> (Self, bool);
fn overflowing_rem(self, i: Self) -> (Self, bool);
fn overflowing_neg(self) -> (Self, bool);
fn overflowing_shl(self, i: u32) -> (Self, bool);
fn overflowing_shr(self, i: u32) -> (Self, bool);
fn saturating_add(self, i: Self) -> Self;
fn saturating_sub(self, i: Self) -> Self;
fn saturating_mul(self, i: Self) -> Self;
#[inline]
fn zero() -> Self {
Self::ZERO
}
#[inline]
fn one() -> Self {
Self::ONE
}
#[inline]
fn two() -> Self {
Self::TWO
}
#[inline]
fn is_zero(self) -> bool {
self == Self::ZERO
}
#[inline]
fn is_one(self) -> bool {
self == Self::ONE
}
#[inline]
fn ceil_divmod(self, y: Self) -> (Self, i32) {
let q = self / y;
let r = self % y;
match r.is_zero() {
true => (q, r.as_i32()),
false => (q + Self::ONE, r.as_i32() - y.as_i32())
}
}
#[inline]
fn ceil_div(self, y: Self) -> Self {
self.ceil_divmod(y).0
}
#[inline]
fn ceil_mod(self, y: Self) -> i32 {
self.ceil_divmod(y).1
}
#[inline]
fn bit_length(self) -> u32 {
Self::BITS as u32 - self.leading_zeros()
}
#[inline]
fn try_u8_or_max(self) -> u8 {
try_cast_or_max(self)
}
#[inline]
fn try_u16_or_max(self) -> u16 {
try_cast_or_max(self)
}
#[inline]
fn try_u32_or_max(self) -> u32 {
try_cast_or_max(self)
}
#[inline]
fn try_u64_or_max(self) -> u64 {
try_cast_or_max(self)
}
#[cfg(has_i128)]
#[inline]
fn try_u128_or_max(self) -> u128 {
try_cast_or_max(self)
}
#[inline]
fn try_usize_or_max(self) -> usize {
try_cast_or_max(self)
}
#[inline]
fn try_i8_or_max(self) -> i8 {
try_cast_or_max(self)
}
#[inline]
fn try_i16_or_max(self) -> i16 {
try_cast_or_max(self)
}
#[inline]
fn try_i32_or_max(self) -> i32 {
try_cast_or_max(self)
}
#[inline]
fn try_i64_or_max(self) -> i64 {
try_cast_or_max(self)
}
#[cfg(has_i128)]
#[inline]
fn try_i128_or_max(self) -> i128 {
try_cast_or_max(self)
}
#[inline]
fn try_isize_or_max(self) -> isize {
try_cast_or_max(self)
}
#[inline]
fn try_u8_or_min(self) -> u8 {
try_cast_or_min(self)
}
#[inline]
fn try_u16_or_min(self) -> u16 {
try_cast_or_min(self)
}
#[inline]
fn try_u32_or_min(self) -> u32 {
try_cast_or_min(self)
}
#[inline]
fn try_u64_or_min(self) -> u64 {
try_cast_or_min(self)
}
#[cfg(has_i128)]
#[inline]
fn try_u128_or_min(self) -> u128 {
try_cast_or_min(self)
}
#[inline]
fn try_usize_or_min(self) -> usize {
try_cast_or_min(self)
}
#[inline]
fn try_i8_or_min(self) -> i8 {
try_cast_or_min(self)
}
#[inline]
fn try_i16_or_min(self) -> i16 {
try_cast_or_min(self)
}
#[inline]
fn try_i32_or_min(self) -> i32 {
try_cast_or_min(self)
}
#[inline]
fn try_i64_or_min(self) -> i64 {
try_cast_or_min(self)
}
#[cfg(has_i128)]
#[inline]
fn try_i128_or_min(self) -> i128 {
try_cast_or_min(self)
}
#[inline]
fn try_isize_or_min(self) -> isize {
try_cast_or_min(self)
}
}
macro_rules! integer_impl {
($($t:tt)*) => ($(
impl Integer for $t {
const ZERO: $t = 0;
const ONE: $t = 1;
const TWO: $t = 2;
const MAX: $t = $t::max_value();
const MIN: $t = $t::min_value();
const BITS: usize = mem::size_of::<$t>() * 8;
#[inline]
fn max_value() -> Self {
Self::MAX
}
#[inline]
fn min_value() -> Self {
Self::MIN
}
#[inline]
fn count_ones(self) -> u32 {
$t::count_ones(self)
}
#[inline]
fn count_zeros(self) -> u32 {
$t::count_zeros(self)
}
#[inline]
fn leading_zeros(self) -> u32 {
$t::leading_zeros(self)
}
#[inline]
fn trailing_zeros(self) -> u32 {
$t::trailing_zeros(self)
}
#[inline]
fn pow(self, i: u32) -> Self {
$t::pow(self, i)
}
#[inline]
fn checked_add(self, i: Self) -> Option<Self> {
$t::checked_add(self, i)
}
#[inline]
fn checked_sub(self, i: Self) -> Option<Self> {
$t::checked_sub(self, i)
}
#[inline]
fn checked_mul(self, i: Self) -> Option<Self> {
$t::checked_mul(self, i)
}
#[inline]
fn checked_div(self, i: Self) -> Option<Self> {
$t::checked_div(self, i)
}
#[inline]
fn checked_rem(self, i: Self) -> Option<Self> {
$t::checked_rem(self, i)
}
#[inline]
fn checked_neg(self) -> Option<Self> {
$t::checked_neg(self)
}
#[inline]
fn checked_shl(self, i: u32) -> Option<Self> {
$t::checked_shl(self,i)
}
#[inline]
fn checked_shr(self, i: u32) -> Option<Self> {
$t::checked_shr(self,i)
}
#[inline]
fn wrapping_add(self, i: Self) -> Self {
$t::wrapping_add(self, i)
}
#[inline]
fn wrapping_sub(self, i: Self) -> Self {
$t::wrapping_sub(self, i)
}
#[inline]
fn wrapping_mul(self, i: Self) -> Self {
$t::wrapping_mul(self, i)
}
#[inline]
fn wrapping_div(self, i: Self) -> Self {
$t::wrapping_div(self, i)
}
#[inline]
fn wrapping_rem(self, i: Self) -> Self {
$t::wrapping_rem(self, i)
}
#[inline]
fn wrapping_neg(self) -> Self {
$t::wrapping_neg(self)
}
#[inline]
fn wrapping_shl(self, i: u32) -> Self {
$t::wrapping_shl(self,i)
}
#[inline]
fn wrapping_shr(self, i: u32) -> Self {
$t::wrapping_shr(self,i)
}
#[inline]
fn overflowing_add(self, i: Self) -> (Self, bool) {
$t::overflowing_add(self, i)
}
#[inline]
fn overflowing_sub(self, i: Self) -> (Self, bool) {
$t::overflowing_sub(self, i)
}
#[inline]
fn overflowing_mul(self, i: Self) -> (Self, bool) {
$t::overflowing_mul(self, i)
}
#[inline]
fn overflowing_div(self, i: Self) -> (Self, bool) {
$t::overflowing_div(self, i)
}
#[inline]
fn overflowing_rem(self, i: Self) -> (Self, bool) {
$t::overflowing_rem(self, i)
}
#[inline]
fn overflowing_neg(self) -> (Self, bool) {
$t::overflowing_neg(self)
}
#[inline]
fn overflowing_shl(self, i: u32) -> (Self, bool) {
$t::overflowing_shl(self,i)
}
#[inline]
fn overflowing_shr(self, i: u32) -> (Self, bool) {
$t::overflowing_shr(self,i)
}
#[inline]
fn saturating_add(self, i: Self) -> Self {
$t::saturating_add(self, i)
}
#[inline]
fn saturating_sub(self, i: Self) -> Self {
$t::saturating_sub(self, i)
}
#[inline]
fn saturating_mul(self, i: Self) -> Self {
$t::saturating_mul(self, i)
}
}
)*)
}
integer_impl! { u8 u16 u32 u64 usize i8 i16 i32 i64 isize }
#[cfg(has_i128)] integer_impl! { u128 i128 }
#[inline]
pub fn unwrap_or_max<T: Integer>(t: Option<T>) -> T {
t.unwrap_or(T::max_value())
}
#[inline]
pub fn unwrap_or_min<T: Integer>(t: Option<T>) -> T {
t.unwrap_or(T::min_value())
}
#[inline]
pub fn try_cast_or_max<U: Integer, T: TryCast<U>>(t: T) -> U {
unwrap_or_max(TryCast::try_cast(t))
}
#[inline]
pub fn try_cast_or_min<U: Integer, T: TryCast<U>>(t: T) -> U {
unwrap_or_min(TryCast::try_cast(t))
}
pub trait SignedInteger: Integer + ops::Neg<Output=Self>
{
fn checked_abs(self) -> Option<Self>;
fn wrapping_abs(self) -> Self;
fn overflowing_abs(self) -> (Self, bool);
}
macro_rules! signed_integer_impl {
($($t:tt)*) => ($(
impl SignedInteger for $t {
fn checked_abs(self) -> Option<Self> {
$t::checked_abs(self)
}
fn wrapping_abs(self) -> Self {
$t::wrapping_abs(self)
}
fn overflowing_abs(self) -> (Self, bool) {
$t::overflowing_abs(self)
}
}
)*)
}
signed_integer_impl! { i8 i16 i32 i64 isize }
#[cfg(has_i128)] signed_integer_impl! { i128 }
pub trait UnsignedInteger: Integer
{
#[inline]
fn is_odd(self) -> bool {
self & Self::ONE == Self::ONE
}
#[inline]
fn is_even(self) -> bool {
!self.is_odd()
}
}
macro_rules! unsigned_integer_impl {
($($t:ty)*) => ($(
impl UnsignedInteger for $t {}
)*)
}
unsigned_integer_impl! { u8 u16 u32 u64 usize }
#[cfg(has_i128)] unsigned_integer_impl! { u128 }
pub trait Float: Number + ops::Neg<Output=Self>
{
type Unsigned: UnsignedInteger;
const ZERO: Self;
const ONE: Self;
const TWO: Self;
const MAX: Self;
const MIN: Self;
const INFINITY: Self;
const NEG_INFINITY: Self;
const NAN: Self;
const BITS: usize;
const SIGN_MASK: Self::Unsigned;
const EXPONENT_MASK: Self::Unsigned;
const HIDDEN_BIT_MASK: Self::Unsigned;
const MANTISSA_MASK: Self::Unsigned;
const INFINITY_BITS: Self::Unsigned;
const NEGATIVE_INFINITY_BITS: Self::Unsigned;
const MANTISSA_SIZE: i32;
const EXPONENT_BIAS: i32;
const DENORMAL_EXPONENT: i32;
const MAX_EXPONENT: i32;
fn abs(self) -> Self;
fn ceil(self) -> Self;
fn exp(self) -> Self;
fn floor(self) -> Self;
fn ln(self) -> Self;
fn powi(self, n: i32) -> Self;
fn powf(self, f: Self) -> Self;
fn round(self) -> Self;
fn to_bits(self) -> Self::Unsigned;
fn from_bits(u: Self::Unsigned) -> Self;
fn is_sign_positive(self) -> bool;
fn is_sign_negative(self) -> bool;
#[inline]
fn is_zero(self) -> bool {
self == Self::ZERO
}
#[inline]
fn is_one(self) -> bool {
self == Self::ONE
}
#[inline]
fn is_denormal(self) -> bool {
self.to_bits() & Self::EXPONENT_MASK == Self::Unsigned::ZERO
}
#[inline]
fn is_special(self) -> bool {
self.to_bits() & Self::EXPONENT_MASK == Self::EXPONENT_MASK
}
#[inline]
fn is_nan(self) -> bool {
self.is_special() && !(self.to_bits() & Self::MANTISSA_MASK).is_zero()
}
#[inline]
fn is_inf(self) -> bool {
self.is_special() && (self.to_bits() & Self::MANTISSA_MASK).is_zero()
}
#[inline]
fn is_odd(self) -> bool {
self.to_bits().is_odd()
}
#[inline]
fn is_even(self) -> bool {
!self.is_odd()
}
#[inline]
fn exponent(self) -> i32 {
if self.is_denormal() {
return Self::DENORMAL_EXPONENT;
}
let bits = self.to_bits();
let biased_e: i32 = AsCast::as_cast((bits & Self::EXPONENT_MASK) >> Self::MANTISSA_SIZE);
biased_e - Self::EXPONENT_BIAS
}
#[inline]
fn mantissa(self) -> Self::Unsigned {
let bits = self.to_bits();
let s = bits & Self::MANTISSA_MASK;
if !self.is_denormal() {
s + Self::HIDDEN_BIT_MASK
} else {
s
}
}
#[inline]
fn next(self) -> Self {
let bits = self.to_bits();
if self.is_sign_negative() && self.is_zero() {
Self::ZERO
} else if bits == Self::INFINITY_BITS {
Self::from_bits(Self::INFINITY_BITS)
} else if self.is_sign_negative() {
Self::from_bits(bits.saturating_sub(Self::Unsigned::ONE))
} else {
Self::from_bits(bits.saturating_add(Self::Unsigned::ONE))
}
}
#[inline]
fn next_positive(self) -> Self {
debug_assert!(self.is_sign_positive() && !self.is_inf());
Self::from_bits(self.to_bits() + Self::Unsigned::ONE)
}
#[inline]
fn prev(self) -> Self {
let bits = self.to_bits();
if self.is_sign_positive() && self.is_zero() {
-Self::ZERO
} else if bits == Self::NEGATIVE_INFINITY_BITS {
Self::from_bits(Self::NEGATIVE_INFINITY_BITS)
} else if self.is_sign_negative() {
Self::from_bits(bits.saturating_add(Self::Unsigned::ONE))
} else {
Self::from_bits(bits.saturating_sub(Self::Unsigned::ONE))
}
}
#[inline]
fn prev_positive(self) -> Self {
debug_assert!(self.is_sign_positive() && !self.is_zero());
return Self::from_bits(self.to_bits() - Self::Unsigned::ONE);
}
#[inline]
fn round_positive_even(self) -> Self {
if self.mantissa().is_odd() {
self.next_positive()
} else {
self
}
}
#[inline]
fn max_finite(self, f: Self) -> Self {
debug_assert!(!self.is_special() && !f.is_special(), "max_finite self={} f={}", self, f);
if self < f { f } else { self }
}
#[inline]
fn min_finite(self, f: Self) -> Self {
debug_assert!(!self.is_special() && !f.is_special(), "min_finite self={} f={}", self, f);
if self < f { self } else { f }
}
}
macro_rules! float_method {
($f:ident, $t:tt, $meth:ident, $intr:ident $(,$i:expr)*) => ({
#[cfg(feature = "std")]
return $t::$meth($f $(,$i)*);
#[cfg(not(feature = "std"))]
return unsafe { core::intrinsics::$intr($f $(,$i)*) };
})
}
macro_rules! float_method_msvc {
($f:ident, $ts:tt, $tl:tt, $meth:ident, $intr:ident $(,$i:expr)*) => ({
#[cfg(feature = "std")]
return $ts::$meth($f $(,$i)*);
#[cfg(all(not(feature = "std"), not(target_env = "msvc")))]
return unsafe { core::intrinsics::$intr($f $(,$i)*) };
#[cfg(all(not(feature = "std"), target_env = "msvc"))]
return ($f as $tl).$meth() as $ts;
})
}
macro_rules! float_method_log_solaris {
($f:ident, $t:tt, $meth:ident, $intr:ident $(,$i:expr)*) => ({
#[cfg(feature = "std")]
return $t::$meth($f $(,$i)*);
#[cfg(all(not(feature = "std"), not(target_os = "solaris")))]
return unsafe { core::intrinsics::$intr($f $(,$i)*) };
#[cfg(all(not(feature = "std"), target_os = "solaris"))] {
if $f.is_nan() {
$f
} else if $f.is_special() {
if $f > $t::ZERO { $f } else { $t::NAN }
} else if $f > $t::ZERO {
unsafe { core::intrinsics::$intr($f $(,$i)*) }
} else if $f.is_zero() {
$t::NEG_INFINITY
} else {
$t::NAN
}
}
})
}
impl Float for f32 {
type Unsigned = u32;
const ZERO: f32 = 0.0;
const ONE: f32 = 1.0;
const TWO: f32 = 2.0;
const MAX: f32 = f32::MAX;
const MIN: f32 = f32::MIN;
const INFINITY: f32 = f32::INFINITY;
const NEG_INFINITY: f32 = f32::NEG_INFINITY;
const NAN: f32 = f32::NAN;
const BITS: usize = 32;
const SIGN_MASK: u32 = 0x80000000;
const EXPONENT_MASK: u32 = 0x7F800000;
const HIDDEN_BIT_MASK: u32 = 0x00800000;
const MANTISSA_MASK: u32 = 0x007FFFFF;
const INFINITY_BITS: u32 = 0x7F800000;
const NEGATIVE_INFINITY_BITS: u32 = Self::INFINITY_BITS | Self::SIGN_MASK;
const MANTISSA_SIZE: i32 = 23;
const EXPONENT_BIAS: i32 = 127 + Self::MANTISSA_SIZE;
const DENORMAL_EXPONENT: i32 = 1 - Self::EXPONENT_BIAS;
const MAX_EXPONENT: i32 = 0xFF - Self::EXPONENT_BIAS;
#[inline]
fn abs(self) -> f32 {
float_method!(self, f32, abs, fabsf32)
}
#[inline]
fn ceil(self) -> f32 {
float_method_msvc!(self, f32, f64, ceil, ceilf32)
}
#[inline]
fn exp(self) -> f32 {
float_method_msvc!(self, f32, f64, exp, expf32)
}
#[inline]
fn floor(self) -> f32 {
float_method_msvc!(self, f32, f64, floor, floorf32)
}
#[inline]
fn ln(self) -> f32 {
float_method_msvc!(self, f32, f64, ln, logf32)
}
#[inline]
fn powi(self, n: i32) -> f32 {
float_method!(self, f32, powi, powif32, n)
}
#[inline]
fn powf(self, n: f32) -> f32 {
float_method_msvc!(self, f32, f64, powf, powf32, n as f32)
}
#[inline]
fn round(self) -> f32 {
float_method!(self, f32, round, roundf32)
}
#[inline]
fn to_bits(self) -> u32 {
f32::to_bits(self)
}
#[inline]
fn from_bits(u: u32) -> f32 {
f32::from_bits(u)
}
#[inline]
fn is_sign_positive(self) -> bool {
f32::is_sign_positive(self)
}
#[inline]
fn is_sign_negative(self) -> bool {
f32::is_sign_negative(self)
}
}
impl Float for f64 {
type Unsigned = u64;
const ZERO: f64 = 0.0;
const ONE: f64 = 1.0;
const TWO: f64 = 2.0;
const MAX: f64 = f64::MAX;
const MIN: f64 = f64::MIN;
const INFINITY: f64 = f64::INFINITY;
const NEG_INFINITY: f64 = f64::NEG_INFINITY;
const NAN: f64 = f64::NAN;
const BITS: usize = 64;
const SIGN_MASK: u64 = 0x8000000000000000;
const EXPONENT_MASK: u64 = 0x7FF0000000000000;
const HIDDEN_BIT_MASK: u64 = 0x0010000000000000;
const MANTISSA_MASK: u64 = 0x000FFFFFFFFFFFFF;
const INFINITY_BITS: u64 = 0x7FF0000000000000;
const NEGATIVE_INFINITY_BITS: u64 = Self::INFINITY_BITS | Self::SIGN_MASK;
const MANTISSA_SIZE: i32 = 52;
const EXPONENT_BIAS: i32 = 1023 + Self::MANTISSA_SIZE;
const DENORMAL_EXPONENT: i32 = 1 - Self::EXPONENT_BIAS;
const MAX_EXPONENT: i32 = 0x7FF - Self::EXPONENT_BIAS;
#[inline]
fn abs(self) -> f64 {
float_method!(self, f64, abs, fabsf64)
}
#[inline]
fn ceil(self) -> f64 {
float_method!(self, f64, ceil, ceilf64)
}
#[inline]
fn exp(self) -> f64 {
float_method!(self, f64, exp, expf64)
}
#[inline]
fn floor(self) -> f64 {
float_method!(self, f64, floor, floorf64)
}
#[inline]
fn ln(self) -> f64 {
float_method_log_solaris!(self, f64, ln, logf64)
}
#[inline]
fn powi(self, n: i32) -> f64 {
float_method!(self, f64, powi, powif64, n)
}
#[inline]
fn powf(self, n: f64) -> f64 {
float_method!(self, f64, powf, powf64, n)
}
#[inline]
fn round(self) -> f64 {
float_method!(self, f64, round, roundf64)
}
#[inline]
fn to_bits(self) -> u64 {
f64::to_bits(self)
}
#[inline]
fn from_bits(u: u64) -> f64 {
f64::from_bits(u)
}
#[inline]
fn is_sign_positive(self) -> bool {
f64::is_sign_positive(self)
}
#[inline]
fn is_sign_negative(self) -> bool {
f64::is_sign_negative(self)
}
}
#[cfg(test)]
mod tests {
use super::*;
fn check_number<T: Number>(x: T, mut y: T) {
let _ = x;
assert!(x < y);
assert!(x != y);
let _ = y + x;
let _ = y - x;
let _ = y * x;
let _ = y / x;
let _ = y % x;
y += x;
y -= x;
y *= x;
y /= x;
y %= x;
}
#[test]
fn number_test() {
check_number(1u8, 5);
check_number(1u16, 5);
check_number(1u32, 5);
check_number(1u64, 5);
check_number(1u128, 5);
check_number(1usize, 5);
check_number(1i8, 5);
check_number(1i16, 5);
check_number(1i32, 5);
check_number(1i64, 5);
check_number(1i128, 5);
check_number(1isize, 5);
check_number(1f32, 5.0);
check_number(1f64, 5.0);
}
fn check_integer<T: Integer>(mut x: T) {
let _ = x;
assert!(x > T::ONE);
assert!(x != T::ONE);
assert_eq!(x.min(T::ONE), T::ONE);
assert_eq!(x.max(T::ONE), x);
let _ = x + T::ONE;
let _ = x - T::ONE;
let _ = x * T::ONE;
let _ = x / T::ONE;
let _ = x % T::ONE;
x += T::ONE;
x -= T::ONE;
x *= T::ONE;
x /= T::ONE;
x %= T::ONE;
let _ = x & T::ONE;
let _ = x | T::ONE;
let _ = x ^ T::ONE;
let _ = !x;
x &= T::ONE;
x |= T::ONE;
x ^= T::ONE;
let _ = x << 1u8;
let _ = x << 1u16;
let _ = x << 1u32;
let _ = x << 1u64;
let _ = x << 1usize;
let _ = x << 1i8;
let _ = x << 1i16;
let _ = x << 1i32;
let _ = x << 1i64;
let _ = x << 1isize;
let _ = x >> 1u8;
let _ = x >> 1u16;
let _ = x >> 1u32;
let _ = x >> 1u64;
let _ = x >> 1usize;
let _ = x >> 1i8;
let _ = x >> 1i16;
let _ = x >> 1i32;
let _ = x >> 1i64;
let _ = x >> 1isize;
x <<= 1u8;
x <<= 1u16;
x <<= 1u32;
x <<= 1u64;
x <<= 1usize;
x <<= 1i8;
x <<= 1i16;
x <<= 1i32;
x <<= 1i64;
x <<= 1isize;
x >>= 1u8;
x >>= 1u16;
x >>= 1u32;
x >>= 1u64;
x >>= 1usize;
x >>= 1i8;
x >>= 1i16;
x >>= 1i32;
x >>= 1i64;
x >>= 1isize;
assert!(T::ZERO.is_zero());
assert!(!T::ONE.is_zero());
assert!(T::ONE.is_one());
assert!(!T::ZERO.is_one());
}
#[test]
fn integer_test() {
check_integer(65u8);
check_integer(65u16);
check_integer(65u32);
check_integer(65u64);
check_integer(65u128);
check_integer(65usize);
check_integer(65i8);
check_integer(65i16);
check_integer(65i32);
check_integer(65i64);
check_integer(65i128);
check_integer(65isize);
}
#[test]
fn ceil_divmod_test() {
assert_eq!(5usize.ceil_divmod(7), (1, -2));
assert_eq!(0usize.ceil_divmod(7), (0, 0));
assert_eq!(35usize.ceil_divmod(7), (5, 0));
assert_eq!(36usize.ceil_divmod(7), (6, -6));
}
#[test]
fn unwrap_or_max_test() {
let x: Option<u8> = None;
assert_eq!(unwrap_or_max(x), u8::max_value());
let x: Option<u8> = Some(1);
assert_eq!(unwrap_or_max(x), 1);
}
#[test]
fn unwrap_or_min_test() {
let x: Option<u8> = None;
assert_eq!(unwrap_or_min(x), u8::min_value());
let x: Option<u8> = Some(1);
assert_eq!(unwrap_or_min(x), 1);
}
#[test]
fn try_cast_or_max_test() {
let x: u8 = try_cast_or_max(u16::min_value());
assert_eq!(x, u8::min_value());
let x: u8 = try_cast_or_max(u8::max_value() as u16);
assert_eq!(x, u8::max_value());
let x: u8 = try_cast_or_max(u16::max_value());
assert_eq!(x, u8::max_value());
}
#[test]
fn try_cast_or_min_test() {
let x: u8 = try_cast_or_min(u16::min_value());
assert_eq!(x, u8::min_value());
let x: u8 = try_cast_or_min(u8::max_value() as u16);
assert_eq!(x, u8::max_value());
let x: u8 = try_cast_or_min(u16::max_value());
assert_eq!(x, u8::min_value());
}
fn check_float<T: Float>(mut x: T) {
let _ = x;
assert!(x > T::ONE);
assert!(x != T::ONE);
let _ = x + T::ONE;
let _ = x - T::ONE;
let _ = x * T::ONE;
let _ = x / T::ONE;
let _ = x % T::ONE;
let _ = -x;
x += T::ONE;
x -= T::ONE;
x *= T::ONE;
x /= T::ONE;
x %= T::ONE;
let _ = x.abs();
let _ = x.ceil();
let _ = x.exp();
let _ = x.floor();
let _ = x.ln();
let _ = x.powi(5);
let _ = x.powf(T::ONE);
let _ = x.round();
let _ = x.to_bits();
assert_eq!(T::from_bits(x.to_bits()), x);
let _ = x.is_sign_positive();
let _ = x.is_sign_negative();
let _ = x.to_bits() & T::SIGN_MASK;
let _ = x.to_bits() & T::EXPONENT_MASK;
let _ = x.to_bits() & T::HIDDEN_BIT_MASK;
let _ = x.to_bits() & T::MANTISSA_MASK;
assert!(T::from_bits(T::INFINITY_BITS).is_special());
}
#[test]
fn float_test() {
check_float(123f32);
check_float(123f64);
let f: f32 = 1e-45;
assert!(f.is_odd());
assert!(f.next().is_even());
assert!(f.next_positive().is_even());
assert!(f.prev().is_even());
assert!(f.prev_positive().is_even());
assert!(f.round_positive_even().is_even());
assert_eq!(f.prev().next(), f);
assert_eq!(f.prev_positive().next_positive(), f);
assert_eq!(f.round_positive_even(), f.next());
let f: f32 = 0.1;
assert!(f.is_odd());
assert!(f.next().is_even());
assert!(f.next_positive().is_even());
assert!(f.prev().is_even());
assert!(f.prev_positive().is_even());
assert!(f.round_positive_even().is_even());
assert_eq!(f.prev().next(), f);
assert_eq!(f.prev_positive().next_positive(), f);
assert_eq!(f.round_positive_even(), f.next());
let f: f32 = 1.0;
assert!(f.is_even());
assert!(f.next().is_odd());
assert!(f.next_positive().is_odd());
assert!(f.prev().is_odd());
assert!(f.prev_positive().is_odd());
assert!(f.round_positive_even().is_even());
assert_eq!(f.prev().next(), f);
assert_eq!(f.prev_positive().next_positive(), f);
assert_ne!(f.round_positive_even(), f.next());
}
}