mirror of https://github.com/FFmpeg/FFmpeg.git
93 lines
2.8 KiB
93 lines
2.8 KiB
/* |
|
* C99-compatible strtod() implementation |
|
* Copyright (c) 2012 Ronald S. Bultje <rsbultje@gmail.com> |
|
* |
|
* This file is part of FFmpeg. |
|
* |
|
* FFmpeg 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. |
|
* |
|
* FFmpeg 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 FFmpeg; if not, write to the Free Software |
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA |
|
*/ |
|
|
|
#include <limits.h> |
|
#include <stdlib.h> |
|
|
|
#include "libavutil/avstring.h" |
|
#include "libavutil/mathematics.h" |
|
|
|
static const char *check_nan_suffix(const char *s) |
|
{ |
|
const char *start = s; |
|
|
|
if (*s++ != '(') |
|
return start; |
|
|
|
while ((*s >= 'a' && *s <= 'z') || (*s >= 'A' && *s <= 'Z') || |
|
(*s >= '0' && *s <= '9') || *s == '_') |
|
s++; |
|
|
|
return *s == ')' ? s + 1 : start; |
|
} |
|
|
|
#undef strtod |
|
double strtod(const char *, char **); |
|
|
|
double avpriv_strtod(const char *nptr, char **endptr) |
|
{ |
|
const char *end; |
|
double res; |
|
|
|
/* Skip leading spaces */ |
|
while (av_isspace(*nptr)) |
|
nptr++; |
|
|
|
if (!av_strncasecmp(nptr, "infinity", 8)) { |
|
end = nptr + 8; |
|
res = INFINITY; |
|
} else if (!av_strncasecmp(nptr, "inf", 3)) { |
|
end = nptr + 3; |
|
res = INFINITY; |
|
} else if (!av_strncasecmp(nptr, "+infinity", 9)) { |
|
end = nptr + 9; |
|
res = INFINITY; |
|
} else if (!av_strncasecmp(nptr, "+inf", 4)) { |
|
end = nptr + 4; |
|
res = INFINITY; |
|
} else if (!av_strncasecmp(nptr, "-infinity", 9)) { |
|
end = nptr + 9; |
|
res = -INFINITY; |
|
} else if (!av_strncasecmp(nptr, "-inf", 4)) { |
|
end = nptr + 4; |
|
res = -INFINITY; |
|
} else if (!av_strncasecmp(nptr, "nan", 3)) { |
|
end = check_nan_suffix(nptr + 3); |
|
res = NAN; |
|
} else if (!av_strncasecmp(nptr, "+nan", 4) || |
|
!av_strncasecmp(nptr, "-nan", 4)) { |
|
end = check_nan_suffix(nptr + 4); |
|
res = NAN; |
|
} else if (!av_strncasecmp(nptr, "0x", 2) || |
|
!av_strncasecmp(nptr, "-0x", 3) || |
|
!av_strncasecmp(nptr, "+0x", 3)) { |
|
/* FIXME this doesn't handle exponents, non-integers (float/double) |
|
* and numbers too large for long long */ |
|
res = strtoll(nptr, (char **)&end, 16); |
|
} else { |
|
res = strtod(nptr, (char **)&end); |
|
} |
|
|
|
if (endptr) |
|
*endptr = (char *)end; |
|
|
|
return res; |
|
}
|
|
|