#pragma once #include "type_traits.hpp" #include #include #include #include #include #include #include #include namespace ss { //////////////// // number converters //////////////// template std::enable_if_t, std::optional> to_num( const char* begin, const char* const end) { T ret; auto answer = fast_float::from_chars(begin, end, ret); if (answer.ec != std::errc() || answer.ptr != end) { return std::nullopt; } return ret; } inline std::optional from_char(char c) { if (c >= '0' && c <= '9') { return c - '0'; } return std::nullopt; } #if defined(__clang__) && defined(__MINGW32__) && !defined(__MINGW64__) #define MINGW32_CLANG #endif // mingw32 clang does not support some of the builtin functions #if (defined(__clang__) || defined(__GNUC__) || defined(__GUNG__)) && \ !defined(MINGW32_CLANG) //////////////// // mul overflow detection //////////////// template bool mul_overflow(T& result, T operand) { return __builtin_mul_overflow(result, operand, &result); } template <> inline bool mul_overflow(int& result, int operand) { return __builtin_smul_overflow(result, operand, &result); } template <> inline bool mul_overflow(long& result, long operand) { return __builtin_smull_overflow(result, operand, &result); } template <> inline bool mul_overflow(long long& result, long long operand) { return __builtin_smulll_overflow(result, operand, &result); } template <> inline bool mul_overflow(unsigned int& result, unsigned int operand) { return __builtin_umul_overflow(result, operand, &result); } template <> inline bool mul_overflow(unsigned long& result, unsigned long operand) { return __builtin_umull_overflow(result, operand, &result); } template <> inline bool mul_overflow(unsigned long long& result, unsigned long long operand) { return __builtin_umulll_overflow(result, operand, &result); } //////////////// // addition overflow detection //////////////// template inline bool add_overflow(T& result, T operand) { return __builtin_add_overflow(result, operand, &result); } template <> inline bool add_overflow(int& result, int operand) { return __builtin_sadd_overflow(result, operand, &result); } template <> inline bool add_overflow(long& result, long operand) { return __builtin_saddl_overflow(result, operand, &result); } template <> inline bool add_overflow(long long& result, long long operand) { return __builtin_saddll_overflow(result, operand, &result); } template <> inline bool add_overflow(unsigned int& result, unsigned int operand) { return __builtin_uadd_overflow(result, operand, &result); } template <> inline bool add_overflow(unsigned long& result, unsigned long operand) { return __builtin_uaddl_overflow(result, operand, &result); } template <> inline bool add_overflow(unsigned long long& result, unsigned long long operand) { return __builtin_uaddll_overflow(result, operand, &result); } //////////////// // substraction overflow detection //////////////// template inline bool sub_overflow(T& result, T operand) { return __builtin_sub_overflow(result, operand, &result); } template <> inline bool sub_overflow(int& result, int operand) { return __builtin_ssub_overflow(result, operand, &result); } template <> inline bool sub_overflow(long& result, long operand) { return __builtin_ssubl_overflow(result, operand, &result); } template <> inline bool sub_overflow(long long& result, long long operand) { return __builtin_ssubll_overflow(result, operand, &result); } template <> inline bool sub_overflow(unsigned int& result, unsigned int operand) { return __builtin_usub_overflow(result, operand, &result); } template <> inline bool sub_overflow(unsigned long& result, unsigned long operand) { return __builtin_usubl_overflow(result, operand, &result); } template <> inline bool sub_overflow(unsigned long long& result, unsigned long long operand) { return __builtin_usubll_overflow(result, operand, &result); } template bool shift_and_add_overflow(T& value, T digit, F add_last_digit_owerflow) { if (mul_overflow(value, 10) || add_last_digit_owerflow(value, digit)) { return true; } return false; } #else #ifndef SS_NO_WARNINGS #warning "Use clang or gcc if possible for performance reasons. Define SS_NO_WARNINGS to supress warning." #endif template bool shift_and_add_overflow(T& value, T digit, U is_negative) { digit = (is_negative) ? -digit : digit; T old_value = value; value = 10 * value + digit; T expected_old_value = (value - digit) / 10; if (old_value != expected_old_value) { return true; } return false; } #endif template std::enable_if_t, std::optional> to_num( const char* begin, const char* end) { if (begin == end) { return std::nullopt; } bool is_negative = false; if constexpr (std::is_signed_v) { is_negative = *begin == '-'; if (is_negative) { ++begin; } } #if (defined(__clang__) || defined(__GNUC__) || defined(__GUNG__)) && \ !defined(MINGW32_CLANG) auto add_last_digit_owerflow = (is_negative) ? sub_overflow : add_overflow; #else auto add_last_digit_owerflow = is_negative; #endif T value = 0; for (auto i = begin; i != end; ++i) { if (auto digit = from_char(*i); !digit || shift_and_add_overflow(value, digit.value(), add_last_digit_owerflow)) { return std::nullopt; } } return value; } //////////////// // extract //////////////// namespace error { template struct unsupported_type { constexpr static bool value = false; }; } /* namespace */ template std::enable_if_t && !std::is_floating_point_v && !is_instance_of_v && !is_instance_of_v, bool> extract(const char*, const char*, T&) { static_assert(error::unsupported_type::value, "Conversion for given type is not defined, an " "\'extract\' function needs to be defined!"); } template std::enable_if_t || std::is_floating_point_v, bool> extract(const char* begin, const char* end, T& value) { auto optional_value = to_num(begin, end); if (!optional_value) { return false; } value = optional_value.value(); return true; } template std::enable_if_t, bool> extract( const char* begin, const char* end, T& value) { typename T::value_type raw_value; if (extract(begin, end, raw_value)) { value = raw_value; } else { value = std::nullopt; } return true; } template bool extract_variant(const char* begin, const char* end, T& value) { using IthType = std::variant_alternative_t>; IthType ithValue; if (extract(begin, end, ithValue)) { value = ithValue; return true; } else if constexpr (I + 1 < std::variant_size_v) { return extract_variant(begin, end, value); } return false; } template std::enable_if_t, bool> extract( const char* begin, const char* end, T& value) { return extract_variant(begin, end, value); } //////////////// // extract specialization //////////////// template <> inline bool extract(const char* begin, const char* end, bool& value) { if (end == begin + 1) { if (*begin == '1') { value = true; } else if (*begin == '0') { value = false; } else { return false; } } else { size_t size = end - begin; if (size == 4 && strncmp(begin, "true", size) == 0) { value = true; } else if (size == 5 && strncmp(begin, "false", size) == 0) { value = false; } else { return false; } } return true; } template <> inline bool extract(const char* begin, const char* end, char& value) { value = *begin; return (end == begin + 1); } template <> inline bool extract(const char* begin, const char* end, std::string& value) { value = std::string(begin, end); return true; } } /* ss */