| /* Copyright (C) 1991-2016 Free Software Foundation, Inc. | 
 |    This file is part of the GNU C Library. | 
 |  | 
 |    The GNU C 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 2.1 of the License, or (at your option) any later version. | 
 |  | 
 |    The GNU C 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 C Library; if not, see | 
 |    <http://www.gnu.org/licenses/>.  */ | 
 |  | 
 | #include <ctype.h> | 
 | #include <locale.h> | 
 | #include <stddef.h> | 
 | #include <stdio.h> | 
 | #include <stdlib.h> | 
 | #include <errno.h> | 
 | #include <string.h> | 
 | #include <math.h> | 
 |  | 
 | struct ltest | 
 |   { | 
 |     const char *str;		/* Convert this.  */ | 
 |     double expect;		/* To get this.  */ | 
 |     char left;			/* With this left over.  */ | 
 |     int err;			/* And this in errno.  */ | 
 |   }; | 
 | static const struct ltest tests[] = | 
 |   { | 
 |     { "12.345", 12.345, '\0', 0 }, | 
 |     { "12.345e19", 12.345e19, '\0', 0 }, | 
 |     { "-.1e+9", -.1e+9, '\0', 0 }, | 
 |     { ".125", .125, '\0', 0 }, | 
 |     { "1e20", 1e20, '\0', 0 }, | 
 |     { "0e-19", 0, '\0', 0 }, | 
 |     { "4\00012", 4.0, '\0', 0 }, | 
 |     { "5.9e-76", 5.9e-76, '\0', 0 }, | 
 |     { "0x1.4p+3", 10.0, '\0', 0 }, | 
 |     { "0xAp0", 10.0, '\0', 0 }, | 
 |     { "0x0Ap0", 10.0, '\0', 0 }, | 
 |     { "0x0A", 10.0, '\0', 0 }, | 
 |     { "0xA0", 160.0, '\0', 0 }, | 
 |     { "0x0.A0p8", 160.0, '\0', 0 }, | 
 |     { "0x0.50p9", 160.0, '\0', 0 }, | 
 |     { "0x0.28p10", 160.0, '\0', 0 }, | 
 |     { "0x0.14p11", 160.0, '\0', 0 }, | 
 |     { "0x0.0A0p12", 160.0, '\0', 0 }, | 
 |     { "0x0.050p13", 160.0, '\0', 0 }, | 
 |     { "0x0.028p14", 160.0, '\0', 0 }, | 
 |     { "0x0.014p15", 160.0, '\0', 0 }, | 
 |     { "0x00.00A0p16", 160.0, '\0', 0 }, | 
 |     { "0x00.0050p17", 160.0, '\0', 0 }, | 
 |     { "0x00.0028p18", 160.0, '\0', 0 }, | 
 |     { "0x00.0014p19", 160.0, '\0', 0 }, | 
 |     { "0x1p-1023", | 
 |       1.11253692925360069154511635866620203210960799023116591527666e-308, | 
 |       '\0', 0 }, | 
 |     { "0x0.8p-1022", | 
 |       1.11253692925360069154511635866620203210960799023116591527666e-308, | 
 |       '\0', 0 }, | 
 |     { "Inf", HUGE_VAL, '\0', 0 }, | 
 |     { "-Inf", -HUGE_VAL, '\0', 0 }, | 
 |     { "+InFiNiTy", HUGE_VAL, '\0', 0 }, | 
 |     { "0x80000Ap-23", 0x80000Ap-23, '\0', 0 }, | 
 |     { "1e-324", 0, '\0', ERANGE }, | 
 |     { "0x100000000000008p0", 0x1p56, '\0', 0 }, | 
 |     { "0x100000000000008.p0", 0x1p56, '\0', 0 }, | 
 |     { "0x100000000000008.00p0", 0x1p56, '\0', 0 }, | 
 |     { "0x10000000000000800p0", 0x1p64, '\0', 0 }, | 
 |     { "0x10000000000000801p0", 0x1.0000000000001p64, '\0', 0 }, | 
 |     { NULL, 0, '\0', 0 } | 
 |   }; | 
 |  | 
 | static void expand (char *dst, int c); | 
 | static int long_dbl (void); | 
 | static int locale_test (void); | 
 |  | 
 | static int | 
 | do_test (void) | 
 | { | 
 |   char buf[100]; | 
 |   const struct ltest *lt; | 
 |   char *ep; | 
 |   int status = 0; | 
 |   int save_errno; | 
 |  | 
 |   for (lt = tests; lt->str != NULL; ++lt) | 
 |     { | 
 |       double d; | 
 |  | 
 |       errno = 0; | 
 |       d = strtod(lt->str, &ep); | 
 |       save_errno = errno; | 
 |       printf ("strtod (\"%s\") test %u", | 
 | 	     lt->str, (unsigned int) (lt - tests)); | 
 |       if (d == lt->expect && *ep == lt->left && save_errno == lt->err) | 
 | 	puts ("\tOK"); | 
 |       else | 
 | 	{ | 
 | 	  puts ("\tBAD"); | 
 | 	  if (d != lt->expect) | 
 | 	    printf ("  returns %.60g, expected %.60g\n", d, lt->expect); | 
 | 	  if (lt->left != *ep) | 
 | 	    { | 
 | 	      char exp1[5], exp2[5]; | 
 | 	      expand (exp1, *ep); | 
 | 	      expand (exp2, lt->left); | 
 | 	      printf ("  leaves '%s', expected '%s'\n", exp1, exp2); | 
 | 	    } | 
 | 	  if (save_errno != lt->err) | 
 | 	    printf ("  errno %d (%s)  instead of %d (%s)\n", | 
 | 		    save_errno, strerror (save_errno), | 
 | 		    lt->err, strerror (lt->err)); | 
 | 	  status = 1; | 
 | 	} | 
 |     } | 
 |  | 
 |   sprintf (buf, "%f", strtod ("-0.0", NULL)); | 
 |   if (strcmp (buf, "-0.000000") != 0) | 
 |     { | 
 |       printf ("  strtod (\"-0.0\", NULL) returns \"%s\"\n", buf); | 
 |       status = 1; | 
 |     } | 
 |  | 
 |   const char input[] = "3752432815e-39"; | 
 |  | 
 |   float f1 = strtold (input, NULL); | 
 |   float f2; | 
 |   float f3 = strtof (input, NULL); | 
 |   sscanf (input, "%g", &f2); | 
 |  | 
 |   if (f1 != f2) | 
 |     { | 
 |       printf ("f1 = %a != f2 = %a\n", f1, f2); | 
 |       status = 1; | 
 |     } | 
 |   if (f1 != f3) | 
 |     { | 
 |       printf ("f1 = %a != f3 = %a\n", f1, f3); | 
 |       status = 1; | 
 |     } | 
 |   if (f2 != f3) | 
 |     { | 
 |       printf ("f2 = %a != f3 = %a\n", f2, f3); | 
 |       status = 1; | 
 |     } | 
 |  | 
 |   const char input2[] = "+1.000000000116415321826934814453125"; | 
 |   if (strtold (input2, NULL) != +1.000000000116415321826934814453125L) | 
 |     { | 
 |       printf ("input2: %La != %La\n", strtold (input2, NULL), | 
 | 	      +1.000000000116415321826934814453125L); | 
 |       status = 1; | 
 |     } | 
 |  | 
 |   static struct { const char *str; long double l; } ltests[] = | 
 |     { | 
 |       { "42.0000000000000000001", 42.0000000000000000001L }, | 
 |       { "42.00000000000000000001", 42.00000000000000000001L }, | 
 |       { "42.000000000000000000001", 42.000000000000000000001L } | 
 |     }; | 
 |   int n; | 
 |   for (n = 0; n < sizeof (ltests) / sizeof (ltests[0]); ++n) | 
 |     if (strtold (ltests[n].str, NULL) != ltests[n].l) | 
 |       { | 
 | 	printf ("ltests[%d]: %La != %La\n", n, | 
 | 		strtold (ltests[n].str, NULL), ltests[n].l); | 
 | 	status = 1; | 
 |       } | 
 |  | 
 |   status |= long_dbl (); | 
 |  | 
 |   status |= locale_test (); | 
 |  | 
 |   return status ? EXIT_FAILURE : EXIT_SUCCESS; | 
 | } | 
 |  | 
 | static void | 
 | expand (char *dst, int c) | 
 | { | 
 |   if (isprint (c)) | 
 |     { | 
 |       dst[0] = c; | 
 |       dst[1] = '\0'; | 
 |     } | 
 |   else | 
 |     (void) sprintf (dst, "%#.3o", (unsigned int) c); | 
 | } | 
 |  | 
 | static int | 
 | long_dbl (void) | 
 | { | 
 |   /* Regenerate this string using | 
 |  | 
 |      echo '(2^53-1)*2^(1024-53)' | bc | sed 's/\([^\]*\)\\*$/    "\1"/' | 
 |  | 
 |   */ | 
 |   static const char longestdbl[] = | 
 |     "17976931348623157081452742373170435679807056752584499659891747680315" | 
 |     "72607800285387605895586327668781715404589535143824642343213268894641" | 
 |     "82768467546703537516986049910576551282076245490090389328944075868508" | 
 |     "45513394230458323690322294816580855933212334827479782620414472316873" | 
 |     "8177180919299881250404026184124858368"; | 
 |   double d = strtod (longestdbl, NULL); | 
 |  | 
 |   printf ("strtod (\"%s\", NULL) = %g\n", longestdbl, d); | 
 |  | 
 |   if (d != 179769313486231570814527423731704356798070567525844996598917476803157260780028538760589558632766878171540458953514382464234321326889464182768467546703537516986049910576551282076245490090389328944075868508455133942304583236903222948165808559332123348274797826204144723168738177180919299881250404026184124858368.000000) | 
 |     return 1; | 
 |  | 
 |   return 0; | 
 | } | 
 |  | 
 | /* Perform a few tests in a locale with thousands separators.  */ | 
 | static int | 
 | locale_test (void) | 
 | { | 
 |   static const struct | 
 |   { | 
 |     const char *loc; | 
 |     const char *str; | 
 |     double exp; | 
 |     ptrdiff_t nread; | 
 |   } tests[] = | 
 |     { | 
 |       { "de_DE.UTF-8", "1,5", 1.5, 3 }, | 
 |       { "de_DE.UTF-8", "1.5", 1.0, 1 }, | 
 |       { "de_DE.UTF-8", "1.500", 1500.0, 5 }, | 
 |       { "de_DE.UTF-8", "36.893.488.147.419.103.232", 0x1.0p65, 26 } | 
 |     }; | 
 | #define ntests (sizeof (tests) / sizeof (tests[0])) | 
 |   size_t n; | 
 |   int result = 0; | 
 |  | 
 |   puts ("\nLocale tests"); | 
 |  | 
 |   for (n = 0; n < ntests; ++n) | 
 |     { | 
 |       double d; | 
 |       char *endp; | 
 |  | 
 |       if (setlocale (LC_ALL, tests[n].loc) == NULL) | 
 | 	{ | 
 | 	  printf ("cannot set locale %s\n", tests[n].loc); | 
 | 	  result = 1; | 
 | 	  continue; | 
 | 	} | 
 |  | 
 |       /* We call __strtod_interal here instead of strtod to tests the | 
 | 	 handling of grouping.  */ | 
 |       d = __strtod_internal (tests[n].str, &endp, 1); | 
 |       if (d != tests[n].exp) | 
 | 	{ | 
 | 	  printf ("strtod(\"%s\") returns %g and not %g\n", | 
 | 		  tests[n].str, d, tests[n].exp); | 
 | 	  result = 1; | 
 | 	} | 
 |       else if (endp - tests[n].str != tests[n].nread) | 
 | 	{ | 
 | 	  printf ("strtod(\"%s\") read %td bytes and not %td\n", | 
 | 		  tests[n].str, endp - tests[n].str, tests[n].nread); | 
 | 	  result = 1; | 
 | 	} | 
 |     } | 
 |  | 
 |   if (result == 0) | 
 |     puts ("all OK"); | 
 |  | 
 |   return result; | 
 | } | 
 |  | 
 | #define TEST_FUNCTION do_test () | 
 | #include "../test-skeleton.c" |