7 #if !defined(CNL_IMPL_OVERFLOW_GENERIC_H)
8 #define CNL_IMPL_OVERFLOW_GENERIC_H
10 #include "../custom_operator/definition.h"
11 #include "../polarity.h"
12 #include "builtin_overflow.h"
13 #include "is_overflow.h"
14 #include "is_overflow_tag.h"
15 #include "overflow_operator.h"
17 #include <type_traits>
22 template<tag Tag1, tag Tag2>
23 struct common_overflow_tag {
26 template<tag OverflowTag, tag Tag>
27 requires(_impl::is_overflow_tag<OverflowTag>::value && !_impl::is_overflow_tag<Tag>::value)
struct common_overflow_tag<OverflowTag, Tag>
28 : std::type_identity<OverflowTag> {
31 template<tag Tag, tag OverflowTag>
32 requires(!_impl::is_overflow_tag<Tag>::value && _impl::is_overflow_tag<OverflowTag>::value)
struct common_overflow_tag<Tag, OverflowTag>
33 : std::type_identity<OverflowTag> {
36 template<overflow_tag OverflowTag>
37 struct common_overflow_tag<OverflowTag, OverflowTag>
38 : std::type_identity<OverflowTag> {
41 template<tag Tag1, tag Tag2>
42 using common_overflow_tag_t =
typename common_overflow_tag<Tag1, Tag2>::type;
46 template<
typename Source, tag SrcTag,
typename Destination, tag DestTag>
47 requires(_impl::is_overflow_tag<DestTag>::value || _impl::is_overflow_tag<SrcTag>::value)
struct custom_operator<_impl::convert_op, op_value<Source, SrcTag>, op_value<Destination, DestTag>> {
48 using overflow_tag = _impl::common_overflow_tag_t<DestTag, SrcTag>;
50 [[nodiscard]] constexpr
auto operator()(Source
const& from)
const
52 return _impl::is_overflow<_impl::convert_op, _impl::polarity::positive>{}
53 .template operator()<Destination>(from)
54 ? _impl::overflow_operator<
55 _impl::convert_op, overflow_tag, _impl::polarity::positive>{}
56 .template operator()<Destination>(from)
57 : _impl::is_overflow<_impl::convert_op, _impl::polarity::negative>{}
58 .template operator()<Destination>(from)
59 ? _impl::overflow_operator<
60 _impl::convert_op, overflow_tag, _impl::polarity::negative>{}
61 .template operator()<Destination>(from)
62 :
static_cast<Destination
>(from);
67 template<_impl::unary_arithmetic_op Operator,
typename Operand, overflow_tag Tag>
68 struct custom_operator<Operator, op_value<Operand, Tag>> {
69 [[nodiscard]] constexpr
auto operator()(Operand
const& rhs)
const
70 -> _impl::op_result<Operator, Operand>
72 return _impl::is_overflow<Operator, _impl::polarity::positive>{}(rhs)
73 ? _impl::overflow_operator<
74 Operator, Tag, _impl::polarity::positive>{}(rhs)
75 : _impl::is_overflow<Operator, _impl::polarity::negative>{}(rhs)
76 ? _impl::overflow_operator<
77 Operator, Tag, _impl::polarity::negative>{}(rhs)
82 #if defined(CNL_BUILTIN_OVERFLOW_ENABLED)
83 template<_impl::binary_arithmetic_op Operator,
typename Lhs, overflow_tag LhsTag,
typename Rhs, overflow_tag RhsTag>
84 requires _impl::builtin_overflow_operator<Operator, Lhs, Rhs>::value
struct custom_operator<Operator, op_value<Lhs, LhsTag>, op_value<Rhs, RhsTag>> {
85 using result_type = _impl::op_result<Operator, Lhs, Rhs>;
87 [[nodiscard]] constexpr
auto operator()(Lhs
const& lhs, Rhs
const& rhs)
const -> result_type
90 if (!_impl::builtin_overflow_operator<Operator, Lhs, Rhs>{}(lhs, rhs, result)) {
94 switch (_impl::overflow_polarity<Operator>{}(lhs, rhs)) {
95 case _impl::polarity::positive:
96 return _impl::overflow_operator<
97 Operator, _impl::common_overflow_tag_t<LhsTag, RhsTag>,
98 _impl::polarity::positive>{}(lhs, rhs);
99 case _impl::polarity::negative:
100 return _impl::overflow_operator<
101 Operator, _impl::common_overflow_tag_t<LhsTag, RhsTag>,
102 _impl::polarity::negative>{}(lhs, rhs);
104 return _impl::unreachable<result_type>(
"CNL internal error");
110 template<_impl::binary_arithmetic_op Operator,
typename Lhs, overflow_tag LhsTag,
typename Rhs, overflow_tag RhsTag>
111 requires(!_impl::builtin_overflow_operator<Operator, Lhs, Rhs>::value)
struct custom_operator<Operator, op_value<Lhs, LhsTag>, op_value<Rhs, RhsTag>> {
112 [[nodiscard]] constexpr
auto operator()(Lhs
const& lhs, Rhs
const& rhs)
const
114 return _impl::is_overflow<Operator, _impl::polarity::positive>{}(lhs, rhs)
115 ? _impl::overflow_operator<
116 Operator, _impl::common_overflow_tag_t<LhsTag, RhsTag>,
117 _impl::polarity::positive>{}(lhs, rhs)
118 : _impl::is_overflow<Operator, _impl::polarity::negative>{}(lhs, rhs)
119 ? _impl::overflow_operator<
120 Operator, _impl::common_overflow_tag_t<LhsTag, RhsTag>,
121 _impl::polarity::negative>{}(lhs, rhs)
122 : Operator{}(lhs, rhs);
126 template<_impl::shift_op Operator,
typename Lhs, overflow_tag LhsTag,
typename Rhs, tag RhsTag>
127 struct custom_operator<Operator, op_value<Lhs, LhsTag>, op_value<Rhs, RhsTag>> {
128 [[nodiscard]] constexpr
auto operator()(Lhs
const& lhs, Rhs
const& rhs)
const
129 -> _impl::op_result<Operator, Lhs, Rhs>
131 return _impl::is_overflow<Operator, _impl::polarity::positive>{}(lhs, rhs)
132 ? _impl::overflow_operator<
133 Operator, _impl::common_overflow_tag_t<LhsTag, RhsTag>,
134 _impl::polarity::positive>{}(lhs, rhs)
135 : _impl::is_overflow<Operator, _impl::polarity::negative>{}(lhs, rhs)
136 ? _impl::overflow_operator<
137 Operator, _impl::common_overflow_tag_t<LhsTag, RhsTag>,
138 _impl::polarity::negative>{}(lhs, rhs)
139 : Operator{}(lhs, rhs);
143 template<_impl::prefix_op Operator,
typename Rhs, overflow_tag OverflowTag>
144 struct custom_operator<Operator, op_value<Rhs, OverflowTag>> {
145 constexpr
auto operator()(Rhs& rhs)
const -> Rhs
147 return _impl::operate<typename _impl::pre_to_assign<Operator>::type, OverflowTag>{}(rhs, 1);
151 template<_impl::postfix_op Operator,
typename Rhs, overflow_tag OverflowTag>
152 struct custom_operator<Operator, op_value<Rhs, OverflowTag>> {
153 constexpr
auto operator()(Rhs& rhs)
const -> Rhs
156 _impl::operate<typename _impl::post_to_assign<Operator>::type, OverflowTag>{}(rhs, 1);
162 #endif // CNL_IMPL_OVERFLOW_GENERIC_H