decimal-scaled 0.4.1

Const-generic base-10 fixed-point decimals (D9/D18/D38/D76/D153/D307) with integer-only transcendentals correctly rounded to within 0.5 ULP — exact at the type's last representable place. Deterministic across every platform; no_std-friendly.
Documentation
//! Generic wide-integer square-root kernel — `isqrt(raw · 10^SCALE)`
//! with a single round step.
//!
//! For a `Dxx<SCALE>` value with raw storage `r`, the logical value is
//! `r / 10^SCALE`, so the square-root raw storage is
//! `round(sqrt(r · 10^SCALE))`. The radicand is formed exactly in a
//! wider integer (so the multiply by `10^SCALE` cannot overflow), the
//! exact integer square root is taken with `isqrt`, and a single
//! round-to-nearest step lands the result on the type's last
//! representable place. The result is within 0.5 ULP under any of the
//! six [`crate::support::rounding::RoundingMode`] values.
//!
//! # Generic kernel + per-tier shims
//!
//! The kernel is parameterised over `(Storage, SqrtWide)` via the
//! [`crate::wide_int::WideStorage`] trait, which exposes the unifrom
//! surface (`ZERO` / `ONE` / `TEN`, `pow`, `isqrt`, `resize_to`, the
//! standard arithmetic ops) every wide signed integer in the family
//! ships. The kernel collapses to a single function:
//!
//! - [`sqrt`] — `pub(crate) fn sqrt<S, W>(raw: S, scale, mode) -> S`
//!   where `S: WideStorage` is the storage type and
//!   `W: WideStorage` is the next-up width used to form the radicand
//!   without overflow.
//!
//! Per-tier free functions ([`sqrt_d57`], [`sqrt_d76`], …) remain as
//! thin shims so the policy callers don't need to know about the
//! `(Storage, SqrtWide)` pairing — each shim picks the right `W` for
//! its tier and forwards to [`sqrt`].
//!
//! Each function returns `Storage::ZERO` for `raw <= 0` (the
//! saturate-not-panic policy matching the typed `sqrt_strict` surface).

use crate::support::rounding::RoundingMode;
use crate::wide_int::WideStorage;

/// Generic square-root kernel for the wide-integer family.
///
/// `S` is the storage type backing `Dxx<SCALE>` and `W` is the
/// next-up width used to form `raw · 10^scale` without overflow.
/// See the module docs for the full pipeline.
#[inline]
#[must_use]
pub(crate) fn sqrt<S, W>(raw: S, scale: u32, mode: RoundingMode) -> S
where
    S: WideStorage,
    W: WideStorage,
{
    if raw <= S::ZERO {
        return S::ZERO;
    }
    let n: W = raw.resize_to::<W>() * W::TEN.pow(scale);
    let q: W = n.isqrt();
    let diff: W = n - q * q;
    let halfway_round_up = diff > q;
    let diff_nonzero = diff != W::ZERO;
    let bump = match mode {
        RoundingMode::HalfToEven
        | RoundingMode::HalfAwayFromZero
        | RoundingMode::HalfTowardZero => halfway_round_up,
        RoundingMode::Trunc | RoundingMode::Floor => false,
        RoundingMode::Ceiling => diff_nonzero,
    };
    let q = if bump { q + W::ONE } else { q };
    q.resize_to::<S>()
}

/// Emits a `sqrt_<tier>(raw, scale, mode) -> Storage` shim around
/// the generic [`sqrt`] kernel.
macro_rules! decl_sqrt_kernel_shim {
    ($name:ident, $Storage:ty, $SqrtWide:ty) => {
        /// Square-root kernel shim for one wide tier. See module docs.
        #[inline]
        #[must_use]
        pub(crate) fn $name(raw: $Storage, scale: u32, mode: RoundingMode) -> $Storage {
            sqrt::<$Storage, $SqrtWide>(raw, scale, mode)
        }
    };
}

#[cfg(any(feature = "d57", feature = "wide"))]
decl_sqrt_kernel_shim!(sqrt_d57, crate::wide_int::Int192, crate::wide_int::Int384);

#[cfg(any(feature = "d76", feature = "wide"))]
decl_sqrt_kernel_shim!(sqrt_d76, crate::wide_int::Int256, crate::wide_int::Int512);

#[cfg(any(feature = "d115", feature = "wide"))]
decl_sqrt_kernel_shim!(sqrt_d115, crate::wide_int::Int384, crate::wide_int::Int768);

#[cfg(any(feature = "d153", feature = "wide"))]
decl_sqrt_kernel_shim!(sqrt_d153, crate::wide_int::Int512, crate::wide_int::Int1024);

#[cfg(any(feature = "d230", feature = "wide"))]
decl_sqrt_kernel_shim!(sqrt_d230, crate::wide_int::Int768, crate::wide_int::Int1536);

#[cfg(any(feature = "d307", feature = "wide", feature = "x-wide"))]
decl_sqrt_kernel_shim!(sqrt_d307, crate::wide_int::Int1024, crate::wide_int::Int2048);

#[cfg(any(feature = "d462", feature = "x-wide"))]
decl_sqrt_kernel_shim!(sqrt_d462, crate::wide_int::Int1536, crate::wide_int::Int3072);

#[cfg(any(feature = "d616", feature = "x-wide"))]
decl_sqrt_kernel_shim!(sqrt_d616, crate::wide_int::Int2048, crate::wide_int::Int4096);

#[cfg(any(feature = "d924", feature = "xx-wide"))]
decl_sqrt_kernel_shim!(sqrt_d924, crate::wide_int::Int3072, crate::wide_int::Int6144);

#[cfg(any(feature = "d1232", feature = "xx-wide"))]
decl_sqrt_kernel_shim!(sqrt_d1232, crate::wide_int::Int4096, crate::wide_int::Int8192);