.
[gnulib.git] / lib / xstrtod.c
1 /* xstrtod.c - error-checking interface to strtod
2    Copyright (C) 1996 Free Software Foundation, Inc.
3
4    This program is free software; you can redistribute it and/or modify
5    it under the terms of the GNU General Public License as published by
6    the Free Software Foundation; either version 2, or (at your option)
7    any later version.
8
9    This program is distributed in the hope that it will be useful,
10    but WITHOUT ANY WARRANTY; without even the implied warranty of
11    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12    GNU General Public License for more details.
13
14    You should have received a copy of the GNU General Public License
15    along with this program; if not, write to the Free Software
16    Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.  */
17
18 /* Written by Jim Meyering.  */
19
20 #ifdef HAVE_CONFIG_H
21 #include <config.h>
22 #endif
23
24 #ifdef STDC_HEADERS
25 #include <stdlib.h>
26 #else
27 double strtod ();
28 #endif
29
30 #include <errno.h>
31 #include <stdio.h>
32 #include <limits.h>
33 #include "xstrtod.h"
34
35 /* An interface to strtod that encapsulates all the error checking
36    one should usually perform.  Like strtod, but upon successful
37    conversion put the result in *RESULT and return zero.  Return
38    non-zero and don't modify *RESULT upon any failure.  */
39
40 int
41 xstrtod (str, ptr, result)
42      const char *str;
43      const char **ptr;
44      double *result;
45 {
46   double val;
47   char *terminator;
48   int fail;
49
50   fail = 0;
51   errno = 0;
52   val = strtod (str, &terminator);
53
54   /* Having a non-zero terminator is an error only when PTR is NULL. */
55   if (terminator == str || (ptr == NULL && *terminator != '\0'))
56     fail = 1;
57   else
58     {
59       /* Allow underflow (in which case strtod returns zero),
60          but flag overflow as an error. */
61       if (val != 0.0 && errno == ERANGE)
62         fail = 1;
63     }
64
65   if (ptr != NULL)
66     *ptr = terminator;
67
68   *result = val;
69   return fail;
70 }
71