feos 0.9.5

FeOs - A framework for equations of state and classical density functional theory.
Documentation
use super::Pets;
use crate::hard_sphere::HardSphereProperties;
use feos_core::StateHD;
use num_dual::DualNum;
use std::f64::consts::{FRAC_PI_3, PI};

pub const A: [f64; 7] = [
    0.690603404,
    1.189317012,
    1.265604153,
    -24.34554201,
    93.67300357,
    -157.8773415,
    96.93736697,
];
pub const B: [f64; 7] = [
    0.664852128,
    2.10733079,
    -9.597951213,
    -17.37871193,
    30.17506222,
    209.3942909,
    -353.2743581,
];

impl Pets {
    pub fn dispersion_helmholtz_energy_density<D: DualNum<f64> + Copy>(
        &self,
        state: &StateHD<D>,
    ) -> D {
        // auxiliary variables
        let n = self.sigma.len();
        let rho = &state.partial_density;

        // temperature dependent segment radius
        let d3 = self.hs_diameter(state.temperature).map(|d| d.powi(3));

        // packing fraction
        let eta = rho.component_mul(&d3).sum() * 0.5 * FRAC_PI_3;

        // mixture densities, crosswise interactions of all segments on all chains
        let mut rho1mix = D::zero();
        let mut rho2mix = D::zero();
        for i in 0..n {
            for j in 0..n {
                let eps_ij = state.temperature.recip() * self.epsilon_k_ij[(i, j)];
                let sigma_ij = self.sigma_ij[(i, j)].powi(3);
                rho1mix += rho[i] * rho[j] * eps_ij * sigma_ij;
                rho2mix += rho[i] * rho[j] * eps_ij * eps_ij * sigma_ij;
            }
        }

        // I1, I2 and C1
        let mut i1 = D::zero();
        let mut i2 = D::zero();
        let mut eta_i = D::one();
        for i in 0..=6 {
            i1 += eta_i * A[i];
            i2 += eta_i * B[i];
            eta_i *= eta;
        }
        let c1 = ((eta * 8.0 - eta.powi(2) * 2.0) / (eta - 1.0).powi(4) + 1.0).recip();

        // Helmholtz energy
        (-rho1mix * i1 * 2.0 - rho2mix * c1 * i2) * PI
    }
}

#[cfg(test)]
mod tests {
    use super::*;
    use crate::pets::parameters::utils::{
        argon_krypton_parameters, argon_parameters, krypton_parameters,
    };
    use approx::assert_relative_eq;
    use nalgebra::dvector;

    #[test]
    fn mix() {
        let argon = Pets::new(argon_parameters());
        let krypton = Pets::new(krypton_parameters());
        let mix = Pets::new(argon_krypton_parameters());
        let t = 250.0;
        let v = 2.5e28;
        let n = 1.0;
        let s = StateHD::new(t, v, &dvector![n]);
        let a1 = argon.dispersion_helmholtz_energy_density(&s);
        let a2 = krypton.dispersion_helmholtz_energy_density(&s);
        let s1m = StateHD::new(t, v, &dvector![n, 0.0]);
        let a1m = mix.dispersion_helmholtz_energy_density(&s1m);
        let s2m = StateHD::new(t, v, &dvector![0.0, n]);
        let a2m = mix.dispersion_helmholtz_energy_density(&s2m);
        assert_relative_eq!(a1, a1m, epsilon = 1e-14);
        assert_relative_eq!(a2, a2m, epsilon = 1e-14);
    }
}