7 #if !defined(CNL_IMPL_DUPLEX_INTEGER_DIGITS_H)
8 #define CNL_IMPL_DUPLEX_INTEGER_DIGITS_H
10 #include "../cnl_assert.h"
11 #include "../num_traits/digits.h"
12 #include "declaration.h"
15 #include <type_traits>
23 template<
typename Result,
typename Lhs>
24 [[nodiscard]] constexpr
auto sensible_right_shift(Lhs
const& lhs,
int rhs) -> Result
25 requires(digits_v<Result> <=
digits_v<decltype(lhs >> rhs)>)
28 using promoted_type = decltype(lhs >> rhs);
29 return static_cast<Result
>(
30 (rhs >= digits_v<promoted_type>)
31 ? lhs >> (digits_v<Lhs> - 1) >> 1
32 : (lhs >> rhs) &
static_cast<promoted_type
>(~Result{}));
35 template<
typename Result,
typename Lhs>
36 [[nodiscard]] constexpr
auto sensible_right_shift(Lhs
const& lhs,
int rhs) -> Result
37 requires(digits_v<Result> >
digits_v<decltype(lhs >> rhs)>)
40 using promoted_type = decltype(lhs >> rhs);
41 return (rhs >= digits_v<promoted_type>) ? Result{}
42 :
static_cast<Result
>(lhs >> rhs);
45 template<
typename Result,
typename Lhs>
46 [[nodiscard]] constexpr
auto sensible_left_shift(Lhs
const& lhs,
int rhs) -> Result
47 requires(digits_v<Result> <=
digits_v<decltype(lhs << rhs)>)
50 using promoted_type = decltype(lhs << rhs);
51 using unsigned_type = numbers::set_signedness_t<promoted_type, false>;
52 return (rhs >= digits_v<promoted_type>)
54 :
static_cast<Result
>(
57 static_cast<unsigned_type
>(
58 lhs & sensible_right_shift<Lhs>(~Result{}, rhs))
62 template<
typename Result,
typename Lhs>
63 [[nodiscard]] constexpr
auto sensible_left_shift(Lhs
const& lhs,
int rhs) -> Result
64 requires(digits_v<Result> >
digits_v<decltype(lhs << rhs)>)
66 return sensible_left_shift<Result>(
static_cast<Result
>(lhs), rhs);
69 template<
typename Result,
typename Lhs>
70 [[nodiscard]] constexpr
auto extra_sensible_right_shift(Lhs
const& lhs,
int rhs) -> Result
72 return (rhs < 0) ? sensible_left_shift<Result>(lhs, -rhs)
73 : sensible_right_shift<Result>(lhs, rhs);
80 template<
typename Upper,
typename Lower>
81 inline constexpr
int digits_v<_impl::duplex_integer<Upper, Lower>> = digits_v<Upper> + digits_v<Lower>;
84 #endif // CNL_IMPL_DUPLEX_INTEGER_DIGITS_H