1#ifndef COOLPROP_NUMERICS_H
2#define COOLPROP_NUMERICS_H
16#if defined(HUGE_VAL) && !defined(_HUGE)
17# define _HUGE HUGE_VAL
20# if defined(HUGE) && !defined(_HUGE)
25template <
typename T,
size_t N>
34 return (x <= DBL_MAX && x >= -DBL_MAX);
38# define M_PI 3.14159265358979323846
59template <
typename X,
typename Y>
67 Spline(
const std::vector<X>& x,
const std::vector<Y>& y) {
68 if (x.size() != y.size()) {
69 std::cerr <<
"X and Y must be the same size " << std::endl;
74 std::cerr <<
"Must have at least three points for interpolation" << std::endl;
78 typedef typename std::vector<X>::difference_type size_type;
80 size_type n = y.size() - 1;
82 std::vector<Y> b(n), d(n), a(n), c(n + 1), l(n + 1), u(n + 1), z(n + 1);
83 std::vector<X> h(n + 1);
90 for (size_type i = 1; i < n; i++) {
91 h[i] = x[i + 1] - x[i];
92 l[i] = Y(2 * (x[i + 1] - x[i - 1])) - Y(h[i - 1]) * u[i - 1];
93 u[i] = Y(h[i]) / l[i];
94 a[i] = (Y(3) / Y(h[i])) * (y[i + 1] - y[i]) - (Y(3) / Y(h[i - 1])) * (y[i] - y[i - 1]);
95 z[i] = (a[i] - Y(h[i - 1]) * z[i - 1]) / l[i];
101 for (size_type j = n - 1; j >= 0; j--) {
102 c[j] = z[j] - u[j] * c[j + 1];
103 b[j] = (y[j + 1] - y[j]) / Y(h[j]) - (Y(h[j]) * (c[j + 1] + Y(2) * c[j])) / Y(3);
104 d[j] = (c[j + 1] - c[j]) / Y(3 * h[j]);
107 for (size_type i = 0; i < n; i++) {
120 typename std::vector<element_type>::const_iterator it;
131 if (
mElements.size() == 0)
return std::vector<Y>(xx.size());
133 typename std::vector<X>::const_iterator it;
134 typename std::vector<element_type>::const_iterator it2;
137 for (it = xx.begin(); it != xx.end(); it++) {
143 ys.push_back(it2->eval(*it));
154 Element(
X _x, Y _a, Y _b, Y _c, Y _d) :
x(_x),
a(_a),
b(_b),
c(_c),
d(_d) {}
158 return a +
b * xix +
c * (xix * xix) +
d * (xix * xix * xix);
180 for (std::size_t i = 0; i < z1.size(); ++i) {
181 T diff = std::abs(z1[i] - z2[i]);
182 if (std::abs(diff) > maxvecdiff) {
191std::vector<T>
linspace(T xmin, T xmax, std::size_t n) {
192 std::vector<T> x(n, 0.0);
194 for (std::size_t i = 0; i < n; ++i) {
195 x[i] = (xmax - xmin) / (n - 1) * i + xmin;
202 std::vector<T> x(n, 0.0);
203 T logxmin = log10(xmin), logxmax = log10(xmax);
205 for (std::size_t i = 0; i < n; ++i) {
206 x[i] = exp((logxmax - logxmin) / (n - 1) * i + logxmin);
212std::vector<T>
logspace(T xmin, T xmax, std::size_t n) {
213 std::vector<T> x(n, 0.0);
214 T logxmin = log(xmin), logxmax = log(xmax);
216 for (std::size_t i = 0; i < n; ++i) {
217 x[i] = exp((logxmax - logxmin) / (n - 1) * i + logxmin);
233 std::size_t N = vec.size(), L = 0, R = N - 1, M = (L + R) / 2;
243 if (L == vec.size() - 1) {
252 std::size_t MR = M, ML = M;
255 if (MR == vec.size() - 1) {
267 T rML = vec[ML] - val;
268 T rMR = vec[MR] - val;
270 if (rR * rML > 0 && rL * rML < 0) {
274 }
else if (rR * rMR < 0 && rL * rMR > 0) {
280 format(
"Unable to bisect segmented vector; neither chunk contains the solution val:%g left:(%g,%g) right:(%g,%g)", val, vec[L],
281 vec[ML], vec[MR], vec[R]));
286 if (rR * rM > 0 && rL * rM < 0) {
313 std::size_t N =
mat[j].size(), L = 0, R = N - 1, M = (L + R) / 2;
321 rR =
mat[R][j] - val;
324 if (L ==
mat.size() - 1) {
329 rL =
mat[L][j] - val;
332 std::size_t MR = M, ML = M;
335 if (MR ==
mat.size() - 1) {
347 T rML =
mat[ML][j] - val;
348 T rMR =
mat[MR][j] - val;
350 if (rR * rMR > 0 && rL * rML < 0) {
353 rR =
mat[ML][j] - val;
354 }
else if (rR * rMR < 0 && rL * rML > 0) {
357 rL =
mat[MR][j] - val;
360 format(
"Unable to bisect segmented vector slice; neither chunk contains the solution %g lef:(%g,%g) right:(%g,%g)", val,
mat[L][j],
365 rM =
mat[M][j] - val;
366 if (rR * rM > 0 && rL * rM < 0) {
369 rR =
mat[R][j] - val;
373 rL =
mat[L][j] - val;
387inline std::set<std::set<std::size_t>>
powerset(std::set<std::size_t>
const& set) {
388 std::set<std::set<std::size_t>> result;
389 std::vector<std::set<std::size_t>::const_iterator> elements;
391 std::set<std::size_t> tmp;
392 std::transform(elements.begin(), elements.end(), std::inserter(tmp, tmp.end()),
powerset_dereference);
394 if (!elements.empty() && ++elements.back() == set.end()) {
397 std::set<std::size_t>::const_iterator iter;
398 if (elements.empty()) {
401 iter = elements.back();
404 for (; iter != set.end(); ++iter) {
405 elements.push_back(iter);
408 }
while (!elements.empty());
415 double max = std::abs(A);
416 double min = std::abs(B);
422 return ((1.0 - min / max * 1e0) < D);
425 format(
"Too small numbers: %f cannot be tested with an accepted error of %f for a machine precision of %f. ", max, D,
DBL_EPSILON));
434 T sumx = std::accumulate(x.begin(), x.end(),
static_cast<T
>(0));
436 for (std::size_t i = 0; i < x.size(); ++i) {
448 std::vector<std::vector<double>>
A;
449 std::vector<double>
B;
456 void add_4value_constraints(
double x1,
double x2,
double x3,
double x4,
double y1,
double y2,
double y3,
double y4);
464 if (n == 0)
return 1;
469template <
class T1,
class T2>
478void MatInv_2(
double A[2][2],
double B[2][2]);
481double interp1d(
const std::vector<double>* x,
const std::vector<double>* y,
double x0);
482double powInt(
double x,
int y);
496#define POW5(x) ((x) * (x) * (x) * (x) * (x))
497#define POW6(x) ((x) * (x) * (x) * (x) * (x) * (x))
498#define POW7(x) ((x) * (x) * (x) * (x) * (x) * (x) * (x))
502 return (y1 - y0) / (x1 - x0) * (x - x0) + y0;
504template <
class T1,
class T2>
505T2
LinearInterp(
const std::vector<T1>& x,
const std::vector<T1>& y, std::size_t i0, std::size_t
i1, T2 val) {
516 L0 = ((x - x1) * (x - x2)) / ((x0 - x1) * (x0 - x2));
517 L1 = ((x - x0) * (x - x2)) / ((x1 - x0) * (x1 - x2));
518 L2 = ((x - x0) * (x - x1)) / ((x2 - x0) * (x2 - x1));
519 return L0 * f0 + L1 * f1 + L2 * f2;
521template <
class T1,
class T2>
522T2
QuadInterp(
const std::vector<T1>& x,
const std::vector<T1>& y, std::size_t i0, std::size_t
i1, std::size_t i2, T2 val) {
523 return QuadInterp(x[i0], x[
i1], x[i2], y[i0], y[
i1], y[i2],
static_cast<T1
>(val));
527T
CubicInterp(T x0, T x1, T x2, T x3, T f0, T f1, T f2, T f3, T x) {
533 L0 = ((x - x1) * (x - x2) * (x - x3)) / ((x0 - x1) * (x0 - x2) * (x0 - x3));
534 L1 = ((x - x0) * (x - x2) * (x - x3)) / ((x1 - x0) * (x1 - x2) * (x1 - x3));
535 L2 = ((x - x0) * (x - x1) * (x - x3)) / ((x2 - x0) * (x2 - x1) * (x2 - x3));
536 L3 = ((x - x0) * (x - x1) * (x - x2)) / ((x3 - x0) * (x3 - x1) * (x3 - x2));
537 return L0 * f0 + L1 * f1 + L2 * f2 + L3 * f3;
544 T L0 = ((x - x1) * (x - x2) * (x - x3)) / ((x0 - x1) * (x0 - x2) * (x0 - x3));
545 T dL0_dx = (1 / (x - x1) + 1 / (x - x2) + 1 / (x - x3)) * L0;
546 T L1 = ((x - x0) * (x - x2) * (x - x3)) / ((x1 - x0) * (x1 - x2) * (x1 - x3));
547 T dL1_dx = (1 / (x - x0) + 1 / (x - x2) + 1 / (x - x3)) * L1;
548 T L2 = ((x - x0) * (x - x1) * (x - x3)) / ((x2 - x0) * (x2 - x1) * (x2 - x3));
549 T dL2_dx = (1 / (x - x0) + 1 / (x - x1) + 1 / (x - x3)) * L2;
550 T L3 = ((x - x0) * (x - x1) * (x - x2)) / ((x3 - x0) * (x3 - x1) * (x3 - x2));
551 T dL3_dx = (1 / (x - x0) + 1 / (x - x1) + 1 / (x - x2)) * L3;
552 return dL0_dx * f0 + dL1_dx * f1 + dL2_dx * f2 + dL3_dx * f3;
554template <
class T1,
class T2>
555T2
CubicInterp(
const std::vector<T1>& x,
const std::vector<T1>& y, std::size_t i0, std::size_t
i1, std::size_t i2, std::size_t i3, T2 val) {
556 return CubicInterp(x[i0], x[
i1], x[i2], x[i3], y[i0], y[
i1], y[i2], y[i3],
static_cast<T1
>(val));
561 return (x >= std::min(x1, x2) && x <= std::max(x1, x2));
577void solve_cubic(
double a,
double b,
double c,
double d,
int& N,
double& x0,
double& x1,
double& x2);
579void solve_quartic(
double a,
double b,
double c,
double d,
double e,
int& N,
double& x0,
double& x1,
double& x2,
double& x3);
582inline T
min3(T x1, T x2, T x3) {
583 return std::min(std::min(x1, x2), x3);
586inline T
max3(T x1, T x2, T x3) {
587 return std::max(std::max(x1, x2), x3);
590inline T
min4(T x1, T x2, T x3, T x4) {
591 return std::min(std::min(std::min(x1, x2), x3), x4);
594inline T
max4(T x1, T x2, T x3, T x4) {
595 return std::max(std::max(std::max(x1, x2), x3), x4);
599 return std::abs(a - b) <= 1 *
DBL_EPSILON * std::max(std::abs(a), std::abs(b));
605 std::size_t N = x.size();
606 for (std::size_t i = 0; i < N; ++i) {
607 T axi = std::abs(x[i]);
618 std::size_t N = x.size();
619 for (std::size_t i = 0; i < N; ++i) {
620 T axi = std::abs(x[i]);
630 return static_cast<int>(1);
632 return static_cast<int>(0);
681double acosh(
double x) {
682 return log(x + sqrt(x * x - 1.0));
684double asinh(
double value) {
686 return log(value + sqrt(value * value + 1));
688 return -log(-value + sqrt(value * value + 1));
694#if defined(__ISPOWERPC__)
696double acosh(
double x) {
697 return log(x + sqrt(x * x - 1.0));
699double asinh(
double value) {
701 return log(value + sqrt(value * value + 1));
703 return -log(-value + sqrt(value * value + 1));
708#if defined(__ISPOWERPC__)