artichoke_backend/extn/core/float/
mod.rs

1use std::fmt;
2
3use crate::extn::core::numeric::{self, Coercion, Outcome};
4use crate::extn::prelude::*;
5
6pub(in crate::extn) mod mruby;
7
8#[repr(transparent)]
9#[derive(Default, Clone, Copy, PartialEq, PartialOrd)]
10pub struct Float(f64);
11
12impl fmt::Debug for Float {
13    fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
14        self.0.fmt(f)
15    }
16}
17
18impl fmt::Display for Float {
19    fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
20        self.0.fmt(f)
21    }
22}
23
24impl ConvertMut<Float, Value> for Artichoke {
25    #[inline]
26    fn convert_mut(&mut self, from: Float) -> Value {
27        self.convert_mut(from.0)
28    }
29}
30
31impl TryConvert<Value, Float> for Artichoke {
32    type Error = Error;
33
34    #[inline]
35    fn try_convert(&self, value: Value) -> Result<Float, Self::Error> {
36        let num = self.try_convert(value)?;
37        Ok(Float(num))
38    }
39}
40
41impl From<f64> for Float {
42    #[inline]
43    fn from(flt: f64) -> Self {
44        Self(flt)
45    }
46}
47
48impl From<Float> for f64 {
49    #[inline]
50    fn from(flt: Float) -> Self {
51        flt.as_f64()
52    }
53}
54
55impl From<Float> for Outcome {
56    #[inline]
57    fn from(flt: Float) -> Self {
58        Self::Float(flt.into())
59    }
60}
61
62impl From<f64> for Outcome {
63    #[inline]
64    fn from(flt: f64) -> Self {
65        Self::Float(flt)
66    }
67}
68
69impl Float {
70    /// The minimum number of significant decimal digits in a double-precision
71    /// floating point.
72    ///
73    /// Usually defaults to 15.
74    pub const DIG: i64 = f64::DIGITS as i64;
75
76    /// The difference between 1 and the smallest double-precision floating
77    /// point number greater than 1.
78    ///
79    /// Usually defaults to 2.2204460492503131e-16.
80    pub const EPSILON: f64 = f64::EPSILON;
81
82    /// An expression representing positive infinity.
83    pub const INFINITY: f64 = f64::INFINITY;
84
85    /// The minimum number of significant decimal digits in a double-precision
86    /// floating point.
87    ///
88    /// Usually defaults to 15.
89    pub const MANT_DIG: i64 = f64::MANTISSA_DIGITS as i64;
90
91    /// The largest possible integer in a double-precision floating point
92    /// number.
93    ///
94    /// Usually defaults to `1.7976931348623157e+308`.
95    pub const MAX: f64 = f64::MAX;
96
97    /// The largest positive exponent in a double-precision floating point where
98    /// 10 raised to this power minus 1.
99    ///
100    /// Usually defaults to 308.
101    pub const MAX_10_EXP: i64 = f64::MAX_10_EXP as i64;
102
103    /// The largest possible exponent value in a double-precision floating
104    /// point.
105    ///
106    /// Usually defaults to 1024.
107    pub const MAX_EXP: i64 = f64::MAX_EXP as i64;
108
109    /// The smallest positive normalized number in a double-precision floating
110    /// point.
111    ///
112    /// Usually defaults to 2.2250738585072014e-308.
113    ///
114    /// If the platform supports denormalized numbers, there are numbers between
115    /// zero and [`Float::MIN`]. `0.0.next_float` returns the smallest positive
116    /// floating point number including denormalized numbers.
117    pub const MIN: f64 = f64::MIN;
118
119    /// The smallest negative exponent in a double-precision floating point
120    /// where 10 raised to this power minus 1.
121    ///
122    /// Usually defaults to -307.
123    pub const MIN_10_EXP: i64 = f64::MIN_10_EXP as i64;
124
125    /// The smallest possible exponent value in a double-precision floating
126    /// point.
127    ///
128    /// Usually defaults to -1021.
129    pub const MIN_EXP: i64 = f64::MIN_EXP as i64;
130
131    /// An expression representing a value which is "not a number".
132    pub const NAN: f64 = f64::NAN;
133
134    pub const NEG_INFINITY: f64 = f64::NEG_INFINITY;
135
136    /// The base of the floating point, or number of unique digits used to
137    /// represent the number.
138    ///
139    /// Usually defaults to 2 on most systems, which would represent a base-10
140    /// decimal.
141    pub const RADIX: i64 = f64::RADIX as i64;
142
143    /// Represents the rounding mode for floating point addition.
144    ///
145    /// Usually defaults to 1, rounding to the nearest number.
146    ///
147    /// Other modes include:
148    ///
149    /// | mode                               | value |
150    /// |------------------------------------|-------|
151    /// | Indeterminable                     | -1    |
152    /// | Rounding towards zero              | 0     |
153    /// | Rounding to the nearest number     | 1     |
154    /// | Rounding towards positive infinity | 2     |
155    /// | Rounding towards negative infinity | 3     |
156    ///
157    /// # Rust Caveats
158    ///
159    /// Rust does not support setting the rounding mode and the behavior from
160    /// LLVM is not documented.  Because of this uncertainty, Artichoke sets its
161    /// rounding mode to `-1`, Indeterminable.
162    ///
163    /// The Rust docs say [`f64::round`][round] rounds "half-way cases away from
164    /// 0.0." Stack Overflow has a
165    /// [discussion around float rounding semantics][stackoverflow] in Rust and
166    /// LLVM.
167    ///
168    /// [stackoverflow]: https://stackoverflow.com/a/28122536
169    /// [round]: https://doc.rust-lang.org/1.42.0/std/primitive.f64.html#method.round
170    pub const ROUNDS: i64 = -1;
171
172    /// Construct a new, zero, float.
173    #[inline]
174    #[must_use]
175    pub const fn new() -> Self {
176        Self(0.0)
177    }
178
179    /// Construct a new `Float` with a given [`f64`].
180    #[inline]
181    #[must_use]
182    pub const fn with_f64(f: f64) -> Self {
183        Self(f)
184    }
185
186    /// Convert self to an `i64` with a saturating cast.
187    #[inline]
188    #[must_use]
189    #[expect(clippy::cast_possible_truncation, reason = "Ruby Float#to_i intentionally truncates")]
190    pub const fn as_i64(self) -> i64 {
191        self.0 as i64
192    }
193
194    /// Return the inner [`f64`].
195    #[inline]
196    #[must_use]
197    pub const fn as_f64(self) -> f64 {
198        self.0
199    }
200
201    #[inline]
202    #[must_use]
203    #[expect(
204        clippy::cast_precision_loss,
205        clippy::cast_possible_truncation,
206        reason = "consts are in f64 lossless range"
207    )]
208    pub fn try_into_fixnum(self) -> Option<i64> {
209        const FIXABLE_MAX: f64 = 2_i64.pow(f64::MANTISSA_DIGITS) as f64;
210        const FIXABLE_MIN: f64 = -(2_i64.pow(f64::MANTISSA_DIGITS)) as f64;
211
212        match self.0 {
213            x if !x.is_finite() => None,
214            x if x > FIXABLE_MAX => None,
215            x if x < FIXABLE_MIN => None,
216            x => Some(x as i64),
217        }
218    }
219
220    /// Compute the remainder of self and other.
221    ///
222    /// Equivalent to `self.as_f64() % other.as_f64()`.
223    #[inline]
224    #[must_use]
225    pub fn modulo(self, other: Self) -> Self {
226        Self(self.0 % other.0)
227    }
228
229    #[inline]
230    pub fn coerced_modulo(self, interp: &mut Artichoke, other: Value) -> Result<Outcome, Error> {
231        if let Ruby::Float = other.ruby_type() {
232            let other = other.try_convert_into::<Float>(interp)?;
233            return Ok(self.modulo(other).into());
234        }
235        let x = interp.convert_mut(self);
236        let coerced = numeric::coerce(interp, x, other)?;
237        match coerced {
238            Coercion::Float(x, y) => Ok((x % y).into()),
239            Coercion::Integer(x, y) => Ok((x % y).into()),
240        }
241    }
242}