From 22ab502072f42256373a94cae84244505366f168 Mon Sep 17 00:00:00 2001 From: Sam Hocevar Date: Wed, 2 Mar 2022 14:28:54 +0100 Subject: [PATCH] Refactor erf() and add erfc() --- include/lol/private/types/real.h | 1 + include/lol/private/types/real.ipp | 70 ++++++++++++++++++------------ 2 files changed, 43 insertions(+), 28 deletions(-) diff --git a/include/lol/private/types/real.h b/include/lol/private/types/real.h index f262efb7..331f1662 100644 --- a/include/lol/private/types/real.h +++ b/include/lol/private/types/real.h @@ -122,6 +122,7 @@ public: template friend real_t exp2(real_t const &x); template friend real_t expm1(real_t const &x); template friend real_t erf(real_t const &x); + template friend real_t erfc(real_t const &x); template friend real_t log(real_t const &x); template friend real_t log2(real_t const &x); template friend real_t log10(real_t const &x); diff --git a/include/lol/private/types/real.ipp b/include/lol/private/types/real.ipp index c2f46d3c..6a05a195 100644 --- a/include/lol/private/types/real.ipp +++ b/include/lol/private/types/real.ipp @@ -1165,8 +1165,8 @@ template real_t erf(real_t const &x) { /* Strategy for erf(x): * - if x<0, erf(x) = -erf(-x) + * - if x>7, erf(x) = 1-erfc(x) * - if x<7, erf(x) = sum((-1)^n·x^(2n+1)/((2n+1)·n!))/sqrt(π/4) - * - if x≥7, erf(x) = 1+exp(-x²)/(x·sqrt(π))·sum((-1)^n·(2n-1)!!/(2x²)^n * * FIXME: do not compute factorials at each iteration, accumulate * them instead (see fast_exp_sub). @@ -1177,39 +1177,53 @@ template real_t erf(real_t const &x) if (x.is_negative()) return -erf(-x); + // FIXME: this test is inefficient; the series below converges slowly for 1≤x≤7, + // but on the other hand there are accuracy issues with the erfc() implementation + // in that range. + if (x > real_t(7)) + return real_t::R_1() - erfc(x); + auto sum = real_t::R_0(); auto x2 = x * x; - - /* FIXME: this test is inefficient; the series converges slowly for x≥1 */ - if (x < real_t(7)) + auto xn = x, xmul = x2; + for (int n = 0;; ++n, xn *= xmul) { - auto xn = x, xmul = x2; - for (int n = 0;; ++n, xn *= xmul) - { - auto tmp = xn / (fast_fact(n) * (2 * n + 1)); - auto newsum = (n & 1) ? sum - tmp : sum + tmp; - if (newsum == sum) - break; - sum = newsum; - } - return sum * real_t::R_2_SQRTPI(); + auto tmp = xn / (fast_fact(n) * (2 * n + 1)); + auto newsum = (n & 1) ? sum - tmp : sum + tmp; + if (newsum == sum) + break; + sum = newsum; } - else - { - auto xn = real_t::R_1(), xmul = inverse(x2 + x2); - /* FIXME: this does not converge well! We need to stop at 30 - * iterations and sacrifice some accuracy. */ - for (int n = 0; n < 30; ++n, xn *= xmul) - { - auto tmp = xn * fast_fact(n * 2 - 1, 2); - auto newsum = (n & 1) ? sum - tmp : sum + tmp; - if (newsum == sum) - break; - sum = newsum; - } - return real_t::R_1() - exp(-x2) / (x * sqrt(real_t::R_PI())) * sum; + return sum * real_t::R_2_SQRTPI(); +} + +template real_t erfc(real_t const &x) +{ + // Strategy for erfc(x): + // - if x<1, erfc(x) = 1-erf(x) + // - if x≥1, erfc(x) = exp(-x²)/(x·sqrt(π))·sum((-1)^n·(2n-1)!!/(2x²)^n) + if (x < real_t::R_1()) + return real_t::R_1() - erf(x); + + auto sum = real_t::R_0(); + auto x2 = x * x; + // XXX: The series does not converge, there is an optimal value around + // ⌊x²+0.5⌋. If maximum accuracy cannot be reached, we must stop summing + // when divergence is detected. + auto prev = x2; + auto xn = real_t::R_1(), xmul = inverse(x2 + x2); + for (int n = 0;; ++n, xn *= xmul) + { + auto tmp = xn * fast_fact(n * 2 - 1, 2); + auto newsum = (n & 1) ? sum - tmp : sum + tmp; + if (newsum == sum || tmp > prev) + break; + sum = newsum; + prev = tmp; } + + return exp(-x2) / (x * sqrt(real_t::R_PI())) * sum; } template real_t sinh(real_t const &x)