CNL  2.0.2 (development)
Compositional Numeric Library
nearest_rounding_tag.h
1 
2 // Copyright John McFarlane 2018.
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_NEAREST_ROUNDING_TAG_H)
8 #define CNL_IMPL_ROUNDING_NEAREST_ROUNDING_TAG_H
9 
10 #include "../custom_operator/definition.h"
11 #include "../custom_operator/native_tag.h"
12 #include "is_rounding_tag.h"
13 #include "is_tag.h"
14 
15 #include <type_traits>
16 
18 namespace cnl {
27  : _impl::homogeneous_deduction_tag_base
28  , _impl::homogeneous_operator_tag_base {
29  };
30 
31  namespace _impl {
32  template<>
33  struct is_rounding_tag<nearest_rounding_tag> : std::true_type {
34  };
35  }
36 
37  template<_impl::unary_arithmetic_op Operator, typename Operand>
38  struct custom_operator<Operator, op_value<Operand, nearest_rounding_tag>>
39  : custom_operator<Operator, op_value<Operand, _impl::native_tag>> {
40  };
41 
42  template<_impl::binary_arithmetic_op Operator, typename Lhs, typename Rhs>
43  struct custom_operator<Operator, op_value<Lhs, nearest_rounding_tag>, op_value<Rhs, nearest_rounding_tag>>
44  : Operator {
45  };
46 
47  template<typename Lhs, typename Rhs>
48  struct custom_operator<_impl::divide_op, op_value<Lhs, nearest_rounding_tag>, op_value<Rhs, nearest_rounding_tag>> {
49  [[nodiscard]] constexpr auto operator()(Lhs const& lhs, Rhs const& rhs) const
50  -> decltype(lhs / rhs)
51  {
52  return (((lhs < 0) ^ (rhs < 0)) ? lhs - (rhs / 2) : lhs + (rhs / 2)) / rhs;
53  }
54  };
55 
56  template<_impl::shift_op Operator, typename Lhs, typename Rhs, tag RhsTag>
57  struct custom_operator<Operator, op_value<Lhs, nearest_rounding_tag>, op_value<Rhs, RhsTag>> : Operator {
58  };
59 
60  template<_impl::prefix_op Operator, typename Rhs>
61  struct custom_operator<Operator, op_value<Rhs, nearest_rounding_tag>> : Operator {
62  };
63 
64  template<_impl::postfix_op Operator, typename Lhs>
65  struct custom_operator<Operator, op_value<Lhs, nearest_rounding_tag>> : Operator {
66  };
67 }
68 
69 #endif // CNL_IMPL_ROUNDING_NEAREST_ROUNDING_TAG_H
std::true_type
cnl
compositional numeric library
Definition: abort.h:15
cnl::nearest_rounding_tag
tag to specify round-to-nearest behavior in arithemtic operations
Definition: nearest_rounding_tag.h:26