Improve error handling of c_strtod.
[gnulib.git] / lib / c-strtod.c
1 /* Convert string to double, using the C locale.
2
3    Copyright (C) 2003, 2004, 2006, 2009 Free Software Foundation, Inc.
4
5    This program is free software: you can redistribute it and/or modify
6    it under the terms of the GNU General Public License as published by
7    the Free Software Foundation; either version 3 of the License, or
8    (at your option) any later version.
9
10    This program is distributed in the hope that it will be useful,
11    but WITHOUT ANY WARRANTY; without even the implied warranty of
12    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
13    GNU General Public License for more details.
14
15    You should have received a copy of the GNU General Public License
16    along with this program.  If not, see <http://www.gnu.org/licenses/>.  */
17
18 /* Written by Paul Eggert.  */
19
20 #include <config.h>
21
22 #include "c-strtod.h"
23
24 #include <errno.h>
25 #include <locale.h>
26 #include <stdlib.h>
27
28 #include "xalloc.h"
29
30 #if LONG
31 # define C_STRTOD c_strtold
32 # define DOUBLE long double
33 # define STRTOD_L strtold_l
34 #else
35 # define C_STRTOD c_strtod
36 # define DOUBLE double
37 # define STRTOD_L strtod_l
38 #endif
39
40 /* c_strtold falls back on strtod if strtold doesn't conform to C99.  */
41 #if LONG && HAVE_C99_STRTOLD
42 # define STRTOD strtold
43 #else
44 # define STRTOD strtod
45 #endif
46
47 DOUBLE
48 C_STRTOD (char const *nptr, char **endptr)
49 {
50   DOUBLE r;
51
52 #ifdef LC_ALL_MASK
53
54   locale_t c_locale;
55   int saved_errno;
56
57   c_locale = newlocale (LC_ALL_MASK, "C", (locale_t) 0);
58   if (!c_locale)
59     return 0; /* errno is set here */
60
61   r = STRTOD_L (nptr, endptr, c_locale);
62
63   saved_errno = errno;
64   freelocale (c_locale);
65   errno = saved_errno;
66
67 #else
68
69   char *saved_locale = setlocale (LC_NUMERIC, NULL);
70
71   if (saved_locale)
72     {
73       saved_locale = xstrdup (saved_locale);
74       setlocale (LC_NUMERIC, "C");
75     }
76
77   r = STRTOD (nptr, endptr);
78
79   if (saved_locale)
80     {
81       int saved_errno = errno;
82
83       setlocale (LC_NUMERIC, saved_locale);
84       free (saved_locale);
85       errno = saved_errno;
86     }
87
88 #endif
89
90   return r;
91 }