decimal-scaled 0.5.0

Const-generic base-10 fixed-point decimals (D18/D38/D76/D153/D307 and the half-width tiers up to D1232) 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
// SPDX-FileCopyrightText: 2026 John Moxley
// SPDX-License-Identifier: MIT OR Apache-2.0

//! Integer square-root policy — the native-vs-Newton algorithm matcher.
//!
//! `Uint<N>::isqrt` delegates to [`dispatch`], which follows the canonical
//! policy shape (see `docs/ARCHITECTURE.md` → "Policy file structure"):
//!
//! 1. an [`Algorithm`] enum — the real isqrt algorithms, no `Default`
//!    variant;
//! 2. a [`Select`] verdict — a settled algorithm or "the value decides";
//! 3. a `const fn` [`select`] keyed on `N`, total over the key;
//! 4. dispatch via an inline `const { select::<N>() }` block, then an
//!    **exhaustive** `match algo` — no `_`, no panic.
//!
//! Because `select` is `const` and keyed only on the const generic `N`,
//! the `const { … }` block folds per monomorphisation and the unchosen arm
//! is dead-arm-eliminated in release: each concrete `Uint<N>` compiles to a
//! direct call to the chosen kernel, no runtime branch.
//!
//! # Algorithm selection
//!
//! The two algorithms correspond directly to the arms of the existing
//! const-`N` ladder in [`crate::int::algos::isqrt::isqrt_mag_fixed::isqrt_mag_fixed`], which
//! this policy formalises:
//!
//! - **`N ∈ {1, 2}`** → [`isqrt_native`]: single hardware instruction via
//!   `u64::isqrt` (`N == 1`) or `u128::isqrt` (`N == 2`). The fastest path
//!   at these widths; genuinely width-bespoke (no generic form).
//! - **`N >= 3`** → [`isqrt_newton`]: width-agnostic Newton iteration with a
//!   hardware-`f64::sqrt` seed over u64 limbs — one algorithm serving every
//!   wider int. Today's `limbs_isqrt_u64` (now in `int/algos/roots.rs`).
//!
//! The `ByValue` arm of [`Select`] is present for canonical-shape uniformity;
//! `select` never returns it (the choice is fully determined by `N`).
//!
//! # Const-ness
//!
//! `dispatch` is **not** `const fn`. The `Newton` arm calls
//! [`isqrt_newton`], which performs Newton iteration and is not
//! const-evaluable. The `Native` arm could in principle be `const`, but
//! because the policy must accommodate both arms a single `const fn` is not
//! possible. `Uint<N>::isqrt` is therefore not `const fn`.

use crate::int::algos::isqrt::isqrt_karatsuba::isqrt_karatsuba as isqrt_karatsuba_kernel;
use crate::int::algos::isqrt::isqrt_mag_fixed::isqrt_mag_fixed;
use crate::int::algos::isqrt::isqrt_schoolbook::isqrt_schoolbook as isqrt_schoolbook_kernel;
use crate::int::types::Uint;

// ── 1. the real isqrt algorithms — NAMED, no `Default` ───────────────

/// The integer square-root algorithms this policy chooses between. Variants
/// are the CamelCase of each kernel fn's name minus the `isqrt_` function
/// prefix — strict 1:1 with the kernel fns.
///
/// Names follow RULES §4: `isqrt_native` → `Native`, `isqrt_newton` →
/// `Newton`.
#[derive(Clone, Copy, PartialEq, Eq)]
enum Algorithm {
    /// [`isqrt_native`] — hardware `u64::isqrt` (`N == 1`) or
    /// `u128::isqrt` (`N == 2`). Width-bespoke; const-split inside the fn.
    Native,
    /// [`isqrt_newton`] — width-agnostic Newton iteration with a
    /// hardware-`f64::sqrt` seed over u64 limbs. Serves `3 <= N < 64`.
    Newton,
    /// [`isqrt_karatsuba`] — Karatsuba Square Root (Zimmermann RR-3805): one
    /// half-width divide per recursion level, `O(log n)` total, versus
    /// Newton's full-width divide per iteration. The `isqrt_ab` N-way A/B
    /// shows its margin shrinks monotonically with width and crosses over the
    /// shipped Newton kernel at the widest tier (`N == 64` / D1232), where it
    /// wins ~1.1-1.4x; Newton stays faster below that.
    Karatsuba,
    /// [`isqrt_schoolbook`] -- two-bits-at-a-time bitwise reference
    /// implementation; pure integer, no division, no floating-point seed.
    /// Serves any `N` as a generic reference baseline.
    #[allow(dead_code)]
    Schoolbook,
}

// ── 2. the verdict ────────────────────────────────────────────────────

/// A settled algorithm, or "the value decides". The isqrt picker always
/// returns `ByAlgorithm`: the choice is fully determined by `N`. `ByValue`
/// is part of the canonical shape for uniformity across functions; `select`
/// never returns it.
#[derive(Clone, Copy)]
enum Select<const N: usize> {
    ByAlgorithm(Algorithm),
    #[allow(dead_code)]
    ByValue(fn(&Uint<N>) -> Algorithm),
}

// ── 3. the matcher: const, keyed on `N`, total over the key ──────────

/// Pick the isqrt algorithm for storage limb count `N`. Total over the key.
///
/// - `N ∈ {1, 2}` → [`Algorithm::Native`] (hardware single-instruction path).
/// - `3 <= N < 64` → [`Algorithm::Newton`] (generic limb Newton).
/// - `N >= 64` (the widest tier, D1232) → [`Algorithm::Karatsuba`]: the
///   half-width-divide recursion crosses over Newton here (the `isqrt_ab` A/B
///   win-region; below it Newton's lower constant factor wins).
const fn select<const N: usize>() -> Select<N> {
    match N {
        1 | 2 => Select::ByAlgorithm(Algorithm::Native),
        64.. => Select::ByAlgorithm(Algorithm::Karatsuba),
        _ => Select::ByAlgorithm(Algorithm::Newton),
    }
}

// ── algorithm fns: thin delegations to the existing kernels ──────────

/// Native hardware integer square root for `Uint<N>` where `N ∈ {1, 2}`.
///
/// Delegates to [`isqrt_mag_fixed`] which const-splits on `N` internally:
/// `N == 1` → `u64::isqrt`, `N == 2` → `u128::isqrt`. Both are single
/// hardware instructions on modern ISAs.
#[inline]
pub(crate) fn isqrt_native<const N: usize>(x: Uint<N>) -> Uint<N> {
    let mut out = [0u64; N];
    isqrt_mag_fixed::<N>(x.as_limbs(), &mut out);
    Uint::<N>::from_limbs(out)
}

/// Newton integer square root for `Uint<N>` where `N >= 3`.
///
/// Delegates to [`isqrt_mag_fixed`] which routes to
/// [`crate::int::algos::isqrt::isqrt_newton::isqrt_newton`] for `N >= 3`: Newton
/// iteration with a hardware-`f64::sqrt` seed over the u64 limbs.
#[inline]
pub(crate) fn isqrt_newton<const N: usize>(x: Uint<N>) -> Uint<N> {
    let mut out = [0u64; N];
    isqrt_mag_fixed::<N>(x.as_limbs(), &mut out);
    Uint::<N>::from_limbs(out)
}

/// Karatsuba Square Root for `Uint<N>` — the widest-tier (`N >= 64`) arm.
///
/// Delegates to
/// [`isqrt_karatsuba_kernel`][`crate::int::algos::isqrt::isqrt_karatsuba::isqrt_karatsuba`]:
/// the RR-3805 recursion whose per-level divide is half-width and runs only
/// `O(log n)` times. Bit-identical to [`isqrt_newton`]; the `isqrt_ab` A/B
/// shows it wins at `N == 64` where Newton's full-width-divide-per-iteration
/// cost finally dominates.
#[inline]
pub(crate) fn isqrt_karatsuba<const N: usize>(x: Uint<N>) -> Uint<N> {
    let mut out = [0u64; N];
    isqrt_karatsuba_kernel(x.as_limbs(), &mut out);
    Uint::<N>::from_limbs(out)
}

/// Schoolbook two-bits-at-a-time integer square root for `Uint<N>`.
///
/// Delegates to
/// [`isqrt_schoolbook_kernel`][`crate::int::algos::isqrt::isqrt_schoolbook::isqrt_schoolbook`]:
/// classical bitwise digit-by-digit algorithm; no division, no float seed.
/// Serves any `N` as a generic reference baseline.
#[allow(dead_code)]
#[inline]
pub(crate) fn isqrt_schoolbook_policy<const N: usize>(x: Uint<N>) -> Uint<N> {
    let mut out = [0u64; N];
    isqrt_schoolbook_kernel(x.as_limbs(), &mut out);
    Uint::<N>::from_limbs(out)
}
// ── 4. the dispatcher: fold the verdict, then dispatch ────────────────

/// Integer square-root dispatcher for `Uint<N>`.
///
/// Resolves the compile-time algorithm verdict via
/// `const { select::<N>() }` (folds per monomorphisation; dead arms are
/// eliminated in release) then dispatches exhaustively over [`Algorithm`].
///
/// Not `const fn`: the `Newton` arm delegates to
/// [`crate::int::algos::isqrt::isqrt_newton::isqrt_newton`] (Newton iteration, not
/// const-evaluable).
#[inline]
pub(crate) fn dispatch<const N: usize>(x: Uint<N>) -> Uint<N> {
    let algo = match const { select::<N>() } {
        Select::ByAlgorithm(a) => a,
        Select::ByValue(f) => f(&x),
    };
    match algo {
        Algorithm::Native => isqrt_native::<N>(x),
        Algorithm::Newton => isqrt_newton::<N>(x),
        Algorithm::Karatsuba => isqrt_karatsuba::<N>(x),
        Algorithm::Schoolbook => isqrt_schoolbook_policy::<N>(x),
    }
}