7 #if !defined(CNL_IMPL_WIDE_TAG_GENERIC_H)
8 #define CNL_IMPL_WIDE_TAG_GENERIC_H
10 #include "../custom_operator/definition.h"
11 #include "../custom_operator/native_tag.h"
12 #include "../num_traits/digits.h"
13 #include "../num_traits/set_width.h"
14 #include "../num_traits/to_rep.h"
15 #include "../num_traits/width.h"
16 #include "../numbers/set_signedness.h"
17 #include "../numbers/signedness.h"
18 #include "definition.h"
19 #include "is_wide_tag.h"
22 #include <type_traits>
26 template<
int ArchetypeDigits,
typename ArchetypeNarrowest,
typename Initializer>
27 struct deduction<wide_tag<ArchetypeDigits, ArchetypeNarrowest>, Initializer> {
30 digits_v<Initializer>,
31 _impl::set_width_t<Initializer, _impl::width<ArchetypeNarrowest>>>;
34 using type = Initializer;
38 template<
typename Src, tag SrcTag,
typename Dest, _impl::any_w
ide_tag DestTag>
39 requires(!_impl::is_wide_tag<SrcTag>)
struct custom_operator<
41 op_value<Src, SrcTag>, op_value<Dest, DestTag>> {
42 [[nodiscard]] constexpr
auto operator()(Src
const& from)
const
44 return custom_operator<_impl::convert_op, op_value<Src, SrcTag>, op_value<Dest>>{}(from);
49 template<
typename Src, _impl::any_w
ide_tag SrcTag,
typename Dest, tag DestTag>
50 struct custom_operator<_impl::convert_op, op_value<Src, SrcTag>, op_value<Dest, DestTag>> {
51 [[nodiscard]] constexpr
auto operator()(Src
const& from)
const
53 return custom_operator<_impl::convert_op, op_value<Src>, op_value<Dest>>{}(from);
57 template<_impl::unary_arithmetic_op Operator,
int Digits,
typename Narrowest,
class Rhs>
58 struct custom_operator<Operator, op_value<Rhs, wide_tag<Digits, Narrowest>>>
63 _impl::binary_arithmetic_op Operator,
int LhsDigits,
class LhsNarrowest,
int RhsDigits,
class RhsNarrowest,
65 struct custom_operator<
67 op_value<Lhs, wide_tag<LhsDigits, LhsNarrowest>>,
68 op_value<Rhs, wide_tag<RhsDigits, RhsNarrowest>>> {
70 static constexpr
auto _max_digits{
std::max(LhsDigits, RhsDigits)};
71 static constexpr
auto _are_signed{
72 numbers::signedness_v<LhsNarrowest> || numbers::signedness_v<RhsNarrowest>};
74 using narrowest = numbers::set_signedness_t<common_type, _are_signed>;
76 using result_tag = wide_tag<_max_digits, narrowest>;
77 using result =
typename result_tag::rep;
80 [[nodiscard]] constexpr
auto operator()(Lhs
const& lhs, Rhs
const& rhs)
const -> result
82 return static_cast<result
>(Operator{}(lhs, rhs));
86 template<_impl::shift_op Operator,
typename Lhs,
int LhsDigits,
typename LhsNarrowest,
typename Rhs>
87 struct custom_operator<
89 op_value<Lhs, wide_tag<LhsDigits, LhsNarrowest>>,
91 [[nodiscard]] constexpr
auto operator()(Lhs
const& lhs, Rhs
const& rhs)
const
93 return Operator{}(lhs, rhs);
97 template<_impl::comparison_op Operator,
int LhsDigits,
class LhsNarrowest,
int RhsDigits,
class RhsNarrowest>
98 struct custom_operator<
99 Operator, wide_tag<LhsDigits, LhsNarrowest>, wide_tag<RhsDigits, RhsNarrowest>>
102 op_value<_impl::native_tag>,
103 op_value<_impl::native_tag>> {
106 template<_impl::prefix_op Operator,
typename Rhs,
int Digits,
typename Narrowest>
107 struct custom_operator<Operator, op_value<Rhs, wide_tag<Digits, Narrowest>>> : Operator {
110 template<_impl::postfix_op Operator,
typename Lhs,
int Digits,
typename Narrowest>
111 struct custom_operator<Operator, op_value<Lhs, wide_tag<Digits, Narrowest>>> : Operator {
115 #endif // CNL_IMPL_WIDE_TAG_GENERIC_H