/* Test whether a basic modff invocation works. */ /* This test is generated by misc/genbasic.c. */ #include #include #include #include "../basic.h" #pragma STDC FENV_ACCESS ON #define MF_UNSPEC1 (1 << 0) #define MF_UNSPEC2 (1 << 1) #define MF_MAYERR (1 << 2) // Soft fail on rounding errors and report only one. int imprecise; static const char* fperrname(int excepts) { switch ( excepts ) { case 0: return "FE_NONE"; case FE_INVALID: return "FE_INVALID"; case FE_DIVBYZERO: return "FE_DIVBYZERO"; case FE_OVERFLOW: return "FE_OVERFLOW"; case FE_UNDERFLOW: return "FE_UNDERFLOW"; default: return "FE_MULTIPLE"; } } void test(int variant, float input1, int errnum, int fperr, float lower, float lower_integral, float expected, float expected_integral, float upper, float upper_integral, int flags) { errno = 0; if ( feclearexcept(FE_ALL_EXCEPT) ) errx(1, "feclearexcept"); float integral; float output = modff(input1, &integral); if ( errnum == 0 && errno ) err(1, "(%d.) modff(%.4f) failed", variant, input1); if ( (math_errhandling & MATH_ERRNO) && errnum && errno != errnum ) errx(1, "(%d.) modff(%.4f) did not %s", variant, input1, strerrno(errnum)); int excepts = fetestexcept(FE_INVALID | FE_DIVBYZERO | FE_OVERFLOW | FE_UNDERFLOW); if ( fperr == 0 && excepts ) errx(1, "(%d.) modff(%.4f) %s", variant, input1, fperrname(excepts)); if ( (math_errhandling & MATH_ERREXCEPT) && fperr != 0 && excepts != fperr && !((flags & MF_MAYERR) && !excepts) ) errx(1, "(%d.) modff(%.4f) did not %s", variant, input1, fperrname(fperr)); if ( !(flags & MF_UNSPEC1) ) { if ( !(isnan(expected) ? isnan(output) : isfinite(expected) && expected != 0.0 ? isfinite(output) && (output == expected || (lower < output && output < upper)) : output == expected) ) { if ( imprecise && isfinite(output) && isfinite(expected) ) return; warnx("(%d.) modff(%.4f) = %.6a, not %.6a, diff %.6a, ratio %.8g", variant, input1, output, expected, output - expected, output / expected); if ( !isfinite(output) || !isfinite(expected) ) exit(1); imprecise = 1; } } if ( !(flags & MF_UNSPEC2) ) { if ( !(isnan(expected_integral) ? isnan(integral) : isfinite(expected_integral) && expected_integral != 0.0 ? isfinite(integral) && (integral == expected_integral || (lower_integral < integral && integral < upper_integral)) : integral == expected_integral) ) { if ( imprecise && isfinite(integral) && isfinite(expected_integral) ) return; warnx("(%d.) modff(%.4f).integral = %.6a, not %.6a, diff %.6a, ratio %.8g", variant, input1, integral, expected_integral, integral - expected_integral, integral / expected_integral); if ( !isfinite(integral) || !isfinite(expected_integral) ) exit(1); imprecise = 1; } } } int main(void) { test(1, 90.01, 0, 0, 0x1.47bffep-7, 0x1.67fffep+6, 0x1.47cp-7, 0x1.68p+6, 0x1.47c002p-7, 0x1.680002p+6, 0); test(2, -12.34, 0, 0, -0x1.5c2902p-2, -0x1.800002p+3, -0x1.5c29p-2, -0x1.8p+3, -0x1.5c28fep-2, -0x1.7ffffep+3, 0); test(3, nanf(""), 0, 0, nanf(""), nanf(""), nanf(""), nanf(""), nanf(""), nanf(""), 0); test(4, strtof("inf", NULL), 0, 0, -0x1.0p-149, strtof("inf", NULL), 0x0.0p+0, strtof("inf", NULL), 0x1.0p-149, strtof("inf", NULL), 0); test(5, strtof("-inf", NULL), 0, 0, -0x1.0p-149, strtof("-inf", NULL), -0x0.0p+0, strtof("-inf", NULL), 0x1.0p-149, strtof("-inf", NULL), 0); test(6, 0.0, 0, 0, -0x1.0p-149, -0x1.0p-149, 0x0.0p+0, 0x0.0p+0, 0x1.0p-149, 0x1.0p-149, 0); return imprecise; }