mirror of
https://github.com/fastfloat/fast_float.git
synced 2025-12-07 01:06:48 +08:00
183 lines
5.7 KiB
C++
183 lines
5.7 KiB
C++
#include "fast_float/fast_float.h"
|
|
|
|
#include <cassert>
|
|
#include <cmath>
|
|
#include <cstdio>
|
|
#include <ios>
|
|
#include <iostream>
|
|
#include <limits>
|
|
#include <stdexcept>
|
|
|
|
#if defined(__CYGWIN__) || defined(__MINGW32__) || defined(__MINGW64__)
|
|
// Anything at all that is related to cygwin, msys and so forth will
|
|
// always use this fallback because we cannot rely on it behaving as normal
|
|
// gcc.
|
|
#include <locale>
|
|
#include <sstream>
|
|
|
|
// workaround for CYGWIN
|
|
double cygwin_strtod_l(char const *start, char **end) {
|
|
double d;
|
|
std::stringstream ss;
|
|
ss.imbue(std::locale::classic());
|
|
ss << start;
|
|
ss >> d;
|
|
if (ss.fail()) {
|
|
*end = nullptr;
|
|
}
|
|
if (ss.eof()) {
|
|
ss.clear();
|
|
}
|
|
auto nread = ss.tellg();
|
|
*end = const_cast<char *>(start) + nread;
|
|
return d;
|
|
}
|
|
|
|
float cygwin_strtof_l(char const *start, char **end) {
|
|
float d;
|
|
std::stringstream ss;
|
|
ss.imbue(std::locale::classic());
|
|
ss << start;
|
|
ss >> d;
|
|
if (ss.fail()) {
|
|
*end = nullptr;
|
|
}
|
|
if (ss.eof()) {
|
|
ss.clear();
|
|
}
|
|
auto nread = ss.tellg();
|
|
*end = const_cast<char *>(start) + nread;
|
|
return d;
|
|
}
|
|
#endif
|
|
|
|
template <typename T> char *to_string(T d, char *buffer) {
|
|
auto written = std::snprintf(buffer, 64, "%.*e",
|
|
std::numeric_limits<T>::max_digits10 - 1, d);
|
|
return buffer + written;
|
|
}
|
|
|
|
void strtof_from_string(char const *st, float &d) {
|
|
char *pr = (char *)st;
|
|
#if defined(__CYGWIN__) || defined(__MINGW32__) || defined(__MINGW64__) || \
|
|
defined(sun) || defined(__sun)
|
|
d = cygwin_strtof_l(st, &pr);
|
|
#elif defined(_WIN32)
|
|
static _locale_t c_locale = _create_locale(LC_ALL, "C");
|
|
d = _strtof_l(st, &pr, c_locale);
|
|
#else
|
|
static locale_t c_locale = newlocale(LC_ALL_MASK, "C", NULL);
|
|
d = strtof_l(st, &pr, c_locale);
|
|
#endif
|
|
if (pr == st) {
|
|
throw std::runtime_error("bug in strtod_from_string");
|
|
}
|
|
}
|
|
|
|
bool allvalues() {
|
|
char buffer[64];
|
|
for (uint64_t w = 0; w <= 0xFFFFFFFF; w++) {
|
|
float v;
|
|
if ((w % 1048576) == 0) {
|
|
std::cout << ".";
|
|
std::cout.flush();
|
|
}
|
|
uint32_t word = uint32_t(w);
|
|
memcpy(&v, &word, sizeof(v));
|
|
if (std::isfinite(v)) {
|
|
float nextf = std::nextafterf(v, INFINITY);
|
|
if (copysign(1, v) != copysign(1, nextf)) {
|
|
continue;
|
|
}
|
|
if (!std::isfinite(nextf)) {
|
|
continue;
|
|
}
|
|
double v1{v};
|
|
assert(float(v1) == v);
|
|
double v2{nextf};
|
|
assert(float(v2) == nextf);
|
|
double midv{v1 + (v2 - v1) / 2};
|
|
float expected_midv = float(midv);
|
|
|
|
char const *string_end = to_string(midv, buffer);
|
|
float str_answer;
|
|
strtof_from_string(buffer, str_answer);
|
|
|
|
float result_value;
|
|
auto result = fast_float::from_chars(buffer, string_end, result_value);
|
|
// Starting with version 4.0 for fast_float, we return result_out_of_range
|
|
// if the value is either too small (too close to zero) or too large
|
|
// (effectively infinity). So std::errc::result_out_of_range is normal for
|
|
// well-formed input strings.
|
|
if (result.ec != std::errc() &&
|
|
result.ec != std::errc::result_out_of_range) {
|
|
std::cerr << "parsing error ? " << buffer << std::endl;
|
|
return false;
|
|
}
|
|
if (std::isnan(v)) {
|
|
if (!std::isnan(result_value)) {
|
|
std::cerr << "not nan" << buffer << std::endl;
|
|
std::cerr << "v " << std::hexfloat << v << std::endl;
|
|
std::cerr << "v2 " << std::hexfloat << v2 << std::endl;
|
|
std::cerr << "midv " << std::hexfloat << midv << std::endl;
|
|
std::cerr << "expected_midv " << std::hexfloat << expected_midv
|
|
<< std::endl;
|
|
return false;
|
|
}
|
|
} else if (copysign(1, result_value) != copysign(1, v)) {
|
|
std::cerr << buffer << std::endl;
|
|
std::cerr << "v " << std::hexfloat << v << std::endl;
|
|
std::cerr << "v2 " << std::hexfloat << v2 << std::endl;
|
|
std::cerr << "midv " << std::hexfloat << midv << std::endl;
|
|
std::cerr << "expected_midv " << std::hexfloat << expected_midv
|
|
<< std::endl;
|
|
std::cerr << "I got " << std::hexfloat << result_value
|
|
<< " but I was expecting " << v << std::endl;
|
|
return false;
|
|
} else if (result_value != str_answer) {
|
|
std::cerr << "no match ? " << buffer << std::endl;
|
|
std::cerr << "v " << std::hexfloat << v << std::endl;
|
|
std::cerr << "v2 " << std::hexfloat << v2 << std::endl;
|
|
std::cerr << "midv " << std::hexfloat << midv << std::endl;
|
|
std::cerr << "expected_midv " << std::hexfloat << expected_midv
|
|
<< std::endl;
|
|
std::cout << "started with " << std::hexfloat << midv << std::endl;
|
|
std::cout << "round down to " << std::hexfloat << str_answer
|
|
<< std::endl;
|
|
std::cout << "got back " << std::hexfloat << result_value << std::endl;
|
|
std::cout << std::dec;
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
std::cout << std::endl;
|
|
return true;
|
|
}
|
|
|
|
inline void Assert(bool Assertion) {
|
|
#if defined(__CYGWIN__) || defined(__MINGW32__) || defined(__MINGW64__) || \
|
|
defined(sun) || defined(__sun)
|
|
if (!Assertion) {
|
|
std::cerr << "Omitting hard failure on msys/cygwin/sun systems.";
|
|
}
|
|
#else
|
|
if (!Assertion) {
|
|
throw std::runtime_error("bug");
|
|
}
|
|
#endif
|
|
}
|
|
|
|
int main() {
|
|
#if defined(__CYGWIN__) || defined(__MINGW32__) || defined(__MINGW64__) || \
|
|
defined(sun) || defined(__sun)
|
|
std::cout << "Warning: msys/cygwin or solaris detected. This particular test "
|
|
"is likely to generate false failures due to our reliance on "
|
|
"the underlying runtime library as a gold standard."
|
|
<< std::endl;
|
|
#endif
|
|
Assert(allvalues());
|
|
std::cout << std::endl;
|
|
std::cout << "all ok" << std::endl;
|
|
return EXIT_SUCCESS;
|
|
}
|