CNL  2.0.2 (development)
Compositional Numeric Library
ctors.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_CTORS_H)
8 #define CNL_IMPL_DUPLEX_INTEGER_CTORS_H
9 
10 #include "../power_value.h"
11 #include "definition.h"
12 #include "operators.h"
13 
14 #include <concepts>
15 #include <limits>
16 
18 namespace cnl {
19  namespace _impl {
20  // cnl::_impl::calculate_lower
21  template<typename Lower, typename Integer>
22  requires(digits_v<Lower> >= digits_v<Integer>)
23  [[nodiscard]] constexpr auto calculate_lower(Integer const& input) -> Lower
24  {
25  return Lower(input) & std::numeric_limits<Lower>::max();
26  }
27 
28  template<typename Lower, typename Integer>
29  requires(digits_v<Lower> < digits_v<Integer>)
30  [[nodiscard]] constexpr auto calculate_lower(Integer const& input) -> Lower
31  {
32  return static_cast<Lower>(input & static_cast<Integer>(std::numeric_limits<Lower>::max()));
33  }
34 
35  // cnl::_impl::calculate upper
36  template<typename Upper, typename Lower, typename Integer>
37  requires(digits_v<Lower> >= digits_v<Integer>)
38  [[nodiscard]] constexpr auto calculate_upper(Integer const& input) -> Upper
39  {
40  // sign-friendly flush
41  // TODO: Not reproduced locally. Investigate.
42  // NOLINTNEXTLINE(clang-analyzer-core.UndefinedBinaryOperatorResult)
43  return static_cast<Upper>((input >> (digits_v<Integer> - 1)) >> 1);
44  }
45 
46  template<typename Upper, typename Lower, typename Integer>
47  requires(digits_v<Lower> < digits_v<Integer>)
48  [[nodiscard]] constexpr auto calculate_upper(Integer const& input) -> Upper
49  {
50  return sensible_right_shift<Upper>(input, digits_v<Lower>);
51  }
52 
53  template<typename Upper, typename Lower>
54  constexpr duplex_integer<Upper, Lower>::duplex_integer(
55  upper_type const& u, lower_type const& l)
56  : _upper(u)
57  , _lower(l)
58  {
59  }
60 
61  template<typename Upper, typename Lower>
62  template<integer Number>
63  constexpr duplex_integer<Upper, Lower>::duplex_integer(Number const& n)
64  : _upper(calculate_upper<Upper, Lower>(n))
65  , _lower(calculate_lower<Lower>(n))
66  {
67  }
68 
69  // Note: floating-point constructor is not a constant expression because it relies on
70  // std::fmod and std::fmod is not declared constexpr. (See wg21.link/p0533 for efforts to
71  // remedy this.)
72  template<typename Upper, typename Lower>
73  template<std::floating_point Number>
74  constexpr duplex_integer<Upper, Lower>::duplex_integer(Number const& n)
75  : _upper(Upper(n / power_value<Number, lower_width, 2>()))
76  , _lower(Lower(std::fmod(n, power_value<Number, lower_width, 2>())))
77  {
78  }
79  }
80 }
81 
82 #endif // CNL_IMPL_DUPLEX_INTEGER_CTORS_H
cnl
compositional numeric library
Definition: abort.h:15
std::fmod
T fmod(T... args)
std
STL namespace.
std::numeric_limits::max
T max(T... args)