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_WIDE_TAG_GENERIC_H)
8 #define CNL_IMPL_WIDE_TAG_GENERIC_H
9 
10 #include "../custom_operator/definition.h"
11 #include "../custom_operator/native_tag.h"
12 #include "../num_traits/digits.h"
13 #include "../num_traits/set_width.h"
14 #include "../num_traits/to_rep.h"
15 #include "../num_traits/width.h"
16 #include "../numbers/set_signedness.h"
17 #include "../numbers/signedness.h"
18 #include "definition.h"
19 #include "is_wide_tag.h"
20 
21 #include <algorithm>
22 #include <type_traits>
23 
25 namespace cnl {
26  template<int ArchetypeDigits, typename ArchetypeNarrowest, typename Initializer>
27  struct deduction<wide_tag<ArchetypeDigits, ArchetypeNarrowest>, Initializer> {
28  // tag associated with deduced type
29  using tag = wide_tag<
30  digits_v<Initializer>,
31  _impl::set_width_t<Initializer, _impl::width<ArchetypeNarrowest>>>;
32 
33  // deduced type
34  using type = Initializer;
35  };
36 
38  template<typename Src, tag SrcTag, typename Dest, _impl::any_wide_tag DestTag>
39  requires(!_impl::is_wide_tag<SrcTag>) struct custom_operator<
40  _impl::convert_op,
41  op_value<Src, SrcTag>, op_value<Dest, DestTag>> {
42  [[nodiscard]] constexpr auto operator()(Src const& from) const
43  {
44  return custom_operator<_impl::convert_op, op_value<Src, SrcTag>, op_value<Dest>>{}(from);
45  }
46  };
48 
49  template<typename Src, _impl::any_wide_tag SrcTag, typename Dest, tag DestTag>
50  struct custom_operator<_impl::convert_op, op_value<Src, SrcTag>, op_value<Dest, DestTag>> {
51  [[nodiscard]] constexpr auto operator()(Src const& from) const
52  {
53  return custom_operator<_impl::convert_op, op_value<Src>, op_value<Dest>>{}(from);
54  }
55  };
56 
57  template<_impl::unary_arithmetic_op Operator, int Digits, typename Narrowest, class Rhs>
58  struct custom_operator<Operator, op_value<Rhs, wide_tag<Digits, Narrowest>>>
59  : Operator {
60  };
61 
62  template<
63  _impl::binary_arithmetic_op Operator, int LhsDigits, class LhsNarrowest, int RhsDigits, class RhsNarrowest,
64  class Lhs, class Rhs>
65  struct custom_operator<
66  Operator,
67  op_value<Lhs, wide_tag<LhsDigits, LhsNarrowest>>,
68  op_value<Rhs, wide_tag<RhsDigits, RhsNarrowest>>> {
69  private:
70  static constexpr auto _max_digits{std::max(LhsDigits, RhsDigits)};
71  static constexpr auto _are_signed{
72  numbers::signedness_v<LhsNarrowest> || numbers::signedness_v<RhsNarrowest>};
73  using common_type = typename std::common_type<LhsNarrowest, RhsNarrowest>::type;
74  using narrowest = numbers::set_signedness_t<common_type, _are_signed>;
75 
76  using result_tag = wide_tag<_max_digits, narrowest>;
77  using result = typename result_tag::rep;
78 
79  public:
80  [[nodiscard]] constexpr auto operator()(Lhs const& lhs, Rhs const& rhs) const -> result
81  {
82  return static_cast<result>(Operator{}(lhs, rhs));
83  }
84  };
85 
86  template<_impl::shift_op Operator, typename Lhs, int LhsDigits, typename LhsNarrowest, typename Rhs>
87  struct custom_operator<
88  Operator,
89  op_value<Lhs, wide_tag<LhsDigits, LhsNarrowest>>,
90  op_value<Rhs>> {
91  [[nodiscard]] constexpr auto operator()(Lhs const& lhs, Rhs const& rhs) const
92  {
93  return Operator{}(lhs, rhs);
94  }
95  };
96 
97  template<_impl::comparison_op Operator, int LhsDigits, class LhsNarrowest, int RhsDigits, class RhsNarrowest>
98  struct custom_operator<
99  Operator, wide_tag<LhsDigits, LhsNarrowest>, wide_tag<RhsDigits, RhsNarrowest>>
100  : custom_operator<
101  Operator,
102  op_value<_impl::native_tag>,
103  op_value<_impl::native_tag>> {
104  };
105 
106  template<_impl::prefix_op Operator, typename Rhs, int Digits, typename Narrowest>
107  struct custom_operator<Operator, op_value<Rhs, wide_tag<Digits, Narrowest>>> : Operator {
108  };
109 
110  template<_impl::postfix_op Operator, typename Lhs, int Digits, typename Narrowest>
111  struct custom_operator<Operator, op_value<Lhs, wide_tag<Digits, Narrowest>>> : Operator {
112  };
113 }
114 
115 #endif // CNL_IMPL_WIDE_TAG_GENERIC_H
std::common_type
cnl
compositional numeric library
Definition: abort.h:15
std::max
T max(T... args)