CNL  2.0.2 (development)
Compositional Numeric Library
definition.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_DEFINITION_H)
8 #define CNL_IMPL_DUPLEX_INTEGER_DEFINITION_H
9 
10 #include "../num_traits/width.h"
11 #include "../numbers/set_signedness.h"
12 #include "../power_value.h"
13 #include "../unreachable.h"
14 #include "declaration.h"
15 #include "digits.h"
16 #include "numbers.h"
17 
18 #include <cmath>
19 #include <concepts>
20 #include <iterator>
21 
23 namespace cnl {
24  namespace _impl {
25  template<typename Integer>
26  [[nodiscard]] constexpr auto is_flushed(Integer const& value)
27  {
28  return value == 0 || value == static_cast<Integer>(~Integer{});
29  }
30 
31  template<typename Result, typename Upper, typename Lower>
32  [[nodiscard]] constexpr auto upper_value(Upper const& upper) -> Result
33  {
34  return (digits_v<Result> <= digits_v<Lower>)
35  ? !is_flushed(upper)
36  ? unreachable<Result>("overflow in narrowing conversion")
37  : Result{}
38  : Result(sensible_left_shift<Result>(upper, digits_v<Lower>));
39  }
40 
41  // Class duplex_integer is bigendian because this is consistent with std::pair.
42  // It makes < faster but possibly makes == slower.
43 
44  template<typename Upper, typename Lower>
45  class duplex_integer {
46  static_assert(!numbers::signedness_v<Lower>, "Lower component must be unsigned.");
47 
48  using upper_type = Upper;
49  using lower_type = Lower;
50 
51  static constexpr int lower_width = width<lower_type>;
52 
53  public:
54  duplex_integer() = default;
55 
56  constexpr duplex_integer(upper_type const& u, lower_type const& l);
57 
58  template<integer Number>
59  constexpr duplex_integer(Number const& n); // NOLINT(hicpp-explicit-conversions,google-explicit-constructor)
60 
61  template<std::floating_point Number>
62  constexpr duplex_integer(Number const& n); // NOLINT(hicpp-explicit-conversions,google-explicit-constructor)
63 
64  [[nodiscard]] constexpr auto upper() const -> upper_type const&
65  {
66  return _upper;
67  }
68 
69  constexpr auto upper() -> upper_type&
70  {
71  return _upper;
72  }
73 
74  [[nodiscard]] constexpr auto lower() const -> lower_type const&
75  {
76  return _lower;
77  }
78 
79  constexpr auto lower() -> lower_type&
80  {
81  return _lower;
82  }
83 
84  [[nodiscard]] explicit constexpr operator bool() const
85  {
86  return _lower || _upper;
87  }
88 
89  template<integer Integer>
90  [[nodiscard]] explicit constexpr operator Integer() const
91  {
92  return upper_value<Integer, Upper, Lower>(_upper) | static_cast<Integer>(_lower);
93  }
94 
95  template<std::floating_point Number>
96  [[nodiscard]] explicit constexpr operator Number() const
97  {
98  return static_cast<Number>(_upper) * power_value<Number, lower_width, 2>()
99  + static_cast<Number>(_lower);
100  }
101 
102  private:
103  // value == _upper<<lower_width + _lower
104  upper_type _upper;
105  lower_type _lower;
106  };
107  }
108 }
109 
110 #endif // CNL_IMPL_DUPLEX_INTEGER_DEFINITION_H
numbers.h
cnl
compositional numeric library
Definition: abort.h:15