use crate::traits::NumberTheory;
use crate::result::NTResult;
use crate::arithmetic::inlineops::*;
use crate::arithmetic::mpz::Mpz;
use crate::arithmetic::sign::Sign;
use crate::arithmetic::sliceops::*;
use crate::data::primes::PRIMELIST;
use crate::data::primes::MERSENNE_LIST;
use std::cmp::Ordering;
impl NumberTheory for Mpz {
fn rng() -> Self {
Mpz::unchecked_new(Sign::Positive, vec![rng_64(), rng_64(), rng_64(), rng_64()])
}
fn residue(&self, ring: &Self) -> Self{
if ring.clone() == Mpz::zero(){
return self.clone()
}
let rem = self.ref_euclidean(&ring).1;
if !self.is_positive(){
return self.ref_subtraction(&rem)
}
return rem
}
fn euclidean_div(&self, other: &Self) -> (Self, Self) {
let (mut quo, mut rem) = self.ref_euclidean(other);
if self.sign == Sign::Negative && other.sign == Sign::Negative {
rem.neg();
} else if self.sign == Sign::Positive && other.sign == Sign::Negative {
quo.neg();
} else if self.sign == Sign::Negative && other.sign == Sign::Positive {
quo.neg();
rem.neg();
}
quo.fix_zero();
rem.fix_zero();
(quo, rem)
}
fn quadratic_residue(&self, n: &Self) -> Self {
if n == &Mpz::zero() {
return Mpz::zero();
}
let mut p = self.clone();
if p.sign == Sign::Negative {
p = p.add_modinv(n);
}
if n.is_power_of_two() {
let reducer = n.ref_subtraction(&Mpz::one());
return p.ref_product(&p).and(&reducer);
}
p.ref_product(&p).ref_euclidean(n).1
}
fn checked_quadratic_residue(&self, n: &Self) -> NTResult<Self>{
NTResult::Eval(self.quadratic_residue(n))
}
fn product_residue(&self, other: &Self, n: &Self) -> Self {
if n == &Mpz::zero() {
return Mpz::zero();
}
let mut p = self.clone();
let mut q = other.clone();
if p.sign == Sign::Negative {
p = p.add_modinv(n);
}
if q.sign == Sign::Negative {
q = q.add_modinv(n)
}
if n.is_power_of_two() {
let reducer = n.ref_subtraction(&Mpz::one());
return p.ref_product(&q).and(&reducer);
}
p.ref_product(&q).ref_euclidean(n).1
}
fn checked_product_residue(&self, other: &Self, n: &Self) -> NTResult<Self>{
NTResult::Eval(self.product_residue(other,n))
}
fn exp_residue(&self, y: &Self, n: &Self) -> Self {
if n == &Mpz::zero() {
match y.to_i64() {
Some(x) => {
if x < 1i64{
panic!("No inverse")
}
return self.pow(x as u64)
},
None => panic!("Incomputably large result"),
}
}
let mut p = self.clone();
if p.sign == Sign::Negative {
p = p.add_modinv(n);
}
p.u_mod_pow(y, n)
}
fn checked_exp_residue(&self, y: &Self, n: &Self) -> NTResult<Self> {
if n == &Mpz::zero() {
match y.to_i64() {
Some(x) => {
if x < 0i64{
return NTResult::DNE
}
return NTResult::Eval(self.pow(x as u64))
},
None => return NTResult::CompExceeded,
}
}
let mut p = self.clone();
if p.sign == Sign::Negative {
p = p.add_modinv(n);
}
if !y.is_positive(){
let (gcd,base,_) = self.eea(n);
if !gcd.is_one(){
return NTResult::DNE
}
return NTResult::Eval(base.u_mod_pow(y,n))
}
NTResult::Eval(p.u_mod_pow(y, n))
}
fn is_sprp(&self, base: &Self) -> bool {
self.sprp(base.clone())
}
#[cfg(not(feature = "parallel"))]
fn is_prime(&self) -> bool {
if self.len() < 3 {
return self.to_u128().unwrap().is_prime();
}
if !self.trial_div() {
return false;
}
if self.is_fermat() {
return false;
}
match self.is_mersenne() {
Some(x) => {
if MERSENNE_LIST.contains(&(x as u32)) {
return true;
} else if x < 57885161 {
return false;
} else {
return self.llt(x);
}
}
None => (),
}
if !self.is_sprp(&Mpz::two()) {
return false;
}
if !self.form_check() {
return false;
}
if !self.jacobi_check_mpz() {
return false;
}
if !self.weighted_sprp() {
return false;
}
true
}
#[cfg(feature = "parallel")]
fn is_prime(&self) -> bool {
if self.len() < 3 {
return self.to_u128().unwrap().is_prime();
}
if !self.trial_div(){
return false
}
if self.is_fermat() {
return false;
}
match self.is_mersenne() {
Some(x) => {
if MERSENNE_LIST.contains(&(x as u32)) {
return true;
} else if x < 57885161 {
return false;
} else {
return self.llt(x);
}
}
None => (),
}
if self.len() < 9 {
if !self.is_sprp(&Mpz::two()) {
return false;
}
if !self.form_check() {
return false;
}
if !self.jacobi_check_mpz() {
return false;
}
if !self.weighted_sprp() {
return false;
}
return true;
} else {
let a = self.clone();
let b = self.clone();
let c = self.clone();
let two = std::thread::spawn(move || a.is_sprp(&Mpz::two()));
let jacobi = std::thread::spawn(move || b.jacobi_check_mpz());
let rand = std::thread::spawn(move || c.weighted_sprp());
if !self.form_check() {
return false;
}
return two.join().unwrap() & jacobi.join().unwrap() & rand.join().unwrap();
}
}
fn prime_proof(&self) -> (bool, Vec<Self>) {
let x_minus = self.ref_subtraction(&Mpz::one());
let fctrs = x_minus
.factor()
.iter()
.step_by(2)
.map(|y| y.clone())
.collect::<Vec<Self>>();
let mut certificate = vec![Mpz::two()];
certificate.extend_from_slice(&fctrs[..]);
loop {
let mut witness = Mpz::rand((self.bit_length()-1) as usize);
'witness: loop {
if witness.gcd(self) == Mpz::one() {
break 'witness;
}
witness.successor();
}
if witness.exp_residue(&x_minus, self) != Mpz::one() {
certificate[0] = witness;
return (false, certificate);
}
'inner: for (idx, j) in fctrs.iter().enumerate() {
if witness.exp_residue(&x_minus.ref_euclidean(j).0, self) == Mpz::one() {
break 'inner;
}
if idx == fctrs.len() - 1 {
certificate[0] = witness;
return (true, certificate);
}
}
}
}
fn prime_list(&self, sup: &Self) -> Vec<Self> {
let mut delta = self.ref_subtraction(sup);
delta.successor();
match delta.to_u64() {
Some(x) => {
let mut primevector = vec![];
let mut p = self.clone();
for _ in 0..x{
if p.is_prime() {
primevector.push(p.clone())
}
p.successor();
}
primevector
}
None => panic!("Incomputably large interval"),
}
}
fn nth_prime(&self) -> NTResult<Self> {
let mut count = Mpz::zero();
let mut start = Mpz::zero();
if *self == Mpz::zero(){
return NTResult::DNE
}
loop {
start.successor();
if start.is_prime() {
count.successor()
}
if count.u_cmp(self) == Ordering::Equal {
return NTResult::Eval(start);
}
}
}
fn pi(&self) -> Self {
match self.to_u128(){
Some(x) => return Mpz::from_u128(x.pi()),
None => (),
}
let mut count = 0u64;
let mut start = Mpz::zero();
loop {
start.successor();
if start.is_prime() {
count += 1;
}
if start.u_cmp(self) == Ordering::Equal {
return Mpz::from_u64(count);
}
}
}
fn prime_gen(x: u32) -> NTResult<Mpz> {
if x < 128 {
return u128::prime_gen(x).map(|y| Mpz::from_u128(y))
}
let mut form = Mpz::one().shl(x as usize-1);
let bitlength = form.ref_subtraction(&Mpz::one());
form.successor();
loop {
let mut k = Mpz::rand(x as usize+1);
k.mut_and(&bitlength);
k.mut_or(&form);
assert_eq!(k.bit_length() as u32,x);
if k.is_prime() {
return NTResult::Eval(k);
}
}
}
fn gcd(&self, other: &Self) -> Self {
let mut a = self.clone();
let mut b = other.clone();
while b != Mpz::zero() {
let t = b.clone();
b = a.ref_euclidean(&b).1;
b.normalize();
a = t;
}
a
}
fn extended_gcd(&self, other: &Self) -> (Self, Self, Self) {
let mut gcd = self.clone();
let mut new_r = other.clone();
let mut bezout_1 = Mpz::one();
let mut new_s = Mpz::zero();
let mut bezout_2 = Mpz::zero();
let mut new_t = Mpz::one();
while new_r != Mpz::zero() {
let (quo, _rem) = gcd.euclidean_div(&new_r);
let mut temp = new_r.clone();
new_r = gcd.ref_subtraction(&quo.ref_product(&temp));
gcd = temp.clone();
temp = new_s.clone();
new_s = bezout_1.ref_subtraction(&quo.ref_product(&temp));
bezout_1 = temp.clone();
temp = new_t.clone();
new_t = bezout_2.ref_subtraction(&quo.ref_product(&temp));
bezout_2 = temp.clone();
}
(gcd, bezout_1, bezout_2)
}
fn lcm(&self, other: &Self) -> Self {
if self == &Mpz::zero() && other == &Mpz::zero(){
return Mpz::zero()
}
let cf = self.gcd(other);
self.euclidean_div(&cf).0.ref_product(&other)
}
fn checked_lcm(&self, other: &Self) -> NTResult<Self> {
NTResult::Eval(self.lcm(other))
}
fn factor(&self) -> Vec<Self> {
let mut n = self.clone();
let mut factors: Vec<Self> = vec![];
let twofactor = n.trailing_zeros();
if twofactor > 0 {
n.mut_shr(twofactor as usize);
factors.push(Mpz::from_u64(2u64));
factors.push(Mpz::from_u64(twofactor));
}
for i in PRIMELIST[1..].iter() {
let (quo, rem) = n.word_div(*i as u64);
if rem == 0 {
let mut count = 1u64;
factors.push(Mpz::from_u64(*i as u64));
n = quo;
'inner: loop {
let (inner_quo, inner_rem) = n.word_div(*i as u64);
if inner_rem != 0 {
break 'inner;
}
n = inner_quo;
count += 1;
}
factors.push(Mpz::from_u64(count));
}
}
n.normalize();
if n.is_one(){
return factors;
}
if n.sprp_check(5) {
factors.push(n.clone());
factors.push(Mpz::one());
return factors;
}
'outer: while !n.is_one() {
let k = n.rho_mpz();
factors.push(k.clone());
let mut count = 0u64;
'secinner: loop {
let (inner_quo, inner_rem) = n.ref_euclidean(&k);
if !inner_rem.is_zero() {
break 'secinner;
}
n = inner_quo;
n.normalize(); count += 1;
}
factors.push(Mpz::from_u64(count));
if n.sprp_check(5) {
factors.push(n);
factors.push(Mpz::one());
break 'outer;
}
}
factors
}
fn checked_factor(&self) -> NTResult<Vec<Self>>{
match self.to_u64(){
Some(x) => {
if x == 0 {
return NTResult::InfiniteSet
}
if x == 1 {
return NTResult::DNE
}
}
None => (),
}
NTResult::Eval(self.factor())
}
fn sqrt(&self) -> (Self, Self) {
if self.len() < 3 {
let k = Mpz::from_u64(self.to_u128().unwrap().sqrt().0 as u64);
if !self.is_positive() {
return (k, Mpz::one());
}
return (k, Mpz::zero());
}
let mut est = self.shr(((self.bit_length() / 2) - 1) as usize).abs();
loop {
let s = est.clone();
let t = s.ref_addition(&self.euclidean_div(&s).0);
est = t.shr(1);
remove_lead_zeros(&mut est.limbs);
if est.u_cmp(&s) == Ordering::Greater || est.u_cmp(&s) == Ordering::Equal {
if self.sign == Sign::Negative {
return (s, Mpz::one());
}
return (s, Mpz::zero());
}
}
}
fn nth_root(&self, n: &Self) -> (Self, Self) {
let y = n.to_u64().unwrap();
let shift = ((self.bit_length() / y) - 1) * (y - 1);
let mut est = self.shr(shift as usize).abs();
let scalar = Mpz::from_u64(y - 1);
let ymp = Mpz::from_u64(y);
loop {
let s = est.clone();
let t = s
.ref_product(&scalar)
.ref_addition(&self.euclidean_div(&s.pow(y - 1)).0);
est = t.euclidean_div(&ymp).0;
if est.u_cmp(&s) == Ordering::Greater || est.u_cmp(&s) == Ordering::Equal {
if self.sign == Sign::Negative && self.is_even() {
return (s, Mpz::one());
}
return (s, Mpz::zero());
}
}
}
fn max_exp(&self) -> (Self,Self){
let mut expo = Mpz::from_u64(self.bit_length());
let mut flag = true;
if self.is_positive(){
flag = false
}
loop{
let mut base = self.abs().nth_root(&expo).0;
if base.pow(expo.to_u64().unwrap()) == self.abs(){
if flag{
base.neg();
}
return(base,expo)
}
expo.inv_successor();
if expo.to_u64().unwrap() == 1{
let mut val = self.clone();
if flag {
val.neg()
}
return (val,Mpz::one())
}
}
}
fn radical(&self) -> NTResult<Self>{
match self.to_u64(){
Some(x) => return x.radical().map(|y| Mpz::from_u64(y)),
None => {
let mut rad = Mpz::one();
for i in self.factor().iter().step_by(2) {
rad = rad.ref_product(i)
}
return NTResult::Eval(rad)
},
}
}
fn k_free(&self, x: &Self) -> bool {
for i in self.factor()[1..].iter().step_by(2) {
if i == x {
return false;
}
}
true
}
fn euler_totient(&self) -> Self {
match self.to_u128(){
Some(x)=> return Mpz::from_u128(x.euler_totient()),
None => (),
}
let mut factors = self.factor();
let mut denominator = Mpz::one();
let mut numerator = Mpz::one();
for i in factors.iter().step_by(2) {
denominator = denominator.ref_product(i)
}
for i in factors.iter_mut().step_by(2) {
sub_slice(&mut i.limbs[..], &[1]);
numerator = numerator.ref_product(i)
}
(self.ref_euclidean(&denominator).0).ref_product(&numerator)
}
fn jordan_totient(&self, k: &Self) -> NTResult<Self> {
if self.clone() == Mpz::zero() || self.is_one(){
return NTResult::Eval(self.clone())
}
let fctr = self.factor();
let mut denom = Mpz::one();
let mut numer = Mpz::one();
let negone = Mpz::unchecked_new(Sign::Negative, vec![1]);
for i in fctr.iter().step_by(2) {
let pow = i.pow(k.to_u64().unwrap());
denom = denom.ref_product(&pow);
numer = numer.ref_product(&pow.ref_addition(&negone));
}
NTResult::Eval(
numer
.ref_product(&self.pow(k.to_u64().unwrap()))
.ref_euclidean(&denom)
.0,
)
}
fn carmichael_totient(&self) -> NTResult<Self>{
match self.to_u128(){
Some(x)=> return x.carmichael_totient().map(|y| Mpz::from_u128(y)),
None => (),
}
let fctr = self.factor();
let base = fctr.iter().step_by(2).map(|z| z.clone()).collect::<Vec<Self>>();
let power = fctr[1..].iter().step_by(2).map(|k| k.to_u64().unwrap()).collect::<Vec<u64>>();
let two = Mpz::two();
let mut result = Mpz::one();
for (idx,el) in base.iter().enumerate(){
if el == &two && power[0] > 2{
let phi = ((el.pow(power[idx]).ref_euclidean(el).0).ref_product(&el.ref_subtraction(&Mpz::one()))).shr(1);
result = result.lcm(&phi);
}
else{
let phi = (el.pow(power[idx]).ref_euclidean(el).0).ref_product(&el.ref_subtraction(&Mpz::one()));
result = result.lcm(&phi);
}
}
NTResult::Eval(result)
}
fn dedekind_psi(&self, k: &Self) -> NTResult<Self> {
if *self == Mpz::zero(){
return NTResult::Infinite
}
let k2 = k.shl(1);
self.jordan_totient(&k2).map(|y| y.ref_euclidean(&self.jordan_totient(k).unwrap()).0)
}
fn legendre(&self, p: &Self) -> i8 {
let mut p_minus = p.clone();
p_minus.set_sign(Sign::Positive);
p_minus.inv_successor();
let pow = p_minus.ref_euclidean(&Mpz::two()).0;
let k = self.exp_residue(&pow, p);
if k == Mpz::one() {
return 1;
};
if k == p_minus {
return -1;
};
0i8
}
fn checked_legendre(&self, p: &Self) -> NTResult<i8> {
let mut plus = p.clone();
plus.set_sign(Sign::Positive);
if plus == Mpz::two() {
return NTResult::Undefined;
}
match plus.is_prime() {
true => NTResult::Eval(self.legendre(&plus)),
false => NTResult::Undefined,
}
}
fn liouville(&self) -> i8 {
match self.to_u128(){
Some(an) => return an.liouville(),
None => (),
}
let mut primeomega = Mpz::zero();
for i in self.factor()[1..].iter().step_by(2) {
primeomega.mut_addition(i.clone());
}
if primeomega.is_even() {
return 1;
}
return -1;
}
fn derivative(&self) -> NTResult<Self> {
if *self == Mpz::zero(){
return NTResult::Eval(Mpz::zero())
}
if self.is_one(){
return NTResult::Eval(Mpz::zero())
}
let fctr = self.factor();
let mut sum = Mpz::zero();
for i in 0..fctr.len() / 2 {
sum.mut_addition(fctr[2*i+1].ref_product(&self.ref_euclidean(&fctr[2*i]).0))
}
NTResult::Eval(sum)
}
fn mangoldt(&self) -> f64 {
match self.to_u128(){
Some(eburum) => return eburum.mangoldt(),
None => (),
}
let fctr = self.factor();
if fctr.len() != 2 {
return 0f64;
}
return fctr[0].ln();
}
fn mobius(&self) -> i8 {
match self.to_u128(){
Some(eburum) => return eburum.mobius(),
None => (),
}
let fctr = self.factor();
if fctr.len() == 1{ return -1
}
for i in 0..fctr.len()/2{
if fctr[2*i+1].to_u64().unwrap() > 1{
return 0
}
}
let fctrsum = fctr[1..].iter().step_by(2).map(|k| k.to_u64().unwrap()).sum::<u64>();
if fctrsum&1 == 1{ return -1
}
return 1
}
fn jacobi(&self, k: &Self) -> i8 {
let mut n = self.clone();
let mut p = k.clone();
let mut t = 1i8;
n = n.euclidean_div(&p).1;
while n != Mpz::zero() {
let zeros = n.trailing_zeros();
n.mut_shr(zeros as usize);
if (p.congruence_u64(8, 3) || p.congruence_u64(8, 5)) && (zeros % 2 == 1) {
t = -t
}
std::mem::swap(&mut n, &mut p);
if n.congruence_u64(4, 3) && p.congruence_u64(4, 3) {
t = -t;
}
n = n.euclidean_div(&p).1;
}
if p == Mpz::one() {
t
} else {
0
}
}
fn checked_jacobi(&self, k: &Self) -> NTResult<i8> {
if k.sign == Sign::Positive && k != &Mpz::zero() && !k.is_even() {
return NTResult::Eval(self.jacobi(k));
}
NTResult::Undefined
}
fn kronecker(&self, k: &Self) -> i8{
let mut multiplier = 1;
if !k.is_positive() && !self.is_positive(){
multiplier = -1;
}
let x = self.clone();
if k.is_zero(){
if x.is_one(){
return 1
}
return 0
}
if k.is_one(){
return 1
}
let fctr = k.factor();
let mut start = 0;
let mut res = 1;
let two = Mpz::two();
if fctr[0] == two{
start = 1;
if x.is_even(){
res = 0;
}
else if x.congruence_u64(8,1) || x.congruence_u64(8,7){
res=1
}
else{
res = (-1i8).pow(fctr[1].to_u64().unwrap() as u32)
}
}
if fctr[0] == two && fctr.len() == 2{
return res*multiplier
}
for i in start..fctr.len()/2{
res*=self.legendre(&fctr[2*i]).pow(fctr[2*i+1].to_u64().unwrap() as u32);
}
return res*multiplier
}
fn smooth(&self) -> NTResult<Self> {
if *self == Mpz::zero(){
return NTResult::Infinite
}
if self.is_one(){
return NTResult::DNE
}
let k = self.factor();
NTResult::Eval(k[k.len() - 2].clone())
}
fn is_smooth(&self, b: &Self) -> bool {
match self.smooth(){
NTResult::Infinite => false,
NTResult::Eval(x) => {
let mut k = b.clone();
k.set_sign(Sign::Positive);
return matches!(x.u_cmp(&k), Ordering::Less)
},
_=> false,
}
}
}