diffsol 0.12.3

A library for solving ordinary differential equations (ODEs) in Rust.
Documentation
use crate::{
    ode_solver::problem::OdeSolverSolution, scalar::scale, Context, DenseMatrix, OdeBuilder,
    OdeEquationsImplicit, OdeSolverProblem, Vector,
};
use num_traits::{FromPrimitive, One};
use std::ops::MulAssign;

// dy/dt = y^2
fn rhs<M: DenseMatrix>(x: &M::V, _p: &M::V, _t: M::T, y: &mut M::V) {
    y.copy_from(x);
    y.component_mul_assign(x);
}

// Jv = 2yv
fn rhs_jac<M: DenseMatrix>(x: &M::V, _p: &M::V, _t: M::T, v: &M::V, y: &mut M::V) {
    y.copy_from(v);
    y.component_mul_assign(x);
    y.mul_assign(scale(M::T::from_f64(2.).unwrap()));
}

#[allow(clippy::type_complexity)]
pub fn dydt_y2_problem<M: DenseMatrix + 'static>(
    use_coloring: bool,
    size: usize,
) -> (
    OdeSolverProblem<impl OdeEquationsImplicit<M = M, V = M::V, T = M::T, C = M::C>>,
    OdeSolverSolution<M::V>,
) {
    let y0 = -200.;
    let tlast = 20.0;
    let problem = OdeBuilder::<M>::new()
        .use_coloring(use_coloring)
        .rtol(1e-4)
        .rhs_implicit(rhs::<M>, rhs_jac::<M>)
        .init(move |_p, _t, y| y.fill(M::T::from_f64(y0).unwrap()), size)
        .build()
        .unwrap();
    let mut soln = OdeSolverSolution::default();
    let y0: Vec<M::T> = [M::T::from_f64(y0).unwrap()].repeat(size);
    let n = 10;
    let dt = tlast / n as f64;
    for i in 0..=n {
        let t = M::T::from_f64(i as f64 * dt).unwrap();
        // y = y0 / (1 - y0 * t)
        let y = y0
            .iter()
            .map(|&y| y / (M::T::one() - y * t))
            .collect::<Vec<_>>();
        soln.push(problem.context().vector_from_vec(y), t);
    }
    (problem, soln)
}