1#ifndef COOLPROP_NUMERICS_H
2#define COOLPROP_NUMERICS_H
17#if defined(HUGE_VAL) && !defined(_HUGE)
18# define _HUGE HUGE_VAL
21# if defined(HUGE) && !defined(_HUGE)
26template <
typename T,
size_t N>
35 return (x <= DBL_MAX && x >= -DBL_MAX);
39# define M_PI 3.14159265358979323846
60template <
typename X,
typename Y>
68 Spline(
const std::vector<X>& x,
const std::vector<Y>& y) {
69 if (x.size() != y.size()) {
70 std::cerr <<
"X and Y must be the same size " <<
'\n';
75 std::cerr <<
"Must have at least three points for interpolation" <<
'\n';
79 using size_type =
typename std::vector<X>::difference_type;
81 size_type n = y.size() - 1;
83 std::vector<Y> b(n), d(n), a(n), c(n + 1), l(n + 1), u(n + 1), z(n + 1);
84 std::vector<X> h(n + 1);
91 for (size_type i = 1; i < n; i++) {
92 h[i] = x[i + 1] - x[i];
93 l[i] = Y(2 * (x[i + 1] - x[i - 1])) - Y(h[i - 1]) * u[i - 1];
94 u[i] = Y(h[i]) / l[i];
95 a[i] = (Y(3) / Y(h[i])) * (y[i + 1] - y[i]) - (Y(3) / Y(h[i - 1])) * (y[i] - y[i - 1]);
96 z[i] = (a[i] - Y(h[i - 1]) * z[i - 1]) / l[i];
102 for (size_type j = n - 1; j >= 0; j--) {
103 c[j] = z[j] - u[j] * c[j + 1];
104 b[j] = (y[j + 1] - y[j]) / Y(h[j]) - (Y(h[j]) * (c[j + 1] + Y(2) * c[j])) / Y(3);
105 d[j] = (c[j + 1] - c[j]) / Y(3 * h[j]);
108 for (size_type i = 0; i < n; i++) {
121 typename std::vector<element_type>::const_iterator it;
132 if (
mElements.size() == 0)
return std::vector<Y>(xx.size());
134 typename std::vector<X>::const_iterator it;
135 typename std::vector<element_type>::const_iterator it2;
138 for (it = xx.begin(); it != xx.end(); it++) {
144 ys.push_back(it2->eval(*it));
155 Element(
X _x, Y _a, Y _b, Y _c, Y _d) :
x(_x),
a(_a),
b(_b),
c(_c),
d(_d) {}
159 return a +
b * xix +
c * (xix * xix) +
d * (xix * xix * xix);
181 for (std::size_t i = 0; i < z1.size(); ++i) {
182 T diff = std::abs(z1[i] - z2[i]);
183 if (std::abs(diff) > maxvecdiff) {
193 std::vector<T> x(n, 0.0);
195 for (std::size_t i = 0; i < n; ++i) {
196 x[i] = (xmax - xmin) / (n - 1) * i + xmin;
203 std::vector<T> x(n, 0.0);
204 T logxmin = log10(xmin), logxmax = log10(xmax);
206 for (std::size_t i = 0; i < n; ++i) {
207 x[i] = exp((logxmax - logxmin) / (n - 1) * i + logxmin);
214 std::vector<T> x(n, 0.0);
215 T logxmin = log(xmin), logxmax = log(xmax);
217 for (std::size_t i = 0; i < n; ++i) {
218 x[i] = exp((logxmax - logxmin) / (n - 1) * i + logxmin);
234 std::size_t N = vec.size(), L = 0, R = N - 1, M = (L + R) / 2;
244 if (L == vec.size() - 1) {
253 std::size_t MR = M, ML = M;
256 if (MR == vec.size() - 1) {
268 T rML = vec[ML] - val;
269 T rMR = vec[MR] - val;
271 if (rR * rML > 0 && rL * rML < 0) {
275 }
else if (rR * rMR < 0 && rL * rMR > 0) {
281 format(
"Unable to bisect segmented vector; neither chunk contains the solution val:%g left:(%g,%g) right:(%g,%g)", val, vec[L],
282 vec[ML], vec[MR], vec[R]));
287 if (rR * rM > 0 && rL * rM < 0) {
314 std::size_t N =
mat[j].size(), L = 0, R = N - 1, M = (L + R) / 2;
322 rR =
mat[R][j] - val;
325 if (L ==
mat.size() - 1) {
330 rL =
mat[L][j] - val;
333 std::size_t MR = M, ML = M;
336 if (MR ==
mat.size() - 1) {
348 T rML =
mat[ML][j] - val;
349 T rMR =
mat[MR][j] - val;
351 if (rR * rMR > 0 && rL * rML < 0) {
354 rR =
mat[ML][j] - val;
355 }
else if (rR * rMR < 0 && rL * rML > 0) {
358 rL =
mat[MR][j] - val;
361 format(
"Unable to bisect segmented vector slice; neither chunk contains the solution %g lef:(%g,%g) right:(%g,%g)", val,
mat[L][j],
366 rM =
mat[M][j] - val;
367 if (rR * rM > 0 && rL * rM < 0) {
370 rR =
mat[R][j] - val;
374 rL =
mat[L][j] - val;
388inline std::set<std::set<std::size_t>>
powerset(std::set<std::size_t>
const& set) {
389 std::set<std::set<std::size_t>> result;
390 std::vector<std::set<std::size_t>::const_iterator> elements;
392 std::set<std::size_t> tmp;
393 std::transform(elements.begin(), elements.end(), std::inserter(tmp, tmp.end()),
powerset_dereference);
395 if (!elements.empty() && ++elements.back() == set.end()) {
398 std::set<std::size_t>::const_iterator iter;
399 if (elements.empty()) {
402 iter = elements.back();
405 for (; iter != set.end(); ++iter) {
406 elements.push_back(iter);
409 }
while (!elements.empty());
416 double max = std::abs(A);
417 double min = std::abs(B);
423 return ((1.0 - min / max * 1e0) < D);
426 format(
"Too small numbers: %f cannot be tested with an accepted error of %f for a machine precision of %f. ", max, D,
DBL_EPSILON));
435 T sumx = std::accumulate(x.begin(), x.end(),
static_cast<T>(0));
437 for (std::size_t i = 0; i < x.size(); ++i) {
449 std::vector<std::vector<double>>
A;
450 std::vector<double>
B;
457 void add_4value_constraints(
double x1,
double x2,
double x3,
double x4,
double y1,
double y2,
double y3,
double y4);
465 if (n == 0)
return 1;
470template <
class T1,
class T2>
479void MatInv_2(
double A[2][2],
double B[2][2]);
482double interp1d(
const std::vector<double>* x,
const std::vector<double>* y,
double x0);
483double powInt(
double x,
int y);
497#define POW5(x) ((x) * (x) * (x) * (x) * (x))
498#define POW6(x) ((x) * (x) * (x) * (x) * (x) * (x))
499#define POW7(x) ((x) * (x) * (x) * (x) * (x) * (x) * (x))
503 return (y1 - y0) / (x1 - x0) * (x - x0) + y0;
505template <
class T1,
class T2>
506T2
LinearInterp(
const std::vector<T1>& x,
const std::vector<T1>& y, std::size_t i0, std::size_t
i1, T2 val) {
517 L0 = ((x - x1) * (x - x2)) / ((x0 - x1) * (x0 - x2));
518 L1 = ((x - x0) * (x - x2)) / ((x1 - x0) * (x1 - x2));
519 L2 = ((x - x0) * (x - x1)) / ((x2 - x0) * (x2 - x1));
520 return L0 * f0 + L1 * f1 + L2 * f2;
522template <
class T1,
class T2>
523T2
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) {
524 return QuadInterp(x[i0], x[
i1], x[i2], y[i0], y[
i1], y[i2],
static_cast<T1
>(val));
534 L0 = ((x - x1) * (x - x2) * (x - x3)) / ((x0 - x1) * (x0 - x2) * (x0 - x3));
535 L1 = ((x - x0) * (x - x2) * (x - x3)) / ((x1 - x0) * (x1 - x2) * (x1 - x3));
536 L2 = ((x - x0) * (x - x1) * (x - x3)) / ((x2 - x0) * (x2 - x1) * (x2 - x3));
537 L3 = ((x - x0) * (x - x1) * (x - x2)) / ((x3 - x0) * (x3 - x1) * (x3 - x2));
538 return L0 * f0 + L1 * f1 + L2 * f2 + L3 * f3;
545 T L0 = ((x - x1) * (x - x2) * (x - x3)) / ((x0 - x1) * (x0 - x2) * (x0 - x3));
546 T dL0_dx = (1 / (x - x1) + 1 / (x - x2) + 1 / (x - x3)) * L0;
547 T L1 = ((x - x0) * (x - x2) * (x - x3)) / ((x1 - x0) * (x1 - x2) * (x1 - x3));
548 T dL1_dx = (1 / (x - x0) + 1 / (x - x2) + 1 / (x - x3)) * L1;
549 T L2 = ((x - x0) * (x - x1) * (x - x3)) / ((x2 - x0) * (x2 - x1) * (x2 - x3));
550 T dL2_dx = (1 / (x - x0) + 1 / (x - x1) + 1 / (x - x3)) * L2;
551 T L3 = ((x - x0) * (x - x1) * (x - x2)) / ((x3 - x0) * (x3 - x1) * (x3 - x2));
552 T dL3_dx = (1 / (x - x0) + 1 / (x - x1) + 1 / (x - x2)) * L3;
553 return dL0_dx * f0 + dL1_dx * f1 + dL2_dx * f2 + dL3_dx * f3;
555template <
class T1,
class T2>
556T2
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) {
557 return CubicInterp(x[i0], x[
i1], x[i2], x[i3], y[i0], y[
i1], y[i2], y[i3],
static_cast<T1
>(val));
562 return (x >= std::min(x1, x2) && x <= std::max(x1, x2));
578void solve_cubic(
double a,
double b,
double c,
double d,
int& N,
double& x0,
double& x1,
double& x2);
580void solve_quartic(
double a,
double b,
double c,
double d,
double e,
int& N,
double& x0,
double& x1,
double& x2,
double& x3);
584 return std::min(std::min(x1, x2), x3);
588 return std::max(std::max(x1, x2), x3);
592 return std::min(std::min(std::min(x1, x2), x3), x4);
596 return std::max(std::max(std::max(x1, x2), x3), x4);
600 return std::abs(a - b) <= 1 *
DBL_EPSILON * std::max(std::abs(a), std::abs(b));
606 std::size_t N = x.size();
607 for (std::size_t i = 0; i < N; ++i) {
608 T axi = std::abs(x[i]);
619 std::size_t N = x.size();
620 for (std::size_t i = 0; i < N; ++i) {
621 T axi = std::abs(x[i]);
631 return static_cast<int>(1);
633 return static_cast<int>(0);
682double acosh(
double x) {
683 return log(x + sqrt(x * x - 1.0));
685double asinh(
double value) {
687 return log(value + sqrt(value * value + 1));
689 return -log(-value + sqrt(value * value + 1));
695#if defined(__ISPOWERPC__)
697double acosh(
double x) {
698 return log(x + sqrt(x * x - 1.0));
700double asinh(
double value) {
702 return log(value + sqrt(value * value + 1));
704 return -log(-value + sqrt(value * value + 1));
709#if defined(__ISPOWERPC__)