multiply.hpp 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439
  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_MUL_HPP
  9. #define BOOST_MP_CPP_INT_MUL_HPP
  10. namespace boost{ namespace multiprecision{ namespace backends{
  11. template <unsigned MinBits1, unsigned MaxBits1, cpp_integer_type SignType1, cpp_int_check_type Checked1, class Allocator1, unsigned MinBits2, unsigned MaxBits2, cpp_integer_type SignType2, cpp_int_check_type Checked2, class Allocator2>
  12. inline typename enable_if_c<!is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value && !is_trivial_cpp_int<cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2> >::value >::type
  13. eval_multiply(
  14. cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& result,
  15. const cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2>& a,
  16. const limb_type& val) BOOST_MP_NOEXCEPT_IF((is_non_throwing_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value))
  17. {
  18. if(!val)
  19. {
  20. result = static_cast<limb_type>(0);
  21. return;
  22. }
  23. if((void*)&a != (void*)&result)
  24. result.resize(a.size(), a.size());
  25. double_limb_type carry = 0;
  26. typename cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>::limb_pointer p = result.limbs();
  27. typename cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>::limb_pointer pe = result.limbs() + result.size();
  28. typename cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2>::const_limb_pointer pa = a.limbs();
  29. while(p != pe)
  30. {
  31. carry += static_cast<double_limb_type>(*pa) * static_cast<double_limb_type>(val);
  32. *p = static_cast<limb_type>(carry);
  33. carry >>= cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>::limb_bits;
  34. ++p, ++pa;
  35. }
  36. if(carry)
  37. {
  38. unsigned i = result.size();
  39. result.resize(i + 1, i + 1);
  40. if(cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>::variable || (result.size() > i))
  41. result.limbs()[i] = static_cast<limb_type>(carry);
  42. }
  43. result.sign(a.sign());
  44. if(!cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>::variable)
  45. result.normalize();
  46. }
  47. //
  48. // resize_for_carry forces a resize of the underlying buffer only if a previous request
  49. // for "required" elements could possibly have failed, *and* we have checking enabled.
  50. // This will cause an overflow error inside resize():
  51. //
  52. template <unsigned MinBits1, unsigned MaxBits1, cpp_integer_type SignType1, cpp_int_check_type Checked1, class Allocator1>
  53. inline void resize_for_carry(cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& /*result*/, unsigned /*required*/){}
  54. template <unsigned MinBits1, unsigned MaxBits1, cpp_integer_type SignType1, cpp_int_check_type Checked1, class Allocator1>
  55. inline void resize_for_carry(cpp_int_backend<MinBits1, MaxBits1, SignType1, checked, void>& result, unsigned required)
  56. {
  57. if(result.size() != required)
  58. result.resize(required, required);
  59. }
  60. template <unsigned MinBits1, unsigned MaxBits1, cpp_integer_type SignType1, cpp_int_check_type Checked1, class Allocator1, unsigned MinBits2, unsigned MaxBits2, cpp_integer_type SignType2, cpp_int_check_type Checked2, class Allocator2, unsigned MinBits3, unsigned MaxBits3, cpp_integer_type SignType3, cpp_int_check_type Checked3, class Allocator3>
  61. inline typename enable_if_c<!is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value && !is_trivial_cpp_int<cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2> >::value && !is_trivial_cpp_int<cpp_int_backend<MinBits3, MaxBits3, SignType3, Checked3, Allocator3> >::value >::type
  62. eval_multiply(
  63. cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& result,
  64. const cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2>& a,
  65. const cpp_int_backend<MinBits3, MaxBits3, SignType3, Checked3, Allocator3>& b) BOOST_MP_NOEXCEPT_IF((is_non_throwing_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value))
  66. {
  67. // Very simple long multiplication, only usable for small numbers of limb_type's
  68. // but that's the typical use case for this type anyway:
  69. //
  70. // Special cases first:
  71. //
  72. unsigned as = a.size();
  73. unsigned bs = b.size();
  74. typename cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2>::const_limb_pointer pa = a.limbs();
  75. typename cpp_int_backend<MinBits3, MaxBits3, SignType3, Checked3, Allocator3>::const_limb_pointer pb = b.limbs();
  76. if(as == 1)
  77. {
  78. bool s = b.sign() != a.sign();
  79. if(bs == 1)
  80. {
  81. result = static_cast<double_limb_type>(*pa) * static_cast<double_limb_type>(*pb);
  82. }
  83. else
  84. {
  85. limb_type l = *pa;
  86. eval_multiply(result, b, l);
  87. }
  88. result.sign(s);
  89. return;
  90. }
  91. if(bs == 1)
  92. {
  93. bool s = b.sign() != a.sign();
  94. limb_type l = *pb;
  95. eval_multiply(result, a, l);
  96. result.sign(s);
  97. return;
  98. }
  99. if((void*)&result == (void*)&a)
  100. {
  101. cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> t(a);
  102. eval_multiply(result, t, b);
  103. return;
  104. }
  105. if((void*)&result == (void*)&b)
  106. {
  107. cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> t(b);
  108. eval_multiply(result, a, t);
  109. return;
  110. }
  111. result.resize(as + bs, as + bs - 1);
  112. typename cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>::limb_pointer pr = result.limbs();
  113. static const double_limb_type limb_max = ~static_cast<limb_type>(0u);
  114. static const double_limb_type double_limb_max = ~static_cast<double_limb_type>(0u);
  115. BOOST_STATIC_ASSERT(double_limb_max - 2 * limb_max >= limb_max * limb_max);
  116. double_limb_type carry = 0;
  117. std::memset(pr, 0, result.size() * sizeof(limb_type));
  118. for(unsigned i = 0; i < as; ++i)
  119. {
  120. unsigned inner_limit = cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>::variable ? bs : (std::min)(result.size() - i, bs);
  121. for(unsigned j = 0; j < inner_limit; ++j)
  122. {
  123. BOOST_ASSERT(i+j < result.size());
  124. #if (!defined(__GLIBCXX__) && !defined(__GLIBCPP__)) || !BOOST_WORKAROUND(BOOST_GCC_VERSION, <= 50100)
  125. BOOST_ASSERT(!std::numeric_limits<double_limb_type>::is_specialized
  126. || ((std::numeric_limits<double_limb_type>::max)() - carry
  127. >
  128. static_cast<double_limb_type>(cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>::max_limb_value) * static_cast<double_limb_type>(cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>::max_limb_value)));
  129. #endif
  130. carry += static_cast<double_limb_type>(pa[i]) * static_cast<double_limb_type>(pb[j]);
  131. BOOST_ASSERT(!std::numeric_limits<double_limb_type>::is_specialized || ((std::numeric_limits<double_limb_type>::max)() - carry >= pr[i+j]));
  132. carry += pr[i + j];
  133. pr[i + j] = static_cast<limb_type>(carry);
  134. carry >>= cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>::limb_bits;
  135. BOOST_ASSERT(carry <= (cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>::max_limb_value));
  136. }
  137. resize_for_carry(result, as + bs); // May throw if checking is enabled
  138. if(cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>::variable || (i + bs < result.size()))
  139. pr[i + bs] = static_cast<limb_type>(carry);
  140. carry = 0;
  141. }
  142. result.normalize();
  143. //
  144. // Set the sign of the result:
  145. //
  146. result.sign(a.sign() != b.sign());
  147. }
  148. template <unsigned MinBits1, unsigned MaxBits1, cpp_integer_type SignType1, cpp_int_check_type Checked1, class Allocator1, unsigned MinBits2, unsigned MaxBits2, cpp_integer_type SignType2, cpp_int_check_type Checked2, class Allocator2>
  149. BOOST_MP_FORCEINLINE typename enable_if_c<!is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value && !is_trivial_cpp_int<cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2> >::value >::type
  150. eval_multiply(
  151. cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& result,
  152. const cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2>& a) BOOST_MP_NOEXCEPT_IF((is_non_throwing_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value))
  153. {
  154. eval_multiply(result, result, a);
  155. }
  156. template <unsigned MinBits1, unsigned MaxBits1, cpp_integer_type SignType1, cpp_int_check_type Checked1, class Allocator1>
  157. BOOST_MP_FORCEINLINE typename enable_if_c<!is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value>::type
  158. eval_multiply(cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& result, const limb_type& val) BOOST_MP_NOEXCEPT_IF((is_non_throwing_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value))
  159. {
  160. eval_multiply(result, result, val);
  161. }
  162. template <unsigned MinBits1, unsigned MaxBits1, cpp_integer_type SignType1, cpp_int_check_type Checked1, class Allocator1, unsigned MinBits2, unsigned MaxBits2, cpp_integer_type SignType2, cpp_int_check_type Checked2, class Allocator2>
  163. BOOST_MP_FORCEINLINE typename enable_if_c<!is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value && !is_trivial_cpp_int<cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2> >::value >::type
  164. eval_multiply(
  165. cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& result,
  166. const cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2>& a,
  167. const double_limb_type& val) BOOST_MP_NOEXCEPT_IF((is_non_throwing_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value))
  168. {
  169. if(val <= (std::numeric_limits<limb_type>::max)())
  170. {
  171. eval_multiply(result, a, static_cast<limb_type>(val));
  172. }
  173. else
  174. {
  175. cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> t(val);
  176. eval_multiply(result, a, t);
  177. }
  178. }
  179. template <unsigned MinBits1, unsigned MaxBits1, cpp_integer_type SignType1, cpp_int_check_type Checked1, class Allocator1>
  180. BOOST_MP_FORCEINLINE typename enable_if_c<!is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value>::type
  181. eval_multiply(cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& result, const double_limb_type& val) BOOST_MP_NOEXCEPT_IF((is_non_throwing_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value))
  182. {
  183. eval_multiply(result, result, val);
  184. }
  185. template <unsigned MinBits1, unsigned MaxBits1, cpp_integer_type SignType1, cpp_int_check_type Checked1, class Allocator1, unsigned MinBits2, unsigned MaxBits2, cpp_integer_type SignType2, cpp_int_check_type Checked2, class Allocator2>
  186. BOOST_MP_FORCEINLINE typename enable_if_c<!is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value && !is_trivial_cpp_int<cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2> >::value >::type
  187. eval_multiply(
  188. cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& result,
  189. const cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2>& a,
  190. const signed_limb_type& val) BOOST_MP_NOEXCEPT_IF((is_non_throwing_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value))
  191. {
  192. if(val > 0)
  193. eval_multiply(result, a, static_cast<limb_type>(val));
  194. else
  195. {
  196. eval_multiply(result, a, static_cast<limb_type>(boost::multiprecision::detail::unsigned_abs(val)));
  197. result.negate();
  198. }
  199. }
  200. template <unsigned MinBits1, unsigned MaxBits1, cpp_integer_type SignType1, cpp_int_check_type Checked1, class Allocator1>
  201. BOOST_MP_FORCEINLINE typename enable_if_c<!is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value>::type
  202. eval_multiply(cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& result, const signed_limb_type& val) BOOST_MP_NOEXCEPT_IF((is_non_throwing_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value))
  203. {
  204. eval_multiply(result, result, val);
  205. }
  206. template <unsigned MinBits1, unsigned MaxBits1, cpp_integer_type SignType1, cpp_int_check_type Checked1, class Allocator1, unsigned MinBits2, unsigned MaxBits2, cpp_integer_type SignType2, cpp_int_check_type Checked2, class Allocator2>
  207. inline typename enable_if_c<!is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value && !is_trivial_cpp_int<cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2> >::value >::type
  208. eval_multiply(
  209. cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& result,
  210. const cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2>& a,
  211. const signed_double_limb_type& val) BOOST_MP_NOEXCEPT_IF((is_non_throwing_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value))
  212. {
  213. if(val > 0)
  214. {
  215. if(val <= (std::numeric_limits<limb_type>::max)())
  216. {
  217. eval_multiply(result, a, static_cast<limb_type>(val));
  218. return;
  219. }
  220. }
  221. else if(val >= -static_cast<signed_double_limb_type>((std::numeric_limits<limb_type>::max)()))
  222. {
  223. eval_multiply(result, a, static_cast<limb_type>(boost::multiprecision::detail::unsigned_abs(val)));
  224. result.negate();
  225. return;
  226. }
  227. cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> t(val);
  228. eval_multiply(result, a, t);
  229. }
  230. template <unsigned MinBits1, unsigned MaxBits1, cpp_integer_type SignType1, cpp_int_check_type Checked1, class Allocator1>
  231. BOOST_MP_FORCEINLINE typename enable_if_c<!is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value>::type
  232. eval_multiply(cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& result, const signed_double_limb_type& val) BOOST_MP_NOEXCEPT_IF((is_non_throwing_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value))
  233. {
  234. eval_multiply(result, result, val);
  235. }
  236. //
  237. // Now over again for trivial cpp_int's:
  238. //
  239. template <unsigned MinBits1, unsigned MaxBits1, cpp_integer_type SignType1, cpp_int_check_type Checked1, class Allocator1>
  240. BOOST_MP_FORCEINLINE typename enable_if_c<
  241. is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value
  242. && is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value
  243. && (is_signed_number<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value
  244. || is_signed_number<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value)
  245. >::type
  246. eval_multiply(
  247. cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& result,
  248. const cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& o) BOOST_MP_NOEXCEPT_IF((is_non_throwing_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value))
  249. {
  250. *result.limbs() = detail::checked_multiply(*result.limbs(), *o.limbs(), typename cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>::checked_type());
  251. result.sign(result.sign() != o.sign());
  252. result.normalize();
  253. }
  254. template <unsigned MinBits1, unsigned MaxBits1, cpp_integer_type SignType1, cpp_int_check_type Checked1, class Allocator1>
  255. BOOST_MP_FORCEINLINE typename enable_if_c<
  256. is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value
  257. && is_unsigned_number<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value
  258. >::type
  259. eval_multiply(
  260. cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& result,
  261. const cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& o) BOOST_MP_NOEXCEPT_IF((is_non_throwing_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value))
  262. {
  263. *result.limbs() = detail::checked_multiply(*result.limbs(), *o.limbs(), typename cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>::checked_type());
  264. result.normalize();
  265. }
  266. template <unsigned MinBits1, unsigned MaxBits1, cpp_integer_type SignType1, cpp_int_check_type Checked1, class Allocator1>
  267. BOOST_MP_FORCEINLINE typename enable_if_c<
  268. is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value
  269. && is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value
  270. && (is_signed_number<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value
  271. || is_signed_number<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value)
  272. >::type
  273. eval_multiply(
  274. cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& result,
  275. const cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& a,
  276. 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))
  277. {
  278. *result.limbs() = detail::checked_multiply(*a.limbs(), *b.limbs(), typename cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>::checked_type());
  279. result.sign(a.sign() != b.sign());
  280. result.normalize();
  281. }
  282. template <unsigned MinBits1, unsigned MaxBits1, cpp_integer_type SignType1, cpp_int_check_type Checked1, class Allocator1>
  283. BOOST_MP_FORCEINLINE typename enable_if_c<
  284. is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value
  285. && is_unsigned_number<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value
  286. >::type
  287. eval_multiply(
  288. cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& result,
  289. const cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& a,
  290. 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))
  291. {
  292. *result.limbs() = detail::checked_multiply(*a.limbs(), *b.limbs(), typename cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>::checked_type());
  293. result.normalize();
  294. }
  295. //
  296. // Special routines for multiplying two integers to obtain a multiprecision result:
  297. //
  298. template <unsigned MinBits1, unsigned MaxBits1, cpp_integer_type SignType1, cpp_int_check_type Checked1, class Allocator1>
  299. BOOST_MP_FORCEINLINE typename enable_if_c<
  300. !is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value
  301. >::type
  302. eval_multiply(
  303. cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& result,
  304. signed_double_limb_type a, signed_double_limb_type b)
  305. {
  306. static const signed_double_limb_type mask = ~static_cast<limb_type>(0);
  307. static const unsigned limb_bits = sizeof(limb_type) * CHAR_BIT;
  308. bool s = false;
  309. double_limb_type w, x, y, z;
  310. if(a < 0)
  311. {
  312. a = -a;
  313. s = true;
  314. }
  315. if(b < 0)
  316. {
  317. b = -b;
  318. s = !s;
  319. }
  320. w = a & mask;
  321. x = a >> limb_bits;
  322. y = b & mask;
  323. z = b >> limb_bits;
  324. result.resize(4, 4);
  325. limb_type* pr = result.limbs();
  326. double_limb_type carry = w * y;
  327. pr[0] = static_cast<limb_type>(carry);
  328. carry >>= limb_bits;
  329. carry += w * z + x * y;
  330. pr[1] = static_cast<limb_type>(carry);
  331. carry >>= limb_bits;
  332. carry += x * z;
  333. pr[2] = static_cast<limb_type>(carry);
  334. pr[3] = static_cast<limb_type>(carry >> limb_bits);
  335. result.sign(s);
  336. result.normalize();
  337. }
  338. template <unsigned MinBits1, unsigned MaxBits1, cpp_integer_type SignType1, cpp_int_check_type Checked1, class Allocator1>
  339. BOOST_MP_FORCEINLINE typename enable_if_c<
  340. !is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value
  341. >::type
  342. eval_multiply(
  343. cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& result,
  344. double_limb_type a, double_limb_type b)
  345. {
  346. static const signed_double_limb_type mask = ~static_cast<limb_type>(0);
  347. static const unsigned limb_bits = sizeof(limb_type) * CHAR_BIT;
  348. double_limb_type w, x, y, z;
  349. w = a & mask;
  350. x = a >> limb_bits;
  351. y = b & mask;
  352. z = b >> limb_bits;
  353. result.resize(4, 4);
  354. limb_type* pr = result.limbs();
  355. double_limb_type carry = w * y;
  356. pr[0] = static_cast<limb_type>(carry);
  357. carry >>= limb_bits;
  358. carry += w * z;
  359. pr[1] = static_cast<limb_type>(carry);
  360. carry >>= limb_bits;
  361. pr[2] = static_cast<limb_type>(carry);
  362. carry = x * y + pr[1];
  363. pr[1] = static_cast<limb_type>(carry);
  364. carry >>= limb_bits;
  365. carry += pr[2] + x * z;
  366. pr[2] = static_cast<limb_type>(carry);
  367. pr[3] = static_cast<limb_type>(carry >> limb_bits);
  368. result.sign(false);
  369. result.normalize();
  370. }
  371. template <unsigned MinBits1, unsigned MaxBits1, cpp_integer_type SignType1, cpp_int_check_type Checked1, class Allocator1,
  372. unsigned MinBits2, unsigned MaxBits2, cpp_integer_type SignType2, cpp_int_check_type Checked2, class Allocator2>
  373. BOOST_MP_FORCEINLINE typename enable_if_c<
  374. !is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value
  375. && is_trivial_cpp_int<cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2> >::value
  376. && is_trivial_cpp_int<cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2> >::value
  377. >::type
  378. eval_multiply(
  379. cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& result,
  380. cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2> const& a,
  381. cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2> const& b)
  382. {
  383. typedef typename boost::multiprecision::detail::canonical<typename cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2>::local_limb_type, cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::type canonical_type;
  384. eval_multiply(result, static_cast<canonical_type>(*a.limbs()), static_cast<canonical_type>(*b.limbs()));
  385. result.sign(a.sign() != b.sign());
  386. }
  387. template <unsigned MinBits1, unsigned MaxBits1, cpp_integer_type SignType1, cpp_int_check_type Checked1, class Allocator1, class SI>
  388. BOOST_MP_FORCEINLINE typename enable_if_c<is_signed<SI>::value && (sizeof(SI) <= sizeof(signed_double_limb_type) / 2)>::type
  389. eval_multiply(
  390. cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& result,
  391. SI a, SI b)
  392. {
  393. result = static_cast<signed_double_limb_type>(a) * static_cast<signed_double_limb_type>(b);
  394. }
  395. template <unsigned MinBits1, unsigned MaxBits1, cpp_integer_type SignType1, cpp_int_check_type Checked1, class Allocator1, class UI>
  396. BOOST_MP_FORCEINLINE typename enable_if_c<is_unsigned<UI>::value && (sizeof(UI) <= sizeof(signed_double_limb_type) / 2)>::type
  397. eval_multiply(
  398. cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& result,
  399. UI a, UI b)
  400. {
  401. result = static_cast<double_limb_type>(a) * static_cast<double_limb_type>(b);
  402. }
  403. }}} // namespaces
  404. #endif