CNL  2.0.2 (development)
Compositional Numeric Library
power_value.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_POWER_H)
8 #define CNL_IMPL_POWER_H
9 
10 #include "../constant.h"
11 #include "num_traits/digits.h"
12 #include "num_traits/from_value.h"
13 
14 #include <limits>
15 #include <type_traits>
16 
18 namespace cnl {
19  namespace _impl {
21  // power_value - one integer raised to another as a floating-point or integer type
22 
23  template<
24  typename S, int Exponent, int Radix, bool PositiveExponent = (0 < Exponent),
25  bool OddExponent = ((Exponent & 1) != 0),
26  bool FloatingPointS = std::numeric_limits<S>::is_iec559>
27  struct power_value_fn;
28 
29  template<typename S, int Radix>
30  struct power_value_fn<S, 0, Radix, false, false, false> {
31  [[nodiscard]] constexpr auto operator()() const
32  {
33  return S{1};
34  }
35  };
36 
37  template<typename S, int Exponent, bool OddExponent>
38  struct power_value_fn<S, Exponent, 2, true, OddExponent, false> {
39  [[nodiscard]] constexpr auto operator()() const
40  {
41  using result_numeric_limits = std::numeric_limits<decltype(
42  decltype(std::declval<S>() >> std::declval<constant<digits_v<S> - 1>>()){1}
43  << constant<Exponent>{})>;
44  static_assert(
46  || Exponent < result_numeric_limits::digits,
47  "attempted operation will result in overflow");
48 
49  // TODO: This expression is so ugly that it might justify
50  // a separate specialization of power for elastic_integer
51  return decltype(std::declval<S>() >> std::declval<constant<digits_v<S> - 1>>()){1}
52  << constant<Exponent>{};
53  }
54  };
55 
56  template<typename S, int Exponent, int Radix, bool OddExponent>
57  struct power_value_fn<S, Exponent, Radix, true, OddExponent, false> {
58  [[nodiscard]] constexpr auto operator()() const
59  {
60  return power_value_fn<S, (Exponent - 1), Radix>{}() * Radix;
61  }
62  };
63 
64  template<typename S, int Exponent, int Radix, bool PositiveExponent, bool OddExponent>
65  struct power_value_fn<S, Exponent, Radix, PositiveExponent, OddExponent, true> {
66  [[nodiscard]] constexpr auto operator()() const -> S
67  {
68  return Exponent ? S(1.) / power_value_fn<S, -Exponent, Radix>{}() : S{1.};
69  }
70  };
71 
72  template<typename S, int Exponent, int Radix>
73  struct power_value_fn<S, Exponent, Radix, true, false, true> {
74  [[nodiscard]] constexpr static auto square(S const& r)
75  {
76  return r * r;
77  }
78 
79  [[nodiscard]] constexpr auto operator()() const
80  {
81  return square(power_value_fn<S, Exponent / 2, Radix>{}());
82  }
83  };
84 
85  template<typename S, int Exponent, int Radix>
86  struct power_value_fn<S, Exponent, Radix, true, true, true> {
87  [[nodiscard]] constexpr static auto square(S const& r)
88  {
89  return r * r;
90  }
91 
92  [[nodiscard]] constexpr auto operator()() const
93  {
94  return S(Radix) * power_value_fn<S, (Exponent - 1), Radix>{}();
95  }
96  };
97 
98  template<typename S, int Exponent, int Radix>
99  [[nodiscard]] constexpr auto power_value()
100  {
101  return power_value_fn<S, Exponent, Radix>{}();
102  }
103  }
104 }
105 
106 #endif // CNL_IMPL_POWER_H
std::is_signed
cnl
compositional numeric library
Definition: abort.h:15
std::is_integral
std::numeric_limits
std::declval
T declval(T... args)