use std::cmp::Ordering;
use std::mem::swap;
use crate::integer::*;
use crate::ordered::{OrderedRing, OrderedRingStore};
use crate::pid::*;
use crate::ring::*;
pub fn eea<R>(a: El<R>, b: El<R>, ring: R) -> (El<R>, El<R>, El<R>)
where
R: RingStore,
R::Type: EuclideanRing,
{
let (mut a, mut b) = (a, b);
let (mut sa, mut ta) = (ring.one(), ring.zero());
let (mut sb, mut tb) = (ring.zero(), ring.one());
while !ring.is_zero(&b) {
let (quo, rem) = ring.euclidean_div_rem(a, &b);
ta = ring.sub(ta, ring.mul_ref(&quo, &tb));
sa = ring.sub(sa, ring.mul_ref_snd(quo, &sb));
a = rem;
swap(&mut a, &mut b);
swap(&mut sa, &mut sb);
swap(&mut ta, &mut tb);
}
return (sa, ta, a);
}
#[stability::unstable(feature = "enable")]
pub fn half_eea<R>(a: El<R>, b: El<R>, ring: R) -> (El<R>, El<R>)
where
R: RingStore,
R::Type: EuclideanRing,
{
let (mut a, mut b) = (a, b);
let (mut s, mut t) = (ring.one(), ring.zero());
while !ring.is_zero(&b) {
let (q, r) = ring.euclidean_div_rem(a, &b);
a = r;
ring.sub_assign(&mut s, ring.mul_ref_snd(q, &t));
swap(&mut a, &mut b);
swap(&mut s, &mut t);
}
return (s, a);
}
pub fn signed_eea<R>(fst: El<R>, snd: El<R>, ring: R) -> (El<R>, El<R>, El<R>)
where
R: RingStore,
R::Type: EuclideanRing + OrderedRing,
{
if ring.is_zero(&fst) {
return match ring.cmp(&snd, &ring.zero()) {
Ordering::Equal => (ring.zero(), ring.zero(), ring.zero()),
Ordering::Less => (ring.zero(), ring.negate(ring.one()), ring.negate(snd)),
Ordering::Greater => (ring.zero(), ring.one(), snd),
};
}
let fst_negative = ring.cmp(&fst, &ring.zero());
let (s, t, d) = eea(fst, snd, &ring);
if ring.cmp(&d, &ring.zero()) == fst_negative {
return (s, t, d);
} else {
return (ring.negate(s), ring.negate(t), ring.negate(d));
}
}
pub fn gcd<R>(a: El<R>, b: El<R>, ring: R) -> El<R>
where
R: RingStore,
R::Type: EuclideanRing,
{
let (mut a, mut b) = (a, b);
while !ring.is_zero(&b) {
let (_, r) = ring.euclidean_div_rem(a, &b);
a = b;
b = r;
}
return a;
}
pub fn signed_gcd<R>(a: El<R>, b: El<R>, ring: R) -> El<R>
where
R: RingStore,
R::Type: EuclideanRing + OrderedRing,
{
let (_, _, d) = signed_eea(a, b, ring);
return d;
}
pub fn signed_lcm<R>(fst: El<R>, snd: El<R>, ring: R) -> El<R>
where
R: RingStore,
R::Type: EuclideanRing + OrderedRing,
{
if ring.is_zero(&fst) || ring.is_zero(&snd) {
ring.zero()
} else {
ring.mul(
ring.euclidean_div(ring.clone_el(&fst), &signed_gcd(fst, ring.clone_el(&snd), &ring)),
snd,
)
}
}
pub fn lcm<R>(fst: El<R>, snd: El<R>, ring: R) -> El<R>
where
R: RingStore,
R::Type: EuclideanRing,
{
if ring.is_zero(&fst) || ring.is_zero(&snd) {
ring.zero()
} else {
ring.euclidean_div(ring.mul_ref(&fst, &snd), &gcd(fst, snd, &ring))
}
}
pub fn inv_crt<I>(a: El<I>, b: El<I>, p: &El<I>, q: &El<I>, ZZ: I) -> El<I>
where
I: RingStore,
I::Type: IntegerRing,
{
let (s, t, d) = signed_eea(ZZ.clone_el(p), ZZ.clone_el(q), &ZZ);
assert!(ZZ.is_one(&d) || ZZ.is_neg_one(&d));
let mut result = ZZ.add(ZZ.prod([a, t, ZZ.clone_el(q)]), ZZ.prod([b, s, ZZ.clone_el(p)]));
let n = ZZ.mul_ref(p, q);
result = ZZ.euclidean_rem(result, &n);
if ZZ.is_neg(&result) {
ZZ.add_assign(&mut result, n);
}
return result;
}
#[cfg(test)]
use crate::primitive_int::*;
#[test]
fn test_gcd() {
assert_eq!(3, gcd(15, 6, &StaticRing::<i64>::RING).abs());
assert_eq!(3, gcd(6, 15, &StaticRing::<i64>::RING).abs());
assert_eq!(7, gcd(0, 7, &StaticRing::<i64>::RING).abs());
assert_eq!(7, gcd(7, 0, &StaticRing::<i64>::RING).abs());
assert_eq!(0, gcd(0, 0, &StaticRing::<i64>::RING).abs());
assert_eq!(1, gcd(9, 1, &StaticRing::<i64>::RING).abs());
assert_eq!(1, gcd(1, 9, &StaticRing::<i64>::RING).abs());
assert_eq!(1, gcd(13, 300, &StaticRing::<i64>::RING).abs());
assert_eq!(1, gcd(300, 13, &StaticRing::<i64>::RING).abs());
assert_eq!(3, gcd(-15, 6, &StaticRing::<i64>::RING).abs());
assert_eq!(3, gcd(6, -15, &StaticRing::<i64>::RING).abs());
assert_eq!(3, gcd(-6, -15, &StaticRing::<i64>::RING).abs());
}
#[test]
fn test_signed_gcd() {
assert_eq!(3, signed_gcd(15, 6, &StaticRing::<i64>::RING));
assert_eq!(3, signed_gcd(6, 15, &StaticRing::<i64>::RING));
assert_eq!(7, signed_gcd(0, 7, &StaticRing::<i64>::RING));
assert_eq!(7, signed_gcd(7, 0, &StaticRing::<i64>::RING));
assert_eq!(0, signed_gcd(0, 0, &StaticRing::<i64>::RING));
assert_eq!(1, signed_gcd(9, 1, &StaticRing::<i64>::RING));
assert_eq!(1, signed_gcd(1, 9, &StaticRing::<i64>::RING));
assert_eq!(1, signed_gcd(13, 300, &StaticRing::<i64>::RING));
assert_eq!(1, signed_gcd(300, 13, &StaticRing::<i64>::RING));
assert_eq!(-3, signed_gcd(-15, 6, &StaticRing::<i64>::RING));
assert_eq!(3, signed_gcd(6, -15, &StaticRing::<i64>::RING));
assert_eq!(-3, signed_gcd(-6, -15, &StaticRing::<i64>::RING));
}
#[test]
fn test_eea_sign() {
assert_eq!((2, -1, 1), signed_eea(3, 5, &StaticRing::<i64>::RING));
assert_eq!((-1, 2, 1), signed_eea(5, 3, &StaticRing::<i64>::RING));
assert_eq!((2, 1, -1), signed_eea(-3, 5, &StaticRing::<i64>::RING));
assert_eq!((-1, -2, 1), signed_eea(5, -3, &StaticRing::<i64>::RING));
assert_eq!((2, 1, 1), signed_eea(3, -5, &StaticRing::<i64>::RING));
assert_eq!((-1, -2, -1), signed_eea(-5, 3, &StaticRing::<i64>::RING));
assert_eq!((2, -1, -1), signed_eea(-3, -5, &StaticRing::<i64>::RING));
assert_eq!((-1, 2, -1), signed_eea(-5, -3, &StaticRing::<i64>::RING));
assert_eq!((0, 0, 0), signed_eea(0, 0, &StaticRing::<i64>::RING));
assert_eq!((1, 0, 4), signed_eea(4, 0, &StaticRing::<i64>::RING));
assert_eq!((0, 1, 4), signed_eea(0, 4, &StaticRing::<i64>::RING));
assert_eq!((1, 0, -4), signed_eea(-4, 0, &StaticRing::<i64>::RING));
assert_eq!((0, -1, 4), signed_eea(0, -4, &StaticRing::<i64>::RING));
}
#[test]
fn test_signed_eea() {
assert_eq!((-1, 1, 2), signed_eea(6, 8, &StaticRing::<i64>::RING));
assert_eq!((2, -1, 5), signed_eea(15, 25, &StaticRing::<i64>::RING));
assert_eq!((4, -7, 2), signed_eea(32, 18, &StaticRing::<i64>::RING));
}
#[test]
fn test_signed_lcm() {
assert_eq!(24, signed_lcm(6, 8, &StaticRing::<i64>::RING));
assert_eq!(24, signed_lcm(-6, 8, &StaticRing::<i64>::RING));
assert_eq!(-24, signed_lcm(6, -8, &StaticRing::<i64>::RING));
assert_eq!(-24, signed_lcm(-6, -8, &StaticRing::<i64>::RING));
assert_eq!(0, signed_lcm(0, 0, &StaticRing::<i64>::RING));
assert_eq!(0, signed_lcm(6, 0, &StaticRing::<i64>::RING));
assert_eq!(0, signed_lcm(0, 8, &StaticRing::<i64>::RING));
}