72 lines
2.0 KiB
C
72 lines
2.0 KiB
C
|
/* error-checking interface to strtod-like functions
|
||
|
|
||
|
Copyright (C) 1996, 1999-2000, 2003-2006, 2009-2020 Free Software
|
||
|
Foundation, Inc.
|
||
|
|
||
|
This program is free software: you can redistribute it and/or modify
|
||
|
it under the terms of the GNU General Public License as published by
|
||
|
the Free Software Foundation; either version 3 of the License, or
|
||
|
(at your option) any later version.
|
||
|
|
||
|
This program 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 General Public License for more details.
|
||
|
|
||
|
You should have received a copy of the GNU General Public License
|
||
|
along with this program. If not, see <https://www.gnu.org/licenses/>. */
|
||
|
|
||
|
/* Written by Jim Meyering. */
|
||
|
|
||
|
#include <config.h>
|
||
|
|
||
|
#include "xstrtod.h"
|
||
|
|
||
|
#include <errno.h>
|
||
|
#include <limits.h>
|
||
|
#include <stdio.h>
|
||
|
|
||
|
#if LONG
|
||
|
# define XSTRTOD xstrtold
|
||
|
# define DOUBLE long double
|
||
|
#else
|
||
|
# define XSTRTOD xstrtod
|
||
|
# define DOUBLE double
|
||
|
#endif
|
||
|
|
||
|
/* An interface to a string-to-floating-point conversion function that
|
||
|
encapsulates all the error checking one should usually perform.
|
||
|
Like strtod/strtold, but stores the conversion in *RESULT,
|
||
|
and returns true upon successful conversion.
|
||
|
CONVERT specifies the conversion function, e.g., strtod itself. */
|
||
|
|
||
|
bool
|
||
|
XSTRTOD (char const *str, char const **ptr, DOUBLE *result,
|
||
|
DOUBLE (*convert) (char const *, char **))
|
||
|
{
|
||
|
DOUBLE val;
|
||
|
char *terminator;
|
||
|
bool ok = true;
|
||
|
|
||
|
errno = 0;
|
||
|
val = convert (str, &terminator);
|
||
|
|
||
|
/* Having a non-zero terminator is an error only when PTR is NULL. */
|
||
|
if (terminator == str || (ptr == NULL && *terminator != '\0'))
|
||
|
ok = false;
|
||
|
else
|
||
|
{
|
||
|
/* Allow underflow (in which case CONVERT returns zero),
|
||
|
but flag overflow as an error. The user can decide
|
||
|
to use the limits in RESULT upon ERANGE. */
|
||
|
if (val != 0 && errno == ERANGE)
|
||
|
ok = false;
|
||
|
}
|
||
|
|
||
|
if (ptr != NULL)
|
||
|
*ptr = terminator;
|
||
|
|
||
|
*result = val;
|
||
|
return ok;
|
||
|
}
|