use std::fmt;
use rb_sys::{rb_float_new_in_heap, VALUE};
use crate::{
into_value::IntoValue,
numeric::Numeric,
try_convert::TryConvertOwned,
value::{
private::{self, ReprValue as _},
NonZeroValue, ReprValue,
},
Error, Float, RFloat, Ruby, TryConvert, Value,
};
impl Ruby {
#[inline]
pub fn flonum_from_f64(&self, n: f64) -> Result<Flonum, RFloat> {
Flonum::from_f64_impl(n)
.ok_or_else(|| unsafe { RFloat::from_rb_value_unchecked(rb_float_new_in_heap(n)) })
}
}
#[derive(Clone, Copy)]
#[repr(transparent)]
pub struct Flonum(NonZeroValue);
impl Flonum {
#[inline]
pub fn from_value(val: Value) -> Option<Self> {
unsafe {
val.is_flonum()
.then(|| Self(NonZeroValue::new_unchecked(val)))
}
}
#[inline]
pub(crate) unsafe fn from_rb_value_unchecked(val: VALUE) -> Self {
Self(NonZeroValue::new_unchecked(Value::new(val)))
}
#[inline]
pub(crate) fn from_f64_impl(d: f64) -> Option<Self> {
let v = d.to_bits();
let bits = (v >> 60) & 0x7;
if v != 0x3000000000000000 && bits.wrapping_sub(3) & !0x01 == 0 {
return Some(unsafe { Self::from_rb_value_unchecked(v.rotate_left(3) & !0x01 | 0x02) });
} else if v == 0 {
return Some(unsafe { Self::from_rb_value_unchecked(0x8000000000000002) });
}
None
}
#[cfg_attr(
not(feature = "old-api"),
deprecated(note = "please use `Ruby::flonum_from_f64` instead")
)]
#[cfg_attr(docsrs, doc(cfg(feature = "old-api")))]
#[inline]
pub fn from_f64(n: f64) -> Result<Self, RFloat> {
get_ruby!().flonum_from_f64(n)
}
#[inline]
pub fn to_f64(self) -> f64 {
let v = self.as_rb_value();
if v != 0x8000000000000002 {
let b63 = v >> 63;
let v = (2_u64.wrapping_sub(b63) | v & !0x03).rotate_right(3);
return f64::from_bits(v);
}
0.0
}
}
impl fmt::Display for Flonum {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
write!(f, "{}", unsafe { self.to_s_infallible() })
}
}
impl fmt::Debug for Flonum {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
write!(f, "{}", self.inspect())
}
}
impl IntoValue for Flonum {
#[inline]
fn into_value_with(self, _: &Ruby) -> Value {
self.0.get()
}
}
impl Numeric for Flonum {}
unsafe impl private::ReprValue for Flonum {}
impl ReprValue for Flonum {}
impl TryConvert for Flonum {
fn try_convert(val: Value) -> Result<Self, Error> {
let float = Float::try_convert(val)?;
if let Some(flonum) = Flonum::from_value(float.as_value()) {
Ok(flonum)
} else {
Err(Error::new(
Ruby::get_with(val).exception_range_error(),
"float out of range for flonum",
))
}
}
}
unsafe impl TryConvertOwned for Flonum {}