7 #if !defined(CNL_IMPL_OVERFLOW_IS_OVERFLOW_H)
8 #define CNL_IMPL_OVERFLOW_IS_OVERFLOW_H
10 #include "../../numeric.h"
11 #include "../custom_operator/op.h"
12 #include "../num_traits/digits.h"
13 #include "../numbers/signedness.h"
14 #include "../polarity.h"
18 #include <type_traits>
26 template<
class T, polarity>
27 struct overflow_digits;
30 struct overflow_digits<T, polarity::positive>
35 struct overflow_digits<T, polarity::negative>
46 template<
typename Operand,
bool IsSigned = numbers::
signedness_v<Operand>>
50 template<
typename Operand>
51 struct has_most_negative_number<Operand, true>
53 bool, std::numeric_limits<Operand>::lowest() < -std::numeric_limits<Operand>::max()> {
59 template<polarity Polarity, bool DestinationIsFloat, bool SourceIsFloat>
60 struct is_overflow_convert;
63 struct is_overflow_convert<polarity::positive, false, false> {
64 template<typename Destination, typename Source>
65 [[nodiscard]] constexpr auto operator()(Source const& rhs) const
67 return overflow_digits<Destination, polarity::positive>::value
68 < overflow_digits<Source, polarity::positive>::value
69 && rhs > static_cast<Source>(std::numeric_limits<Destination>::max());
74 struct is_overflow_convert<polarity::positive, false, true> {
75 template<typename Destination, typename Source>
76 [[nodiscard]] constexpr auto operator()(Source const& rhs) const
78 return rhs > static_cast<Source>(std::numeric_limits<Destination>::max());
83 struct is_overflow_convert<polarity::positive, true, false> {
84 template<
typename Destination,
typename Source>
85 [[nodiscard]] constexpr
auto operator()(Source
const& rhs)
const
92 struct is_overflow_convert<polarity::positive, true, true> {
93 template<
typename Destination,
typename Source>
94 [[nodiscard]] constexpr
auto operator()(Source
const&)
const
101 struct is_overflow_convert<polarity::negative, false, false> {
102 template<
typename Destination,
typename Source>
103 [[nodiscard]] constexpr
auto operator()(Source
const& rhs)
const
105 return overflow_digits<Destination, polarity::negative>::value
106 < overflow_digits<Source, polarity::negative>::value
112 struct is_overflow_convert<polarity::negative, false, true> {
113 template<
typename Destination,
typename Source>
114 [[nodiscard]] constexpr
auto operator()(Source
const& rhs)
const
121 struct is_overflow_convert<polarity::negative, true, false> {
122 template<
typename Destination,
typename Source>
123 [[nodiscard]] constexpr
auto operator()(Source
const& rhs)
const
130 struct is_overflow_convert<polarity::negative, true, true> {
131 template<
typename Destination,
typename Source>
132 [[nodiscard]] constexpr
auto operator()(Source
const&)
const
141 template<op Operator,
typename... Operands>
142 struct operator_overflow_traits {
143 using result = op_result<Operator, Operands...>;
145 static constexpr
int positive_digits =
146 _impl::overflow_digits<result, polarity::positive>::value;
147 static constexpr
int negative_digits =
148 _impl::overflow_digits<result, polarity::negative>::value;
150 [[nodiscard]]
static constexpr
auto lowest()
155 [[nodiscard]]
static constexpr
auto max()
160 template<
typename Operand>
161 [[nodiscard]]
static constexpr
auto leading_bits(Operand
const& operand)
163 return cnl::leading_bits(
static_cast<result
>(operand));
170 template<
typename Operator, polarity Polarity>
172 template<
typename... Operands>
173 [[nodiscard]] constexpr
auto operator()(Operands
const&...)
const
179 template<polarity Polarity>
180 struct is_overflow<convert_op, Polarity> {
181 template<
typename Destination,
typename Source>
182 [[nodiscard]] constexpr
auto operator()(Source
const& from)
const
184 using is_overflow_convert = cnl::_impl::is_overflow_convert<
187 return is_overflow_convert{}.template operator()<Destination>(from);
191 #if defined(_MSC_VER)
192 #pragma warning(push)
193 #pragma warning(disable : 4146)
196 struct is_overflow<minus_op, polarity::positive> {
197 template<
typename Rhs>
198 [[nodiscard]] constexpr
auto operator()(Rhs
const& rhs)
const
205 struct is_overflow<minus_op, polarity::negative> {
206 template<
typename Rhs>
207 [[nodiscard]] constexpr
auto operator()(Rhs
const& rhs)
const
209 return !numbers::signedness_v<Rhs> && rhs;
212 #if defined(_MSC_VER)
217 struct is_overflow<add_op, polarity::positive> {
218 template<
typename Lhs,
typename Rhs>
219 [[nodiscard]] constexpr
auto operator()(Lhs
const& lhs, Rhs
const& rhs)
const
221 using traits = operator_overflow_traits<add_op, Lhs, Rhs>;
222 return (
std::max(overflow_digits<Lhs, polarity::positive>::value, overflow_digits<Rhs, polarity::positive>::value)
224 > traits::positive_digits)
225 && lhs > Lhs{0} && rhs > Rhs{0} &&
226 typename traits::result(lhs) > traits::max() - rhs;
231 struct is_overflow<add_op, polarity::negative> {
232 template<
typename Lhs,
typename Rhs>
233 [[nodiscard]] constexpr
auto operator()(Lhs
const& lhs, Rhs
const& rhs)
const
235 using traits = operator_overflow_traits<add_op, Lhs, Rhs>;
236 return (
std::max(overflow_digits<Lhs, polarity::positive>::value, overflow_digits<Rhs, polarity::positive>::value)
238 > traits::positive_digits)
239 && lhs < Lhs{0} && rhs < Rhs{0} &&
240 typename traits::result(lhs) < traits::lowest() - rhs;
244 #if defined(__GNUC__)
245 #pragma GCC diagnostic push
246 #pragma GCC diagnostic ignored "-Wstrict-overflow"
249 struct is_overflow<subtract_op, polarity::positive> {
250 template<
typename Lhs,
typename Rhs>
251 [[nodiscard]] constexpr
auto operator()(Lhs
const& lhs, Rhs
const& rhs)
const
253 using traits = operator_overflow_traits<subtract_op, Lhs, Rhs>;
254 return (
std::max(overflow_digits<Lhs, polarity::positive>::value, overflow_digits<Rhs, polarity::negative>::value)
256 > traits::positive_digits)
263 struct is_overflow<subtract_op, polarity::negative> {
264 template<
typename Lhs,
typename Rhs>
265 [[nodiscard]] constexpr
auto operator()(Lhs
const& lhs, Rhs
const& rhs)
const
267 using traits = operator_overflow_traits<subtract_op, Lhs, Rhs>;
268 return (
std::max(overflow_digits<Lhs, polarity::positive>::value, overflow_digits<Rhs, polarity::positive>::value)
270 > traits::positive_digits)
271 && (rhs >= 0) && lhs < traits::lowest() + rhs;
274 #if defined(__GNUC__)
275 #pragma GCC diagnostic pop
278 #if defined(__GNUC__)
279 #pragma GCC diagnostic push
280 #pragma GCC diagnostic ignored "-Wsign-compare"
283 struct is_overflow<multiply_op, polarity::positive> {
284 template<
typename Lhs,
typename Rhs>
285 [[nodiscard]] constexpr
auto operator()(Lhs
const& lhs, Rhs
const& rhs)
const
287 using traits = operator_overflow_traits<multiply_op, Lhs, Rhs>;
288 return (overflow_digits<Lhs, polarity::positive>::value
289 + overflow_digits<Rhs, polarity::positive>::value
290 > traits::positive_digits)
291 && ((lhs > Lhs{0}) ? (rhs > Rhs{0}) && (traits::max() / rhs) < lhs
292 : (rhs < Rhs{0}) && (traits::max() / rhs) > lhs);
297 struct is_overflow<multiply_op, polarity::negative> {
298 template<
typename Lhs,
typename Rhs>
299 [[nodiscard]] constexpr
auto operator()(Lhs
const& lhs, Rhs
const& rhs)
const
301 using traits = operator_overflow_traits<multiply_op, Lhs, Rhs>;
302 return (overflow_digits<Lhs, polarity::positive>::value
303 + overflow_digits<Rhs, polarity::positive>::value
304 > traits::positive_digits)
305 && ((lhs < Lhs{0}) ? (rhs > Rhs{0}) && (traits::lowest() / rhs) > lhs
306 : (rhs < Rhs{0}) && (traits::lowest() / rhs) < lhs);
309 #if defined(__GNUC__)
310 #pragma GCC diagnostic pop
314 struct is_overflow<divide_op, polarity::positive> {
315 template<
typename Lhs,
typename Rhs>
316 [[nodiscard]] constexpr
auto operator()(Lhs
const& lhs, Rhs
const& rhs)
const
318 using traits = operator_overflow_traits<divide_op, Lhs, Rhs>;
319 return (has_most_negative_number<Lhs>::value) ? rhs == -1 && lhs == traits::lowest()
325 struct is_overflow<shift_left_op, polarity::negative> {
326 template<
typename Lhs,
typename Rhs>
327 requires numbers::signedness_v<Lhs>
328 [[nodiscard]] constexpr
auto operator()(Lhs
const& lhs, Rhs
const& rhs)
const
330 using traits = operator_overflow_traits<shift_left_op, Lhs, Rhs>;
331 return lhs < 0 ? rhs > 0 ? rhs < traits::positive_digits
332 ? (lhs >> (traits::positive_digits - rhs)) != -1
338 template<
typename Lhs,
typename Rhs>
339 requires(!numbers::signedness_v<Lhs>)
340 [[nodiscard]] constexpr
auto
341 operator()(Lhs
const&, Rhs
const&)
const
348 struct is_overflow<shift_left_op, polarity::positive> {
349 template<
typename Lhs,
typename Rhs>
350 [[nodiscard]] constexpr
auto operator()(Lhs
const& lhs, Rhs
const& rhs)
const
352 using traits = operator_overflow_traits<shift_left_op, Lhs, Rhs>;
353 return lhs > 0 ? rhs > 0 ? rhs < traits::positive_digits
354 ? (lhs >> (traits::positive_digits - rhs)) != 0
363 #endif // CNL_IMPL_OVERFLOW_IS_OVERFLOW_H