7 #if !defined(CNL_IMPL_DUPLEX_INTEGER_SHIFT_H)
8 #define CNL_IMPL_DUPLEX_INTEGER_SHIFT_H
10 #include "../custom_operator/definition.h"
11 #include "../custom_operator/op.h"
12 #include "definition.h"
16 template<
typename Upper,
typename Lower,
typename Rhs>
17 struct custom_operator<
19 op_value<_impl::duplex_integer<Upper, Lower>>,
22 using duplex_integer = _impl::duplex_integer<Upper, Lower>;
23 [[nodiscard]] constexpr
auto with_int(duplex_integer
const& lhs,
int const& rhs)
const
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));
34 [[nodiscard]] constexpr
auto operator()(duplex_integer
const& lhs, Rhs
const& rhs)
const
37 return with_int(lhs,
static_cast<int>(rhs));
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>>,
47 using duplex_integer = _impl::duplex_integer<Upper, Lower>;
49 [[nodiscard]] constexpr
auto with_int(duplex_integer
const& lhs,
int rhs)
const
52 return duplex_integer(calculate_upper(lhs, rhs), calculate_lower(lhs, rhs));
55 [[nodiscard]] constexpr
auto calculate_upper(duplex_integer
const& lhs,
int rhs)
const
58 return _impl::sensible_right_shift<Upper>(lhs.upper(), rhs);
61 [[nodiscard]] constexpr
auto calculate_lower(duplex_integer
const& lhs,
int rhs)
const
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>));
71 [[nodiscard]] constexpr
auto operator()(duplex_integer
const& lhs, Rhs
const& rhs)
const
74 return with_int(lhs,
static_cast<int>(rhs));
79 #endif // CNL_IMPL_DUPLEX_INTEGER_SHIFT_H