CNL  2.0.2 (development)
Compositional Numeric Library
digits.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_DIGITS_H)
8 #define CNL_IMPL_DUPLEX_INTEGER_DIGITS_H
9 
10 #include "../cnl_assert.h"
11 #include "../num_traits/digits.h"
12 #include "declaration.h"
13 #include "numbers.h"
14 
15 #include <type_traits>
16 
18 namespace cnl {
19  namespace _impl {
21  // 'sensible' shift operations - for when the compiler is being obtuse
22 
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)>)
26  {
27  CNL_ASSERT(rhs >= 0);
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{}));
33  }
34 
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)>)
38  {
39  CNL_ASSERT(rhs >= 0);
40  using promoted_type = decltype(lhs >> rhs);
41  return (rhs >= digits_v<promoted_type>) ? Result{}
42  : static_cast<Result>(lhs >> rhs);
43  }
44 
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)>)
48  {
49  CNL_ASSERT(rhs >= 0);
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>)
53  ? Result{}
54  : static_cast<Result>(
55  // TODO: Not reproduced locally. Investigate.
56  // NOLINTNEXTLINE(clang-analyzer-core.UndefinedBinaryOperatorResult)
57  static_cast<unsigned_type>(
58  lhs & sensible_right_shift<Lhs>(~Result{}, rhs))
59  << rhs);
60  }
61 
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)>)
65  {
66  return sensible_left_shift<Result>(static_cast<Result>(lhs), rhs);
67  }
68 
69  template<typename Result, typename Lhs>
70  [[nodiscard]] constexpr auto extra_sensible_right_shift(Lhs const& lhs, int rhs) -> Result
71  {
72  return (rhs < 0) ? sensible_left_shift<Result>(lhs, -rhs)
73  : sensible_right_shift<Result>(lhs, rhs);
74  }
75  }
76 
78  // cnl::digits_v<cnl::_impl::duplex_integer<>>
79 
80  template<typename Upper, typename Lower>
81  inline constexpr int digits_v<_impl::duplex_integer<Upper, Lower>> = digits_v<Upper> + digits_v<Lower>;
82 }
83 
84 #endif // CNL_IMPL_DUPLEX_INTEGER_DIGITS_H
cnl::digits_v
constexpr int digits_v
provide number of binary digits of the given type
Definition: digits.h:31
numbers.h
cnl
compositional numeric library
Definition: abort.h:15