add.hpp 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529
  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_ADD_HPP
  9. #define BOOST_MP_CPP_INT_ADD_HPP
  10. namespace boost{ namespace multiprecision{ namespace backends{
  11. //
  12. // This is the key addition routine where all the argument types are non-trivial cpp_int's:
  13. //
  14. template <class CppInt1, class CppInt2, class CppInt3>
  15. inline void add_unsigned(CppInt1& result, const CppInt2& a, const CppInt3& b) BOOST_MP_NOEXCEPT_IF(is_non_throwing_cpp_int<CppInt1>::value)
  16. {
  17. using std::swap;
  18. // Nothing fancy, just let uintmax_t take the strain:
  19. double_limb_type carry = 0;
  20. unsigned m, x;
  21. unsigned as = a.size();
  22. unsigned bs = b.size();
  23. minmax(as, bs, m, x);
  24. if(x == 1)
  25. {
  26. bool s = a.sign();
  27. result = static_cast<double_limb_type>(*a.limbs()) + static_cast<double_limb_type>(*b.limbs());
  28. result.sign(s);
  29. return;
  30. }
  31. result.resize(x, x);
  32. typename CppInt2::const_limb_pointer pa = a.limbs();
  33. typename CppInt3::const_limb_pointer pb = b.limbs();
  34. typename CppInt1::limb_pointer pr = result.limbs();
  35. typename CppInt1::limb_pointer pr_end = pr + m;
  36. if(as < bs)
  37. swap(pa, pb);
  38. // First where a and b overlap:
  39. while(pr != pr_end)
  40. {
  41. carry += static_cast<double_limb_type>(*pa) + static_cast<double_limb_type>(*pb);
  42. *pr = static_cast<limb_type>(carry);
  43. carry >>= CppInt1::limb_bits;
  44. ++pr, ++pa, ++pb;
  45. }
  46. pr_end += x - m;
  47. // Now where only a has digits:
  48. while(pr != pr_end)
  49. {
  50. if(!carry)
  51. {
  52. if(pa != pr)
  53. #if BOOST_WORKAROUND(BOOST_MSVC, >= 1600)
  54. std::copy(pa, pa + (pr_end - pr), stdext::checked_array_iterator<limb_type*>(pr, result.size()));
  55. #else
  56. std::copy(pa, pa + (pr_end - pr), pr);
  57. #endif
  58. break;
  59. }
  60. carry += static_cast<double_limb_type>(*pa);
  61. *pr = static_cast<limb_type>(carry);
  62. carry >>= CppInt1::limb_bits;
  63. ++pr, ++pa;
  64. }
  65. if(carry)
  66. {
  67. // We overflowed, need to add one more limb:
  68. result.resize(x + 1, x + 1);
  69. if(CppInt1::variable || (result.size() > x))
  70. result.limbs()[x] = static_cast<limb_type>(carry);
  71. }
  72. result.normalize();
  73. result.sign(a.sign());
  74. }
  75. //
  76. // As above, but for adding a single limb to a non-trivial cpp_int:
  77. //
  78. template <class CppInt1, class CppInt2>
  79. inline void add_unsigned(CppInt1& result, const CppInt2& a, const limb_type& o) BOOST_MP_NOEXCEPT_IF(is_non_throwing_cpp_int<CppInt1>::value)
  80. {
  81. // Addition using modular arithmetic.
  82. // Nothing fancy, just let uintmax_t take the strain:
  83. if(&result != &a)
  84. result.resize(a.size(), a.size());
  85. double_limb_type carry = o;
  86. typename CppInt1::limb_pointer pr = result.limbs();
  87. typename CppInt2::const_limb_pointer pa = a.limbs();
  88. unsigned i = 0;
  89. // Addition with carry until we either run out of digits or carry is zero:
  90. for(; carry && (i < result.size()); ++i)
  91. {
  92. carry += static_cast<double_limb_type>(pa[i]);
  93. pr[i] = static_cast<limb_type>(carry);
  94. carry >>= CppInt1::limb_bits;
  95. }
  96. // Just copy any remaining digits:
  97. if(&a != &result)
  98. {
  99. for(; i < result.size(); ++i)
  100. pr[i] = pa[i];
  101. }
  102. if(carry)
  103. {
  104. // We overflowed, need to add one more limb:
  105. unsigned x = result.size();
  106. result.resize(x + 1, x + 1);
  107. if(CppInt1::variable || (result.size() > x))
  108. result.limbs()[x] = static_cast<limb_type>(carry);
  109. }
  110. result.normalize();
  111. result.sign(a.sign());
  112. }
  113. //
  114. // Core subtraction routine for all non-trivial cpp_int's:
  115. //
  116. template <class CppInt1, class CppInt2, class CppInt3>
  117. inline void subtract_unsigned(CppInt1& result, const CppInt2& a, const CppInt3& b) BOOST_MP_NOEXCEPT_IF(is_non_throwing_cpp_int<CppInt1>::value)
  118. {
  119. using std::swap;
  120. // Nothing fancy, just let uintmax_t take the strain:
  121. double_limb_type borrow = 0;
  122. unsigned m, x;
  123. minmax(a.size(), b.size(), m, x);
  124. //
  125. // special cases for small limb counts:
  126. //
  127. if(x == 1)
  128. {
  129. bool s = a.sign();
  130. limb_type al = *a.limbs();
  131. limb_type bl = *b.limbs();
  132. if(bl > al)
  133. {
  134. std::swap(al, bl);
  135. s = !s;
  136. }
  137. result = al - bl;
  138. result.sign(s);
  139. return;
  140. }
  141. // This isn't used till later, but comparison has to occur before we resize the result,
  142. // as that may also resize a or b if this is an inplace operation:
  143. int c = a.compare_unsigned(b);
  144. // Set up the result vector:
  145. result.resize(x, x);
  146. // Now that a, b, and result are stable, get pointers to their limbs:
  147. typename CppInt2::const_limb_pointer pa = a.limbs();
  148. typename CppInt3::const_limb_pointer pb = b.limbs();
  149. typename CppInt1::limb_pointer pr = result.limbs();
  150. bool swapped = false;
  151. if(c < 0)
  152. {
  153. swap(pa, pb);
  154. swapped = true;
  155. }
  156. else if(c == 0)
  157. {
  158. result = static_cast<limb_type>(0);
  159. return;
  160. }
  161. unsigned i = 0;
  162. // First where a and b overlap:
  163. while(i < m)
  164. {
  165. borrow = static_cast<double_limb_type>(pa[i]) - static_cast<double_limb_type>(pb[i]) - borrow;
  166. pr[i] = static_cast<limb_type>(borrow);
  167. borrow = (borrow >> CppInt1::limb_bits) & 1u;
  168. ++i;
  169. }
  170. // Now where only a has digits, only as long as we've borrowed:
  171. while(borrow && (i < x))
  172. {
  173. borrow = static_cast<double_limb_type>(pa[i]) - borrow;
  174. pr[i] = static_cast<limb_type>(borrow);
  175. borrow = (borrow >> CppInt1::limb_bits) & 1u;
  176. ++i;
  177. }
  178. // Any remaining digits are the same as those in pa:
  179. if((x != i) && (pa != pr))
  180. #if BOOST_WORKAROUND(BOOST_MSVC, >= 1600)
  181. std::copy(pa + i, pa + x, stdext::checked_array_iterator<limb_type*>(pr + i, result.size() - i));
  182. #else
  183. std::copy(pa + i, pa + x, pr + i);
  184. #endif
  185. BOOST_ASSERT(0 == borrow);
  186. //
  187. // We may have lost digits, if so update limb usage count:
  188. //
  189. result.normalize();
  190. result.sign(a.sign());
  191. if(swapped)
  192. result.negate();
  193. }
  194. //
  195. // And again to subtract a single limb:
  196. //
  197. template <class CppInt1, class CppInt2>
  198. inline void subtract_unsigned(CppInt1& result, const CppInt2& a, const limb_type& b) BOOST_MP_NOEXCEPT_IF(is_non_throwing_cpp_int<CppInt1>::value)
  199. {
  200. // Subtract one limb.
  201. // Nothing fancy, just let uintmax_t take the strain:
  202. BOOST_STATIC_CONSTANT(double_limb_type, borrow = static_cast<double_limb_type>(CppInt1::max_limb_value) + 1);
  203. result.resize(a.size(), a.size());
  204. typename CppInt1::limb_pointer pr = result.limbs();
  205. typename CppInt2::const_limb_pointer pa = a.limbs();
  206. if(*pa >= b)
  207. {
  208. *pr = *pa - b;
  209. if(&result != &a)
  210. {
  211. #if BOOST_WORKAROUND(BOOST_MSVC, >= 1600)
  212. std::copy(pa + 1, pa + a.size(), stdext::checked_array_iterator<limb_type*>(pr + 1, result.size() - 1));
  213. #else
  214. std::copy(pa + 1, pa + a.size(), pr + 1);
  215. #endif
  216. result.sign(a.sign());
  217. }
  218. else if((result.size() == 1) && (*pr == 0))
  219. {
  220. result.sign(false); // zero is unsigned.
  221. }
  222. }
  223. else if(result.size() == 1)
  224. {
  225. *pr = b - *pa;
  226. result.sign(!a.sign());
  227. }
  228. else
  229. {
  230. *pr = static_cast<limb_type>((borrow + *pa) - b);
  231. unsigned i = 1;
  232. while(!pa[i])
  233. {
  234. pr[i] = CppInt1::max_limb_value;
  235. ++i;
  236. }
  237. pr[i] = pa[i] - 1;
  238. if(&result != &a)
  239. {
  240. ++i;
  241. #if BOOST_WORKAROUND(BOOST_MSVC, >= 1600)
  242. std::copy(pa + i, pa + a.size(), stdext::checked_array_iterator<limb_type*>(pr + i, result.size() - i));
  243. #else
  244. std::copy(pa + i, pa + a.size(), pr + i);
  245. #endif
  246. }
  247. result.normalize();
  248. result.sign(a.sign());
  249. }
  250. }
  251. //
  252. // Now the actual functions called by the front end, all of which forward to one of the above:
  253. //
  254. 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>
  255. 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
  256. eval_add(
  257. cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& result,
  258. 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))
  259. {
  260. eval_add(result, result, o);
  261. }
  262. 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>
  263. 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
  264. eval_add(
  265. cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& result,
  266. const cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2>& a,
  267. 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))
  268. {
  269. if(a.sign() != b.sign())
  270. {
  271. subtract_unsigned(result, a, b);
  272. return;
  273. }
  274. add_unsigned(result, a, b);
  275. }
  276. template <unsigned MinBits1, unsigned MaxBits1, cpp_integer_type SignType1, cpp_int_check_type Checked1, class Allocator1>
  277. BOOST_MP_FORCEINLINE typename enable_if_c<!is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value>::type
  278. eval_add(cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& result, const limb_type& o) BOOST_MP_NOEXCEPT_IF((is_non_throwing_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value))
  279. {
  280. if(result.sign())
  281. {
  282. subtract_unsigned(result, result, o);
  283. }
  284. else
  285. add_unsigned(result, result, o);
  286. }
  287. 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>
  288. 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
  289. eval_add(
  290. cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& result,
  291. const cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2>& a,
  292. const limb_type& o) BOOST_MP_NOEXCEPT_IF((is_non_throwing_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value))
  293. {
  294. if(a.sign())
  295. {
  296. subtract_unsigned(result, a, o);
  297. }
  298. else
  299. add_unsigned(result, a, o);
  300. }
  301. template <unsigned MinBits1, unsigned MaxBits1, cpp_integer_type SignType1, cpp_int_check_type Checked1, class Allocator1>
  302. BOOST_MP_FORCEINLINE typename enable_if_c<!is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value>::type
  303. eval_add(
  304. cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& result,
  305. const signed_limb_type& o) BOOST_MP_NOEXCEPT_IF((is_non_throwing_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value))
  306. {
  307. if(o < 0)
  308. eval_subtract(result, static_cast<limb_type>(boost::multiprecision::detail::unsigned_abs(o)));
  309. else if(o > 0)
  310. eval_add(result, static_cast<limb_type>(o));
  311. }
  312. 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>
  313. 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
  314. eval_add(
  315. cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& result,
  316. const cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2>& a,
  317. const signed_limb_type& o) BOOST_MP_NOEXCEPT_IF((is_non_throwing_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value))
  318. {
  319. if(o < 0)
  320. eval_subtract(result, a, static_cast<limb_type>(boost::multiprecision::detail::unsigned_abs(o)));
  321. else if(o > 0)
  322. eval_add(result, a, static_cast<limb_type>(o));
  323. else if(&result != &a)
  324. result = a;
  325. }
  326. template <unsigned MinBits1, unsigned MaxBits1, cpp_integer_type SignType1, cpp_int_check_type Checked1, class Allocator1>
  327. BOOST_MP_FORCEINLINE typename enable_if_c<!is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value>::type
  328. eval_subtract(
  329. cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& result,
  330. const limb_type& o) BOOST_MP_NOEXCEPT_IF((is_non_throwing_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value))
  331. {
  332. if(result.sign())
  333. {
  334. add_unsigned(result, result, o);
  335. }
  336. else
  337. subtract_unsigned(result, result, o);
  338. }
  339. 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>
  340. 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
  341. eval_subtract(
  342. cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& result,
  343. const cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2>& a,
  344. const limb_type& o) BOOST_MP_NOEXCEPT_IF((is_non_throwing_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value))
  345. {
  346. if(a.sign())
  347. {
  348. add_unsigned(result, a, o);
  349. }
  350. else
  351. {
  352. subtract_unsigned(result, a, o);
  353. }
  354. }
  355. template <unsigned MinBits1, unsigned MaxBits1, cpp_integer_type SignType1, cpp_int_check_type Checked1, class Allocator1>
  356. BOOST_MP_FORCEINLINE typename enable_if_c<!is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value>::type
  357. eval_subtract(
  358. cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& result,
  359. const signed_limb_type& o) BOOST_MP_NOEXCEPT_IF((is_non_throwing_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value))
  360. {
  361. if(o)
  362. {
  363. if(o < 0)
  364. eval_add(result, static_cast<limb_type>(boost::multiprecision::detail::unsigned_abs(o)));
  365. else
  366. eval_subtract(result, static_cast<limb_type>(o));
  367. }
  368. }
  369. 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>
  370. 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
  371. eval_subtract(
  372. cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& result,
  373. const cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2>& a,
  374. const signed_limb_type& o) BOOST_MP_NOEXCEPT_IF((is_non_throwing_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value))
  375. {
  376. if(o)
  377. {
  378. if(o < 0)
  379. eval_add(result, a, static_cast<limb_type>(boost::multiprecision::detail::unsigned_abs(o)));
  380. else
  381. eval_subtract(result, a, static_cast<limb_type>(o));
  382. }
  383. else if(&result != &a)
  384. result = a;
  385. }
  386. template <unsigned MinBits1, unsigned MaxBits1, cpp_integer_type SignType1, cpp_int_check_type Checked1, class Allocator1>
  387. BOOST_MP_FORCEINLINE typename enable_if_c<!is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value>::type
  388. eval_increment(cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& result) BOOST_MP_NOEXCEPT_IF((is_non_throwing_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value))
  389. {
  390. static const limb_type one = 1;
  391. if(!result.sign() && (result.limbs()[0] < cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>::max_limb_value))
  392. ++result.limbs()[0];
  393. else if(result.sign() && result.limbs()[0])
  394. --result.limbs()[0];
  395. else
  396. eval_add(result, one);
  397. }
  398. template <unsigned MinBits1, unsigned MaxBits1, cpp_integer_type SignType1, cpp_int_check_type Checked1, class Allocator1>
  399. BOOST_MP_FORCEINLINE typename enable_if_c<!is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value>::type
  400. eval_decrement(cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& result) BOOST_MP_NOEXCEPT_IF((is_non_throwing_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value))
  401. {
  402. static const limb_type one = 1;
  403. if(!result.sign() && result.limbs()[0])
  404. --result.limbs()[0];
  405. else if(result.sign() && (result.limbs()[0] < cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>::max_limb_value))
  406. ++result.limbs()[0];
  407. else
  408. eval_subtract(result, one);
  409. }
  410. 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>
  411. 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
  412. eval_subtract(
  413. cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& result,
  414. 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))
  415. {
  416. eval_subtract(result, result, o);
  417. }
  418. 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>
  419. 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 && !is_trivial_cpp_int<cpp_int_backend<MinBits3, MaxBits3, SignType3, Checked3, Allocator3> >::value >::type
  420. eval_subtract(
  421. cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& result,
  422. const cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2>& a,
  423. 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))
  424. {
  425. if(a.sign() != b.sign())
  426. {
  427. add_unsigned(result, a, b);
  428. return;
  429. }
  430. subtract_unsigned(result, a, b);
  431. }
  432. //
  433. // Simple addition and subtraction routine for trivial cpp_int's come last:
  434. //
  435. // One of the arguments is signed:
  436. //
  437. template <unsigned MinBits1, unsigned MaxBits1, cpp_integer_type SignType1, cpp_int_check_type Checked1, class Allocator1>
  438. inline typename enable_if_c<
  439. is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value
  440. && is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value
  441. && (is_signed_number<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value || is_signed_number<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value)
  442. >::type
  443. eval_add(
  444. cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& result,
  445. 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))
  446. {
  447. if(result.sign() != o.sign())
  448. {
  449. if(*o.limbs() > *result.limbs())
  450. {
  451. *result.limbs() = detail::checked_subtract(*o.limbs(), *result.limbs(), typename cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>::checked_type());
  452. result.negate();
  453. }
  454. else
  455. *result.limbs() = detail::checked_subtract(*result.limbs(), *o.limbs(), typename cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>::checked_type());
  456. }
  457. else
  458. *result.limbs() = detail::checked_add(*result.limbs(), *o.limbs(), typename cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>::checked_type());
  459. result.normalize();
  460. }
  461. // Simple version for two unsigned arguments:
  462. template <unsigned MinBits1, unsigned MaxBits1, cpp_integer_type SignType1, cpp_int_check_type Checked1, class Allocator1>
  463. BOOST_MP_FORCEINLINE typename enable_if_c<
  464. is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value
  465. && is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value
  466. && is_unsigned_number<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value
  467. && is_unsigned_number<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value
  468. >::type
  469. eval_add(cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& result,
  470. 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))
  471. {
  472. *result.limbs() = detail::checked_add(*result.limbs(), *o.limbs(), typename cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>::checked_type());
  473. result.normalize();
  474. }
  475. // signed subtraction:
  476. template <unsigned MinBits1, unsigned MaxBits1, cpp_integer_type SignType1, cpp_int_check_type Checked1, class Allocator1>
  477. inline typename enable_if_c<
  478. is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value
  479. && is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value
  480. && (is_signed_number<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value || is_signed_number<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value)
  481. >::type
  482. eval_subtract(
  483. cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& result,
  484. 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))
  485. {
  486. if(result.sign() != o.sign())
  487. {
  488. *result.limbs() = detail::checked_add(*result.limbs(), *o.limbs(), typename cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>::checked_type());
  489. }
  490. else if(*result.limbs() < *o.limbs())
  491. {
  492. *result.limbs() = detail::checked_subtract(*o.limbs(), *result.limbs(), typename cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>::checked_type());
  493. result.negate();
  494. }
  495. else
  496. *result.limbs() = detail::checked_subtract(*result.limbs(), *o.limbs(), typename cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>::checked_type());
  497. result.normalize();
  498. }
  499. template <unsigned MinBits1, unsigned MaxBits1, cpp_integer_type SignType1, cpp_int_check_type Checked1, class Allocator1>
  500. BOOST_MP_FORCEINLINE typename enable_if_c<
  501. is_trivial_cpp_int<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value
  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. && is_unsigned_number<cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1> >::value
  505. >::type
  506. eval_subtract(
  507. cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>& result,
  508. 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))
  509. {
  510. *result.limbs() = detail::checked_subtract(*result.limbs(), *o.limbs(), typename cpp_int_backend<MinBits1, MaxBits1, SignType1, Checked1, Allocator1>::checked_type());
  511. result.normalize();
  512. }
  513. }}} // namespaces
  514. #endif