CNL  2.0.2 (development)
Compositional Numeric Library
binary_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_SCALED_BINARY_OPERATOR_H)
8 #define CNL_IMPL_SCALED_BINARY_OPERATOR_H
9 
10 #include "../custom_operator/definition.h"
11 #include "../custom_operator/tagged.h"
12 #include "../num_traits/scale.h"
13 #include "definition.h"
14 #include "is_scaled_tag.h"
15 
16 #include <algorithm>
17 
19 namespace cnl {
20  template<_impl::binary_op Operator, typename Lhs, typename Rhs, int Exponent, int Radix>
21  struct custom_operator<Operator, op_value<Lhs, power<Exponent, Radix>>, op_value<Rhs, power<Exponent, Radix>>>
22  : Operator {
23  };
24 
25  namespace _impl {
26  template<binary_op Operator>
27  struct is_zero_degree : std::true_type {
28  };
29  template<>
30  struct is_zero_degree<multiply_op> : std::false_type {
31  };
32  template<>
33  struct is_zero_degree<divide_op> : std::false_type {
34  };
35  template<>
36  struct is_zero_degree<modulo_op> : std::false_type {
37  };
38  }
39 
40  template<_impl::binary_op Operator, typename Lhs, int LhsExponent, int RhsExponent, typename Rhs, int Radix>
41  requires(LhsExponent != RhsExponent && _impl::is_zero_degree<Operator>::value) struct custom_operator<
42  Operator,
43  op_value<Lhs, power<LhsExponent, Radix>>,
44  op_value<Rhs, power<RhsExponent, Radix>>> {
45  private:
46  static constexpr int _common_exponent = std::min(LhsExponent, RhsExponent);
47  using common_power = power<_common_exponent, Radix>;
48  static constexpr int _lhs_left_shift = LhsExponent - _common_exponent;
49  static constexpr int _rhs_left_shift = RhsExponent - _common_exponent;
50 
51  public:
52  [[nodiscard]] constexpr auto operator()(Lhs const& lhs, Rhs const& rhs) const
53  {
54  return _impl::operate<Operator, common_power>{}(
55  _impl::scale<_lhs_left_shift, Radix>(lhs),
56  _impl::scale<_rhs_left_shift, Radix>(rhs));
57  }
58  };
59 
60  template<_impl::binary_op Operator, typename Lhs, int LhsExponent, typename Rhs, int RhsExponent, int Radix>
61  requires(LhsExponent != RhsExponent && !_impl::is_zero_degree<Operator>::value) struct custom_operator<
62  Operator,
63  op_value<Lhs, power<LhsExponent, Radix>>,
64  op_value<Rhs, power<RhsExponent, Radix>>>
65  : Operator {
66  };
67 
68  template<_impl::shift_op Operator, typename LhsRep, scaled_tag LhsTag, typename Rhs>
69  requires(!_impl::is_constant<Rhs>::value) struct custom_operator<
70  Operator,
71  op_value<LhsRep, LhsTag>,
72  op_value<Rhs>> {
73  [[nodiscard]] constexpr auto operator()(LhsRep const& lhs, Rhs const& rhs) const
74  {
75  return Operator{}(lhs, rhs);
76  }
77  };
78 }
79 
80 #endif // CNL_IMPL_SCALED_BINARY_OPERATOR_H
std::true_type
cnl
compositional numeric library
Definition: abort.h:15
std::min
T min(T... args)