7 #if !defined(CNL_IMPL_SCALED_BINARY_OPERATOR_H)
8 #define CNL_IMPL_SCALED_BINARY_OPERATOR_H
10 #include "../custom_operator/definition.h"
11 #include "../custom_operator/tagged.h"
12 #include "../num_traits/scale.h"
13 #include "definition.h"
14 #include "is_scaled_tag.h"
20 template<_impl::binary_op Operator,
typename Lhs,
typename Rhs,
int Exponent,
int Radix>
21 struct custom_operator<Operator, op_value<Lhs, power<Exponent, Radix>>, op_value<Rhs, power<Exponent, Radix>>>
26 template<binary_op Operator>
40 template<_impl::binary_op Operator,
typename Lhs,
int LhsExponent,
int RhsExponent,
typename Rhs,
int Radix>
41 requires(LhsExponent != RhsExponent && _impl::is_zero_degree<Operator>::value)
struct custom_operator<
43 op_value<Lhs, power<LhsExponent, Radix>>,
44 op_value<Rhs, power<RhsExponent, Radix>>> {
46 static constexpr
int _common_exponent =
std::min(LhsExponent, RhsExponent);
47 using common_power = power<_common_exponent, Radix>;
48 static constexpr
int _lhs_left_shift = LhsExponent - _common_exponent;
49 static constexpr
int _rhs_left_shift = RhsExponent - _common_exponent;
52 [[nodiscard]] constexpr
auto operator()(Lhs
const& lhs, Rhs
const& rhs)
const
54 return _impl::operate<Operator, common_power>{}(
55 _impl::scale<_lhs_left_shift, Radix>(lhs),
56 _impl::scale<_rhs_left_shift, Radix>(rhs));
60 template<_impl::binary_op Operator,
typename Lhs,
int LhsExponent,
typename Rhs,
int RhsExponent,
int Radix>
61 requires(LhsExponent != RhsExponent && !_impl::is_zero_degree<Operator>::value)
struct custom_operator<
63 op_value<Lhs, power<LhsExponent, Radix>>,
64 op_value<Rhs, power<RhsExponent, Radix>>>
68 template<_impl::shift_op Operator,
typename LhsRep, scaled_tag LhsTag,
typename Rhs>
69 requires(!_impl::is_constant<Rhs>::value)
struct custom_operator<
71 op_value<LhsRep, LhsTag>,
73 [[nodiscard]] constexpr
auto operator()(LhsRep
const& lhs, Rhs
const& rhs)
const
75 return Operator{}(lhs, rhs);
80 #endif // CNL_IMPL_SCALED_BINARY_OPERATOR_H