a8a8e5f56a
- Fix up last few digits of a lot of known-answer tests. Confirmed with GNU mpfr to 200 bits of precision and cross-checked with whatever libm Ubuntu ships with. - Test relative error, not absolute error. - Set bounds in terms of *_EPSILON, not magic numbers. *_EPSILON is twice the largest relative error of a correctly rounded operation, and equal to the largest relative error of an operation with up to 1ulp error. Most of the operations we're testing are not correctly rounded, but they ought to be no more than 1ulp away. For the few cases where that's not a priori clear (like comparing cbrt and pow(x, 1/3)), use twice *_EPSILON to allow some leeway. - Write the success condition positively as error <= eps. This comes out false if the result is a NaN, meaning failure. In contrast, if we write error > eps for the _failure_ condition, then if the result is a NaN, it will also come out false, but meaning success, which is not what we want. - Fix the trigonometric test cases near bad spots. sin(pi - d) for nonzero d is not zero; it is d + O(d^3). pi is not a floating-point number, so these results should be approximately the nonzero error of our approximation to pi. Likewise with cos(pi/2 - d) and tan(pi + d). (Yes, I know the sin _function_ is ill-conditioned near pi so you shouldn't pass approximate inputs near there, but that's separate from whether a sin _implementation_ gives an answer that is wrong by quintillions of ulps.) Since on x86 (i386 and amd64 alike) we currently use x87 hardware trigonometric instructions, which are bad, these are marked xfail on x86 for now until we switch to software implementations (coming soon to a repository near you). - Use %.8g, %.17g, %.35g to print float, double, long double in failures. This should be enough to identify the problematic outputs and/or reproduce the computation, even if long double is binary128 with 115 bits of precision. If there are any new libm test failures after this, tell me what architecture you're on and send me the atf output and I'll try to figure it out.
63 lines
2.1 KiB
C
63 lines
2.1 KiB
C
/* $NetBSD: t_libm.h,v 1.7 2018/11/07 03:59:36 riastradh Exp $ */
|
|
|
|
/*
|
|
* Check result of fn(arg) is correct within the bounds.
|
|
* Should be ok to do the checks using 'double' for 'float' functions.
|
|
* On i386 float and double values are returned on the x87 stack and might
|
|
* be out of range for the function - so save and print as 'long double'.
|
|
* (otherwise you can get 'inf != inf' reported!)
|
|
*/
|
|
#define T_LIBM_CHECK(subtest, fn, arg, expect_, epsilon_) do { \
|
|
long double epsilon = epsilon_; \
|
|
long double expect = expect_; \
|
|
long double r = fn(arg); \
|
|
long double e = fabsl((r - expect)/expect); \
|
|
if (!(r == expect || e <= epsilon)) \
|
|
atf_tc_fail_nonfatal( \
|
|
"subtest %u: " #fn "(%g) is %Lg (%.14La) " \
|
|
"not %Lg (%.13La), error %Lg (%.6La) > %Lg", \
|
|
subtest, arg, r, r, expect, expect, e, e, epsilon); \
|
|
} while (0)
|
|
|
|
/* Check that the result of fn(arg) is NaN */
|
|
#ifndef __vax__
|
|
#define T_LIBM_CHECK_NAN(subtest, fn, arg) do { \
|
|
double r = fn(arg); \
|
|
if (!isnan(r)) \
|
|
atf_tc_fail_nonfatal("subtest %u: " #fn "(%g) is %g not NaN", \
|
|
subtest, arg, r); \
|
|
} while (0)
|
|
#else
|
|
/* vax doesn't support NaN */
|
|
#define T_LIBM_CHECK_NAN(subtest, fn, arg) (void)(arg)
|
|
#endif
|
|
|
|
/* Check that the result of fn(arg) is +0.0 */
|
|
#define T_LIBM_CHECK_PLUS_ZERO(subtest, fn, arg) do { \
|
|
double r = fn(arg); \
|
|
if (fabs(r) > 0.0 || signbit(r) != 0) \
|
|
atf_tc_fail_nonfatal("subtest %u: " #fn "(%g) is %g not +0.0", \
|
|
subtest, arg, r); \
|
|
} while (0)
|
|
|
|
/* Check that the result of fn(arg) is -0.0 */
|
|
#define T_LIBM_CHECK_MINUS_ZERO(subtest, fn, arg) do { \
|
|
double r = fn(arg); \
|
|
if (fabs(r) > 0.0 || signbit(r) == 0) \
|
|
atf_tc_fail_nonfatal("subtest %u: " #fn "(%g) is %g not -0.0", \
|
|
subtest, arg, r); \
|
|
} while (0)
|
|
|
|
/* Some useful constants (for test vectors) */
|
|
#ifndef __vax__ /* no NAN nor +/- INF on vax */
|
|
#define T_LIBM_NAN (0.0 / 0.0)
|
|
#define T_LIBM_PLUS_INF (+1.0 / 0.0)
|
|
#define T_LIBM_MINUS_INF (-1.0 / 0.0)
|
|
#endif
|
|
|
|
/* One line definition of a simple test */
|
|
#define ATF_LIBM_TEST(name, description) \
|
|
ATF_TC(name); \
|
|
ATF_TC_HEAD(name, tc) { atf_tc_set_md_var(tc, "descr", description); } \
|
|
ATF_TC_BODY(name, tc)
|