mirror of
https://github.com/pezkuwichain/pezkuwi-subxt.git
synced 2026-04-26 09:57:56 +00:00
Use sensible maths for from_rational (#13660)
* Use sensible maths for from_rational * Fixes * Fixes * More fixes * Remove debugging * Add fuzzer tests Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> * Prevent panics Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> * docs Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> * Clean up old code Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> * Test all rounding modes of from_rational Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> * Clean up code Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> * Revert "Prevent panics" This reverts commit 7e88ac76138a1b590e68b68318505b69efe1e1f6. * fix imports Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> * cleanup Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> * Fuzz test multiply_rational Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> * Fix import Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> * fmt Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> * Return None in multiply_rational on zero div Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> --------- Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> Co-authored-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io>
This commit is contained in:
@@ -182,8 +182,8 @@ mod double128 {
|
||||
}
|
||||
}
|
||||
|
||||
/// Returns `a * b / c` and `(a * b) % c` (wrapping to 128 bits) or `None` in the case of
|
||||
/// overflow and c = 0.
|
||||
/// Returns `a * b / c` (wrapping to 128 bits) or `None` in the case of
|
||||
/// overflow.
|
||||
pub const fn multiply_by_rational_with_rounding(
|
||||
a: u128,
|
||||
b: u128,
|
||||
|
||||
@@ -45,7 +45,7 @@ pub use per_things::{
|
||||
InnerOf, MultiplyArg, PerThing, PerU16, Perbill, Percent, Permill, Perquintill, RationalArg,
|
||||
ReciprocalArg, Rounding, SignedRounding, UpperOf,
|
||||
};
|
||||
pub use rational::{Rational128, RationalInfinite};
|
||||
pub use rational::{MultiplyRational, Rational128, RationalInfinite};
|
||||
|
||||
use sp_std::{cmp::Ordering, fmt::Debug, prelude::*};
|
||||
use traits::{BaseArithmetic, One, SaturatedConversion, Unsigned, Zero};
|
||||
|
||||
@@ -26,7 +26,7 @@ use codec::{CompactAs, Encode};
|
||||
use num_traits::{Pow, SaturatingAdd, SaturatingSub};
|
||||
use sp_std::{
|
||||
fmt, ops,
|
||||
ops::{Add, AddAssign, Div, Rem, Sub},
|
||||
ops::{Add, Sub},
|
||||
prelude::*,
|
||||
};
|
||||
|
||||
@@ -46,6 +46,7 @@ pub trait RationalArg:
|
||||
+ Unsigned
|
||||
+ Zero
|
||||
+ One
|
||||
+ crate::MultiplyRational
|
||||
{
|
||||
}
|
||||
|
||||
@@ -58,7 +59,8 @@ impl<
|
||||
+ ops::AddAssign<Self>
|
||||
+ Unsigned
|
||||
+ Zero
|
||||
+ One,
|
||||
+ One
|
||||
+ crate::MultiplyRational,
|
||||
> RationalArg for T
|
||||
{
|
||||
}
|
||||
@@ -105,7 +107,7 @@ pub trait PerThing:
|
||||
+ Pow<usize, Output = Self>
|
||||
{
|
||||
/// The data type used to build this per-thingy.
|
||||
type Inner: BaseArithmetic + Unsigned + Copy + Into<u128> + fmt::Debug;
|
||||
type Inner: BaseArithmetic + Unsigned + Copy + Into<u128> + fmt::Debug + crate::MultiplyRational;
|
||||
|
||||
/// A data type larger than `Self::Inner`, used to avoid overflow in some computations.
|
||||
/// It must be able to compute `ACCURACY^2`.
|
||||
@@ -115,7 +117,8 @@ pub trait PerThing:
|
||||
+ TryInto<Self::Inner>
|
||||
+ UniqueSaturatedInto<Self::Inner>
|
||||
+ Unsigned
|
||||
+ fmt::Debug;
|
||||
+ fmt::Debug
|
||||
+ crate::MultiplyRational;
|
||||
|
||||
/// The accuracy of this type.
|
||||
const ACCURACY: Self::Inner;
|
||||
@@ -538,39 +541,6 @@ where
|
||||
rem_mul_div_inner.into()
|
||||
}
|
||||
|
||||
/// Just a simple generic integer divide with custom rounding.
|
||||
fn div_rounded<N>(n: N, d: N, r: Rounding) -> N
|
||||
where
|
||||
N: Clone
|
||||
+ Eq
|
||||
+ Ord
|
||||
+ Zero
|
||||
+ One
|
||||
+ AddAssign
|
||||
+ Add<Output = N>
|
||||
+ Rem<Output = N>
|
||||
+ Div<Output = N>,
|
||||
{
|
||||
let mut o = n.clone() / d.clone();
|
||||
use Rounding::*;
|
||||
let two = || N::one() + N::one();
|
||||
if match r {
|
||||
Up => !((n % d).is_zero()),
|
||||
NearestPrefDown => {
|
||||
let rem = n % d.clone();
|
||||
rem > d / two()
|
||||
},
|
||||
NearestPrefUp => {
|
||||
let rem = n % d.clone();
|
||||
rem >= d.clone() / two() + d % two()
|
||||
},
|
||||
Down => false,
|
||||
} {
|
||||
o += N::one()
|
||||
}
|
||||
o
|
||||
}
|
||||
|
||||
macro_rules! implement_per_thing {
|
||||
(
|
||||
$name:ident,
|
||||
@@ -687,7 +657,8 @@ macro_rules! implement_per_thing {
|
||||
+ ops::AddAssign<N>
|
||||
+ Unsigned
|
||||
+ Zero
|
||||
+ One,
|
||||
+ One
|
||||
+ $crate::MultiplyRational,
|
||||
Self::Inner: Into<N>
|
||||
{
|
||||
// q cannot be zero.
|
||||
@@ -695,30 +666,8 @@ macro_rules! implement_per_thing {
|
||||
// p should not be bigger than q.
|
||||
if p > q { return Err(()) }
|
||||
|
||||
let factor = div_rounded::<N>(q.clone(), $max.into(), Rounding::Up).max(One::one());
|
||||
|
||||
// q cannot overflow: (q / (q/$max)) < $max. p < q hence p also cannot overflow.
|
||||
let q_reduce: $type = div_rounded(q, factor.clone(), r)
|
||||
.try_into()
|
||||
.map_err(|_| "Failed to convert")
|
||||
.expect(
|
||||
"`q / ceil(q/$max) < $max`; macro prevents any type being created that \
|
||||
does not satisfy this; qed"
|
||||
);
|
||||
let p_reduce: $type = div_rounded(p, factor, r)
|
||||
.try_into()
|
||||
.map_err(|_| "Failed to convert")
|
||||
.expect(
|
||||
"`p / ceil(p/$max) < $max`; macro prevents any type being created that \
|
||||
does not satisfy this; qed"
|
||||
);
|
||||
|
||||
// `p_reduced` and `q_reduced` are within `Self::Inner`. Multiplication by another
|
||||
// `$max` will always fit in `$upper_type`. This is guaranteed by the macro tests.
|
||||
let n = p_reduce as $upper_type * <$upper_type>::from($max);
|
||||
let d = q_reduce as $upper_type;
|
||||
let part = div_rounded(n, d, r);
|
||||
Ok($name(part as Self::Inner))
|
||||
let max: N = $max.into();
|
||||
max.multiply_rational(p, q, r).ok_or(())?.try_into().map(|x| $name(x)).map_err(|_| ())
|
||||
}
|
||||
}
|
||||
|
||||
@@ -1862,6 +1811,22 @@ fn from_rational_with_rounding_works_in_extreme_case() {
|
||||
}
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn from_rational_with_rounding_breakage() {
|
||||
let n = 372633774963620730670986667244911905u128;
|
||||
let d = 512593663333074177468745541591173060u128;
|
||||
let q = Perquintill::from_rational_with_rounding(n, d, Rounding::Down).unwrap();
|
||||
assert!(q * d <= n);
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn from_rational_with_rounding_breakage_2() {
|
||||
let n = 36893488147419103230u128;
|
||||
let d = 36893488147419103630u128;
|
||||
let q = Perquintill::from_rational_with_rounding(n, d, Rounding::Up).unwrap();
|
||||
assert!(q * d >= n);
|
||||
}
|
||||
|
||||
implement_per_thing!(Percent, test_per_cent, [u32, u64, u128], 100u8, u8, u16, "_Percent_",);
|
||||
implement_per_thing_with_perthousand!(
|
||||
PerU16,
|
||||
|
||||
@@ -284,6 +284,54 @@ impl PartialEq for Rational128 {
|
||||
}
|
||||
}
|
||||
|
||||
pub trait MultiplyRational: Sized {
|
||||
fn multiply_rational(self, n: Self, d: Self, r: Rounding) -> Option<Self>;
|
||||
}
|
||||
|
||||
macro_rules! impl_rrm {
|
||||
($ulow:ty, $uhi:ty) => {
|
||||
impl MultiplyRational for $ulow {
|
||||
fn multiply_rational(self, n: Self, d: Self, r: Rounding) -> Option<Self> {
|
||||
if d.is_zero() {
|
||||
return None
|
||||
}
|
||||
|
||||
let sn = (self as $uhi) * (n as $uhi);
|
||||
let mut result = sn / (d as $uhi);
|
||||
let remainder = (sn % (d as $uhi)) as $ulow;
|
||||
if match r {
|
||||
Rounding::Up => remainder > 0,
|
||||
// cannot be `(d + 1) / 2` since `d` might be `max_value` and overflow.
|
||||
Rounding::NearestPrefUp => remainder >= d / 2 + d % 2,
|
||||
Rounding::NearestPrefDown => remainder > d / 2,
|
||||
Rounding::Down => false,
|
||||
} {
|
||||
result = match result.checked_add(1) {
|
||||
Some(v) => v,
|
||||
None => return None,
|
||||
};
|
||||
}
|
||||
if result > (<$ulow>::max_value() as $uhi) {
|
||||
None
|
||||
} else {
|
||||
Some(result as $ulow)
|
||||
}
|
||||
}
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
impl_rrm!(u8, u16);
|
||||
impl_rrm!(u16, u32);
|
||||
impl_rrm!(u32, u64);
|
||||
impl_rrm!(u64, u128);
|
||||
|
||||
impl MultiplyRational for u128 {
|
||||
fn multiply_rational(self, n: Self, d: Self, r: Rounding) -> Option<Self> {
|
||||
crate::helpers_128bit::multiply_by_rational_with_rounding(self, n, d, r)
|
||||
}
|
||||
}
|
||||
|
||||
#[cfg(test)]
|
||||
mod tests {
|
||||
use super::{helpers_128bit::*, *};
|
||||
|
||||
@@ -32,6 +32,8 @@ use sp_std::ops::{
|
||||
Add, AddAssign, Div, DivAssign, Mul, MulAssign, Rem, RemAssign, Shl, Shr, Sub, SubAssign,
|
||||
};
|
||||
|
||||
use crate::MultiplyRational;
|
||||
|
||||
/// A meta trait for arithmetic type operations, regardless of any limitation on size.
|
||||
pub trait BaseArithmetic:
|
||||
From<u8>
|
||||
@@ -186,9 +188,9 @@ pub trait AtLeast32Bit: BaseArithmetic + From<u16> + From<u32> {}
|
||||
impl<T: BaseArithmetic + From<u16> + From<u32>> AtLeast32Bit for T {}
|
||||
|
||||
/// A meta trait for arithmetic. Same as [`AtLeast32Bit `], but also bounded to be unsigned.
|
||||
pub trait AtLeast32BitUnsigned: AtLeast32Bit + Unsigned {}
|
||||
pub trait AtLeast32BitUnsigned: AtLeast32Bit + Unsigned + MultiplyRational {}
|
||||
|
||||
impl<T: AtLeast32Bit + Unsigned> AtLeast32BitUnsigned for T {}
|
||||
impl<T: AtLeast32Bit + Unsigned + MultiplyRational> AtLeast32BitUnsigned for T {}
|
||||
|
||||
/// Just like `From` except that if the source value is too big to fit into the destination type
|
||||
/// then it'll saturate the destination.
|
||||
|
||||
Reference in New Issue
Block a user