7 #if !defined(CNL_IMPL_ELASTIC_TAG_GENERIC_H)
8 #define CNL_IMPL_ELASTIC_TAG_GENERIC_H
10 #include "../../constant.h"
11 #include "../custom_operator/overloads.h"
12 #include "../num_traits/set_digits.h"
13 #include "../num_traits/width.h"
14 #include "../numbers/set_signedness.h"
15 #include "definition.h"
18 #include <type_traits>
23 typename Source,
int SrcDigits,
class SrcNarrowest,
24 typename Destination,
int DestDigits,
class DestNarrowest>
25 struct custom_operator<
27 op_value<Source, cnl::elastic_tag<SrcDigits, SrcNarrowest>>,
28 op_value<Destination, cnl::elastic_tag<DestDigits, DestNarrowest>>>
32 op_value<Destination>> {
35 template<
typename Source,
typename Destination,
int DestDigits,
class DestNarrowest>
36 struct custom_operator<
39 op_value<Destination, cnl::elastic_tag<DestDigits, DestNarrowest>>> : custom_operator<_impl::convert_op, op_value<Source>, op_value<Destination>> {
42 template<
typename Source,
int Digits,
class Narrowest,
typename Destination>
43 struct custom_operator<
45 op_value<Source, cnl::elastic_tag<Digits, Narrowest>>,
46 op_value<Destination>>
47 : custom_operator<_impl::convert_op, op_value<Source>, op_value<Destination>> {
51 _impl::binary_arithmetic_op Operator,
52 typename Lhs,
int LhsDigits,
class LhsNarrowest,
53 typename Rhs,
int RhsDigits,
class RhsNarrowest>
54 struct custom_operator<
56 op_value<Lhs, elastic_tag<LhsDigits, LhsNarrowest>>,
57 op_value<Rhs, elastic_tag<RhsDigits, RhsNarrowest>>> {
58 static_assert(digits_v<Lhs> >= LhsDigits,
"LHS number is not wide enough");
59 static_assert(digits_v<Rhs> >= RhsDigits,
"RHS number is not wide enough");
61 using result_tag = decltype(Operator{}(
62 std::declval<elastic_tag<LhsDigits, LhsNarrowest>>(),
64 using result_rep =
typename result_tag::rep;
66 [[nodiscard]] constexpr
auto operator()(Lhs
const& lhs, Rhs
const& rhs)
const
68 return Operator()(
static_cast<result_rep
>(lhs),
static_cast<result_rep
>(rhs));
73 template<_impl::shift_op Operator,
typename Lhs,
int LhsDigits,
typename LhsNarrowest,
typename Rhs>
74 requires(!_impl::is_constant<Rhs>::value)
struct custom_operator<
76 op_value<Lhs, elastic_tag<LhsDigits, LhsNarrowest>>,
78 [[nodiscard]] constexpr
auto operator()(Lhs
const& lhs, Rhs
const& rhs)
const
80 return Operator{}(lhs, rhs);
86 _impl::shift_op Operator,
87 typename Lhs,
int LhsDigits,
typename LhsNarrowest,
int RhsDigits,
88 typename Rhs,
typename RhsNarrowest>
89 struct custom_operator<
91 op_value<Lhs, elastic_tag<LhsDigits, LhsNarrowest>>,
92 op_value<Rhs, elastic_tag<RhsDigits, RhsNarrowest>>>
95 op_value<Lhs, elastic_tag<LhsDigits, LhsNarrowest>>,
96 op_value<Rhs, _impl::native_tag>> {
100 #endif // CNL_IMPL_ELASTIC_TAG_GENERIC_H