#include "fast_float/fast_float.h" #include #include #include #include #include #include #include #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 #include // workaround for CYGWIN double cygwin_strtod_l(const char *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(start) + nread; return d; } float cygwin_strtof_l(const char *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(start) + nread; return d; } #endif template char *to_string(T d, char *buffer) { auto written = std::snprintf(buffer, 64, "%.*e", std::numeric_limits::max_digits10 - 1, d); return buffer + written; } void strtof_from_string(const char *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); const char *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; }