bitwise.hpp 38 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843
  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_BIT_HPP
  9. #define BOOST_MP_CPP_INT_BIT_HPP
  10. #ifdef _MSC_VER
  11. #pragma warning(push)
  12. #pragma warning(disable:4319)
  13. #endif
  14. namespace boost{ namespace multiprecision{ namespace backends{
  15. 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>
  16. void is_valid_bitwise_op(
  17. cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& result,
  18. const cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2>& o, const mpl::int_<checked>&)
  19. {
  20. if(result.sign() || o.sign())
  21. BOOST_THROW_EXCEPTION(std::range_error("Bitwise operations on negative values results in undefined behavior."));
  22. }
  23. 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>
  24. void is_valid_bitwise_op(
  25. cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>&,
  26. const cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2>& , const mpl::int_<unchecked>&){}
  27. template <unsigned MinBits1, unsigned MaxBits1, cpp_int_check_type Checked1, class Allocator1>
  28. void is_valid_bitwise_op(
  29. const cpp_int_backend<MinBits1, MaxBits1, signed_magnitude, Checked1, Allocator1>& result, const mpl::int_<checked>&)
  30. {
  31. if(result.sign())
  32. BOOST_THROW_EXCEPTION(std::range_error("Bitwise operations on negative values results in undefined behavior."));
  33. }
  34. template <unsigned MinBits1, unsigned MaxBits1, cpp_int_check_type Checked1, class Allocator1>
  35. void is_valid_bitwise_op(
  36. const cpp_int_backend<MinBits1, MaxBits1, unsigned_magnitude, Checked1, Allocator1>&, const mpl::int_<checked>&){}
  37. template <unsigned MinBits1, unsigned MaxBits1, cpp_integer_type SignType1, cpp_int_check_type Checked1, class Allocator1>
  38. void is_valid_bitwise_op(
  39. cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>&, const mpl::int_<unchecked>&){}
  40. template <class CppInt1, class CppInt2, class Op>
  41. void bitwise_op(
  42. CppInt1& result,
  43. const CppInt2& o,
  44. Op op, const mpl::true_&) BOOST_MP_NOEXCEPT_IF((is_non_throwing_cpp_int<CppInt1>::value))
  45. {
  46. //
  47. // There are 4 cases:
  48. // * Both positive.
  49. // * result negative, o positive.
  50. // * o negative, result positive.
  51. // * Both negative.
  52. //
  53. // When one arg is negative we convert to 2's complement form "on the fly",
  54. // and then convert back to signed-magnitude form at the end.
  55. //
  56. // Note however, that if the type is checked, then bitwise ops on negative values
  57. // are not permitted and an exception will result.
  58. //
  59. is_valid_bitwise_op(result, o, typename CppInt1::checked_type());
  60. //
  61. // First figure out how big the result needs to be and set up some data:
  62. //
  63. unsigned rs = result.size();
  64. unsigned os = o.size();
  65. unsigned m, x;
  66. minmax(rs, os, m, x);
  67. result.resize(x, x);
  68. typename CppInt1::limb_pointer pr = result.limbs();
  69. typename CppInt2::const_limb_pointer po = o.limbs();
  70. for(unsigned i = rs; i < x; ++i)
  71. pr[i] = 0;
  72. limb_type next_limb = 0;
  73. if(!result.sign())
  74. {
  75. if(!o.sign())
  76. {
  77. for(unsigned i = 0; i < os; ++i)
  78. pr[i] = op(pr[i], po[i]);
  79. for(unsigned i = os; i < x; ++i)
  80. pr[i] = op(pr[i], limb_type(0));
  81. }
  82. else
  83. {
  84. // "o" is negative:
  85. double_limb_type carry = 1;
  86. for(unsigned i = 0; i < os; ++i)
  87. {
  88. carry += static_cast<double_limb_type>(~po[i]);
  89. pr[i] = op(pr[i], static_cast<limb_type>(carry));
  90. carry >>= CppInt1::limb_bits;
  91. }
  92. for(unsigned i = os; i < x; ++i)
  93. {
  94. carry += static_cast<double_limb_type>(~limb_type(0));
  95. pr[i] = op(pr[i], static_cast<limb_type>(carry));
  96. carry >>= CppInt1::limb_bits;
  97. }
  98. // Set the overflow into the "extra" limb:
  99. carry += static_cast<double_limb_type>(~limb_type(0));
  100. next_limb = op(limb_type(0), static_cast<limb_type>(carry));
  101. }
  102. }
  103. else
  104. {
  105. if(!o.sign())
  106. {
  107. // "result" is negative:
  108. double_limb_type carry = 1;
  109. for(unsigned i = 0; i < os; ++i)
  110. {
  111. carry += static_cast<double_limb_type>(~pr[i]);
  112. pr[i] = op(static_cast<limb_type>(carry), po[i]);
  113. carry >>= CppInt1::limb_bits;
  114. }
  115. for(unsigned i = os; i < x; ++i)
  116. {
  117. carry += static_cast<double_limb_type>(~pr[i]);
  118. pr[i] = op(static_cast<limb_type>(carry), limb_type(0));
  119. carry >>= CppInt1::limb_bits;
  120. }
  121. // Set the overflow into the "extra" limb:
  122. carry += static_cast<double_limb_type>(~limb_type(0));
  123. next_limb = op(static_cast<limb_type>(carry), limb_type(0));
  124. }
  125. else
  126. {
  127. // both are negative:
  128. double_limb_type r_carry = 1;
  129. double_limb_type o_carry = 1;
  130. for(unsigned i = 0; i < os; ++i)
  131. {
  132. r_carry += static_cast<double_limb_type>(~pr[i]);
  133. o_carry += static_cast<double_limb_type>(~po[i]);
  134. pr[i] = op(static_cast<limb_type>(r_carry), static_cast<limb_type>(o_carry));
  135. r_carry >>= CppInt1::limb_bits;
  136. o_carry >>= CppInt1::limb_bits;
  137. }
  138. for(unsigned i = os; i < x; ++i)
  139. {
  140. r_carry += static_cast<double_limb_type>(~pr[i]);
  141. o_carry += static_cast<double_limb_type>(~limb_type(0));
  142. pr[i] = op(static_cast<limb_type>(r_carry), static_cast<limb_type>(o_carry));
  143. r_carry >>= CppInt1::limb_bits;
  144. o_carry >>= CppInt1::limb_bits;
  145. }
  146. // Set the overflow into the "extra" limb:
  147. r_carry += static_cast<double_limb_type>(~limb_type(0));
  148. o_carry += static_cast<double_limb_type>(~limb_type(0));
  149. next_limb = op(static_cast<limb_type>(r_carry), static_cast<limb_type>(o_carry));
  150. }
  151. }
  152. //
  153. // See if the result is negative or not:
  154. //
  155. if(static_cast<signed_limb_type>(next_limb) < 0)
  156. {
  157. double_limb_type carry = 1;
  158. for(unsigned i = 0; i < x; ++i)
  159. {
  160. carry += static_cast<double_limb_type>(~pr[i]);
  161. pr[i] = static_cast<limb_type>(carry);
  162. carry >>= CppInt1::limb_bits;
  163. }
  164. if(carry)
  165. {
  166. result.resize(x + 1, x);
  167. if(result.size() > x)
  168. result.limbs()[x] = static_cast<limb_type>(carry);
  169. }
  170. result.sign(true);
  171. }
  172. else
  173. result.sign(false);
  174. result.normalize();
  175. }
  176. template <class CppInt1, class CppInt2, class Op>
  177. void bitwise_op(
  178. CppInt1& result,
  179. const CppInt2& o,
  180. Op op, const mpl::false_&) BOOST_MP_NOEXCEPT_IF((is_non_throwing_cpp_int<CppInt1>::value))
  181. {
  182. //
  183. // Both arguments are unsigned types, very simple case handled as a special case.
  184. //
  185. // First figure out how big the result needs to be and set up some data:
  186. //
  187. unsigned rs = result.size();
  188. unsigned os = o.size();
  189. unsigned m, x;
  190. minmax(rs, os, m, x);
  191. result.resize(x, x);
  192. typename CppInt1::limb_pointer pr = result.limbs();
  193. typename CppInt2::const_limb_pointer po = o.limbs();
  194. for(unsigned i = rs; i < x; ++i)
  195. pr[i] = 0;
  196. for(unsigned i = 0; i < os; ++i)
  197. pr[i] = op(pr[i], po[i]);
  198. for(unsigned i = os; i < x; ++i)
  199. pr[i] = op(pr[i], limb_type(0));
  200. result.normalize();
  201. }
  202. struct bit_and{ limb_type operator()(limb_type a, limb_type b)const BOOST_NOEXCEPT { return a & b; } };
  203. struct bit_or { limb_type operator()(limb_type a, limb_type b)const BOOST_NOEXCEPT { return a | b; } };
  204. struct bit_xor{ limb_type operator()(limb_type a, limb_type b)const BOOST_NOEXCEPT { return a ^ b; } };
  205. 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>
  206. 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
  207. eval_bitwise_and(
  208. cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& result,
  209. const cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2>& o) BOOST_MP_NOEXCEPT_IF((is_non_throwing_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value))
  210. {
  211. bitwise_op(result, o, bit_and(),
  212. mpl::bool_<std::numeric_limits<number<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> > >::is_signed || std::numeric_limits<number<cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2> > >::is_signed>());
  213. }
  214. 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>
  215. 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
  216. eval_bitwise_or(
  217. cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& result,
  218. const cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2>& o) BOOST_MP_NOEXCEPT_IF((is_non_throwing_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value))
  219. {
  220. bitwise_op(result, o, bit_or(),
  221. mpl::bool_<std::numeric_limits<number<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> > >::is_signed || std::numeric_limits<number<cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2> > >::is_signed>());
  222. }
  223. 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>
  224. 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
  225. eval_bitwise_xor(
  226. cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& result,
  227. const cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2>& o) BOOST_MP_NOEXCEPT_IF((is_non_throwing_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value))
  228. {
  229. bitwise_op(result, o, bit_xor(),
  230. mpl::bool_<std::numeric_limits<number<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> > >::is_signed || std::numeric_limits<number<cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2> > >::is_signed>());
  231. }
  232. //
  233. // Again for operands which are single limbs:
  234. //
  235. template <unsigned MinBits1, unsigned MaxBits1, cpp_int_check_type Checked1, class Allocator1>
  236. BOOST_MP_FORCEINLINE typename enable_if_c<!is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, unsigned_magnitude, Checked1, Allocator1> >::value>::type
  237. eval_bitwise_and(
  238. cpp_int_backend<MinBits1, MaxBits1, unsigned_magnitude, Checked1, Allocator1>& result,
  239. limb_type l) BOOST_NOEXCEPT
  240. {
  241. result.limbs()[0] &= l;
  242. result.resize(1, 1);
  243. }
  244. template <unsigned MinBits1, unsigned MaxBits1, cpp_int_check_type Checked1, class Allocator1>
  245. BOOST_MP_FORCEINLINE typename enable_if_c<!is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, unsigned_magnitude, Checked1, Allocator1> >::value>::type
  246. eval_bitwise_or(
  247. cpp_int_backend<MinBits1, MaxBits1, unsigned_magnitude, Checked1, Allocator1>& result,
  248. limb_type l) BOOST_NOEXCEPT
  249. {
  250. result.limbs()[0] |= l;
  251. }
  252. template <unsigned MinBits1, unsigned MaxBits1, cpp_int_check_type Checked1, class Allocator1>
  253. BOOST_MP_FORCEINLINE typename enable_if_c<!is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, unsigned_magnitude, Checked1, Allocator1> >::value>::type
  254. eval_bitwise_xor(
  255. cpp_int_backend<MinBits1, MaxBits1, unsigned_magnitude, Checked1, Allocator1>& result,
  256. limb_type l) BOOST_NOEXCEPT
  257. {
  258. result.limbs()[0] ^= l;
  259. }
  260. 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>
  261. BOOST_MP_FORCEINLINE typename enable_if_c<is_signed_number<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value && !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
  262. eval_complement(
  263. cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& result,
  264. const cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2>& o) BOOST_MP_NOEXCEPT_IF((is_non_throwing_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value))
  265. {
  266. BOOST_STATIC_ASSERT_MSG(((Checked1 != checked) || (Checked2 != checked)), "Attempt to take the complement of a signed type results in undefined behavior.");
  267. // Increment and negate:
  268. result = o;
  269. eval_increment(result);
  270. result.negate();
  271. }
  272. template <unsigned MinBits1, unsigned MaxBits1, cpp_integer_type SignType1, cpp_int_check_type Checked1, class Allocator1>
  273. BOOST_MP_FORCEINLINE typename enable_if_c<is_unsigned_number<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value && !is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value && !is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value >::type
  274. eval_complement(
  275. cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& result,
  276. 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))
  277. {
  278. unsigned os = o.size();
  279. result.resize(UINT_MAX, os);
  280. for(unsigned i = 0; i < os; ++i)
  281. result.limbs()[i] = ~o.limbs()[i];
  282. for(unsigned i = os; i < result.size(); ++i)
  283. result.limbs()[i] = ~static_cast<limb_type>(0);
  284. result.normalize();
  285. }
  286. template <class Int>
  287. inline void left_shift_byte(Int& result, double_limb_type s)
  288. {
  289. limb_type offset = static_cast<limb_type>(s / Int::limb_bits);
  290. limb_type shift = static_cast<limb_type>(s % Int::limb_bits);
  291. unsigned ors = result.size();
  292. if((ors == 1) && (!*result.limbs()))
  293. return; // shifting zero yields zero.
  294. unsigned rs = ors;
  295. if(shift && (result.limbs()[ors - 1] >> (Int::limb_bits - shift)))
  296. ++rs; // Most significant limb will overflow when shifted
  297. rs += offset;
  298. result.resize(rs, rs);
  299. rs = result.size();
  300. typename Int::limb_pointer pr = result.limbs();
  301. if(rs != ors)
  302. pr[rs - 1] = 0u;
  303. std::size_t bytes = static_cast<std::size_t>(s / CHAR_BIT);
  304. std::size_t len = std::min(ors * sizeof(limb_type), rs * sizeof(limb_type) - bytes);
  305. if(bytes >= rs * sizeof(limb_type))
  306. result = static_cast<limb_type>(0u);
  307. else
  308. {
  309. unsigned char* pc = reinterpret_cast<unsigned char*>(pr);
  310. std::memmove(pc + bytes, pc, len);
  311. std::memset(pc, 0, bytes);
  312. }
  313. }
  314. template <class Int>
  315. inline void left_shift_limb(Int& result, double_limb_type s)
  316. {
  317. limb_type offset = static_cast<limb_type>(s / Int::limb_bits);
  318. limb_type shift = static_cast<limb_type>(s % Int::limb_bits);
  319. unsigned ors = result.size();
  320. if((ors == 1) && (!*result.limbs()))
  321. return; // shifting zero yields zero.
  322. unsigned rs = ors;
  323. if(shift && (result.limbs()[ors - 1] >> (Int::limb_bits - shift)))
  324. ++rs; // Most significant limb will overflow when shifted
  325. rs += offset;
  326. result.resize(rs, rs);
  327. bool truncated = result.size() != rs;
  328. typename Int::limb_pointer pr = result.limbs();
  329. if(offset > rs)
  330. {
  331. // The result is shifted past the end of the result:
  332. result = static_cast<limb_type>(0);
  333. return;
  334. }
  335. unsigned i = rs - result.size();
  336. for(; i < ors; ++i)
  337. pr[rs - 1 - i] = pr[ors - 1 - i];
  338. for(; i < rs; ++i)
  339. pr[rs - 1 - i] = 0;
  340. }
  341. template <class Int>
  342. inline void left_shift_generic(Int& result, double_limb_type s)
  343. {
  344. limb_type offset = static_cast<limb_type>(s / Int::limb_bits);
  345. limb_type shift = static_cast<limb_type>(s % Int::limb_bits);
  346. unsigned ors = result.size();
  347. if((ors == 1) && (!*result.limbs()))
  348. return; // shifting zero yields zero.
  349. unsigned rs = ors;
  350. if(shift && (result.limbs()[ors - 1] >> (Int::limb_bits - shift)))
  351. ++rs; // Most significant limb will overflow when shifted
  352. rs += offset;
  353. result.resize(rs, rs);
  354. bool truncated = result.size() != rs;
  355. typename Int::limb_pointer pr = result.limbs();
  356. if(offset > rs)
  357. {
  358. // The result is shifted past the end of the result:
  359. result = static_cast<limb_type>(0);
  360. return;
  361. }
  362. unsigned i = rs - result.size();
  363. // This code only works when shift is non-zero, otherwise we invoke undefined behaviour!
  364. BOOST_ASSERT(shift);
  365. if(!truncated)
  366. {
  367. if(rs > ors + offset)
  368. {
  369. pr[rs - 1 - i] = pr[ors - 1 - i] >> (Int::limb_bits - shift);
  370. --rs;
  371. }
  372. else
  373. {
  374. pr[rs - 1 - i] = pr[ors - 1 - i] << shift;
  375. if(ors > 1)
  376. pr[rs - 1 - i] |= pr[ors - 2 - i] >> (Int::limb_bits - shift);
  377. ++i;
  378. }
  379. }
  380. for(; ors > 1 + i; ++i)
  381. {
  382. pr[rs - 1 - i] = pr[ors - 1 - i] << shift;
  383. pr[rs - 1 - i] |= pr[ors - 2 - i] >> (Int::limb_bits - shift);
  384. }
  385. if(ors >= 1 + i)
  386. {
  387. pr[rs - 1 - i] = pr[ors - 1 - i] << shift;
  388. ++i;
  389. }
  390. for(; i < rs; ++i)
  391. pr[rs - 1 - i] = 0;
  392. }
  393. template <unsigned MinBits1, unsigned MaxBits1, cpp_integer_type SignType1, cpp_int_check_type Checked1, class Allocator1>
  394. inline typename enable_if_c<!is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value>::type
  395. eval_left_shift(
  396. cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& result,
  397. double_limb_type s) BOOST_MP_NOEXCEPT_IF((is_non_throwing_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value))
  398. {
  399. is_valid_bitwise_op(result, typename cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>::checked_type());
  400. if(!s)
  401. return;
  402. #if defined(BOOST_LITTLE_ENDIAN) && defined(BOOST_MP_USE_LIMB_SHIFT)
  403. static const limb_type limb_shift_mask = cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>::limb_bits - 1;
  404. static const limb_type byte_shift_mask = CHAR_BIT - 1;
  405. if((s & limb_shift_mask) == 0)
  406. {
  407. left_shift_limb(result, s);
  408. }
  409. else if((s & byte_shift_mask) == 0)
  410. {
  411. left_shift_byte(result, s);
  412. }
  413. #elif defined(BOOST_LITTLE_ENDIAN)
  414. static const limb_type byte_shift_mask = CHAR_BIT - 1;
  415. if((s & byte_shift_mask) == 0)
  416. {
  417. left_shift_byte(result, s);
  418. }
  419. #else
  420. static const limb_type limb_shift_mask = cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>::limb_bits - 1;
  421. if((s & limb_shift_mask) == 0)
  422. {
  423. left_shift_limb(result, s);
  424. }
  425. #endif
  426. else
  427. {
  428. left_shift_generic(result, s);
  429. }
  430. //
  431. // We may have shifted off the end and have leading zeros:
  432. //
  433. result.normalize();
  434. }
  435. template <class Int>
  436. inline void right_shift_byte(Int& result, double_limb_type s)
  437. {
  438. limb_type offset = static_cast<limb_type>(s / Int::limb_bits);
  439. limb_type shift;
  440. BOOST_ASSERT((s % CHAR_BIT) == 0);
  441. unsigned ors = result.size();
  442. unsigned rs = ors;
  443. if(offset >= rs)
  444. {
  445. result = limb_type(0);
  446. return;
  447. }
  448. rs -= offset;
  449. typename Int::limb_pointer pr = result.limbs();
  450. unsigned char* pc = reinterpret_cast<unsigned char*>(pr);
  451. shift = static_cast<limb_type>(s / CHAR_BIT);
  452. std::memmove(pc, pc + shift, ors * sizeof(pr[0]) - shift);
  453. shift = (sizeof(limb_type) - shift % sizeof(limb_type)) * CHAR_BIT;
  454. if(shift < Int::limb_bits)
  455. {
  456. pr[ors - offset - 1] &= (static_cast<limb_type>(1u) << shift) - 1;
  457. if(!pr[ors - offset - 1] && (rs > 1))
  458. --rs;
  459. }
  460. result.resize(rs, rs);
  461. }
  462. template <class Int>
  463. inline void right_shift_limb(Int& result, double_limb_type s)
  464. {
  465. limb_type offset = static_cast<limb_type>(s / Int::limb_bits);
  466. BOOST_ASSERT((s % Int::limb_bits) == 0);
  467. unsigned ors = result.size();
  468. unsigned rs = ors;
  469. if(offset >= rs)
  470. {
  471. result = limb_type(0);
  472. return;
  473. }
  474. rs -= offset;
  475. typename Int::limb_pointer pr = result.limbs();
  476. unsigned i = 0;
  477. for(; i < rs; ++i)
  478. pr[i] = pr[i + offset];
  479. result.resize(rs, rs);
  480. }
  481. template <class Int>
  482. inline void right_shift_generic(Int& result, double_limb_type s)
  483. {
  484. limb_type offset = static_cast<limb_type>(s / Int::limb_bits);
  485. limb_type shift = static_cast<limb_type>(s % Int::limb_bits);
  486. unsigned ors = result.size();
  487. unsigned rs = ors;
  488. if(offset >= rs)
  489. {
  490. result = limb_type(0);
  491. return;
  492. }
  493. rs -= offset;
  494. typename Int::limb_pointer pr = result.limbs();
  495. if((pr[ors - 1] >> shift) == 0)
  496. {
  497. if(--rs == 0)
  498. {
  499. result = limb_type(0);
  500. return;
  501. }
  502. }
  503. unsigned i = 0;
  504. // This code only works for non-zero shift, otherwise we invoke undefined behaviour!
  505. BOOST_ASSERT(shift);
  506. for(; i + offset + 1 < ors; ++i)
  507. {
  508. pr[i] = pr[i + offset] >> shift;
  509. pr[i] |= pr[i + offset + 1] << (Int::limb_bits - shift);
  510. }
  511. pr[i] = pr[i + offset] >> shift;
  512. result.resize(rs, rs);
  513. }
  514. template <unsigned MinBits1, unsigned MaxBits1, cpp_int_check_type Checked1, class Allocator1>
  515. inline typename enable_if_c<!is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, unsigned_magnitude, Checked1, Allocator1> >::value>::type
  516. eval_right_shift(
  517. cpp_int_backend<MinBits1, MaxBits1, unsigned_magnitude, Checked1, Allocator1>& result,
  518. double_limb_type s) BOOST_MP_NOEXCEPT_IF((is_non_throwing_cpp_int<cpp_int_backend<MinBits1, MaxBits1, unsigned_magnitude, Checked1, Allocator1> >::value))
  519. {
  520. is_valid_bitwise_op(result, typename cpp_int_backend<MinBits1, MaxBits1, unsigned_magnitude, Checked1, Allocator1>::checked_type());
  521. if(!s)
  522. return;
  523. #if defined(BOOST_LITTLE_ENDIAN) && defined(BOOST_MP_USE_LIMB_SHIFT)
  524. static const limb_type limb_shift_mask = cpp_int_backend<MinBits1, MaxBits1, signed_magnitude, Checked1, Allocator1>::limb_bits - 1;
  525. static const limb_type byte_shift_mask = CHAR_BIT - 1;
  526. if((s & limb_shift_mask) == 0)
  527. right_shift_limb(result, s);
  528. else if((s & byte_shift_mask) == 0)
  529. right_shift_byte(result, s);
  530. #elif defined(BOOST_LITTLE_ENDIAN)
  531. static const limb_type byte_shift_mask = CHAR_BIT - 1;
  532. if((s & byte_shift_mask) == 0)
  533. right_shift_byte(result, s);
  534. #else
  535. static const limb_type limb_shift_mask = cpp_int_backend<MinBits1, MaxBits1, signed_magnitude, Checked1, Allocator1>::limb_bits - 1;
  536. if((s & limb_shift_mask) == 0)
  537. right_shift_limb(result, s);
  538. #endif
  539. else
  540. right_shift_generic(result, s);
  541. }
  542. template <unsigned MinBits1, unsigned MaxBits1, cpp_int_check_type Checked1, class Allocator1>
  543. inline typename enable_if_c<!is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, signed_magnitude, Checked1, Allocator1> >::value>::type
  544. eval_right_shift(
  545. cpp_int_backend<MinBits1, MaxBits1, signed_magnitude, Checked1, Allocator1>& result,
  546. double_limb_type s) BOOST_MP_NOEXCEPT_IF((is_non_throwing_cpp_int<cpp_int_backend<MinBits1, MaxBits1, signed_magnitude, Checked1, Allocator1> >::value))
  547. {
  548. is_valid_bitwise_op(result, typename cpp_int_backend<MinBits1, MaxBits1, signed_magnitude, Checked1, Allocator1>::checked_type());
  549. if(!s)
  550. return;
  551. bool is_neg = result.sign();
  552. if(is_neg)
  553. eval_increment(result);
  554. #if defined(BOOST_LITTLE_ENDIAN) && defined(BOOST_MP_USE_LIMB_SHIFT)
  555. static const limb_type limb_shift_mask = cpp_int_backend<MinBits1, MaxBits1, signed_magnitude, Checked1, Allocator1>::limb_bits - 1;
  556. static const limb_type byte_shift_mask = CHAR_BIT - 1;
  557. if((s & limb_shift_mask) == 0)
  558. right_shift_limb(result, s);
  559. else if((s & byte_shift_mask) == 0)
  560. right_shift_byte(result, s);
  561. #elif defined(BOOST_LITTLE_ENDIAN)
  562. static const limb_type byte_shift_mask = CHAR_BIT - 1;
  563. if((s & byte_shift_mask) == 0)
  564. right_shift_byte(result, s);
  565. #else
  566. static const limb_type limb_shift_mask = cpp_int_backend<MinBits1, MaxBits1, signed_magnitude, Checked1, Allocator1>::limb_bits - 1;
  567. if((s & limb_shift_mask) == 0)
  568. right_shift_limb(result, s);
  569. #endif
  570. else
  571. right_shift_generic(result, s);
  572. if(is_neg)
  573. eval_decrement(result);
  574. }
  575. //
  576. // Over again for trivial cpp_int's:
  577. //
  578. template <unsigned MinBits1, unsigned MaxBits1, cpp_integer_type SignType1, cpp_int_check_type Checked1, class Allocator1, class T>
  579. BOOST_MP_FORCEINLINE typename enable_if<is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> > >::type
  580. eval_left_shift(cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& result, T s) BOOST_MP_NOEXCEPT_IF((is_non_throwing_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value))
  581. {
  582. is_valid_bitwise_op(result, typename cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>::checked_type());
  583. *result.limbs() = detail::checked_left_shift(*result.limbs(), s, typename cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>::checked_type());
  584. result.normalize();
  585. }
  586. template <unsigned MinBits1, unsigned MaxBits1, cpp_integer_type SignType1, cpp_int_check_type Checked1, class Allocator1, class T>
  587. BOOST_MP_FORCEINLINE typename enable_if<is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> > >::type
  588. eval_right_shift(cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& result, T s) BOOST_MP_NOEXCEPT_IF((is_non_throwing_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value))
  589. {
  590. // Nothing to check here... just make sure we don't invoke undefined behavior:
  591. is_valid_bitwise_op(result, typename cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>::checked_type());
  592. *result.limbs() = (static_cast<unsigned>(s) >= sizeof(*result.limbs()) * CHAR_BIT) ? 0 : (result.sign() ? ((--*result.limbs()) >> s) + 1 : *result.limbs() >> s);
  593. if(result.sign() && (*result.limbs() == 0))
  594. result = static_cast<signed_limb_type>(-1);
  595. }
  596. 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>
  597. inline typename enable_if_c<
  598. is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value
  599. && is_trivial_cpp_int<cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2> >::value
  600. && (is_signed_number<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value || is_signed_number<cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2> >::value)
  601. >::type
  602. eval_complement(
  603. cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& result,
  604. const cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2>& o) BOOST_MP_NOEXCEPT_IF((is_non_throwing_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value))
  605. {
  606. BOOST_STATIC_ASSERT_MSG(((Checked1 != checked) || (Checked2 != checked)), "Attempt to take the complement of a signed type results in undefined behavior.");
  607. //
  608. // If we're not checked then emulate 2's complement behavior:
  609. //
  610. if(o.sign())
  611. {
  612. *result.limbs() = *o.limbs() - 1;
  613. result.sign(false);
  614. }
  615. else
  616. {
  617. *result.limbs() = 1 + *o.limbs();
  618. result.sign(true);
  619. }
  620. result.normalize();
  621. }
  622. 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>
  623. inline typename enable_if_c<
  624. is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value
  625. && is_trivial_cpp_int<cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2> >::value
  626. && is_unsigned_number<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value
  627. && is_unsigned_number<cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2> >::value
  628. >::type
  629. eval_complement(
  630. cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& result,
  631. const cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2>& o) BOOST_MP_NOEXCEPT_IF((is_non_throwing_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value))
  632. {
  633. *result.limbs() = ~*o.limbs();
  634. result.normalize();
  635. }
  636. 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>
  637. inline typename enable_if_c<
  638. is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value
  639. && is_trivial_cpp_int<cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2> >::value
  640. && is_unsigned_number<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value
  641. && is_unsigned_number<cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2> >::value
  642. >::type
  643. eval_bitwise_and(
  644. cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& result,
  645. const cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2>& o) BOOST_MP_NOEXCEPT_IF((is_non_throwing_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value))
  646. {
  647. *result.limbs() &= *o.limbs();
  648. }
  649. 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>
  650. inline typename enable_if_c<
  651. is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value
  652. && is_trivial_cpp_int<cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2> >::value
  653. && (is_signed_number<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value || is_signed_number<cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2> >::value)
  654. >::type
  655. eval_bitwise_and(
  656. cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& result,
  657. const cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2>& o) BOOST_MP_NOEXCEPT_IF((is_non_throwing_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value))
  658. {
  659. is_valid_bitwise_op(result, o, typename cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>::checked_type());
  660. using default_ops::eval_bit_test;
  661. using default_ops::eval_increment;
  662. if(result.sign() || o.sign())
  663. {
  664. static const unsigned m = static_unsigned_max<static_unsigned_max<MinBits1, MinBits2>::value, static_unsigned_max<MaxBits1, MaxBits2>::value>::value;
  665. cpp_int_backend<m + 1, m + 1, unsigned_magnitude, unchecked, void> t1(result);
  666. cpp_int_backend<m + 1, m + 1, unsigned_magnitude, unchecked, void> t2(o);
  667. eval_bitwise_and(t1, t2);
  668. bool s = eval_bit_test(t1, m + 1);
  669. if(s)
  670. {
  671. eval_complement(t1, t1);
  672. eval_increment(t1);
  673. }
  674. result = t1;
  675. result.sign(s);
  676. }
  677. else
  678. {
  679. *result.limbs() &= *o.limbs();
  680. }
  681. }
  682. 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>
  683. inline typename enable_if_c<
  684. is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value
  685. && is_trivial_cpp_int<cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2> >::value
  686. && is_unsigned_number<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value
  687. && is_unsigned_number<cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2> >::value
  688. >::type
  689. eval_bitwise_or(
  690. cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& result,
  691. const cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2>& o) BOOST_MP_NOEXCEPT_IF((is_non_throwing_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value))
  692. {
  693. *result.limbs() |= *o.limbs();
  694. }
  695. 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>
  696. inline typename enable_if_c<
  697. is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value
  698. && is_trivial_cpp_int<cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2> >::value
  699. && (is_signed_number<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value || is_signed_number<cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2> >::value)
  700. >::type
  701. eval_bitwise_or(
  702. cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& result,
  703. const cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2>& o) BOOST_MP_NOEXCEPT_IF((is_non_throwing_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value))
  704. {
  705. is_valid_bitwise_op(result, o, typename cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>::checked_type());
  706. using default_ops::eval_bit_test;
  707. using default_ops::eval_increment;
  708. if(result.sign() || o.sign())
  709. {
  710. static const unsigned m = static_unsigned_max<static_unsigned_max<MinBits1, MinBits2>::value, static_unsigned_max<MaxBits1, MaxBits2>::value>::value;
  711. cpp_int_backend<m + 1, m + 1, unsigned_magnitude, unchecked, void> t1(result);
  712. cpp_int_backend<m + 1, m + 1, unsigned_magnitude, unchecked, void> t2(o);
  713. eval_bitwise_or(t1, t2);
  714. bool s = eval_bit_test(t1, m + 1);
  715. if(s)
  716. {
  717. eval_complement(t1, t1);
  718. eval_increment(t1);
  719. }
  720. result = t1;
  721. result.sign(s);
  722. }
  723. else
  724. {
  725. *result.limbs() |= *o.limbs();
  726. result.normalize();
  727. }
  728. }
  729. 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>
  730. inline typename enable_if_c<
  731. is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value
  732. && is_trivial_cpp_int<cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2> >::value
  733. && is_unsigned_number<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value
  734. && is_unsigned_number<cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2> >::value
  735. >::type
  736. eval_bitwise_xor(
  737. cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& result,
  738. const cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2>& o) BOOST_MP_NOEXCEPT_IF((is_non_throwing_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value))
  739. {
  740. *result.limbs() ^= *o.limbs();
  741. }
  742. 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>
  743. inline typename enable_if_c<
  744. is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value
  745. && is_trivial_cpp_int<cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2> >::value
  746. && (is_signed_number<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value || is_signed_number<cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2> >::value)
  747. >::type
  748. eval_bitwise_xor(
  749. cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& result,
  750. const cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2>& o) BOOST_MP_NOEXCEPT_IF((is_non_throwing_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value))
  751. {
  752. is_valid_bitwise_op(result, o, typename cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>::checked_type());
  753. using default_ops::eval_bit_test;
  754. using default_ops::eval_increment;
  755. if(result.sign() || o.sign())
  756. {
  757. static const unsigned m = static_unsigned_max<static_unsigned_max<MinBits1, MinBits2>::value, static_unsigned_max<MaxBits1, MaxBits2>::value>::value;
  758. cpp_int_backend<m + 1, m + 1, unsigned_magnitude, unchecked, void> t1(result);
  759. cpp_int_backend<m + 1, m + 1, unsigned_magnitude, unchecked, void> t2(o);
  760. eval_bitwise_xor(t1, t2);
  761. bool s = eval_bit_test(t1, m + 1);
  762. if(s)
  763. {
  764. eval_complement(t1, t1);
  765. eval_increment(t1);
  766. }
  767. result = t1;
  768. result.sign(s);
  769. }
  770. else
  771. {
  772. *result.limbs() ^= *o.limbs();
  773. }
  774. }
  775. }}} // namespaces
  776. #ifdef _MSC_VER
  777. #pragma warning(pop)
  778. #endif
  779. #endif