deep_causality_physics 0.6.2

Standard library of physics formulas and engineering primitives for DeepCausality.
Documentation
/*
 * SPDX-License-Identifier: MIT
 * Copyright (c) 2023 - 2026. The DeepCausality Authors and Contributors. All Rights Reserved.
 */

use core::f64::consts::PI;
/// Haruna's Gauge Field Formalism for Logical Quantum Gates.
///
/// Based on "Note on Logical Gates by Gauge Field Formalism of Quantum Error Correction"
/// by Junichi Haruna (2025).
/// See: https://arxiv.org/abs/2511.15224
use deep_causality_multivector::CausalMultiVector;
use deep_causality_num::{Complex, DivisionAlgebra, FromPrimitive, One, RealField};

/// Helper function to compute the exponential of a multivector: $e^A = \sum A^n/n!$
/// Uses Taylor series expansion.
fn exp<R>(mv: &CausalMultiVector<Complex<R>>) -> CausalMultiVector<Complex<R>>
where
    R: RealField + FromPrimitive,
{
    let small = R::from_f64(1e-15).expect("R::from_f64(1e-15)");
    let tol = R::from_f64(1e-12).expect("R::from_f64(1e-12)");
    let max_norm = R::from_f64(1e6).expect("R::from_f64(1e6)");

    // Fast path: exp(0) = I
    if mv
        .data()
        .iter()
        .all(|c| c.re.abs() < small && c.im.abs() < small)
    {
        return CausalMultiVector::scalar(Complex::one(), mv.metric());
    }

    let metric = mv.metric();
    let one_complex = Complex::<R>::one();
    let mut term = CausalMultiVector::scalar(one_complex, metric);
    let mut sum = term.clone();

    let max_iters = 64;

    // Simple bound to detect pathological exponents; if ||mv|| is huge, series may overflow
    let mv_norm_sq: R = mv
        .data()
        .iter()
        .map(|c| c.norm_sqr())
        .fold(R::zero(), |acc, x| acc + x);
    let mv_norm = mv_norm_sq.sqrt();
    if !mv_norm.is_finite() || mv_norm > max_norm {
        // Return zero-order approximation to avoid producing NaNs downstream
        return CausalMultiVector::scalar(Complex::one(), metric);
    }

    for n in 1..=max_iters {
        let n_r = R::from_f64(n as f64).expect("R::from_f64(n)");
        let n_inv = Complex::new(R::one() / n_r, R::zero());
        term = &term * mv;
        term *= n_inv;

        if term
            .data()
            .iter()
            .any(|c| !c.re.is_finite() || !c.im.is_finite())
        {
            return sum;
        }

        let prev = sum.clone();
        sum += &term;

        let diff = &sum - &prev;
        let delta_sq: R = diff
            .data()
            .iter()
            .map(|c| c.norm_sqr())
            .fold(R::zero(), |acc, x| acc + x);
        let delta = delta_sq.sqrt();

        if !delta.is_finite() {
            return prev;
        }
        if delta < tol {
            return sum;
        }
    }

    // Ensure finiteness of result
    if sum
        .data()
        .iter()
        .any(|c| !c.re.is_finite() || !c.im.is_finite())
    {
        return CausalMultiVector::scalar(Complex::one(), metric);
    }
    sum
}

/// Calculates the Logical Z gate: $Z(\gamma) = \exp(i \pi a(\gamma))$.
pub fn logical_z<R>(a_gamma: &CausalMultiVector<Complex<R>>) -> CausalMultiVector<Complex<R>>
where
    R: RealField + FromPrimitive,
{
    let pi = R::from_f64(PI).expect("R::from_f64(PI)");
    let i_pi = Complex::new(R::zero(), pi);
    let exponent = a_gamma.clone() * i_pi;
    exp(&exponent)
}

/// Calculates the Logical X gate: $X(\tilde{\gamma}) = \exp(i \pi b(\tilde{\gamma}))$.
pub fn logical_x<R>(b_gamma_tilde: &CausalMultiVector<Complex<R>>) -> CausalMultiVector<Complex<R>>
where
    R: RealField + FromPrimitive,
{
    let pi = R::from_f64(PI).expect("R::from_f64(PI)");
    let i_pi = Complex::new(R::zero(), pi);
    let exponent = b_gamma_tilde.clone() * i_pi;
    exp(&exponent)
}

/// Calculates the Logical S gate: $S(\gamma) = \exp(i \frac{\pi}{2} a(\gamma)^2)$.
pub fn logical_s<R>(a_gamma: &CausalMultiVector<Complex<R>>) -> CausalMultiVector<Complex<R>>
where
    R: RealField + FromPrimitive,
{
    let pi_2 = R::from_f64(PI / 2.0).expect("R::from_f64(PI/2)");
    let i_pi_2 = Complex::new(R::zero(), pi_2);
    let a_sq = a_gamma.clone() * a_gamma.clone();
    let exponent = a_sq * i_pi_2;
    exp(&exponent)
}

/// Calculates the Logical Hadamard gate.
pub fn logical_hadamard<R>(
    a_gamma: &CausalMultiVector<Complex<R>>,
    b_gamma_tilde: &CausalMultiVector<Complex<R>>,
) -> CausalMultiVector<Complex<R>>
where
    R: RealField + FromPrimitive,
{
    // phase_scalar = exp(-i pi/4) = cos(-pi/4) + i sin(-pi/4)
    let neg_pi_4 = R::from_f64(-PI / 4.0).expect("R::from_f64(-PI/4)");
    let phase_scalar = Complex::new(neg_pi_4.cos(), neg_pi_4.sin());

    let s_a = logical_s(a_gamma);

    let pi_2 = R::from_f64(PI / 2.0).expect("R::from_f64(PI/2)");
    let i_pi_2 = Complex::new(R::zero(), pi_2);
    let b_sq = b_gamma_tilde.clone() * b_gamma_tilde.clone();
    let mid = exp(&(b_sq * i_pi_2));

    let step1 = &s_a * &mid;
    let step2 = &step1 * &s_a;
    step2 * phase_scalar
}

/// Calculates the Logical CZ gate: $CZ(\gamma_1, \gamma_2) = \exp(i \pi a(\gamma_1) a(\gamma_2))$.
pub fn logical_cz<R>(
    a_gamma1: &CausalMultiVector<Complex<R>>,
    a_gamma2: &CausalMultiVector<Complex<R>>,
) -> CausalMultiVector<Complex<R>>
where
    R: RealField + FromPrimitive,
{
    let pi = R::from_f64(PI).expect("R::from_f64(PI)");
    let i_pi = Complex::new(R::zero(), pi);
    let interaction = a_gamma1.clone() * a_gamma2.clone();
    let exponent = interaction * i_pi;
    exp(&exponent)
}

/// Calculates the Logical T gate.
pub fn logical_t<R>(a_gamma: &CausalMultiVector<Complex<R>>) -> CausalMultiVector<Complex<R>>
where
    R: RealField + FromPrimitive,
{
    let a = a_gamma.clone();
    let a2 = &a * &a;
    let a3 = &a2 * &a;

    let half = R::from_f64(0.5).expect("R::from_f64(0.5)");
    let neg_three_quarters = R::from_f64(-0.75).expect("R::from_f64(-0.75)");

    let c1 = Complex::new(half, R::zero());
    let c2 = Complex::new(neg_three_quarters, R::zero());
    let c3 = Complex::new(half, R::zero());

    let term1 = a3 * c1;
    let term2 = a2 * c2;
    let term3 = a * c3;

    let sum = term1 + term2 + term3;
    let pi = R::from_f64(PI).expect("R::from_f64(PI)");
    let i_pi = Complex::new(R::zero(), pi);

    exp(&(sum * i_pi))
}