/* tprintf.c -- test file for mpfr_printf and mpfr_vprintf Copyright 2008-2021 Free Software Foundation, Inc. Contributed by the AriC and Caramba projects, INRIA. This file is part of the GNU MPFR Library. The GNU MPFR Library is free software; you can redistribute it and/or modify it under the terms of the GNU Lesser General Public License as published by the Free Software Foundation; either version 3 of the License, or (at your option) any later version. The GNU MPFR Library is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. You should have received a copy of the GNU Lesser General Public License along with the GNU MPFR Library; see the file COPYING.LESSER. If not, see https://www.gnu.org/licenses/ or write to the Free Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA. */ /* FIXME: The output is not tested (thus coverage data are meaningless); only the return value is tested (output string length). Knowing the implementation, we may need only some minimal checks: all the formatted output functions are based on mpfr_vasnprintf_aux and full checks are done via tsprintf. */ /* Needed due to the tests on HAVE_STDARG and MPFR_USE_MINI_GMP */ #ifdef HAVE_CONFIG_H # include "config.h" #endif #if defined(HAVE_STDARG) && !defined(MPFR_USE_MINI_GMP) #include #include #include #ifdef HAVE_LOCALE_H #include #endif #define MPFR_NEED_INTMAX_H #include "mpfr-test.h" #define STDOUT_FILENO 1 #define QUOTE(X) NAME(X) #define NAME(X) #X /* unlike other tests, we print out errors to stderr because stdout might be redirected */ #define check_length(num_test, var, value, var_spec) \ if ((var) != (value)) \ { \ fprintf (stderr, "Error in test #%d: mpfr_printf printed %" \ QUOTE(var_spec)" characters instead of %d\n", \ (num_test), (var), (value)); \ exit (1); \ } #define check_length_with_cmp(num_test, var, value, cmp, var_spec) \ if (cmp != 0) \ { \ mpfr_fprintf (stderr, "Error in test #%d, mpfr_printf printed %" \ QUOTE(var_spec)" characters instead of %d\n", \ (num_test), (var), (value)); \ exit (1); \ } /* limit for random precision in random() */ const int prec_max_printf = 5000; /* boolean: is stdout redirected to a file ? */ int stdout_redirect; static void check (const char *fmt, mpfr_ptr x) { if (mpfr_printf (fmt, x) == -1) { fprintf (stderr, "Error 1 in mpfr_printf(\"%s\", ...)\n", fmt); exit (1); } putchar ('\n'); } static void check_vprintf (const char *fmt, ...) { va_list ap; va_start (ap, fmt); if (mpfr_vprintf (fmt, ap) == -1) { fprintf (stderr, "Error 2 in mpfr_vprintf(\"%s\", ...)\n", fmt); va_end (ap); exit (1); } putchar ('\n'); va_end (ap); } static unsigned int check_vprintf_failure (const char *fmt, ...) { va_list ap; int r, e; va_start (ap, fmt); errno = 0; r = mpfr_vprintf (fmt, ap); e = errno; va_end (ap); if (r != -1 #ifdef EOVERFLOW || e != EOVERFLOW #endif ) { putchar ('\n'); fprintf (stderr, "Error 3 in mpfr_vprintf(\"%s\", ...)\n" "Got r = %d, errno = %d\n", fmt, r, e); return 1; } putchar ('\n'); return 0; } /* The goal of this test is to check cases where more INT_MAX characters are output, in which case, it should be a failure, because like C's *printf functions, the return type is int and the returned value must be either the number of characters printed or a negative value. */ static void check_long_string (void) { /* this test is VERY expensive both in time (~1 min on core2 @ 2.40GHz) and in memory (~2.5 GB) */ mpfr_t x; long large_prec = 2147483647; size_t min_memory_limit, old_memory_limit; old_memory_limit = tests_memory_limit; /* With a 32-bit (4GB) address space, a realloc failure has been noticed with a 2G precision (though allocating up to 4GB is possible): MPFR: Can't reallocate memory (old_size=4096 new_size=2147487744) The implementation might be improved to use less memory and avoid this problem. In the mean time, let's choose a smaller precision, but this will generally have the effect to disable the test. */ if (sizeof (void *) == 4) large_prec /= 2; /* We assume that the precision won't be increased internally. */ if (large_prec > MPFR_PREC_MAX) large_prec = MPFR_PREC_MAX; /* Increase tests_memory_limit if need be in order to avoid an obvious failure due to insufficient memory. Note that such an increase is necessary, but is not guaranteed to be sufficient in all cases (e.g. with logging activated). */ min_memory_limit = large_prec / MPFR_BYTES_PER_MP_LIMB; if (min_memory_limit > (size_t) -1 / 32) min_memory_limit = (size_t) -1; else min_memory_limit *= 32; if (tests_memory_limit > 0 && tests_memory_limit < min_memory_limit) tests_memory_limit = min_memory_limit; mpfr_init2 (x, large_prec); mpfr_set_ui (x, 1, MPFR_RNDN); mpfr_nextabove (x); if (large_prec >= INT_MAX - 512) { unsigned int err = 0; #define LS1 "%Rb %512d" #define LS2 "%RA %RA %Ra %Ra %512d" err |= check_vprintf_failure (LS1, x, 1); err |= check_vprintf_failure (LS2, x, x, x, x, 1); if (sizeof (long) * CHAR_BIT > 40) { long n1, n2; n1 = large_prec + 517; n2 = -17; err |= check_vprintf_failure (LS1 "%ln", x, 1, &n2); if (n1 != n2) { fprintf (stderr, "Error in check_long_string(\"%s\", ...)\n" "Expected n = %ld\n" "Got n = %ld\n", LS1 "%ln", n1, n2); err = 1; } n1 = ((large_prec - 2) / 4) * 4 + 548; n2 = -17; err |= check_vprintf_failure (LS2 "%ln", x, x, x, x, 1, &n2); if (n1 != n2) { fprintf (stderr, "Error in check_long_string(\"%s\", ...)\n" "Expected n = %ld\n" "Got n = %ld\n", LS2 "%ln", n1, n2); err = 1; } } if (err) exit (1); } mpfr_clear (x); tests_memory_limit = old_memory_limit; } static void check_special (void) { mpfr_t x; mpfr_init (x); mpfr_set_inf (x, 1); check ("%Ra", x); check ("%Rb", x); check ("%Re", x); check ("%Rf", x); check ("%Rg", x); check_vprintf ("%Ra", x); check_vprintf ("%Rb", x); check_vprintf ("%Re", x); check_vprintf ("%Rf", x); check_vprintf ("%Rg", x); mpfr_set_inf (x, -1); check ("%Ra", x); check ("%Rb", x); check ("%Re", x); check ("%Rf", x); check ("%Rg", x); check_vprintf ("%Ra", x); check_vprintf ("%Rb", x); check_vprintf ("%Re", x); check_vprintf ("%Rf", x); check_vprintf ("%Rg", x); mpfr_set_nan (x); check ("%Ra", x); check ("%Rb", x); check ("%Re", x); check ("%Rf", x); check ("%Rg", x); check_vprintf ("%Ra", x); check_vprintf ("%Rb", x); check_vprintf ("%Re", x); check_vprintf ("%Rf", x); check_vprintf ("%Rg", x); mpfr_clear (x); } static void check_mixed (void) { int ch = 'a'; #ifndef NPRINTF_HH signed char sch = -1; unsigned char uch = 1; #endif short sh = -1; unsigned short ush = 1; int i = -1; int j = 1; unsigned int ui = 1; long lo = -1; unsigned long ulo = 1; float f = -1.25; double d = -1.25; #if defined(PRINTF_T) || defined(PRINTF_L) long double ld = -1.25; #endif #ifdef PRINTF_T ptrdiff_t p = 1, saved_p; #endif size_t sz = 1; mpz_t mpz; mpq_t mpq; mpf_t mpf; mpfr_rnd_t rnd = MPFR_RNDN; mpfr_t mpfr; mpfr_prec_t prec; mpz_init (mpz); mpz_set_ui (mpz, ulo); mpq_init (mpq); mpq_set_si (mpq, lo, ulo); mpf_init (mpf); mpf_set_q (mpf, mpq); mpfr_init (mpfr); mpfr_set_f (mpfr, mpf, MPFR_RNDN); prec = mpfr_get_prec (mpfr); check_vprintf ("a. %Ra, b. %u, c. %lx%n", mpfr, ui, ulo, &j); check_length (1, j, 22, d); check_vprintf ("a. %c, b. %Rb, c. %u, d. %li%ln", i, mpfr, i, lo, &ulo); check_length (2, ulo, 36, lu); check_vprintf ("a. %hi, b. %*f, c. %Re%hn", ush, 3, f, mpfr, &ush); check_length (3, ush, 46, hu); check_vprintf ("a. %hi, b. %f, c. %#.2Rf%n", sh, d, mpfr, &i); check_length (4, i, 29, d); check_vprintf ("a. %R*A, b. %Fe, c. %i%zn", rnd, mpfr, mpf, sz, &sz); check_length (5, (unsigned long) sz, 34, lu); /* no format specifier '%zu' in C90 */ check_vprintf ("a. %Pu, b. %c, c. %RUG, d. %Zi%Zn", prec, ch, mpfr, mpz, &mpz); check_length_with_cmp (6, mpz, 24, mpz_cmp_ui (mpz, 24), Zi); check_vprintf ("%% a. %#.0RNg, b. %Qx%Rn c. %p", mpfr, mpq, &mpfr, (void *) &i); check_length_with_cmp (7, mpfr, 15, mpfr_cmp_ui (mpfr, 15), Rg); #ifdef PRINTF_T saved_p = p; check_vprintf ("%% a. %RNg, b. %Qx, c. %td%tn", mpfr, mpq, p, &p); if (p != 20) { mpfr_fprintf (stderr, "Error in test 8, got '%% a. %RNg, b. %Qx, c. %td'\n", mpfr, mpq, saved_p); #if defined(__MINGW32__) || defined(__MINGW64__) fprintf (stderr, "Your MinGW may be too old, in which case compiling GMP\n" "with -D__USE_MINGW_ANSI_STDIO might be required.\n"); #endif } check_length (8, (long) p, 20, ld); /* no format specifier '%td' in C90 */ #endif #ifdef PRINTF_L check_vprintf ("a. %RA, b. %Lf, c. %QX%zn", mpfr, ld, mpq, &sz); check_length (9, (unsigned long) sz, 30, lu); /* no format specifier '%zu' in C90 */ #endif #ifndef NPRINTF_HH check_vprintf ("a. %hhi, b. %Ra, c. %hhu%hhn", sch, mpfr, uch, &uch); check_length (10, (unsigned int) uch, 22, u); /* no format specifier '%hhu' in C90 */ #endif #if defined(HAVE_LONG_LONG) && !defined(NPRINTF_LL) { long long llo = -1; unsigned long long ullo = 1; check_vprintf ("a. %Re, b. %llx%Qn", mpfr, ullo, &mpq); check_length_with_cmp (11, mpq, 31, mpq_cmp_ui (mpq, 31, 1), Qu); check_vprintf ("a. %lli, b. %Rf%lln", llo, mpfr, &ullo); check_length (12, ullo, 19, llu); } #endif #if defined(_MPFR_H_HAVE_INTMAX_T) && !defined(NPRINTF_J) { intmax_t im = -1; uintmax_t uim = 1; check_vprintf ("a. %*RA, b. %ji%Fn", 10, mpfr, im, &mpf); check_length_with_cmp (31, mpf, 20, mpf_cmp_ui (mpf, 20), Fg); check_vprintf ("a. %.*Re, b. %jx%jn", 10, mpfr, uim, &im); check_length (32, (long) im, 25, li); /* no format specifier "%ji" in C90 */ } #endif mpfr_clear (mpfr); mpf_clear (mpf); mpq_clear (mpq); mpz_clear (mpz); } static void check_random (int nb_tests) { int i; mpfr_t x; mpfr_rnd_t rnd; char flag[] = { '-', '+', ' ', '#', '0', /* no ambiguity: first zeros are flag zero*/ '\'' }; char specifier[] = { 'a', 'b', 'e', 'f', 'g' }; mpfr_exp_t old_emin, old_emax; old_emin = mpfr_get_emin (); old_emax = mpfr_get_emax (); mpfr_init (x); for (i = 0; i < nb_tests; ++i) { int ret; int j, jmax; int spec, prec; #define FMT_SIZE 13 char fmt[FMT_SIZE]; /* at most something like "%-+ #0'.*R*f" */ char *ptr = fmt; tests_default_random (x, 256, MPFR_EMIN_MIN, MPFR_EMAX_MAX, 0); rnd = (mpfr_rnd_t) RND_RAND (); spec = (int) (randlimb () % 5); jmax = (spec == 3 || spec == 4) ? 6 : 5; /* ' flag only with %f or %g */ /* advantage small precision */ prec = (randlimb () % 2) ? 10 : prec_max_printf; prec = (int) (randlimb () % prec); if (spec == 3 && (mpfr_get_exp (x) > prec_max_printf || mpfr_get_exp (x) < -prec_max_printf)) /* change style 'f' to style 'e' when number x is very large or very small*/ --spec; *ptr++ = '%'; for (j = 0; j < jmax; j++) { if (randlimb () % 3 == 0) *ptr++ = flag[j]; } *ptr++ = '.'; *ptr++ = '*'; *ptr++ = 'R'; *ptr++ = '*'; *ptr++ = specifier[spec]; *ptr = '\0'; MPFR_ASSERTD (ptr - fmt < FMT_SIZE); mpfr_printf ("mpfr_printf(\"%s\", %d, %s, %Re)\n", fmt, prec, mpfr_print_rnd_mode (rnd), x); ret = mpfr_printf (fmt, prec, rnd, x); if (ret == -1) { if (spec == 3 && (MPFR_GET_EXP (x) > INT_MAX || MPFR_GET_EXP (x) < -INT_MAX)) /* normal failure: x is too large to be output with full precision */ { mpfr_printf ("too large !"); } else { printf ("Error in mpfr_printf(\"%s\", %d, %s, ...)", fmt, prec, mpfr_print_rnd_mode (rnd)); if (stdout_redirect) { if ((fflush (stdout) == EOF) || (fclose (stdout) == EOF)) { perror ("check_random"); exit (1); } } exit (1); } } putchar ('\n'); } set_emin (old_emin); set_emax (old_emax); mpfr_clear (x); } #if defined(HAVE_LOCALE_H) && defined(HAVE_SETLOCALE) static void test_locale (void) { const char * const tab_locale[] = { "en_US", "en_US.iso88591", "en_US.iso885915", "en_US.utf8" }; int i; mpfr_t x; int count; char v[] = "99999999999999999999999.5"; for (i = 0; i < numberof(tab_locale); i++) { char *s; s = setlocale (LC_ALL, tab_locale[i]); if (s != NULL && MPFR_THOUSANDS_SEPARATOR == ',') break; } if (i == numberof(tab_locale)) { if (getenv ("MPFR_CHECK_LOCALES") == NULL) return; fprintf (stderr, "Cannot find a locale with ',' thousands separator.\n" "Please install one of the en_US based locales.\n"); exit (1); } mpfr_init2 (x, 113); mpfr_set_ui (x, 10000, MPFR_RNDN); count = mpfr_printf ("(1) 10000=%'Rg \n", x); check_length (10000, count, 18, d); count = mpfr_printf ("(2) 10000=%'Rf \n", x); check_length (10001, count, 25, d); mpfr_set_ui (x, 1000, MPFR_RNDN); count = mpfr_printf ("(3) 1000=%'Rf \n", x); check_length (10002, count, 23, d); for (i = 1; i <= sizeof (v) - 3; i++) { mpfr_set_str (x, v + sizeof (v) - 3 - i, 10, MPFR_RNDN); count = mpfr_printf ("(4) 10^i=%'.0Rf \n", x); check_length (10002 + i, count, 12 + i + i/3, d); } #define N0 20 for (i = 1; i <= N0; i++) { char s[N0+4]; int j, rnd; s[0] = '1'; for (j = 1; j <= i; j++) s[j] = '0'; s[i+1] = '\0'; mpfr_set_str (x, s, 10, MPFR_RNDN); RND_LOOP (rnd) { count = mpfr_printf ("(5) 10^i=%'.0R*f \n", (mpfr_rnd_t) rnd, x); check_length (11000 + 10 * i + rnd, count, 12 + i + i/3, d); } strcat (s + (i + 1), ".5"); count = mpfr_printf ("(5) 10^i=%'.0Rf \n", x); check_length (11000 + 10 * i + 9, count, 12 + i + i/3, d); } mpfr_set_str (x, "1000", 10, MPFR_RNDN); count = mpfr_printf ("%'012.3Rg\n", x); check_length (12000, count, 13, d); count = mpfr_printf ("%'012.4Rg\n", x); check_length (12001, count, 13, d); count = mpfr_printf ("%'013.4Rg\n", x); check_length (12002, count, 14, d); mpfr_clear (x); } #else static void test_locale (void) { if (getenv ("MPFR_CHECK_LOCALES") != NULL) { fprintf (stderr, "Cannot test locales.\n"); exit (1); } } #endif int main (int argc, char *argv[]) { int N; tests_start_mpfr (); /* with no argument: prints to /dev/null, tprintf N: prints N tests to stdout */ if (argc == 1) { N = 1000; stdout_redirect = 1; if (freopen ("/dev/null", "w", stdout) == NULL) { /* We failed to open this device, try with a dummy file */ if (freopen ("tprintf_out.txt", "w", stdout) == NULL) { /* Output the error message to stderr since it is not a message about a wrong result in MPFR. Anyway the standard output may have changed. */ fprintf (stderr, "Can't open /dev/null or a temporary file\n"); exit (1); } } } else { stdout_redirect = 0; N = atoi (argv[1]); } check_special (); check_mixed (); /* expensive tests */ if (getenv ("MPFR_CHECK_LARGEMEM") != NULL) check_long_string(); check_random (N); test_locale (); if (stdout_redirect) { if ((fflush (stdout) == EOF) || (fclose (stdout) == EOF)) perror ("main"); } tests_end_mpfr (); return 0; } #else /* HAVE_STDARG */ int main (void) { /* We have nothing to test. */ return 77; } #endif /* HAVE_STDARG */