7 #if !defined(CNL_IMPL_POWER_H)
8 #define CNL_IMPL_POWER_H
10 #include "../constant.h"
11 #include "num_traits/digits.h"
12 #include "num_traits/from_value.h"
15 #include <type_traits>
24 typename S,
int Exponent,
int Radix,
bool PositiveExponent = (0 < Exponent),
25 bool OddExponent = ((Exponent & 1) != 0),
27 struct power_value_fn;
29 template<
typename S,
int Radix>
30 struct power_value_fn<S, 0, Radix, false, false, false> {
31 [[nodiscard]] constexpr
auto operator()()
const
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
42 decltype(std::declval<S>() >>
std::declval<constant<digits_v<S> - 1>>()){1}
43 << constant<Exponent>{})>;
46 || Exponent < result_numeric_limits::digits,
47 "attempted operation will result in overflow");
51 return decltype(std::declval<S>() >>
std::declval<constant<digits_v<S> - 1>>()){1}
52 << constant<Exponent>{};
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
60 return power_value_fn<S, (Exponent - 1), Radix>{}() * Radix;
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
68 return Exponent ? S(1.) / power_value_fn<S, -Exponent, Radix>{}() : S{1.};
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)
79 [[nodiscard]] constexpr
auto operator()()
const
81 return square(power_value_fn<S, Exponent / 2, Radix>{}());
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)
92 [[nodiscard]] constexpr
auto operator()()
const
94 return S(Radix) * power_value_fn<S, (Exponent - 1), Radix>{}();
98 template<
typename S,
int Exponent,
int Radix>
99 [[nodiscard]] constexpr
auto power_value()
101 return power_value_fn<S, Exponent, Radix>{}();
106 #endif // CNL_IMPL_POWER_H