// Copyright 2013-2014 The Rust Project Developers. See the COPYRIGHT // file at the top-level directory of this distribution and at // http://rust-lang.org/COPYRIGHT. // // Licensed under the Apache License, Version 2.0 or the MIT license // , at your // option. This file may not be copied, modified, or distributed // except according to those terms. //! Numeric traits for generic mathematics #![doc(html_logo_url = "https://rust-num.github.io/num/rust-logo-128x128-blk-v2.png", html_favicon_url = "https://rust-num.github.io/num/favicon.ico", html_root_url = "https://rust-num.github.io/num/", html_playground_url = "http://play.integer32.com/")] use std::ops::{Add, Sub, Mul, Div, Rem}; use std::num::Wrapping; pub use bounds::Bounded; pub use float::{Float, FloatConst}; pub use identities::{Zero, One, zero, one}; pub use ops::checked::*; pub use ops::wrapping::*; pub use ops::saturating::Saturating; pub use sign::{Signed, Unsigned, abs, abs_sub, signum}; pub use cast::*; pub use int::PrimInt; pub use pow::{pow, checked_pow}; pub mod identities; pub mod sign; pub mod ops; pub mod bounds; pub mod float; pub mod cast; pub mod int; pub mod pow; /// The base trait for numeric types pub trait Num: PartialEq + Zero + One + Add + Sub + Mul + Div + Rem { type FromStrRadixErr; /// Convert from a string and radix <= 36. /// /// # Examples /// /// ```rust /// use num_traits::Num; /// /// let result = ::from_str_radix("27", 10); /// assert_eq!(result, Ok(27)); /// /// let result = ::from_str_radix("foo", 10); /// assert!(result.is_err()); /// ``` fn from_str_radix(str: &str, radix: u32) -> Result; } macro_rules! int_trait_impl { ($name:ident for $($t:ty)*) => ($( impl $name for $t { type FromStrRadixErr = ::std::num::ParseIntError; #[inline] fn from_str_radix(s: &str, radix: u32) -> Result { <$t>::from_str_radix(s, radix) } } )*) } int_trait_impl!(Num for usize u8 u16 u32 u64 isize i8 i16 i32 i64); impl Num for Wrapping where Wrapping: Add> + Sub> + Mul> + Div> + Rem> { type FromStrRadixErr = T::FromStrRadixErr; fn from_str_radix(str: &str, radix: u32) -> Result { T::from_str_radix(str, radix).map(Wrapping) } } #[derive(Debug)] pub enum FloatErrorKind { Empty, Invalid, } // FIXME: std::num::ParseFloatError is stable in 1.0, but opaque to us, // so there's not really any way for us to reuse it. #[derive(Debug)] pub struct ParseFloatError { pub kind: FloatErrorKind, } // FIXME: The standard library from_str_radix on floats was deprecated, so we're stuck // with this implementation ourselves until we want to make a breaking change. // (would have to drop it from `Num` though) macro_rules! float_trait_impl { ($name:ident for $($t:ty)*) => ($( impl $name for $t { type FromStrRadixErr = ParseFloatError; fn from_str_radix(src: &str, radix: u32) -> Result { use self::FloatErrorKind::*; use self::ParseFloatError as PFE; // Special values match src { "inf" => return Ok(Float::infinity()), "-inf" => return Ok(Float::neg_infinity()), "NaN" => return Ok(Float::nan()), _ => {}, } fn slice_shift_char(src: &str) -> Option<(char, &str)> { src.chars().nth(0).map(|ch| (ch, &src[1..])) } let (is_positive, src) = match slice_shift_char(src) { None => return Err(PFE { kind: Empty }), Some(('-', "")) => return Err(PFE { kind: Empty }), Some(('-', src)) => (false, src), Some((_, _)) => (true, src), }; // The significand to accumulate let mut sig = if is_positive { 0.0 } else { -0.0 }; // Necessary to detect overflow let mut prev_sig = sig; let mut cs = src.chars().enumerate(); // Exponent prefix and exponent index offset let mut exp_info = None::<(char, usize)>; // Parse the integer part of the significand for (i, c) in cs.by_ref() { match c.to_digit(radix) { Some(digit) => { // shift significand one digit left sig = sig * (radix as $t); // add/subtract current digit depending on sign if is_positive { sig = sig + ((digit as isize) as $t); } else { sig = sig - ((digit as isize) as $t); } // Detect overflow by comparing to last value, except // if we've not seen any non-zero digits. if prev_sig != 0.0 { if is_positive && sig <= prev_sig { return Ok(Float::infinity()); } if !is_positive && sig >= prev_sig { return Ok(Float::neg_infinity()); } // Detect overflow by reversing the shift-and-add process if is_positive && (prev_sig != (sig - digit as $t) / radix as $t) { return Ok(Float::infinity()); } if !is_positive && (prev_sig != (sig + digit as $t) / radix as $t) { return Ok(Float::neg_infinity()); } } prev_sig = sig; }, None => match c { 'e' | 'E' | 'p' | 'P' => { exp_info = Some((c, i + 1)); break; // start of exponent }, '.' => { break; // start of fractional part }, _ => { return Err(PFE { kind: Invalid }); }, }, } } // If we are not yet at the exponent parse the fractional // part of the significand if exp_info.is_none() { let mut power = 1.0; for (i, c) in cs.by_ref() { match c.to_digit(radix) { Some(digit) => { // Decrease power one order of magnitude power = power / (radix as $t); // add/subtract current digit depending on sign sig = if is_positive { sig + (digit as $t) * power } else { sig - (digit as $t) * power }; // Detect overflow by comparing to last value if is_positive && sig < prev_sig { return Ok(Float::infinity()); } if !is_positive && sig > prev_sig { return Ok(Float::neg_infinity()); } prev_sig = sig; }, None => match c { 'e' | 'E' | 'p' | 'P' => { exp_info = Some((c, i + 1)); break; // start of exponent }, _ => { return Err(PFE { kind: Invalid }); }, }, } } } // Parse and calculate the exponent let exp = match exp_info { Some((c, offset)) => { let base = match c { 'E' | 'e' if radix == 10 => 10.0, 'P' | 'p' if radix == 16 => 2.0, _ => return Err(PFE { kind: Invalid }), }; // Parse the exponent as decimal integer let src = &src[offset..]; let (is_positive, exp) = match slice_shift_char(src) { Some(('-', src)) => (false, src.parse::()), Some(('+', src)) => (true, src.parse::()), Some((_, _)) => (true, src.parse::()), None => return Err(PFE { kind: Invalid }), }; match (is_positive, exp) { (true, Ok(exp)) => base.powi(exp as i32), (false, Ok(exp)) => 1.0 / base.powi(exp as i32), (_, Err(_)) => return Err(PFE { kind: Invalid }), } }, None => 1.0, // no exponent }; Ok(sig * exp) } } )*) } float_trait_impl!(Num for f32 f64); /// A value bounded by a minimum and a maximum /// /// If input is less than min then this returns min. /// If input is greater than max then this returns max. /// Otherwise this returns input. #[inline] pub fn clamp(input: T, min: T, max: T) -> T { debug_assert!(min <= max, "min must be less than or equal to max"); if input < min { min } else if input > max { max } else { input } } #[test] fn clamp_test() { // Int test assert_eq!(1, clamp(1, -1, 2)); assert_eq!(-1, clamp(-2, -1, 2)); assert_eq!(2, clamp(3, -1, 2)); // Float test assert_eq!(1.0, clamp(1.0, -1.0, 2.0)); assert_eq!(-1.0, clamp(-2.0, -1.0, 2.0)); assert_eq!(2.0, clamp(3.0, -1.0, 2.0)); } #[test] fn from_str_radix_unwrap() { // The Result error must impl Debug to allow unwrap() let i: i32 = Num::from_str_radix("0", 10).unwrap(); assert_eq!(i, 0); let f: f32 = Num::from_str_radix("0.0", 10).unwrap(); assert_eq!(f, 0.0); } macro_rules! test_wrapping_from_str_radix { ($($t:ty)+) => { $( for &(s, r) in &[("42", 10), ("42", 2), ("-13.0", 10), ("foo", 10)] { let w = Wrapping::<$t>::from_str_radix(s, r).map(|w| w.0); assert_eq!(w, <$t as Num>::from_str_radix(s, r)); } )+ }; } #[test] fn wrapping_from_str_radix() { test_wrapping_from_str_radix!(usize u8 u16 u32 u64 isize i8 i16 i32 i64); }