use core::cmp::Ordering;
use dashu_base::{AbsOrd, EstimatedLog2, Sign};
use dashu_int::{IBig, UBig};
use crate::{
fbig::FBig,
repr::Repr,
repr::Word,
round::Round,
utils::{shl_digits, shl_digits_in_place},
};
impl<R1: Round, R2: Round, const B: Word> PartialEq<FBig<R2, B>> for FBig<R1, B> {
#[inline]
fn eq(&self, other: &FBig<R2, B>) -> bool {
match (self.repr.is_infinite(), other.repr.is_infinite()) {
(true, true) => !((self.repr.exponent >= 0) ^ (other.repr.exponent >= 0)),
(false, false) => self.repr == other.repr,
(_, _) => false,
}
}
}
impl<R: Round, const B: Word> Eq for FBig<R, B> {}
fn repr_cmp_same_base<const B: Word, const ABS: bool>(
lhs: &Repr<B>,
rhs: &Repr<B>,
precision: Option<(usize, usize)>,
) -> Ordering {
match (lhs.is_infinite(), rhs.is_infinite()) {
(true, true) => {
return if ABS {
Ordering::Equal
} else {
lhs.exponent.cmp(&rhs.exponent)
}
}
(false, true) => {
return match ABS || rhs.exponent >= 0 {
true => Ordering::Less,
false => Ordering::Greater,
}
}
(true, false) => {
return match ABS || lhs.exponent >= 0 {
true => Ordering::Greater,
false => Ordering::Less,
}
}
_ => {}
};
let sign = if ABS {
Sign::Positive
} else {
match (lhs.significand.sign(), rhs.significand.sign()) {
(Sign::Positive, Sign::Positive) => Sign::Positive,
(Sign::Positive, Sign::Negative) => return Ordering::Greater,
(Sign::Negative, Sign::Positive) => return Ordering::Less,
(Sign::Negative, Sign::Negative) => Sign::Negative,
}
};
match (lhs.is_zero(), rhs.is_zero()) {
(true, true) => return Ordering::Equal,
(true, false) => {
return Ordering::Less;
}
(false, true) => {
return Ordering::Greater;
}
_ => {}
}
let (lhs_exp, rhs_exp) = (lhs.exponent, rhs.exponent);
if let Some((lhs_prec, rhs_prec)) = precision {
if lhs_prec != 0 && rhs_prec != 0 {
if lhs_exp > rhs_exp + rhs_prec as isize {
return sign * Ordering::Greater;
}
if rhs_exp > lhs_exp + lhs_prec as isize {
return sign * Ordering::Less;
}
}
}
let (lhs_digits, rhs_digits) = (lhs.digits_ub(), rhs.digits_ub());
if lhs_exp > rhs_exp + rhs_digits as isize {
return sign * Ordering::Greater;
}
if rhs_exp > lhs_exp + lhs_digits as isize {
return sign * Ordering::Less;
}
let (lhs_signif, rhs_signif) = (&lhs.significand, &rhs.significand);
if ABS {
match lhs_exp.cmp(&rhs_exp) {
Ordering::Equal => lhs_signif.abs_cmp(rhs_signif),
Ordering::Greater => {
shl_digits::<B>(lhs_signif, (lhs_exp - rhs_exp) as usize).abs_cmp(rhs_signif)
}
Ordering::Less => {
lhs_signif.abs_cmp(&shl_digits::<B>(rhs_signif, (rhs_exp - lhs_exp) as usize))
}
}
} else {
match lhs_exp.cmp(&rhs_exp) {
Ordering::Equal => lhs_signif.cmp(rhs_signif),
Ordering::Greater => {
shl_digits::<B>(lhs_signif, (lhs_exp - rhs_exp) as usize).cmp(rhs_signif)
}
Ordering::Less => {
lhs_signif.cmp(&shl_digits::<B>(rhs_signif, (rhs_exp - lhs_exp) as usize))
}
}
}
}
impl<const B: Word> PartialOrd for Repr<B> {
#[inline]
fn partial_cmp(&self, other: &Self) -> Option<Ordering> {
Some(self.cmp(other))
}
}
impl<const B: Word> Ord for Repr<B> {
#[inline]
fn cmp(&self, other: &Self) -> Ordering {
repr_cmp_same_base::<B, false>(self, other, None)
}
}
impl<R1: Round, R2: Round, const B: Word> PartialOrd<FBig<R2, B>> for FBig<R1, B> {
#[inline]
fn partial_cmp(&self, other: &FBig<R2, B>) -> Option<Ordering> {
Some(repr_cmp_same_base::<B, false>(
&self.repr,
&other.repr,
Some((self.context.precision, other.context.precision)),
))
}
}
impl<R: Round, const B: Word> Ord for FBig<R, B> {
#[inline]
fn cmp(&self, other: &Self) -> Ordering {
repr_cmp_same_base::<B, false>(
&self.repr,
&other.repr,
Some((self.context.precision, other.context.precision)),
)
}
}
impl<R: Round, const B: Word> AbsOrd for FBig<R, B> {
#[inline]
fn abs_cmp(&self, other: &Self) -> Ordering {
repr_cmp_same_base::<B, true>(
&self.repr,
&other.repr,
Some((self.context.precision, other.context.precision)),
)
}
}
pub(crate) fn repr_cmp_ubig<const B: Word, const ABS: bool>(lhs: &Repr<B>, rhs: &UBig) -> Ordering {
if lhs.is_infinite() {
return if lhs.exponent > 0 || ABS {
Ordering::Greater
} else {
Ordering::Less
};
}
if !ABS && lhs.significand.sign() == Sign::Negative {
return Ordering::Less;
}
let (lhs_lo, lhs_hi) = lhs.log2_bounds();
let (rhs_lo, rhs_hi) = rhs.log2_bounds();
if lhs_lo > rhs_hi {
return Ordering::Greater;
}
if lhs_hi < rhs_lo {
return Ordering::Less;
}
let mut rhs: IBig = rhs.clone().into();
if lhs.exponent < 0 {
shl_digits_in_place::<B>(&mut rhs, (-lhs.exponent) as usize);
lhs.significand.cmp(&rhs)
} else {
shl_digits::<B>(&lhs.significand, lhs.exponent as usize).cmp(&rhs)
}
}
pub(crate) fn repr_cmp_ibig<const B: Word, const ABS: bool>(lhs: &Repr<B>, rhs: &IBig) -> Ordering {
if lhs.is_infinite() {
return if lhs.exponent > 0 || ABS {
Ordering::Greater
} else {
Ordering::Less
};
}
let sign = if ABS {
Sign::Positive
} else {
match (lhs.significand.sign(), rhs.sign()) {
(Sign::Positive, Sign::Positive) => Sign::Positive,
(Sign::Positive, Sign::Negative) => return Ordering::Greater,
(Sign::Negative, Sign::Positive) => return Ordering::Less,
(Sign::Negative, Sign::Negative) => Sign::Negative,
}
};
let (lhs_lo, lhs_hi) = lhs.log2_bounds();
let (rhs_lo, rhs_hi) = rhs.log2_bounds();
if lhs_lo > rhs_hi {
return sign * Ordering::Greater;
}
if lhs_hi < rhs_lo {
return sign * Ordering::Less;
}
if lhs.exponent < 0 {
lhs.significand
.cmp(&shl_digits::<B>(rhs, (-lhs.exponent) as usize))
} else {
shl_digits::<B>(&lhs.significand, lhs.exponent as usize).cmp(rhs)
}
}
macro_rules! impl_abs_ord_with_method {
($T:ty, $method:ident) => {
impl<const B: Word> AbsOrd<$T> for Repr<B> {
#[inline]
fn abs_cmp(&self, other: &$T) -> Ordering {
$method::<B, true>(self, other)
}
}
impl<const B: Word> AbsOrd<Repr<B>> for $T {
#[inline]
fn abs_cmp(&self, other: &Repr<B>) -> Ordering {
$method::<B, true>(other, self).reverse()
}
}
impl<R: Round, const B: Word> AbsOrd<$T> for FBig<R, B> {
#[inline]
fn abs_cmp(&self, other: &$T) -> Ordering {
$method::<B, true>(&self.repr, other)
}
}
impl<R: Round, const B: Word> AbsOrd<FBig<R, B>> for $T {
#[inline]
fn abs_cmp(&self, other: &FBig<R, B>) -> Ordering {
$method::<B, true>(&other.repr, self).reverse()
}
}
};
}
impl_abs_ord_with_method!(UBig, repr_cmp_ubig);
impl_abs_ord_with_method!(IBig, repr_cmp_ibig);