#![allow(dead_code)]
use util::*;
cfg_if! {
if #[cfg(limb_width_64)] {
pub type Limb = u64;
type Wide = u128;
type SignedWide = i128;
} else {
pub type Limb = u32;
type Wide = u64;
type SignedWide = i64;
}}
#[inline]
pub(super) fn as_limb<T: Integer>(t: T)
-> Limb
{
as_cast(t)
}
#[inline]
fn as_wide<T: Integer>(t: T)
-> Wide
{
as_cast(t)
}
#[inline]
fn as_signed_wide<T: Integer>(t: T)
-> SignedWide
{
as_cast(t)
}
#[inline]
fn split_u16(x: u16) -> [Limb; 1] {
[as_limb(x)]
}
#[inline]
fn split_u32(x: u32) -> [Limb; 1] {
[as_limb(x)]
}
#[cfg(limb_width_32)]
#[inline]
fn split_u64(x: u64) -> [Limb; 2] {
[as_limb(x), as_limb(x >> 32)]
}
#[cfg(limb_width_64)]
#[inline]
fn split_u64(x: u64) -> [Limb; 1] {
[as_limb(x)]
}
#[cfg(all(has_i128, limb_width_32))]
#[inline]
fn split_u128(x: u128) -> [Limb; 4] {
[as_limb(x), as_limb(x >> 32), as_limb(x >> 64), as_limb(x >> 96)]
}
#[cfg(all(has_i128, limb_width_64))]
#[inline]
fn split_u128(x: u128) -> [Limb; 2] {
[as_limb(x), as_limb(x >> 64)]
}
#[inline]
pub fn nonzero<T: Integer>(x: &[T], rindex: usize) -> bool {
let len = x.len();
let slc = &x[..len-rindex];
slc.iter().rev().any(|&x| x != T::ZERO)
}
#[inline]
fn u16_to_hi16_1(r0: u16) -> (u16, bool) {
debug_assert!(r0 != 0);
let ls = r0.leading_zeros();
(r0 << ls, false)
}
#[inline]
fn u16_to_hi16_2(r0: u16, r1: u16) -> (u16, bool) {
debug_assert!(r0 != 0);
let ls = r0.leading_zeros();
let rs = 16 - ls;
let v = match ls {
0 => r0,
_ => (r0 << ls) | (r1 >> rs),
};
let n = r1 << ls != 0;
(v, n)
}
#[inline]
fn u32_to_hi16_1(r0: u32) -> (u16, bool) {
let r0 = u32_to_hi32_1(r0).0;
((r0 >> 16).as_u16(), r0.as_u16() != 0)
}
#[inline]
fn u32_to_hi16_2(r0: u32, r1: u32) -> (u16, bool) {
let (r0, n) = u32_to_hi32_2(r0, r1);
((r0 >> 16).as_u16(), n || r0.as_u16() != 0)
}
#[inline]
fn u64_to_hi16_1(r0: u64) -> (u16, bool) {
let r0 = u64_to_hi64_1(r0).0;
((r0 >> 48).as_u16(), r0.as_u16() != 0)
}
#[inline]
fn u64_to_hi16_2(r0: u64, r1: u64) -> (u16, bool) {
let (r0, n) = u64_to_hi64_2(r0, r1);
((r0 >> 48).as_u16(), n || r0.as_u16() != 0)
}
trait Hi16<T>: SliceLike<T> {
fn hi16_1(&self) -> (u16, bool);
fn hi16_2(&self) -> (u16, bool);
fn hi16(&self) -> (u16, bool) {
match self.len() {
0 => (0, false),
1 => self.hi16_1(),
_ => self.hi16_2(),
}
}
}
impl Hi16<u16> for [u16] {
#[inline]
fn hi16_1(&self) -> (u16, bool) {
debug_assert!(self.len() == 1);
let rview = self.rview();
let r0 = rview[0];
u16_to_hi16_1(r0)
}
#[inline]
fn hi16_2(&self) -> (u16, bool) {
debug_assert!(self.len() == 2);
let rview = self.rview();
let r0 = rview[0];
let r1 = rview[1];
let (v, n) = u16_to_hi16_2(r0, r1);
(v, n || nonzero(self, 2))
}
}
impl Hi16<u32> for [u32] {
#[inline]
fn hi16_1(&self) -> (u16, bool) {
debug_assert!(self.len() == 1);
let rview = self.rview();
let r0 = rview[0];
u32_to_hi16_1(r0)
}
#[inline]
fn hi16_2(&self) -> (u16, bool) {
debug_assert!(self.len() == 2);
let rview = self.rview();
let r0 = rview[0];
let r1 = rview[1];
let (v, n) = u32_to_hi16_2(r0, r1);
(v, n || nonzero(self, 2))
}
}
impl Hi16<u64> for [u64] {
#[inline]
fn hi16_1(&self) -> (u16, bool) {
debug_assert!(self.len() == 1);
let rview = self.rview();
let r0 = rview[0];
u64_to_hi16_1(r0)
}
#[inline]
fn hi16_2(&self) -> (u16, bool) {
debug_assert!(self.len() == 2);
let rview = self.rview();
let r0 = rview[0];
let r1 = rview[1];
let (v, n) = u64_to_hi16_2(r0, r1);
(v, n || nonzero(self, 2))
}
}
#[inline]
fn u32_to_hi32_1(r0: u32) -> (u32, bool) {
debug_assert!(r0 != 0);
let ls = r0.leading_zeros();
(r0 << ls, false)
}
#[inline]
fn u32_to_hi32_2(r0: u32, r1: u32) -> (u32, bool) {
debug_assert!(r0 != 0);
let ls = r0.leading_zeros();
let rs = 32 - ls;
let v = match ls {
0 => r0,
_ => (r0 << ls) | (r1 >> rs),
};
let n = r1 << ls != 0;
(v, n)
}
#[inline]
fn u64_to_hi32_1(r0: u64) -> (u32, bool) {
let r0 = u64_to_hi64_1(r0).0;
((r0 >> 32).as_u32(), r0.as_u32() != 0)
}
#[inline]
fn u64_to_hi32_2(r0: u64, r1: u64) -> (u32, bool) {
let (r0, n) = u64_to_hi64_2(r0, r1);
((r0 >> 32).as_u32(), n || r0.as_u32() != 0)
}
trait Hi32<T>: SliceLike<T> {
fn hi32_1(&self) -> (u32, bool);
fn hi32_2(&self) -> (u32, bool);
fn hi32_3(&self) -> (u32, bool);
fn hi32(&self) -> (u32, bool) {
match self.len() {
0 => (0, false),
1 => self.hi32_1(),
2 => self.hi32_2(),
_ => self.hi32_3(),
}
}
}
impl Hi32<u16> for [u16] {
#[inline]
fn hi32_1(&self) -> (u32, bool) {
debug_assert!(self.len() == 1);
let rview = self.rview();
u32_to_hi32_1(rview[0].as_u32())
}
#[inline]
fn hi32_2(&self) -> (u32, bool) {
debug_assert!(self.len() == 2);
let rview = self.rview();
let r0 = rview[0].as_u32() << 16;
let r1 = rview[1].as_u32();
u32_to_hi32_1(r0 | r1)
}
#[inline]
fn hi32_3(&self) -> (u32, bool) {
debug_assert!(self.len() >= 3);
let rview = self.rview();
let r0 = rview[0].as_u32();
let r1 = rview[1].as_u32() << 16;
let r2 = rview[2].as_u32();
let (v, n) = u32_to_hi32_2( r0, r1 | r2);
(v, n || nonzero(self, 3))
}
}
impl Hi32<u32> for [u32] {
#[inline]
fn hi32_1(&self) -> (u32, bool) {
debug_assert!(self.len() == 1);
let rview = self.rview();
let r0 = rview[0];
u32_to_hi32_1(r0)
}
#[inline]
fn hi32_2(&self) -> (u32, bool) {
debug_assert!(self.len() >= 2);
let rview = self.rview();
let r0 = rview[0];
let r1 = rview[1];
let (v, n) = u32_to_hi32_2(r0, r1);
(v, n || nonzero(self, 2))
}
#[inline]
fn hi32_3(&self) -> (u32, bool) {
self.hi32_2()
}
}
impl Hi32<u64> for [u64] {
#[inline]
fn hi32_1(&self) -> (u32, bool) {
debug_assert!(self.len() == 1);
let rview = self.rview();
let r0 = rview[0];
u64_to_hi32_1(r0)
}
#[inline]
fn hi32_2(&self) -> (u32, bool) {
debug_assert!(self.len() >= 2);
let rview = self.rview();
let r0 = rview[0];
let r1 = rview[1];
let (v, n) = u64_to_hi32_2(r0, r1);
(v, n || nonzero(self, 2))
}
#[inline]
fn hi32_3(&self) -> (u32, bool) {
self.hi32_2()
}
}
#[inline]
fn u64_to_hi64_1(r0: u64) -> (u64, bool) {
debug_assert!(r0 != 0);
let ls = r0.leading_zeros();
(r0 << ls, false)
}
#[inline]
fn u64_to_hi64_2(r0: u64, r1: u64) -> (u64, bool) {
debug_assert!(r0 != 0);
let ls = r0.leading_zeros();
let rs = 64 - ls;
let v = match ls {
0 => r0,
_ => (r0 << ls) | (r1 >> rs),
};
let n = r1 << ls != 0;
(v, n)
}
trait Hi64<T>: SliceLike<T> {
fn hi64_1(&self) -> (u64, bool);
fn hi64_2(&self) -> (u64, bool);
fn hi64_3(&self) -> (u64, bool);
fn hi64_4(&self) -> (u64, bool);
fn hi64_5(&self) -> (u64, bool);
fn hi64(&self) -> (u64, bool) {
match self.len() {
0 => (0, false),
1 => self.hi64_1(),
2 => self.hi64_2(),
3 => self.hi64_3(),
4 => self.hi64_4(),
_ => self.hi64_5(),
}
}
}
impl Hi64<u16> for [u16] {
#[inline]
fn hi64_1(&self) -> (u64, bool) {
debug_assert!(self.len() == 1);
let rview = self.rview();
let r0 = rview[0].as_u64();
u64_to_hi64_1(r0)
}
#[inline]
fn hi64_2(&self) -> (u64, bool) {
debug_assert!(self.len() == 2);
let rview = self.rview();
let r0 = rview[0].as_u64() << 16;
let r1 = rview[1].as_u64();
u64_to_hi64_1(r0 | r1)
}
#[inline]
fn hi64_3(&self) -> (u64, bool) {
debug_assert!(self.len() == 3);
let rview = self.rview();
let r0 = rview[0].as_u64() << 32;
let r1 = rview[1].as_u64() << 16;
let r2 = rview[2].as_u64();
u64_to_hi64_1(r0 | r1 | r2)
}
#[inline]
fn hi64_4(&self) -> (u64, bool) {
debug_assert!(self.len() == 4);
let rview = self.rview();
let r0 = rview[0].as_u64() << 48;
let r1 = rview[1].as_u64() << 32;
let r2 = rview[2].as_u64() << 16;
let r3 = rview[3].as_u64();
u64_to_hi64_1(r0 | r1 | r2 | r3)
}
#[inline]
fn hi64_5(&self) -> (u64, bool) {
debug_assert!(self.len() >= 5);
let rview = self.rview();
let r0 = rview[0].as_u64();
let r1 = rview[1].as_u64() << 48;
let r2 = rview[2].as_u64() << 32;
let r3 = rview[3].as_u64() << 16;
let r4 = rview[4].as_u64();
let (v, n) = u64_to_hi64_2(r0, r1 | r2 | r3 | r4);
(v, n || nonzero(self, 5))
}
}
impl Hi64<u32> for [u32] {
#[inline]
fn hi64_1(&self) -> (u64, bool) {
debug_assert!(self.len() == 1);
let rview = self.rview();
let r0 = rview[0].as_u64();
u64_to_hi64_1(r0)
}
#[inline]
fn hi64_2(&self) -> (u64, bool) {
debug_assert!(self.len() == 2);
let rview = self.rview();
let r0 = rview[0].as_u64() << 32;
let r1 = rview[1].as_u64();
u64_to_hi64_1(r0 | r1)
}
#[inline]
fn hi64_3(&self) -> (u64, bool) {
debug_assert!(self.len() >= 3);
let rview = self.rview();
let r0 = rview[0].as_u64();
let r1 = rview[1].as_u64() << 32;
let r2 = rview[2].as_u64();
let (v, n) = u64_to_hi64_2(r0, r1 | r2);
(v, n || nonzero(self, 3))
}
#[inline]
fn hi64_4(&self) -> (u64, bool) {
self.hi64_3()
}
#[inline]
fn hi64_5(&self) -> (u64, bool) {
self.hi64_3()
}
}
impl Hi64<u64> for [u64] {
#[inline]
fn hi64_1(&self) -> (u64, bool) {
debug_assert!(self.len() == 1);
let rview = self.rview();
let r0 = rview[0];
u64_to_hi64_1(r0)
}
#[inline]
fn hi64_2(&self) -> (u64, bool) {
debug_assert!(self.len() >= 2);
let rview = self.rview();
let r0 = rview[0];
let r1 = rview[1];
let (v, n) = u64_to_hi64_2(r0, r1);
(v, n || nonzero(self, 2))
}
#[inline]
fn hi64_3(&self) -> (u64, bool) {
self.hi64_2()
}
#[inline]
fn hi64_4(&self) -> (u64, bool) {
self.hi64_2()
}
#[inline]
fn hi64_5(&self) -> (u64, bool) {
self.hi64_2()
}
}
#[cfg(has_i128)]
#[inline]
fn u128_to_hi128_1(r0: u128) -> (u128, bool) {
let ls = r0.leading_zeros();
(r0 << ls, false)
}
#[cfg(has_i128)]
#[inline]
fn u128_to_hi128_2(r0: u128, r1: u128) -> (u128, bool) {
let ls = r0.leading_zeros();
let rs = 128 - ls;
let v = (r0 << ls) | (r1 >> rs);
let n = r1 << ls != 0;
(v, n)
}
#[cfg(has_i128)]
trait Hi128<T>: SliceLike<T> {
fn hi128_1(&self) -> (u128, bool);
fn hi128_2(&self) -> (u128, bool);
fn hi128_3(&self) -> (u128, bool);
fn hi128_4(&self) -> (u128, bool);
fn hi128_5(&self) -> (u128, bool);
fn hi128_6(&self) -> (u128, bool);
fn hi128_7(&self) -> (u128, bool);
fn hi128_8(&self) -> (u128, bool);
fn hi128_9(&self) -> (u128, bool);
fn hi128(&self) -> (u128, bool) {
match self.len() {
0 => (0, false),
1 => self.hi128_1(),
2 => self.hi128_2(),
3 => self.hi128_3(),
4 => self.hi128_4(),
6 => self.hi128_6(),
7 => self.hi128_7(),
8 => self.hi128_8(),
_ => self.hi128_9(),
}
}
}
#[cfg(has_i128)]
impl Hi128<u16> for [u16] {
#[inline]
fn hi128_1(&self) -> (u128, bool) {
debug_assert!(self.len() == 1);
let rview = self.rview();
let r0 = rview[0].as_u128();
u128_to_hi128_1(r0)
}
#[inline]
fn hi128_2(&self) -> (u128, bool) {
debug_assert!(self.len() == 2);
let rview = self.rview();
let r0 = rview[0].as_u128() << 16;
let r1 = rview[1].as_u128();
u128_to_hi128_1(r0 | r1)
}
#[inline]
fn hi128_3(&self) -> (u128, bool) {
debug_assert!(self.len() == 3);
let rview = self.rview();
let r0 = rview[0].as_u128() << 32;
let r1 = rview[1].as_u128() << 16;
let r2 = rview[2].as_u128();
u128_to_hi128_1(r0 | r1 | r2)
}
#[inline]
fn hi128_4(&self) -> (u128, bool) {
debug_assert!(self.len() == 4);
let rview = self.rview();
let r0 = rview[0].as_u128() << 48;
let r1 = rview[1].as_u128() << 32;
let r2 = rview[2].as_u128() << 16;
let r3 = rview[3].as_u128();
u128_to_hi128_1(r0 | r1 | r2 | r3)
}
#[inline]
fn hi128_5(&self) -> (u128, bool) {
debug_assert!(self.len() == 5);
let rview = self.rview();
let r0 = rview[0].as_u128() << 64;
let r1 = rview[1].as_u128() << 48;
let r2 = rview[2].as_u128() << 32;
let r3 = rview[3].as_u128() << 16;
let r4 = rview[4].as_u128();
u128_to_hi128_1(r0 | r1 | r2 | r3 | r4)
}
#[inline]
fn hi128_6(&self) -> (u128, bool) {
debug_assert!(self.len() == 6);
let rview = self.rview();
let r0 = rview[0].as_u128() << 80;
let r1 = rview[1].as_u128() << 64;
let r2 = rview[2].as_u128() << 48;
let r3 = rview[3].as_u128() << 32;
let r4 = rview[4].as_u128() << 16;
let r5 = rview[5].as_u128();
u128_to_hi128_1(r0 | r1 | r2 | r3 | r4 | r5)
}
#[inline]
fn hi128_7(&self) -> (u128, bool) {
debug_assert!(self.len() == 7);
let rview = self.rview();
let r0 = rview[0].as_u128() << 96;
let r1 = rview[1].as_u128() << 80;
let r2 = rview[2].as_u128() << 64;
let r3 = rview[3].as_u128() << 48;
let r4 = rview[4].as_u128() << 32;
let r5 = rview[5].as_u128() << 16;
let r6 = rview[6].as_u128();
u128_to_hi128_1(r0 | r1 | r2 | r3 | r4 | r5 | r6)
}
#[inline]
fn hi128_8(&self) -> (u128, bool) {
debug_assert!(self.len() == 8);
let rview = self.rview();
let r0 = rview[0].as_u128() << 112;
let r1 = rview[1].as_u128() << 96;
let r2 = rview[2].as_u128() << 80;
let r3 = rview[3].as_u128() << 64;
let r4 = rview[4].as_u128() << 48;
let r5 = rview[5].as_u128() << 32;
let r6 = rview[6].as_u128() << 16;
let r7 = rview[7].as_u128();
u128_to_hi128_1(r0 | r1 | r2 | r3 | r4 | r5 | r6 | r7)
}
#[inline]
fn hi128_9(&self) -> (u128, bool) {
debug_assert!(self.len() >= 9);
let rview = self.rview();
let r0 = rview[0].as_u128();
let r1 = rview[1].as_u128() << 112;
let r2 = rview[2].as_u128() << 96;
let r3 = rview[3].as_u128() << 80;
let r4 = rview[4].as_u128() << 64;
let r5 = rview[5].as_u128() << 48;
let r6 = rview[6].as_u128() << 32;
let r7 = rview[7].as_u128() << 16;
let r8 = rview[8].as_u128();
let (v, n) = u128_to_hi128_2(r0, r1 | r2 | r3 | r4 | r5 | r6 | r7 | r8);
(v, n || nonzero(self, 9))
}
}
#[cfg(has_i128)]
impl Hi128<u32> for [u32] {
#[inline]
fn hi128_1(&self) -> (u128, bool) {
debug_assert!(self.len() == 1);
let rview = self.rview();
let r0 = rview[0].as_u128();
u128_to_hi128_1(r0)
}
#[inline]
fn hi128_2(&self) -> (u128, bool) {
debug_assert!(self.len() == 2);
let rview = self.rview();
let r0 = rview[0].as_u128() << 32;
let r1 = rview[1].as_u128();
u128_to_hi128_1(r0 | r1)
}
#[inline]
fn hi128_3(&self) -> (u128, bool) {
debug_assert!(self.len() == 3);
let rview = self.rview();
let r0 = rview[0].as_u128() << 64;
let r1 = rview[1].as_u128() << 32;
let r2 = rview[2].as_u128();
u128_to_hi128_1(r0 | r1 | r2)
}
#[inline]
fn hi128_4(&self) -> (u128, bool) {
debug_assert!(self.len() == 4);
let rview = self.rview();
let r0 = rview[0].as_u128() << 96;
let r1 = rview[1].as_u128() << 64;
let r2 = rview[2].as_u128() << 32;
let r3 = rview[3].as_u128();
u128_to_hi128_1(r0 | r1 | r2 | r3)
}
#[inline]
fn hi128_5(&self) -> (u128, bool) {
debug_assert!(self.len() >= 5);
let rview = self.rview();
let r0 = rview[0].as_u128();
let r1 = rview[1].as_u128() << 96;
let r2 = rview[2].as_u128() << 64;
let r3 = rview[3].as_u128() << 32;
let r4 = rview[4].as_u128();
let (v, n) = u128_to_hi128_2(r0, r1 | r2 | r3 | r4);
(v, n || nonzero(self, 5))
}
#[inline]
fn hi128_6(&self) -> (u128, bool) {
self.hi128_5()
}
#[inline]
fn hi128_7(&self) -> (u128, bool) {
self.hi128_5()
}
#[inline]
fn hi128_8(&self) -> (u128, bool) {
self.hi128_5()
}
#[inline]
fn hi128_9(&self) -> (u128, bool) {
self.hi128_5()
}
}
#[cfg(has_i128)]
impl Hi128<u64> for [u64] {
#[inline]
fn hi128_1(&self) -> (u128, bool) {
debug_assert!(self.len() == 1);
let rview = self.rview();
let r0 = rview[0].as_u128();
u128_to_hi128_1(r0)
}
#[inline]
fn hi128_2(&self) -> (u128, bool) {
debug_assert!(self.len() == 2);
let rview = self.rview();
let r0 = rview[0].as_u128() << 64;
let r1 = rview[1].as_u128();
u128_to_hi128_1(r0 | r1)
}
#[inline]
fn hi128_3(&self) -> (u128, bool) {
debug_assert!(self.len() >= 3);
let rview = self.rview();
let r0 = rview[0].as_u128();
let r1 = rview[1].as_u128() << 64;
let r2 = rview[2].as_u128();
let (v, n) = u128_to_hi128_2(r0, r1 | r2);
(v, n || nonzero(self, 3))
}
#[inline]
fn hi128_4(&self) -> (u128, bool) {
self.hi128_3()
}
#[inline]
fn hi128_5(&self) -> (u128, bool) {
self.hi128_3()
}
#[inline]
fn hi128_6(&self) -> (u128, bool) {
self.hi128_3()
}
#[inline]
fn hi128_7(&self) -> (u128, bool) {
self.hi128_3()
}
#[inline]
fn hi128_8(&self) -> (u128, bool) {
self.hi128_3()
}
#[inline]
fn hi128_9(&self) -> (u128, bool) {
self.hi128_3()
}
}
pub(in atof::algorithm) mod scalar {
use super::*;
#[inline]
pub fn add(x: Limb, y: Limb)
-> (Limb, bool)
{
x.overflowing_add(y)
}
#[inline]
pub fn iadd(x: &mut Limb, y: Limb)
-> bool
{
let t = add(*x, y);
*x = t.0;
t.1
}
#[inline]
pub fn sub(x: Limb, y: Limb)
-> (Limb, bool)
{
x.overflowing_sub(y)
}
#[inline]
pub fn isub(x: &mut Limb, y: Limb)
-> bool
{
let t = sub(*x, y);
*x = t.0;
t.1
}
#[inline]
pub fn mul(x: Limb, y: Limb, carry: Limb)
-> (Limb, Limb)
{
let z: Wide = as_wide(x) * as_wide(y) + as_wide(carry);
(as_limb(z), as_limb(z >> Limb::BITS))
}
#[inline]
pub fn imul(x: &mut Limb, y: Limb, carry: Limb)
-> Limb
{
let t = mul(*x, y, carry);
*x = t.0;
t.1
}
#[inline]
pub fn div(x: Limb, y: Limb, rem: Limb)
-> (Limb, Limb)
{
let x = as_wide(x) | (as_wide(rem) << Limb::BITS);
let y = as_wide(y);
(as_limb(x / y), as_limb(x % y))
}
#[inline]
pub fn idiv(x: &mut Limb, y: Limb, rem: Limb)
-> Limb
{
let t = div(*x, y, rem);
*x = t.0;
t.1
}
}
pub(in atof::algorithm) mod small {
use lib::iter;
use super::*;
use super::super::small_powers::*;
use super::super::large_powers::*;
#[inline]
pub fn leading_zero_limbs(_: &[Limb]) -> usize {
0
}
#[inline]
pub fn trailing_zero_limbs(x: &[Limb]) -> usize {
let mut iter = x.iter().enumerate();
let opt = iter.find(|&tup| !tup.1.is_zero());
let value = opt
.map(|t| t.0)
.unwrap_or(x.len());
value
}
#[inline]
pub fn leading_zeros(x: &[Limb]) -> usize {
if x.is_empty() {
0
} else {
x.rindex(0).leading_zeros().as_usize()
}
}
#[inline]
pub fn trailing_zeros(x: &[Limb]) -> usize {
let index = trailing_zero_limbs(x);
let mut count = index.saturating_mul(Limb::BITS);
if let Some(value) = x.get(index) {
count = count.saturating_add(value.trailing_zeros().as_usize());
}
count
}
#[inline]
pub fn bit_length(x: &[Limb]) -> usize {
let nlz = leading_zeros(x);
Limb::BITS.checked_mul(x.len())
.map(|v| v - nlz)
.unwrap_or(usize::max_value())
}
#[inline]
pub fn limb_length(x: &[Limb]) -> usize {
x.len()
}
#[inline]
pub fn ishr_bits<T>(x: &mut T, n: usize)
-> Limb
where T: CloneableVecLike<Limb>
{
let bits = Limb::BITS;
debug_assert!(n < bits && n != 0);
let lshift = bits - n;
let rshift = n;
let mut prev: Limb = 0;
for xi in x.iter_mut().rev() {
let tmp = *xi;
*xi >>= rshift;
*xi |= prev << lshift;
prev = tmp;
}
prev & lower_n_mask(as_limb(rshift))
}
#[inline]
pub fn ishr_limbs<T>(x: &mut T, n: usize)
-> bool
where T: CloneableVecLike<Limb>
{
debug_assert!(n != 0);
if n >= x.len() {
x.clear();
false
} else {
let is_zero = (&x[..n]).iter().all(|v| v.is_zero());
x.remove_many(0..n);
is_zero
}
}
pub fn ishr<T>(x: &mut T, n: usize)
-> bool
where T: CloneableVecLike<Limb>
{
let bits = Limb::BITS;
let rem = n % bits;
let div = n / bits;
let is_zero = match div.is_zero() {
true => true,
false => ishr_limbs(x, div),
};
let truncated = match rem.is_zero() {
true => 0,
false => ishr_bits(x, rem),
};
let roundup = {
let halfway = lower_n_halfway(as_limb(rem));
if truncated > halfway {
true
} else if truncated == halfway {
!is_zero || x[0].is_odd()
} else {
false
}
};
normalize(x);
roundup
}
#[inline]
pub fn shr<T>(x: &[Limb], n: usize)
-> (T, bool)
where T: CloneableVecLike<Limb>
{
let mut z = T::default();
z.extend_from_slice(x);
let roundup = ishr(&mut z, n);
(z, roundup)
}
#[inline]
pub fn ishl_bits<T>(x: &mut T, n: usize)
where T: CloneableVecLike<Limb>
{
let bits = Limb::BITS;
debug_assert!(n < bits);
if n.is_zero() {
return;
}
let rshift = bits - n;
let lshift = n;
let mut prev: Limb = 0;
for xi in x.iter_mut() {
let tmp = *xi;
*xi <<= lshift;
*xi |= prev >> rshift;
prev = tmp;
}
let carry = prev >> rshift;
if carry != 0 {
x.push(carry);
}
}
#[inline]
pub fn shl_bits<T>(x: &[Limb], n: usize)
-> T
where T: CloneableVecLike<Limb>
{
let mut z = T::default();
z.extend_from_slice(x);
ishl_bits(&mut z, n);
z
}
#[inline]
pub fn ishl_limbs<T>(x: &mut T, n: usize)
where T: CloneableVecLike<Limb>
{
debug_assert!(n != 0);
if !x.is_empty() {
x.insert_many(0, iter::repeat(0).take(n));
}
}
#[inline]
pub fn ishl<T>(x: &mut T, n: usize)
where T: CloneableVecLike<Limb>
{
let bits = Limb::BITS;
let rem = n % bits;
let div = n / bits;
ishl_bits(x, rem);
if !div.is_zero() {
ishl_limbs(x, div);
}
}
#[inline]
pub fn shl<T>(x: &[Limb], n: usize)
-> T
where T: CloneableVecLike<Limb>
{
let mut z = T::default();
z.extend_from_slice(x);
ishl(&mut z, n);
z
}
#[inline]
pub fn normalize<T>(x: &mut T)
where T: CloneableVecLike<Limb>
{
while !x.is_empty() && x.rindex(0).is_zero() {
x.pop();
}
}
pub fn iadd_impl<T>(x: &mut T, y: Limb, xstart: usize)
where T: CloneableVecLike<Limb>
{
if x.len() <= xstart {
x.push(y);
} else {
let mut carry = scalar::iadd(&mut x[xstart], y);
let mut size = xstart + 1;
while carry && size < x.len() {
carry = scalar::iadd(&mut x[size], 1);
size += 1;
}
if carry {
x.push(1);
}
}
}
#[inline]
pub fn iadd<T>(x: &mut T, y: Limb)
where T: CloneableVecLike<Limb>
{
iadd_impl(x, y, 0);
}
#[inline]
pub fn add<T>(x: &[Limb], y: Limb)
-> T
where T: CloneableVecLike<Limb>
{
let mut z = T::default();
z.extend_from_slice(x);
iadd(&mut z, y);
z
}
pub fn isub_impl<T>(x: &mut T, y: Limb, xstart: usize)
where T: CloneableVecLike<Limb>
{
debug_assert!(x.len() > xstart && (x[xstart] >= y || x.len() > xstart+1));
let mut carry = scalar::isub(&mut x[xstart], y);
let mut size = xstart + 1;
while carry && size < x.len() {
carry = scalar::isub(&mut x[size], 1);
size += 1;
}
normalize(x);
}
#[inline]
pub fn isub<T>(x: &mut T, y: Limb)
where T: CloneableVecLike<Limb>
{
isub_impl(x, y, 0);
}
#[inline]
pub fn sub<T>(x: &[Limb], y: Limb)
-> T
where T: CloneableVecLike<Limb>
{
let mut z = T::default();
z.extend_from_slice(x);
isub(&mut z, y);
z
}
#[inline]
pub fn imul<T>(x: &mut T, y: Limb)
where T: CloneableVecLike<Limb>
{
let mut carry: Limb = 0;
for xi in x.iter_mut() {
carry = scalar::imul(xi, y, carry);
}
if carry != 0 {
x.push(carry);
}
}
#[inline]
pub fn mul<T>(x: &[Limb], y: Limb)
-> T
where T: CloneableVecLike<Limb>
{
let mut z = T::default();
z.extend_from_slice(x);
imul(&mut z, y);
z
}
pub fn imul_power<T>(x: &mut T, radix: u32, n: u32)
where T: CloneableVecLike<Limb>
{
use super::large::KARATSUBA_CUTOFF;
let small_powers = get_small_powers(radix);
let large_powers = get_large_powers(radix);
if n == 0 {
return;
}
let bit_length = 32 - n.leading_zeros().as_usize();
debug_assert!(bit_length != 0 && bit_length <= large_powers.len());
if x.len() + large_powers[bit_length-1].len() < 2*KARATSUBA_CUTOFF {
let step = small_powers.len() - 1;
let power = small_powers[step];
let mut n = n.as_usize();
while n >= step {
imul(x, power);
n -= step;
}
imul(x, small_powers[n]);
} else {
let mut idx: usize = 0;
let mut bit: usize = 1;
let mut n = n.as_usize();
while n != 0 {
if n & bit != 0 {
debug_assert!(idx < large_powers.len());
large::imul(x, large_powers[idx]);
n ^= bit;
}
idx += 1;
bit <<= 1;
}
}
}
#[inline]
pub fn mul_power<T>(x: &[Limb], radix: u32, n: u32)
-> T
where T: CloneableVecLike<Limb>
{
let mut z = T::default();
z.extend_from_slice(x);
imul_power(&mut z, radix, n);
z
}
pub fn idiv<T>(x: &mut T, y: Limb)
-> Limb
where T: CloneableVecLike<Limb>
{
let mut rem: Limb = 0;
for xi in x.iter_mut().rev() {
rem = scalar::idiv(xi, y, rem);
}
normalize(x);
rem
}
#[inline]
pub fn div<T>(x: &[Limb], y: Limb)
-> (T, Limb)
where T: CloneableVecLike<Limb>
{
let mut z = T::default();
z.extend_from_slice(x);
let rem = idiv(&mut z, y);
(z, rem)
}
pub fn ipow<T>(x: &mut T, mut n: Limb)
where T: CloneableVecLike<Limb>
{
let mut base = T::default();
base.push(1);
mem::swap(x, &mut base);
loop {
if n.is_odd() {
large::imul(x, &base);
}
n /= 2;
if n.is_zero() {
break;
} else {
base = large::mul(&base, &base);
}
}
}
#[inline]
pub fn pow<T>(x: &[Limb], n: Limb)
-> T
where T: CloneableVecLike<Limb>
{
let mut z = T::default();
z.extend_from_slice(x);
ipow(&mut z, n);
z
}
}
pub(in atof::algorithm) mod large {
use lib::cmp;
use super::*;
#[inline]
pub fn compare(x: &[Limb], y: &[Limb]) -> cmp::Ordering {
if x.len() > y.len() {
return cmp::Ordering::Greater;
} else if x.len() < y.len() {
return cmp::Ordering::Less;
} else {
let iter = x.iter().rev().zip(y.iter().rev());
for (&xi, &yi) in iter {
if xi > yi {
return cmp::Ordering::Greater;
} else if xi < yi {
return cmp::Ordering::Less;
}
}
return cmp::Ordering::Equal;
}
}
#[inline]
pub fn greater(x: &[Limb], y: &[Limb]) -> bool {
compare(x, y) == cmp::Ordering::Greater
}
#[inline]
pub fn greater_equal(x: &[Limb], y: &[Limb]) -> bool {
!less(x, y)
}
#[inline]
pub fn less(x: &[Limb], y: &[Limb]) -> bool {
compare(x, y) == cmp::Ordering::Less
}
#[inline]
pub fn less_equal(x: &[Limb], y: &[Limb]) -> bool {
!greater(x, y)
}
#[inline]
pub fn equal(x: &[Limb], y: &[Limb]) -> bool {
let mut iter = x.iter().rev().zip(y.iter().rev());
x.len() == y.len() && iter.all(|(&xi, &yi)| xi == yi)
}
pub fn iadd_impl<T>(x: &mut T, y: &[Limb], xstart: usize)
where T: CloneableVecLike<Limb>
{
if y.len() > x.len() - xstart {
x.resize(y.len() + xstart, 0);
}
let mut carry = false;
for (xi, yi) in (&mut x[xstart..]).iter_mut().zip(y.iter()) {
let mut tmp = scalar::iadd(xi, *yi);
if carry {
tmp |= scalar::iadd(xi, 1);
}
carry = tmp;
}
if carry {
small::iadd_impl(x, 1, y.len()+xstart);
}
}
#[inline]
pub fn iadd<T>(x: &mut T, y: &[Limb])
where T: CloneableVecLike<Limb>
{
iadd_impl(x, y, 0)
}
#[inline]
pub fn add<T>(x: &[Limb], y: &[Limb])
-> T
where T: CloneableVecLike<Limb>
{
let mut z = T::default();
z.extend_from_slice(x);
iadd(&mut z, y);
z
}
pub fn isub<T>(x: &mut T, y: &[Limb])
where T: CloneableVecLike<Limb>
{
debug_assert!(greater_equal(x, y));
let mut carry = false;
for (xi, yi) in x.iter_mut().zip(y.iter()) {
let mut tmp = scalar::isub(xi, *yi);
if carry {
tmp |= scalar::isub(xi, 1);
}
carry = tmp;
}
if carry {
small::isub_impl(x, 1, y.len());
} else {
small::normalize(x);
}
}
#[inline]
pub fn sub<T>(x: &[Limb], y: &[Limb])
-> T
where T: CloneableVecLike<Limb>
{
let mut z = T::default();
z.extend_from_slice(x);
isub(&mut z, y);
z
}
pub const KARATSUBA_CUTOFF: usize = 32;
fn long_mul<T>(x: &[Limb], y: &[Limb])
-> T
where T: CloneableVecLike<Limb>
{
let mut z: T = small::mul(x, y[0]);
z.resize(x.len() + y.len(), 0);
for (i, &yi) in y[1..].iter().enumerate() {
let zi: T = small::mul(x, yi);
iadd_impl(&mut z, &zi, i+1);
}
small::normalize(&mut z);
z
}
#[inline]
pub fn karatsuba_split<'a>(z: &'a [Limb], m: usize)
-> (&'a [Limb], &'a [Limb])
{
(&z[..m], &z[m..])
}
fn karatsuba_mul<T>(x: &[Limb], y: &[Limb])
-> T
where T: CloneableVecLike<Limb>
{
if y.len() <= KARATSUBA_CUTOFF {
long_mul(x, y)
} else if x.len() < y.len() / 2 {
karatsuba_uneven_mul(x, y)
} else {
let m = y.len() / 2;
let (xl, xh) = karatsuba_split(x, m);
let (yl, yh) = karatsuba_split(y, m);
let sumx: T = add(xl, xh);
let sumy: T = add(yl, yh);
let z0: T = karatsuba_mul(xl, yl);
let mut z1: T = karatsuba_mul(&sumx, &sumy);
let z2: T = karatsuba_mul(xh, yh);
isub(&mut z1, &z2);
isub(&mut z1, &z0);
let mut result = T::default();
let len = z0.len().max(m + z1.len()).max(2*m + z2.len());
result.reserve_exact(len);
result.extend_from_slice(&z0);
iadd_impl(&mut result, &z1, m);
iadd_impl(&mut result, &z2, 2*m);
result
}
}
fn karatsuba_uneven_mul<T>(x: &[Limb], mut y: &[Limb])
-> T
where T: CloneableVecLike<Limb>
{
let mut result = T::default();
result.resize(x.len() + y.len(), 0);
let mut start = 0;
while y.len() != 0 {
let m = x.len().min(y.len());
let (yl, yh) = karatsuba_split(y, m);
let prod: T = karatsuba_mul(x, yl);
iadd_impl(&mut result, &prod, start);
y = yh;
start += m;
}
small::normalize(&mut result);
result
}
#[inline]
fn karatsuba_mul_fwd<T>(x: &[Limb], y: &[Limb])
-> T
where T: CloneableVecLike<Limb>
{
if x.len() < y.len() {
karatsuba_mul(x, y)
} else {
karatsuba_mul(y, x)
}
}
#[inline]
pub fn imul<T>(x: &mut T, y: &[Limb])
where T: CloneableVecLike<Limb>
{
if y.len() == 1 {
small::imul(x, y[0]);
} else {
*x = karatsuba_mul_fwd(x, y);
}
}
#[inline]
pub fn mul<T>(x: &[Limb], y: &[Limb])
-> T
where T: CloneableVecLike<Limb>
{
let mut z = T::default();
z.extend_from_slice(x);
imul(&mut z, y);
z
}
const ALGORITHM_D_B: Wide = 1 << Limb::BITS;
const ALGORITHM_D_M: Wide = ALGORITHM_D_B - 1;
#[inline]
fn calculate_qhat(x: &[Limb], y: &[Limb], j: usize)
-> Wide
{
let n = y.len();
let x_jn = as_wide(x[j+n]);
let x_jn1 = as_wide(x[j+n-1]);
let num = (x_jn << Limb::BITS) + x_jn1;
let den = as_wide(y[n-1]);
let mut qhat = num / den;
let mut rhat = num - qhat * den;
let x_jn2 = as_wide(x[j+n-2]);
let y_n2 = as_wide(y[n-2]);
let y_n1 = as_wide(y[n-1]);
while qhat >= ALGORITHM_D_B || qhat * y_n2 > (rhat << Limb::BITS) + x_jn2 {
qhat -= 1;
rhat += y_n1;
if rhat >= ALGORITHM_D_B {
break;
}
}
qhat
}
#[inline]
fn multiply_and_subtract<T>(x: &mut T, y: &T, qhat: Wide, j: usize)
-> SignedWide
where T: CloneableVecLike<Limb>
{
let n = y.len();
let mut k: SignedWide = 0;
let mut t: SignedWide;
for i in 0..n {
let x_ij = as_signed_wide(x[i+j]);
let y_i = as_wide(y[i]);
let p = qhat * y_i;
t = x_ij.wrapping_sub(k).wrapping_sub(as_signed_wide(p & ALGORITHM_D_M));
x[i+j] = as_limb(t);
k = as_signed_wide(p >> Limb::BITS) - (t >> Limb::BITS);
}
t = as_signed_wide(x[j+n]) - k;
x[j+n] = as_limb(t);
t
}
#[inline]
fn test_quotient(qhat: Wide, t: SignedWide)
-> Wide
{
if t < 0 {
qhat - 1
} else {
qhat
}
}
#[inline]
fn add_back<T>(x: &mut T, y: &T, mut t: SignedWide, j: usize)
where T: CloneableVecLike<Limb>
{
let n = y.len();
if t < 0 {
let mut k: SignedWide = 0;
for i in 0..n {
t = as_signed_wide(as_wide(x[i+j]) + as_wide(y[i])) + k;
x[i+j] = as_limb(t);
k = t >> Limb::BITS;
}
let x_jn = as_signed_wide(x[j+n]) + k;
x[j+n] = as_limb(x_jn);
}
}
#[inline]
fn calculate_remainder<T>(x: &[Limb], y: &[Limb], s: usize)
-> T
where T: CloneableVecLike<Limb>
{
let n = y.len();
let mut r = T::default();
r.reserve_exact(n);
let rs = Limb::BITS - s;
for i in 0..n-1 {
let xi = as_wide(x[i]) >> s;
let xi1 = as_wide(x[i+1]) << rs;
let ri = xi | xi1;
r.push(as_limb(ri));
}
let x_n1 = x[n-1] >> s;
r.push(as_limb(x_n1));
r
}
fn algorithm_d_div<T>(x: &[Limb], y: &[Limb])
-> (T, T)
where T: CloneableVecLike<Limb>
{
let s = y.rindex(0).leading_zeros().as_usize();
let m = x.len();
let n = y.len();
let mut xn: T = small::shl_bits(x, s);
let yn: T = small::shl_bits(y, s);
xn.push(0);
let mut q = T::default();
q.resize(m-n+1, 0);
for j in (0..m-n+1).rev() {
let mut qhat = calculate_qhat(&xn, &yn, j);
if qhat != 0 {
let t = multiply_and_subtract(&mut xn, &yn, qhat, j);
qhat = test_quotient(qhat, t);
add_back(&mut xn, &yn, t, j);
}
q[j] = as_limb(qhat);
}
let mut r = calculate_remainder(&xn, &yn, s);
small::normalize(&mut q);
small::normalize(&mut r);
(q, r)
}
pub fn idiv<T>(x: &mut T, y: &[Limb])
-> T
where T: CloneableVecLike<Limb>
{
debug_assert!(y.len() != 0);
if x.len() < y.len() {
let mut r = T::default();
mem::swap(x, &mut r);
r
} else if y.len() == 1 {
let mut r = T::default();
r.push(small::idiv(x, y[0]));
r
} else {
let (q, r) = algorithm_d_div(x, y);
*x = q;
r
}
}
#[inline]
pub fn div<T>(x: &[Limb], y: &[Limb])
-> (T, T)
where T: CloneableVecLike<Limb>
{
let mut z = T::default();
z.extend_from_slice(x);
let rem = idiv(&mut z, y);
(z, rem)
}
pub fn quorem<T>(x: &mut T, y: &T)
-> Limb
where T: CloneableVecLike<Limb>
{
debug_assert!(y.len() > 0);
let mask = as_wide(Limb::max_value());
let m = x.len();
let n = y.len();
if m < n {
return 0;
}
let mut q = x[m-1] / (y[n-1] + 1);
if q != 0 {
let mut borrow: Wide = 0;
let mut carry: Wide = 0;
for j in 0..m {
let p = as_wide(y[j]) * as_wide(q) + carry;
carry = p >> Limb::BITS;
let t = as_wide(x[j]).wrapping_sub(p & mask).wrapping_sub(borrow);
borrow = (t >> Limb::BITS) & 1;
x[j] = as_limb(t);
}
small::normalize(x);
}
if greater_equal(x, y) {
q += 1;
let mut borrow: Wide = 0;
let mut carry: Wide = 0;
for j in 0..m {
let p = as_wide(y[j]) + carry;
carry = p >> Limb::BITS;
let t = as_wide(x[j]).wrapping_sub(p & mask).wrapping_sub(borrow);
borrow = (t >> Limb::BITS) & 1;
x[j] = as_limb(t);
}
small::normalize(x);
}
q
}
}
use lib::cmp;
use super::small_powers::*;
use super::large_powers::*;
macro_rules! imul_power {
($name:ident, $base:expr) => (
#[inline]
fn $name(&mut self, n: u32) {
self.imul_power_impl($base, n)
}
);
}
pub(in atof::algorithm) trait SharedOps: Clone + Sized + Default {
type StorageType: CloneableVecLike<Limb>;
fn data<'a>(&'a self) -> &'a Self::StorageType;
fn data_mut<'a>(&'a mut self) -> &'a mut Self::StorageType;
#[inline]
fn is_zero(&self) -> bool {
self.limb_length() == 0
}
#[inline]
fn compare(&self, y: &Self) -> cmp::Ordering {
large::compare(self.data(), y.data())
}
#[inline]
fn greater(&self, y: &Self) -> bool {
large::greater(self.data(), y.data())
}
#[inline]
fn greater_equal(&self, y: &Self) -> bool {
large::greater_equal(self.data(), y.data())
}
#[inline]
fn less(&self, y: &Self) -> bool {
large::less(self.data(), y.data())
}
#[inline]
fn less_equal(&self, y: &Self) -> bool {
large::less_equal(self.data(), y.data())
}
#[inline]
fn equal(&self, y: &Self) -> bool {
large::equal(self.data(), y.data())
}
#[inline]
fn leading_zero_limbs(&self) -> usize {
small::leading_zero_limbs(self.data())
}
#[inline]
fn trailing_zero_limbs(&self) -> usize {
small::trailing_zero_limbs(self.data())
}
#[inline]
fn leading_zeros(&self) -> usize {
small::leading_zeros(self.data())
}
#[inline]
fn trailing_zeros(&self) -> usize {
small::trailing_zeros(self.data())
}
#[inline]
fn bit_length(&self) -> usize {
small::bit_length(self.data())
}
#[inline]
fn limb_length(&self) -> usize {
small::limb_length(self.data())
}
#[inline]
fn hi16(&self) -> (u16, bool) {
self.data().as_slice().hi16()
}
#[inline]
fn hi32(&self) -> (u32, bool) {
self.data().as_slice().hi32()
}
#[inline]
fn hi64(&self) -> (u64, bool) {
self.data().as_slice().hi64()
}
#[cfg(has_i128)]
#[inline]
fn hi128(&self) -> (u128, bool) {
self.data().as_slice().hi128()
}
#[inline]
fn pad_zero_digits(&mut self, n: usize) -> usize {
small::ishl_limbs(self.data_mut(), n);
n
}
#[inline]
fn from_u16(x: u16) -> Self {
let mut v = Self::default();
let slc = split_u16(x);
v.data_mut().extend_from_slice(&slc);
v.normalize();
v
}
#[inline]
fn from_u32(x: u32) -> Self {
let mut v = Self::default();
let slc = split_u32(x);
v.data_mut().extend_from_slice(&slc);
v.normalize();
v
}
#[inline]
fn from_u64(x: u64) -> Self {
let mut v = Self::default();
let slc = split_u64(x);
v.data_mut().extend_from_slice(&slc);
v.normalize();
v
}
#[cfg(has_i128)]
#[inline]
fn from_u128(x: u128) -> Self {
let mut v = Self::default();
let slc = split_u128(x);
v.data_mut().extend_from_slice(&slc);
v.normalize();
v
}
#[inline]
fn normalize(&mut self) {
small::normalize(self.data_mut());
}
#[inline]
fn is_normalized(&self) -> bool {
self.data().is_empty() || !self.data().rindex(0).is_zero()
}
#[inline]
fn ishl(&mut self, n: usize) {
small::ishl(self.data_mut(), n);
}
fn shl(&self, n: usize) -> Self {
let mut x = self.clone();
x.ishl(n);
x
}
fn ishr(&mut self, n: usize, mut roundup: bool) {
roundup &= small::ishr(self.data_mut(), n);
if roundup {
if self.data().is_empty() {
self.data_mut().push(1);
} else {
self.data_mut()[0] += 1;
}
}
}
fn shr(&self, n: usize, roundup: bool) -> Self {
let mut x = self.clone();
x.ishr(n, roundup);
x
}
}
pub(in atof::algorithm) trait SmallOps: SharedOps {
#[inline]
fn small_powers(radix: u32) -> &'static [Limb] {
get_small_powers(radix)
}
#[inline]
fn large_powers(radix: u32) -> &'static [&'static [Limb]] {
get_large_powers(radix)
}
#[inline]
fn iadd_small(&mut self, y: Limb) {
small::iadd(self.data_mut(), y);
}
#[inline]
fn add_small(&self, y: Limb) -> Self {
let mut x = self.clone();
x.iadd_small(y);
x
}
#[inline]
fn isub_small(&mut self, y: Limb) {
small::isub(self.data_mut(), y);
}
#[inline]
fn sub_small(&mut self, y: Limb) -> Self {
let mut x = self.clone();
x.isub_small(y);
x
}
#[inline]
fn imul_small(&mut self, y: Limb) {
small::imul(self.data_mut(), y);
}
#[inline]
fn mul_small(&self, y: Limb) -> Self {
let mut x = self.clone();
x.imul_small(y);
x
}
#[inline]
fn imul_power_impl(&mut self, radix: u32, n: u32) {
small::imul_power(self.data_mut(), radix, n);
}
fn imul_power(&mut self, radix: u32, n: u32) {
match radix {
2 => self.imul_pow2(n),
3 => self.imul_pow3(n),
4 => self.imul_pow4(n),
5 => self.imul_pow5(n),
6 => self.imul_pow6(n),
7 => self.imul_pow7(n),
8 => self.imul_pow8(n),
9 => self.imul_pow9(n),
10 => self.imul_pow10(n),
11 => self.imul_pow11(n),
12 => self.imul_pow12(n),
13 => self.imul_pow13(n),
14 => self.imul_pow14(n),
15 => self.imul_pow15(n),
16 => self.imul_pow16(n),
17 => self.imul_pow17(n),
18 => self.imul_pow18(n),
19 => self.imul_pow19(n),
20 => self.imul_pow20(n),
21 => self.imul_pow21(n),
22 => self.imul_pow22(n),
23 => self.imul_pow23(n),
24 => self.imul_pow24(n),
25 => self.imul_pow25(n),
26 => self.imul_pow26(n),
27 => self.imul_pow27(n),
28 => self.imul_pow28(n),
29 => self.imul_pow29(n),
30 => self.imul_pow30(n),
31 => self.imul_pow31(n),
32 => self.imul_pow32(n),
33 => self.imul_pow33(n),
34 => self.imul_pow34(n),
35 => self.imul_pow35(n),
36 => self.imul_pow36(n),
_ => unreachable!()
}
}
fn imul_pow2(&mut self, n: u32) {
self.ishl(n.as_usize())
}
imul_power!(imul_pow3, 3);
#[inline]
fn imul_pow4(&mut self, n: u32) {
self.imul_pow2(2*n);
}
imul_power!(imul_pow5, 5);
#[inline]
fn imul_pow6(&mut self, n: u32) {
self.imul_pow3(n);
self.imul_pow2(n);
}
imul_power!(imul_pow7, 7);
#[inline]
fn imul_pow8(&mut self, n: u32) {
self.imul_pow2(3*n);
}
#[inline]
fn imul_pow9(&mut self, n: u32) {
self.imul_pow3(n);
self.imul_pow3(n);
}
#[inline]
fn imul_pow10(&mut self, n: u32) {
self.imul_pow5(n);
self.imul_pow2(n);
}
imul_power!(imul_pow11, 11);
#[inline]
fn imul_pow12(&mut self, n: u32) {
self.imul_pow3(n);
self.imul_pow4(n);
}
imul_power!(imul_pow13, 13);
#[inline]
fn imul_pow14(&mut self, n: u32) {
self.imul_pow7(n);
self.imul_pow2(n);
}
#[inline]
fn imul_pow15(&mut self, n: u32) {
self.imul_pow3(n);
self.imul_pow5(n);
}
#[inline]
fn imul_pow16(&mut self, n: u32) {
self.imul_pow2(4*n);
}
imul_power!(imul_pow17, 17);
#[inline]
fn imul_pow18(&mut self, n: u32) {
self.imul_pow9(n);
self.imul_pow2(n);
}
imul_power!(imul_pow19, 19);
#[inline]
fn imul_pow20(&mut self, n: u32) {
self.imul_pow5(n);
self.imul_pow4(n);
}
#[inline]
fn imul_pow21(&mut self, n: u32) {
self.imul_pow3(n);
self.imul_pow7(n);
}
#[inline]
fn imul_pow22(&mut self, n: u32) {
self.imul_pow11(n);
self.imul_pow2(n);
}
imul_power!(imul_pow23, 23);
#[inline]
fn imul_pow24(&mut self, n: u32) {
self.imul_pow3(n);
self.imul_pow8(n);
}
#[inline]
fn imul_pow25(&mut self, n: u32) {
self.imul_pow5(n);
self.imul_pow5(n);
}
#[inline]
fn imul_pow26(&mut self, n: u32) {
self.imul_pow13(n);
self.imul_pow2(n);
}
#[inline]
fn imul_pow27(&mut self, n: u32) {
self.imul_pow9(n);
self.imul_pow3(n);
}
#[inline]
fn imul_pow28(&mut self, n: u32) {
self.imul_pow7(n);
self.imul_pow4(n);
}
imul_power!(imul_pow29, 29);
#[inline]
fn imul_pow30(&mut self, n: u32) {
self.imul_pow15(n);
self.imul_pow2(n);
}
imul_power!(imul_pow31, 31);
#[inline]
fn imul_pow32(&mut self, n: u32) {
self.imul_pow2(5*n);
}
#[inline]
fn imul_pow33(&mut self, n: u32) {
self.imul_pow3(n);
self.imul_pow11(n);
}
#[inline]
fn imul_pow34(&mut self, n: u32) {
self.imul_pow17(n);
self.imul_pow2(n);
}
#[inline]
fn imul_pow35(&mut self, n: u32) {
self.imul_pow5(n);
self.imul_pow7(n);
}
#[inline]
fn imul_pow36(&mut self, n: u32) {
self.imul_pow9(n);
self.imul_pow4(n);
}
#[inline]
fn idiv_small(&mut self, y: Limb) -> Limb {
small::idiv(self.data_mut(), y)
}
#[inline]
fn div_small(&self, y: Limb) -> (Self, Limb) {
let mut x = self.clone();
let rem = x.idiv_small(y);
(x, rem)
}
#[inline]
fn ipow(&mut self, n: Limb) {
small::ipow(self.data_mut(), n);
}
#[inline]
fn pow(&self, n: Limb) -> Self {
let mut x = self.clone();
x.ipow(n);
x
}
}
pub(in atof::algorithm) trait LargeOps: SmallOps {
#[inline]
fn iadd_large(&mut self, y: &Self) {
large::iadd(self.data_mut(), y.data());
}
#[inline]
fn add_large(&mut self, y: &Self) -> Self {
let mut x = self.clone();
x.iadd_large(y);
x
}
#[inline]
fn isub_large(&mut self, y: &Self) {
large::isub(self.data_mut(), y.data());
}
#[inline]
fn sub_large(&mut self, y: &Self) -> Self {
let mut x = self.clone();
x.isub_large(y);
x
}
#[inline]
fn imul_large(&mut self, y: &Self) {
large::imul(self.data_mut(), y.data());
}
#[inline]
fn mul_large(&mut self, y: &Self) -> Self {
let mut x = self.clone();
x.imul_large(y);
x
}
#[inline]
fn idiv_large(&mut self, y: &Self) -> Self {
let mut rem = Self::default();
*rem.data_mut() = large::idiv(self.data_mut(), y.data());
rem
}
#[inline]
fn div_large(&mut self, y: &Self) -> (Self, Self) {
let mut x = self.clone();
let rem = x.idiv_large(y);
(x, rem)
}
#[inline]
fn quorem(&mut self, y: &Self) -> Limb {
large::quorem(self.data_mut(), y.data())
}
}
#[cfg(test)]
mod tests {
use util::test::*;
use super::*;
#[derive(Clone, Default)]
struct Bigint {
data: DataType,
}
impl Bigint {
#[inline]
pub fn new() -> Bigint {
Bigint { data: stackvec![] }
}
}
impl SharedOps for Bigint {
type StorageType = DataType;
#[inline]
fn data<'a>(&'a self) -> &'a Self::StorageType {
&self.data
}
#[inline]
fn data_mut<'a>(&'a mut self) -> &'a mut Self::StorageType {
&mut self.data
}
}
impl SmallOps for Bigint {
}
impl LargeOps for Bigint {
}
#[test]
fn greater_test() {
let x = Bigint { data: from_u32(&[1]) };
let y = Bigint { data: from_u32(&[2]) };
assert!(!x.greater(&y));
assert!(!x.greater(&x));
assert!(y.greater(&x));
let x = Bigint { data: from_u32(&[5, 1]) };
let y = Bigint { data: from_u32(&[2]) };
assert!(x.greater(&y));
assert!(!x.greater(&x));
assert!(!y.greater(&x));
let x = Bigint { data: from_u32(&[5, 1, 9]) };
let y = Bigint { data: from_u32(&[6, 2, 8]) };
assert!(x.greater(&y));
assert!(!x.greater(&x));
assert!(!y.greater(&x));
let x = Bigint { data: from_u32(&[0, 1, 9]) };
let y = Bigint { data: from_u32(&[4294967295, 0, 9]) };
assert!(x.greater(&y));
assert!(!x.greater(&x));
assert!(!y.greater(&x));
}
#[test]
fn greater_equal_test() {
let x = Bigint { data: from_u32(&[1]) };
let y = Bigint { data: from_u32(&[2]) };
assert!(!x.greater_equal(&y));
assert!(x.greater_equal(&x));
assert!(y.greater_equal(&x));
let x = Bigint { data: from_u32(&[5, 1]) };
let y = Bigint { data: from_u32(&[2]) };
assert!(x.greater_equal(&y));
assert!(x.greater_equal(&x));
assert!(!y.greater_equal(&x));
let x = Bigint { data: from_u32(&[5, 1, 9]) };
let y = Bigint { data: from_u32(&[6, 2, 8]) };
assert!(x.greater_equal(&y));
assert!(x.greater_equal(&x));
assert!(!y.greater_equal(&x));
let x = Bigint { data: from_u32(&[0, 1, 9]) };
let y = Bigint { data: from_u32(&[4294967295, 0, 9]) };
assert!(x.greater_equal(&y));
assert!(x.greater_equal(&x));
assert!(!y.greater_equal(&x));
}
#[test]
fn equal_test() {
let x = Bigint { data: from_u32(&[1]) };
let y = Bigint { data: from_u32(&[2]) };
assert!(!x.equal(&y));
assert!(x.equal(&x));
assert!(!y.equal(&x));
let x = Bigint { data: from_u32(&[5, 1]) };
let y = Bigint { data: from_u32(&[2]) };
assert!(!x.equal(&y));
assert!(x.equal(&x));
assert!(!y.equal(&x));
let x = Bigint { data: from_u32(&[5, 1, 9]) };
let y = Bigint { data: from_u32(&[6, 2, 8]) };
assert!(!x.equal(&y));
assert!(x.equal(&x));
assert!(!y.equal(&x));
let x = Bigint { data: from_u32(&[0, 1, 9]) };
let y = Bigint { data: from_u32(&[4294967295, 0, 9]) };
assert!(!x.equal(&y));
assert!(x.equal(&x));
assert!(!y.equal(&x));
}
#[test]
fn less_test() {
let x = Bigint { data: from_u32(&[1]) };
let y = Bigint { data: from_u32(&[2]) };
assert!(x.less(&y));
assert!(!x.less(&x));
assert!(!y.less(&x));
let x = Bigint { data: from_u32(&[5, 1]) };
let y = Bigint { data: from_u32(&[2]) };
assert!(!x.less(&y));
assert!(!x.less(&x));
assert!(y.less(&x));
let x = Bigint { data: from_u32(&[5, 1, 9]) };
let y = Bigint { data: from_u32(&[6, 2, 8]) };
assert!(!x.less(&y));
assert!(!x.less(&x));
assert!(y.less(&x));
let x = Bigint { data: from_u32(&[0, 1, 9]) };
let y = Bigint { data: from_u32(&[4294967295, 0, 9]) };
assert!(!x.less(&y));
assert!(!x.less(&x));
assert!(y.less(&x));
}
#[test]
fn less_equal_test() {
let x = Bigint { data: from_u32(&[1]) };
let y = Bigint { data: from_u32(&[2]) };
assert!(x.less_equal(&y));
assert!(x.less_equal(&x));
assert!(!y.less_equal(&x));
let x = Bigint { data: from_u32(&[5, 1]) };
let y = Bigint { data: from_u32(&[2]) };
assert!(!x.less_equal(&y));
assert!(x.less_equal(&x));
assert!(y.less_equal(&x));
let x = Bigint { data: from_u32(&[5, 1, 9]) };
let y = Bigint { data: from_u32(&[6, 2, 8]) };
assert!(!x.less_equal(&y));
assert!(x.less_equal(&x));
assert!(y.less_equal(&x));
let x = Bigint { data: from_u32(&[0, 1, 9]) };
let y = Bigint { data: from_u32(&[4294967295, 0, 9]) };
assert!(!x.less_equal(&y));
assert!(x.less_equal(&x));
assert!(y.less_equal(&x));
}
#[test]
fn leading_zero_limbs_test() {
assert_eq!(Bigint::new().leading_zero_limbs(), 0);
assert_eq!(Bigint::from_u16(0xF).leading_zero_limbs(), 0);
assert_eq!(Bigint::from_u32(0xFF).leading_zero_limbs(), 0);
assert_eq!(Bigint::from_u64(0xFF00000000).leading_zero_limbs(), 0);
assert_eq!(Bigint::from_u128(0xFF000000000000000000000000).leading_zero_limbs(), 0);
assert_eq!(Bigint::from_u16(0xF).leading_zero_limbs(), 0);
assert_eq!(Bigint::from_u32(0xF).leading_zero_limbs(), 0);
assert_eq!(Bigint::from_u64(0xF00000000).leading_zero_limbs(), 0);
assert_eq!(Bigint::from_u128(0xF000000000000000000000000).leading_zero_limbs(), 0);
assert_eq!(Bigint::from_u16(0xF0).leading_zero_limbs(), 0);
assert_eq!(Bigint::from_u32(0xF0).leading_zero_limbs(), 0);
assert_eq!(Bigint::from_u64(0xF000000000).leading_zero_limbs(), 0);
assert_eq!(Bigint::from_u128(0xF0000000000000000000000000).leading_zero_limbs(), 0);
}
#[test]
fn trailing_zero_limbs_test() {
assert_eq!(Bigint::new().trailing_zero_limbs(), 0);
assert_eq!(Bigint { data: stackvec![0xFF] }.trailing_zero_limbs(), 0);
assert_eq!(Bigint { data: stackvec![0, 0xFF000] }.trailing_zero_limbs(), 1);
assert_eq!(Bigint { data: stackvec![0, 0, 0, 0xFF000] }.trailing_zero_limbs(), 3);
}
#[test]
fn leading_zeros_test() {
assert_eq!(Bigint::new().leading_zeros(), 0);
assert_eq!(Bigint::from_u16(0xFF).leading_zeros(), Limb::BITS-8);
assert_eq!(Bigint::from_u32(0xFF).leading_zeros(), Limb::BITS-8);
assert_eq!(Bigint::from_u64(0xFF00000000).leading_zeros(), 24);
assert_eq!(Bigint::from_u128(0xFF000000000000000000000000).leading_zeros(), 24);
assert_eq!(Bigint::from_u16(0xF).leading_zeros(), Limb::BITS-4);
assert_eq!(Bigint::from_u32(0xF).leading_zeros(), Limb::BITS-4);
assert_eq!(Bigint::from_u64(0xF00000000).leading_zeros(), 28);
assert_eq!(Bigint::from_u128(0xF000000000000000000000000).leading_zeros(), 28);
assert_eq!(Bigint::from_u16(0xF0).leading_zeros(), Limb::BITS-8);
assert_eq!(Bigint::from_u32(0xF0).leading_zeros(), Limb::BITS-8);
assert_eq!(Bigint::from_u64(0xF000000000).leading_zeros(), 24);
assert_eq!(Bigint::from_u128(0xF0000000000000000000000000).leading_zeros(), 24);
}
#[test]
fn trailing_zeros_test() {
assert_eq!(Bigint::new().trailing_zeros(), 0);
assert_eq!(Bigint::from_u16(0xFF).trailing_zeros(), 0);
assert_eq!(Bigint::from_u32(0xFF).trailing_zeros(), 0);
assert_eq!(Bigint::from_u64(0xFF00000000).trailing_zeros(), 32);
assert_eq!(Bigint::from_u128(0xFF000000000000000000000000).trailing_zeros(), 96);
assert_eq!(Bigint::from_u16(0xF).trailing_zeros(), 0);
assert_eq!(Bigint::from_u32(0xF).trailing_zeros(), 0);
assert_eq!(Bigint::from_u64(0xF00000000).trailing_zeros(), 32);
assert_eq!(Bigint::from_u128(0xF000000000000000000000000).trailing_zeros(), 96);
assert_eq!(Bigint::from_u16(0xF0).trailing_zeros(), 4);
assert_eq!(Bigint::from_u32(0xF0).trailing_zeros(), 4);
assert_eq!(Bigint::from_u64(0xF000000000).trailing_zeros(), 36);
assert_eq!(Bigint::from_u128(0xF0000000000000000000000000).trailing_zeros(), 100);
}
#[test]
fn hi32_test() {
assert_eq!(Bigint::from_u16(0xA).hi32(), (0xA0000000, false));
assert_eq!(Bigint::from_u32(0xAB).hi32(), (0xAB000000, false));
assert_eq!(Bigint::from_u64(0xAB00000000).hi32(), (0xAB000000, false));
assert_eq!(Bigint::from_u64(0xA23456789A).hi32(), (0xA2345678, true));
}
#[test]
fn hi64_test() {
assert_eq!(Bigint::from_u16(0xA).hi64(), (0xA000000000000000, false));
assert_eq!(Bigint::from_u32(0xAB).hi64(), (0xAB00000000000000, false));
assert_eq!(Bigint::from_u64(0xAB00000000).hi64(), (0xAB00000000000000, false));
assert_eq!(Bigint::from_u64(0xA23456789A).hi64(), (0xA23456789A000000, false));
assert_eq!(Bigint::from_u128(0xABCDEF0123456789ABCDEF0123).hi64(), (0xABCDEF0123456789, true));
}
#[test]
fn hi128_test() {
assert_eq!(Bigint::from_u128(0xABCDEF0123456789ABCDEF0123).hi128(), (0xABCDEF0123456789ABCDEF0123000000, false));
assert_eq!(Bigint::from_u128(0xABCDEF0123456789ABCDEF0123456789).hi128(), (0xABCDEF0123456789ABCDEF0123456789, false));
assert_eq!(Bigint { data: from_u32(&[0x34567890, 0xBCDEF012, 0x3456789A, 0xBCDEF012, 0xA]) }.hi128(), (0xABCDEF0123456789ABCDEF0123456789, false));
assert_eq!(Bigint { data: from_u32(&[0x34567891, 0xBCDEF012, 0x3456789A, 0xBCDEF012, 0xA]) }.hi128(), (0xABCDEF0123456789ABCDEF0123456789, true));
}
#[test]
fn pad_zero_digits_test() {
let mut x = Bigint { data: stackvec![0, 0, 0, 1] };
x.pad_zero_digits(3);
assert_eq!(x.data.as_slice(), &[0, 0, 0, 0, 0, 0, 1]);
let mut x = Bigint { data: stackvec![1] };
x.pad_zero_digits(1);
assert_eq!(x.data.as_slice(), &[0, 1]);
}
#[test]
fn shl_test() {
let mut big = Bigint { data: from_u32(&[0xD2210408]) };
big.ishl(5);
assert_eq!(big.data, from_u32(&[0x44208100, 0x1A]));
big.ishl(32);
assert_eq!(big.data, from_u32(&[0, 0x44208100, 0x1A]));
big.ishl(27);
assert_eq!(big.data, from_u32(&[0, 0, 0xD2210408]));
let mut big = Bigint { data: from_u32(&[0x20020010, 0x8040100, 0xD2210408]) };
big.ishl(5);
assert_eq!(big.data, from_u32(&[0x400200, 0x802004, 0x44208101, 0x1A]));
big.ishl(32);
assert_eq!(big.data, from_u32(&[0, 0x400200, 0x802004, 0x44208101, 0x1A]));
big.ishl(27);
assert_eq!(big.data, from_u32(&[0, 0, 0x20020010, 0x8040100, 0xD2210408]));
}
#[test]
fn shr_test() {
let mut big = Bigint { data: from_u32(&[0xD2210408]) };
big.ishr(5, false);
assert_eq!(big.data, from_u32(&[0x6910820]));
big.ishr(27, false);
assert_eq!(big.data, from_u32(&[]));
let mut big = Bigint { data: from_u32(&[0x20020010, 0x8040100, 0xD2210408]) };
big.ishr(5, false);
assert_eq!(big.data, from_u32(&[0x1001000, 0x40402008, 0x6910820]));
big.ishr(32, false);
assert_eq!(big.data, from_u32(&[0x40402008, 0x6910820]));
big.ishr(27, false);
assert_eq!(big.data, from_u32(&[0xD2210408]));
let mut big = Bigint { data: from_u32(&[0xD2210408]) };
big.ishr(3, true);
assert_eq!(big.data, from_u32(&[0x1A442081]));
big.ishr(1, true);
assert_eq!(big.data, from_u32(&[0xD221040]));
let mut big = Bigint { data: from_u32(&[0xD2210408]) };
big.ishr(4, true);
assert_eq!(big.data, from_u32(&[0xD221040]));
let mut big = Bigint { data: from_u32(&[0xD2210438]) };
big.ishr(3, true);
assert_eq!(big.data, from_u32(&[0x1A442087]));
big.ishr(1, true);
assert_eq!(big.data, from_u32(&[0xD221044]));
let mut big = Bigint { data: from_u32(&[0xD2210438]) };
big.ishr(5, true);
assert_eq!(big.data, from_u32(&[0x6910822]));
}
#[test]
fn bit_length_test() {
let x = Bigint { data: from_u32(&[0, 0, 0, 1]) };
assert_eq!(x.bit_length(), 97);
let x = Bigint { data: from_u32(&[0, 0, 0, 3]) };
assert_eq!(x.bit_length(), 98);
let x = Bigint { data: from_u32(&[1<<31]) };
assert_eq!(x.bit_length(), 32);
}
#[test]
fn iadd_small_test() {
let mut x = Bigint { data: from_u32(&[4294967295]) };
x.iadd_small(5);
assert_eq!(x.data, from_u32(&[4, 1]));
let mut x = Bigint { data: from_u32(&[5]) };
x.iadd_small(7);
assert_eq!(x.data, from_u32(&[12]));
let mut x = Bigint::from_u64(0x80000000FFFFFFFF);
x.iadd_small(7);
assert_eq!(x.data, from_u32(&[6, 0x80000001]));
let mut x = Bigint::from_u64(0xFFFFFFFFFFFFFFFF);
x.iadd_small(7);
assert_eq!(x.data, from_u32(&[6, 0, 1]));
}
#[test]
fn isub_small_test() {
let mut x = Bigint { data: from_u32(&[4, 1]) };
x.isub_small(5);
assert_eq!(x.data, from_u32(&[4294967295]));
let mut x = Bigint { data: from_u32(&[12]) };
x.isub_small(7);
assert_eq!(x.data, from_u32(&[5]));
let mut x = Bigint { data: from_u32(&[6, 0x80000001]) };
x.isub_small(7);
assert_eq!(x.data, from_u32(&[0xFFFFFFFF, 0x80000000]));
let mut x = Bigint { data: from_u32(&[6, 0, 1]) };
x.isub_small(7);
assert_eq!(x.data, from_u32(&[0xFFFFFFFF, 0xFFFFFFFF]));
}
#[test]
fn imul_small_test() {
let mut x = Bigint { data: from_u32(&[5]) };
x.imul_small(7);
assert_eq!(x.data, from_u32(&[35]));
let mut x = Bigint::from_u64(0x4000000040000);
x.imul_small(5);
assert_eq!(x.data, from_u32(&[0x00140000, 0x140000]));
let mut x = Bigint { data: from_u32(&[0x33333334]) };
x.imul_small(5);
assert_eq!(x.data, from_u32(&[4, 1]));
let mut x = Bigint::from_u64(0x133333334);
x.imul_small(5);
assert_eq!(x.data, from_u32(&[4, 6]));
let mut x = Bigint::from_u64(0x3333333333333334);
x.imul_small(5);
assert_eq!(x.data, from_u32(&[4, 0, 1]));
}
#[test]
fn idiv_small_test() {
let mut x = Bigint { data: from_u32(&[4]) };
assert_eq!(x.idiv_small(7), 4);
assert_eq!(x.data, from_u32(&[]));
let mut x = Bigint { data: from_u32(&[3]) };
assert_eq!(x.idiv_small(7), 3);
assert_eq!(x.data, from_u32(&[]));
let mut x = Bigint { data: from_u32(&[15]) };
assert_eq!(x.idiv_small(10), 5);
assert_eq!(x.data, from_u32(&[1]));
let mut x = Bigint::from_u64(0x133333334);
assert_eq!(x.idiv_small(5), 1);
assert_eq!(x.data, from_u32(&[0x3D70A3D7]));
let mut x = Bigint::from_u64(0x3333333333333334);
assert_eq!(x.idiv_small(5), 4);
assert_eq!(x.data, from_u32(&[0xD70A3D70, 0xA3D70A3]));
}
#[test]
fn ipow_test() {
let x = Bigint { data: from_u32(&[5]) };
assert_eq!(x.pow(2).data, from_u32(&[25]));
assert_eq!(x.pow(15).data, from_u32(&[452807053, 7]));
assert_eq!(x.pow(16).data, from_u32(&[2264035265, 35]));
assert_eq!(x.pow(17).data, from_u32(&[2730241733, 177]));
assert_eq!(x.pow(302).data, from_u32(&[2443090281, 2149694430, 2297493928, 1584384001, 1279504719, 1930002239, 3312868939, 3735173465, 3523274756, 2025818732, 1641675015, 2431239749, 4292780461, 3719612855, 4174476133, 3296847770, 2677357556, 638848153, 2198928114, 3285049351, 2159526706, 626302612]));
}
#[test]
fn iadd_large_test() {
let mut x = Bigint { data: from_u32(&[4294967295]) };
let y = Bigint { data: from_u32(&[5]) };
x.iadd_large(&y);
assert_eq!(x.data, from_u32(&[4, 1]));
let mut x = Bigint { data: from_u32(&[5]) };
let y = Bigint { data: from_u32(&[7]) };
x.iadd_large(&y);
assert_eq!(x.data, from_u32(&[12]));
let mut x = Bigint::from_u64(0x80000000FFFFFFFF);
let y = Bigint { data: from_u32(&[7]) };
x.iadd_large(&y);
assert_eq!(x.data, from_u32(&[6, 0x80000001]));
let mut x = Bigint::from_u64(0x7FFFFFFFFFFFFFFF);
let y = Bigint::from_u64(0x7FFFFFFFFFFFFFFF);
x.iadd_large(&y);
assert_eq!(x.data, from_u32(&[0xFFFFFFFE, 0xFFFFFFFF]));
let mut x = Bigint::from_u64(0x8FFFFFFFFFFFFFFF);
let y = Bigint::from_u64(0x7FFFFFFFFFFFFFFF);
x.iadd_large(&y);
assert_eq!(x.data, from_u32(&[0xFFFFFFFE, 0x0FFFFFFF, 1]));
}
#[test]
fn isub_large_test() {
let mut x = Bigint { data: from_u32(&[4, 1]) };
let y = Bigint { data: from_u32(&[5]) };
x.isub_large(&y);
assert_eq!(x.data, from_u32(&[4294967295]));
let mut x = Bigint { data: from_u32(&[12]) };
let y = Bigint { data: from_u32(&[7]) };
x.isub_large(&y);
assert_eq!(x.data, from_u32(&[5]));
let mut x = Bigint { data: from_u32(&[6, 0x80000001]) };
let y = Bigint { data: from_u32(&[7]) };
x.isub_large(&y);
assert_eq!(x.data, from_u32(&[0xFFFFFFFF, 0x80000000]));
let mut x = Bigint { data: from_u32(&[0xFFFFFFFF, 0x7FFFFFFF]) };
let y = Bigint { data: from_u32(&[0xFFFFFFFF, 0x7FFFFFFF]) };
x.isub_large(&y);
assert_eq!(x.data, from_u32(&[]));
let mut x = Bigint { data: from_u32(&[0xFFFFFFFE, 0x80000000]) };
let y = Bigint { data: from_u32(&[0xFFFFFFFF, 0x7FFFFFFF]) };
x.isub_large(&y);
assert_eq!(x.data, from_u32(&[0xFFFFFFFF]));
}
#[test]
fn imul_large_test() {
let mut x = Bigint { data: from_u32(&[0xFFFFFFFF]) };
let y = Bigint { data: from_u32(&[]) };
x.imul_large(&y);
assert_eq!(x.data, from_u32(&[]));
let mut x = Bigint { data: from_u32(&[0xFFFFFFFF]) };
let y = Bigint { data: from_u32(&[5]) };
x.imul_large(&y);
assert_eq!(x.data, from_u32(&[0xFFFFFFFB, 0x4]));
let mut x = Bigint { data: from_u32(&[0xFFFFFFFF]) };
let y = Bigint { data: from_u32(&[0xFFFFFFFE]) };
x.imul_large(&y);
assert_eq!(x.data, from_u32(&[0x2, 0xFFFFFFFD]));
let mut x = Bigint { data: from_u32(&[0xFFFFFFFE, 0x0FFFFFFF, 1]) };
let y = Bigint { data: from_u32(&[0x99999999, 0x99999999, 0xCCCD9999, 0xCCCC]) };
x.imul_large(&y);
assert_eq!(x.data, from_u32(&[0xCCCCCCCE, 0x5CCCCCCC, 0x9997FFFF, 0x33319999, 0x999A7333, 0xD999]));
}
#[test]
fn imul_karatsuba_mul_test() {
let mut x = Bigint { data: from_u32(&[1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16]) };
let y = Bigint { data: from_u32(&[4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19]) };
x.imul_large(&y);
assert_eq!(x.data, from_u32(&[4, 13, 28, 50, 80, 119, 168, 228, 300, 385, 484, 598, 728, 875, 1040, 1224, 1340, 1435, 1508, 1558, 1584, 1585, 1560, 1508, 1428, 1319, 1180, 1010, 808, 573, 304]));
let mut x = Bigint { data: from_u32(&[1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16]) };
let y = Bigint { data: from_u32(&[4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 26, 27, 28, 29, 30, 31, 32, 33, 34, 35, 36, 37]) };
x.imul_large(&y);
assert_eq!(x.data, from_u32(&[4, 13, 28, 50, 80, 119, 168, 228, 300, 385, 484, 598, 728, 875, 1040, 1224, 1360, 1496, 1632, 1768, 1904, 2040, 2176, 2312, 2448, 2584, 2720, 2856, 2992, 3128, 3264, 3400, 3536, 3672, 3770, 3829, 3848, 3826, 3762, 3655, 3504, 3308, 3066, 2777, 2440, 2054, 1618, 1131, 592]));
}
#[test]
fn idiv_large_test() {
let mut x = Bigint { data: from_u32(&[0xFFFFFFFF]) };
let y = Bigint { data: from_u32(&[5]) };
let rem = x.idiv_large(&y);
assert_eq!(x.data, from_u32(&[0x33333333]));
assert_eq!(rem.data, from_u32(&[0]));
let mut x = Bigint { data: from_u32(&[0x2, 0xFFFFFFFF]) };
let y = Bigint { data: from_u32(&[0xFFFFFFFE]) };
let rem = x.idiv_large(&y);
assert_eq!(x.data, from_u32(&[1, 1]));
assert_eq!(rem.data, from_u32(&[4]));
let mut x = Bigint { data: from_u32(&[0xCCCCCCCF, 0x5CCCCCCC, 0x9997FFFF, 0x33319999, 0x999A7333, 0xD999]) };
let y = Bigint { data: from_u32(&[0x99999999, 0x99999999, 0xCCCD9999, 0xCCCC]) };
let rem = x.idiv_large(&y);
assert_eq!(x.data, from_u32(&[0xFFFFFFFE, 0x0FFFFFFF, 1]));
assert_eq!(rem.data, from_u32(&[1]));
let mut x = Bigint { data: from_u32(&[4, 13, 29, 50, 80, 119, 168, 228, 300, 385, 484, 598, 728, 875, 1040, 1224, 1340, 1435, 1508, 1558, 1584, 1585, 1560, 1508, 1428, 1319, 1180, 1010, 808, 573, 304]) };
let y = Bigint { data: from_u32(&[4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19]) };
let rem = x.idiv_large(&y);
assert_eq!(x.data, from_u32(&[1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16]));
assert_eq!(rem.data, from_u32(&[0, 0, 1]));
}
#[test]
fn quorem_test() {
let mut x = Bigint::from_u128(42535295865117307932921825928971026432);
let y = Bigint::from_u128(17218479456385750618067377696052635483);
assert_eq!(x.quorem(&y), 2);
assert_eq!(x.data, from_u32(&[1873752394, 3049207402, 3024501058, 102215382]));
}
}