mirror of
https://github.com/fastfloat/fast_float.git
synced 2026-06-15 00:16:11 +08:00
tests: parallelize the exhaustive midpoint sweep across hardware threads
This commit is contained in:
parent
7790aa6231
commit
b20c420964
@ -4,6 +4,10 @@ cmake_minimum_required(VERSION 3.11 FATAL_ERROR)
|
||||
|
||||
include(FetchContent)
|
||||
|
||||
# Some tests (the exhaustive sweeps) parallelize across std::thread.
|
||||
set(THREADS_PREFER_PTHREAD_FLAG ON)
|
||||
find_package(Threads REQUIRED)
|
||||
|
||||
option(SYSTEM_DOCTEST "Use system copy of doctest" OFF)
|
||||
option(FASTFLOAT_SUPPLEMENTAL_TESTS "Run supplemental tests" ON)
|
||||
|
||||
@ -49,6 +53,7 @@ function(fast_float_add_cpp_test TEST_NAME)
|
||||
target_compile_options(${TEST_NAME} PUBLIC -Wsign-compare -Wshadow -Wwrite-strings -Wpointer-arith -Winit-self -Wconversion -Wsign-conversion)
|
||||
endif()
|
||||
target_link_libraries(${TEST_NAME} PUBLIC fast_float supplemental-data)
|
||||
target_link_libraries(${TEST_NAME} PUBLIC Threads::Threads)
|
||||
if (NOT SYSTEM_DOCTEST)
|
||||
target_link_libraries(${TEST_NAME} PUBLIC doctest)
|
||||
else ()
|
||||
|
||||
@ -1,5 +1,6 @@
|
||||
#include "fast_float/fast_float.h"
|
||||
|
||||
#include <atomic>
|
||||
#include <cassert>
|
||||
#include <cmath>
|
||||
#include <cstdio>
|
||||
@ -7,6 +8,8 @@
|
||||
#include <iostream>
|
||||
#include <limits>
|
||||
#include <stdexcept>
|
||||
#include <thread>
|
||||
#include <vector>
|
||||
|
||||
#if defined(__CYGWIN__) || defined(__MINGW32__) || defined(__MINGW64__)
|
||||
// Anything at all that is related to cygwin, msys and so forth will
|
||||
@ -74,86 +77,107 @@ void strtof_from_string(char const *st, float &d) {
|
||||
}
|
||||
}
|
||||
|
||||
bool allvalues() {
|
||||
// Checks a single 32-bit word (interpreted as a float). Returns true if the
|
||||
// parser agrees with the reference, false (after logging) on a mismatch.
|
||||
bool check_word(uint32_t word) {
|
||||
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;
|
||||
}
|
||||
}
|
||||
float v;
|
||||
memcpy(&v, &word, sizeof(v));
|
||||
if (!std::isfinite(v)) {
|
||||
return true;
|
||||
}
|
||||
float nextf = std::nextafterf(v, INFINITY);
|
||||
if (copysign(1, v) != copysign(1, nextf)) {
|
||||
return true;
|
||||
}
|
||||
if (!std::isfinite(nextf)) {
|
||||
return true;
|
||||
}
|
||||
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;
|
||||
}
|
||||
|
||||
// Sweeps the whole 2^32 float space, split across hardware threads (the values
|
||||
// are independent). Returns false as soon as any word mismatches.
|
||||
bool allvalues() {
|
||||
unsigned int nthreads = std::thread::hardware_concurrency();
|
||||
if (nthreads == 0) {
|
||||
nthreads = 1;
|
||||
}
|
||||
std::atomic<bool> ok{true};
|
||||
std::vector<std::thread> workers;
|
||||
workers.reserve(nthreads);
|
||||
for (unsigned int t = 0; t < nthreads; t++) {
|
||||
workers.emplace_back([t, nthreads, &ok]() {
|
||||
for (uint64_t w = t;
|
||||
w <= 0xFFFFFFFF && ok.load(std::memory_order_relaxed);
|
||||
w += nthreads) {
|
||||
if (!check_word(uint32_t(w))) {
|
||||
ok.store(false, std::memory_order_relaxed);
|
||||
return;
|
||||
}
|
||||
}
|
||||
});
|
||||
}
|
||||
for (std::thread &worker : workers) {
|
||||
worker.join();
|
||||
}
|
||||
return ok.load();
|
||||
}
|
||||
|
||||
inline void Assert(bool Assertion) {
|
||||
#if defined(__CYGWIN__) || defined(__MINGW32__) || defined(__MINGW64__) || \
|
||||
defined(sun) || defined(__sun)
|
||||
|
||||
Loading…
x
Reference in New Issue
Block a user