#include "fast_float/fast_float.h" #include #include #include #include #include #include #include #include #include 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; } // Checks a single 32-bit word (interpreted as a float); aborts on a mismatch. void check_word(uint32_t word) { char buffer[64]; float v; memcpy(&v, &word, sizeof(v)); char const *string_end = to_string(v, buffer); 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; abort(); } if (std::isnan(v)) { if (!std::isnan(result_value)) { std::cerr << "not nan" << buffer << std::endl; abort(); } } else if (copysign(1, result_value) != copysign(1, v)) { std::cerr << "I got " << std::hexfloat << result_value << " but I was expecting " << v << std::endl; abort(); } else if (result_value != v) { std::cerr << "no match ? " << buffer << std::endl; std::cout << "started with " << std::hexfloat << v << std::endl; std::cout << "got back " << std::hexfloat << result_value << std::endl; std::cout << std::dec; abort(); } } // Sweeps the whole 2^32 float space, split across hardware threads (the values // are independent); check_word() aborts on the first mismatch. void allvalues() { unsigned int nthreads = std::thread::hardware_concurrency(); if (nthreads == 0) { nthreads = 1; } std::vector workers; workers.reserve(nthreads); for (unsigned int t = 0; t < nthreads; t++) { workers.emplace_back([t, nthreads]() { for (uint64_t w = t; w <= 0xFFFFFFFF; w += nthreads) { check_word(uint32_t(w)); } }); } for (std::thread &worker : workers) { worker.join(); } std::cout << std::endl; } int main() { allvalues(); std::cout << std::endl; std::cout << "all ok" << std::endl; return EXIT_SUCCESS; }