#include "fast_float/fast_float.h" #include #include #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; } bool basic_test_64bit(std::string vals, double val) { double result_value; auto result = fast_float::from_chars(vals.data(), vals.data() + vals.size(), result_value); if (result.ec != std::errc() && result.ec != std::errc::result_out_of_range) { std::cerr << " I could not parse " << vals << std::endl; return false; } if (std::isnan(val)) { if (!std::isnan(result_value)) { std::cerr << vals << std::endl; std::cerr << "not nan" << result_value << std::endl; return false; } } else if (copysign(1, result_value) != copysign(1, val)) { std::cerr << "I got " << std::hexfloat << result_value << " but I was expecting " << val << std::endl; return false; } else if (result_value != val) { std::cerr << vals << std::endl; std::cerr << "I got " << std::hexfloat << result_value << " but I was expecting " << val << std::endl; std::cerr << std::dec; std::cerr << "string: " << vals << std::endl; return false; } return true; } // Sweeps the whole 2^32 float space (widened to double), split across hardware // threads (the values are independent); stops at the first mismatch. void all_32bit_values() { unsigned int nthreads = std::thread::hardware_concurrency(); if (nthreads == 0) { nthreads = 1; } std::atomic ok{true}; std::vector workers; workers.reserve(nthreads); for (unsigned int t = 0; t < nthreads; t++) { workers.emplace_back([t, nthreads, &ok]() { char buffer[64]; for (uint64_t w = t; w <= 0xFFFFFFFF && ok.load(std::memory_order_relaxed); w += nthreads) { float v32; uint32_t word = uint32_t(w); memcpy(&v32, &word, sizeof(v32)); double v = v32; char const *string_end = to_string(v, buffer); std::string s(buffer, size_t(string_end - buffer)); if (!basic_test_64bit(s, v)) { ok.store(false, std::memory_order_relaxed); return; } } }); } for (std::thread &worker : workers) { worker.join(); } std::cout << std::endl; } int main() { all_32bit_values(); std::cout << std::endl; std::cout << "all ok" << std::endl; return EXIT_SUCCESS; }