deep_causality_num 0.3.3

Number utils for for deep_causality crate.
Documentation
/*
 * SPDX-License-Identifier: MIT
 * Copyright (c) 2023 - 2026. The DeepCausality Authors and Contributors. All Rights Reserved.
 */

use crate::Real;

mod algebra;
mod arithmetic;
mod convert;
mod display;
mod from_primitive;
mod identity;
mod real;

/// A dual number `a + b·ε` where the infinitesimal `ε` satisfies `ε² = 0`.
///
/// Dual numbers are the type-based primitive for **forward-mode automatic
/// differentiation**. Evaluating any function composed from the arithmetic and
/// elementary operations on `Dual::variable(x0)` (which is `x0 + 1·ε`) yields
/// `f(x0)` in the real part and `f'(x0)` in the `ε` part, exact to machine
/// precision — the derivative falls out of the trait impls by the chain rule.
///
/// `Dual<T>` is built over `T: Real` (the analytic real-scalar trait), not
/// `RealField`: a dual's component needs the elementary functions but never a field
/// inverse. `Dual<T>` itself implements [`Real`](crate::Real) — so a dual is a
/// first-class analytic scalar that drops into any `Real`-generic code and **nests**
/// (`Dual<Dual<T>>` gives second derivatives) — but it does **not** implement
/// [`Field`](crate::Field)/[`RealField`](crate::RealField), because `ε` is a zero
/// divisor (`ε·ε = 0`) and has no multiplicative inverse.
///
/// # Examples
///
/// ```
/// use deep_causality_num::Dual;
///
/// // f(x) = x³ + 2x, evaluated with its derivative at x = 3.
/// let x = Dual::variable(3.0_f64);
/// let y = x * x * x + x + x;
/// assert_eq!(y.value(), 27.0 + 6.0); // 3³ + 2·3 = 33
/// assert_eq!(y.derivative(), 27.0 + 2.0); // 3·3² + 2 = 29
/// ```
#[derive(Copy, Clone, PartialEq, PartialOrd, Debug)]
pub struct Dual<T: Real> {
    /// The real part `a` — the function value.
    pub re: T,
    /// The infinitesimal coefficient `b` — the derivative carried in the `ε` channel.
    pub du: T,
}

impl<T: Real> Dual<T> {
    /// Constructs a dual number `re + du·ε` from both components.
    #[inline]
    pub fn new(re: T, du: T) -> Self {
        Self { re, du }
    }

    /// Constructs a constant `re + 0·ε` (a value with zero derivative).
    #[inline]
    pub fn constant(re: T) -> Self {
        Self { re, du: T::zero() }
    }

    /// Constructs the differentiation seed `re + 1·ε` (the independent variable).
    #[inline]
    pub fn variable(re: T) -> Self {
        Self { re, du: T::one() }
    }

    /// Returns the real part `a` — the function value `f(x0)`.
    #[inline]
    pub fn value(&self) -> T {
        self.re
    }

    /// Returns the infinitesimal coefficient `b` — the derivative `f'(x0)`.
    #[inline]
    pub fn derivative(&self) -> T {
        self.du
    }
}