CNL  2.0.2 (development)
Compositional Numeric Library
divide.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_DUPLEX_INTEGER_DIVIDE_H)
8 #define CNL_IMPL_DUPLEX_INTEGER_DIVIDE_H
9 
10 #include "../custom_operator/definition.h"
11 #include "../custom_operator/native_tag.h"
12 #include "../custom_operator/op.h"
13 #include "../numbers/set_signedness.h"
14 #include "../wide_integer/definition.h"
15 #include "ctors.h"
16 #include "definition.h"
17 #include "numbers.h"
18 #include "numeric_limits.h"
19 
20 #include <algorithm>
21 #include <limits>
22 
24 namespace cnl {
25  namespace _impl {
26  // cnl::_impl::heterogeneous_duplex_divide_operator
27  template<typename Lhs, typename Rhs>
28  struct heterogeneous_duplex_divide_operator {
29  using common_type = rep_of_t<wide_integer<
30  std::max(digits_v<Lhs>, digits_v<Rhs>),
31  numbers::set_signedness_t<int, numbers::signedness_v<Lhs> | numbers::signedness_v<Rhs>>>>;
32 
33  [[nodiscard]] constexpr auto operator()(Lhs const& lhs, Rhs const& rhs) const -> Lhs
34  {
35  return static_cast<Lhs>(
36  static_cast<common_type>(lhs) / static_cast<common_type>(rhs));
37  }
38  };
39  }
40 
41  // duplex_integer<> / duplex_integer<>
42  template<typename Upper, typename Lower>
43  struct custom_operator<
44  _impl::divide_op,
45  op_value<_impl::duplex_integer<Upper, Lower>>,
46  op_value<_impl::duplex_integer<Upper, Lower>>> {
47  private:
48  using duplex_integer = _impl::duplex_integer<Upper, Lower>;
49  using unsigned_duplex_integer = numbers::set_signedness_t<duplex_integer, false>;
50 
51  public:
52  [[nodiscard]] constexpr auto operator()(
53  duplex_integer const& lhs, duplex_integer const& rhs) const -> duplex_integer
54  {
55  return (lhs < duplex_integer{0}) ? (rhs < duplex_integer{0})
56  ? non_negative_division(-lhs, -rhs)
57  : -non_negative_division(-lhs, rhs)
58  : (rhs < duplex_integer{0}) ? -non_negative_division(lhs, -rhs)
59  : non_negative_division(lhs, rhs);
60  }
61 
62  // lifted from:
63  // https://github.com/torvalds/linux/blob/5ac94332248ee017964ba368cdda4ce647e3aba7/lib/math/div64.c#L142
64  static constexpr auto non_negative_division(
65  unsigned_duplex_integer const& dividend, unsigned_duplex_integer const& divisor)
66  -> unsigned_duplex_integer
67  {
68  auto const high = divisor.upper();
69  if (!high) {
70  return div_by_lower(dividend, divisor.lower());
71  }
72 
73  int n = fls(high);
74  auto quot = div_by_lower(dividend >> n, (divisor >> n).lower());
75 
76  if (quot) {
77  --quot;
78  }
79  if ((dividend - quot * divisor) >= divisor) {
80  ++quot;
81  }
82 
83  return quot;
84  }
85 
86  static constexpr auto fls(Upper n) -> int
87  {
88  auto half_digits = std::numeric_limits<duplex_integer>::digits / 2;
89 
90  if (!n) {
91  return 0;
92  }
93  auto const msd = Upper{1} << (half_digits - 1);
94  for (int r = half_digits;; n <<= 1, r--) {
95  if (n & msd) {
96  return r;
97  }
98  }
99  };
100 
101  // from Linux div64_32
102  static constexpr auto div_by_lower(
103  unsigned_duplex_integer const& dividend, Lower const& divisor)
104  -> unsigned_duplex_integer
105  {
106  unsigned_duplex_integer rem = dividend;
107  unsigned_duplex_integer b = divisor;
108  unsigned_duplex_integer d = 1;
109 
110  using unsigned_upper = numbers::set_signedness_t<Upper, false>;
111  auto high = rem.upper();
112 
113  unsigned_duplex_integer quot = 0;
114  if (static_cast<unsigned_upper>(high) >= divisor) {
115  high /= divisor; // NOLINT(clang-analyzer-core.DivideZero)
116  quot = unsigned_duplex_integer{high, 0};
117  rem -= unsigned_duplex_integer(high * divisor, 0);
118  }
119 
120  while (b < rem) {
121  b <<= 1;
122  d <<= 1;
123  }
124 
125  do {
126  if (rem >= b) {
127  rem -= b;
128  quot += d;
129  }
130  b >>= 1;
131  d >>= 1;
132  } while (d);
133 
134  return quot;
135  };
136  };
137 
138  template<typename LhsUpper, typename LhsLower, typename RhsUpper, typename RhsLower>
139  struct custom_operator<
140  _impl::divide_op,
141  op_value<_impl::duplex_integer<LhsUpper, LhsLower>>,
142  op_value<_impl::duplex_integer<RhsUpper, RhsLower>>>
143  : _impl::heterogeneous_duplex_divide_operator<
144  _impl::duplex_integer<LhsUpper, LhsLower>,
145  _impl::duplex_integer<RhsUpper, RhsLower>> {
146  };
147 
148  template<typename Lhs, typename RhsUpper, typename RhsLower>
149  struct custom_operator<
150  _impl::divide_op,
151  op_value<Lhs>,
152  op_value<_impl::duplex_integer<RhsUpper, RhsLower>>>
153  : _impl::heterogeneous_duplex_divide_operator<
154  Lhs, _impl::duplex_integer<RhsUpper, RhsLower>> {
155  };
156 
157  template<typename LhsUpper, typename LhsLower, typename Rhs>
158  struct custom_operator<
159  _impl::divide_op,
160  op_value<_impl::duplex_integer<LhsUpper, LhsLower>>,
161  op_value<Rhs>>
162  : _impl::heterogeneous_duplex_divide_operator<
163  _impl::duplex_integer<LhsUpper, LhsLower>, Rhs> {
164  };
165 }
166 
167 #endif // CNL_IMPL_DUPLEX_INTEGER_DIVIDE_H
numbers.h
cnl::wide_integer
_impl::wrapper< typename wide_tag< Digits, Narrowest >::rep, wide_tag< Digits, Narrowest > > wide_integer
An integer of limitless width.
Definition: definition.h:19
cnl
compositional numeric library
Definition: abort.h:15
std::max
T max(T... args)
std::numeric_limits