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_ELASTIC_TAG_GENERIC_H)
8 #define CNL_IMPL_ELASTIC_TAG_GENERIC_H
9 
10 #include "../../constant.h"
11 #include "../custom_operator/overloads.h"
12 #include "../num_traits/set_digits.h"
13 #include "../num_traits/width.h"
14 #include "../numbers/set_signedness.h"
15 #include "definition.h"
16 #include "policy.h"
17 
18 #include <type_traits>
19 
21 namespace cnl {
22  template<
23  typename Source, int SrcDigits, class SrcNarrowest,
24  typename Destination, int DestDigits, class DestNarrowest>
25  struct custom_operator<
26  _impl::convert_op,
27  op_value<Source, cnl::elastic_tag<SrcDigits, SrcNarrowest>>,
28  op_value<Destination, cnl::elastic_tag<DestDigits, DestNarrowest>>>
29  : custom_operator<
30  _impl::convert_op,
31  op_value<Source>,
32  op_value<Destination>> {
33  };
34 
35  template<typename Source, typename Destination, int DestDigits, class DestNarrowest>
36  struct custom_operator<
37  _impl::convert_op,
38  op_value<Source>,
39  op_value<Destination, cnl::elastic_tag<DestDigits, DestNarrowest>>> : custom_operator<_impl::convert_op, op_value<Source>, op_value<Destination>> {
40  };
41 
42  template<typename Source, int Digits, class Narrowest, typename Destination>
43  struct custom_operator<
44  _impl::convert_op,
45  op_value<Source, cnl::elastic_tag<Digits, Narrowest>>,
46  op_value<Destination>>
47  : custom_operator<_impl::convert_op, op_value<Source>, op_value<Destination>> {
48  };
49 
50  template<
51  _impl::binary_arithmetic_op Operator,
52  typename Lhs, int LhsDigits, class LhsNarrowest,
53  typename Rhs, int RhsDigits, class RhsNarrowest>
54  struct custom_operator<
55  Operator,
56  op_value<Lhs, elastic_tag<LhsDigits, LhsNarrowest>>,
57  op_value<Rhs, elastic_tag<RhsDigits, RhsNarrowest>>> {
58  static_assert(digits_v<Lhs> >= LhsDigits, "LHS number is not wide enough");
59  static_assert(digits_v<Rhs> >= RhsDigits, "RHS number is not wide enough");
60 
61  using result_tag = decltype(Operator{}(
62  std::declval<elastic_tag<LhsDigits, LhsNarrowest>>(),
63  std::declval<elastic_tag<RhsDigits, RhsNarrowest>>()));
64  using result_rep = typename result_tag::rep;
65 
66  [[nodiscard]] constexpr auto operator()(Lhs const& lhs, Rhs const& rhs) const
67  {
68  return Operator()(static_cast<result_rep>(lhs), static_cast<result_rep>(rhs));
69  }
70  };
71 
72  // shift_operator of scaled_integer and scaled_integer
73  template<_impl::shift_op Operator, typename Lhs, int LhsDigits, typename LhsNarrowest, typename Rhs>
74  requires(!_impl::is_constant<Rhs>::value) struct custom_operator<
75  Operator,
76  op_value<Lhs, elastic_tag<LhsDigits, LhsNarrowest>>,
77  op_value<Rhs>> {
78  [[nodiscard]] constexpr auto operator()(Lhs const& lhs, Rhs const& rhs) const
79  {
80  return Operator{}(lhs, rhs);
81  }
82  };
83 
84  // shift_operator of scaled_integer and something else
85  template<
86  _impl::shift_op Operator,
87  typename Lhs, int LhsDigits, typename LhsNarrowest, int RhsDigits,
88  typename Rhs, typename RhsNarrowest>
89  struct custom_operator<
90  Operator,
91  op_value<Lhs, elastic_tag<LhsDigits, LhsNarrowest>>,
92  op_value<Rhs, elastic_tag<RhsDigits, RhsNarrowest>>>
93  : custom_operator<
94  Operator,
95  op_value<Lhs, elastic_tag<LhsDigits, LhsNarrowest>>,
96  op_value<Rhs, _impl::native_tag>> {
97  };
98 }
99 
100 #endif // CNL_IMPL_ELASTIC_TAG_GENERIC_H
cnl
compositional numeric library
Definition: abort.h:15
std::declval
T declval(T... args)