From f6c6de5d63a0c68e71d846604779867ce126d91b Mon Sep 17 00:00:00 2001 From: Srinivas Vasudevan Date: Tue, 14 Jan 2020 21:32:48 +0000 Subject: Ensure Igamma does not NaN or Inf for large values. --- .../src/SpecialFunctions/SpecialFunctionsImpl.h | 38 +++++++++++++++++++--- 1 file changed, 33 insertions(+), 5 deletions(-) (limited to 'unsupported/Eigen/src') diff --git a/unsupported/Eigen/src/SpecialFunctions/SpecialFunctionsImpl.h b/unsupported/Eigen/src/SpecialFunctions/SpecialFunctionsImpl.h index fe3f6d710..425231aa6 100644 --- a/unsupported/Eigen/src/SpecialFunctions/SpecialFunctionsImpl.h +++ b/unsupported/Eigen/src/SpecialFunctions/SpecialFunctionsImpl.h @@ -713,6 +713,18 @@ struct cephes_helper { enum IgammaComputationMode { VALUE, DERIVATIVE, SAMPLE_DERIVATIVE }; +template +static EIGEN_STRONG_INLINE Scalar main_igamma_term(Scalar a, Scalar x) { + /* Compute x**a * exp(-x) / gamma(a) */ + Scalar logax = a * numext::log(x) - x - lgamma_impl::run(a); + if (logax < -numext::log(NumTraits::highest()) || + // Assuming x and a aren't Nan. + (numext::isnan)(logax)) { + return Scalar(0); + } + return numext::exp(logax); +} + template EIGEN_DEVICE_FUNC int igamma_num_iterations() { @@ -755,6 +767,15 @@ struct igammac_cf_impl { return zero; } + Scalar ax = main_igamma_term(a, x); + // This is independent of mode. If this value is zero, + // then the function value is zero. If the function value is zero, + // then we are in a neighborhood where the function value evalutes to zero, + // so the derivative is zero. + if (ax == zero) { + return zero; + } + // continued fraction Scalar y = one - a; Scalar z = x + y + one; @@ -825,9 +846,7 @@ struct igammac_cf_impl { } /* Compute x**a * exp(-x) / gamma(a) */ - Scalar logax = a * numext::log(x) - x - lgamma_impl::run(a); Scalar dlogax_da = numext::log(x) - digamma_impl::run(a); - Scalar ax = numext::exp(logax); Scalar dax_da = ax * dlogax_da; switch (mode) { @@ -858,6 +877,18 @@ struct igamma_series_impl { const Scalar one = 1; const Scalar machep = cephes_helper::machep(); + Scalar ax = main_igamma_term(a, x); + + // This is independent of mode. If this value is zero, + // then the function value is zero. If the function value is zero, + // then we are in a neighborhood where the function value evalutes to zero, + // so the derivative is zero. + if (ax == zero) { + return zero; + } + + ax /= a; + /* power series */ Scalar r = a; Scalar c = one; @@ -886,10 +917,7 @@ struct igamma_series_impl { } } - /* Compute x**a * exp(-x) / gamma(a + 1) */ - Scalar logax = a * numext::log(x) - x - lgamma_impl::run(a + one); Scalar dlogax_da = numext::log(x) - digamma_impl::run(a + one); - Scalar ax = numext::exp(logax); Scalar dax_da = ax * dlogax_da; switch (mode) { -- cgit v1.2.3