diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index 2abccdcf6a..388a06c0b2 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -214,7 +214,7 @@ jobs: run: config_info_travis working-directory: ../boost-root/libs/config/test - name: Test - run: ..\..\..\b2 --hash %ARGS% define=CI_SUPPRESS_KNOWN_ISSUES ${{ matrix.suite }} + run: ..\..\..\b2 --hash %ARGS% define=CI_SUPPRESS_KNOWN_ISSUES debug-symbols=off ${{ matrix.suite }} working-directory: ../boost-root/libs/math/test windows_gcc: runs-on: windows-2019 diff --git a/include/boost/math/distributions/detail/common_error_handling.hpp b/include/boost/math/distributions/detail/common_error_handling.hpp index b91d2036f9..f03f2c49b8 100644 --- a/include/boost/math/distributions/detail/common_error_handling.hpp +++ b/include/boost/math/distributions/detail/common_error_handling.hpp @@ -185,11 +185,12 @@ inline bool check_non_centrality( RealType* result, const Policy& pol) { - if((ncp < 0) || !(boost::math::isfinite)(ncp)) - { // Assume scale == 0 is NOT valid for any distribution. + static const RealType upper_limit = static_cast((std::numeric_limits::max)()) - boost::math::policies::get_max_root_iterations(); + if((ncp < 0) || !(boost::math::isfinite)(ncp) || ncp > upper_limit) + { *result = policies::raise_domain_error( function, - "Non centrality parameter is %1%, but must be > 0 !", ncp, pol); + "Non centrality parameter is %1%, but must be > 0, and a countable value such that x+1 != x", ncp, pol); return false; } return true; diff --git a/include/boost/math/distributions/non_central_beta.hpp b/include/boost/math/distributions/non_central_beta.hpp index 9d1eedecca..9266c9e576 100644 --- a/include/boost/math/distributions/non_central_beta.hpp +++ b/include/boost/math/distributions/non_central_beta.hpp @@ -430,7 +430,7 @@ namespace boost static_cast(p), &r, Policy())) - return (RealType)r; + return static_cast(r); // // Special cases first: // @@ -624,7 +624,7 @@ namespace boost static_cast(x), &r, Policy())) - return (RealType)r; + return static_cast(r); if(l == 0) return pdf(boost::math::beta_distribution(dist.alpha(), dist.beta()), x); @@ -761,7 +761,7 @@ namespace boost l, &r, Policy())) - return (RealType)r; + return static_cast(r); RealType c = a + b + l / 2; RealType mean = 1 - (b / c) * (1 + l / (2 * c * c)); return detail::generic_find_mode_01( @@ -872,7 +872,7 @@ namespace boost x, &r, Policy())) - return (RealType)r; + return static_cast(r); if(l == 0) return cdf(beta_distribution(a, b), x); @@ -909,7 +909,7 @@ namespace boost x, &r, Policy())) - return (RealType)r; + return static_cast(r); if(l == 0) return cdf(complement(beta_distribution(a, b), x)); diff --git a/include/boost/math/distributions/non_central_chi_squared.hpp b/include/boost/math/distributions/non_central_chi_squared.hpp index 5a84c734f2..c85c1d08d1 100644 --- a/include/boost/math/distributions/non_central_chi_squared.hpp +++ b/include/boost/math/distributions/non_central_chi_squared.hpp @@ -88,7 +88,7 @@ namespace boost // stable direction for the gamma function // recurrences: // - int i; + long long i; for(i = k; static_cast(i-k) < max_iter; ++i) { T term = poisf * gamf; @@ -299,7 +299,7 @@ namespace boost if(pois == 0) return 0; T poisb = pois; - for(int i = k; ; ++i) + for(long long i = k; ; ++i) { sum += pois; if(pois / sum < errtol) @@ -310,7 +310,7 @@ namespace boost "Series did not converge, closest value was %1%", sum, pol); pois *= l2 * x2 / ((i + 1) * (n2 + i)); } - for(int i = k - 1; i >= 0; --i) + for(long long i = k - 1; i >= 0; --i) { poisb *= (i + 1) * (n2 + i) / (l2 * x2); sum += poisb; @@ -428,7 +428,7 @@ namespace boost static_cast(p), &r, Policy())) - return (RealType)r; + return static_cast(r); // // Special cases get short-circuited first: // @@ -519,7 +519,7 @@ namespace boost (value_type)x, &r, Policy())) - return (RealType)r; + return static_cast(r); if(l == 0) return pdf(boost::math::chi_squared_distribution(dist.degrees_of_freedom()), x); @@ -821,7 +821,7 @@ namespace boost l, &r, Policy())) - return r; + return static_cast(r); return k + l; } // mean @@ -842,7 +842,7 @@ namespace boost l, &r, Policy())) - return (RealType)r; + return static_cast(r); bool asymptotic_mode = k < l/4; RealType starting_point = asymptotic_mode ? k + l - RealType(3) : RealType(1) + k; return detail::generic_find_mode(dist, starting_point, function); @@ -864,7 +864,7 @@ namespace boost l, &r, Policy())) - return r; + return static_cast(r); return 2 * (2 * l + k); } @@ -887,7 +887,7 @@ namespace boost l, &r, Policy())) - return r; + return static_cast(r); BOOST_MATH_STD_USING return pow(2 / (k + 2 * l), RealType(3)/2) * (k + 3 * l); } @@ -908,7 +908,7 @@ namespace boost l, &r, Policy())) - return r; + return static_cast(r); return 12 * (k + 4 * l) / ((k + 2 * l) * (k + 2 * l)); } // kurtosis_excess @@ -946,7 +946,7 @@ namespace boost x, &r, Policy())) - return r; + return static_cast(r); return detail::non_central_chi_squared_cdf(x, k, l, false, Policy()); } // cdf @@ -975,7 +975,7 @@ namespace boost x, &r, Policy())) - return r; + return static_cast(r); return detail::non_central_chi_squared_cdf(x, k, l, true, Policy()); } // ccdf diff --git a/include/boost/math/distributions/non_central_f.hpp b/include/boost/math/distributions/non_central_f.hpp index bb122029b8..e93d03e597 100644 --- a/include/boost/math/distributions/non_central_f.hpp +++ b/include/boost/math/distributions/non_central_f.hpp @@ -106,7 +106,7 @@ namespace boost l, &r, Policy())) - return r; + return r; if(v2 <= 2) return policies::raise_domain_error( function, @@ -137,7 +137,7 @@ namespace boost l, &r, Policy())) - return r; + return r; RealType guess = m > 2 ? RealType(m * (n + l) / (n * (m - 2))) : RealType(1); return detail::generic_find_mode( dist, @@ -166,7 +166,7 @@ namespace boost l, &r, Policy())) - return r; + return r; if(m <= 4) return policies::raise_domain_error( function, @@ -203,7 +203,7 @@ namespace boost l, &r, Policy())) - return r; + return r; if(m <= 6) return policies::raise_domain_error( function, @@ -240,7 +240,7 @@ namespace boost l, &r, Policy())) - return r; + return r; if(m <= 8) return policies::raise_domain_error( function, @@ -309,7 +309,7 @@ namespace boost dist.non_centrality(), &r, Policy())) - return r; + return r; if((x < 0) || !(boost::math::isfinite)(x)) { @@ -350,7 +350,7 @@ namespace boost c.dist.non_centrality(), &r, Policy())) - return r; + return r; if((c.param < 0) || !(boost::math::isfinite)(c.param)) { diff --git a/include/boost/math/distributions/non_central_t.hpp b/include/boost/math/distributions/non_central_t.hpp index 9468a56c5e..cb9af7e2e5 100644 --- a/include/boost/math/distributions/non_central_t.hpp +++ b/include/boost/math/distributions/non_central_t.hpp @@ -307,9 +307,9 @@ namespace boost function, v, &r, Policy()) || - !detail::check_finite( + !detail::check_non_centrality( function, - delta, + static_cast(delta * delta), &r, Policy()) || @@ -730,9 +730,9 @@ namespace boost detail::check_df_gt0_to_inf( function, v, &r, Policy()); - detail::check_finite( + detail::check_non_centrality( function, - lambda, + static_cast(lambda * lambda), &r, Policy()); } // non_central_t_distribution constructor. @@ -874,12 +874,12 @@ namespace boost function, v, &r, Policy()) || - !detail::check_finite( + !detail::check_non_centrality( function, - l, + static_cast(l * l), &r, Policy())) - return (RealType)r; + return static_cast(r); BOOST_MATH_STD_USING @@ -912,12 +912,12 @@ namespace boost function, v, &r, Policy()) || - !detail::check_finite( + !detail::check_non_centrality( function, - l, + static_cast(l * l), &r, Policy())) - return (RealType)r; + return static_cast(r); if(v <= 1) return policies::raise_domain_error( function, @@ -947,12 +947,12 @@ namespace boost function, v, &r, Policy()) || - !detail::check_finite( + !detail::check_non_centrality( function, - l, + static_cast(l * l), &r, Policy())) - return (RealType)r; + return static_cast(r); if(v <= 2) return policies::raise_domain_error( function, @@ -982,12 +982,12 @@ namespace boost function, v, &r, Policy()) || - !detail::check_finite( + !detail::check_non_centrality( function, - l, + static_cast(l * l), &r, Policy())) - return (RealType)r; + return static_cast(r); if(v <= 3) return policies::raise_domain_error( function, @@ -1014,12 +1014,12 @@ namespace boost function, v, &r, Policy()) || - !detail::check_finite( + !detail::check_non_centrality( function, - l, + static_cast(l * l), &r, Policy())) - return (RealType)r; + return static_cast(r); if(v <= 4) return policies::raise_domain_error( function, @@ -1053,9 +1053,9 @@ namespace boost function, v, &r, Policy()) || - !detail::check_finite( + !detail::check_non_centrality( function, - l, + static_cast(l * l), // we need l^2 to be countable. &r, Policy()) || @@ -1064,7 +1064,7 @@ namespace boost t, &r, Policy())) - return (RealType)r; + return static_cast(r); return policies::checked_narrowing_cast( detail::non_central_t_pdf(static_cast(v), static_cast(l), @@ -1093,9 +1093,9 @@ namespace boost function, v, &r, Policy()) || - !detail::check_finite( + !detail::check_non_centrality( function, - l, + static_cast(l * l), &r, Policy()) || @@ -1104,8 +1104,8 @@ namespace boost x, &r, Policy())) - return (RealType)r; - if ((boost::math::isinf)(v)) + return static_cast(r); + if ((boost::math::isinf)(v)) { // Infinite degrees of freedom, so use normal distribution located at delta. normal_distribution n(l, 1); cdf(n, x); @@ -1147,9 +1147,9 @@ namespace boost function, v, &r, Policy()) || - !detail::check_finite( + !detail::check_non_centrality( function, - l, + static_cast(l * l), &r, Policy()) || @@ -1158,7 +1158,7 @@ namespace boost x, &r, Policy())) - return (RealType)r; + return static_cast(r); if ((boost::math::isinf)(v)) { // Infinite degrees of freedom, so use normal distribution located at delta. diff --git a/include/boost/math/special_functions/detail/ibeta_inverse.hpp b/include/boost/math/special_functions/detail/ibeta_inverse.hpp index 9f257793d1..79eb95c45f 100644 --- a/include/boost/math/special_functions/detail/ibeta_inverse.hpp +++ b/include/boost/math/special_functions/detail/ibeta_inverse.hpp @@ -118,9 +118,17 @@ T temme_method_1_ibeta_inverse(T a, T b, T z, const Policy& pol) x = 0.5; else x = (1 + eta * sqrt((1 + c) / eta_2)) / 2; - - BOOST_MATH_ASSERT(x >= 0); - BOOST_MATH_ASSERT(x <= 1); + // + // These are post-conditions of the method, but the addition above + // may result in us being out by 1ulp either side of the boundary, + // so just check that we're in bounds and adjust as needed. + // See https://github.com/boostorg/math/issues/961 + // + if (x < 0) + x = 0; + else if (x > 1) + x = 1; + BOOST_MATH_ASSERT(eta * (x - 0.5) >= 0); #ifdef BOOST_INSTRUMENT std::cout << "Estimating x with Temme method 1: " << x << std::endl; @@ -901,6 +909,8 @@ T ibeta_inv_imp(T a, T b, T p, T q, const Policy& pol, T* py) if(x < lower) x = lower; } + std::uintmax_t max_iter = policies::get_max_root_iterations(); + std::uintmax_t max_iter_used = 0; // // Figure out how many digits to iterate towards: // @@ -923,10 +933,9 @@ T ibeta_inv_imp(T a, T b, T p, T q, const Policy& pol, T* py) // Now iterate, we can use either p or q as the target here // depending on which is smaller: // - std::uintmax_t max_iter = policies::get_max_root_iterations(); x = boost::math::tools::halley_iterate( boost::math::detail::ibeta_roots(a, b, (p < q ? p : q), (p < q ? false : true)), x, lower, upper, digits, max_iter); - policies::check_root_iterations("boost::math::ibeta<%1%>(%1%, %1%, %1%)", max_iter, pol); + policies::check_root_iterations("boost::math::ibeta<%1%>(%1%, %1%, %1%)", max_iter + max_iter_used, pol); // // We don't really want these asserts here, but they are useful for sanity // checking that we have the limits right, uncomment if you suspect bugs *only*. diff --git a/include/boost/math/tools/roots.hpp b/include/boost/math/tools/roots.hpp index 6493e81228..9fd7bee488 100644 --- a/include/boost/math/tools/roots.hpp +++ b/include/boost/math/tools/roots.hpp @@ -586,8 +586,8 @@ namespace detail { // we can jump way out of bounds if we're not careful. // See https://svn.boost.org/trac/boost/ticket/8314. delta = f0 / f1; - if (fabs(delta) > 2 * fabs(guess)) - delta = (delta < 0 ? -1 : 1) * 2 * fabs(guess); + if (fabs(delta) > 2 * fabs(result)) + delta = (delta < 0 ? -1 : 1) * 2 * fabs(result); } } else @@ -600,9 +600,19 @@ namespace detail { if ((convergence > 0.8) && (convergence < 2)) { // last two steps haven't converged. - delta = (delta > 0) ? (result - min) / 2 : (result - max) / 2; - if ((result != 0) && (fabs(delta) > result)) - delta = sign(delta) * fabs(result) * 0.9f; // protect against huge jumps! + if (fabs(min) < 1 ? fabs(1000 * min) < fabs(max) : fabs(max / min) > 1000) + { + if(delta > 0) + delta = bracket_root_towards_min(f, result, f0, min, max, count); + else + delta = bracket_root_towards_max(f, result, f0, min, max, count); + } + else + { + delta = (delta > 0) ? (result - min) / 2 : (result - max) / 2; + if ((result != 0) && (fabs(delta) > result)) + delta = sign(delta) * fabs(result) * 0.9f; // protect against huge jumps! + } // reset delta2 so that this branch will *not* be taken on the // next iteration: delta2 = delta * 3; @@ -641,6 +651,8 @@ namespace detail { result = guess - delta; if (result <= min) result = float_next(min); + if (result >= max) + result = float_prior(max); guess = min; continue; } @@ -669,6 +681,8 @@ namespace detail { result = guess - delta; if (result >= max) result = float_prior(max); + if (result <= min) + result = float_next(min); guess = min; continue; } diff --git a/test/Jamfile.v2 b/test/Jamfile.v2 index 8aa2bdcff0..daff6520e5 100644 --- a/test/Jamfile.v2 +++ b/test/Jamfile.v2 @@ -168,6 +168,7 @@ test-suite special_fun : [ run git_issue_705.cpp ../../test/build//boost_unit_test_framework ] [ run git_issue_810.cpp ../../test/build//boost_unit_test_framework ] [ run git_issue_826.cpp ../../test/build//boost_unit_test_framework ] + [ run git_issue_961.cpp ] [ run special_functions_test.cpp ../../test/build//boost_unit_test_framework ] [ run test_airy.cpp test_instances//test_instances pch_light ../../test/build//boost_unit_test_framework ] [ run test_bessel_j.cpp test_instances//test_instances pch_light ../../test/build//boost_unit_test_framework ] diff --git a/test/git_issue_961.cpp b/test/git_issue_961.cpp new file mode 100644 index 0000000000..7594dcd6fc --- /dev/null +++ b/test/git_issue_961.cpp @@ -0,0 +1,76 @@ +// (C) Copyright John Maddock 2023. +// Use, modification and distribution are subject to the +// Boost Software License, Version 1.0. (See accompanying file +// LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt) + +#include +#include +#include +#include + +using namespace std; +using namespace boost::math; + +void show_fp_exception_flags() +{ + if (std::fetestexcept(FE_DIVBYZERO)) { + cout << " FE_DIVBYZERO"; + } + // FE_INEXACT is common and not interesting. + // if (std::fetestexcept(FE_INEXACT)) { + // cout << " FE_INEXACT"; + // } + if (std::fetestexcept(FE_INVALID)) { + cout << " FE_INVALID"; + } + if (std::fetestexcept(FE_OVERFLOW)) { + cout << " FE_OVERFLOW"; + } + if (std::fetestexcept(FE_UNDERFLOW)) { + cout << " FE_UNDERFLOW"; + } + cout << endl; +} + +template +int test() +{ + double a = 14.208308325339239; + double b = a; + + double p = 6.4898872103239473e-300; // Throws exception: Assertion `x >= 0' failed. + // double p = 7.8e-307; // No flags set, returns 8.57354094063444939e-23 + // double p = 7.7e-307; // FE_UNDERFLOW set, returns 0.0 + + while (p > (std::numeric_limits::min)()) + { + std::feclearexcept(FE_ALL_EXCEPT); + + try { + + double x = ibeta_inv(a, b, p, Policy()); + + show_fp_exception_flags(); + + std::cout << std::scientific << std::setw(24) + << std::setprecision(17) << x << std::endl; + } + catch (const std::exception& e) + { + std::cout << e.what() << std::endl; + return 1; + } + + p /= 1.25; + } + + return 0; +} + +int main(int argc, char* argv[]) +{ + using namespace boost::math::policies; + if (test>()) + return 1; + return test>>(); +} diff --git a/test/test_nc_beta.cpp b/test/test_nc_beta.cpp index a2623e731b..3e7c08d0f3 100644 --- a/test/test_nc_beta.cpp +++ b/test/test_nc_beta.cpp @@ -257,6 +257,22 @@ void test_spots(RealType) BOOST_MATH_CHECK_THROW(skewness(dist), boost::math::evaluation_error); BOOST_MATH_CHECK_THROW(kurtosis(dist), boost::math::evaluation_error); BOOST_MATH_CHECK_THROW(kurtosis_excess(dist), boost::math::evaluation_error); + // + // Some special error handling tests, if the non-centrality param is too large + // then we have no evaluation method and should get a domain_error: + // + using std::ldexp; + using distro1 = boost::math::non_central_beta_distribution; + using distro2 = boost::math::non_central_beta_distribution>>; + using de = std::domain_error; + BOOST_MATH_CHECK_THROW(distro1(2, 3, ldexp(RealType(1), 100)), de); + if (std::numeric_limits::has_quiet_NaN) + { + distro2 d2(2, 3, ldexp(RealType(1), 100)); + BOOST_CHECK(boost::math::isnan(pdf(d2, 0.5))); + BOOST_CHECK(boost::math::isnan(cdf(d2, 0.5))); + BOOST_CHECK(boost::math::isnan(cdf(complement(d2, 0.5)))); + } } // template void test_spots(RealType) diff --git a/test/test_nc_chi_squared.hpp b/test/test_nc_chi_squared.hpp index d65d1b628f..b2fa6d75be 100644 --- a/test/test_nc_chi_squared.hpp +++ b/test/test_nc_chi_squared.hpp @@ -3,7 +3,10 @@ // Boost Software License, Version 1.0. (See accompanying file // LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt) +#ifndef BOOST_MATH_OVERFLOW_ERROR_POLICY #define BOOST_MATH_OVERFLOW_ERROR_POLICY ignore_error +#endif + #include #define BOOST_TEST_MAIN #include @@ -271,6 +274,22 @@ void test_spots(RealType) BOOST_MATH_CHECK_THROW(pdf(boost::math::non_central_chi_squared_distribution(1, -1), 0), std::domain_error); BOOST_MATH_CHECK_THROW(quantile(boost::math::non_central_chi_squared_distribution(1, 1), -1), std::domain_error); BOOST_MATH_CHECK_THROW(quantile(boost::math::non_central_chi_squared_distribution(1, 1), 2), std::domain_error); + // + // Some special error handling tests, if the non-centrality param is too large + // then we have no evaluation method and should get a domain_error: + // + using std::ldexp; + using distro1 = boost::math::non_central_chi_squared_distribution; + using distro2 = boost::math::non_central_chi_squared_distribution>>; + using de = std::domain_error; + BOOST_MATH_CHECK_THROW(distro1(2, ldexp(RealType(1), 100)), de); + if (std::numeric_limits::has_quiet_NaN) + { + distro2 d2(2, ldexp(RealType(1), 100)); + BOOST_CHECK(boost::math::isnan(pdf(d2, 0.5))); + BOOST_CHECK(boost::math::isnan(cdf(d2, 0.5))); + BOOST_CHECK(boost::math::isnan(cdf(complement(d2, 0.5)))); + } #endif } // template void test_spots(RealType) diff --git a/test/test_nc_f.cpp b/test/test_nc_f.cpp index b3123980ba..1cf411516d 100644 --- a/test/test_nc_f.cpp +++ b/test/test_nc_f.cpp @@ -293,6 +293,21 @@ void test_spots(RealType) BOOST_MATH_CHECK_THROW(pdf(boost::math::non_central_f_distribution(1, -1, 1), 0), std::domain_error); BOOST_MATH_CHECK_THROW(quantile(boost::math::non_central_f_distribution(1, 1, 1), -1), std::domain_error); BOOST_MATH_CHECK_THROW(quantile(boost::math::non_central_f_distribution(1, 1, 1), 2), std::domain_error); + // + // Some special error handling tests, if the non-centrality param is too large + // then we have no evaluation method and should get a domain_error: + // + using std::ldexp; + using distro1 = boost::math::non_central_f_distribution; + using distro2 = boost::math::non_central_f_distribution>>; + using de = std::domain_error; + BOOST_MATH_CHECK_THROW(distro1(2, 3, ldexp(RealType(1), 100)), de); + if (std::numeric_limits::has_quiet_NaN) + { + distro2 d2(2, 3, ldexp(RealType(1), 100)); + BOOST_CHECK(boost::math::isnan(pdf(d2, 0.5))); + BOOST_CHECK(boost::math::isnan(cdf(d2, 0.5))); + } } // template void test_spots(RealType) BOOST_AUTO_TEST_CASE( test_main ) diff --git a/test/test_nc_t.hpp b/test/test_nc_t.hpp index 42f10f9b3e..8fb6accdf3 100644 --- a/test/test_nc_t.hpp +++ b/test/test_nc_t.hpp @@ -3,7 +3,10 @@ // Boost Software License, Version 1.0. (See accompanying file // LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt) +#ifndef BOOST_MATH_OVERFLOW_ERROR_POLICY #define BOOST_MATH_OVERFLOW_ERROR_POLICY ignore_error +#endif + #include #define BOOST_TEST_MAIN #include @@ -302,6 +305,21 @@ void test_spots(RealType) BOOST_MATH_CHECK_THROW(pdf(boost::math::non_central_t_distribution(-1, 1), 0), std::domain_error); BOOST_MATH_CHECK_THROW(quantile(boost::math::non_central_t_distribution(1, 1), -1), std::domain_error); BOOST_MATH_CHECK_THROW(quantile(boost::math::non_central_t_distribution(1, 1), 2), std::domain_error); + // + // Some special error handling tests, if the non-centrality param is too large + // then we have no evaluation method and should get a domain_error: + // + using std::ldexp; + using distro1 = boost::math::non_central_t_distribution; + using distro2 = boost::math::non_central_t_distribution>>; + using de = std::domain_error; + BOOST_MATH_CHECK_THROW(distro1(2, ldexp(RealType(1), 100)), de); + if (std::numeric_limits::has_quiet_NaN) + { + distro2 d2(2, ldexp(RealType(1), 100)); + BOOST_CHECK(boost::math::isnan(pdf(d2, 0.5))); + BOOST_CHECK(boost::math::isnan(cdf(d2, 0.5))); + } } // template void test_spots(RealType) template