use crate::algorithms::fft::cooley_tuckey::CooleyTuckeyButterfly;
use crate::delegate::DelegateRing;
use crate::impl_eq_based_self_iso;
use crate::impl_wrap_unwrap_homs;
use crate::impl_wrap_unwrap_isos;
use crate::ordered::OrderedRingStore;
use crate::primitive_int::*;
use crate::integer::*;
use crate::pid::*;
use crate::ring::*;
use crate::homomorphism::*;
use crate::rings::rust_bigint::*;
use super::*;
use super::zn_barett;
fn high(x: u128) -> u64 {
(x >> 64) as u64
}
fn low(x: u128) -> u64 {
(x & ((1 << 64) - 1)) as u64
}
fn mulhi(lhs: u64, rhs: u64) -> u64 {
high(lhs as u128 * rhs as u128)
}
fn mullo(lhs: u64, rhs: u64) -> u64 {
lhs.wrapping_mul(rhs)
}
#[derive(Clone, Copy)]
pub struct ZnBase {
modulus: i64,
modulus_times_three: u64,
inv_modulus: u128
}
const ZZ: StaticRing<i64> = StaticRing::<i64>::RING;
#[allow(non_upper_case_globals)]
const ZZbig: BigIntRing = BigIntRing::RING;
impl ZnBase {
pub fn new(modulus: u64) -> Self {
assert!(modulus > 1);
assert!(modulus <= ((1 << 62) / 9));
assert!(modulus as u128 * 6 <= u64::MAX as u128);
let inv_modulus = ZZbig.euclidean_div(ZZbig.power_of_two(128), &ZZbig.coerce(&ZZ, modulus as i64));
debug_assert!(ZZbig.is_lt(&ZZbig.mul_ref_fst(&inv_modulus, ZZbig.pow(ZZbig.int_hom().mul_map(ZZbig.coerce(&ZZ, modulus as i64), 6), 2)), &ZZbig.power_of_two(192)));
let inv_modulus = if ZZbig.eq_el(&inv_modulus, &ZZbig.power_of_two(127)) {
1u128 << 127
} else {
int_cast(inv_modulus, &StaticRing::<i128>::RING, &ZZbig) as u128
};
Self {
modulus: modulus as i64,
inv_modulus: inv_modulus,
modulus_times_three: modulus * 3
}
}
fn modulus_u64(&self) -> u64 {
self.modulus as u64
}
fn repr_bound(&self) -> u64 {
self.modulus_u64() * 6
}
fn bounded_reduce(&self, value: u128) -> u64 {
debug_assert!(value <= self.repr_bound() as u128 * self.repr_bound() as u128);
let (in_low, in_high) = (low(value), high(value));
let (invmod_low, invmod_high) = (low(self.inv_modulus), high(self.inv_modulus));
let approx_quotient = mulhi(in_low, invmod_high) + mulhi(in_high, invmod_low) + mullo(in_high, invmod_high);
let result = low(value).wrapping_sub(mullo(approx_quotient, self.modulus_u64()));
debug_assert!(result < self.modulus_times_three);
debug_assert!((value - result as u128) % (self.modulus_u64() as u128) == 0);
return result;
}
fn potential_reduce(&self, mut value: u64) -> u64 {
if value >= self.repr_bound() {
value -= self.repr_bound();
}
if value >= self.modulus_times_three {
value -= self.modulus_times_three;
}
return value;
}
pub fn promise_is_reduced(&self, value: u64) -> ZnEl {
debug_assert!(value <= self.repr_bound());
ZnEl(value)
}
fn complete_reduce(&self, mut value: u64) -> u64 {
debug_assert!(value <= self.repr_bound());
if value >= 4 * self.modulus_u64() {
value -= 4 * self.modulus_u64();
}
if value >= 2 * self.modulus_u64() {
value -= 2 * self.modulus_u64();
}
if value >= self.modulus_u64() {
value -= self.modulus_u64();
}
debug_assert!(value < self.modulus_u64());
return value;
}
}
pub type Zn = RingValue<ZnBase>;
impl Zn {
pub fn new(modulus: u64) -> Self {
RingValue::from(ZnBase::new(modulus))
}
}
#[derive(Clone, Copy)]
pub struct ZnEl(u64);
impl PartialEq for ZnBase {
fn eq(&self, other: &Self) -> bool {
self.modulus == other.modulus
}
}
impl RingBase for ZnBase {
type Element = ZnEl;
fn clone_el(&self, val: &Self::Element) -> Self::Element {
*val
}
fn add_assign(&self, lhs: &mut Self::Element, rhs: Self::Element) {
debug_assert!(lhs.0 <= self.repr_bound());
debug_assert!(rhs.0 <= self.repr_bound());
lhs.0 = self.potential_reduce(lhs.0 + rhs.0);
}
fn negate_inplace(&self, lhs: &mut Self::Element) {
debug_assert!(lhs.0 <= self.repr_bound());
lhs.0 = self.repr_bound() - lhs.0;
}
fn mul_assign(&self, lhs: &mut Self::Element, rhs: Self::Element) {
debug_assert!(lhs.0 <= self.repr_bound());
debug_assert!(rhs.0 <= self.repr_bound());
lhs.0 = self.bounded_reduce(lhs.0 as u128 * rhs.0 as u128);
}
fn from_int(&self, value: i32) -> Self::Element {
RingRef::new(self).coerce(&StaticRing::<i32>::RING, value)
}
fn eq_el(&self, lhs: &Self::Element, rhs: &Self::Element) -> bool {
debug_assert!(lhs.0 <= self.repr_bound());
debug_assert!(rhs.0 <= self.repr_bound());
if lhs.0 > rhs.0 {
self.is_zero(&self.promise_is_reduced(lhs.0 - rhs.0))
} else {
self.is_zero(&self.promise_is_reduced(rhs.0 - lhs.0))
}
}
fn is_zero(&self, val: &Self::Element) -> bool {
debug_assert!(val.0 <= self.repr_bound());
self.complete_reduce(val.0) == 0
}
fn is_commutative(&self) -> bool { true }
fn is_noetherian(&self) -> bool { true }
fn dbg<'a>(&self, value: &Self::Element, out: &mut std::fmt::Formatter<'a>) -> std::fmt::Result {
write!(out, "{}", self.complete_reduce(value.0))
}
fn characteristic<I: IntegerRingStore>(&self, other_ZZ: &I) -> Option<El<I>>
where I::Type: IntegerRing
{
self.size(other_ZZ)
}
fn sum<I>(&self, mut els: I) -> Self::Element
where I: Iterator<Item = Self::Element>
{
let mut result = self.zero();
while let Some(ZnEl(start)) = els.next() {
let mut current = start as u128;
for ZnEl(c) in els.by_ref().take(self.repr_bound() as usize / 2 - 1) {
current += c as u128;
}
self.add_assign(&mut result, self.promise_is_reduced(self.bounded_reduce(current)));
}
debug_assert!(result.0 <= self.repr_bound());
return result;
}
}
impl_eq_based_self_iso!{ ZnBase }
impl<I: IntegerRingStore> CanHomFrom<zn_barett::ZnBase<I>> for ZnBase
where I::Type: IntegerRing
{
type Homomorphism = ();
fn has_canonical_hom(&self, from: &zn_barett::ZnBase<I>) -> Option<Self::Homomorphism> {
if from.integer_ring().get_ring().representable_bits().is_none() || from.integer_ring().get_ring().representable_bits().unwrap() >= self.integer_ring().abs_log2_ceil(self.modulus()).unwrap() {
if from.integer_ring().eq_el(from.modulus(), &int_cast(*self.modulus(), from.integer_ring(), self.integer_ring())) {
Some(())
} else {
None
}
} else {
None
}
}
fn map_in(&self, from: &zn_barett::ZnBase<I>, el: <zn_barett::ZnBase<I> as RingBase>::Element, _: &Self::Homomorphism) -> Self::Element {
self.promise_is_reduced(int_cast(from.smallest_positive_lift(el), self.integer_ring(), from.integer_ring()) as u64)
}
}
impl<I: IntegerRingStore> CanIsoFromTo<zn_barett::ZnBase<I>> for ZnBase
where I::Type: IntegerRing
{
type Isomorphism = <zn_barett::ZnBase<I> as CanHomFrom<StaticRingBase<i64>>>::Homomorphism;
fn has_canonical_iso(&self, from: &zn_barett::ZnBase<I>) -> Option<Self::Isomorphism> {
if from.integer_ring().get_ring().representable_bits().is_none() || from.integer_ring().get_ring().representable_bits().unwrap() >= self.integer_ring().abs_log2_ceil(self.modulus()).unwrap() {
if from.integer_ring().eq_el(from.modulus(), &int_cast(*self.modulus(), from.integer_ring(), self.integer_ring())) {
from.has_canonical_hom(self.integer_ring().get_ring())
} else {
None
}
} else {
None
}
}
fn map_out(&self, from: &zn_barett::ZnBase<I>, el: <Self as RingBase>::Element, iso: &Self::Isomorphism) -> <zn_barett::ZnBase<I> as RingBase>::Element {
from.map_in(self.integer_ring().get_ring(), el.0 as i64, iso)
}
}
#[allow(deprecated)]
impl CanHomFrom<zn_42::ZnBase> for ZnBase {
type Homomorphism = ();
fn has_canonical_hom(&self, from: &zn_42::ZnBase) -> Option<Self::Homomorphism> {
if *self.modulus() == *from.modulus() {
Some(())
} else {
None
}
}
fn map_in(&self, from: &zn_42::ZnBase, el: <zn_42::ZnBase as RingBase>::Element, _: &Self::Homomorphism) -> Self::Element {
self.promise_is_reduced(from.smallest_positive_lift(el) as u64)
}
}
#[allow(deprecated)]pub enum ToZn42Iso {
Trivial, ReduceRequired(<zn_42::ZnBase as CanHomFrom<StaticRingBase<i64>>>::Homomorphism)
}
#[allow(deprecated)]
impl CanIsoFromTo<zn_42::ZnBase> for ZnBase {
type Isomorphism = ToZn42Iso;
fn has_canonical_iso(&self, from: &zn_42::ZnBase) -> Option<Self::Isomorphism> {
if *self.modulus() == *from.modulus() {
if from.repr_bound() >= self.repr_bound() {
Some(ToZn42Iso::Trivial)
} else {
Some(ToZn42Iso::ReduceRequired(from.has_canonical_hom(self.integer_ring().get_ring())?))
}
} else {
None
}
}
fn map_out(&self, from: &zn_42::ZnBase, el: <Self as RingBase>::Element, iso: &Self::Isomorphism) -> <zn_42::ZnBase as RingBase>::Element {
match iso {
ToZn42Iso::Trivial => from.from_bounded(el.0),
ToZn42Iso::ReduceRequired(reduce_hom) => from.map_in(self.integer_ring().get_ring(), el.0 as i64, reduce_hom)
}
}
}
impl DivisibilityRing for ZnBase {
fn checked_left_div(&self, lhs: &Self::Element, rhs: &Self::Element) -> Option<Self::Element> {
super::generic_impls::checked_left_div(RingRef::new(self), lhs, rhs, self.modulus())
}
}
trait ImplGenericIntHomomorphismMarker: IntegerRing + CanIsoFromTo<StaticRingBase<i128>> + CanIsoFromTo<StaticRingBase<i64>> {}
impl ImplGenericIntHomomorphismMarker for RustBigintRingBase {}
#[cfg(feature = "mpir")]
impl ImplGenericIntHomomorphismMarker for crate::rings::mpir::MPZBase {}
impl<I: ?Sized + ImplGenericIntHomomorphismMarker> CanHomFrom<I> for ZnBase {
type Homomorphism = super::generic_impls::IntegerToZnHom<I, StaticRingBase<i128>, Self>;
fn has_canonical_hom(&self, from: &I) -> Option<Self::Homomorphism> {
super::generic_impls::has_canonical_hom_from_int(from, self, StaticRing::<i128>::RING.get_ring(), Some(&(self.repr_bound() as i128 * self.repr_bound() as i128)))
}
fn map_in(&self, from: &I, el: I::Element, hom: &Self::Homomorphism) -> Self::Element {
super::generic_impls::map_in_from_int(from, self, StaticRing::<i128>::RING.get_ring(), el, hom, |n| {
debug_assert!((n as u64) < self.modulus_u64());
self.promise_is_reduced(n as u64)
}, |n| {
debug_assert!(n <= (self.repr_bound() as i128 * self.repr_bound() as i128));
self.promise_is_reduced(self.bounded_reduce(n as u128))
})
}
}
macro_rules! impl_static_int_to_zn {
($($int:ident),*) => {
$(
impl CanHomFrom<StaticRingBase<$int>> for ZnBase {
type Homomorphism = ($int, $int);
fn has_canonical_hom(&self, _from: &StaticRingBase<$int>) -> Option<Self::Homomorphism> {
let bounded_reduce_bound = self.repr_bound() as i128 * self.repr_bound() as i128;
Some((self.repr_bound() as $int, if bounded_reduce_bound > $int::MAX as i128 { $int::MAX } else { bounded_reduce_bound as $int }))
}
fn map_in(&self, _from: &StaticRingBase<$int>, el: $int, hom: &($int, $int)) -> Self::Element {
if el.abs() <= hom.0 {
if el < 0 {
self.negate(self.promise_is_reduced(-(el as i128) as u64))
} else {
self.promise_is_reduced(el as u64)
}
} else if el.abs() <= hom.1 {
if el < 0 {
self.negate(self.promise_is_reduced(self.bounded_reduce(-(el as i128) as u128)))
} else {
self.promise_is_reduced(self.bounded_reduce(el as u128))
}
} else {
if el < 0 {
self.promise_is_reduced(((el as i128 % self.modulus as i128) as i64 + self.modulus) as u64)
} else {
self.promise_is_reduced((el as i128 % self.modulus as i128) as u64)
}
}
}
}
)*
};
}
impl_static_int_to_zn!{ i8, i16, i32, i64, i128 }
#[derive(Clone, Copy)]
pub struct ZnBaseElementsIter<'a> {
ring: &'a ZnBase,
current: u64
}
impl<'a> Iterator for ZnBaseElementsIter<'a> {
type Item = ZnEl;
fn next(&mut self) -> Option<Self::Item> {
if self.current < self.ring.modulus_u64() {
let result = self.current;
self.current += 1;
return Some(self.ring.promise_is_reduced(result));
} else {
return None;
}
}
}
impl FiniteRing for ZnBase {
type ElementsIter<'a> = ZnBaseElementsIter<'a>;
fn elements<'a>(&'a self) -> Self::ElementsIter<'a> {
ZnBaseElementsIter {
ring: self,
current: 0
}
}
fn random_element<G: FnMut() -> u64>(&self, rng: G) -> <Self as RingBase>::Element {
super::generic_impls::random_element(self, rng)
}
fn size<I: IntegerRingStore>(&self, other_ZZ: &I) -> Option<El<I>>
where I::Type: IntegerRing
{
if other_ZZ.get_ring().representable_bits().is_none() || self.integer_ring().abs_log2_ceil(&(self.modulus() + 1)) <= other_ZZ.get_ring().representable_bits() {
Some(int_cast(*self.modulus(), other_ZZ, self.integer_ring()))
} else {
None
}
}
}
impl PrincipalIdealRing for ZnBase {
fn extended_ideal_gen(&self, lhs: &Self::Element, rhs: &Self::Element) -> (Self::Element, Self::Element, Self::Element) {
let (s, t, d) = StaticRing::<i64>::RING.extended_ideal_gen(&(lhs.0 as i64), &(rhs.0 as i64));
let quo = RingRef::new(self).into_can_hom(StaticRing::<i64>::RING).ok().unwrap();
(quo.map(s), quo.map(t), quo.map(d))
}
}
impl ZnRing for ZnBase {
type IntegerRingBase = StaticRingBase<i64>;
type Integers = StaticRing<i64>;
fn integer_ring(&self) -> &Self::Integers {
&StaticRing::<i64>::RING
}
fn smallest_positive_lift(&self, el: Self::Element) -> El<Self::Integers> {
self.complete_reduce(el.0) as i64
}
fn modulus(&self) -> &El<Self::Integers> {
&self.modulus
}
}
impl HashableElRing for ZnBase {
fn hash<H: std::hash::Hasher>(&self, el: &Self::Element, h: &mut H) {
self.integer_ring().hash(&self.smallest_positive_lift(*el), h)
}
}
#[derive(Clone, Copy)]
pub struct ZnFastmulBase {
base: RingValue<ZnBase>
}
pub type ZnFastmul = RingValue<ZnFastmulBase>;
impl ZnFastmul {
pub fn new(base: Zn) -> Self {
RingValue::from(ZnFastmulBase { base })
}
}
impl PartialEq for ZnFastmulBase {
fn eq(&self, other: &Self) -> bool {
self.base.get_ring() == other.base.get_ring()
}
}
impl PrincipalIdealRing for ZnFastmulBase {
fn extended_ideal_gen(&self, lhs: &Self::Element, rhs: &Self::Element) -> (Self::Element, Self::Element, Self::Element) {
let (s, t, d) = self.get_delegate().extended_ideal_gen(self.delegate_ref(lhs), self.delegate_ref(rhs));
(self.rev_delegate(s), self.rev_delegate(t), self.rev_delegate(d))
}
}
impl_eq_based_self_iso!{ ZnFastmulBase }
#[derive(Clone, Copy)]
pub struct ZnFastmulEl {
el: ZnEl,
value_invmod_shifted: u128
}
impl DelegateRing for ZnFastmulBase {
type Base = ZnBase;
type Element = ZnFastmulEl;
fn get_delegate(&self) -> &Self::Base {
self.base.get_ring()
}
fn delegate(&self, el: Self::Element) -> <Self::Base as RingBase>::Element {
el.el
}
fn delegate_mut<'a>(&self, el: &'a mut Self::Element) -> &'a mut <Self::Base as RingBase>::Element {
&mut el.el
}
fn delegate_ref<'a>(&self, el: &'a Self::Element) -> &'a <Self::Base as RingBase>::Element {
&el.el
}
fn postprocess_delegate_mut(&self, el: &mut Self::Element) {
assert!(el.el.0 <= self.base.get_ring().repr_bound());
let value = el.el.0;
el.value_invmod_shifted = ((value as u128) << 64) / self.base.get_ring().modulus_u64() as u128;
}
fn rev_delegate(&self, el: <Self::Base as RingBase>::Element) -> Self::Element {
let mut result = ZnFastmulEl {
el: el,
value_invmod_shifted: 0
};
self.postprocess_delegate_mut(&mut result);
return result;
}
}
impl CanHomFrom<ZnBase> for ZnFastmulBase {
type Homomorphism = ();
fn has_canonical_hom(&self, from: &ZnBase) -> Option<Self::Homomorphism> {
if from == self.base.get_ring() {
Some(())
} else {
None
}
}
fn map_in(&self, _from: &ZnBase, el: <ZnBase as RingBase>::Element, _hom: &Self::Homomorphism) -> Self::Element {
self.rev_delegate(el)
}
}
impl CanHomFrom<ZnFastmulBase> for ZnBase {
type Homomorphism = ();
fn has_canonical_hom(&self, from: &ZnFastmulBase) -> Option<Self::Homomorphism> {
if self == from.base.get_ring() {
Some(())
} else {
None
}
}
fn map_in(&self, _from: &ZnFastmulBase, el: <ZnFastmulBase as RingBase>::Element, _hom: &Self::Homomorphism) -> Self::Element {
el.el
}
fn mul_assign_map_in(&self, _from: &ZnFastmulBase, lhs: &mut Self::Element, rhs: <ZnFastmulBase as RingBase>::Element, _hom: &Self::Homomorphism) {
debug_assert!(lhs.0 <= self.repr_bound());
let lhs_original = lhs.0;
let product = mullo(lhs.0, rhs.el.0);
let approx_quotient = mullo(lhs.0, high(rhs.value_invmod_shifted)).wrapping_add(mulhi(lhs.0, low(rhs.value_invmod_shifted)));
lhs.0 = product.wrapping_sub(mullo(approx_quotient, self.modulus_u64()));
debug_assert!(lhs.0 < self.modulus_times_three);
debug_assert!((lhs_original as u128 * rhs.el.0 as u128 - lhs.0 as u128) % (self.modulus_u64() as u128) == 0);
}
fn mul_assign_map_in_ref(&self, from: &ZnFastmulBase, lhs: &mut Self::Element, rhs: &<ZnFastmulBase as RingBase>::Element, hom: &Self::Homomorphism) {
self.mul_assign_map_in(from, lhs, *rhs, hom);
}
}
impl CanIsoFromTo<ZnFastmulBase> for ZnBase {
type Isomorphism = <ZnFastmulBase as CanHomFrom<Self>>::Homomorphism;
fn has_canonical_iso(&self, from: &ZnFastmulBase) -> Option<Self::Isomorphism> {
from.has_canonical_hom(self)
}
fn map_out(&self, from: &ZnFastmulBase, el: Self::Element, iso: &Self::Isomorphism) -> <ZnFastmulBase as RingBase>::Element {
from.map_in(self, el, iso)
}
}
impl CooleyTuckeyButterfly<ZnFastmulBase> for ZnBase {
#[inline(always)]
fn butterfly<V: crate::vector::VectorViewMut<Self::Element>, H: Homomorphism<ZnFastmulBase, Self>>(&self, hom: &H, values: &mut V, twiddle: &<ZnFastmulBase as RingBase>::Element, i1: usize, i2: usize) {
let mut a = *values.at(i1);
if a.0 >= self.modulus_times_three {
a.0 -= self.modulus_times_three;
}
let mut b = *values.at(i2);
hom.mul_assign_map_ref(&mut b, twiddle);
debug_assert!(a.0 <= self.modulus_times_three);
debug_assert!(b.0 < self.modulus_times_three);
debug_assert!(self.repr_bound() >= self.modulus_u64() * 6);
*values.at_mut(i1) = self.promise_is_reduced(a.0 + b.0);
*values.at_mut(i2) = self.promise_is_reduced(a.0 + self.modulus_times_three - b.0);
}
fn inv_butterfly<V: crate::vector::VectorViewMut<Self::Element>, H: Homomorphism<ZnFastmulBase, Self>>(&self, hom: &H, values: &mut V, twiddle: &<ZnFastmulBase as RingBase>::Element, i1: usize, i2: usize) {
let a = *values.at(i1);
let b = *values.at(i2);
*values.at_mut(i1) = self.add(a, b);
*values.at_mut(i2) = self.promise_is_reduced(a.0 + self.modulus_times_three - b.0);
hom.mul_assign_map_ref(values.at_mut(i2), twiddle);
}
}
impl<I: ?Sized + IntegerRing> CanHomFrom<I> for ZnFastmulBase
where ZnBase: CanHomFrom<I>
{
type Homomorphism = <ZnBase as CanHomFrom<I>>::Homomorphism;
fn has_canonical_hom(&self, from: &I) -> Option<Self::Homomorphism> {
self.base.get_ring().has_canonical_hom(from)
}
fn map_in(&self, from: &I, el: I::Element, hom: &Self::Homomorphism) -> Self::Element {
self.rev_delegate(self.base.get_ring().map_in(from, el, hom))
}
}
impl_wrap_unwrap_homs!{ ZnBase, ZnBase }
impl_wrap_unwrap_isos!{ ZnBase, ZnBase }
#[cfg(test)]
use test::Bencher;
#[cfg(test)]
fn elements<'a>(ring: &'a Zn) -> impl 'a + Iterator<Item = El<Zn>> {
(0..63).map(|i| ring.coerce(&ZZ, 1 << i))
}
#[test]
fn test_sum() {
for n in [(1 << 41) - 1, (1 << 42) - 1, (1 << 58) - 1, (1 << 58) + 1, (3 << 57) - 1, (3 << 57) + 1] {
let Zn = Zn::new(n);
assert_el_eq!(&Zn, &Zn.int_hom().map(10001 * 5000), &Zn.sum((0..=10000).map(|x| Zn.int_hom().map(x))));
}
}
#[test]
fn test_ring_axioms() {
for n in [2, 5, 7, 17] {
let Zn = Zn::new(n);
crate::ring::generic_tests::test_ring_axioms(&Zn, Zn.elements());
}
for n in [(1 << 41) - 1, (1 << 42) - 1, (1 << 58) - 1, (1 << 58) + 1, (3 << 57) - 1, (3 << 57) + 1] {
let Zn = Zn::new(n);
crate::ring::generic_tests::test_ring_axioms(&Zn, elements(&Zn));
}
}
#[test]
#[allow(deprecated)]
fn test_iso_zn_42() {
for n in [2, 5, 17, (1 << 41) - 1] {
let R1 = Zn::new(n);
let R2 = zn_42::Zn::new(n);
let elements = (1..42).map(|i| 1 << i).map(|x| R2.coerce(&ZZ, x));
crate::ring::generic_tests::test_hom_axioms(&R2, &R1, elements.clone());
crate::ring::generic_tests::test_iso_axioms(&R2, &R1, elements);
}
}
#[test]
fn test_divisibility_axioms() {
for n in [2, 5, 7, 17] {
let Zn = Zn::new(n);
crate::divisibility::generic_tests::test_divisibility_axioms(&Zn, Zn.elements());
}
for n in [(1 << 41) - 1, (1 << 42) - 1, (1 << 58) - 1, (1 << 58) + 1, (3 << 57) - 1, (3 << 57) + 1] {
let Zn = Zn::new(n);
crate::divisibility::generic_tests::test_divisibility_axioms(&Zn, elements(&Zn));
}
}
#[test]
fn test_zn_axioms() {
for n in [2, 5, 7, 17] {
let Zn = Zn::new(n);
super::generic_tests::test_zn_axioms(&Zn);
}
}
#[test]
fn test_principal_ideal_ring_axioms() {
let R = Zn::new(17);
crate::pid::generic_tests::test_principal_ideal_ring_axioms(R, R.elements());
let R = Zn::new(63);
crate::pid::generic_tests::test_principal_ideal_ring_axioms(R, R.elements());
}
#[test]
fn test_hom_from_fastmul() {
for n in [2, 5, 7, 17] {
let Zn = Zn::new(n);
let Zn_fastmul = ZnFastmul::new(Zn);
crate::ring::generic_tests::test_hom_axioms(Zn_fastmul, Zn, Zn.elements().map(|x| Zn_fastmul.coerce(&Zn, x)));
}
for n in [(1 << 41) - 1, (1 << 42) - 1, (1 << 58) - 1, (1 << 58) + 1, (3 << 57) - 1, (3 << 57) + 1] {
let Zn = Zn::new(n);
let Zn_fastmul = ZnFastmul::new(Zn);
crate::ring::generic_tests::test_hom_axioms(Zn_fastmul, Zn, elements(&Zn).map(|x| Zn_fastmul.coerce(&Zn, x)));
}
}
#[test]
fn test_finite_field_axioms() {
crate::rings::finite::generic_tests::test_finite_ring_axioms(&Zn::new(128));
crate::rings::finite::generic_tests::test_finite_ring_axioms(&Zn::new(15));
crate::rings::finite::generic_tests::test_finite_ring_axioms(&Zn::new(1 << 32));
}
#[test]
fn test_from_int_hom() {
for n in [2, 5, 7, 17] {
let Zn = Zn::new(n);
crate::ring::generic_tests::test_hom_axioms(StaticRing::<i8>::RING, Zn, -8..8);
crate::ring::generic_tests::test_hom_axioms(StaticRing::<i16>::RING, Zn, -8..8);
crate::ring::generic_tests::test_hom_axioms(StaticRing::<i32>::RING, Zn, -8..8);
crate::ring::generic_tests::test_hom_axioms(StaticRing::<i64>::RING, Zn, -8..8);
crate::ring::generic_tests::test_hom_axioms(StaticRing::<i128>::RING, Zn, -8..8);
}
let Zn = Zn::new(5);
assert_el_eq!(&Zn, &Zn.int_hom().map(3), &Zn.can_hom(&StaticRing::<i64>::RING).unwrap().map(-1596802));
}
#[bench]
fn bench_hom_from_i64(bencher: &mut Bencher) {
let Zn = Zn::new(36028797018963971 );
bencher.iter(|| {
let hom = Zn.can_hom(&StaticRing::<i64>::RING).unwrap();
assert_el_eq!(&Zn, &Zn.int_hom().map(-1300), &Zn.sum((0..100).flat_map(|_| (0..=56).map(|k| 1 << k)).map(|x| hom.map(x))))
});
}