CNL  2.0.2 (development)
Compositional Numeric Library
builtin_overflow.h
1 
2 // Copyright John McFarlane 2019.
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_OVERFLOW_BUILTIN_OVERFLOW_H)
8 #define CNL_IMPL_OVERFLOW_BUILTIN_OVERFLOW_H
9 
10 #include "../config.h"
11 #include "../custom_operator/op.h"
12 #include "../polarity.h"
13 #include "overflow_operator.h"
14 
15 #include <concepts>
16 
18 namespace cnl {
19  namespace _impl {
21  // cnl::_impl::overflow_polarity
22 
23  template<op Operator>
24  struct overflow_polarity;
25 
26  template<>
27  struct overflow_polarity<convert_op> {
28  template<typename Destination, typename Source>
29  [[nodiscard]] constexpr auto operator()(Source const& from) const
30  {
31  return measure_polarity(from);
32  }
33  };
34 
35  template<>
36  struct overflow_polarity<minus_op> {
37  template<typename Rhs>
38  [[nodiscard]] constexpr auto operator()(Rhs const& rhs) const
39  {
40  return -measure_polarity(rhs);
41  }
42  };
43 
44  template<>
45  struct overflow_polarity<add_op> {
46  template<typename Lhs, typename Rhs>
47  [[nodiscard]] constexpr auto operator()(Lhs const&, Rhs const& rhs) const
48  {
49  return measure_polarity(rhs);
50  }
51  };
52 
53  template<>
54  struct overflow_polarity<subtract_op> {
55  template<typename Lhs, typename Rhs>
56  [[nodiscard]] constexpr auto operator()(Lhs const&, Rhs const& rhs) const
57  {
58  return -measure_polarity(rhs);
59  }
60  };
61 
62  template<>
63  struct overflow_polarity<multiply_op> {
64  template<typename Lhs, typename Rhs>
65  [[nodiscard]] constexpr auto operator()(Lhs const& lhs, Rhs const& rhs) const
66  {
67  return measure_polarity(lhs) * measure_polarity(rhs);
68  }
69  };
70 
72  // cnl::_impl::are_builtin_operands
73 
74  template<typename Lhs, typename Rhs>
75  struct are_builtin_operands
77  bool, std::is_integral_v<Lhs> && std::is_integral_v<Rhs>> {
78  };
79 
81  // cnl::_impl::builtin_overflow_operator
82 
83  template<binary_arithmetic_op Operator, typename Lhs, typename Rhs>
84  struct builtin_overflow_operator : std::false_type {
85  };
86 
87 #if defined(CNL_BUILTIN_OVERFLOW_ENABLED)
88  template<typename Lhs, typename Rhs>
89  struct builtin_overflow_operator<add_op, Lhs, Rhs> : are_builtin_operands<Lhs, Rhs> {
90  template<typename Result>
91  [[nodiscard]] constexpr auto operator()(
92  Lhs const& lhs, Rhs const& rhs, Result& result) const
93  {
94  return __builtin_add_overflow(lhs, rhs, &result);
95  }
96  };
97 
98  template<typename Lhs, typename Rhs>
99  struct builtin_overflow_operator<subtract_op, Lhs, Rhs> : are_builtin_operands<Lhs, Rhs> {
100  template<typename Result>
101  [[nodiscard]] constexpr auto operator()(
102  Lhs const& lhs, Rhs const& rhs, Result& result) const
103  {
104  return __builtin_sub_overflow(lhs, rhs, &result);
105  }
106  };
107 
108  template<typename Lhs, typename Rhs>
109  struct builtin_overflow_operator<multiply_op, Lhs, Rhs> : are_builtin_operands<Lhs, Rhs> {
110  template<typename Result>
111  [[nodiscard]] constexpr auto operator()(
112  Lhs const& lhs, Rhs const& rhs, Result& result) const
113  {
114  return __builtin_mul_overflow(lhs, rhs, &result);
115  }
116  };
117 #endif
118  }
119 }
120 
121 #endif // CNL_IMPL_OVERFLOW_BUILTIN_OVERFLOW_H
std::integral_constant
cnl
compositional numeric library
Definition: abort.h:15