Newer
Older
#define CATCH_CONFIG_MAIN
#include "catch/catch.hpp"
#include "teqp/core.hpp"
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
auto build_vdW_argon() {
double Omega_b = 1.0 / 8, Omega_a = 27.0 / 64;
double Tcrit = 150.687, pcrit = 4863000.0; // Argon
double R = get_R_gas<double>(); // Universal gas constant
double b = Omega_b * R * Tcrit / pcrit;
double ba = Omega_b / Omega_a / Tcrit / R;
double a = b / ba;
auto vdW = vdWEOS1(a, b);
return vdW;
}
auto build_simple() {
// Argon + Xenon
std::valarray<double> Tc_K = { 150.687, 289.733 };
std::valarray<double> pc_Pa = { 4863000.0, 5842000.0 };
auto R = 1.380649e-23 * 6.02214076e23; ///< Exact value, given by k_B*N_A
int i = 0;
double ai = 27.0 / 64.0 * pow(R * Tc_K[i], 2) / pc_Pa[i];
double bi = 1.0 / 8.0 * R * Tc_K[i] / pc_Pa[i];
return vdWEOS1(ai, bi);
}
auto build_vdW() {
// Argon + Xenon
std::valarray<double> Tc_K = { 150.687, 289.733 };
std::valarray<double> pc_Pa = { 4863000.0, 5842000.0 };
return vdWEOS(Tc_K, pc_Pa);
}
TEST_CASE("Check virial coefficients for vdW", "[virial]")
{
auto vdW = build_vdW_argon();
double Omega_b = 1.0 / 8, Omega_a = 27.0 / 64;
double Tcrit = 150.687, pcrit = 4863000.0; // Argon
double R = get_R_gas<double>(); // Universal gas constant
double b = Omega_b * R * Tcrit / pcrit;
double ba = Omega_b / Omega_a / Tcrit / R;
double a = b / ba;
Eigen::ArrayXd molefrac(1); molefrac = 1.0;
using vd = VirialDerivatives<decltype(vdW)>;
auto Bn = vd::get_Bnvir<Nvir, ADBackends::autodiff>(vdW, T, molefrac);
auto Bnmcx = vd::get_Bnvir<Nvir, ADBackends::multicomplex>(vdW, T, molefrac);
// Exact solutions for virial coefficients for van der Waals
auto get_vdW_exacts = [a, b, R, T](int Nmax) {
std::map<int, double> o = { {2, b - a / (R * T)} };
for (auto i = 3; i <= Nmax; ++i) {
o[i] = pow(b, i - 1);
}
return o;
};
auto Bnexact = get_vdW_exacts(Nvir);
// This one with complex step derivatives as another check
double B2 = vd::get_B2vir(vdW, T, molefrac);
double B2exact = b - a / (R * T);
CHECK(std::abs(B2exact-Bnexact[2]) < 1e-15);
CHECK(std::abs(B2-Bnexact[2]) < 1e-15);
// And all the remaining derivatives
for (auto i = 2; i <= Nvir; ++i) {
auto numeric = Bn[i];
auto exact = Bnexact[i];
auto err = std::abs(numeric-exact);
auto relerr = err / std::abs(Bn[i]);
CAPTURE(numeric);
CAPTURE(exact);
CAPTURE(i);
CAPTURE(relerr);
CHECK(relerr < 1e-15);
}
}
TEST_CASE("Check neff", "[virial]")
{
double T = 298.15;
double rho = 3.0;
const Eigen::Array2d molefrac = { 0.5, 0.5 };
auto f = [&T, &rho, &molefrac](const auto& model) {
auto neff = TDXDerivatives<decltype(model)>::get_neff(model, T, rho, molefrac);
CAPTURE(neff);
CHECK(neff > 0);
CHECK(neff < 100);
};
// This quantity is undefined for the van der Waals EOS because Ar20 is always 0
//SECTION("vdW") {
// f(build_simple());
//}
SECTION("PCSAFT") {
std::vector<std::string> names = { "Methane", "Ethane" };
f(PCSAFTMixture(names));
}
}
const Eigen::Array2d rhovec = { rho / 2, rho / 2 };
auto get_worst_error = [&T, &rhovec](const auto &model){
using id = IsochoricDerivatives <decltype(model)>;
auto H1 = id::build_Psir_Hessian_autodiff(model, T, rhovec);
auto H2 = id::build_Psir_Hessian_mcx(model, T, rhovec);
auto err = (H1.array() - H2).abs().maxCoeff();
CAPTURE(err);
CHECK(err < 1e-15);
return;
};
SECTION("simple") {
get_worst_error(build_simple());
}
SECTION("less_simple") {
get_worst_error(build_vdW());
}
}
{
auto model = build_simple();
const double T = 298.15;
const double rho = 3000.0;
const auto rhovec = (Eigen::ArrayXd(2) << rho / 2, rho / 2).finished();
const auto molefrac = rhovec/rhovec.sum();
// Exact solution from EOS
auto pexact = model.p(T, 1/rho);
// Numerical solution from alphar
using id = IsochoricDerivatives<decltype(model)>;
auto pfromderiv = rho*model.R*T + id::get_pr(model, T, rhovec);
using tdx = TDXDerivatives<decltype(model)>;
auto p_ar0n = rho*model.R*T*(1 + tdx::get_Ar0n<3>(model, T, rho, molefrac)[1]);
constexpr int Nvir = 8;
using vd = VirialDerivatives<decltype(model)>;
auto Bn = vd::get_Bnvir<Nvir>(model, T, molefrac);
auto Z = 1.0;
for (auto i = 2; i <= Nvir; ++i){
Z += Bn[i]*pow(rho, i-1);
auto pvir = Z * rho * model.R * T;
}
auto pvir = Z*rho*model.R*T;
CHECK(std::abs(pfromderiv - pexact)/pexact < 1e-15);
CHECK(std::abs(pvir - pexact)/pexact < 1e-8);
CHECK(std::abs(p_ar0n - pexact) / pexact < 1e-8);
}
TEST_CASE("Check 0n derivatives", "[virial][p]")
{
std::vector<std::string> names = { "Methane", "Ethane" };
auto model = PCSAFTMixture(names);
const double T = 100.0;
const double rho = 126.1856883066021;
const auto rhovec = (Eigen::ArrayXd(2) << rho, 0).finished();
const auto molefrac = rhovec / rhovec.sum();
using tdx = TDXDerivatives<decltype(model)>;
auto Ar02 = tdx::get_Ar02(model, T, rho, molefrac);
auto Ar02n = tdx::get_Ar0n<2>(model, T, rho, molefrac)[2];
CHECK(std::abs(Ar02 - Ar02n) < 1e-13);
auto Ar01 = tdx::get_Ar01(model, T, rho, molefrac);
auto Ar01n = tdx::get_Ar0n<4>(model, T, rho, molefrac)[1];
CHECK(std::abs(Ar01 - Ar01n) < 1e-13);
}
TEST_CASE("Trace critical locus for vdW", "[vdW][crit]")
{
// Argon + Xenon
std::valarray<double> Tc_K = { 150.687, 289.733 };
std::valarray<double> pc_Pa = { 4863000.0, 5842000.0 };
vdWEOS<double> vdW(Tc_K, pc_Pa);
auto Zc = 3.0/8.0;
auto rhoc0 = pc_Pa[0] / (vdW.R * Tc_K[0]) / Zc;
double T0 = Tc_K[0];
Eigen::ArrayXd rhovec0(2); rhovec0 << rhoc0, 0.0 ;
auto tic0 = std::chrono::steady_clock::now();
std::string filename = "";
trace_critical_arclength_binary(vdW, T0, rhovec0, filename);
auto tic1 = std::chrono::steady_clock::now();
}
TEST_CASE("TEST B12", "") {
const auto model = build_vdW();
const double T = 298.15;
const auto molefrac = (Eigen::ArrayXd(2) << 1/3, 2/3).finished();
using vd = VirialDerivatives<decltype(model)>;
auto B12 = vd::get_B12vir(model, T, molefrac);
}
TEST_CASE("Test psir gradient", "") {
const auto model = build_vdW();
const double T = 298.15;
using id = IsochoricDerivatives<decltype(model)>;
const Eigen::Array2d rhovec = { 1, 2 };
const Eigen::Array2d molefrac = { 1 / 3, 2 / 3 };
auto psirfunc2 = [&model](const auto& T, const auto& rho_) {
auto rhotot_ = rho_.sum();
auto molefrac = (rho_ / rhotot_);
return model.alphar(T, rhotot_, molefrac) * model.R * T * rhotot_;
};
auto chk0 = derivrhoi(psirfunc2, T, rhovec, 0);
auto chk1 = derivrhoi(psirfunc2, T, rhovec, 1);
auto grad = id::build_Psir_gradient_autodiff(model, T, rhovec);
auto err0 = std::abs((chk0 - grad[0])/chk0);
auto err1 = std::abs((chk1 - grad[1])/chk1);
CAPTURE(err0);
CAPTURE(err1);
CHECK(err0 < 1e-12);
CHECK(err1 < 1e-12);
}
TEST_CASE("Test pure VLE", "") {
const auto model = build_vdW_argon();
double T = 100.0;
auto resid = IsothermPureVLEResiduals(model, T);
auto rhovec = (Eigen::ArrayXd(2) << 22834.056386882046, 1025.106554560764).finished();
auto r0 = resid.call(rhovec);
auto J = resid.Jacobian(rhovec);
auto v = J.matrix().colPivHouseholderQr().solve(-r0.matrix()).array().eval();
auto rhovec1 = rhovec + v;
auto r1 = resid.call(rhovec1);
CHECK((r0.cwiseAbs() > r1.cwiseAbs()).eval().all());
do_pure_VLE_T(resid, T, 22834.056386882046, 1025.106554560764, 20);