diff --git a/.gitmodules b/.gitmodules index 7d6379c24f..7a238f398a 100644 --- a/.gitmodules +++ b/.gitmodules @@ -41,3 +41,6 @@ [submodule "3party/googletest"] path = 3party/googletest url = https://github.com/google/googletest.git +[submodule "3party/fast_double_parser"] + path = 3party/fast_double_parser + url = https://github.com/lemire/fast_double_parser.git diff --git a/3party/fast_double_parser b/3party/fast_double_parser new file mode 160000 index 0000000000..111ad80417 --- /dev/null +++ b/3party/fast_double_parser @@ -0,0 +1 @@ +Subproject commit 111ad8041765d1623195bd6eb8b6bc7d97c44507 diff --git a/base/CMakeLists.txt b/base/CMakeLists.txt index 615afd5235..855619eebb 100644 --- a/base/CMakeLists.txt +++ b/base/CMakeLists.txt @@ -1,6 +1,8 @@ project(base) set(SRC + # Added to automatically recompile our sources if the submodule is updated. + ../3party/fast_double_parser/include/fast_double_parser.h array_adapters.hpp assert.hpp atomic_shared_ptr.hpp diff --git a/base/string_utils.cpp b/base/string_utils.cpp index 0f76f9612b..b12348431d 100644 --- a/base/string_utils.cpp +++ b/base/string_utils.cpp @@ -2,6 +2,8 @@ #include "base/assert.hpp" +#include "3party/fast_double_parser/include/fast_double_parser.h" + #include #include #include @@ -20,12 +22,14 @@ T RealConverter(char const * start, char ** stop); template <> float RealConverter(char const * start, char ** stop) { + // . or , parsing depends on locale! return std::strtof(start, stop); } template <> double RealConverter(char const * start, char ** stop) { + // . or , parsing depends on locale! return std::strtod(start, stop); } @@ -38,14 +42,24 @@ bool IsFinite(T t) template bool ToReal(char const * start, T & result) { - char * stop; - auto const tmp = RealConverter(start, &stop); - - if (*stop != 0 || start == stop || !IsFinite(tmp)) - return false; - - result = tmp; - return true; + // Try faster implementation first. + double d; + char const * endptr = fast_double_parser::parse_number(start, &d); + if (endptr == nullptr) + { + // Fallback to our implementation, it supports numbers like "1." + char * stop; + result = RealConverter(start, &stop); + if (*stop == 0 && start != stop && IsFinite(result)) + return true; + } + else if (*endptr == 0) + { + result = static_cast(d); + return true; + } + // Do not parse strings that contain additional non-number characters. + return false; } } // namespace