extern "C" { fn extF80M_add(x: *const F80, y: *const F80, ptr: *mut F80); fn extF80M_sub(x: *const F80, y: *const F80, ptr: *mut F80); fn extF80M_mul(x: *const F80, y: *const F80, ptr: *mut F80); fn extF80M_div(x: *const F80, y: *const F80, ptr: *mut F80); //fn extF80M_rem(x: *const F80, y: *const F80, ptr: *mut F80); fn extF80M_sqrt(x: *const F80, ptr: *mut F80); fn extF80M_roundToInt(x: *const F80, rounding_mode: u8, raise_inexact: bool, dst: *mut F80); fn extF80M_eq(x: *const F80, y: *const F80) -> bool; //fn extF80M_eq_signaling(x: *const F80, y: *const F80) -> bool; //fn extF80M_le(x: *const F80, y: *const F80) -> bool; //fn extF80M_le_quiet(x: *const F80, y: *const F80) -> bool; fn extF80M_lt(x: *const F80, y: *const F80) -> bool; fn extF80M_lt_quiet(x: *const F80, y: *const F80) -> bool; fn extF80M_to_i32(src: *const F80, rounding_mode: u8, raise_inexact: bool) -> i32; fn extF80M_to_i64(src: *const F80, rounding_mode: u8, raise_inexact: bool) -> i64; fn i32_to_extF80M(src: i32, dst: *mut F80); fn i64_to_extF80M(src: i64, dst: *mut F80); fn f32_to_extF80M(src: i32, dst: *mut F80); fn f64_to_extF80M(src: u64, dst: *mut F80); fn extF80M_to_f32(src: *const F80) -> i32; fn extF80M_to_f64(src: *const F80) -> u64; static mut softfloat_roundingMode: u8; static mut extF80_roundingPrecision: u8; static mut softfloat_exceptionFlags: u8; } pub enum RoundingMode { NearEven, Trunc, Floor, Ceil, } pub enum Precision { P80, P64, P32, } #[repr(C)] #[derive(Copy, Clone)] pub struct F80 { pub mantissa: u64, pub sign_exponent: u16, } impl F80 { pub const ZERO: F80 = F80 { mantissa: 0, sign_exponent: 0, }; pub const ONE: F80 = F80 { mantissa: 0x8000000000000000, sign_exponent: 0x3FFF, }; pub const LN_10: F80 = F80 { mantissa: 0x935D8DDDAAA8B000, sign_exponent: 0x4000, }; pub const LN_2: F80 = F80 { mantissa: 0xB17217F7D1CF7800, sign_exponent: 0x3FFE, }; pub const PI: F80 = F80 { mantissa: 0xC90FDAA22168C000, sign_exponent: 0x4000, }; pub const LOG2_E: F80 = F80 { mantissa: 0xB8AA3B295C17F000, sign_exponent: 0x3FFF, }; pub const INDEFINITE_NAN: F80 = F80 { mantissa: 0xC000000000000000, sign_exponent: 0x7FFF, }; pub const POS_INFINITY: F80 = F80 { mantissa: 0x8000000000000000, sign_exponent: 0x7FFF, }; pub const NEG_INFINITY: F80 = F80 { mantissa: 0x8000000000000000, sign_exponent: 0xFFFF, }; pub fn sign(&self) -> bool { (self.sign_exponent >> 15) == 1 } pub fn exponent(&self) -> i16 { (self.sign_exponent as i16 & 0x7FFF) - 0x3FFF } pub fn of_i32(src: i32) -> F80 { let mut x = F80::ZERO; unsafe { i32_to_extF80M(src, &mut x) }; x } pub fn of_i64(src: i64) -> F80 { let mut x = F80::ZERO; unsafe { i64_to_extF80M(src, &mut x) }; x } pub fn of_f32(src: i32) -> F80 { let mut x = F80::ZERO; unsafe { f32_to_extF80M(src, &mut x) }; x } pub fn of_f64(src: u64) -> F80 { let mut x = F80::ZERO; unsafe { f64_to_extF80M(src, &mut x) }; x } fn of_f64x(src: f64) -> F80 { F80::of_f64(f64::to_bits(src)) } pub fn to_f32(&self) -> i32 { unsafe { extF80M_to_f32(self) } } pub fn to_f64(&self) -> u64 { unsafe { extF80M_to_f64(self) } } fn to_f64x(&self) -> f64 { f64::from_bits(self.to_f64()) } pub fn to_i32(&self) -> i32 { unsafe { extF80M_to_i32(self, softfloat_roundingMode, false) } } pub fn to_i64(&self) -> i64 { unsafe { extF80M_to_i64(self, softfloat_roundingMode, false) } } pub fn truncate_to_i32(&self) -> i32 { unsafe { extF80M_to_i32(self, 1, false) } } pub fn truncate_to_i64(&self) -> i64 { unsafe { extF80M_to_i64(self, 1, false) } } pub fn cos(self) -> F80 { F80::of_f64x(self.to_f64x().cos()) } pub fn sin(self) -> F80 { F80::of_f64x(self.to_f64x().sin()) } pub fn tan(self) -> F80 { F80::of_f64x(self.to_f64x().tan()) } pub fn atan(self) -> F80 { F80::of_f64x(self.to_f64x().atan()) } pub fn atan2(self, other: F80) -> F80 { F80::of_f64x(self.to_f64x().atan2(other.to_f64x())) } pub fn log2(self) -> F80 { F80::of_f64x(self.to_f64x().log2()) } pub fn ln(self) -> F80 { F80::of_f64x(self.to_f64x().ln()) } pub fn abs(self) -> F80 { F80 { mantissa: self.mantissa, sign_exponent: self.sign_exponent & !0x8000, } } pub fn two_pow(self) -> F80 { F80::of_f64x(2.0f64.powf(self.to_f64x())) } pub fn round(self) -> F80 { let mut result = F80::ZERO; unsafe { extF80M_roundToInt(&self, softfloat_roundingMode, false, &mut result) }; result } pub fn trunc(self) -> F80 { let mut result = F80::ZERO; unsafe { extF80M_roundToInt(&self, 1, false, &mut result) }; result } pub fn sqrt(self) -> F80 { let mut result = F80::ZERO; unsafe { extF80M_sqrt(&self, &mut result) }; result } pub fn is_finite(self) -> bool { // TODO: Can probably be done more efficiently self != F80::POS_INFINITY && self != F80::NEG_INFINITY } pub fn is_nan(self) -> bool { // TODO: Can probably be done more efficiently self != self } pub fn set_rounding_mode(mode: RoundingMode) { unsafe { softfloat_roundingMode = match mode { RoundingMode::NearEven => 0, RoundingMode::Trunc => 1, RoundingMode::Floor => 2, RoundingMode::Ceil => 3, } }; } pub fn set_precision(precision: Precision) { unsafe { extF80_roundingPrecision = match precision { Precision::P80 => 80, Precision::P64 => 64, Precision::P32 => 32, } }; } pub fn get_exception_flags() -> u8 { let f = unsafe { softfloat_exceptionFlags }; // translate softfloat's flags to x87 status flags f >> 4 & 1 | f >> 1 & 4 | f << 3 & 16 } pub fn clear_exception_flags() { unsafe { softfloat_exceptionFlags = 0 } } pub fn partial_cmp_quiet(&self, other: &Self) -> Option { // TODO: Can probably be done more efficiently if unsafe { extF80M_lt_quiet(self, other) } { Some(std::cmp::Ordering::Less) } else if unsafe { extF80M_lt_quiet(other, self) } { Some(std::cmp::Ordering::Greater) } else if self == other { Some(std::cmp::Ordering::Equal) } else { None } } } impl std::ops::Add for F80 { type Output = F80; fn add(self, other: Self) -> Self { let mut result = F80::ZERO; unsafe { extF80M_add(&self, &other, &mut result) }; result } } impl std::ops::Sub for F80 { type Output = F80; fn sub(self, other: Self) -> Self { let mut result = F80::ZERO; unsafe { extF80M_sub(&self, &other, &mut result) }; result } } impl std::ops::Neg for F80 { type Output = F80; fn neg(self) -> Self { let mut result = self; result.sign_exponent ^= 1 << 15; result } } impl std::ops::Mul for F80 { type Output = F80; fn mul(self, other: Self) -> Self { let mut result = F80::ZERO; unsafe { extF80M_mul(&self, &other, &mut result) }; result } } impl std::ops::Div for F80 { type Output = F80; fn div(self, other: Self) -> Self { let mut result = F80::ZERO; unsafe { extF80M_div(&self, &other, &mut result) }; result } } impl std::ops::Rem for F80 { type Output = F80; fn rem(self, other: Self) -> Self { let quot = (self / other).trunc(); self - quot * other // Uses round-to-nearest instead of truncation //let mut result = F80::ZERO; //unsafe { // extF80M_rem(&self, &other, &mut result) //}; //result } } impl PartialEq for F80 { fn eq(&self, other: &Self) -> bool { unsafe { extF80M_eq(self, other) } } } impl PartialOrd for F80 { fn partial_cmp(&self, other: &Self) -> Option { // TODO: Can probably be done more efficiently if unsafe { extF80M_lt(self, other) } { Some(std::cmp::Ordering::Less) } else if unsafe { extF80M_lt(other, self) } { Some(std::cmp::Ordering::Greater) } else if self == other { Some(std::cmp::Ordering::Equal) } else { None } } }