feos 0.9.5

FeOs - A framework for equations of state and classical density functional theory.
Documentation
use super::BarkerHenderson;
use crate::uvtheory::parameters::UVTheoryPars;
use nalgebra::{DMatrix, DVector, dvector};
use num_dual::DualNum;

const BH_CONSTANTS_ETA_B: [[f64; 2]; 3] = [
    [-0.960919783, -0.921097447],
    [-0.547468020, -3.508014069],
    [-2.253750186, 3.581161364],
];

const BH_CONSTANTS_ETA_A: [[f64; 4]; 4] = [
    [-1.217417282, 6.754987582, -0.5919326153, -28.99719604],
    [1.579548775, -26.93879416, 0.3998915410, 106.9446266],
    [-1.993990512, 44.11863355, -40.10916106, -29.6130848],
    [0.0, 0.0, 0.0, 0.0],
];

/// Dimensionless Hard-sphere diameter according to Barker-Henderson division.
/// Eq. S23 and S24.
impl BarkerHenderson {
    pub fn diameter_bh<D: DualNum<f64> + Copy>(
        parameters: &UVTheoryPars,
        temperature: D,
    ) -> DVector<D> {
        parameters
            .cd_bh_pure
            .iter()
            .enumerate()
            .map(|(i, c)| {
                let t = temperature / parameters.epsilon_k[i];
                let d = t.powf(0.25) * c[1] + t.powf(0.75) * c[2] + t.powf(1.25) * c[3];
                (t * c[0] + d * (t + 1.0).ln() + t.powi(2) * c[4] + 1.0)
                    .powf(-0.5 / parameters.rep[i])
                    * parameters.sigma[i]
            })
            .collect::<Vec<_>>()
            .into()
    }
}

pub(super) fn packing_fraction<D: DualNum<f64> + Copy>(
    partial_density: &DVector<D>,
    diameter: &DVector<D>,
) -> D {
    (0..partial_density.len()).fold(D::zero(), |acc, i| {
        acc + partial_density[i] * diameter[i].powi(3) * (std::f64::consts::PI / 6.0)
    })
}

pub(super) fn packing_fraction_b<D: DualNum<f64> + Copy>(
    parameters: &UVTheoryPars,
    diameter: &DVector<D>,
    eta: D,
) -> DMatrix<D> {
    let n = parameters.att.len();
    DMatrix::from_fn(n, n, |i, j| {
        let tau =
            -(diameter[i] / parameters.sigma[i] + diameter[j] / parameters.sigma[j]) * 0.5 + 1.0; //dimensionless
        let tau2 = tau * tau;

        let c = dvector![
            tau * BH_CONSTANTS_ETA_B[0][0] + tau2 * BH_CONSTANTS_ETA_B[0][1],
            tau * BH_CONSTANTS_ETA_B[1][0] + tau2 * BH_CONSTANTS_ETA_B[1][1],
            tau * BH_CONSTANTS_ETA_B[2][0] + tau2 * BH_CONSTANTS_ETA_B[2][1],
        ];
        eta + eta * c[0] + eta * eta * c[1] + eta.powi(3) * c[2]
    })
}

pub(super) fn packing_fraction_a<D: DualNum<f64> + Copy>(
    parameters: &UVTheoryPars,
    diameter: &DVector<D>,
    eta: D,
) -> DMatrix<D> {
    let n = parameters.att.len();
    DMatrix::from_fn(n, n, |i, j| {
        let tau =
            -(diameter[i] / parameters.sigma[i] + diameter[j] / parameters.sigma[j]) * 0.5 + 1.0;
        let tau2 = tau * tau;
        let rep_inv = 1.0 / parameters.rep_ij[(i, j)];

        let c = dvector![
            tau * (BH_CONSTANTS_ETA_A[0][0] + BH_CONSTANTS_ETA_A[0][1] * rep_inv)
                + tau2 * (BH_CONSTANTS_ETA_A[0][2] + BH_CONSTANTS_ETA_A[0][3] * rep_inv),
            tau * (BH_CONSTANTS_ETA_A[1][0] + BH_CONSTANTS_ETA_A[1][1] * rep_inv)
                + tau2 * (BH_CONSTANTS_ETA_A[1][2] + BH_CONSTANTS_ETA_A[1][3] * rep_inv),
            tau * (BH_CONSTANTS_ETA_A[2][0] + BH_CONSTANTS_ETA_A[2][1] * rep_inv)
                + tau2 * (BH_CONSTANTS_ETA_A[2][2] + BH_CONSTANTS_ETA_A[2][3] * rep_inv),
            tau * (BH_CONSTANTS_ETA_A[3][0] + BH_CONSTANTS_ETA_A[3][1] * rep_inv)
                + tau2 * (BH_CONSTANTS_ETA_A[3][2] + BH_CONSTANTS_ETA_A[3][3] * rep_inv),
        ];

        eta + eta * c[0] + eta * eta * c[1] + eta.powi(3) * c[2] + eta.powi(4) * c[3]
    })
}

#[cfg(test)]
#[expect(clippy::excessive_precision)]
mod test {
    use super::*;
    use crate::uvtheory::{
        Perturbation::BarkerHenderson as BH,
        parameters::utils::{methane_parameters, test_parameters},
    };

    #[test]
    fn test_bh_diameter() {
        let p = test_parameters(12.0, 6.0, 1.0, 1.0, BH);
        assert_eq!(
            BarkerHenderson::diameter_bh(&p, 2.0)[0],
            0.95777257352360246
        );
        let p = test_parameters(24.0, 6.0, 1.0, 1.0, BH);
        assert_eq!(
            BarkerHenderson::diameter_bh(&p, 5.0)[0],
            0.95583586434435486
        );

        // Methane
        let p = UVTheoryPars::new(&methane_parameters(12.0, 6.0), BH);
        assert_eq!(
            BarkerHenderson::diameter_bh(&p, 2.0 * p.epsilon_k[0])[0] / p.sigma[0],
            0.95777257352360246
        );
        let p = UVTheoryPars::new(&methane_parameters(24.0, 6.0), BH);
        assert_eq!(
            BarkerHenderson::diameter_bh(&p, 5.0 * p.epsilon_k[0])[0] / p.sigma[0],
            0.95583586434435486
        );
    }
}