misc.hpp 26 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609
  1. ///////////////////////////////////////////////////////////////
  2. // Copyright 2012 John Maddock. Distributed under the Boost
  3. // Software License, Version 1.0. (See accompanying file
  4. // LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_
  5. //
  6. // Comparison operators for cpp_int_backend:
  7. //
  8. #ifndef BOOST_MP_CPP_INT_MISC_HPP
  9. #define BOOST_MP_CPP_INT_MISC_HPP
  10. #include <boost/multiprecision/detail/bitscan.hpp> // lsb etc
  11. #include <boost/integer/common_factor_rt.hpp> // gcd/lcm
  12. #ifdef BOOST_MSVC
  13. #pragma warning(push)
  14. #pragma warning(disable:4702)
  15. #endif
  16. namespace boost{ namespace multiprecision{ namespace backends{
  17. template <class R, class CppInt>
  18. void check_in_range(const CppInt& val, const mpl::int_<checked>&)
  19. {
  20. typedef typename boost::multiprecision::detail::canonical<R, CppInt>::type cast_type;
  21. if(val.sign())
  22. {
  23. if(val.compare(static_cast<cast_type>((std::numeric_limits<R>::min)())) < 0)
  24. BOOST_THROW_EXCEPTION(std::overflow_error("Could not convert to the target type - -value is out of range."));
  25. }
  26. else
  27. {
  28. if(val.compare(static_cast<cast_type>((std::numeric_limits<R>::max)())) > 0)
  29. BOOST_THROW_EXCEPTION(std::overflow_error("Could not convert to the target type - -value is out of range."));
  30. }
  31. }
  32. template <class R, class CppInt>
  33. inline void check_in_range(const CppInt& /*val*/, const mpl::int_<unchecked>&) BOOST_NOEXCEPT {}
  34. inline void check_is_negative(const mpl::true_&) BOOST_NOEXCEPT {}
  35. inline void check_is_negative(const mpl::false_&)
  36. {
  37. BOOST_THROW_EXCEPTION(std::range_error("Attempt to assign a negative value to an unsigned type."));
  38. }
  39. template <class Integer>
  40. inline Integer negate_integer(Integer i, const mpl::true_&) BOOST_NOEXCEPT
  41. {
  42. return -i;
  43. }
  44. template <class Integer>
  45. inline Integer negate_integer(Integer i, const mpl::false_&) BOOST_NOEXCEPT
  46. {
  47. return ~(i-1);
  48. }
  49. template <class R, unsigned MinBits1, unsigned MaxBits1, cpp_integer_type SignType1, cpp_int_check_type Checked1, class Allocator1>
  50. inline typename enable_if_c<is_integral<R>::value && !is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value, void>::type
  51. eval_convert_to(R* result, const cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& backend) BOOST_MP_NOEXCEPT_IF((is_non_throwing_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value))
  52. {
  53. typedef mpl::int_<Checked1> checked_type;
  54. check_in_range<R>(backend, checked_type());
  55. *result = static_cast<R>(backend.limbs()[0]);
  56. unsigned shift = cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>::limb_bits;
  57. for(unsigned i = 1; (i < backend.size()) && (shift < static_cast<unsigned>(std::numeric_limits<R>::digits)); ++i)
  58. {
  59. *result += static_cast<R>(backend.limbs()[i]) << shift;
  60. shift += cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>::limb_bits;
  61. }
  62. if(backend.sign())
  63. {
  64. check_is_negative(boost::is_signed<R>());
  65. *result = negate_integer(*result, boost::is_signed<R>());
  66. }
  67. }
  68. template <class R, unsigned MinBits1, unsigned MaxBits1, cpp_integer_type SignType1, cpp_int_check_type Checked1, class Allocator1>
  69. inline typename enable_if_c<is_floating_point<R>::value && !is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value, void>::type
  70. eval_convert_to(R* result, const cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& backend) BOOST_MP_NOEXCEPT_IF(is_arithmetic<R>::value)
  71. {
  72. typename cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>::const_limb_pointer p = backend.limbs();
  73. unsigned shift = cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>::limb_bits;
  74. *result = static_cast<R>(*p);
  75. for(unsigned i = 1; i < backend.size(); ++i)
  76. {
  77. *result += static_cast<R>(std::ldexp(static_cast<long double>(p[i]), shift));
  78. shift += cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>::limb_bits;
  79. }
  80. if(backend.sign())
  81. *result = -*result;
  82. }
  83. template <unsigned MinBits1, unsigned MaxBits1, cpp_integer_type SignType1, cpp_int_check_type Checked1, class Allocator1>
  84. BOOST_MP_FORCEINLINE typename enable_if_c<!is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value, bool>::type
  85. eval_is_zero(const cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& val) BOOST_NOEXCEPT
  86. {
  87. return (val.size() == 1) && (val.limbs()[0] == 0);
  88. }
  89. template <unsigned MinBits1, unsigned MaxBits1, cpp_integer_type SignType1, cpp_int_check_type Checked1, class Allocator1>
  90. BOOST_MP_FORCEINLINE typename enable_if_c<!is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value, int>::type
  91. eval_get_sign(const cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& val) BOOST_NOEXCEPT
  92. {
  93. return eval_is_zero(val) ? 0 : val.sign() ? -1 : 1;
  94. }
  95. template <unsigned MinBits1, unsigned MaxBits1, cpp_integer_type SignType1, cpp_int_check_type Checked1, class Allocator1>
  96. BOOST_MP_FORCEINLINE typename enable_if_c<!is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value>::type
  97. eval_abs(cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& result, const cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& val) BOOST_MP_NOEXCEPT_IF((is_non_throwing_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value))
  98. {
  99. result = val;
  100. result.sign(false);
  101. }
  102. //
  103. // Get the location of the least-significant-bit:
  104. //
  105. template <unsigned MinBits1, unsigned MaxBits1, cpp_integer_type SignType1, cpp_int_check_type Checked1, class Allocator1>
  106. inline typename enable_if_c<!is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value, unsigned>::type
  107. eval_lsb(const cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& a)
  108. {
  109. using default_ops::eval_get_sign;
  110. if(eval_get_sign(a) == 0)
  111. {
  112. BOOST_THROW_EXCEPTION(std::range_error("No bits were set in the operand."));
  113. }
  114. if(a.sign())
  115. {
  116. BOOST_THROW_EXCEPTION(std::range_error("Testing individual bits in negative values is not supported - results are undefined."));
  117. }
  118. //
  119. // Find the index of the least significant limb that is non-zero:
  120. //
  121. unsigned index = 0;
  122. while(!a.limbs()[index] && (index < a.size()))
  123. ++index;
  124. //
  125. // Find the index of the least significant bit within that limb:
  126. //
  127. unsigned result = boost::multiprecision::detail::find_lsb(a.limbs()[index]);
  128. return result + index * cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>::limb_bits;
  129. }
  130. //
  131. // Get the location of the most-significant-bit:
  132. //
  133. template <unsigned MinBits1, unsigned MaxBits1, cpp_integer_type SignType1, cpp_int_check_type Checked1, class Allocator1>
  134. inline typename enable_if_c<!is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value, unsigned>::type
  135. eval_msb(const cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& a)
  136. {
  137. using default_ops::eval_get_sign;
  138. if(eval_get_sign(a) == 0)
  139. {
  140. BOOST_THROW_EXCEPTION(std::range_error("No bits were set in the operand."));
  141. }
  142. if(a.sign())
  143. {
  144. BOOST_THROW_EXCEPTION(std::range_error("Testing individual bits in negative values is not supported - results are undefined."));
  145. }
  146. //
  147. // Find the index of the most significant bit that is non-zero:
  148. //
  149. return (a.size() - 1) * cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>::limb_bits + boost::multiprecision::detail::find_msb(a.limbs()[a.size() - 1]);
  150. }
  151. template <unsigned MinBits1, unsigned MaxBits1, cpp_integer_type SignType1, cpp_int_check_type Checked1, class Allocator1>
  152. inline typename enable_if_c<!is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value, bool>::type
  153. eval_bit_test(const cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& val, unsigned index) BOOST_NOEXCEPT
  154. {
  155. unsigned offset = index / cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>::limb_bits;
  156. unsigned shift = index % cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>::limb_bits;
  157. limb_type mask = shift ? limb_type(1u) << shift : limb_type(1u);
  158. if(offset >= val.size())
  159. return false;
  160. return val.limbs()[offset] & mask ? true : false;
  161. }
  162. template <unsigned MinBits1, unsigned MaxBits1, cpp_integer_type SignType1, cpp_int_check_type Checked1, class Allocator1>
  163. inline typename enable_if_c<!is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value>::type
  164. eval_bit_set(cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& val, unsigned index)
  165. {
  166. unsigned offset = index / cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>::limb_bits;
  167. unsigned shift = index % cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>::limb_bits;
  168. limb_type mask = shift ? limb_type(1u) << shift : limb_type(1u);
  169. if(offset >= val.size())
  170. {
  171. unsigned os = val.size();
  172. val.resize(offset + 1, offset + 1);
  173. if(offset >= val.size())
  174. return; // fixed precision overflow
  175. for(unsigned i = os; i <= offset; ++i)
  176. val.limbs()[i] = 0;
  177. }
  178. val.limbs()[offset] |= mask;
  179. }
  180. template <unsigned MinBits1, unsigned MaxBits1, cpp_integer_type SignType1, cpp_int_check_type Checked1, class Allocator1>
  181. inline typename enable_if_c<!is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value>::type
  182. eval_bit_unset(cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& val, unsigned index) BOOST_NOEXCEPT
  183. {
  184. unsigned offset = index / cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>::limb_bits;
  185. unsigned shift = index % cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>::limb_bits;
  186. limb_type mask = shift ? limb_type(1u) << shift : limb_type(1u);
  187. if(offset >= val.size())
  188. return;
  189. val.limbs()[offset] &= ~mask;
  190. val.normalize();
  191. }
  192. template <unsigned MinBits1, unsigned MaxBits1, cpp_integer_type SignType1, cpp_int_check_type Checked1, class Allocator1>
  193. inline typename enable_if_c<!is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value>::type
  194. eval_bit_flip(cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& val, unsigned index)
  195. {
  196. unsigned offset = index / cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>::limb_bits;
  197. unsigned shift = index % cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>::limb_bits;
  198. limb_type mask = shift ? limb_type(1u) << shift : limb_type(1u);
  199. if(offset >= val.size())
  200. {
  201. unsigned os = val.size();
  202. val.resize(offset + 1, offset + 1);
  203. if(offset >= val.size())
  204. return; // fixed precision overflow
  205. for(unsigned i = os; i <= offset; ++i)
  206. val.limbs()[i] = 0;
  207. }
  208. val.limbs()[offset] ^= mask;
  209. val.normalize();
  210. }
  211. template <unsigned MinBits1, unsigned MaxBits1, cpp_integer_type SignType1, cpp_int_check_type Checked1, class Allocator1>
  212. inline typename enable_if_c<!is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value>::type
  213. eval_qr(
  214. const cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& x,
  215. const cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& y,
  216. cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& q,
  217. cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& r) BOOST_MP_NOEXCEPT_IF((is_non_throwing_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value))
  218. {
  219. divide_unsigned_helper(&q, x, y, r);
  220. q.sign(x.sign() != y.sign());
  221. r.sign(x.sign());
  222. }
  223. template <unsigned MinBits1, unsigned MaxBits1, cpp_integer_type SignType1, cpp_int_check_type Checked1, class Allocator1>
  224. inline typename enable_if_c<!is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value>::type
  225. eval_qr(
  226. const cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& x,
  227. limb_type y,
  228. cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& q,
  229. cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& r) BOOST_MP_NOEXCEPT_IF((is_non_throwing_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value))
  230. {
  231. divide_unsigned_helper(&q, x, y, r);
  232. q.sign(x.sign());
  233. r.sign(x.sign());
  234. }
  235. template <unsigned MinBits1, unsigned MaxBits1, cpp_integer_type SignType1, cpp_int_check_type Checked1, class Allocator1, class U>
  236. inline typename enable_if_c<is_integral<U>::value>::type eval_qr(
  237. const cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& x,
  238. U y,
  239. cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& q,
  240. cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& r) BOOST_MP_NOEXCEPT_IF((is_non_throwing_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value))
  241. {
  242. using default_ops::eval_qr;
  243. cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> t(y);
  244. eval_qr(x, t, q, r);
  245. }
  246. template <unsigned MinBits1, unsigned MaxBits1, cpp_integer_type SignType1, cpp_int_check_type Checked1, class Allocator1, class Integer>
  247. inline typename enable_if_c<is_unsigned<Integer>::value && !is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value, Integer>::type
  248. eval_integer_modulus(const cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& x, Integer val)
  249. {
  250. if((sizeof(Integer) <= sizeof(limb_type)) || (val <= (std::numeric_limits<limb_type>::max)()))
  251. {
  252. cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> d;
  253. divide_unsigned_helper(static_cast<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>*>(0), x, static_cast<limb_type>(val), d);
  254. return d.limbs()[0];
  255. }
  256. else
  257. {
  258. return default_ops::eval_integer_modulus(x, val);
  259. }
  260. }
  261. template <unsigned MinBits1, unsigned MaxBits1, cpp_integer_type SignType1, cpp_int_check_type Checked1, class Allocator1, class Integer>
  262. BOOST_MP_FORCEINLINE typename enable_if_c<is_signed<Integer>::value && !is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value, Integer>::type
  263. eval_integer_modulus(const cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& x, Integer val)
  264. {
  265. return eval_integer_modulus(x, boost::multiprecision::detail::unsigned_abs(val));
  266. }
  267. inline limb_type integer_gcd_reduce(limb_type u, limb_type v)
  268. {
  269. do
  270. {
  271. if(u > v)
  272. std::swap(u, v);
  273. if(u == v)
  274. break;
  275. v -= u;
  276. v >>= boost::multiprecision::detail::find_lsb(v);
  277. } while(true);
  278. return u;
  279. }
  280. inline double_limb_type integer_gcd_reduce(double_limb_type u, double_limb_type v)
  281. {
  282. do
  283. {
  284. if(u > v)
  285. std::swap(u, v);
  286. if(u == v)
  287. break;
  288. if(v <= ~static_cast<limb_type>(0))
  289. {
  290. u = integer_gcd_reduce(static_cast<limb_type>(v), static_cast<limb_type>(u));
  291. break;
  292. }
  293. v -= u;
  294. while((static_cast<unsigned>(v) & 1u) == 0)
  295. v >>= 1;
  296. } while(true);
  297. return u;
  298. }
  299. template <unsigned MinBits1, unsigned MaxBits1, cpp_integer_type SignType1, cpp_int_check_type Checked1, class Allocator1>
  300. inline typename enable_if_c<!is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value>::type
  301. eval_gcd(
  302. cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& result,
  303. const cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& a,
  304. limb_type v)
  305. {
  306. using default_ops::eval_lsb;
  307. using default_ops::eval_is_zero;
  308. using default_ops::eval_get_sign;
  309. int shift;
  310. cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> u(a);
  311. int s = eval_get_sign(u);
  312. /* GCD(0,x) := x */
  313. if(s < 0)
  314. {
  315. u.negate();
  316. }
  317. else if(s == 0)
  318. {
  319. result = v;
  320. return;
  321. }
  322. if(v == 0)
  323. {
  324. result = u;
  325. return;
  326. }
  327. /* Let shift := lg K, where K is the greatest power of 2
  328. dividing both u and v. */
  329. unsigned us = eval_lsb(u);
  330. unsigned vs = boost::multiprecision::detail::find_lsb(v);
  331. shift = (std::min)(us, vs);
  332. eval_right_shift(u, us);
  333. if(vs)
  334. v >>= vs;
  335. do
  336. {
  337. /* Now u and v are both odd, so diff(u, v) is even.
  338. Let u = min(u, v), v = diff(u, v)/2. */
  339. if(u.size() <= 2)
  340. {
  341. if(u.size() == 1)
  342. v = integer_gcd_reduce(*u.limbs(), v);
  343. else
  344. {
  345. double_limb_type i;
  346. i = u.limbs()[0] | (static_cast<double_limb_type>(u.limbs()[1]) << sizeof(limb_type) * CHAR_BIT);
  347. v = static_cast<limb_type>(integer_gcd_reduce(i, static_cast<double_limb_type>(v)));
  348. }
  349. break;
  350. }
  351. eval_subtract(u, v);
  352. us = eval_lsb(u);
  353. eval_right_shift(u, us);
  354. }
  355. while(true);
  356. result = v;
  357. eval_left_shift(result, shift);
  358. }
  359. template <unsigned MinBits1, unsigned MaxBits1, cpp_integer_type SignType1, cpp_int_check_type Checked1, class Allocator1, class Integer>
  360. inline typename enable_if_c<is_unsigned<Integer>::value && (sizeof(Integer) <= sizeof(limb_type)) && !is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value>::type
  361. eval_gcd(
  362. cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& result,
  363. const cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& a,
  364. const Integer& v)
  365. {
  366. eval_gcd(result, a, static_cast<limb_type>(v));
  367. }
  368. template <unsigned MinBits1, unsigned MaxBits1, cpp_integer_type SignType1, cpp_int_check_type Checked1, class Allocator1, class Integer>
  369. inline typename enable_if_c<is_signed<Integer>::value && (sizeof(Integer) <= sizeof(limb_type)) && !is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value>::type
  370. eval_gcd(
  371. cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& result,
  372. const cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& a,
  373. const Integer& v)
  374. {
  375. eval_gcd(result, a, static_cast<limb_type>(v < 0 ? -v : v));
  376. }
  377. template <unsigned MinBits1, unsigned MaxBits1, cpp_integer_type SignType1, cpp_int_check_type Checked1, class Allocator1>
  378. inline typename enable_if_c<!is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value>::type
  379. eval_gcd(
  380. cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& result,
  381. const cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& a,
  382. const cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& b)
  383. {
  384. using default_ops::eval_lsb;
  385. using default_ops::eval_is_zero;
  386. using default_ops::eval_get_sign;
  387. if(a.size() == 1)
  388. {
  389. eval_gcd(result, b, *a.limbs());
  390. return;
  391. }
  392. if(b.size() == 1)
  393. {
  394. eval_gcd(result, a, *b.limbs());
  395. return;
  396. }
  397. int shift;
  398. cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> u(a), v(b);
  399. int s = eval_get_sign(u);
  400. /* GCD(0,x) := x */
  401. if(s < 0)
  402. {
  403. u.negate();
  404. }
  405. else if(s == 0)
  406. {
  407. result = v;
  408. return;
  409. }
  410. s = eval_get_sign(v);
  411. if(s < 0)
  412. {
  413. v.negate();
  414. }
  415. else if(s == 0)
  416. {
  417. result = u;
  418. return;
  419. }
  420. /* Let shift := lg K, where K is the greatest power of 2
  421. dividing both u and v. */
  422. unsigned us = eval_lsb(u);
  423. unsigned vs = eval_lsb(v);
  424. shift = (std::min)(us, vs);
  425. eval_right_shift(u, us);
  426. eval_right_shift(v, vs);
  427. do
  428. {
  429. /* Now u and v are both odd, so diff(u, v) is even.
  430. Let u = min(u, v), v = diff(u, v)/2. */
  431. s = u.compare(v);
  432. if(s > 0)
  433. u.swap(v);
  434. if(s == 0)
  435. break;
  436. if(v.size() <= 2)
  437. {
  438. if(v.size() == 1)
  439. u = integer_gcd_reduce(*v.limbs(), *u.limbs());
  440. else
  441. {
  442. double_limb_type i, j;
  443. i = v.limbs()[0] | (static_cast<double_limb_type>(v.limbs()[1]) << sizeof(limb_type) * CHAR_BIT);
  444. j = (u.size() == 1) ? *u.limbs() : u.limbs()[0] | (static_cast<double_limb_type>(u.limbs()[1]) << sizeof(limb_type) * CHAR_BIT);
  445. u = integer_gcd_reduce(i, j);
  446. }
  447. break;
  448. }
  449. eval_subtract(v, u);
  450. vs = eval_lsb(v);
  451. eval_right_shift(v, vs);
  452. }
  453. while(true);
  454. result = u;
  455. eval_left_shift(result, shift);
  456. }
  457. //
  458. // Now again for trivial backends:
  459. //
  460. template <unsigned MinBits1, unsigned MaxBits1, cpp_integer_type SignType1, cpp_int_check_type Checked1, class Allocator1>
  461. BOOST_MP_FORCEINLINE typename enable_if_c<is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value>::type
  462. eval_gcd(cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& result, const cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& a, const cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& b) BOOST_NOEXCEPT
  463. {
  464. *result.limbs() = boost::integer::gcd(*a.limbs(), *b.limbs());
  465. }
  466. // This one is only enabled for unchecked cpp_int's, for checked int's we need the checking in the default version:
  467. template <unsigned MinBits1, unsigned MaxBits1, cpp_integer_type SignType1, cpp_int_check_type Checked1, class Allocator1>
  468. BOOST_MP_FORCEINLINE typename enable_if_c<is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value && (Checked1 == unchecked)>::type
  469. eval_lcm(cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& result, const cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& a, const cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& b) BOOST_MP_NOEXCEPT_IF((is_non_throwing_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value))
  470. {
  471. *result.limbs() = boost::integer::lcm(*a.limbs(), *b.limbs());
  472. result.normalize(); // result may overflow the specified number of bits
  473. }
  474. inline void conversion_overflow(const mpl::int_<checked>&)
  475. {
  476. BOOST_THROW_EXCEPTION(std::overflow_error("Overflow in conversion to narrower type"));
  477. }
  478. inline void conversion_overflow(const mpl::int_<unchecked>&){}
  479. template <class R, unsigned MinBits1, unsigned MaxBits1, cpp_integer_type SignType1, cpp_int_check_type Checked1, class Allocator1>
  480. inline typename enable_if_c<
  481. is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value
  482. && is_signed_number<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value
  483. >::type
  484. eval_convert_to(R* result, const cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& val)
  485. {
  486. typedef typename common_type<R, typename cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>::local_limb_type>::type common_type;
  487. if(std::numeric_limits<R>::is_specialized && (static_cast<common_type>(*val.limbs()) > static_cast<common_type>((std::numeric_limits<R>::max)())))
  488. {
  489. conversion_overflow(typename cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>::checked_type());
  490. *result = (std::numeric_limits<R>::max)();
  491. }
  492. else
  493. *result = static_cast<R>(*val.limbs());
  494. if(val.isneg())
  495. {
  496. check_is_negative(mpl::bool_<boost::is_signed<R>::value || boost::is_floating_point<R>::value>());
  497. *result = negate_integer(*result, mpl::bool_<boost::is_signed<R>::value || boost::is_floating_point<R>::value>());
  498. }
  499. }
  500. template <class R, unsigned MinBits1, unsigned MaxBits1, cpp_integer_type SignType1, cpp_int_check_type Checked1, class Allocator1>
  501. inline typename enable_if_c<
  502. is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value
  503. && is_unsigned_number<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value
  504. >::type
  505. eval_convert_to(R* result, const cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& val)
  506. {
  507. typedef typename common_type<R, typename cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>::local_limb_type>::type common_type;
  508. if(std::numeric_limits<R>::is_specialized && (static_cast<common_type>(*val.limbs()) > static_cast<common_type>((std::numeric_limits<R>::max)())))
  509. {
  510. conversion_overflow(typename cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>::checked_type());
  511. *result = (std::numeric_limits<R>::max)();
  512. }
  513. else
  514. *result = static_cast<R>(*val.limbs());
  515. }
  516. template <unsigned MinBits1, unsigned MaxBits1, cpp_integer_type SignType1, cpp_int_check_type Checked1, class Allocator1>
  517. inline typename enable_if_c<is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value, unsigned>::type
  518. eval_lsb(const cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& a)
  519. {
  520. using default_ops::eval_get_sign;
  521. if(eval_get_sign(a) == 0)
  522. {
  523. BOOST_THROW_EXCEPTION(std::range_error("No bits were set in the operand."));
  524. }
  525. if(a.sign())
  526. {
  527. BOOST_THROW_EXCEPTION(std::range_error("Testing individual bits in negative values is not supported - results are undefined."));
  528. }
  529. //
  530. // Find the index of the least significant bit within that limb:
  531. //
  532. return boost::multiprecision::detail::find_lsb(*a.limbs());
  533. }
  534. template <unsigned MinBits1, unsigned MaxBits1, cpp_integer_type SignType1, cpp_int_check_type Checked1, class Allocator1>
  535. inline typename enable_if_c<is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value, unsigned>::type
  536. eval_msb(const cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& a)
  537. {
  538. using default_ops::eval_get_sign;
  539. if(eval_get_sign(a) == 0)
  540. {
  541. BOOST_THROW_EXCEPTION(std::range_error("No bits were set in the operand."));
  542. }
  543. if(a.sign())
  544. {
  545. BOOST_THROW_EXCEPTION(std::range_error("Testing individual bits in negative values is not supported - results are undefined."));
  546. }
  547. //
  548. // Find the index of the least significant bit within that limb:
  549. //
  550. return boost::multiprecision::detail::find_msb(*a.limbs());
  551. }
  552. #ifdef BOOST_MSVC
  553. #pragma warning(pop)
  554. #endif
  555. }}} // namespaces
  556. #endif