use ndarray::Array1;
use stochastic_rs_core::simd_rng::Deterministic;
use stochastic_rs_core::simd_rng::SeedExt;
use stochastic_rs_core::simd_rng::Unseeded;
use stochastic_rs_distributions::normal::SimdNormal;
use crate::traits::FloatExt;
use crate::traits::ProcessExt;
#[derive(Debug, Clone)]
pub struct Garch<T: FloatExt, S: SeedExt = Unseeded> {
pub omega: T,
pub alpha: Array1<T>,
pub beta: Array1<T>,
pub n: usize,
pub seed: S,
}
impl<T: FloatExt> Garch<T> {
pub fn new(omega: T, alpha: Array1<T>, beta: Array1<T>, n: usize) -> Self {
assert!(omega > T::zero(), "Garch requires omega > 0");
Garch {
omega,
alpha,
beta,
n,
seed: Unseeded,
}
}
}
impl<T: FloatExt> Garch<T, Deterministic> {
pub fn seeded(omega: T, alpha: Array1<T>, beta: Array1<T>, n: usize, seed: u64) -> Self {
assert!(omega > T::zero(), "Garch requires omega > 0");
Garch {
omega,
alpha,
beta,
n,
seed: Deterministic::new(seed),
}
}
}
impl<T: FloatExt, S: SeedExt> ProcessExt<T> for Garch<T, S> {
type Output = Array1<T>;
fn sample(&self) -> Self::Output {
let p = self.alpha.len();
let q = self.beta.len();
let mut z = Array1::<T>::zeros(self.n);
if self.n > 0 {
let slice = z.as_slice_mut().expect("contiguous");
let normal = SimdNormal::<T>::from_seed_source(T::zero(), T::one(), &self.seed);
normal.fill_slice_fast(slice);
}
let mut x = Array1::<T>::zeros(self.n);
let mut sigma2 = Array1::<T>::zeros(self.n);
let var_floor = T::from_f64_fast(1e-12);
let sum_alpha = self.alpha.iter().cloned().sum();
let sum_beta = self.beta.iter().cloned().sum();
let denom = T::one() - sum_alpha - sum_beta;
assert!(
denom > T::zero(),
"Garch requires sum(alpha) + sum(beta) < 1 for finite unconditional variance"
);
for t in 0..self.n {
if t == 0 {
sigma2[t] = self.omega / denom;
} else {
let mut var_t = self.omega;
for i in 1..=p {
if t >= i {
var_t += self.alpha[i - 1] * x[t - i].powi(2);
}
}
for j in 1..=q {
if t >= j {
var_t += self.beta[j - 1] * sigma2[t - j];
}
}
sigma2[t] = var_t;
}
assert!(
sigma2[t].is_finite() && sigma2[t] > T::zero(),
"Garch produced non-positive or non-finite conditional variance at t={}",
t
);
x[t] = sigma2[t].max(var_floor).sqrt() * z[t];
}
x
}
}
py_process_1d!(PyGarch, Garch,
sig: (omega, alpha, beta, n, seed=None, dtype=None),
params: (omega: f64, alpha: Vec<f64>, beta: Vec<f64>, n: usize)
);