diff --git a/utilities/Cargo.toml b/utilities/Cargo.toml index 59772f20b631ccb8dad528f713861b5badf3334d..72ab82977dd4553305fc187894c496b7886080ec 100644 --- a/utilities/Cargo.toml +++ b/utilities/Cargo.toml @@ -7,6 +7,7 @@ edition = "2018" [dependencies] serde = { version = "1.0", optional = true } codec = { package = "parity-scale-codec", version = "1.0.0", default-features = false } +primitives = { package = "substrate-primitives", git = "https://github.com/paritytech/substrate.git", default-features = false } sr-primitives = { git = "https://github.com/paritytech/substrate.git", default-features = false } runtime-io = { package = "sr-io", git = "https://github.com/paritytech/substrate.git", default-features = false } rstd = { package = "sr-std", git = "https://github.com/paritytech/substrate.git", default-features = false } @@ -15,7 +16,6 @@ frame-system = { git = "https://github.com/paritytech/substrate.git", default-fe frame-support = { git = "https://github.com/paritytech/substrate.git", default-features = false } [dev-dependencies] -primitives = { package = "substrate-primitives", git = "https://github.com/paritytech/substrate.git", default-features = false } clear_on_drop = { version = "0.2.3", features = ["no_cc"] } # https://github.com/paritytech/substrate/issues/4179 diff --git a/utilities/src/fixed128.rs b/utilities/src/fixed128.rs new file mode 100644 index 0000000000000000000000000000000000000000..0085c769cb36b310bc9f83b5f037785cd606edb4 --- /dev/null +++ b/utilities/src/fixed128.rs @@ -0,0 +1,256 @@ +use codec::{Decode, Encode}; +use primitives::U256; +use rstd::convert::{Into, TryFrom, TryInto}; +use sr_primitives::traits::{Bounded, Saturating}; + +/// An unsigned fixed point number. Can hold any value in the range [0, 340_282_366_920_938_463_464] +/// with fixed point accuracy of 10 ** 18 +#[derive(Encode, Decode, Default, Copy, Clone, PartialEq, Eq, PartialOrd, Ord)] +pub struct FixedU128(u128); + +const DIV: u128 = 1_000_000_000_000_000_000; + +impl FixedU128 { + /// create self from a natural number + /// + /// Note that this might be lossy + pub fn from_natural(int: u128) -> Self { + Self(int.saturating_mul(DIV)) + } + + pub fn accuracy() -> u128 { + DIV + } + + /// raw constructor. Equal to `parts / DIV`. + pub fn from_parts(parts: u128) -> Self { + Self(parts) + } + + /// creates self from a rational number. Equal to `n/d` + /// + /// Note that this might be lossy + pub fn from_rational(n: u128, d: u128) -> Self { + Self( + (U256::from(n).saturating_mul(U256::from(DIV)) / U256::from(d).max(U256::one())) + .try_into() + .unwrap_or_else(|_| Bounded::max_value()), + ) + } + + /// consume self and return the inner value. + /// + /// This should only be used for testing. + #[cfg(any(feature = "std", test))] + pub fn into_inner(self) -> u128 { + self.0 + } + + // checked add for FixedU128 + pub fn checked_add(&self, rhs: &Self) -> Option<Self> { + self.0.checked_add(rhs.0).map(Self) + } + + // checked sub for FixedU128 + pub fn checked_sub(&self, rhs: &Self) -> Option<Self> { + self.0.checked_sub(rhs.0).map(Self) + } + + // checked mul for FixedU128 + pub fn checked_mul(&self, rhs: &Self) -> Option<Self> { + if let Some(r) = U256::from(self.0) + .checked_mul(U256::from(rhs.0)) + .and_then(|n| n.checked_div(U256::from(DIV))) + { + if let Ok(r) = TryInto::<u128>::try_into(r) { + return Some(Self(r)); + } + } + + None + } + + // checked div for FixedU128 + pub fn checked_div(&self, rhs: &Self) -> Option<Self> { + if let Some(r) = U256::from(self.0) + .checked_mul(U256::from(DIV)) + .and_then(|n| n.checked_div(U256::from(rhs.0))) + { + if let Ok(r) = TryInto::<u128>::try_into(r) { + return Some(Self(r)); + } + } + + None + } + + /// checked mul for type N + pub fn checked_mul_int<N>(&self, other: &N) -> Option<N> + where + N: Copy + TryFrom<u128> + TryInto<u128>, + { + if let Ok(n) = N::try_into(*other) { + if let Some(n) = U256::from(self.0) + .checked_mul(U256::from(n)) + .and_then(|n| n.checked_div(U256::from(DIV))) + { + if let Ok(r) = TryInto::<u128>::try_into(n) { + if let Ok(r) = TryInto::<N>::try_into(r) { + return Some(r); + } + } + } + } + + None + } + + /// checked div for type N + pub fn checked_div_int<N>(&self, other: &N) -> Option<N> + where + N: Copy + TryFrom<u128> + TryInto<u128>, + { + if let Ok(n) = N::try_into(*other) { + if let Some(n) = self.0.checked_div(n).and_then(|n| n.checked_div(DIV)) { + if let Ok(r) = TryInto::<N>::try_into(n) { + return Some(r); + } + } + } + + None + } +} + +impl Saturating for FixedU128 { + fn saturating_add(self, rhs: Self) -> Self { + Self(self.0.saturating_add(rhs.0)) + } + + fn saturating_mul(self, rhs: Self) -> Self { + Self( + (U256::from(self.0).saturating_mul(U256::from(rhs.0)) / U256::from(DIV)) + .try_into() + .unwrap_or_else(|_| Bounded::max_value()), + ) + } + + fn saturating_sub(self, rhs: Self) -> Self { + Self(self.0.saturating_sub(rhs.0)) + } +} + +impl Bounded for FixedU128 { + fn max_value() -> Self { + Self(u128::max_value()) + } + + fn min_value() -> Self { + Self(0u128) + } +} + +impl rstd::fmt::Debug for FixedU128 { + #[cfg(feature = "std")] + fn fmt(&self, f: &mut rstd::fmt::Formatter) -> rstd::fmt::Result { + write!(f, "FixedU128({},{})", self.0 / DIV, (self.0 % DIV) / 1000) + } + + #[cfg(not(feature = "std"))] + fn fmt(&self, _: &mut rstd::fmt::Formatter) -> rstd::fmt::Result { + Ok(()) + } +} + +#[cfg(test)] +mod tests { + use super::*; + + fn max() -> FixedU128 { + FixedU128::from_parts(u128::max_value()) + } + + #[test] + fn fixed128_semantics() { + assert_eq!(FixedU128::from_rational(5, 2).0, 5 * 1_000_000_000_000_000_000 / 2); + assert_eq!(FixedU128::from_rational(5, 2), FixedU128::from_rational(10, 4)); + assert_eq!(FixedU128::from_rational(5, 0), FixedU128::from_rational(5, 1)); + + // biggest value that can be created. + assert_ne!(max(), FixedU128::from_natural(340_282_366_920_938_463_463)); + assert_eq!(max(), FixedU128::from_natural(340_282_366_920_938_463_464)); + } + + #[test] + fn fixed128_operation() { + let a = FixedU128::from_natural(2); + let b = FixedU128::from_natural(1); + assert_eq!(a.checked_add(&b), Some(FixedU128::from_natural(1 + 2))); + assert_eq!(a.checked_sub(&b), Some(FixedU128::from_natural(2 - 1))); + assert_eq!(a.checked_mul(&b), Some(FixedU128::from_natural(1 * 2))); + assert_eq!(a.checked_div(&b), Some(FixedU128::from_rational(2, 1))); + + let a = FixedU128::from_rational(5, 2); + let b = FixedU128::from_rational(3, 2); + assert_eq!(a.checked_add(&b), Some(FixedU128::from_rational(8, 2))); + assert_eq!(a.checked_sub(&b), Some(FixedU128::from_rational(2, 2))); + assert_eq!(a.checked_mul(&b), Some(FixedU128::from_rational(15, 4))); + assert_eq!(a.checked_div(&b), Some(FixedU128::from_rational(10, 6))); + + let a = FixedU128::from_natural(120); + let b = 2i32; + assert_eq!(a.checked_div_int::<i32>(&b), Some(60)); + + let a = FixedU128::from_rational(20, 1); + let b = 2i32; + assert_eq!(a.checked_div_int::<i32>(&b), Some(10)); + + let a = FixedU128::from_natural(120); + let b = 2i32; + assert_eq!(a.checked_mul_int::<i32>(&b), Some(240)); + + let a = FixedU128::from_rational(1, 2); + let b = 20i32; + assert_eq!(a.checked_mul_int::<i32>(&b), Some(10)); + } + + #[test] + fn checked_div_with_zero_should_be_none() { + let a = FixedU128::from_natural(1); + let b = FixedU128::from_natural(0); + assert_eq!(a.checked_div(&b), None); + } + + #[test] + fn checked_div_int_with_zero_should_be_none() { + let a = FixedU128::from_natural(1); + let b = 0i32; + assert_eq!(a.checked_div_int(&b), None); + } + + #[test] + fn under_flow_should_be_none() { + let a = FixedU128::from_natural(2); + let b = FixedU128::from_natural(3); + assert_eq!(a.checked_sub(&b), None); + } + + #[test] + fn over_flow_should_be_none() { + let a = FixedU128::from_parts(u128::max_value() - 1); + let b = FixedU128::from_parts(2); + assert_eq!(a.checked_add(&b), None); + + let a = FixedU128::max_value(); + let b = FixedU128::from_rational(2, 1); + assert_eq!(a.checked_mul(&b), None); + + let a = FixedU128::from_natural(255); + let b = 2u8; + assert_eq!(a.checked_mul_int(&b), None); + + let a = FixedU128::from_natural(256); + let b = 1u8; + assert_eq!(a.checked_div_int(&b), None); + } +} diff --git a/utilities/src/lib.rs b/utilities/src/lib.rs index 6b864cc0748b4ff660f614de5dfc7b822e8af9c4..d158e8b59ef60f2561488a81355ecc8f48917fbf 100644 --- a/utilities/src/lib.rs +++ b/utilities/src/lib.rs @@ -1,5 +1,7 @@ #![cfg_attr(not(feature = "std"), no_std)] +pub mod fixed128; pub mod linked_item; +pub use fixed128::FixedU128; pub use linked_item::{LinkedItem, LinkedList};