CNL  2.0.2 (development)
Compositional Numeric Library
binary_arithmetic_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_WRAPPER_BINARY_OPERATOR_H)
8 #define CNL_IMPL_WRAPPER_BINARY_OPERATOR_H
9 
10 #include "../custom_operator/definition.h"
11 #include "../custom_operator/is_same_tag_family.h"
12 #include "../custom_operator/native_tag.h"
13 #include "../custom_operator/op.h"
14 #include "../custom_operator/overloads.h"
15 #include "../num_traits/set_rep.h"
16 #include "../num_traits/set_tag.h"
17 #include "is_wrapper.h"
18 #include "make_wrapper.h"
19 #include "numbers.h"
20 #include "operator_helpers.h"
21 
22 #include <concepts>
23 #include <type_traits>
24 
26 namespace cnl {
27  // higher OP any_wrapper
28  template<_impl::binary_arithmetic_op Operator, std::floating_point Lhs, _impl::any_wrapper Rhs>
29  struct custom_operator<Operator, op_value<Lhs>, op_value<Rhs>> {
30  [[nodiscard]] constexpr auto operator()(Lhs const& lhs, Rhs const& rhs) const
31  {
32  return Operator()(lhs, static_cast<Lhs>(rhs));
33  }
34  };
35 
36  // any_wrapper OP higher
37  template<_impl::binary_arithmetic_op Operator, _impl::any_wrapper Lhs, std::floating_point Rhs>
38  struct custom_operator<Operator, op_value<Lhs>, op_value<Rhs>> {
39  [[nodiscard]] constexpr auto operator()(Lhs const& lhs, Rhs const& rhs) const
40  {
41  return Operator()(static_cast<Rhs>(lhs), rhs);
42  }
43  };
44 
45  // lower OP any_wrapper
46  template<_impl::binary_arithmetic_op Operator, class Lhs, class Rhs>
47  requires _impl::number_can_wrap<Rhs, Lhs>::value struct custom_operator<Operator, op_value<Lhs>, op_value<Rhs>> {
48  [[nodiscard]] constexpr auto operator()(Lhs const& lhs, Rhs const& rhs) const
49  {
50  return Operator()(_impl::from_value<Rhs>(lhs), rhs);
51  }
52  };
53 
54  // any_wrapper OP lower
55  template<_impl::binary_arithmetic_op Operator, class Lhs, class Rhs>
56  requires _impl::number_can_wrap<Lhs, Rhs>::value struct custom_operator<Operator, op_value<Lhs>, op_value<Rhs>> {
57  [[nodiscard]] constexpr auto operator()(Lhs const& lhs, Rhs const& rhs) const
58  {
59  return Operator()(lhs, from_value<Lhs, Rhs>{}(rhs));
60  }
61  };
62 
63  template<_impl::binary_arithmetic_op Operator, _impl::any_wrapper Lhs, _impl::any_wrapper Rhs>
64  requires(_impl::is_same_tag_family<_impl::tag_of_t<Lhs>, _impl::tag_of_t<Rhs>>::value) struct custom_operator<Operator, op_value<Lhs>, op_value<Rhs>> {
65  [[nodiscard]] constexpr auto operator()(Lhs const& lhs, Rhs const& rhs) const
66  {
67  auto const lhs_rep{_impl::to_rep(lhs)};
68  auto const rhs_rep{_impl::to_rep(rhs)};
69  using rep_operator = custom_operator<
70  Operator,
71  op_value<std::remove_const_t<decltype(lhs_rep)>, _impl::tag_of_t<Lhs>>,
72  op_value<std::remove_const_t<decltype(rhs_rep)>, _impl::tag_of_t<Rhs>>>;
73  auto const result_rep{rep_operator{}(lhs_rep, rhs_rep)};
74  using result_rep_type = std::remove_const_t<decltype(result_rep)>;
75  using result_tag = _impl::op_result<Operator, _impl::tag_of_t<Lhs>, _impl::tag_of_t<Rhs>>;
76  using result_archetype = _impl::set_rep_t<_impl::set_tag_t<Lhs, result_tag>, result_rep_type>;
77 
78  return _impl::from_rep<result_archetype>(result_rep);
79  }
80  };
81 }
82 
83 #endif // CNL_IMPL_WRAPPER_BINARY_OPERATOR_H
cnl
compositional numeric library
Definition: abort.h:15
numbers.h