15 #include <type_traits>
28 template <
typename BoolCondition>
29 using EnableIf =
typename std::enable_if<BoolCondition::value, void>::type;
58 template <
typename T1,
typename T2>
64 template <
typename From,
typename To>
67 if(from >
static_cast<From
>(std::numeric_limits<To>::max()))
69 throw std::runtime_error(
"Value outside the max numerical limit.");
73 template <
typename From,
typename To>
78 if(from != 0 && from != 1)
80 throw std::runtime_error(
"Implicit casting to bool is not allowed");
83 else if(from < std::numeric_limits<To>::min())
85 throw std::runtime_error(
"Value outside the lovest numerical limit.");
89 template <
typename From,
typename To>
93 if constexpr(std::is_integral_v<From> && std::is_floating_point_v<To>)
97 bool doesnt_fit =
false;
98 if constexpr(!std::is_signed_v<From>)
100 doesnt_fit =
static_cast<uint64_t
>(from) > max_exact;
104 doesnt_fit = std::abs(
static_cast<int64_t
>(from)) >
static_cast<int64_t
>(max_exact);
108 throw std::runtime_error(
"Loss of precision when converting a large integer number "
109 "to floating point:" +
114 else if constexpr(std::is_floating_point_v<From> && std::is_integral_v<To>)
116 if(from >
static_cast<From
>(std::numeric_limits<To>::max()) ||
117 from <
static_cast<From
>(std::numeric_limits<To>::lowest()) ||
118 from != std::nearbyint(from))
120 throw std::runtime_error(
"Invalid floating point to integer conversion");
126 if(from >
static_cast<From
>(std::numeric_limits<To>::max()) ||
127 from <
static_cast<From
>(std::numeric_limits<To>::lowest()))
129 throw std::runtime_error(
"Value outside numeric limits");
131 To as_target =
static_cast<To
>(from);
132 From back_to_source =
static_cast<From
>(as_target);
133 if(from != back_to_source)
135 throw std::runtime_error(
"Value truncated in conversion");
142 template <
typename SRC,
typename DST>
145 static_assert(is_convertible_type<SRC>() && is_convertible_type<DST>(),
"Not "
148 constexpr
bool both_integers = is_integer<SRC>() && is_integer<DST>();
150 if constexpr(is_signed<SRC>() && !is_signed<DST>())
154 throw std::runtime_error(
"Value is negative and can't be converted to unsigned");
160 if constexpr(is_same<SRC, DST>() || (is_same<SRC, float>() && is_same<DST, double>()))
163 target =
static_cast<DST
>(source);
165 else if constexpr(both_integers)
167 if constexpr(
sizeof(SRC) ==
sizeof(DST) && !is_signed<SRC>() && is_signed<DST>())
169 checkUpperLimit<SRC, DST>(source);
172 else if constexpr(
sizeof(SRC) >
sizeof(DST))
174 if constexpr(is_signed<SRC>())
176 checkLowerLimit<SRC, DST>(source);
178 checkUpperLimit<SRC, DST>(source);
180 target =
static_cast<DST
>(source);
183 else if constexpr(is_convertible_to_bool<SRC>() && is_same<DST, bool>())
185 target =
static_cast<DST
>(source);
196 checkTruncation<SRC, DST>(source);
198 target =
static_cast<DST
>(source);