CNL  2.0.2 (development)
Compositional Numeric Library
neg_inf_rounding_tag.h
1 
2 // Copyright Heikki Berg 2020
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_ROUNDING_FLOOR_ROUNDING_TAG_H)
8 #define CNL_IMPL_ROUNDING_FLOOR_ROUNDING_TAG_H
9 
10 #include "../cmath/abs.h"
11 #include "../custom_operator/definition.h"
12 #include "../custom_operator/native_tag.h"
13 #include "is_rounding_tag.h"
14 #include "is_tag.h"
15 
16 #include <type_traits>
17 
19 namespace cnl {
29  : _impl::homogeneous_deduction_tag_base
30  , _impl::homogeneous_operator_tag_base {
31  };
32 
33  namespace _impl {
34  template<>
35  struct is_rounding_tag<neg_inf_rounding_tag> : std::true_type {
36  };
37  }
38 
39  template<_impl::unary_arithmetic_op Operator, typename Operand>
40  struct custom_operator<Operator, op_value<Operand, neg_inf_rounding_tag>>
41  : custom_operator<Operator, op_value<Operand, _impl::native_tag>> {
42  };
43 
44  template<_impl::binary_arithmetic_op Operator, typename Lhs, typename Rhs>
45  struct custom_operator<Operator, op_value<Lhs, neg_inf_rounding_tag>, op_value<Rhs, neg_inf_rounding_tag>>
46  : Operator {
47  };
48 
49  template<typename Lhs, typename Rhs>
50  struct custom_operator<_impl::divide_op, op_value<Lhs, neg_inf_rounding_tag>, op_value<Rhs, neg_inf_rounding_tag>> {
51  private:
52  using result_type = decltype(std::declval<Lhs>() / std::declval<Rhs>());
53  [[nodiscard]] constexpr auto remainder(Lhs const& lhs, Rhs const& rhs) const -> result_type
54  {
55  return lhs % rhs;
56  }
57  [[nodiscard]] constexpr auto division(Lhs const& lhs, Rhs const& rhs) const -> result_type
58  {
59  return lhs / rhs;
60  }
61  [[nodiscard]] constexpr auto division_neg(Lhs const& lhs, Rhs const& rhs) const
62  -> result_type
63  {
64  return lhs / rhs - 1;
65  }
66 
67  public:
68  [[nodiscard]] constexpr auto operator()(Lhs const& lhs, Rhs const& rhs) const -> result_type
69  {
70  return ((remainder(lhs, rhs) != 0) && ((remainder(lhs, rhs) < 0) != (rhs < 0)))
71  ? division_neg(lhs, rhs)
72  : division(lhs, rhs);
73  }
74  };
75 
76  template<_impl::shift_op Operator, tag RhsTag, typename Lhs, typename Rhs>
77  struct custom_operator<Operator, op_value<Lhs, neg_inf_rounding_tag>, op_value<Rhs, RhsTag>> : Operator {
78  };
79 
80  template<_impl::prefix_op Operator, typename Rhs>
81  struct custom_operator<Operator, op_value<Rhs, neg_inf_rounding_tag>> : Operator {
82  };
83 
84  template<_impl::postfix_op Operator, typename Lhs>
85  struct custom_operator<Operator, op_value<Lhs, neg_inf_rounding_tag>> : Operator {
86  };
87 }
88 
89 #endif // CNL_IMPL_ROUNDING_FLOOR_ROUNDING_TAG_H
std::true_type
std::remainder
T remainder(T... args)
cnl::neg_inf_rounding_tag
tag to specify floor or round towards minus infinity rounding behavior in arithmetic operations
Definition: neg_inf_rounding_tag.h:28
cnl
compositional numeric library
Definition: abort.h:15