CNL  2.0.2 (development)
Compositional Numeric Library
custom_operator.h
1 
2 // Copyright John McFarlane 2019.
3 // Distributed under the Boost Software License, Version 1.0.
4 // (See accompanying file ../LICENSE_1_0.txt or copy at
5 // http://www.boost.org/LICENSE_1_0.txt)
6 
7 #if !defined(CNL_IMPL_OVERFLOW_GENERIC_H)
8 #define CNL_IMPL_OVERFLOW_GENERIC_H
9 
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"
16 
17 #include <type_traits>
18 
20 namespace cnl {
21  namespace _impl {
22  template<tag Tag1, tag Tag2>
23  struct common_overflow_tag {
24  };
25 
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> {
29  };
30 
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> {
34  };
35 
36  template<overflow_tag OverflowTag>
37  struct common_overflow_tag<OverflowTag, OverflowTag>
38  : std::type_identity<OverflowTag> {
39  };
40 
41  template<tag Tag1, tag Tag2>
42  using common_overflow_tag_t = typename common_overflow_tag<Tag1, Tag2>::type;
43  }
44 
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>;
49 
50  [[nodiscard]] constexpr auto operator()(Source const& from) const
51  {
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);
63  }
64  };
66 
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>
71  {
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)
78  : Operator{}(rhs);
79  }
80  };
81 
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>;
86 
87  [[nodiscard]] constexpr auto operator()(Lhs const& lhs, Rhs const& rhs) const -> result_type
88  {
89  result_type result{};
90  if (!_impl::builtin_overflow_operator<Operator, Lhs, Rhs>{}(lhs, rhs, result)) {
91  return result;
92  }
93 
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);
103  default:
104  return _impl::unreachable<result_type>("CNL internal error");
105  }
106  }
107  };
108 #endif
109 
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
113  {
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);
123  }
124  };
125 
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>
130  {
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);
140  }
141  };
142 
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
146  {
147  return _impl::operate<typename _impl::pre_to_assign<Operator>::type, OverflowTag>{}(rhs, 1);
148  }
149  };
150 
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
154  {
155  auto copy = rhs;
156  _impl::operate<typename _impl::post_to_assign<Operator>::type, OverflowTag>{}(rhs, 1);
157  return copy;
158  }
159  };
160 }
161 
162 #endif // CNL_IMPL_OVERFLOW_GENERIC_H
std::copy
T copy(T... args)
cnl
compositional numeric library
Definition: abort.h:15