From c957e099b0e983409c93c3b0565bd4b84247e8a1 Mon Sep 17 00:00:00 2001 From: Massimiliano Fasi Date: Sat, 1 Jun 2024 08:11:46 +0100 Subject: [PATCH] Add support for formats without infinities --- mex/cpfloat.c | 42 ++++-- mex/cpfloat.m | 5 + src/cpfloat_definitions.h | 19 +++ src/cpfloat_template.h | 15 +- test/cpfloat_test.m | 16 +++ test/cpfloat_test.ts | 283 +++++++++++++++++++------------------- 6 files changed, 222 insertions(+), 158 deletions(-) diff --git a/mex/cpfloat.c b/mex/cpfloat.c index 1e5dd62..15a717c 100644 --- a/mex/cpfloat.c +++ b/mex/cpfloat.c @@ -42,6 +42,7 @@ void mexFunction(int nlhs, fpopts->emin = -14; fpopts->emax = 15; fpopts->explim = CPFLOAT_EXPRANGE_TARG; + fpopts->infinity = CPFLOAT_INF_USE; fpopts->round = CPFLOAT_RND_NE; fpopts->saturation = CPFLOAT_SAT_NO; fpopts->subnormal = CPFLOAT_SUBN_USE; @@ -57,6 +58,7 @@ void mexFunction(int nlhs, /* Parse second argument and populate fpopts structure. */ if (nrhs > 1) { bool is_subn_rnd_default = false; + bool is_inf_no_default = false; if(!mxIsEmpty(prhs[1]) && !mxIsStruct(prhs[1])) { mexErrMsgIdAndTxt("cpfloat:invalidstruct", "Second argument must be a struct."); @@ -83,6 +85,7 @@ void mexFunction(int nlhs, fpopts->precision = 4; fpopts->emin = -6; fpopts->emax = 8; + is_inf_no_default = true; } else if (!strcmp(fpopts->format, "q52") || !strcmp(fpopts->format, "fp8-e5m2") || !strcmp(fpopts->format, "E5M2")) { @@ -137,6 +140,7 @@ void mexFunction(int nlhs, mexErrMsgIdAndTxt("cpfloat:invalidformat", "Invalid floating-point format specified."); } + /* Set default values to be compatible with MATLAB chop. */ tmp = mxGetField(prhs[1], 0, "subnormal"); if (tmp != NULL) { @@ -147,9 +151,8 @@ void mexFunction(int nlhs, } else { if (is_subn_rnd_default) fpopts->subnormal = CPFLOAT_SUBN_RND; /* Default for bfloat16. */ - else - fpopts->subnormal = CPFLOAT_SUBN_USE; } + tmp = mxGetField(prhs[1], 0, "explim"); if (tmp != NULL) { if (mxGetM(tmp) == 0 && mxGetN(tmp) == 0) @@ -157,6 +160,18 @@ void mexFunction(int nlhs, else if (mxGetClassID(tmp) == mxDOUBLE_CLASS) fpopts->explim = *((double *)mxGetData(tmp)); } + + tmp = mxGetField(prhs[1], 0, "infinity"); + if (tmp != NULL) { + if (mxGetM(tmp) == 0 && mxGetN(tmp) == 0) + fpopts->infinity = CPFLOAT_INF_USE; + else if (mxGetClassID(tmp) == mxDOUBLE_CLASS) + fpopts->infinity = *((double *)mxGetData(tmp)); + } else { + if (is_inf_no_default) + fpopts->infinity = CPFLOAT_INF_NO; /* Default for E4M5. */ + } + tmp = mxGetField(prhs[1], 0, "round"); if (tmp != NULL) { if (mxGetM(tmp) == 0 && mxGetN(tmp) == 0) @@ -164,15 +179,15 @@ void mexFunction(int nlhs, else if (mxGetClassID(tmp) == mxDOUBLE_CLASS) fpopts->round = *((double *)mxGetData(tmp)); } + tmp = mxGetField(prhs[1], 0, "saturation"); if (tmp != NULL) { if (mxGetM(tmp) == 0 && mxGetN(tmp) == 0) fpopts->saturation = CPFLOAT_SAT_NO; else if (mxGetClassID(tmp) == mxDOUBLE_CLASS) fpopts->saturation = *((double *)mxGetData(tmp)); - } else { - fpopts->saturation = CPFLOAT_SAT_NO; } + tmp = mxGetField(prhs[1], 0, "subnormal"); if (tmp != NULL) { if (mxGetM(tmp) == 0 && mxGetN(tmp) == 0) @@ -313,11 +328,11 @@ void mexFunction(int nlhs, /* Allocate and return second output. */ if (nlhs > 1) { - const char* field_names[] = {"format", "params", "explim", + const char* field_names[] = {"format", "params", "explim", "infinity", "round", "saturation", "subnormal", "flip", "p"}; mwSize dims[2] = {1, 1}; - plhs[1] = mxCreateStructArray(2, dims, 8, field_names); + plhs[1] = mxCreateStructArray(2, dims, 9, field_names); mxSetFieldByNumber(plhs[1], 0, 0, mxCreateString(fpopts->format)); mxArray *outparams = mxCreateDoubleMatrix(1,3,mxREAL); @@ -332,30 +347,35 @@ void mexFunction(int nlhs, outexplimptr[0] = fpopts->explim; mxSetFieldByNumber(plhs[1], 0, 2, outexplim); + mxArray *outinfinity = mxCreateDoubleMatrix(1, 1, mxREAL); + double *outinfinityptr = mxGetData(outinfinity); + outinfinityptr[0] = fpopts->infinity; + mxSetFieldByNumber(plhs[1], 0, 3, outinfinity); + mxArray *outround = mxCreateDoubleMatrix(1,1,mxREAL); double *outroundptr = mxGetData(outround); outroundptr[0] = fpopts->round; - mxSetFieldByNumber(plhs[1], 0, 3, outround); + mxSetFieldByNumber(plhs[1], 0, 4, outround); mxArray *outsaturation = mxCreateDoubleMatrix(1,1,mxREAL); double *outsaturationptr = mxGetData(outsaturation); outsaturationptr[0] = fpopts->saturation; - mxSetFieldByNumber(plhs[1], 0, 4, outsaturation); + mxSetFieldByNumber(plhs[1], 0, 5, outsaturation); mxArray *outsubnormal = mxCreateDoubleMatrix(1,1,mxREAL); double *outsubnormalptr = mxGetData(outsubnormal); outsubnormalptr[0] = fpopts->subnormal; - mxSetFieldByNumber(plhs[1], 0, 5, outsubnormal); + mxSetFieldByNumber(plhs[1], 0, 6, outsubnormal); mxArray *outflip = mxCreateDoubleMatrix(1,1,mxREAL); double *outflipptr = mxGetData(outflip); outflipptr[0] = fpopts->flip; - mxSetFieldByNumber(plhs[1], 0, 6, outflip); + mxSetFieldByNumber(plhs[1], 0, 7, outflip); mxArray *outp = mxCreateDoubleMatrix(1,1,mxREAL); double *outpptr = mxGetData(outp); outpptr[0] = fpopts->p; - mxSetFieldByNumber(plhs[1], 0, 7, outp); + mxSetFieldByNumber(plhs[1], 0, 8, outp); } if (nlhs > 2) diff --git a/mex/cpfloat.m b/mex/cpfloat.m index 2dbf9b0..3e27dea 100644 --- a/mex/cpfloat.m +++ b/mex/cpfloat.m @@ -45,6 +45,11 @@ % this field is set to 0, and the exponent range of the format specified in % FPOPTS.format otherwise. The default value for this field is 1. % +% * The scalar FPOPTS.infinity specifies whether infinities are supported. The +% target floating-point format will support infinities if this field is set +% to 1, and they will be replaced by NaNs otherwise. The default value for +% this field is 0 if the target format is 'E4M3' and 1 otherwise. +% % * The scalar FPOPTS.round specifies the rounding mode. Possible values are: % -1 for round-to-nearest with ties-to-away; % 0 for round-to-nearest with ties-to-zero; diff --git a/src/cpfloat_definitions.h b/src/cpfloat_definitions.h index 354d7b1..9493f4d 100644 --- a/src/cpfloat_definitions.h +++ b/src/cpfloat_definitions.h @@ -9,6 +9,7 @@ * defines the enumerated types * * + @ref cpfloat_explim_t, + * + @ref cpfloat_infinity_t, * + @ref cpfloat_rounding_t, * + @ref cpfloat_saturation_t, * + @ref cpfloat_softerr_t, @@ -63,6 +64,16 @@ typedef enum { CPFLOAT_EXPRANGE_TARG = 1 } cpfloat_explim_t; +/** + * @brief Infinity support modes available in CPFloat. + */ +typedef enum { + /** Use infinities in target format. */ + CPFLOAT_INF_NO = 0, + /** Replace infinities with NaNs in target format. */ + CPFLOAT_INF_USE = 1, +} cpfloat_infinity_t; + /** * @brief Rounding modes available in CPFloat. */ @@ -234,6 +245,14 @@ typedef struct { * `CPFLOAT_EXPRANGE_STOR`. */ cpfloat_explim_t explim; + /** + * @brief Support for infinities in target format. + * + * @details If this field is set to `CPFLOAT_INF_USE`, the target format + * supports signed infinities. If the field is set to `CPFLOAT_INF_NO`, + * infinities are replaced with a quiet NaN. + */ + cpfloat_infinity_t infinity; /** * @brief Rounding mode to be used for the conversion. * diff --git a/src/cpfloat_template.h b/src/cpfloat_template.h index 28cf71c..7695e87 100644 --- a/src/cpfloat_template.h +++ b/src/cpfloat_template.h @@ -160,6 +160,7 @@ typedef struct { cpfloat_precision_t precision; cpfloat_exponent_t emin; cpfloat_exponent_t emax; + cpfloat_infinity_t infinity; cpfloat_rounding_t round; cpfloat_saturation_t saturation; cpfloat_subnormal_t subnormal; @@ -322,13 +323,19 @@ static inline FPPARAMS COMPUTE_GLOBAL_PARAMS(const optstruct *fpopts, FPTYPE xmax = ldexp(1., emax) * (2-ldexp(1., 1-precision)); FPTYPE xbnd = ldexp(1., emax) * (2-ldexp(1., -precision)); - FPTYPE ofvalue = (fpopts->saturation == CPFLOAT_SAT_USE) ? xmax : INFINITY; + /* + * Here, fpopts->saturation takes precedence over fpopts->infinity. Therefore, + * when saturation arithmetic is used, infinities are not produced even when + * the target format supports them. + */ + FPTYPE ofvalue = (fpopts->saturation == CPFLOAT_SAT_USE) ? xmax : + (fpopts->infinity == CPFLOAT_INF_USE ? INFINITY : NAN); /* Bitmasks. */ INTTYPE leadmask = FULLMASK << (DEFPREC-precision); /* To keep. */ INTTYPE trailmask = leadmask ^ FULLMASK; /* To discard. */ - FPPARAMS params = {precision, emin, emax, fpopts->round, + FPPARAMS params = {precision, emin, emax, fpopts->infinity, fpopts->round, fpopts->saturation, fpopts->subnormal, ftzthreshold, ofvalue, xmin, xmax, xbnd, leadmask, trailmask, NULL, NULL}; @@ -656,7 +663,9 @@ static inline void UPDATE_LOCAL_PARAMS(const FPTYPE *A, numelem, p, lp) \ PARALLEL_STRING(PARALLEL) \ { \ - if (p->emax == DEFEMAX && p->saturation == CPFLOAT_SAT_NO) { \ + if (p->emax == DEFEMAX \ + && p->saturation == CPFLOAT_SAT_NO \ + && p->infinity == CPFLOAT_INF_USE) { \ FOR_STRING(PARALLEL) \ for (size_t i=0; iround = 1; fpopts->explim = CPFLOAT_EXPRANGE_STOR; + fpopts->infinity = CPFLOAT_INF_USE; fpopts->saturation = CPFLOAT_SAT_NO; fpopts->flip = 0; @@ -1319,162 +1320,156 @@ double *refinfd = malloc(n * sizeof(*refinfd)); float *xf = malloc(n * sizeof(*xf)); float *refxmaxf = malloc(n * sizeof(*refxmaxf)); float *refinff = malloc(n * sizeof(*refinff)); -for (size_t saturation = 0; saturation < 2; saturation++) { - fpopts->saturation = saturation; - for (size_t mode = 1; mode < 3; mode++) { - double *yd = allocate_array_double(xd, n, mode); - float *yf = allocate_array_float(xf, n, mode); - for (size_t i = 0; i < nformats; i++) { - fpopts->precision = precision[i]; - fpopts->emax = emax[i]; - fpopts->emin = emin[i]; - double xmax = maxnormal(fpopts); - double xbound = maxbound(fpopts); - double xd_imm [] = {nextafter(xmax, INFINITY), - nextafter(xbound, 0), - xbound, - nextafter(xbound, INFINITY)}; - double refd [] = {xmax, xmax, inf_double(), inf_double()}; - if (saturation == 0) { - for (size_t j = 0; j < n; j++) { - refxmaxd[j] = xmax; - refinfd[j] = inf_double(); - } - } else { +for (size_t infinity = 0; infinity < 2; infinity++) { + fpopts->infinity = infinity; + for (size_t saturation = 0; saturation < 2; saturation++) { + fpopts->saturation = saturation; + for (size_t mode = 1; mode < 3; mode++) { + double *yd = allocate_array_double(xd, n, mode); + float *yf = allocate_array_float(xf, n, mode); + for (size_t i = 0; i < nformats; i++) { + fpopts->precision = precision[i]; + fpopts->emax = emax[i]; + fpopts->emin = emin[i]; + + double xmax = maxnormal(fpopts); + double xbound = maxbound(fpopts); + + double of_value = saturation == 1 ? xmax : + (infinity == 1 ? inf_double() : NAN); + + double xd_imm [] = {nextafter(xmax, INFINITY), + nextafter(xbound, 0), + xbound, + nextafter(xbound, INFINITY)}; + double refd [] = {xmax, xmax, of_value, of_value}; for (size_t j = 0; j < n; j++) { - refd[2] = xmax; - refd[3] = xmax; refxmaxd[j] = xmax; - refinfd[j] = xmax; + refinfd[j] = of_value; } - } - reset_array_double(yd, n, mode); - copy_array_double(xd, (double *)&xd_imm, n); - select_tests_det_double(yd, xd, refd, n, fpopts, -1, -1, i, i, -1, 1); - reset_array_double(yd, n, mode); - copy_array_double(xd, (double *)&xd_imm, n); - select_tests_det_double(yd, xd, refd, n, fpopts, 1, 1, i, i, -1, 1); - refd[2] = xmax; - reset_array_double(yd, n, mode); - copy_array_double(xd, (double *)&xd_imm, n); - select_tests_det_double(yd, xd, refd, n, fpopts, 0, 0, i, i, -1, 1); - if (saturation == 0) - refd[2] = inf_double(); - reset_array_double(yd, n, mode); - copy_array_double(xd, (double *)&xd_imm, n); - select_tests_det_double(yd, xd, refinfd, n, fpopts, 2, 2, i, i, -1, 1); - reset_array_double(yd, n, mode); - copy_array_double(xd, (double *)&xd_imm, n); - select_tests_det_double(yd, xd, refxmaxd, n, fpopts, 3, 4, i, i, -1, 1); - reset_array_double(yd, n, mode); - copy_array_double(xd, (double *)&xd_imm, n); - select_tests_det_double(yd, xd, refxmaxd, n, fpopts, 7, 7, i, i, -1, 1); - csign_intarray_double((uint64_t *)xd_imm, n); - csign_intarray_double((uint64_t *)refd, n); - csign_intarray_double((uint64_t *)refxmaxd, n); - csign_intarray_double((uint64_t *)refinfd, n); - reset_array_double(yd, n, mode); - copy_array_double(xd, (double *)&xd_imm, n); - select_tests_det_double(yd, xd, refd, n, fpopts, -1, -1, i, i, -1, 1); - reset_array_double(yd, n, mode); - copy_array_double(xd, (double *)&xd_imm, n); - select_tests_det_double(yd, xd, refd, n, fpopts, 1, 1, i, i, -1, 1); - refd[2] = -xmax; - reset_array_double(yd, n, mode); - copy_array_double(xd, (double *)&xd_imm, n); - select_tests_det_double(yd, xd, refd, n, fpopts, 0, 0, i, i, -1, 1); - reset_array_double(yd, n, mode); - copy_array_double(xd, (double *)&xd_imm, n); - select_tests_det_double(yd, xd, refxmaxd, n, fpopts, 2, 2, i, i, -1, 1); - reset_array_double(yd, n, mode); - copy_array_double(xd, (double *)&xd_imm, n); - select_tests_det_double(yd, xd, refinfd, n, fpopts, 3, 3, i, i, -1, 1); - reset_array_double(yd, n, mode); - copy_array_double(xd, (double *)&xd_imm, n); - select_tests_det_double(yd, xd, refxmaxd, n, fpopts, 4, 4, i, i, -1, 1); - reset_array_double(yd, n, mode); - copy_array_double(xd, (double *)&xd_imm, n); - select_tests_det_double(yd, xd, refxmaxd, n, fpopts, 7, 7, i, i, -1, 1); - - float xf_imm [] = {nextafterf(xmax, INFINITY), - nextafterf(xbound, 0), - xbound, - nextafterf(xbound, INFINITY)}; - float reff [] = {xmax, xmax, inf_float(), inf_float()}; - if (saturation == 0) { + + reset_array_double(yd, n, mode); + copy_array_double(xd, (double *)&xd_imm, n); + select_tests_det_double(yd, xd, refd, n, fpopts, -1, -1, i, i, -1, 1); + reset_array_double(yd, n, mode); + copy_array_double(xd, (double *)&xd_imm, n); + select_tests_det_double(yd, xd, refd, n, fpopts, 1, 1, i, i, -1, 1); + refd[2] = xmax; + reset_array_double(yd, n, mode); + copy_array_double(xd, (double *)&xd_imm, n); + select_tests_det_double(yd, xd, refd, n, fpopts, 0, 0, i, i, -1, 1); + refd[2] = of_value; + reset_array_double(yd, n, mode); + copy_array_double(xd, (double *)&xd_imm, n); + select_tests_det_double(yd, xd, refinfd, n, fpopts, 2, 2, i, i, -1, 1); + reset_array_double(yd, n, mode); + copy_array_double(xd, (double *)&xd_imm, n); + select_tests_det_double(yd, xd, refxmaxd, n, fpopts, 3, 4, i, i, -1, 1); + reset_array_double(yd, n, mode); + copy_array_double(xd, (double *)&xd_imm, n); + select_tests_det_double(yd, xd, refxmaxd, n, fpopts, 7, 7, i, i, -1, 1); + csign_intarray_double((uint64_t *)xd_imm, n); + csign_intarray_double((uint64_t *)refd, n); + csign_intarray_double((uint64_t *)refxmaxd, n); + csign_intarray_double((uint64_t *)refinfd, n); + reset_array_double(yd, n, mode); + copy_array_double(xd, (double *)&xd_imm, n); + select_tests_det_double(yd, xd, refd, n, fpopts, -1, -1, i, i, -1, 1); + reset_array_double(yd, n, mode); + copy_array_double(xd, (double *)&xd_imm, n); + select_tests_det_double(yd, xd, refd, n, fpopts, 1, 1, i, i, -1, 1); + refd[2] = -xmax; + reset_array_double(yd, n, mode); + copy_array_double(xd, (double *)&xd_imm, n); + select_tests_det_double(yd, xd, refd, n, fpopts, 0, 0, i, i, -1, 1); + reset_array_double(yd, n, mode); + copy_array_double(xd, (double *)&xd_imm, n); + select_tests_det_double(yd, xd, refxmaxd, n, fpopts, 2, 2, i, i, -1, 1); + reset_array_double(yd, n, mode); + copy_array_double(xd, (double *)&xd_imm, n); + select_tests_det_double(yd, xd, refinfd, n, fpopts, 3, 3, i, i, -1, 1); + reset_array_double(yd, n, mode); + copy_array_double(xd, (double *)&xd_imm, n); + select_tests_det_double(yd, xd, refxmaxd, n, fpopts, 4, 4, i, i, -1, 1); + reset_array_double(yd, n, mode); + copy_array_double(xd, (double *)&xd_imm, n); + select_tests_det_double(yd, xd, refxmaxd, n, fpopts, 7, 7, i, i, -1, 1); + + float of_valuef = saturation == 1 ? xmax : + (infinity == 1 ? inf_float() : NAN); + float xf_imm [] = {nextafterf(xmax, INFINITY), + nextafterf(xbound, 0), + xbound, + nextafterf(xbound, INFINITY)}; + float reff [] = {xmax, xmax, of_valuef, of_valuef}; for (size_t j = 0; j < n; j++) { refxmaxf[j] = xmax; - refinff[j] = inf_float(); + refinff[j] = of_valuef; } - } else { + + reset_array_float(yf, n, mode); + copy_array_float(xf, (float *)&xf_imm, n); + select_tests_det_float(yf, xf, reff, n, fpopts, -1, -1, i, i, -1, 1); + reset_array_float(yf, n, mode); + copy_array_float(xf, (float *)&xf_imm, n); + select_tests_det_float(yf, xf, reff, n, fpopts, 1, 1, i, i, -1, 1); reff[2] = xmax; - reff[3] = xmax; - for (size_t j = 0; j < n; j++) { - refxmaxf[j] = xmax; - refinff[j] = xmax; - } + reset_array_float(yf, n, mode); + copy_array_float(xf, (float *)&xf_imm, n); + select_tests_det_float(yf, xf, reff, n, fpopts, 0, 0, i, i, -1, 1); + reff[2] = of_valuef; + reset_array_float(yf, n, mode); + copy_array_float(xf, (float *)&xf_imm, n); + select_tests_det_float(yf, xf, refinff, n, fpopts, 2, 2, i, i, -1, 1); + reset_array_float(yf, n, mode); + copy_array_float(xf, (float *)&xf_imm, n); + select_tests_det_float(yf, xf, refxmaxf, n, fpopts, 3, 4, i, i, -1, 1); + reset_array_float(yf, n, mode); + copy_array_float(xf, (float *)&xf_imm, n); + select_tests_det_float(yf, xf, refxmaxf, n, fpopts, 7, 7, i, i, -1, 1); + reset_array_float(yf, n, mode); + copy_array_float(xf, (float *)&xf_imm, n); + csign_intarray_float((uint32_t *)xf_imm, n); + csign_intarray_float((uint32_t *)reff, n); + csign_intarray_float((uint32_t *)refxmaxf, n); + csign_intarray_float((uint32_t *)refinff, n); + reset_array_float(yf, n, mode); + copy_array_float(xf, (float *)&xf_imm, n); + select_tests_det_float(yf, xf, reff, n, fpopts, -1, -1, i, i, -1, 1); + reset_array_float(yf, n, mode); + copy_array_float(xf, (float *)&xf_imm, n); + select_tests_det_float(yf, xf, reff, n, fpopts, 1, 1, i, i, -1, 1); + reff[2] = -xmax; + reset_array_float(yf, n, mode); + copy_array_float(xf, (float *)&xf_imm, n); + select_tests_det_float(yf, xf, reff, n, fpopts, 0, 0, i, i, -1, 1); + reset_array_float(yf, n, mode); + copy_array_float(xf, (float *)&xf_imm, n); + select_tests_det_float(yf, xf, refxmaxf, n, fpopts, 2, 2, i, i, -1, 1); + reset_array_float(yf, n, mode); + copy_array_float(xf, (float *)&xf_imm, n); + select_tests_det_float(yf, xf, refinff, n, fpopts, 3, 3, i, i, -1, 1); + reset_array_float(yf, n, mode); + copy_array_float(xf, (float *)&xf_imm, n); + select_tests_det_float(yf, xf, refxmaxf, n, fpopts, 4, 4, i, i, -1, 1); + reset_array_float(yf, n, mode); + copy_array_float(xf, (float *)&xf_imm, n); + select_tests_det_float(yf, xf, refxmaxf, n, fpopts, 7, 7, i, i, -1, 1); } - reset_array_float(yf, n, mode); - copy_array_float(xf, (float *)&xf_imm, n); - select_tests_det_float(yf, xf, reff, n, fpopts, -1, -1, i, i, -1, 1); - reset_array_float(yf, n, mode); - copy_array_float(xf, (float *)&xf_imm, n); - select_tests_det_float(yf, xf, reff, n, fpopts, 1, 1, i, i, -1, 1); - reff[2] = xmax; - reset_array_float(yf, n, mode); - copy_array_float(xf, (float *)&xf_imm, n); - select_tests_det_float(yf, xf, reff, n, fpopts, 0, 0, i, i, -1, 1); - if (saturation == 0) - reff[2] = inf_float(); - reset_array_float(yf, n, mode); - copy_array_float(xf, (float *)&xf_imm, n); - select_tests_det_float(yf, xf, refinff, n, fpopts, 2, 2, i, i, -1, 1); - reset_array_float(yf, n, mode); - copy_array_float(xf, (float *)&xf_imm, n); - select_tests_det_float(yf, xf, refxmaxf, n, fpopts, 3, 4, i, i, -1, 1); - reset_array_float(yf, n, mode); - copy_array_float(xf, (float *)&xf_imm, n); - select_tests_det_float(yf, xf, refxmaxf, n, fpopts, 7, 7, i, i, -1, 1); - reset_array_float(yf, n, mode); - copy_array_float(xf, (float *)&xf_imm, n); - csign_intarray_float((uint32_t *)xf_imm, n); - csign_intarray_float((uint32_t *)reff, n); - csign_intarray_float((uint32_t *)refxmaxf, n); - csign_intarray_float((uint32_t *)refinff, n); - reset_array_float(yf, n, mode); - copy_array_float(xf, (float *)&xf_imm, n); - select_tests_det_float(yf, xf, reff, n, fpopts, -1, -1, i, i, -1, 1); - reset_array_float(yf, n, mode); - copy_array_float(xf, (float *)&xf_imm, n); - select_tests_det_float(yf, xf, reff, n, fpopts, 1, 1, i, i, -1, 1); - reff[2] = -xmax; - reset_array_float(yf, n, mode); - copy_array_float(xf, (float *)&xf_imm, n); - select_tests_det_float(yf, xf, reff, n, fpopts, 0, 0, i, i, -1, 1); - reset_array_float(yf, n, mode); - copy_array_float(xf, (float *)&xf_imm, n); - select_tests_det_float(yf, xf, refxmaxf, n, fpopts, 2, 2, i, i, -1, 1); - reset_array_float(yf, n, mode); - copy_array_float(xf, (float *)&xf_imm, n); - select_tests_det_float(yf, xf, refinff, n, fpopts, 3, 3, i, i, -1, 1); - reset_array_float(yf, n, mode); - copy_array_float(xf, (float *)&xf_imm, n); - select_tests_det_float(yf, xf, refxmaxf, n, fpopts, 4, 4, i, i, -1, 1); - reset_array_float(yf, n, mode); - copy_array_float(xf, (float *)&xf_imm, n); - select_tests_det_float(yf, xf, refxmaxf, n, fpopts, 7, 7, i, i, -1, 1); + free_array_double(yd, mode); + free_array_float(yf, mode); } - free_array_double(yd, mode); - free_array_float(yf, mode); } + free(xd); + free(refxmaxd); + free(refinfd); + free(xf); + free(refxmaxf); + free(refinff); } -free(xd); -free(refxmaxd); -free(refinfd); -free(xf); -free(refxmaxf); -free(refinff); fpopts->saturation = CPFLOAT_SAT_NO; +fpopts->infinity = CPFLOAT_INF_USE; + /* Rounding of numbers in the subnormal range. */ #test deterministic_rounding_subnormal_numbers