use std::cmp::Ordering;
use std::convert::TryFrom;
use std::convert::TryInto;
use std::ffi::{CStr, CString};
use std::fmt;
use std::iter::{Product, Sum};
use std::marker::PhantomData;
use std::mem::MaybeUninit;
use std::ops::{
Add, AddAssign, Div, DivAssign, Mul, MulAssign, Neg, Rem, RemAssign, Sub, SubAssign,
};
use std::str::FromStr;
use libc::c_char;
#[cfg(feature = "num-traits")]
use num_traits::{MulAdd, MulAddAssign, One, Zero};
use crate::context::{Class, Context};
use crate::decimal::Decimal;
use crate::decimal128::Decimal128;
use crate::decimal32::Decimal32;
use crate::error::ParseDecimalError;
#[repr(transparent)]
#[derive(Clone, Copy)]
pub struct Decimal64 {
pub(crate) inner: decnumber_sys::decDouble,
}
impl Decimal64 {
pub const NAN: Decimal64 = Decimal64::from_ne_bytes(if cfg!(target_endian = "little") {
[0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x7c]
} else {
[0x7c, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0]
});
pub const ZERO: Decimal64 = Decimal64::from_ne_bytes(if cfg!(target_endian = "little") {
[0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x38, 0x22]
} else {
[0x22, 0x38, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0]
});
pub const ONE: Decimal64 = Decimal64::from_ne_bytes(if cfg!(target_endian = "little") {
[0x1, 0x0, 0x0, 0x0, 0x0, 0x0, 0x38, 0x22]
} else {
[0x22, 0x38, 0x0, 0x0, 0x0, 0x0, 0x0, 0x1]
});
const TWO_POW_32: Decimal64 = Decimal64::from_ne_bytes(if cfg!(target_endian = "little") {
[0x7A, 0xB5, 0xAF, 0x15, 0x1, 0x0, 0x38, 0x22]
} else {
[0x22, 0x38, 0x0, 0x1, 0x15, 0xAF, 0xB5, 0x7A]
});
pub fn from_le_bytes(mut bytes: [u8; 8]) -> Decimal64 {
if cfg!(target_endian = "big") {
bytes.reverse();
}
Decimal64::from_ne_bytes(bytes)
}
pub fn from_be_bytes(mut bytes: [u8; 8]) -> Decimal64 {
if cfg!(target_endian = "little") {
bytes.reverse();
}
Decimal64::from_ne_bytes(bytes)
}
pub const fn from_ne_bytes(bytes: [u8; 8]) -> Decimal64 {
Decimal64 {
inner: decnumber_sys::decDouble { bytes },
}
}
pub fn to_le_bytes(&self) -> [u8; 8] {
let mut bytes = self.to_ne_bytes();
if cfg!(target_endian = "big") {
bytes.reverse();
}
bytes
}
pub fn to_be_bytes(&self) -> [u8; 8] {
let mut bytes = self.to_ne_bytes();
if cfg!(target_endian = "little") {
bytes.reverse();
}
bytes
}
pub fn to_ne_bytes(&self) -> [u8; 8] {
self.inner.bytes
}
pub fn class(&self) -> Class {
Class::from_c(unsafe { decnumber_sys::decDoubleClass(&self.inner) })
}
pub fn digits(&self) -> u32 {
unsafe { decnumber_sys::decDoubleDigits(&self.inner) }
}
pub fn coefficient(&self) -> i64 {
let mut dpd = if cfg!(target_endian = "big") {
u64::from_be_bytes(self.inner.bytes)
} else {
u64::from_le_bytes(self.inner.bytes)
};
let dpd_mask = 0b11_1111_1111;
let mut r = i64::from(unsafe { decnumber_sys::DPD2BIN[dpd as usize & dpd_mask] });
dpd >>= 10;
r += i64::from(unsafe { decnumber_sys::DPD2BINK[dpd as usize & dpd_mask] });
dpd >>= 10;
r += i64::from(unsafe { decnumber_sys::DPD2BINM[dpd as usize & dpd_mask] });
dpd >>= 10;
let mut r_1: i64 = 0;
r_1 += i64::from(unsafe { decnumber_sys::DPD2BIN[dpd as usize & dpd_mask] });
dpd >>= 10;
r_1 += i64::from(unsafe { decnumber_sys::DPD2BINK[dpd as usize & dpd_mask] });
r += r_1 * 10_000_000_00;
let h = i64::from(unsafe { decnumber_sys::DECCOMBMSD[(dpd >> 18) as usize] });
if h > 0 {
r += h * 1_000_000_000_000_000;
}
if self.is_negative() {
r *= -1;
}
r
}
pub fn coefficient_digits(&self) -> [u8; decnumber_sys::DECDOUBLE_Pmax] {
let mut buf = [0u8; decnumber_sys::DECDOUBLE_Pmax];
unsafe {
decnumber_sys::decDoubleGetCoefficient(&self.inner, buf.as_mut_ptr() as *mut u8);
}
buf
}
pub fn exponent(&self) -> i32 {
unsafe { decnumber_sys::decDoubleGetExponent(&self.inner) }
}
pub fn canonical(mut self) -> Decimal64 {
unsafe {
decnumber_sys::decDoubleCanonical(&mut self.inner, &self.inner);
}
self
}
pub fn is_canonical(&self) -> bool {
unsafe { decnumber_sys::decDoubleIsCanonical(&self.inner) != 0 }
}
pub fn is_finite(&self) -> bool {
unsafe { decnumber_sys::decDoubleIsFinite(&self.inner) != 0 }
}
pub fn is_infinite(&self) -> bool {
unsafe { decnumber_sys::decDoubleIsInfinite(&self.inner) != 0 }
}
pub fn is_integer(&self) -> bool {
unsafe { decnumber_sys::decDoubleIsInteger(&self.inner) != 0 }
}
pub fn is_logical(&self) -> bool {
unsafe { decnumber_sys::decDoubleIsInteger(&self.inner) != 0 }
}
pub fn is_nan(&self) -> bool {
unsafe { decnumber_sys::decDoubleIsNaN(&self.inner) != 0 }
}
pub fn is_negative(&self) -> bool {
unsafe { decnumber_sys::decDoubleIsNegative(&self.inner) != 0 }
}
pub fn is_normal(&self) -> bool {
unsafe { decnumber_sys::decDoubleIsNormal(&self.inner) != 0 }
}
pub fn is_positive(&self) -> bool {
unsafe { decnumber_sys::decDoubleIsPositive(&self.inner) != 0 }
}
pub fn is_signaling_nan(&self) -> bool {
unsafe { decnumber_sys::decDoubleIsSignaling(&self.inner) != 0 }
}
pub fn is_signed(&self) -> bool {
unsafe { decnumber_sys::decDoubleIsSigned(&self.inner) != 0 }
}
pub fn is_subnormal(&self) -> bool {
unsafe { decnumber_sys::decDoubleIsSubnormal(&self.inner) != 0 }
}
pub fn is_zero(&self) -> bool {
unsafe { decnumber_sys::decDoubleIsZero(&self.inner) != 0 }
}
pub fn quantum_matches(&self, rhs: &Decimal64) -> bool {
unsafe { decnumber_sys::decDoubleSameQuantum(&self.inner, &rhs.inner) != 0 }
}
pub fn total_cmp(&self, rhs: &Decimal64) -> Ordering {
let mut d = Decimal64::ZERO;
unsafe {
decnumber_sys::decDoubleCompareTotal(&mut d.inner, &self.inner, &rhs.inner);
}
if d.is_positive() {
Ordering::Greater
} else if d.is_negative() {
Ordering::Less
} else {
debug_assert!(d.is_zero());
Ordering::Equal
}
}
pub fn to_standard_notation_string(&self) -> String {
if !self.is_finite() {
return self.to_string();
}
let mut digits = [b'0'; decnumber_sys::DECDOUBLE_Pmax];
let mut digits_idx = 0;
let (sourlo, sourhi) = if cfg!(target_endian = "little") {
(
u32::from_le_bytes(self.inner.bytes[0..4].try_into().unwrap()) as usize,
u32::from_le_bytes(self.inner.bytes[4..8].try_into().unwrap()) as usize,
)
} else {
(
u32::from_be_bytes(self.inner.bytes[4..8].try_into().unwrap()) as usize,
u32::from_be_bytes(self.inner.bytes[0..4].try_into().unwrap()) as usize,
)
};
let comb = ((sourhi >> 26) & 0x1f) as usize;
let msd = unsafe { decnumber_sys::DECCOMBMSD[comb] };
if msd > 0 {
digits[digits_idx] = b'0' + msd as u8;
digits_idx += 1;
}
#[allow(unused_assignments)]
let mut dpd: usize = 0;
dpd = (sourhi >> 8) & 0x3ff; dpd2char!(dpd, digits, digits_idx);
dpd = ((sourhi & 0xff) << 2) | (sourlo >> 30); dpd2char!(dpd, digits, digits_idx);
dpd = (sourlo >> 20) & 0x3ff; dpd2char!(dpd, digits, digits_idx);
dpd = (sourlo >> 10) & 0x3ff; dpd2char!(dpd, digits, digits_idx);
dpd = (sourlo) & 0x3ff; dpd2char!(dpd, digits, digits_idx);
stringify_digits!(self, digits, digits_idx)
}
}
#[test]
fn test() {
let mut cx = Context::<Decimal64>::default();
let d = cx.parse("0").unwrap();
println!("{:?}", d.to_standard_notation_string());
}
impl Default for Decimal64 {
fn default() -> Decimal64 {
Decimal64::ZERO
}
}
impl fmt::Debug for Decimal64 {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
fmt::Display::fmt(self, f)
}
}
impl fmt::Display for Decimal64 {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
let mut buf = ['\0'; decnumber_sys::DECDOUBLE_String];
let c_str = unsafe {
if f.alternate() {
decnumber_sys::decDoubleToEngString(&self.inner, buf.as_mut_ptr() as *mut c_char);
} else {
decnumber_sys::decDoubleToString(&self.inner, buf.as_mut_ptr() as *mut c_char);
}
CStr::from_ptr(buf.as_ptr() as *const c_char)
};
f.write_str(
c_str
.to_str()
.expect("decDoubleToString yields valid UTF-8"),
)
}
}
impl FromStr for Decimal64 {
type Err = ParseDecimalError;
fn from_str(s: &str) -> Result<Decimal64, ParseDecimalError> {
Context::<Decimal64>::default().parse(s)
}
}
impl From<i32> for Decimal64 {
fn from(n: i32) -> Decimal64 {
let mut d = Decimal64::ZERO;
unsafe {
decnumber_sys::decDoubleFromInt32(&mut d.inner, n);
};
d
}
}
impl From<u32> for Decimal64 {
fn from(n: u32) -> Decimal64 {
let mut d = Decimal64::ZERO;
unsafe {
decnumber_sys::decDoubleFromUInt32(&mut d.inner, n);
};
d
}
}
impl From<Decimal32> for Decimal64 {
fn from(d32: Decimal32) -> Decimal64 {
let mut d64 = Decimal64::ZERO;
unsafe {
decnumber_sys::decSingleToWider(&d32.inner, &mut d64.inner);
}
d64
}
}
impl PartialOrd for Decimal64 {
fn partial_cmp(&self, other: &Self) -> Option<Ordering> {
Context::<Decimal64>::default().partial_cmp(*self, *other)
}
}
impl PartialEq for Decimal64 {
fn eq(&self, other: &Self) -> bool {
self.partial_cmp(other) == Some(Ordering::Equal)
}
}
impl Neg for Decimal64 {
type Output = Decimal64;
fn neg(self) -> Decimal64 {
Context::<Decimal64>::default().minus(self)
}
}
impl Add<Decimal64> for Decimal64 {
type Output = Decimal64;
fn add(self, rhs: Decimal64) -> Decimal64 {
Context::<Decimal64>::default().add(self, rhs)
}
}
impl AddAssign<Decimal64> for Decimal64 {
fn add_assign(&mut self, rhs: Decimal64) {
*self = Context::<Decimal64>::default().add(*self, rhs);
}
}
impl Div<Decimal64> for Decimal64 {
type Output = Decimal64;
fn div(self, rhs: Decimal64) -> Decimal64 {
Context::<Decimal64>::default().div(self, rhs)
}
}
impl DivAssign<Decimal64> for Decimal64 {
fn div_assign(&mut self, rhs: Decimal64) {
*self = Context::<Decimal64>::default().div(*self, rhs);
}
}
impl Mul<Decimal64> for Decimal64 {
type Output = Decimal64;
fn mul(self, rhs: Decimal64) -> Decimal64 {
Context::<Decimal64>::default().mul(self, rhs)
}
}
impl MulAssign<Decimal64> for Decimal64 {
fn mul_assign(&mut self, rhs: Decimal64) {
*self = Context::<Decimal64>::default().mul(*self, rhs);
}
}
impl Rem<Decimal64> for Decimal64 {
type Output = Decimal64;
fn rem(self, rhs: Decimal64) -> Decimal64 {
Context::<Decimal64>::default().rem(self, rhs)
}
}
impl RemAssign<Decimal64> for Decimal64 {
fn rem_assign(&mut self, rhs: Decimal64) {
*self = Context::<Decimal64>::default().rem(*self, rhs);
}
}
impl Sub<Decimal64> for Decimal64 {
type Output = Decimal64;
fn sub(self, rhs: Decimal64) -> Decimal64 {
Context::<Decimal64>::default().sub(self, rhs)
}
}
impl SubAssign<Decimal64> for Decimal64 {
fn sub_assign(&mut self, rhs: Decimal64) {
*self = Context::<Decimal64>::default().sub(*self, rhs);
}
}
impl Sum for Decimal64 {
fn sum<I>(iter: I) -> Self
where
I: Iterator<Item = Decimal64>,
{
let mut cx = Context::<Decimal64>::default();
let mut sum = Decimal64::ZERO;
for d in iter {
sum = cx.add(sum, d);
}
sum
}
}
impl<'a> Sum<&'a Decimal64> for Decimal64 {
fn sum<I>(iter: I) -> Self
where
I: Iterator<Item = &'a Decimal64>,
{
iter.copied().sum()
}
}
impl Product for Decimal64 {
fn product<I>(iter: I) -> Self
where
I: Iterator<Item = Decimal64>,
{
let mut cx = Context::<Decimal64>::default();
let mut product = Decimal64::ONE;
for d in iter {
product = cx.mul(product, d);
}
product
}
}
impl<'a> Product<&'a Decimal64> for Decimal64 {
fn product<I>(iter: I) -> Self
where
I: Iterator<Item = &'a Decimal64>,
{
iter.copied().product()
}
}
impl Default for Context<Decimal64> {
fn default() -> Context<Decimal64> {
let mut ctx = MaybeUninit::<decnumber_sys::decContext>::uninit();
let ctx = unsafe {
decnumber_sys::decContextDefault(ctx.as_mut_ptr(), decnumber_sys::DEC_INIT_DECDOUBLE);
ctx.assume_init()
};
Context {
inner: ctx,
_phantom: PhantomData,
}
}
}
impl Context<Decimal64> {
pub fn parse<S>(&mut self, s: S) -> Result<Decimal64, ParseDecimalError>
where
S: Into<Vec<u8>>,
{
let c_string = CString::new(s).map_err(|_| ParseDecimalError)?;
let mut d = Decimal64::ZERO;
unsafe {
decnumber_sys::decDoubleFromString(&mut d.inner, c_string.as_ptr(), &mut self.inner);
};
if (self.inner.status & decnumber_sys::DEC_Conversion_syntax) != 0 {
Err(ParseDecimalError)
} else {
Ok(d)
}
}
pub fn from_decimal128(&mut self, d128: Decimal128) -> Decimal64 {
let mut d64 = Decimal64::ZERO;
unsafe {
decnumber_sys::decDoubleFromWider(&mut d64.inner, &d128.inner, &mut self.inner);
};
d64
}
pub fn from_decimal<const N: usize>(&mut self, d: &Decimal<N>) -> Decimal64 {
let mut d64 = Decimal64::ZERO;
unsafe {
decnumber_sys::decimal64FromNumber(&mut d64.inner, d.as_ptr(), &mut self.inner);
};
d64
}
pub fn from_i64(&mut self, n: i64) -> Decimal64 {
from_signed_int!(Decimal64, self, n)
}
pub fn from_u64(&mut self, n: u64) -> Decimal64 {
from_unsigned_int!(Decimal64, self, n)
}
pub fn abs(&mut self, mut n: Decimal64) -> Decimal64 {
let n_inner = &mut n.inner as *mut decnumber_sys::decDouble;
unsafe {
decnumber_sys::decDoubleAbs(n_inner, n_inner, &mut self.inner);
}
n
}
pub fn add(&mut self, mut lhs: Decimal64, rhs: Decimal64) -> Decimal64 {
let lhs_inner = &mut lhs.inner as *mut decnumber_sys::decDouble;
unsafe {
decnumber_sys::decDoubleAdd(lhs_inner, lhs_inner, &rhs.inner, &mut self.inner);
}
lhs
}
pub fn and(&mut self, mut lhs: Decimal64, rhs: Decimal64) -> Decimal64 {
let lhs_inner = &mut lhs.inner as *mut decnumber_sys::decDouble;
unsafe {
decnumber_sys::decDoubleAnd(lhs_inner, lhs_inner, &rhs.inner, &mut self.inner);
}
lhs
}
pub fn div(&mut self, mut lhs: Decimal64, rhs: Decimal64) -> Decimal64 {
let lhs_inner = &mut lhs.inner as *mut decnumber_sys::decDouble;
unsafe {
decnumber_sys::decDoubleDivide(lhs_inner, lhs_inner, &rhs.inner, &mut self.inner);
}
lhs
}
pub fn div_integer(&mut self, mut lhs: Decimal64, rhs: Decimal64) -> Decimal64 {
let lhs_inner = &mut lhs.inner as *mut decnumber_sys::decDouble;
unsafe {
decnumber_sys::decDoubleDivideInteger(
lhs_inner,
lhs_inner,
&rhs.inner,
&mut self.inner,
);
}
lhs
}
pub fn fma(&mut self, mut x: Decimal64, y: Decimal64, z: Decimal64) -> Decimal64 {
let x_inner = &mut x.inner as *mut decnumber_sys::decDouble;
unsafe {
decnumber_sys::decDoubleFMA(x_inner, x_inner, &y.inner, &z.inner, &mut self.inner);
}
x
}
pub fn invert(&mut self, mut n: Decimal64) -> Decimal64 {
let n_inner = &mut n.inner as *mut decnumber_sys::decDouble;
unsafe {
decnumber_sys::decDoubleInvert(n_inner, n_inner, &mut self.inner);
}
n
}
pub fn logb(&mut self, mut n: Decimal64) -> Decimal64 {
let n_inner = &mut n.inner as *mut decnumber_sys::decDouble;
unsafe {
decnumber_sys::decDoubleLogB(n_inner, n_inner, &mut self.inner);
}
n
}
pub fn max(&mut self, mut lhs: Decimal64, rhs: Decimal64) -> Decimal64 {
let lhs_inner = &mut lhs.inner as *mut decnumber_sys::decDouble;
unsafe {
decnumber_sys::decDoubleMax(lhs_inner, lhs_inner, &rhs.inner, &mut self.inner);
}
lhs
}
pub fn max_abs(&mut self, mut lhs: Decimal64, rhs: Decimal64) -> Decimal64 {
let lhs_inner = &mut lhs.inner as *mut decnumber_sys::decDouble;
unsafe {
decnumber_sys::decDoubleMaxMag(lhs_inner, lhs_inner, &rhs.inner, &mut self.inner);
}
lhs
}
pub fn min(&mut self, mut lhs: Decimal64, rhs: Decimal64) -> Decimal64 {
let lhs_inner = &mut lhs.inner as *mut decnumber_sys::decDouble;
unsafe {
decnumber_sys::decDoubleMin(lhs_inner, lhs_inner, &rhs.inner, &mut self.inner);
}
lhs
}
pub fn min_abs(&mut self, mut lhs: Decimal64, rhs: Decimal64) -> Decimal64 {
let lhs_inner = &mut lhs.inner as *mut decnumber_sys::decDouble;
unsafe {
decnumber_sys::decDoubleMinMag(lhs_inner, lhs_inner, &rhs.inner, &mut self.inner);
}
lhs
}
pub fn minus(&mut self, mut n: Decimal64) -> Decimal64 {
let n_inner = &mut n.inner as *mut decnumber_sys::decDouble;
unsafe {
decnumber_sys::decDoubleMinus(n_inner, n_inner, &mut self.inner);
}
n
}
pub fn mul(&mut self, mut lhs: Decimal64, rhs: Decimal64) -> Decimal64 {
unsafe {
decnumber_sys::decDoubleMultiply(
&mut lhs.inner,
&lhs.inner,
&rhs.inner,
&mut self.inner,
);
}
lhs
}
pub fn next_minus(&mut self, mut n: Decimal64) -> Decimal64 {
let n_inner = &mut n.inner as *mut decnumber_sys::decDouble;
unsafe {
decnumber_sys::decDoubleNextMinus(n_inner, n_inner, &mut self.inner);
}
n
}
pub fn next_plus(&mut self, mut n: Decimal64) -> Decimal64 {
let n_inner = &mut n.inner as *mut decnumber_sys::decDouble;
unsafe {
decnumber_sys::decDoubleNextPlus(n_inner, n_inner, &mut self.inner);
}
n
}
pub fn next_toward(&mut self, mut x: Decimal64, y: Decimal64) -> Decimal64 {
let x_inner = &mut x.inner as *mut decnumber_sys::decDouble;
unsafe {
decnumber_sys::decDoubleNextToward(x_inner, x_inner, &y.inner, &mut self.inner);
}
x
}
pub fn partial_cmp(&mut self, lhs: Decimal64, rhs: Decimal64) -> Option<Ordering> {
let mut d = Decimal64::ZERO;
unsafe {
decnumber_sys::decDoubleCompare(&mut d.inner, &lhs.inner, &rhs.inner, &mut self.inner);
}
if d.is_positive() {
Some(Ordering::Greater)
} else if d.is_negative() {
Some(Ordering::Less)
} else if d.is_zero() {
Some(Ordering::Equal)
} else {
debug_assert!(d.is_nan());
None
}
}
pub fn plus(&mut self, mut n: Decimal64) -> Decimal64 {
let n_inner = &mut n.inner as *mut decnumber_sys::decDouble;
unsafe {
decnumber_sys::decDoublePlus(n_inner, n_inner, &mut self.inner);
}
n
}
pub fn quantize(&mut self, mut lhs: Decimal64, rhs: Decimal64) -> Decimal64 {
let lhs_inner = &mut lhs.inner as *mut decnumber_sys::decDouble;
unsafe {
decnumber_sys::decDoubleQuantize(lhs_inner, lhs_inner, &rhs.inner, &mut self.inner);
}
lhs
}
pub fn reduce(&mut self, mut n: Decimal64) -> Decimal64 {
let n_inner = &mut n.inner as *mut decnumber_sys::decDouble;
unsafe {
decnumber_sys::decDoubleReduce(n_inner, n_inner, &mut self.inner);
}
n
}
pub fn rem(&mut self, mut lhs: Decimal64, rhs: Decimal64) -> Decimal64 {
let lhs_inner = &mut lhs.inner as *mut decnumber_sys::decDouble;
unsafe {
decnumber_sys::decDoubleRemainder(lhs_inner, lhs_inner, &rhs.inner, &mut self.inner);
}
lhs
}
pub fn rem_near(&mut self, mut lhs: Decimal64, rhs: Decimal64) -> Decimal64 {
let lhs_inner = &mut lhs.inner as *mut decnumber_sys::decDouble;
unsafe {
decnumber_sys::decDoubleRemainderNear(
lhs_inner,
lhs_inner,
&rhs.inner,
&mut self.inner,
);
}
lhs
}
pub fn rotate(&mut self, mut lhs: Decimal64, rhs: Decimal64) -> Decimal64 {
let lhs_inner = &mut lhs.inner as *mut decnumber_sys::decDouble;
unsafe {
decnumber_sys::decDoubleRotate(lhs_inner, lhs_inner, &rhs.inner, &mut self.inner);
}
lhs
}
pub fn round(&mut self, mut n: Decimal64) -> Decimal64 {
let n_inner = &mut n.inner as *mut decnumber_sys::decDouble;
unsafe {
decnumber_sys::decDoubleToIntegralExact(n_inner, n_inner, &mut self.inner);
}
n
}
pub fn scaleb(&mut self, mut x: Decimal64, y: Decimal64) -> Decimal64 {
let x_inner = &mut x.inner as *mut decnumber_sys::decDouble;
unsafe {
decnumber_sys::decDoubleScaleB(x_inner, x_inner, &y.inner, &mut self.inner);
}
x
}
pub fn set_exponent(&mut self, d: &mut Decimal64, e: i32) {
unsafe {
decnumber_sys::decDoubleSetExponent(&mut d.inner, &mut self.inner, e);
}
}
pub fn shift(&mut self, mut lhs: Decimal64, rhs: Decimal64) -> Decimal64 {
let lhs_inner = &mut lhs.inner as *mut decnumber_sys::decDouble;
unsafe {
decnumber_sys::decDoubleShift(lhs_inner, lhs_inner, &rhs.inner, &mut self.inner);
}
lhs
}
pub fn rescale(&mut self, x: &mut Decimal64, s: i32) {
let e = x.exponent();
*x = self.shift(*x, Decimal64::from(e - s));
self.set_exponent(x, s);
}
pub fn sub(&mut self, mut lhs: Decimal64, rhs: Decimal64) -> Decimal64 {
unsafe {
decnumber_sys::decDoubleSubtract(
&mut lhs.inner,
&lhs.inner,
&rhs.inner,
&mut self.inner,
);
}
lhs
}
pub fn or(&mut self, mut lhs: Decimal64, rhs: Decimal64) -> Decimal64 {
let lhs_inner = &mut lhs.inner as *mut decnumber_sys::decDouble;
unsafe {
decnumber_sys::decDoubleOr(lhs_inner, lhs_inner, &rhs.inner, &mut self.inner);
}
lhs
}
pub fn xor(&mut self, mut lhs: Decimal64, rhs: Decimal64) -> Decimal64 {
let lhs_inner = &mut lhs.inner as *mut decnumber_sys::decDouble;
unsafe {
decnumber_sys::decDoubleXor(lhs_inner, lhs_inner, &rhs.inner, &mut self.inner);
}
lhs
}
}
#[cfg(feature = "num-traits")]
impl One for Decimal64 {
#[inline]
fn one() -> Self {
Self::ONE
}
}
#[cfg(feature = "num-traits")]
impl Zero for Decimal64 {
#[inline]
fn zero() -> Self {
Self::ZERO
}
#[inline]
fn is_zero(&self) -> bool {
self.is_zero()
}
}
#[cfg(feature = "num-traits")]
impl MulAdd for Decimal64 {
type Output = Self;
fn mul_add(self, a: Self, b: Self) -> Self::Output {
Context::<Self>::default().fma(self, a, b)
}
}
#[cfg(feature = "num-traits")]
impl MulAddAssign for Decimal64 {
#[inline]
fn mul_add_assign(&mut self, a: Self, b: Self) {
*self = self.mul_add(a, b)
}
}