TEST(math, nexttowardl) { ASSERT_DOUBLE_EQ(0.0L, nexttowardl(0.0L, 0.0L)); // Use a runtime value to accomodate the case when // sizeof(double) == sizeof(long double) long double smallest_positive = ldexpl(1.0L, LDBL_MIN_EXP - LDBL_MANT_DIG); ASSERT_DOUBLE_EQ(smallest_positive, nexttowardl(0.0L, 1.0L)); ASSERT_DOUBLE_EQ(-smallest_positive, nexttowardl(0.0L, -1.0L)); }
void test_nexttoward() { static_assert((std::is_same<decltype(nexttoward((double)0, (long double)0)), double>::value), ""); static_assert((std::is_same<decltype(nexttowardf(0, (long double)0)), float>::value), ""); static_assert((std::is_same<decltype(nexttowardl(0, (long double)0)), long double>::value), ""); assert(nexttoward(0, 1) == hexfloat<double>(0x1, 0, -1074)); }
int main(int argc, char *argv[]) { static const int ex_under = FE_UNDERFLOW | FE_INEXACT; /* shorthand */ static const int ex_over = FE_OVERFLOW | FE_INEXACT; long double ldbl_small, ldbl_eps, ldbl_max; printf("1..5\n"); #ifdef __i386__ fpsetprec(FP_PE); #endif /* * We can't use a compile-time constant here because gcc on * FreeBSD/i386 assumes long doubles are truncated to the * double format. */ ldbl_small = ldexpl(1.0, LDBL_MIN_EXP - LDBL_MANT_DIG); ldbl_eps = LDBL_EPSILON; ldbl_max = ldexpl(1.0 - ldbl_eps / 2, LDBL_MAX_EXP); /* * Special cases involving zeroes. */ #define ztest(prec) \ test##prec(copysign##prec(1.0, nextafter##prec(0.0, -0.0)), -1.0, 0); \ test##prec(copysign##prec(1.0, nextafter##prec(-0.0, 0.0)), 1.0, 0); \ test##prec(copysign##prec(1.0, nexttoward##prec(0.0, -0.0)), -1.0, 0);\ test##prec(copysign##prec(1.0, nexttoward##prec(-0.0, 0.0)), 1.0, 0) ztest(); ztest(f); ztest(l); #undef ztest #define stest(next, eps, prec) \ test##prec(next(-0.0, 42.0), eps, ex_under); \ test##prec(next(0.0, -42.0), -eps, ex_under); \ test##prec(next(0.0, INFINITY), eps, ex_under); \ test##prec(next(-0.0, -INFINITY), -eps, ex_under) stest(nextafter, 0x1p-1074, ); stest(nextafterf, 0x1p-149f, f); stest(nextafterl, ldbl_small, l); stest(nexttoward, 0x1p-1074, ); stest(nexttowardf, 0x1p-149f, f); stest(nexttowardl, ldbl_small, l); #undef stest printf("ok 1 - next\n"); /* * `x == y' and NaN tests */ testall(42.0, 42.0, 42.0, 0); testall(-42.0, -42.0, -42.0, 0); testall(INFINITY, INFINITY, INFINITY, 0); testall(-INFINITY, -INFINITY, -INFINITY, 0); testall(NAN, 42.0, NAN, 0); testall(42.0, NAN, NAN, 0); testall(NAN, NAN, NAN, 0); printf("ok 2 - next\n"); /* * Tests where x is an ordinary normalized number */ testboth(1.0, 2.0, 1.0 + DBL_EPSILON, 0, ); testboth(1.0, -INFINITY, 1.0 - DBL_EPSILON/2, 0, ); testboth(1.0, 2.0, 1.0 + FLT_EPSILON, 0, f); testboth(1.0, -INFINITY, 1.0 - FLT_EPSILON/2, 0, f); testboth(1.0, 2.0, 1.0 + ldbl_eps, 0, l); testboth(1.0, -INFINITY, 1.0 - ldbl_eps/2, 0, l); testboth(-1.0, 2.0, -1.0 + DBL_EPSILON/2, 0, ); testboth(-1.0, -INFINITY, -1.0 - DBL_EPSILON, 0, ); testboth(-1.0, 2.0, -1.0 + FLT_EPSILON/2, 0, f); testboth(-1.0, -INFINITY, -1.0 - FLT_EPSILON, 0, f); testboth(-1.0, 2.0, -1.0 + ldbl_eps/2, 0, l); testboth(-1.0, -INFINITY, -1.0 - ldbl_eps, 0, l); /* Cases where nextafter(...) != nexttoward(...) */ test(nexttoward(1.0, 1.0 + ldbl_eps), 1.0 + DBL_EPSILON, 0); testf(nexttowardf(1.0, 1.0 + ldbl_eps), 1.0 + FLT_EPSILON, 0); testl(nexttowardl(1.0, 1.0 + ldbl_eps), 1.0 + ldbl_eps, 0); printf("ok 3 - next\n"); /* * Tests at word boundaries, normalization boundaries, etc. */ testboth(0x1.87654ffffffffp+0, INFINITY, 0x1.87655p+0, 0, ); testboth(0x1.87655p+0, -INFINITY, 0x1.87654ffffffffp+0, 0, ); testboth(0x1.fffffffffffffp+0, INFINITY, 0x1p1, 0, ); testboth(0x1p1, -INFINITY, 0x1.fffffffffffffp+0, 0, ); testboth(0x0.fffffffffffffp-1022, INFINITY, 0x1p-1022, 0, ); testboth(0x1p-1022, -INFINITY, 0x0.fffffffffffffp-1022, ex_under, ); testboth(0x1.fffffep0f, INFINITY, 0x1p1, 0, f); testboth(0x1p1, -INFINITY, 0x1.fffffep0f, 0, f); testboth(0x0.fffffep-126f, INFINITY, 0x1p-126f, 0, f); testboth(0x1p-126f, -INFINITY, 0x0.fffffep-126f, ex_under, f); #if LDBL_MANT_DIG == 53 testboth(0x1.87654ffffffffp+0L, INFINITY, 0x1.87655p+0L, 0, l); testboth(0x1.87655p+0L, -INFINITY, 0x1.87654ffffffffp+0L, 0, l); testboth(0x1.fffffffffffffp+0L, INFINITY, 0x1p1L, 0, l); testboth(0x1p1L, -INFINITY, 0x1.fffffffffffffp+0L, 0, l); testboth(0x0.fffffffffffffp-1022L, INFINITY, 0x1p-1022L, 0, l); testboth(0x1p-1022L, -INFINITY, 0x0.fffffffffffffp-1022L, ex_under, l); #elif LDBL_MANT_DIG == 64 && !defined(__i386) testboth(0x1.87654321fffffffep+0L, INFINITY, 0x1.87654322p+0L, 0, l); testboth(0x1.87654322p+0L, -INFINITY, 0x1.87654321fffffffep+0L, 0, l); testboth(0x1.fffffffffffffffep0L, INFINITY, 0x1p1L, 0, l); testboth(0x1p1L, -INFINITY, 0x1.fffffffffffffffep0L, 0, l); testboth(0x0.fffffffffffffffep-16382L, INFINITY, 0x1p-16382L, 0, l); testboth(0x1p-16382L, -INFINITY, 0x0.fffffffffffffffep-16382L, ex_under, l); #elif LDBL_MANT_DIG == 113 testboth(0x1.876543210987ffffffffffffffffp+0L, INFINITY, 0x1.876543210988p+0, 0, l); testboth(0x1.876543210988p+0L, -INFINITY, 0x1.876543210987ffffffffffffffffp+0L, 0, l); testboth(0x1.ffffffffffffffffffffffffffffp0L, INFINITY, 0x1p1L, 0, l); testboth(0x1p1L, -INFINITY, 0x1.ffffffffffffffffffffffffffffp0L, 0, l); testboth(0x0.ffffffffffffffffffffffffffffp-16382L, INFINITY, 0x1p-16382L, 0, l); testboth(0x1p-16382L, -INFINITY, 0x0.ffffffffffffffffffffffffffffp-16382L, ex_under, l); #endif printf("ok 4 - next\n"); /* * Overflow tests */ test(idd(nextafter(DBL_MAX, INFINITY)), INFINITY, ex_over); test(idd(nextafter(INFINITY, 0.0)), DBL_MAX, 0); test(idd(nexttoward(DBL_MAX, DBL_MAX * 2.0L)), INFINITY, ex_over); #if LDBL_MANT_DIG > 53 test(idd(nexttoward(INFINITY, DBL_MAX * 2.0L)), DBL_MAX, 0); #endif testf(idf(nextafterf(FLT_MAX, INFINITY)), INFINITY, ex_over); testf(idf(nextafterf(INFINITY, 0.0)), FLT_MAX, 0); testf(idf(nexttowardf(FLT_MAX, FLT_MAX * 2.0)), INFINITY, ex_over); testf(idf(nexttowardf(INFINITY, FLT_MAX * 2.0)), FLT_MAX, 0); testboth(ldbl_max, INFINITY, INFINITY, ex_over, l); testboth(INFINITY, 0.0, ldbl_max, 0, l); printf("ok 5 - next\n"); return (0); }
static int testl(long double long_double_x, int int_x, long long_x) { int r = 0; r += __finitel(long_double_x); r += __fpclassifyl(long_double_x); r += __isinfl(long_double_x); r += __isnanl(long_double_x); r += __signbitl(long_double_x); r += acoshl(long_double_x); r += acosl(long_double_x); r += asinhl(long_double_x); r += asinl(long_double_x); r += atan2l(long_double_x, long_double_x); r += atanhl(long_double_x); r += atanl(long_double_x); r += cbrtl(long_double_x); r += ceill(long_double_x); r += copysignl(long_double_x, long_double_x); r += coshl(long_double_x); r += cosl(long_double_x); r += erfcl(long_double_x); r += erfl(long_double_x); r += exp2l(long_double_x); r += expl(long_double_x); r += expm1l(long_double_x); r += fabsl(long_double_x); r += fdiml(long_double_x, long_double_x); r += floorl(long_double_x); r += fmal(long_double_x, long_double_x, long_double_x); r += fmaxl(long_double_x, long_double_x); r += fminl(long_double_x, long_double_x); r += fmodl(long_double_x, long_double_x); r += frexpl(long_double_x, &int_x); r += hypotl(long_double_x, long_double_x); r += ilogbl(long_double_x); r += ldexpl(long_double_x, int_x); r += lgammal(long_double_x); r += llrintl(long_double_x); r += llroundl(long_double_x); r += log10l(long_double_x); r += log1pl(long_double_x); r += log2l(long_double_x); r += logbl(long_double_x); r += logl(long_double_x); r += lrintl(long_double_x); r += lroundl(long_double_x); r += modfl(long_double_x, &long_double_x); r += nearbyintl(long_double_x); r += nextafterl(long_double_x, long_double_x); r += nexttowardl(long_double_x, long_double_x); r += powl(long_double_x, long_double_x); r += remainderl(long_double_x, long_double_x); r += remquol(long_double_x, long_double_x, &int_x); r += rintl(long_double_x); r += roundl(long_double_x); r += scalblnl(long_double_x, long_x); r += scalbnl(long_double_x, int_x); r += sinhl(long_double_x); r += sinl(long_double_x); r += sqrtl(long_double_x); r += tanhl(long_double_x); r += tanl(long_double_x); r += tgammal(long_double_x); r += truncl(long_double_x); return r; }
static int do_test (void) { int result = 0; #ifndef NO_LONG_DOUBLE { long double x = 0x100000001ll + (long double) 0.5; long double q; long double r; r = modfl (x, &q); if (q != (long double) 0x100000001ll || r != 0.5) { printf ("modfl (%Lg, ...) failed\n", x); result = 1; } } { long double x; long double m; long double r; int e; int i; # if LDBL_MANT_DIG == 64 m = 0xf.fffffffffffffffp-4L; # elif LDBL_MANT_DIG == 106 /* This has to match the mantissa of LDBL_MAX which actually does have a missing bit in the middle. */ m = 0x1.fffffffffffff7ffffffffffff8p-1L; # elif LDBL_MANT_DIG == 113 m = 0x1.ffffffffffffffffffffffffffffp-1L; # else # error "Please adjust" # endif for (i = LDBL_MAX_EXP, x = LDBL_MAX; i >= LDBL_MIN_EXP; --i, x /= 2.0L) { printf ("2^%d: ", i); r = frexpl (x, &e); if (r != m) { printf ("mantissa incorrect: %.20La\n", r); result = 1; continue; } if (e != i) { printf ("exponent wrong %d (%.20Lg)\n", e, x); result = 1; continue; } puts ("ok"); } for (i = LDBL_MIN_EXP, x = LDBL_MIN; i >= LDBL_MIN_EXP - LDBL_MANT_DIG + 1; --i, x /= 2.0L) { printf ("2^%d: ", i); r = frexpl (x, &e); if (r != 0.5L) { printf ("mantissa incorrect: %.20La\n", r); result = 1; continue; } if (e != i) { printf ("exponent wrong %d (%.20Lg)\n", e, x); result = 1; continue; } puts ("ok"); } } # if 0 { int e; long double r = frexpl (LDBL_MIN * LDBL_EPSILON, &e); if (r != 0.5) { printf ("frexpl (LDBL_MIN * LDBL_EPSILON, ...): mantissa wrong: %Lg\n", r); result = 1; } else if (e != -16444) { printf ("frexpl (LDBL_MIN * LDBL_EPSILON, ...): exponent wrong: %d\n", e); result = 1; } } # endif #endif { double x = 0x100000001ll + (double) 0.5; double q; double r; r = modf (x, &q); if (q != (double) 0x100000001ll || r != 0.5) { printf ("modf (%g, ...) failed\n", x); result = 1; } } { union ieee754_float v1; union ieee754_float v2; float f; v1.f = f = FLT_MIN; if (fpclassify (f) != FP_NORMAL) { printf ("fpclassify (FLT_MIN) failed: %d\n", fpclassify (f)); result = 1; } f = nextafterf (f, FLT_MIN / 2.0f); if (fpclassify (f) != FP_SUBNORMAL) { printf ("fpclassify (FLT_MIN-epsilon) failed: %d\n", fpclassify (f)); result = 1; } v2.f = f = nextafterf (f, FLT_MIN); if (fpclassify (f) != FP_NORMAL) { printf ("fpclassify (FLT_MIN-epsilon+epsilon) failed: %d\n", fpclassify (f)); result = 1; } if (v1.ieee.mantissa != v2.ieee.mantissa) { printf ("FLT_MIN: mantissa differs: %8x vs %8x\n", v1.ieee.mantissa, v2.ieee.mantissa); result = 1; } if (v1.ieee.exponent != v2.ieee.exponent) { printf ("FLT_MIN: exponent differs: %4x vs %4x\n", v1.ieee.exponent, v2.ieee.exponent); result = 1; } if (v1.ieee.negative != v2.ieee.negative) { printf ("FLT_MIN: negative differs: %d vs %d\n", v1.ieee.negative, v2.ieee.negative); result = 1; } v1.f = f = -FLT_MIN; if (fpclassify (f) != FP_NORMAL) { printf ("fpclassify (-FLT_MIN) failed: %d\n", fpclassify (f)); result = 1; } f = nextafterf (f, -FLT_MIN / 2.0f); if (fpclassify (f) != FP_SUBNORMAL) { printf ("fpclassify (-FLT_MIN-epsilon) failed: %d\n", fpclassify (f)); result = 1; } v2.f = f = nextafterf (f, -FLT_MIN); if (fpclassify (f) != FP_NORMAL) { printf ("fpclassify (-FLT_MIN-epsilon+epsilon) failed: %d\n", fpclassify (f)); result = 1; } if (v1.ieee.mantissa != v2.ieee.mantissa) { printf ("-FLT_MIN: mantissa differs: %8x vs %8x\n", v1.ieee.mantissa, v2.ieee.mantissa); result = 1; } if (v1.ieee.exponent != v2.ieee.exponent) { printf ("-FLT_MIN: exponent differs: %4x vs %4x\n", v1.ieee.exponent, v2.ieee.exponent); result = 1; } if (v1.ieee.negative != v2.ieee.negative) { printf ("-FLT_MIN: negative differs: %d vs %d\n", v1.ieee.negative, v2.ieee.negative); result = 1; } f = FLT_MAX; if (fpclassify (f) != FP_NORMAL) { printf ("fpclassify (FLT_MAX) failed: %d\n", fpclassify (f)); result = 1; } f = nextafterf (f, INFINITY); if (fpclassify (f) != FP_INFINITE) { printf ("fpclassify (FLT_MAX+epsilon) failed: %d\n", fpclassify (f)); result = 1; } f = -FLT_MAX; if (fpclassify (f) != FP_NORMAL) { printf ("fpclassify (-FLT_MAX) failed: %d\n", fpclassify (f)); result = 1; } f = nextafterf (f, -INFINITY); if (fpclassify (f) != FP_INFINITE) { printf ("fpclassify (-FLT_MAX-epsilon) failed: %d\n", fpclassify (f)); result = 1; } v1.f = f = 0.0625; f = nextafterf (f, 0.0); v2.f = f = nextafterf (f, 1.0); if (v1.ieee.mantissa != v2.ieee.mantissa) { printf ("0.0625f down: mantissa differs: %8x vs %8x\n", v1.ieee.mantissa, v2.ieee.mantissa); result = 1; } if (v1.ieee.exponent != v2.ieee.exponent) { printf ("0.0625f down: exponent differs: %4x vs %4x\n", v1.ieee.exponent, v2.ieee.exponent); result = 1; } if (v1.ieee.negative != v2.ieee.negative) { printf ("0.0625f down: negative differs: %d vs %d\n", v1.ieee.negative, v2.ieee.negative); result = 1; } v1.f = f = 0.0625; f = nextafterf (f, 1.0); v2.f = f = nextafterf (f, 0.0); if (v1.ieee.mantissa != v2.ieee.mantissa) { printf ("0.0625f up: mantissa differs: %8x vs %8x\n", v1.ieee.mantissa, v2.ieee.mantissa); result = 1; } if (v1.ieee.exponent != v2.ieee.exponent) { printf ("0.0625f up: exponent differs: %4x vs %4x\n", v1.ieee.exponent, v2.ieee.exponent); result = 1; } if (v1.ieee.negative != v2.ieee.negative) { printf ("0.0625f up: negative differs: %d vs %d\n", v1.ieee.negative, v2.ieee.negative); result = 1; } v1.f = f = -0.0625; f = nextafterf (f, 0.0); v2.f = f = nextafterf (f, -1.0); if (v1.ieee.mantissa != v2.ieee.mantissa) { printf ("-0.0625f up: mantissa differs: %8x vs %8x\n", v1.ieee.mantissa, v2.ieee.mantissa); result = 1; } if (v1.ieee.exponent != v2.ieee.exponent) { printf ("-0.0625f up: exponent differs: %4x vs %4x\n", v1.ieee.exponent, v2.ieee.exponent); result = 1; } if (v1.ieee.negative != v2.ieee.negative) { printf ("-0.0625f up: negative differs: %d vs %d\n", v1.ieee.negative, v2.ieee.negative); result = 1; } v1.f = f = -0.0625; f = nextafterf (f, -1.0); v2.f = f = nextafterf (f, 0.0); if (v1.ieee.mantissa != v2.ieee.mantissa) { printf ("-0.0625f down: mantissa differs: %8x vs %8x\n", v1.ieee.mantissa, v2.ieee.mantissa); result = 1; } if (v1.ieee.exponent != v2.ieee.exponent) { printf ("-0.0625f down: exponent differs: %4x vs %4x\n", v1.ieee.exponent, v2.ieee.exponent); result = 1; } if (v1.ieee.negative != v2.ieee.negative) { printf ("-0.0625f down: negative differs: %d vs %d\n", v1.ieee.negative, v2.ieee.negative); result = 1; } v1.f = f = 0.0f; f = nextafterf (f, 1.0); v2.f = nextafterf (f, -1.0); if (v1.ieee.mantissa != v2.ieee.mantissa) { printf ("0.0f up: mantissa differs: %8x vs %8x\n", v1.ieee.mantissa, v2.ieee.mantissa); result = 1; } if (v1.ieee.exponent != v2.ieee.exponent) { printf ("0.0f up: exponent differs: %4x vs %4x\n", v1.ieee.exponent, v2.ieee.exponent); result = 1; } if (0 != v2.ieee.negative) { printf ("0.0f up: negative differs: 0 vs %d\n", v2.ieee.negative); result = 1; } v1.f = f = 0.0f; f = nextafterf (f, -1.0); v2.f = nextafterf (f, 1.0); if (v1.ieee.mantissa != v2.ieee.mantissa) { printf ("0.0f down: mantissa differs: %8x vs %8x\n", v1.ieee.mantissa, v2.ieee.mantissa); result = 1; } if (v1.ieee.exponent != v2.ieee.exponent) { printf ("0.0f down: exponent differs: %4x vs %4x\n", v1.ieee.exponent, v2.ieee.exponent); result = 1; } if (1 != v2.ieee.negative) { printf ("0.0f down: negative differs: 1 vs %d\n", v2.ieee.negative); result = 1; } if (nextafterf (0.0f, INFINITY) != nextafterf (0.0f, 1.0f) || nextafterf (-0.0f, INFINITY) != nextafterf (-0.0f, 1.0f) || nextafterf (0.0f, -INFINITY) != nextafterf (0.0f, -1.0f) || nextafterf (-0.0f, -INFINITY) != nextafterf (-0.0f, -1.0f)) { printf ("nextafterf (+-0, +-Inf) != nextafterf (+-0, +-1)\n"); result = 1; } if (nexttowardf (0.0f, INFINITY) != nexttowardf (0.0f, 1.0f) || nexttowardf (-0.0f, INFINITY) != nexttowardf (-0.0f, 1.0f) || nexttowardf (0.0f, -INFINITY) != nexttowardf (0.0f, -1.0f) || nexttowardf (-0.0f, -INFINITY) != nexttowardf (-0.0f, -1.0f)) { printf ("nexttowardf (+-0, +-Inf) != nexttowardf (+-0, +-1)\n"); result = 1; } } { union ieee754_double v1; union ieee754_double v2; double d; v1.d = d = DBL_MIN; if (fpclassify (d) != FP_NORMAL) { printf ("fpclassify (DBL_MIN) failed: %d\n", fpclassify (d)); result = 1; } d = nextafter (d, DBL_MIN / 2.0); if (fpclassify (d) != FP_SUBNORMAL) { printf ("fpclassify (DBL_MIN-epsilon) failed: %d\n", fpclassify (d)); result = 1; } v2.d = d = nextafter (d, DBL_MIN); if (fpclassify (d) != FP_NORMAL) { printf ("fpclassify (DBL_MIN-epsilon+epsilon) failed: %d\n", fpclassify (d)); result = 1; } if (v1.ieee.mantissa0 != v2.ieee.mantissa0) { printf ("DBL_MIN: mantissa0 differs: %8x vs %8x\n", v1.ieee.mantissa0, v2.ieee.mantissa0); result = 1; } if (v1.ieee.mantissa1 != v2.ieee.mantissa1) { printf ("DBL_MIN: mantissa1 differs: %8x vs %8x\n", v1.ieee.mantissa1, v2.ieee.mantissa1); result = 1; } if (v1.ieee.exponent != v2.ieee.exponent) { printf ("DBL_MIN: exponent differs: %4x vs %4x\n", v1.ieee.exponent, v2.ieee.exponent); result = 1; } if (v1.ieee.negative != v2.ieee.negative) { printf ("DBL_MIN: negative differs: %d vs %d\n", v1.ieee.negative, v2.ieee.negative); result = 1; } v1.d = d = -DBL_MIN; if (fpclassify (d) != FP_NORMAL) { printf ("fpclassify (-DBL_MIN) failed: %d\n", fpclassify (d)); result = 1; } d = nextafter (d, -DBL_MIN / 2.0); if (fpclassify (d) != FP_SUBNORMAL) { printf ("fpclassify (-DBL_MIN-epsilon) failed: %d\n", fpclassify (d)); result = 1; } v2.d = d = nextafter (d, -DBL_MIN); if (fpclassify (d) != FP_NORMAL) { printf ("fpclassify (-DBL_MIN-epsilon+epsilon) failed: %d\n", fpclassify (d)); result = 1; } if (v1.ieee.mantissa0 != v2.ieee.mantissa0) { printf ("-DBL_MIN: mantissa0 differs: %8x vs %8x\n", v1.ieee.mantissa0, v2.ieee.mantissa0); result = 1; } if (v1.ieee.mantissa1 != v2.ieee.mantissa1) { printf ("-DBL_MIN: mantissa1 differs: %8x vs %8x\n", v1.ieee.mantissa1, v2.ieee.mantissa1); result = 1; } if (v1.ieee.exponent != v2.ieee.exponent) { printf ("-DBL_MIN: exponent differs: %4x vs %4x\n", v1.ieee.exponent, v2.ieee.exponent); result = 1; } if (v1.ieee.negative != v2.ieee.negative) { printf ("-DBL_MIN: negative differs: %d vs %d\n", v1.ieee.negative, v2.ieee.negative); result = 1; } d = DBL_MAX; if (fpclassify (d) != FP_NORMAL) { printf ("fpclassify (DBL_MAX) failed: %d\n", fpclassify (d)); result = 1; } d = nextafter (d, INFINITY); if (fpclassify (d) != FP_INFINITE) { printf ("fpclassify (DBL_MAX+epsilon) failed: %d\n", fpclassify (d)); result = 1; } d = -DBL_MAX; if (fpclassify (d) != FP_NORMAL) { printf ("fpclassify (-DBL_MAX) failed: %d\n", fpclassify (d)); result = 1; } d = nextafter (d, -INFINITY); if (fpclassify (d) != FP_INFINITE) { printf ("fpclassify (-DBL_MAX-epsilon) failed: %d\n", fpclassify (d)); result = 1; } v1.d = d = 0.0625; d = nextafter (d, 0.0); v2.d = d = nextafter (d, 1.0); if (v1.ieee.mantissa0 != v2.ieee.mantissa0) { printf ("0.0625 down: mantissa0 differs: %8x vs %8x\n", v1.ieee.mantissa0, v2.ieee.mantissa0); result = 1; } if (v1.ieee.mantissa1 != v2.ieee.mantissa1) { printf ("0.0625 down: mantissa1 differs: %8x vs %8x\n", v1.ieee.mantissa1, v2.ieee.mantissa1); result = 1; } if (v1.ieee.exponent != v2.ieee.exponent) { printf ("0.0625 down: exponent differs: %4x vs %4x\n", v1.ieee.exponent, v2.ieee.exponent); result = 1; } if (v1.ieee.negative != v2.ieee.negative) { printf ("0.0625 down: negative differs: %d vs %d\n", v1.ieee.negative, v2.ieee.negative); result = 1; } v1.d = d = 0.0625; d = nextafter (d, 1.0); v2.d = d = nextafter (d, 0.0); if (v1.ieee.mantissa0 != v2.ieee.mantissa0) { printf ("0.0625 up: mantissa0 differs: %8x vs %8x\n", v1.ieee.mantissa0, v2.ieee.mantissa0); result = 1; } if (v1.ieee.mantissa1 != v2.ieee.mantissa1) { printf ("0.0625 up: mantissa1 differs: %8x vs %8x\n", v1.ieee.mantissa1, v2.ieee.mantissa1); result = 1; } if (v1.ieee.exponent != v2.ieee.exponent) { printf ("0.0625 up: exponent differs: %4x vs %4x\n", v1.ieee.exponent, v2.ieee.exponent); result = 1; } if (v1.ieee.negative != v2.ieee.negative) { printf ("0.0625 up: negative differs: %d vs %d\n", v1.ieee.negative, v2.ieee.negative); result = 1; } v1.d = d = -0.0625; d = nextafter (d, 0.0); v2.d = d = nextafter (d, -1.0); if (v1.ieee.mantissa0 != v2.ieee.mantissa0) { printf ("-0.0625 up: mantissa0 differs: %8x vs %8x\n", v1.ieee.mantissa0, v2.ieee.mantissa0); result = 1; } if (v1.ieee.mantissa1 != v2.ieee.mantissa1) { printf ("-0.0625 up: mantissa1 differs: %8x vs %8x\n", v1.ieee.mantissa1, v2.ieee.mantissa1); result = 1; } if (v1.ieee.exponent != v2.ieee.exponent) { printf ("-0.0625 up: exponent differs: %4x vs %4x\n", v1.ieee.exponent, v2.ieee.exponent); result = 1; } if (v1.ieee.negative != v2.ieee.negative) { printf ("-0.0625 up: negative differs: %d vs %d\n", v1.ieee.negative, v2.ieee.negative); result = 1; } v1.d = d = -0.0625; d = nextafter (d, -1.0); v2.d = d = nextafter (d, 0.0); if (v1.ieee.mantissa0 != v2.ieee.mantissa0) { printf ("-0.0625 down: mantissa0 differs: %8x vs %8x\n", v1.ieee.mantissa0, v2.ieee.mantissa0); result = 1; } if (v1.ieee.mantissa1 != v2.ieee.mantissa1) { printf ("-0.0625 down: mantissa1 differs: %8x vs %8x\n", v1.ieee.mantissa1, v2.ieee.mantissa1); result = 1; } if (v1.ieee.exponent != v2.ieee.exponent) { printf ("-0.0625 down: exponent differs: %4x vs %4x\n", v1.ieee.exponent, v2.ieee.exponent); result = 1; } if (v1.ieee.negative != v2.ieee.negative) { printf ("-0.0625 down: negative differs: %d vs %d\n", v1.ieee.negative, v2.ieee.negative); result = 1; } v1.d = d = 0.0; d = nextafter (d, 1.0); v2.d = nextafter (d, -1.0); if (v1.ieee.mantissa0 != v2.ieee.mantissa0) { printf ("0.0 up: mantissa0 differs: %8x vs %8x\n", v1.ieee.mantissa0, v2.ieee.mantissa0); result = 1; } if (v1.ieee.mantissa1 != v2.ieee.mantissa1) { printf ("0.0 up: mantissa1 differs: %8x vs %8x\n", v1.ieee.mantissa1, v2.ieee.mantissa1); result = 1; } if (v1.ieee.exponent != v2.ieee.exponent) { printf ("0.0 up: exponent differs: %4x vs %4x\n", v1.ieee.exponent, v2.ieee.exponent); result = 1; } if (0 != v2.ieee.negative) { printf ("0.0 up: negative differs: 0 vs %d\n", v2.ieee.negative); result = 1; } v1.d = d = 0.0; d = nextafter (d, -1.0); v2.d = nextafter (d, 1.0); if (v1.ieee.mantissa0 != v2.ieee.mantissa0) { printf ("0.0 down: mantissa0 differs: %8x vs %8x\n", v1.ieee.mantissa0, v2.ieee.mantissa0); result = 1; } if (v1.ieee.mantissa1 != v2.ieee.mantissa1) { printf ("0.0 down: mantissa1 differs: %8x vs %8x\n", v1.ieee.mantissa1, v2.ieee.mantissa1); result = 1; } if (v1.ieee.exponent != v2.ieee.exponent) { printf ("0.0 down: exponent differs: %4x vs %4x\n", v1.ieee.exponent, v2.ieee.exponent); result = 1; } if (1 != v2.ieee.negative) { printf ("0.0 down: negative differs: 1 vs %d\n", v2.ieee.negative); result = 1; } if (nextafter (0.0, INFINITY) != nextafter (0.0, 1.0) || nextafter (-0.0, INFINITY) != nextafter (-0.0, 1.0) || nextafter (0.0, -INFINITY) != nextafter (0.0, -1.0) || nextafter (-0.0, -INFINITY) != nextafter (-0.0, -1.0)) { printf ("nextafter (+-0, +-Inf) != nextafter (+-0, +-1)\n"); result = 1; } if (nexttoward (0.0, INFINITY) != nexttoward (0.0, 1.0) || nexttoward (-0.0, INFINITY) != nexttoward (-0.0, 1.0) || nexttoward (0.0, -INFINITY) != nexttoward (0.0, -1.0) || nexttoward (-0.0, -INFINITY) != nexttoward (-0.0, -1.0)) { printf ("nexttoward (+-0, +-Inf) != nexttoward (+-0, +-1)\n"); result = 1; } } #ifndef NO_LONG_DOUBLE { long double v1, v2; v1 = LDBL_MIN; if (fpclassify (v1) != FP_NORMAL) { printf ("fpclassify (LDBL_MIN) failed: %d (%La)\n", fpclassify (v1), v1); result = 1; } v2 = nextafterl (v1, LDBL_MIN / 2.0); if (fpclassify (v2) != FP_SUBNORMAL) { printf ("fpclassify (LDBL_MIN-epsilon) failed: %d (%La)\n", fpclassify (v2), v2); result = 1; } v2 = nextafterl (v2, LDBL_MIN); if (fpclassify (v2) != FP_NORMAL) { printf ("fpclassify (LDBL_MIN-epsilon+epsilon) failed: %d (%La)\n", fpclassify (v2), v2); result = 1; } if (v1 != v2) { printf ("LDBL_MIN-epsilon+epsilon != LDBL_MIN: %La vs %La\n", v2, v1); result = 1; } v1 = -LDBL_MIN; if (fpclassify (v1) != FP_NORMAL) { printf ("fpclassify (-LDBL_MIN) failed: %d (%La)\n", fpclassify (v1), v1); result = 1; } v2 = nextafterl (v1, -LDBL_MIN / 2.0); if (fpclassify (v2) != FP_SUBNORMAL) { printf ("fpclassify (-LDBL_MIN-epsilon) failed: %d (%La)\n", fpclassify (v2), v2); result = 1; } v2 = nextafterl (v2, -LDBL_MIN); if (fpclassify (v2) != FP_NORMAL) { printf ("fpclassify (-LDBL_MIN-epsilon+epsilon) failed: %d (%La)\n", fpclassify (v2), v2); result = 1; } if (v1 != v2) { printf ("-LDBL_MIN-epsilon+epsilon != -LDBL_MIN: %La vs %La\n", v2, v1); result = 1; } v1 = LDBL_MAX; if (fpclassify (v1) != FP_NORMAL) { printf ("fpclassify (LDBL_MAX) failed: %d (%La)\n", fpclassify (v1), v1); result = 1; } v2 = nextafterl (v1, INFINITY); if (fpclassify (v2) != FP_INFINITE) { printf ("fpclassify (LDBL_MAX+epsilon) failed: %d (%La)\n", fpclassify (v2), v2); result = 1; } v1 = -LDBL_MAX; if (fpclassify (v1) != FP_NORMAL) { printf ("fpclassify (-LDBL_MAX) failed: %d (%La)\n", fpclassify (v1), v1); result = 1; } v2 = nextafterl (v1, -INFINITY); if (fpclassify (v2) != FP_INFINITE) { printf ("fpclassify (-LDBL_MAX-epsilon) failed: %d (%La)\n", fpclassify (v2), v2); result = 1; } v1 = 0.0625; v2 = nextafterl (v1, 0.0); v2 = nextafterl (v2, 1.0); if (v1 != v2) { printf ("0.0625L-epsilon+epsilon != 0.0625L: %La vs %La\n", v2, v1); result = 1; } v1 = 0.0625; v2 = nextafterl (v1, 1.0); v2 = nextafterl (v2, 0.0); if (v1 != v2) { printf ("0.0625L+epsilon-epsilon != 0.0625L: %La vs %La\n", v2, v1); result = 1; } v1 = -0.0625; v2 = nextafterl (v1, 0.0); v2 = nextafterl (v2, -1.0); if (v1 != v2) { printf ("-0.0625L+epsilon-epsilon != -0.0625L: %La vs %La\n", v2, v1); result = 1; } v1 = -0.0625; v2 = nextafterl (v1, -1.0); v2 = nextafterl (v2, 0.0); if (v1 != v2) { printf ("-0.0625L-epsilon+epsilon != -0.0625L: %La vs %La\n", v2, v1); result = 1; } v1 = 0.0; v2 = nextafterl (v1, 1.0); v2 = nextafterl (v2, -1.0); if (v1 != v2) { printf ("0.0+epsilon-epsilon != 0.0L: %La vs %La\n", v2, v1); result = 1; } if (signbit (v2)) { printf ("0.0+epsilon-epsilon is negative\n"); result = 1; } v1 = 0.0; v2 = nextafterl (v1, -1.0); v2 = nextafterl (v2, 1.0); if (v1 != v2) { printf ("0.0-epsilon+epsilon != 0.0L: %La vs %La\n", v2, v1); result = 1; } if (!signbit (v2)) { printf ("0.0-epsilon+epsilon is positive\n"); result = 1; } if (nextafterl (0.0, INFINITY) != nextafterl (0.0, 1.0) || nextafterl (-0.0, INFINITY) != nextafterl (-0.0, 1.0) || nextafterl (0.0, -INFINITY) != nextafterl (0.0, -1.0) || nextafterl (-0.0, -INFINITY) != nextafterl (-0.0, -1.0)) { printf ("nextafterl (+-0, +-Inf) != nextafterl (+-0, +-1)\n"); result = 1; } if (nexttowardl (0.0L, INFINITY) != nexttowardl (0.0L, 1.0L) || nexttowardl (-0.0L, INFINITY) != nexttowardl (-0.0L, 1.0L) || nexttowardl (0.0L, -INFINITY) != nexttowardl (0.0L, -1.0L) || nexttowardl (-0.0L, -INFINITY) != nexttowardl (-0.0L, -1.0L)) { printf ("nexttowardl (+-0, +-Inf) != nexttowardl (+-0, +-1)\n"); result = 1; } } #endif if (! isnormal (FLT_MIN)) { puts ("isnormal (FLT_MIN) failed"); result = 1; } if (! isnormal (DBL_MIN)) { puts ("isnormal (DBL_MIN) failed"); result = 1; } #ifndef NO_LONG_DOUBLE if (! isnormal (LDBL_MIN)) { puts ("isnormal (LDBL_MIN) failed"); result = 1; } #endif #if defined (__i386__) || defined (__x86_64__) /* This is a test for the strange long doubles in x86 FPUs. */ { union { char b[10]; long double d; } u = { .b = { 0, 0, 0, 0, 0, 0, 0, 0x80, 0, 0 } }; if (fpclassify (u.d) != FP_NORMAL) { printf ("fpclassify (0x00008000000000000000) failed: %d (%Lg)\n", fpclassify (u.d), u.d); result = 1; } } /* Special qNaNs in x86 long double. Test for scalbl. */ { union { char b[10]; long double d; } u = { .b = { 0, 1, 0, 0, 0, 0, 0, 0xc0, 0xff, 0x7f } }; long double r; r = scalbl (u.d, 0.0); if (!isnan (r)) { puts ("scalbl (qNaN, 0) does not return NaN"); result = 1; } else if (memcmp (&r, &u.d, sizeof (double)) != 0) { puts ("scalbl (qNaN, 0) does not return the same NaN"); result = 1; } } #endif #ifndef NO_LONG_DOUBLE { long double r; feclearexcept (FE_ALL_EXCEPT); r = scalbl (LDBL_MIN, 2147483647); if (! isinf (r)) { puts ("scalbl (LDBL_MIN, 2147483647) does not return Inf"); result = 1; } else if (signbit (r) != 0) { puts ("scalbl (LDBL_MIN, 2147483647) returns -Inf"); result = 1; } else if (fetestexcept (FE_UNDERFLOW)) { puts ("scalbl (LDBL_MIN, 2147483647) raises underflow exception"); result = 1; } feclearexcept (FE_ALL_EXCEPT); r = scalbl (LDBL_MAX, -2147483647); if (r != 0.0) { puts ("scalbl (LDBL_MAX, -2147483647) does not return 0"); result = 1; } else if (signbit (r) != 0) { puts ("scalbl (LDBL_MAX, -2147483647) returns -Inf"); result = 1; } else if (fetestexcept (FE_OVERFLOW)) { puts ("scalbl (LDBL_MAX, -2147483647) raises overflow exception"); result = 1; } } #endif /* The tests here are very similar to tests earlier in this file, the important difference is just that there are no intervening union variables that cause some GCC versions to hide possible bugs in nextafter* implementation. */ if (nextafterf (nextafterf (FLT_MIN, FLT_MIN / 2.0), FLT_MIN) != FLT_MIN) { puts ("nextafterf FLT_MIN test failed"); result = 1; } if (nextafterf (nextafterf (-FLT_MIN, -FLT_MIN / 2.0), -FLT_MIN) != -FLT_MIN) { puts ("nextafterf -FLT_MIN test failed"); result = 1; } if (nextafter (nextafter (DBL_MIN, DBL_MIN / 2.0), DBL_MIN) != DBL_MIN) { puts ("nextafter DBL_MIN test failed"); result = 1; } if (nextafter (nextafter (-DBL_MIN, -DBL_MIN / 2.0), -DBL_MIN) != -DBL_MIN) { puts ("nextafter -DBL_MIN test failed"); result = 1; } #ifndef NO_LONG_DOUBLE if (nextafterl (nextafterl (LDBL_MIN, LDBL_MIN / 2.0), LDBL_MIN) != LDBL_MIN) { puts ("nextafterl LDBL_MIN test failed"); result = 1; } if (nextafterl (nextafterl (-LDBL_MIN, -LDBL_MIN / 2.0), -LDBL_MIN) != -LDBL_MIN) { puts ("nextafterl -LDBL_MIN test failed"); result = 1; } #endif volatile float f1 = FLT_MAX; volatile float f2 = FLT_MAX / 2; (void) &f1; (void) &f2; feclearexcept (FE_ALL_EXCEPT); f2 += f1; #if defined(FE_OVERFLOW) && defined(FE_INEXACT) int fe = fetestexcept (FE_ALL_EXCEPT); if (EXCEPTION_TESTS (float) && fe != (FE_OVERFLOW | FE_INEXACT)) { printf ("float overflow test failed: %x\n", fe); result = 1; } #endif volatile double d1 = DBL_MAX; volatile double d2 = DBL_MAX / 2; (void) &d1; (void) &d2; feclearexcept (FE_ALL_EXCEPT); d2 += d1; #if defined(FE_OVERFLOW) && defined(FE_INEXACT) fe = fetestexcept (FE_ALL_EXCEPT); if (EXCEPTION_TESTS (double) && fe != (FE_OVERFLOW | FE_INEXACT)) { printf ("double overflow test failed: %x\n", fe); result = 1; } #endif #ifndef NO_LONG_DOUBLE volatile long double ld1 = LDBL_MAX; volatile long double ld2 = LDBL_MAX / 2; (void) &ld1; (void) &ld2; feclearexcept (FE_ALL_EXCEPT); ld2 += ld1; # if defined(FE_OVERFLOW) && defined(FE_INEXACT) fe = fetestexcept (FE_ALL_EXCEPT); if (EXCEPTION_TESTS (long double) && fe != (FE_OVERFLOW | FE_INEXACT)) { printf ("long double overflow test failed: %x\n", fe); result = 1; } # endif #endif #if !defined NO_LONG_DOUBLE && LDBL_MANT_DIG == 113 volatile long double ld3 = 0x1.0000000000010000000100000001p+1; volatile long double ld4 = 0x1.0000000000000000000000000001p+1; (void) &ld3; (void) &ld4; ld3 -= ld4; if (ld3 != 0x1.0p-47) { printf ("long double subtraction test failed %.28La\n", ld3); result = 1; } #endif /* Skip testing IBM long double format, for 2 reasons: 1) it only supports FE_TONEAREST 2) nextafter (0.0, 1.0) == nextafterl (0.0L, 1.0L), so nextafter (0.0, 1.0) / 16.0L will be 0.0L. */ #if !defined NO_LONG_DOUBLE && LDBL_MANT_DIG >= DBL_MANT_DIG + 4 \ && LDBL_MANT_DIG != 106 int oldmode = fegetround (); int j; for (j = 0; j < 4; j++) { int mode; int i; int k = 0; const char *mstr; switch (j) { #ifdef FE_TONEAREST case 0: mode = FE_TONEAREST; mstr = "nearest"; k = 8; break; #endif #ifdef FE_DOWNWARD case 1: mode = FE_DOWNWARD; mstr = "-inf"; break; #endif #ifdef FE_UPWARD case 2: mode = FE_UPWARD; mstr = "+inf"; k = 15; break; #endif #ifdef FE_TOWARDZERO case 3: mode = FE_TOWARDZERO; mstr = "0"; break; #endif default: continue; } volatile long double ld5 = nextafter (0.0, 1.0) / 16.0L; volatile double d5; (void) &ld5; for (i = 0; i <= 32; i++) { if (fesetround (mode)) { printf ("failed to set rounding mode to %s\n", mstr); if (ROUNDING_TESTS (long double, mode) && ROUNDING_TESTS (double, mode)) result = 1; else puts ("ignoring this failure"); break; } d5 = ld5 * i; (void) &d5; fesetround (oldmode); if (d5 != ((j == 0 && i == 8) ? 0 : (i + k) / 16) * nextafter (0.0, 1.0)) { printf ("%La incorrectly rounded to %s as %a\n", ld5 * i, mstr, d5); if (ROUNDING_TESTS (long double, mode) && ROUNDING_TESTS (double, mode)) result = 1; else puts ("ignoring this failure"); }