CNL  2.0.2 (development)
Compositional Numeric Library
shift.h
1 
2 // Copyright John McFarlane 2018.
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_DUPLEX_INTEGER_SHIFT_H)
8 #define CNL_IMPL_DUPLEX_INTEGER_SHIFT_H
9 
10 #include "../custom_operator/definition.h"
11 #include "../custom_operator/op.h"
12 #include "definition.h"
13 
15 namespace cnl {
16  template<typename Upper, typename Lower, typename Rhs>
17  struct custom_operator<
18  _impl::shift_left_op,
19  op_value<_impl::duplex_integer<Upper, Lower>>,
20  op_value<Rhs>> {
21  private:
22  using duplex_integer = _impl::duplex_integer<Upper, Lower>;
23  [[nodiscard]] constexpr auto with_int(duplex_integer const& lhs, int const& rhs) const
24  -> duplex_integer
25  {
26  return duplex_integer(
27  _impl::sensible_left_shift<Upper>(lhs.upper(), rhs)
28  | _impl::extra_sensible_right_shift<Upper>(
29  lhs.lower(), _impl::width<Lower> - rhs),
30  _impl::sensible_left_shift<Lower>(lhs.lower(), rhs));
31  }
32 
33  public:
34  [[nodiscard]] constexpr auto operator()(duplex_integer const& lhs, Rhs const& rhs) const
35  -> duplex_integer
36  {
37  return with_int(lhs, static_cast<int>(rhs));
38  }
39  };
40 
41  template<typename Upper, typename Lower, typename Rhs>
42  struct custom_operator<
43  _impl::shift_right_op,
44  op_value<_impl::duplex_integer<Upper, Lower>>,
45  op_value<Rhs>> {
46  private:
47  using duplex_integer = _impl::duplex_integer<Upper, Lower>;
48 
49  [[nodiscard]] constexpr auto with_int(duplex_integer const& lhs, int rhs) const
50  -> duplex_integer
51  {
52  return duplex_integer(calculate_upper(lhs, rhs), calculate_lower(lhs, rhs));
53  }
54 
55  [[nodiscard]] constexpr auto calculate_upper(duplex_integer const& lhs, int rhs) const
56  -> Upper
57  {
58  return _impl::sensible_right_shift<Upper>(lhs.upper(), rhs);
59  }
60 
61  [[nodiscard]] constexpr auto calculate_lower(duplex_integer const& lhs, int rhs) const
62  -> Lower
63  {
64  return static_cast<Lower>(
65  _impl::sensible_right_shift<Lower>(lhs.lower(), rhs)
66  | _impl::extra_sensible_right_shift<Lower>(
67  lhs.upper(), rhs - _impl::width<Lower>));
68  }
69 
70  public:
71  [[nodiscard]] constexpr auto operator()(duplex_integer const& lhs, Rhs const& rhs) const
72  -> duplex_integer
73  {
74  return with_int(lhs, static_cast<int>(rhs));
75  }
76  };
77 }
78 
79 #endif // CNL_IMPL_DUPLEX_INTEGER_SHIFT_H
cnl
compositional numeric library
Definition: abort.h:15