cpp_int.hpp 82 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944
  1. //////////////////3/////////////////////////////////////////////
  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. #ifndef BOOST_MP_CPP_INT_HPP
  6. #define BOOST_MP_CPP_INT_HPP
  7. #include <iostream>
  8. #include <iomanip>
  9. #include <boost/cstdint.hpp>
  10. #include <boost/multiprecision/number.hpp>
  11. #include <boost/multiprecision/detail/integer_ops.hpp>
  12. #include <boost/array.hpp>
  13. #include <boost/type_traits/is_integral.hpp>
  14. #include <boost/type_traits/is_floating_point.hpp>
  15. #include <boost/multiprecision/cpp_int/cpp_int_config.hpp>
  16. #include <boost/multiprecision/rational_adaptor.hpp>
  17. #include <boost/multiprecision/traits/is_byte_container.hpp>
  18. #include <boost/detail/endian.hpp>
  19. #include <boost/integer/static_min_max.hpp>
  20. #include <boost/type_traits/common_type.hpp>
  21. #include <boost/type_traits/make_signed.hpp>
  22. #include <boost/multiprecision/cpp_int/checked.hpp>
  23. #ifdef BOOST_MP_USER_DEFINED_LITERALS
  24. #include <boost/multiprecision/cpp_int/value_pack.hpp>
  25. #endif
  26. namespace boost{
  27. namespace multiprecision{
  28. namespace backends{
  29. using boost::enable_if;
  30. #ifdef BOOST_MSVC
  31. // warning C4127: conditional expression is constant
  32. #pragma warning(push)
  33. #pragma warning(disable:4127 4351 4293 4996 4307 4702 6285)
  34. #endif
  35. template <unsigned MinBits = 0, unsigned MaxBits = 0, boost::multiprecision::cpp_integer_type SignType = signed_magnitude, cpp_int_check_type Checked = unchecked, class Allocator = typename mpl::if_c<MinBits && (MinBits == MaxBits), void, std::allocator<limb_type> >::type >
  36. struct cpp_int_backend;
  37. } // namespace backends
  38. namespace detail {
  39. template <unsigned MinBits, unsigned MaxBits, boost::multiprecision::cpp_integer_type SignType, cpp_int_check_type Checked, class Allocator>
  40. struct is_byte_container<backends::cpp_int_backend<MinBits, MaxBits, SignType, Checked, Allocator> > : public boost::false_type {};
  41. } // namespace detail
  42. namespace backends{
  43. template <unsigned MinBits, unsigned MaxBits, cpp_integer_type SignType, cpp_int_check_type Checked, class Allocator, bool trivial = false>
  44. struct cpp_int_base;
  45. //
  46. // Traits class determines the maximum and minimum precision values:
  47. //
  48. template <class T> struct max_precision;
  49. template <unsigned MinBits, unsigned MaxBits, cpp_integer_type SignType, cpp_int_check_type Checked, class Allocator>
  50. struct max_precision<cpp_int_backend<MinBits, MaxBits, SignType, Checked, Allocator> >
  51. {
  52. static const unsigned value = is_void<Allocator>::value ?
  53. static_unsigned_max<MinBits, MaxBits>::value
  54. : (((MaxBits >= MinBits) && MaxBits) ? MaxBits : UINT_MAX);
  55. };
  56. template <class T> struct min_precision;
  57. template <unsigned MinBits, unsigned MaxBits, cpp_integer_type SignType, cpp_int_check_type Checked, class Allocator>
  58. struct min_precision<cpp_int_backend<MinBits, MaxBits, SignType, Checked, Allocator> >
  59. {
  60. static const unsigned value = (is_void<Allocator>::value ? static_unsigned_max<MinBits, MaxBits>::value : MinBits);
  61. };
  62. //
  63. // Traits class determines whether the number of bits precision requested could fit in a native type,
  64. // we call this a "trivial" cpp_int:
  65. //
  66. template <class T>
  67. struct is_trivial_cpp_int
  68. {
  69. static const bool value = false;
  70. };
  71. template <unsigned MinBits, unsigned MaxBits, cpp_integer_type SignType, cpp_int_check_type Checked, class Allocator>
  72. struct is_trivial_cpp_int<cpp_int_backend<MinBits, MaxBits, SignType, Checked, Allocator> >
  73. {
  74. typedef cpp_int_backend<MinBits, MaxBits, SignType, Checked, Allocator> self;
  75. static const bool value = is_void<Allocator>::value && (max_precision<self>::value <= (sizeof(double_limb_type) * CHAR_BIT) - (SignType == signed_packed ? 1 : 0));
  76. };
  77. template <unsigned MinBits, unsigned MaxBits, cpp_integer_type SignType, cpp_int_check_type Checked, class Allocator>
  78. struct is_trivial_cpp_int<cpp_int_base<MinBits, MaxBits, SignType, Checked, Allocator, true> >
  79. {
  80. static const bool value = true;
  81. };
  82. } // namespace backends
  83. //
  84. // Traits class to determine whether a cpp_int_backend is signed or not:
  85. //
  86. template <unsigned MinBits, unsigned MaxBits, cpp_integer_type SignType, cpp_int_check_type Checked, class Allocator>
  87. struct is_unsigned_number<backends::cpp_int_backend<MinBits, MaxBits, SignType, Checked, Allocator> >
  88. : public mpl::bool_<(SignType == unsigned_magnitude) || (SignType == unsigned_packed)>{};
  89. namespace backends{
  90. //
  91. // Traits class determines whether T should be implicitly convertible to U, or
  92. // whether the constructor should be made explicit. The latter happens if we
  93. // are losing the sign, or have fewer digits precision in the target type:
  94. //
  95. template <class T, class U>
  96. struct is_implicit_cpp_int_conversion;
  97. template <unsigned MinBits, unsigned MaxBits, cpp_integer_type SignType, cpp_int_check_type Checked, class Allocator, unsigned MinBits2, unsigned MaxBits2, cpp_integer_type SignType2, cpp_int_check_type Checked2, class Allocator2>
  98. struct is_implicit_cpp_int_conversion<cpp_int_backend<MinBits, MaxBits, SignType, Checked, Allocator>, cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2> >
  99. {
  100. typedef cpp_int_backend<MinBits, MaxBits, SignType, Checked, Allocator> t1;
  101. typedef cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2> t2;
  102. static const bool value =
  103. (is_signed_number<t2>::value || !is_signed_number<t1>::value)
  104. && (max_precision<t1>::value <= max_precision<t2>::value);
  105. };
  106. //
  107. // Traits class to determine whether operations on a cpp_int may throw:
  108. //
  109. template <class T>
  110. struct is_non_throwing_cpp_int : public mpl::false_{};
  111. template <unsigned MinBits, unsigned MaxBits, cpp_integer_type SignType>
  112. struct is_non_throwing_cpp_int<cpp_int_backend<MinBits, MaxBits, SignType, unchecked, void> > : public mpl::true_ {};
  113. //
  114. // Traits class, determines whether the cpp_int is fixed precision or not:
  115. //
  116. template <class T>
  117. struct is_fixed_precision;
  118. template <unsigned MinBits, unsigned MaxBits, cpp_integer_type SignType, cpp_int_check_type Checked, class Allocator>
  119. struct is_fixed_precision<cpp_int_backend<MinBits, MaxBits, SignType, Checked, Allocator> >
  120. : public mpl::bool_<max_precision<cpp_int_backend<MinBits, MaxBits, SignType, Checked, Allocator> >::value != UINT_MAX> {};
  121. namespace detail{
  122. inline void verify_new_size(unsigned new_size, unsigned min_size, const mpl::int_<checked>&)
  123. {
  124. if(new_size < min_size)
  125. BOOST_THROW_EXCEPTION(std::overflow_error("Unable to allocate sufficient storage for the value of the result: value overflows the maximum allowable magnitude."));
  126. }
  127. inline void verify_new_size(unsigned /*new_size*/, unsigned /*min_size*/, const mpl::int_<unchecked>&){}
  128. template <class U>
  129. inline void verify_limb_mask(bool b, U limb, U mask, const mpl::int_<checked>&)
  130. {
  131. // When we mask out "limb" with "mask", do we loose bits? If so it's an overflow error:
  132. if(b && (limb & ~mask))
  133. BOOST_THROW_EXCEPTION(std::overflow_error("Overflow in cpp_int arithmetic: there is insufficient precision in the target type to hold all of the bits of the result."));
  134. }
  135. template <class U>
  136. inline void verify_limb_mask(bool /*b*/, U /*limb*/, U /*mask*/, const mpl::int_<unchecked>&){}
  137. }
  138. //
  139. // Now define the various data layouts that are possible as partial specializations of the base class,
  140. // starting with the default arbitrary precision signed integer type:
  141. //
  142. template <unsigned MinBits, unsigned MaxBits, cpp_int_check_type Checked, class Allocator>
  143. struct cpp_int_base<MinBits, MaxBits, signed_magnitude, Checked, Allocator, false> : private Allocator::template rebind<limb_type>::other
  144. {
  145. typedef typename Allocator::template rebind<limb_type>::other allocator_type;
  146. typedef typename allocator_type::pointer limb_pointer;
  147. typedef typename allocator_type::const_pointer const_limb_pointer;
  148. typedef mpl::int_<Checked> checked_type;
  149. //
  150. // Interface invariants:
  151. //
  152. BOOST_STATIC_ASSERT(!is_void<Allocator>::value);
  153. private:
  154. struct limb_data
  155. {
  156. unsigned capacity;
  157. limb_pointer data;
  158. };
  159. public:
  160. BOOST_STATIC_CONSTANT(unsigned, limb_bits = sizeof(limb_type) * CHAR_BIT);
  161. BOOST_STATIC_CONSTANT(limb_type, max_limb_value = ~static_cast<limb_type>(0u));
  162. BOOST_STATIC_CONSTANT(limb_type, sign_bit_mask = static_cast<limb_type>(1u) << (limb_bits - 1));
  163. BOOST_STATIC_CONSTANT(unsigned, internal_limb_count =
  164. MinBits
  165. ? (MinBits / limb_bits + ((MinBits % limb_bits) ? 1 : 0))
  166. : (sizeof(limb_data) / sizeof(limb_type)));
  167. BOOST_STATIC_CONSTANT(bool, variable = true);
  168. private:
  169. union data_type
  170. {
  171. limb_data ld;
  172. limb_type la[internal_limb_count];
  173. limb_type first;
  174. double_limb_type double_first;
  175. BOOST_CONSTEXPR data_type() : first(0) {}
  176. BOOST_CONSTEXPR data_type(limb_type i) : first(i) {}
  177. BOOST_CONSTEXPR data_type(signed_limb_type i) : first(i < 0 ? static_cast<limb_type>(boost::multiprecision::detail::unsigned_abs(i)) : i) {}
  178. #ifdef BOOST_LITTLE_ENDIAN
  179. BOOST_CONSTEXPR data_type(double_limb_type i) : double_first(i) {}
  180. BOOST_CONSTEXPR data_type(signed_double_limb_type i) : double_first(i < 0 ? static_cast<double_limb_type>(boost::multiprecision::detail::unsigned_abs(i)) : i) {}
  181. #endif
  182. };
  183. data_type m_data;
  184. unsigned m_limbs;
  185. bool m_sign, m_internal;
  186. public:
  187. //
  188. // Direct construction:
  189. //
  190. BOOST_MP_FORCEINLINE BOOST_CONSTEXPR cpp_int_base(limb_type i)BOOST_NOEXCEPT
  191. : m_data(i), m_limbs(1), m_sign(false), m_internal(true) { }
  192. BOOST_MP_FORCEINLINE BOOST_CONSTEXPR cpp_int_base(signed_limb_type i)BOOST_NOEXCEPT
  193. : m_data(i), m_limbs(1), m_sign(i < 0), m_internal(true) { }
  194. #if defined(BOOST_LITTLE_ENDIAN)
  195. BOOST_MP_FORCEINLINE BOOST_CONSTEXPR cpp_int_base(double_limb_type i)BOOST_NOEXCEPT
  196. : m_data(i), m_limbs(i > max_limb_value ? 2 : 1), m_sign(false), m_internal(true) { }
  197. BOOST_MP_FORCEINLINE BOOST_CONSTEXPR cpp_int_base(signed_double_limb_type i)BOOST_NOEXCEPT
  198. : m_data(i), m_limbs(i < 0 ? (static_cast<double_limb_type>(boost::multiprecision::detail::unsigned_abs(i)) > static_cast<double_limb_type>(max_limb_value) ? 2 : 1) : (i > max_limb_value ? 2 : 1)),
  199. m_sign(i < 0), m_internal(true) { }
  200. #endif
  201. //
  202. // Helper functions for getting at our internal data, and manipulating storage:
  203. //
  204. BOOST_MP_FORCEINLINE allocator_type& allocator() BOOST_NOEXCEPT { return *this; }
  205. BOOST_MP_FORCEINLINE const allocator_type& allocator()const BOOST_NOEXCEPT { return *this; }
  206. BOOST_MP_FORCEINLINE unsigned size()const BOOST_NOEXCEPT { return m_limbs; }
  207. BOOST_MP_FORCEINLINE limb_pointer limbs() BOOST_NOEXCEPT { return m_internal ? m_data.la : m_data.ld.data; }
  208. BOOST_MP_FORCEINLINE const_limb_pointer limbs()const BOOST_NOEXCEPT { return m_internal ? m_data.la : m_data.ld.data; }
  209. BOOST_MP_FORCEINLINE unsigned capacity()const BOOST_NOEXCEPT { return m_internal ? internal_limb_count : m_data.ld.capacity; }
  210. BOOST_MP_FORCEINLINE bool sign()const BOOST_NOEXCEPT { return m_sign; }
  211. void sign(bool b) BOOST_NOEXCEPT
  212. {
  213. m_sign = b;
  214. // Check for zero value:
  215. if(m_sign && (m_limbs == 1))
  216. {
  217. if(limbs()[0] == 0)
  218. m_sign = false;
  219. }
  220. }
  221. void resize(unsigned new_size, unsigned min_size)
  222. {
  223. static const unsigned max_limbs = MaxBits / (CHAR_BIT * sizeof(limb_type)) + ((MaxBits % (CHAR_BIT * sizeof(limb_type))) ? 1 : 0);
  224. // We never resize beyond MaxSize:
  225. if(new_size > max_limbs)
  226. new_size = max_limbs;
  227. detail::verify_new_size(new_size, min_size, checked_type());
  228. // See if we have enough capacity already:
  229. unsigned cap = capacity();
  230. if(new_size > cap)
  231. {
  232. // Allocate a new buffer and copy everything over:
  233. cap = (std::min)((std::max)(cap * 4, new_size), max_limbs);
  234. limb_pointer pl = allocator().allocate(cap);
  235. std::memcpy(pl, limbs(), size() * sizeof(limbs()[0]));
  236. if(!m_internal)
  237. allocator().deallocate(limbs(), capacity());
  238. else
  239. m_internal = false;
  240. m_limbs = new_size;
  241. m_data.ld.capacity = cap;
  242. m_data.ld.data = pl;
  243. }
  244. else
  245. {
  246. m_limbs = new_size;
  247. }
  248. }
  249. BOOST_MP_FORCEINLINE void normalize() BOOST_NOEXCEPT
  250. {
  251. limb_pointer p = limbs();
  252. while((m_limbs-1) && !p[m_limbs - 1])--m_limbs;
  253. }
  254. BOOST_MP_FORCEINLINE BOOST_CONSTEXPR cpp_int_base() BOOST_NOEXCEPT : m_data(), m_limbs(1), m_sign(false), m_internal(true) {}
  255. BOOST_MP_FORCEINLINE cpp_int_base(const cpp_int_base& o) : allocator_type(o), m_limbs(0), m_internal(true)
  256. {
  257. resize(o.size(), o.size());
  258. std::memcpy(limbs(), o.limbs(), o.size() * sizeof(limbs()[0]));
  259. m_sign = o.m_sign;
  260. }
  261. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  262. cpp_int_base(cpp_int_base&& o)
  263. : allocator_type(static_cast<allocator_type&&>(o)), m_limbs(o.m_limbs), m_sign(o.m_sign), m_internal(o.m_internal)
  264. {
  265. if(m_internal)
  266. {
  267. std::memcpy(limbs(), o.limbs(), o.size() * sizeof(limbs()[0]));
  268. }
  269. else
  270. {
  271. m_data.ld = o.m_data.ld;
  272. o.m_limbs = 0;
  273. o.m_internal = true;
  274. }
  275. }
  276. cpp_int_base& operator = (cpp_int_base&& o) BOOST_NOEXCEPT
  277. {
  278. if(!m_internal)
  279. allocator().deallocate(m_data.ld.data, m_data.ld.capacity);
  280. *static_cast<allocator_type*>(this) = static_cast<allocator_type&&>(o);
  281. m_limbs = o.m_limbs;
  282. m_sign = o.m_sign;
  283. m_internal = o.m_internal;
  284. if(m_internal)
  285. {
  286. std::memcpy(limbs(), o.limbs(), o.size() * sizeof(limbs()[0]));
  287. }
  288. else
  289. {
  290. m_data.ld = o.m_data.ld;
  291. o.m_limbs = 0;
  292. o.m_internal = true;
  293. }
  294. return *this;
  295. }
  296. #endif
  297. BOOST_MP_FORCEINLINE ~cpp_int_base() BOOST_NOEXCEPT
  298. {
  299. if(!m_internal)
  300. allocator().deallocate(limbs(), capacity());
  301. }
  302. void assign(const cpp_int_base& o)
  303. {
  304. if(this != &o)
  305. {
  306. static_cast<allocator_type&>(*this) = static_cast<const allocator_type&>(o);
  307. m_limbs = 0;
  308. resize(o.size(), o.size());
  309. std::memcpy(limbs(), o.limbs(), o.size() * sizeof(limbs()[0]));
  310. m_sign = o.m_sign;
  311. }
  312. }
  313. BOOST_MP_FORCEINLINE void negate() BOOST_NOEXCEPT
  314. {
  315. m_sign = !m_sign;
  316. // Check for zero value:
  317. if(m_sign && (m_limbs == 1))
  318. {
  319. if(limbs()[0] == 0)
  320. m_sign = false;
  321. }
  322. }
  323. BOOST_MP_FORCEINLINE bool isneg()const BOOST_NOEXCEPT
  324. {
  325. return m_sign;
  326. }
  327. BOOST_MP_FORCEINLINE void do_swap(cpp_int_base& o) BOOST_NOEXCEPT
  328. {
  329. std::swap(m_data, o.m_data);
  330. std::swap(m_sign, o.m_sign);
  331. std::swap(m_internal, o.m_internal);
  332. std::swap(m_limbs, o.m_limbs);
  333. }
  334. protected:
  335. template <class A>
  336. void check_in_range(const A&) BOOST_NOEXCEPT {}
  337. };
  338. #ifndef BOOST_NO_INCLASS_MEMBER_INITIALIZATION
  339. template <unsigned MinBits, unsigned MaxBits, cpp_int_check_type Checked, class Allocator>
  340. const unsigned cpp_int_base<MinBits, MaxBits, signed_magnitude, Checked, Allocator, false>::limb_bits;
  341. template <unsigned MinBits, unsigned MaxBits, cpp_int_check_type Checked, class Allocator>
  342. const limb_type cpp_int_base<MinBits, MaxBits, signed_magnitude, Checked, Allocator, false>::max_limb_value;
  343. template <unsigned MinBits, unsigned MaxBits, cpp_int_check_type Checked, class Allocator>
  344. const limb_type cpp_int_base<MinBits, MaxBits, signed_magnitude, Checked, Allocator, false>::sign_bit_mask;
  345. template <unsigned MinBits, unsigned MaxBits, cpp_int_check_type Checked, class Allocator>
  346. const unsigned cpp_int_base<MinBits, MaxBits, signed_magnitude, Checked, Allocator, false>::internal_limb_count;
  347. template <unsigned MinBits, unsigned MaxBits, cpp_int_check_type Checked, class Allocator>
  348. const bool cpp_int_base<MinBits, MaxBits, signed_magnitude, Checked, Allocator, false>::variable;
  349. #endif
  350. template <unsigned MinBits, unsigned MaxBits, cpp_int_check_type Checked, class Allocator>
  351. struct cpp_int_base<MinBits, MaxBits, unsigned_magnitude, Checked, Allocator, false> : private Allocator::template rebind<limb_type>::other
  352. {
  353. //
  354. // There is currently no support for unsigned arbitrary precision arithmetic, largely
  355. // because it's not clear what subtraction should do:
  356. //
  357. BOOST_STATIC_ASSERT_MSG(((sizeof(Allocator) == 0) && !is_void<Allocator>::value), "There is curently no support for unsigned arbitrary precision integers.");
  358. };
  359. //
  360. // Fixed precision (i.e. no allocator), signed-magnitude type with limb-usage count:
  361. //
  362. template <unsigned MinBits, cpp_int_check_type Checked>
  363. struct cpp_int_base<MinBits, MinBits, signed_magnitude, Checked, void, false>
  364. {
  365. typedef limb_type* limb_pointer;
  366. typedef const limb_type* const_limb_pointer;
  367. typedef mpl::int_<Checked> checked_type;
  368. //
  369. // Interface invariants:
  370. //
  371. BOOST_STATIC_ASSERT_MSG(MinBits > sizeof(double_limb_type) * CHAR_BIT, "Template parameter MinBits is inconsistent with the parameter trivial - did you mistakingly try to override the trivial parameter?");
  372. public:
  373. BOOST_STATIC_CONSTANT(unsigned, limb_bits = sizeof(limb_type) * CHAR_BIT);
  374. BOOST_STATIC_CONSTANT(limb_type, max_limb_value = ~static_cast<limb_type>(0u));
  375. BOOST_STATIC_CONSTANT(limb_type, sign_bit_mask = static_cast<limb_type>(1u) << (limb_bits - 1));
  376. BOOST_STATIC_CONSTANT(unsigned, internal_limb_count = MinBits / limb_bits + ((MinBits % limb_bits) ? 1 : 0));
  377. BOOST_STATIC_CONSTANT(bool, variable = false);
  378. BOOST_STATIC_CONSTANT(limb_type, upper_limb_mask = (MinBits % limb_bits) ? (limb_type(1) << (MinBits % limb_bits)) -1 : (~limb_type(0)));
  379. BOOST_STATIC_ASSERT_MSG(internal_limb_count >= 2, "A fixed precision integer type must have at least 2 limbs");
  380. private:
  381. union data_type{
  382. limb_type m_data[internal_limb_count];
  383. limb_type m_first_limb;
  384. double_limb_type m_double_first_limb;
  385. BOOST_CONSTEXPR data_type() : m_first_limb(0) {}
  386. BOOST_CONSTEXPR data_type(limb_type i) : m_first_limb(i) {}
  387. BOOST_CONSTEXPR data_type(double_limb_type i) : m_double_first_limb(i) {}
  388. #if defined(BOOST_MP_USER_DEFINED_LITERALS)
  389. template <limb_type...VALUES>
  390. BOOST_CONSTEXPR data_type(literals::detail::value_pack<VALUES...>) : m_data{ VALUES... } {}
  391. #endif
  392. } m_wrapper;
  393. boost::uint16_t m_limbs;
  394. bool m_sign;
  395. public:
  396. //
  397. // Direct construction:
  398. //
  399. BOOST_MP_FORCEINLINE BOOST_CONSTEXPR cpp_int_base(limb_type i)BOOST_NOEXCEPT
  400. : m_wrapper(i), m_limbs(1), m_sign(false) {}
  401. BOOST_MP_FORCEINLINE BOOST_CONSTEXPR cpp_int_base(signed_limb_type i)BOOST_NOEXCEPT
  402. : m_wrapper(limb_type(i < 0 ? static_cast<limb_type>(-static_cast<signed_double_limb_type>(i)) : i)), m_limbs(1), m_sign(i < 0) {}
  403. #if defined(BOOST_LITTLE_ENDIAN)
  404. BOOST_MP_FORCEINLINE BOOST_CONSTEXPR cpp_int_base(double_limb_type i)BOOST_NOEXCEPT
  405. : m_wrapper(i), m_limbs(i > max_limb_value ? 2 : 1), m_sign(false) {}
  406. BOOST_MP_FORCEINLINE BOOST_CONSTEXPR cpp_int_base(signed_double_limb_type i)BOOST_NOEXCEPT
  407. : m_wrapper(double_limb_type(i < 0 ? static_cast<double_limb_type>(boost::multiprecision::detail::unsigned_abs(i)) : i)),
  408. m_limbs(i < 0 ? (static_cast<double_limb_type>(boost::multiprecision::detail::unsigned_abs(i)) > max_limb_value ? 2 : 1) : (i > max_limb_value ? 2 : 1)),
  409. m_sign(i < 0) {}
  410. #endif
  411. #if defined(BOOST_MP_USER_DEFINED_LITERALS)
  412. template <limb_type...VALUES>
  413. BOOST_CONSTEXPR cpp_int_base(literals::detail::value_pack<VALUES...> i)
  414. : m_wrapper(i), m_limbs(sizeof...(VALUES)), m_sign(false) {}
  415. BOOST_CONSTEXPR cpp_int_base(literals::detail::value_pack<> i)
  416. : m_wrapper(i), m_limbs(1), m_sign(false) {}
  417. BOOST_CONSTEXPR cpp_int_base(const cpp_int_base& a, const literals::detail::negate_tag&)
  418. : m_wrapper(a.m_wrapper), m_limbs(a.m_limbs), m_sign((a.m_limbs == 1) && (*a.limbs() == 0) ? false : !a.m_sign) {}
  419. #endif
  420. //
  421. // Helper functions for getting at our internal data, and manipulating storage:
  422. //
  423. BOOST_MP_FORCEINLINE unsigned size()const BOOST_NOEXCEPT { return m_limbs; }
  424. BOOST_MP_FORCEINLINE limb_pointer limbs() BOOST_NOEXCEPT { return m_wrapper.m_data; }
  425. BOOST_MP_FORCEINLINE BOOST_CONSTEXPR const_limb_pointer limbs()const BOOST_NOEXCEPT { return m_wrapper.m_data; }
  426. BOOST_MP_FORCEINLINE BOOST_CONSTEXPR bool sign()const BOOST_NOEXCEPT { return m_sign; }
  427. BOOST_MP_FORCEINLINE void sign(bool b) BOOST_NOEXCEPT
  428. {
  429. m_sign = b;
  430. // Check for zero value:
  431. if(m_sign && (m_limbs == 1))
  432. {
  433. if(limbs()[0] == 0)
  434. m_sign = false;
  435. }
  436. }
  437. BOOST_MP_FORCEINLINE void resize(unsigned new_size, unsigned min_size) BOOST_MP_NOEXCEPT_IF((Checked == unchecked))
  438. {
  439. m_limbs = static_cast<boost::uint16_t>((std::min)(new_size, internal_limb_count));
  440. detail::verify_new_size(m_limbs, min_size, checked_type());
  441. }
  442. BOOST_MP_FORCEINLINE void normalize() BOOST_MP_NOEXCEPT_IF((Checked == unchecked))
  443. {
  444. limb_pointer p = limbs();
  445. detail::verify_limb_mask(m_limbs == internal_limb_count, p[internal_limb_count-1], upper_limb_mask, checked_type());
  446. p[internal_limb_count-1] &= upper_limb_mask;
  447. while((m_limbs-1) && !p[m_limbs - 1])--m_limbs;
  448. if((m_limbs == 1) && (!*p)) m_sign = false; // zero is always unsigned
  449. }
  450. BOOST_MP_FORCEINLINE BOOST_CONSTEXPR cpp_int_base()BOOST_NOEXCEPT : m_wrapper(limb_type(0u)), m_limbs(1), m_sign(false) {}
  451. // Not defaulted, it breaks constexpr support in the Intel compiler for some reason:
  452. BOOST_MP_FORCEINLINE BOOST_CONSTEXPR cpp_int_base(const cpp_int_base& o)BOOST_NOEXCEPT
  453. : m_wrapper(o.m_wrapper), m_limbs(o.m_limbs), m_sign(o.m_sign) {}
  454. // Defaulted functions:
  455. //~cpp_int_base() BOOST_NOEXCEPT {}
  456. void assign(const cpp_int_base& o) BOOST_NOEXCEPT
  457. {
  458. if(this != &o)
  459. {
  460. m_limbs = o.m_limbs;
  461. std::memcpy(limbs(), o.limbs(), o.size() * sizeof(o.limbs()[0]));
  462. m_sign = o.m_sign;
  463. }
  464. }
  465. BOOST_MP_FORCEINLINE void negate() BOOST_NOEXCEPT
  466. {
  467. m_sign = !m_sign;
  468. // Check for zero value:
  469. if(m_sign && (m_limbs == 1))
  470. {
  471. if(limbs()[0] == 0)
  472. m_sign = false;
  473. }
  474. }
  475. BOOST_MP_FORCEINLINE bool isneg()const BOOST_NOEXCEPT
  476. {
  477. return m_sign;
  478. }
  479. BOOST_MP_FORCEINLINE void do_swap(cpp_int_base& o) BOOST_NOEXCEPT
  480. {
  481. for(unsigned i = 0; i < (std::max)(size(), o.size()); ++i)
  482. std::swap(m_wrapper.m_data[i], o.m_wrapper.m_data[i]);
  483. std::swap(m_sign, o.m_sign);
  484. std::swap(m_limbs, o.m_limbs);
  485. }
  486. protected:
  487. template <class A>
  488. void check_in_range(const A&) BOOST_NOEXCEPT {}
  489. };
  490. #ifndef BOOST_NO_INCLASS_MEMBER_INITIALIZATION
  491. template <unsigned MinBits, cpp_int_check_type Checked>
  492. const unsigned cpp_int_base<MinBits, MinBits, signed_magnitude, Checked, void, false>::limb_bits;
  493. template <unsigned MinBits, cpp_int_check_type Checked>
  494. const limb_type cpp_int_base<MinBits, MinBits, signed_magnitude, Checked, void, false>::max_limb_value;
  495. template <unsigned MinBits, cpp_int_check_type Checked>
  496. const limb_type cpp_int_base<MinBits, MinBits, signed_magnitude, Checked, void, false>::sign_bit_mask;
  497. template <unsigned MinBits, cpp_int_check_type Checked>
  498. const unsigned cpp_int_base<MinBits, MinBits, signed_magnitude, Checked, void, false>::internal_limb_count;
  499. template <unsigned MinBits, cpp_int_check_type Checked>
  500. const bool cpp_int_base<MinBits, MinBits, signed_magnitude, Checked, void, false>::variable;
  501. #endif
  502. //
  503. // Fixed precision (i.e. no allocator), unsigned type with limb-usage count:
  504. //
  505. template <unsigned MinBits, cpp_int_check_type Checked>
  506. struct cpp_int_base<MinBits, MinBits, unsigned_magnitude, Checked, void, false>
  507. {
  508. typedef limb_type* limb_pointer;
  509. typedef const limb_type* const_limb_pointer;
  510. typedef mpl::int_<Checked> checked_type;
  511. //
  512. // Interface invariants:
  513. //
  514. BOOST_STATIC_ASSERT_MSG(MinBits > sizeof(double_limb_type) * CHAR_BIT, "Template parameter MinBits is inconsistent with the parameter trivial - did you mistakingly try to override the trivial parameter?");
  515. public:
  516. BOOST_STATIC_CONSTANT(unsigned, limb_bits = sizeof(limb_type) * CHAR_BIT);
  517. BOOST_STATIC_CONSTANT(limb_type, max_limb_value = ~static_cast<limb_type>(0u));
  518. BOOST_STATIC_CONSTANT(limb_type, sign_bit_mask = static_cast<limb_type>(1u) << (limb_bits - 1));
  519. BOOST_STATIC_CONSTANT(unsigned, internal_limb_count = MinBits / limb_bits + ((MinBits % limb_bits) ? 1 : 0));
  520. BOOST_STATIC_CONSTANT(bool, variable = false);
  521. BOOST_STATIC_CONSTANT(limb_type, upper_limb_mask = (MinBits % limb_bits) ? (limb_type(1) << (MinBits % limb_bits)) -1 : (~limb_type(0)));
  522. BOOST_STATIC_ASSERT_MSG(internal_limb_count >= 2, "A fixed precision integer type must have at least 2 limbs");
  523. private:
  524. union data_type{
  525. limb_type m_data[internal_limb_count];
  526. limb_type m_first_limb;
  527. double_limb_type m_double_first_limb;
  528. BOOST_CONSTEXPR data_type() : m_first_limb(0) {}
  529. BOOST_CONSTEXPR data_type(limb_type i) : m_first_limb(i) {}
  530. BOOST_CONSTEXPR data_type(double_limb_type i) : m_double_first_limb(i) {}
  531. #if defined(BOOST_MP_USER_DEFINED_LITERALS)
  532. template <limb_type...VALUES>
  533. BOOST_CONSTEXPR data_type(literals::detail::value_pack<VALUES...>) : m_data{ VALUES... } {}
  534. #endif
  535. } m_wrapper;
  536. limb_type m_limbs;
  537. public:
  538. //
  539. // Direct construction:
  540. //
  541. BOOST_MP_FORCEINLINE BOOST_CONSTEXPR cpp_int_base(limb_type i)BOOST_NOEXCEPT
  542. : m_wrapper(i), m_limbs(1) {}
  543. BOOST_MP_FORCEINLINE cpp_int_base(signed_limb_type i)BOOST_MP_NOEXCEPT_IF((Checked == unchecked))
  544. : m_wrapper(limb_type(i < 0 ? static_cast<limb_type>(-static_cast<signed_double_limb_type>(i)) : i)), m_limbs(1) { if(i < 0) negate(); }
  545. #ifdef BOOST_LITTLE_ENDIAN
  546. BOOST_MP_FORCEINLINE BOOST_CONSTEXPR cpp_int_base(double_limb_type i)BOOST_NOEXCEPT
  547. : m_wrapper(i), m_limbs(i > max_limb_value ? 2 : 1) {}
  548. BOOST_MP_FORCEINLINE cpp_int_base(signed_double_limb_type i)BOOST_MP_NOEXCEPT_IF((Checked == unchecked))
  549. : m_wrapper(double_limb_type(i < 0 ? static_cast<double_limb_type>(boost::multiprecision::detail::unsigned_abs(i)) : i)),
  550. m_limbs(i < 0 ? (static_cast<double_limb_type>(boost::multiprecision::detail::unsigned_abs(i)) > max_limb_value ? 2 : 1) : (i > max_limb_value ? 2 : 1))
  551. {
  552. if (i < 0) negate();
  553. }
  554. #endif
  555. #if defined(BOOST_MP_USER_DEFINED_LITERALS)
  556. template <limb_type...VALUES>
  557. BOOST_CONSTEXPR cpp_int_base(literals::detail::value_pack<VALUES...> i)
  558. : m_wrapper(i), m_limbs(sizeof...(VALUES)) {}
  559. BOOST_CONSTEXPR cpp_int_base(literals::detail::value_pack<>)
  560. : m_wrapper(static_cast<limb_type>(0u)), m_limbs(1) {}
  561. #endif
  562. //
  563. // Helper functions for getting at our internal data, and manipulating storage:
  564. //
  565. BOOST_MP_FORCEINLINE unsigned size()const BOOST_NOEXCEPT { return m_limbs; }
  566. BOOST_MP_FORCEINLINE limb_pointer limbs() BOOST_NOEXCEPT { return m_wrapper.m_data; }
  567. BOOST_MP_FORCEINLINE BOOST_CONSTEXPR const_limb_pointer limbs()const BOOST_NOEXCEPT { return m_wrapper.m_data; }
  568. BOOST_MP_FORCEINLINE BOOST_CONSTEXPR bool sign()const BOOST_NOEXCEPT { return false; }
  569. BOOST_MP_FORCEINLINE void sign(bool b) BOOST_MP_NOEXCEPT_IF((Checked == unchecked)) { if(b) negate(); }
  570. BOOST_MP_FORCEINLINE void resize(unsigned new_size, unsigned min_size) BOOST_MP_NOEXCEPT_IF((Checked == unchecked))
  571. {
  572. m_limbs = (std::min)(new_size, internal_limb_count);
  573. detail::verify_new_size(m_limbs, min_size, checked_type());
  574. }
  575. BOOST_MP_FORCEINLINE void normalize() BOOST_MP_NOEXCEPT_IF((Checked == unchecked))
  576. {
  577. limb_pointer p = limbs();
  578. detail::verify_limb_mask(m_limbs == internal_limb_count, p[internal_limb_count-1], upper_limb_mask, checked_type());
  579. p[internal_limb_count-1] &= upper_limb_mask;
  580. while((m_limbs-1) && !p[m_limbs - 1])--m_limbs;
  581. }
  582. BOOST_MP_FORCEINLINE BOOST_CONSTEXPR cpp_int_base() BOOST_NOEXCEPT
  583. : m_wrapper(limb_type(0u)), m_limbs(1) {}
  584. BOOST_MP_FORCEINLINE BOOST_CONSTEXPR cpp_int_base(const cpp_int_base& o) BOOST_NOEXCEPT
  585. : m_wrapper(o.m_wrapper), m_limbs(o.m_limbs) {}
  586. // Defaulted functions:
  587. //~cpp_int_base() BOOST_NOEXCEPT {}
  588. BOOST_MP_FORCEINLINE void assign(const cpp_int_base& o) BOOST_NOEXCEPT
  589. {
  590. if(this != &o)
  591. {
  592. m_limbs = o.m_limbs;
  593. std::memcpy(limbs(), o.limbs(), o.size() * sizeof(limbs()[0]));
  594. }
  595. }
  596. private:
  597. void check_negate(const mpl::int_<checked>&)
  598. {
  599. BOOST_THROW_EXCEPTION(std::range_error("Attempt to negate an unsigned number."));
  600. }
  601. void check_negate(const mpl::int_<unchecked>&){}
  602. public:
  603. void negate() BOOST_MP_NOEXCEPT_IF((Checked == unchecked))
  604. {
  605. // Not so much a negate as a complement - this gets called when subtraction
  606. // would result in a "negative" number:
  607. unsigned i;
  608. if((m_limbs == 1) && (m_wrapper.m_data[0] == 0))
  609. return; // negating zero is always zero, and always OK.
  610. check_negate(checked_type());
  611. for(i = m_limbs; i < internal_limb_count; ++i)
  612. m_wrapper.m_data[i] = 0;
  613. m_limbs = internal_limb_count;
  614. for(i = 0; i < internal_limb_count; ++i)
  615. m_wrapper.m_data[i] = ~m_wrapper.m_data[i];
  616. normalize();
  617. eval_increment(static_cast<cpp_int_backend<MinBits, MinBits, unsigned_magnitude, Checked, void>& >(*this));
  618. }
  619. BOOST_MP_FORCEINLINE BOOST_CONSTEXPR bool isneg()const BOOST_NOEXCEPT
  620. {
  621. return false;
  622. }
  623. BOOST_MP_FORCEINLINE void do_swap(cpp_int_base& o) BOOST_NOEXCEPT
  624. {
  625. for(unsigned i = 0; i < (std::max)(size(), o.size()); ++i)
  626. std::swap(m_wrapper.m_data[i], o.m_wrapper.m_data[i]);
  627. std::swap(m_limbs, o.m_limbs);
  628. }
  629. protected:
  630. template <class A>
  631. void check_in_range(const A&) BOOST_NOEXCEPT {}
  632. };
  633. #ifndef BOOST_NO_INCLASS_MEMBER_INITIALIZATION
  634. template <unsigned MinBits, cpp_int_check_type Checked>
  635. const unsigned cpp_int_base<MinBits, MinBits, unsigned_magnitude, Checked, void, false>::limb_bits;
  636. template <unsigned MinBits, cpp_int_check_type Checked>
  637. const limb_type cpp_int_base<MinBits, MinBits, unsigned_magnitude, Checked, void, false>::max_limb_value;
  638. template <unsigned MinBits, cpp_int_check_type Checked>
  639. const limb_type cpp_int_base<MinBits, MinBits, unsigned_magnitude, Checked, void, false>::sign_bit_mask;
  640. template <unsigned MinBits, cpp_int_check_type Checked>
  641. const unsigned cpp_int_base<MinBits, MinBits, unsigned_magnitude, Checked, void, false>::internal_limb_count;
  642. template <unsigned MinBits, cpp_int_check_type Checked>
  643. const bool cpp_int_base<MinBits, MinBits, unsigned_magnitude, Checked, void, false>::variable;
  644. #endif
  645. //
  646. // Traits classes to figure out a native type with N bits, these vary from boost::uint_t<N> only
  647. // because some platforms have native integer types longer than boost::long_long_type, "really boost::long_long_type" anyone??
  648. //
  649. template <unsigned N, bool s>
  650. struct trivial_limb_type_imp
  651. {
  652. typedef double_limb_type type;
  653. };
  654. template <unsigned N>
  655. struct trivial_limb_type_imp<N, true>
  656. {
  657. typedef typename boost::uint_t<N>::least type;
  658. };
  659. template <unsigned N>
  660. struct trivial_limb_type : public trivial_limb_type_imp<N, N <= sizeof(boost::long_long_type) * CHAR_BIT> {};
  661. //
  662. // Backend for fixed precision signed-magnitude type which will fit entirely inside a "double_limb_type":
  663. //
  664. template <unsigned MinBits, cpp_int_check_type Checked>
  665. struct cpp_int_base<MinBits, MinBits, signed_magnitude, Checked, void, true>
  666. {
  667. typedef typename trivial_limb_type<MinBits>::type local_limb_type;
  668. typedef local_limb_type* limb_pointer;
  669. typedef const local_limb_type* const_limb_pointer;
  670. typedef mpl::int_<Checked> checked_type;
  671. protected:
  672. BOOST_STATIC_CONSTANT(unsigned, limb_bits = sizeof(local_limb_type) * CHAR_BIT);
  673. BOOST_STATIC_CONSTANT(local_limb_type, limb_mask = (MinBits < limb_bits) ? local_limb_type((local_limb_type(~local_limb_type(0))) >> (limb_bits - MinBits)) : local_limb_type(~local_limb_type(0)));
  674. private:
  675. local_limb_type m_data;
  676. bool m_sign;
  677. //
  678. // Interface invariants:
  679. //
  680. BOOST_STATIC_ASSERT_MSG(MinBits <= sizeof(double_limb_type) * CHAR_BIT, "Template parameter MinBits is inconsistent with the parameter trivial - did you mistakingly try to override the trivial parameter?");
  681. protected:
  682. template <class T>
  683. typename boost::disable_if_c<!boost::is_integral<T>::value || (std::numeric_limits<T>::is_specialized && (std::numeric_limits<T>::digits <= (int)MinBits))>::type
  684. check_in_range(T val, const mpl::int_<checked>&)
  685. {
  686. typedef typename common_type<typename make_unsigned<T>::type, local_limb_type>::type common_type;
  687. if(static_cast<common_type>(boost::multiprecision::detail::unsigned_abs(val)) > static_cast<common_type>(limb_mask))
  688. BOOST_THROW_EXCEPTION(std::range_error("The argument to a cpp_int constructor exceeded the largest value it can represent."));
  689. }
  690. template <class T>
  691. typename boost::disable_if_c<boost::is_integral<T>::value || (std::numeric_limits<T>::is_specialized && (std::numeric_limits<T>::digits <= (int)MinBits))>::type
  692. check_in_range(T val, const mpl::int_<checked>&)
  693. {
  694. using std::abs;
  695. typedef typename common_type<T, local_limb_type>::type common_type;
  696. if (static_cast<common_type>(abs(val)) > static_cast<common_type>(limb_mask))
  697. BOOST_THROW_EXCEPTION(std::range_error("The argument to a cpp_int constructor exceeded the largest value it can represent."));
  698. }
  699. template <class T, int C>
  700. void check_in_range(T, const mpl::int_<C>&) BOOST_NOEXCEPT {}
  701. template <class T>
  702. void check_in_range(T val) BOOST_MP_NOEXCEPT_IF(noexcept(std::declval<cpp_int_base>().check_in_range(std::declval<T>(), checked_type())))
  703. {
  704. check_in_range(val, checked_type());
  705. }
  706. public:
  707. //
  708. // Direct construction:
  709. //
  710. template <class SI>
  711. BOOST_MP_FORCEINLINE BOOST_CONSTEXPR cpp_int_base(SI i, typename boost::enable_if_c<is_signed<SI>::value && (Checked == unchecked) >::type const* = 0) BOOST_MP_NOEXCEPT_IF(noexcept(std::declval<cpp_int_base>().check_in_range(std::declval<SI>())))
  712. : m_data(i < 0 ? static_cast<local_limb_type>(static_cast<typename make_unsigned<SI>::type>(boost::multiprecision::detail::unsigned_abs(i)) & limb_mask) : static_cast<local_limb_type>(i)& limb_mask), m_sign(i < 0) {}
  713. template <class SI>
  714. BOOST_MP_FORCEINLINE cpp_int_base(SI i, typename boost::enable_if_c<is_signed<SI>::value && (Checked == checked) >::type const* = 0) BOOST_MP_NOEXCEPT_IF(noexcept(std::declval<cpp_int_base>().check_in_range(std::declval<SI>())))
  715. : m_data(i < 0 ? (static_cast<local_limb_type>(static_cast<typename make_unsigned<SI>::type>(boost::multiprecision::detail::unsigned_abs(i)) & limb_mask)) : static_cast<local_limb_type>(i)& limb_mask), m_sign(i < 0)
  716. { check_in_range(i); }
  717. template <class UI>
  718. BOOST_MP_FORCEINLINE BOOST_CONSTEXPR cpp_int_base(UI i, typename boost::enable_if_c<is_unsigned<UI>::value && (Checked == unchecked)>::type const* = 0) BOOST_NOEXCEPT
  719. : m_data(static_cast<local_limb_type>(i) & limb_mask), m_sign(false) {}
  720. template <class UI>
  721. BOOST_MP_FORCEINLINE cpp_int_base(UI i, typename boost::enable_if_c<is_unsigned<UI>::value && (Checked == checked)>::type const* = 0) BOOST_MP_NOEXCEPT_IF(noexcept(std::declval<cpp_int_base>().check_in_range(std::declval<UI>())))
  722. : m_data(static_cast<local_limb_type>(i) & limb_mask), m_sign(false) { check_in_range(i); }
  723. template <class F>
  724. BOOST_MP_FORCEINLINE BOOST_CONSTEXPR cpp_int_base(F i, typename boost::enable_if_c<is_floating_point<F>::value && (Checked == unchecked)>::type const* = 0) BOOST_NOEXCEPT
  725. : m_data(static_cast<local_limb_type>(std::fabs(i)) & limb_mask), m_sign(i < 0) {}
  726. template <class F>
  727. BOOST_MP_FORCEINLINE cpp_int_base(F i, typename boost::enable_if_c<is_floating_point<F>::value && (Checked == checked)>::type const* = 0)
  728. : m_data(static_cast<local_limb_type>(std::fabs(i)) & limb_mask), m_sign(i < 0) { check_in_range(i); }
  729. #if defined(BOOST_MP_USER_DEFINED_LITERALS)
  730. BOOST_CONSTEXPR cpp_int_base(literals::detail::value_pack<>) BOOST_NOEXCEPT
  731. : m_data(static_cast<local_limb_type>(0u)), m_sign(false) {}
  732. template <limb_type a>
  733. BOOST_CONSTEXPR cpp_int_base(literals::detail::value_pack<a>)BOOST_NOEXCEPT
  734. : m_data(static_cast<local_limb_type>(a)), m_sign(false) {}
  735. template <limb_type a, limb_type b>
  736. BOOST_CONSTEXPR cpp_int_base(literals::detail::value_pack<a, b>)BOOST_NOEXCEPT
  737. : m_data(static_cast<local_limb_type>(a) | (static_cast<local_limb_type>(b) << bits_per_limb)), m_sign(false) {}
  738. BOOST_CONSTEXPR cpp_int_base(const cpp_int_base& a, const literals::detail::negate_tag&)BOOST_NOEXCEPT
  739. : m_data(a.m_data), m_sign(a.m_data ? !a.m_sign : false) {}
  740. #endif
  741. //
  742. // Helper functions for getting at our internal data, and manipulating storage:
  743. //
  744. BOOST_MP_FORCEINLINE BOOST_CONSTEXPR unsigned size()const BOOST_NOEXCEPT { return 1; }
  745. BOOST_MP_FORCEINLINE limb_pointer limbs() BOOST_NOEXCEPT { return &m_data; }
  746. BOOST_MP_FORCEINLINE BOOST_CONSTEXPR const_limb_pointer limbs()const BOOST_NOEXCEPT { return &m_data; }
  747. BOOST_MP_FORCEINLINE bool sign()const BOOST_NOEXCEPT { return m_sign; }
  748. BOOST_MP_FORCEINLINE void sign(bool b) BOOST_NOEXCEPT
  749. {
  750. m_sign = b;
  751. // Check for zero value:
  752. if(m_sign && !m_data)
  753. {
  754. m_sign = false;
  755. }
  756. }
  757. BOOST_MP_FORCEINLINE void resize(unsigned new_size, unsigned min_size)
  758. {
  759. detail::verify_new_size(2, min_size, checked_type());
  760. }
  761. BOOST_MP_FORCEINLINE void normalize() BOOST_MP_NOEXCEPT_IF((Checked == unchecked))
  762. {
  763. if(!m_data)
  764. m_sign = false; // zero is always unsigned
  765. detail::verify_limb_mask(true, m_data, limb_mask, checked_type());
  766. m_data &= limb_mask;
  767. }
  768. BOOST_MP_FORCEINLINE BOOST_CONSTEXPR cpp_int_base() BOOST_NOEXCEPT : m_data(0), m_sign(false) {}
  769. BOOST_MP_FORCEINLINE BOOST_CONSTEXPR cpp_int_base(const cpp_int_base& o) BOOST_NOEXCEPT
  770. : m_data(o.m_data), m_sign(o.m_sign) {}
  771. //~cpp_int_base() BOOST_NOEXCEPT {}
  772. BOOST_MP_FORCEINLINE void assign(const cpp_int_base& o) BOOST_NOEXCEPT
  773. {
  774. m_data = o.m_data;
  775. m_sign = o.m_sign;
  776. }
  777. BOOST_MP_FORCEINLINE void negate() BOOST_NOEXCEPT
  778. {
  779. m_sign = !m_sign;
  780. // Check for zero value:
  781. if(m_data == 0)
  782. {
  783. m_sign = false;
  784. }
  785. }
  786. BOOST_MP_FORCEINLINE bool isneg()const BOOST_NOEXCEPT
  787. {
  788. return m_sign;
  789. }
  790. BOOST_MP_FORCEINLINE void do_swap(cpp_int_base& o) BOOST_NOEXCEPT
  791. {
  792. std::swap(m_sign, o.m_sign);
  793. std::swap(m_data, o.m_data);
  794. }
  795. };
  796. //
  797. // Backend for unsigned fixed precision (i.e. no allocator) type which will fit entirely inside a "double_limb_type":
  798. //
  799. template <unsigned MinBits, cpp_int_check_type Checked>
  800. struct cpp_int_base<MinBits, MinBits, unsigned_magnitude, Checked, void, true>
  801. {
  802. typedef typename trivial_limb_type<MinBits>::type local_limb_type;
  803. typedef local_limb_type* limb_pointer;
  804. typedef const local_limb_type* const_limb_pointer;
  805. private:
  806. BOOST_STATIC_CONSTANT(unsigned, limb_bits = sizeof(local_limb_type) * CHAR_BIT);
  807. BOOST_STATIC_CONSTANT(local_limb_type, limb_mask = limb_bits != MinBits ?
  808. static_cast<local_limb_type>(static_cast<local_limb_type>(~local_limb_type(0)) >> (limb_bits - MinBits))
  809. : static_cast<local_limb_type>(~local_limb_type(0)));
  810. local_limb_type m_data;
  811. typedef mpl::int_<Checked> checked_type;
  812. //
  813. // Interface invariants:
  814. //
  815. BOOST_STATIC_ASSERT_MSG(MinBits <= sizeof(double_limb_type) * CHAR_BIT, "Template parameter MinBits is inconsistent with the parameter trivial - did you mistakingly try to override the trivial parameter?");
  816. protected:
  817. template <class T>
  818. typename boost::disable_if_c<std::numeric_limits<T>::is_specialized && (std::numeric_limits<T>::digits <= (int)MinBits)>::type
  819. check_in_range(T val, const mpl::int_<checked>&, const boost::false_type&)
  820. {
  821. typedef typename common_type<T, local_limb_type>::type common_type;
  822. if(static_cast<common_type>(val) > limb_mask)
  823. BOOST_THROW_EXCEPTION(std::range_error("The argument to a cpp_int constructor exceeded the largest value it can represent."));
  824. }
  825. template <class T>
  826. void check_in_range(T val, const mpl::int_<checked>&, const boost::true_type&)
  827. {
  828. typedef typename common_type<T, local_limb_type>::type common_type;
  829. if(static_cast<common_type>(val) > limb_mask)
  830. BOOST_THROW_EXCEPTION(std::range_error("The argument to a cpp_int constructor exceeded the largest value it can represent."));
  831. if(val < 0)
  832. BOOST_THROW_EXCEPTION(std::range_error("The argument to an unsigned cpp_int constructor was negative."));
  833. }
  834. template <class T, int C, bool B>
  835. BOOST_MP_FORCEINLINE void check_in_range(T, const mpl::int_<C>&, const boost::integral_constant<bool, B>&) BOOST_NOEXCEPT {}
  836. template <class T>
  837. BOOST_MP_FORCEINLINE void check_in_range(T val) BOOST_MP_NOEXCEPT_IF(noexcept(std::declval<cpp_int_base>().check_in_range(std::declval<T>(), checked_type(), is_signed<T>())))
  838. {
  839. check_in_range(val, checked_type(), is_signed<T>());
  840. }
  841. public:
  842. //
  843. // Direct construction:
  844. //
  845. template <class SI>
  846. BOOST_MP_FORCEINLINE BOOST_CONSTEXPR cpp_int_base(SI i, typename boost::enable_if_c<is_signed<SI>::value && (Checked == unchecked) >::type const* = 0) BOOST_NOEXCEPT
  847. : m_data(i < 0 ? (1 + ~static_cast<local_limb_type>(-i)) & limb_mask : static_cast<local_limb_type>(i) & limb_mask) {}
  848. template <class SI>
  849. BOOST_MP_FORCEINLINE cpp_int_base(SI i, typename boost::enable_if_c<is_signed<SI>::value && (Checked == checked) >::type const* = 0) BOOST_MP_NOEXCEPT_IF(noexcept(std::declval<cpp_int_base>().check_in_range(std::declval<SI>())))
  850. : m_data(i < 0 ? 1 + ~static_cast<local_limb_type>(-i) : static_cast<local_limb_type>(i)) { check_in_range(i); }
  851. template <class UI>
  852. BOOST_MP_FORCEINLINE BOOST_CONSTEXPR cpp_int_base(UI i, typename boost::enable_if_c<is_unsigned<UI>::value && (Checked == unchecked) >::type const* = 0) BOOST_NOEXCEPT
  853. : m_data(static_cast<local_limb_type>(i) & limb_mask) {}
  854. template <class UI>
  855. BOOST_MP_FORCEINLINE cpp_int_base(UI i, typename boost::enable_if_c<is_unsigned<UI>::value && (Checked == checked) >::type const* = 0) BOOST_MP_NOEXCEPT_IF(noexcept(std::declval<cpp_int_base>().check_in_range(std::declval<UI>())))
  856. : m_data(static_cast<local_limb_type>(i)) { check_in_range(i); }
  857. template <class F>
  858. BOOST_MP_FORCEINLINE cpp_int_base(F i, typename boost::enable_if<is_floating_point<F> >::type const* = 0) BOOST_MP_NOEXCEPT_IF((Checked == unchecked))
  859. : m_data(static_cast<local_limb_type>(std::fabs(i)) & limb_mask)
  860. {
  861. check_in_range(i);
  862. if(i < 0)
  863. negate();
  864. }
  865. #if defined(BOOST_MP_USER_DEFINED_LITERALS)
  866. BOOST_CONSTEXPR cpp_int_base(literals::detail::value_pack<>) BOOST_NOEXCEPT
  867. : m_data(static_cast<local_limb_type>(0u)) {}
  868. template <limb_type a>
  869. BOOST_CONSTEXPR cpp_int_base(literals::detail::value_pack<a>) BOOST_NOEXCEPT
  870. : m_data(static_cast<local_limb_type>(a)) {}
  871. template <limb_type a, limb_type b>
  872. BOOST_CONSTEXPR cpp_int_base(literals::detail::value_pack<a, b>) BOOST_NOEXCEPT
  873. : m_data(static_cast<local_limb_type>(a) | (static_cast<local_limb_type>(b) << bits_per_limb)) {}
  874. #endif
  875. //
  876. // Helper functions for getting at our internal data, and manipulating storage:
  877. //
  878. BOOST_MP_FORCEINLINE BOOST_CONSTEXPR unsigned size()const BOOST_NOEXCEPT { return 1; }
  879. BOOST_MP_FORCEINLINE limb_pointer limbs() BOOST_NOEXCEPT { return &m_data; }
  880. BOOST_MP_FORCEINLINE BOOST_CONSTEXPR const_limb_pointer limbs()const BOOST_NOEXCEPT { return &m_data; }
  881. BOOST_MP_FORCEINLINE BOOST_CONSTEXPR bool sign()const BOOST_NOEXCEPT { return false; }
  882. BOOST_MP_FORCEINLINE void sign(bool b) BOOST_MP_NOEXCEPT_IF((Checked == unchecked))
  883. {
  884. if(b)
  885. negate();
  886. }
  887. BOOST_MP_FORCEINLINE void resize(unsigned, unsigned min_size)
  888. {
  889. detail::verify_new_size(2, min_size, checked_type());
  890. }
  891. BOOST_MP_FORCEINLINE void normalize() BOOST_MP_NOEXCEPT_IF((Checked == unchecked))
  892. {
  893. detail::verify_limb_mask(true, m_data, limb_mask, checked_type());
  894. m_data &= limb_mask;
  895. }
  896. BOOST_MP_FORCEINLINE BOOST_CONSTEXPR cpp_int_base() BOOST_NOEXCEPT : m_data(0) {}
  897. BOOST_MP_FORCEINLINE BOOST_CONSTEXPR cpp_int_base(const cpp_int_base& o) BOOST_NOEXCEPT
  898. : m_data(o.m_data) {}
  899. //~cpp_int_base() BOOST_NOEXCEPT {}
  900. BOOST_MP_FORCEINLINE void assign(const cpp_int_base& o) BOOST_NOEXCEPT
  901. {
  902. m_data = o.m_data;
  903. }
  904. BOOST_MP_FORCEINLINE void negate() BOOST_MP_NOEXCEPT_IF((Checked == unchecked))
  905. {
  906. if(Checked == checked)
  907. {
  908. BOOST_THROW_EXCEPTION(std::range_error("Attempt to negate an unsigned type."));
  909. }
  910. m_data = ~m_data;
  911. ++m_data;
  912. }
  913. BOOST_MP_FORCEINLINE BOOST_CONSTEXPR bool isneg()const BOOST_NOEXCEPT
  914. {
  915. return false;
  916. }
  917. BOOST_MP_FORCEINLINE void do_swap(cpp_int_base& o) BOOST_NOEXCEPT
  918. {
  919. std::swap(m_data, o.m_data);
  920. }
  921. };
  922. //
  923. // Traits class, lets us know whether type T can be directly converted to the base type,
  924. // used to enable/disable constructors etc:
  925. //
  926. template <class Arg, class Base>
  927. struct is_allowed_cpp_int_base_conversion : public mpl::if_c<
  928. is_same<Arg, limb_type>::value || is_same<Arg, signed_limb_type>::value
  929. #ifdef BOOST_LITTLE_ENDIAN
  930. || is_same<Arg, double_limb_type>::value || is_same<Arg, signed_double_limb_type>::value
  931. #endif
  932. #if defined(BOOST_MP_USER_DEFINED_LITERALS)
  933. || literals::detail::is_value_pack<Arg>::value
  934. #endif
  935. || (is_trivial_cpp_int<Base>::value && is_arithmetic<Arg>::value),
  936. mpl::true_,
  937. mpl::false_
  938. >::type
  939. {};
  940. //
  941. // Now the actual backend, normalising parameters passed to the base class:
  942. //
  943. template <unsigned MinBits, unsigned MaxBits, cpp_integer_type SignType, cpp_int_check_type Checked, class Allocator>
  944. struct cpp_int_backend
  945. : public cpp_int_base<
  946. min_precision<cpp_int_backend<MinBits, MaxBits, SignType, Checked, Allocator> >::value,
  947. max_precision<cpp_int_backend<MinBits, MaxBits, SignType, Checked, Allocator> >::value,
  948. SignType,
  949. Checked,
  950. Allocator,
  951. is_trivial_cpp_int<cpp_int_backend<MinBits, MaxBits, SignType, Checked, Allocator> >::value>
  952. {
  953. typedef cpp_int_backend<MinBits, MaxBits, SignType, Checked, Allocator> self_type;
  954. typedef cpp_int_base<
  955. min_precision<self_type>::value,
  956. max_precision<self_type>::value,
  957. SignType,
  958. Checked,
  959. Allocator,
  960. is_trivial_cpp_int<self_type>::value> base_type;
  961. typedef mpl::bool_<is_trivial_cpp_int<self_type>::value> trivial_tag;
  962. public:
  963. typedef typename mpl::if_<
  964. trivial_tag,
  965. mpl::list<
  966. signed char, short, int, long,
  967. boost::long_long_type, signed_double_limb_type>,
  968. mpl::list<signed_limb_type, signed_double_limb_type>
  969. >::type signed_types;
  970. typedef typename mpl::if_<
  971. trivial_tag,
  972. mpl::list<unsigned char, unsigned short, unsigned,
  973. unsigned long, boost::ulong_long_type, double_limb_type>,
  974. mpl::list<limb_type, double_limb_type>
  975. >::type unsigned_types;
  976. typedef typename mpl::if_<
  977. trivial_tag,
  978. mpl::list<float, double, long double>,
  979. mpl::list<long double>
  980. >::type float_types;
  981. typedef mpl::int_<Checked> checked_type;
  982. BOOST_MP_FORCEINLINE BOOST_CONSTEXPR cpp_int_backend() BOOST_NOEXCEPT{}
  983. BOOST_MP_FORCEINLINE BOOST_CONSTEXPR cpp_int_backend(const cpp_int_backend& o) BOOST_MP_NOEXCEPT_IF(boost::is_void<Allocator>::value) : base_type(o) {}
  984. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  985. BOOST_MP_FORCEINLINE BOOST_CONSTEXPR cpp_int_backend(cpp_int_backend&& o) BOOST_NOEXCEPT
  986. : base_type(static_cast<base_type&&>(o)) {}
  987. #endif
  988. //
  989. // Direct construction from arithmetic type:
  990. //
  991. template <class Arg>
  992. BOOST_MP_FORCEINLINE BOOST_CONSTEXPR cpp_int_backend(Arg i, typename boost::enable_if_c<is_allowed_cpp_int_base_conversion<Arg, base_type>::value >::type const* = 0)BOOST_MP_NOEXCEPT_IF(noexcept(base_type(std::declval<Arg>())))
  993. : base_type(i) {}
  994. private:
  995. template <unsigned MinBits2, unsigned MaxBits2, cpp_integer_type SignType2, cpp_int_check_type Checked2, class Allocator2>
  996. void do_assign(const cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2>& other, mpl::true_ const&, mpl::true_ const &)
  997. {
  998. // Assigning trivial type to trivial type:
  999. this->check_in_range(*other.limbs());
  1000. *this->limbs() = static_cast<typename self_type::local_limb_type>(*other.limbs());
  1001. this->sign(other.sign());
  1002. this->normalize();
  1003. }
  1004. template <unsigned MinBits2, unsigned MaxBits2, cpp_integer_type SignType2, cpp_int_check_type Checked2, class Allocator2>
  1005. void do_assign(const cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2>& other, mpl::true_ const&, mpl::false_ const &)
  1006. {
  1007. // non-trivial to trivial narrowing conversion:
  1008. double_limb_type v = *other.limbs();
  1009. if(other.size() > 1)
  1010. {
  1011. v |= static_cast<double_limb_type>(other.limbs()[1]) << bits_per_limb;
  1012. if((Checked == checked) && (other.size() > 2))
  1013. {
  1014. BOOST_THROW_EXCEPTION(std::range_error("Assignment of a cpp_int that is out of range for the target type."));
  1015. }
  1016. }
  1017. *this = v;
  1018. this->sign(other.sign());
  1019. this->normalize();
  1020. }
  1021. template <unsigned MinBits2, unsigned MaxBits2, cpp_integer_type SignType2, cpp_int_check_type Checked2, class Allocator2>
  1022. void do_assign(const cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2>& other, mpl::false_ const&, mpl::true_ const &)
  1023. {
  1024. // trivial to non-trivial, treat the trivial argument as if it were an unsigned arithmetic type, then set the sign afterwards:
  1025. *this = static_cast<
  1026. typename boost::multiprecision::detail::canonical<
  1027. typename cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2>::local_limb_type,
  1028. cpp_int_backend<MinBits, MaxBits, SignType, Checked, Allocator>
  1029. >::type
  1030. >(*other.limbs());
  1031. this->sign(other.sign());
  1032. }
  1033. template <unsigned MinBits2, unsigned MaxBits2, cpp_integer_type SignType2, cpp_int_check_type Checked2, class Allocator2>
  1034. void do_assign(const cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2>& other, mpl::false_ const&, mpl::false_ const &)
  1035. {
  1036. // regular non-trivial to non-trivial assign:
  1037. this->resize(other.size(), other.size());
  1038. std::memcpy(this->limbs(), other.limbs(), (std::min)(other.size(), this->size()) * sizeof(this->limbs()[0]));
  1039. this->sign(other.sign());
  1040. this->normalize();
  1041. }
  1042. public:
  1043. template <unsigned MinBits2, unsigned MaxBits2, cpp_integer_type SignType2, cpp_int_check_type Checked2, class Allocator2>
  1044. cpp_int_backend(
  1045. const cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2>& other,
  1046. typename boost::enable_if_c<is_implicit_cpp_int_conversion<cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2>, self_type>::value>::type* = 0)
  1047. : base_type()
  1048. {
  1049. do_assign(
  1050. other,
  1051. mpl::bool_<is_trivial_cpp_int<self_type>::value>(),
  1052. mpl::bool_<is_trivial_cpp_int<cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2> >::value>());
  1053. }
  1054. template <unsigned MinBits2, unsigned MaxBits2, cpp_integer_type SignType2, cpp_int_check_type Checked2, class Allocator2>
  1055. explicit cpp_int_backend(
  1056. const cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2>& other,
  1057. typename boost::disable_if_c<is_implicit_cpp_int_conversion<cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2>, self_type>::value>::type* = 0)
  1058. : base_type()
  1059. {
  1060. do_assign(
  1061. other,
  1062. mpl::bool_<is_trivial_cpp_int<self_type>::value>(),
  1063. mpl::bool_<is_trivial_cpp_int<cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2> >::value>());
  1064. }
  1065. template <unsigned MinBits2, unsigned MaxBits2, cpp_integer_type SignType2, cpp_int_check_type Checked2, class Allocator2>
  1066. cpp_int_backend& operator=(
  1067. const cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2>& other)
  1068. {
  1069. do_assign(
  1070. other,
  1071. mpl::bool_<is_trivial_cpp_int<self_type>::value>(),
  1072. mpl::bool_<is_trivial_cpp_int<cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2> >::value>());
  1073. return *this;
  1074. }
  1075. #ifdef BOOST_MP_USER_DEFINED_LITERALS
  1076. BOOST_CONSTEXPR cpp_int_backend(const cpp_int_backend& a, const literals::detail::negate_tag& tag)
  1077. : base_type(static_cast<const base_type&>(a), tag){}
  1078. #endif
  1079. BOOST_MP_FORCEINLINE cpp_int_backend& operator = (const cpp_int_backend& o) BOOST_MP_NOEXCEPT_IF(noexcept(std::declval<cpp_int_backend>().assign(std::declval<const cpp_int_backend&>())))
  1080. {
  1081. this->assign(o);
  1082. return *this;
  1083. }
  1084. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  1085. BOOST_MP_FORCEINLINE cpp_int_backend& operator = (cpp_int_backend&& o) BOOST_MP_NOEXCEPT_IF(noexcept(std::declval<base_type&>() = std::declval<base_type>()))
  1086. {
  1087. *static_cast<base_type*>(this) = static_cast<base_type&&>(o);
  1088. return *this;
  1089. }
  1090. #endif
  1091. private:
  1092. template <class A>
  1093. typename boost::enable_if<is_unsigned<A> >::type do_assign_arithmetic(A val, const mpl::true_&)
  1094. BOOST_MP_NOEXCEPT_IF(noexcept(std::declval<cpp_int_backend>().check_in_range(std::declval<A>())))
  1095. {
  1096. this->check_in_range(val);
  1097. *this->limbs() = static_cast<typename self_type::local_limb_type>(val);
  1098. this->normalize();
  1099. }
  1100. template <class A>
  1101. typename boost::disable_if_c<is_unsigned<A>::value || !is_integral<A>::value >::type do_assign_arithmetic(A val, const mpl::true_&)
  1102. BOOST_MP_NOEXCEPT_IF(noexcept(std::declval<cpp_int_backend>().check_in_range(std::declval<A>())) && noexcept(std::declval<cpp_int_backend>().sign(true)))
  1103. {
  1104. this->check_in_range(val);
  1105. *this->limbs() = (val < 0) ? static_cast<typename self_type::local_limb_type>(boost::multiprecision::detail::unsigned_abs(val)) : static_cast<typename self_type::local_limb_type>(val);
  1106. this->sign(val < 0);
  1107. this->normalize();
  1108. }
  1109. template <class A>
  1110. typename boost::enable_if_c< !is_integral<A>::value>::type do_assign_arithmetic(A val, const mpl::true_&)
  1111. {
  1112. this->check_in_range(val);
  1113. *this->limbs() = (val < 0) ? static_cast<typename self_type::local_limb_type>(boost::multiprecision::detail::abs(val)) : static_cast<typename self_type::local_limb_type>(val);
  1114. this->sign(val < 0);
  1115. this->normalize();
  1116. }
  1117. BOOST_MP_FORCEINLINE void do_assign_arithmetic(limb_type i, const mpl::false_&) BOOST_NOEXCEPT
  1118. {
  1119. this->resize(1, 1);
  1120. *this->limbs() = i;
  1121. this->sign(false);
  1122. }
  1123. BOOST_MP_FORCEINLINE void do_assign_arithmetic(signed_limb_type i, const mpl::false_&) BOOST_MP_NOEXCEPT_IF(noexcept(std::declval<cpp_int_backend>().sign(true)))
  1124. {
  1125. this->resize(1, 1);
  1126. *this->limbs() = static_cast<limb_type>(boost::multiprecision::detail::unsigned_abs(i));
  1127. this->sign(i < 0);
  1128. }
  1129. void do_assign_arithmetic(double_limb_type i, const mpl::false_&) BOOST_NOEXCEPT
  1130. {
  1131. BOOST_STATIC_ASSERT(sizeof(i) == 2 * sizeof(limb_type));
  1132. BOOST_STATIC_ASSERT(base_type::internal_limb_count >= 2);
  1133. typename base_type::limb_pointer p = this->limbs();
  1134. *p = static_cast<limb_type>(i);
  1135. p[1] = static_cast<limb_type>(i >> base_type::limb_bits);
  1136. this->resize(p[1] ? 2 : 1, p[1] ? 2 : 1);
  1137. this->sign(false);
  1138. }
  1139. void do_assign_arithmetic(signed_double_limb_type i, const mpl::false_&) BOOST_MP_NOEXCEPT_IF(noexcept(std::declval<cpp_int_backend>().sign(true)))
  1140. {
  1141. BOOST_STATIC_ASSERT(sizeof(i) == 2 * sizeof(limb_type));
  1142. BOOST_STATIC_ASSERT(base_type::internal_limb_count >= 2);
  1143. bool s = false;
  1144. double_limb_type ui;
  1145. if(i < 0)
  1146. s = true;
  1147. ui = static_cast<double_limb_type>(boost::multiprecision::detail::unsigned_abs(i));
  1148. typename base_type::limb_pointer p = this->limbs();
  1149. *p = static_cast<limb_type>(ui);
  1150. p[1] = static_cast<limb_type>(ui >> base_type::limb_bits);
  1151. this->resize(p[1] ? 2 : 1, p[1] ? 2 : 1);
  1152. this->sign(s);
  1153. }
  1154. void do_assign_arithmetic(long double a, const mpl::false_&)
  1155. {
  1156. using default_ops::eval_add;
  1157. using default_ops::eval_subtract;
  1158. using std::frexp;
  1159. using std::ldexp;
  1160. using std::floor;
  1161. if(a < 0)
  1162. {
  1163. do_assign_arithmetic(-a, mpl::false_());
  1164. this->sign(true);
  1165. return;
  1166. }
  1167. if (a == 0) {
  1168. *this = static_cast<limb_type>(0u);
  1169. }
  1170. if (a == 1) {
  1171. *this = static_cast<limb_type>(1u);
  1172. }
  1173. BOOST_ASSERT(!(boost::math::isinf)(a));
  1174. BOOST_ASSERT(!(boost::math::isnan)(a));
  1175. int e;
  1176. long double f, term;
  1177. *this = static_cast<limb_type>(0u);
  1178. f = frexp(a, &e);
  1179. static const limb_type shift = std::numeric_limits<limb_type>::digits;
  1180. while(f)
  1181. {
  1182. // extract int sized bits from f:
  1183. f = ldexp(f, shift);
  1184. term = floor(f);
  1185. e -= shift;
  1186. eval_left_shift(*this, shift);
  1187. if(term > 0)
  1188. eval_add(*this, static_cast<limb_type>(term));
  1189. else
  1190. eval_subtract(*this, static_cast<limb_type>(-term));
  1191. f -= term;
  1192. }
  1193. if(e > 0)
  1194. eval_left_shift(*this, e);
  1195. else if(e < 0)
  1196. eval_right_shift(*this, -e);
  1197. }
  1198. public:
  1199. template <class Arithmetic>
  1200. BOOST_MP_FORCEINLINE typename boost::enable_if_c<!boost::multiprecision::detail::is_byte_container<Arithmetic>::value, cpp_int_backend&>::type operator = (Arithmetic val) BOOST_MP_NOEXCEPT_IF(noexcept(std::declval<cpp_int_backend>().do_assign_arithmetic(std::declval<Arithmetic>(), trivial_tag())))
  1201. {
  1202. do_assign_arithmetic(val, trivial_tag());
  1203. return *this;
  1204. }
  1205. private:
  1206. void do_assign_string(const char* s, const mpl::true_&)
  1207. {
  1208. std::size_t n = s ? std::strlen(s) : 0;
  1209. *this = 0;
  1210. unsigned radix = 10;
  1211. bool isneg = false;
  1212. if(n && (*s == '-'))
  1213. {
  1214. --n;
  1215. ++s;
  1216. isneg = true;
  1217. }
  1218. if(n && (*s == '0'))
  1219. {
  1220. if((n > 1) && ((s[1] == 'x') || (s[1] == 'X')))
  1221. {
  1222. radix = 16;
  1223. s +=2;
  1224. n -= 2;
  1225. }
  1226. else
  1227. {
  1228. radix = 8;
  1229. n -= 1;
  1230. }
  1231. }
  1232. if(n)
  1233. {
  1234. unsigned val;
  1235. while(*s)
  1236. {
  1237. if(*s >= '0' && *s <= '9')
  1238. val = *s - '0';
  1239. else if(*s >= 'a' && *s <= 'f')
  1240. val = 10 + *s - 'a';
  1241. else if(*s >= 'A' && *s <= 'F')
  1242. val = 10 + *s - 'A';
  1243. else
  1244. val = radix + 1;
  1245. if(val >= radix)
  1246. {
  1247. BOOST_THROW_EXCEPTION(std::runtime_error("Unexpected content found while parsing character string."));
  1248. }
  1249. *this->limbs() = detail::checked_multiply(*this->limbs(), static_cast<typename base_type::local_limb_type>(radix), checked_type());
  1250. *this->limbs() = detail::checked_add(*this->limbs(), static_cast<typename base_type::local_limb_type>(val), checked_type());
  1251. ++s;
  1252. }
  1253. }
  1254. if(isneg)
  1255. this->negate();
  1256. }
  1257. void do_assign_string(const char* s, const mpl::false_&)
  1258. {
  1259. using default_ops::eval_multiply;
  1260. using default_ops::eval_add;
  1261. std::size_t n = s ? std::strlen(s) : 0;
  1262. *this = static_cast<limb_type>(0u);
  1263. unsigned radix = 10;
  1264. bool isneg = false;
  1265. if(n && (*s == '-'))
  1266. {
  1267. --n;
  1268. ++s;
  1269. isneg = true;
  1270. }
  1271. if(n && (*s == '0'))
  1272. {
  1273. if((n > 1) && ((s[1] == 'x') || (s[1] == 'X')))
  1274. {
  1275. radix = 16;
  1276. s +=2;
  1277. n -= 2;
  1278. }
  1279. else
  1280. {
  1281. radix = 8;
  1282. n -= 1;
  1283. }
  1284. }
  1285. //
  1286. // Exception guarentee: create the result in stack variable "result"
  1287. // then do a swap at the end. In the event of a throw, *this will
  1288. // be left unchanged.
  1289. //
  1290. cpp_int_backend result;
  1291. if(n)
  1292. {
  1293. if(radix == 16)
  1294. {
  1295. while(*s == '0') ++s;
  1296. std::size_t bitcount = 4 * std::strlen(s);
  1297. limb_type val;
  1298. std::size_t limb, shift;
  1299. if(bitcount > 4)
  1300. bitcount -= 4;
  1301. else
  1302. bitcount = 0;
  1303. std::size_t newsize = bitcount / (sizeof(limb_type) * CHAR_BIT) + 1;
  1304. result.resize(static_cast<unsigned>(newsize), static_cast<unsigned>(newsize)); // will throw if this is a checked integer that cannot be resized
  1305. std::memset(result.limbs(), 0, result.size() * sizeof(limb_type));
  1306. while(*s)
  1307. {
  1308. if(*s >= '0' && *s <= '9')
  1309. val = *s - '0';
  1310. else if(*s >= 'a' && *s <= 'f')
  1311. val = 10 + *s - 'a';
  1312. else if(*s >= 'A' && *s <= 'F')
  1313. val = 10 + *s - 'A';
  1314. else
  1315. {
  1316. BOOST_THROW_EXCEPTION(std::runtime_error("Unexpected content found while parsing character string."));
  1317. }
  1318. limb = bitcount / (sizeof(limb_type) * CHAR_BIT);
  1319. shift = bitcount % (sizeof(limb_type) * CHAR_BIT);
  1320. val <<= shift;
  1321. if(result.size() > limb)
  1322. {
  1323. result.limbs()[limb] |= val;
  1324. }
  1325. ++s;
  1326. bitcount -= 4;
  1327. }
  1328. result.normalize();
  1329. }
  1330. else if(radix == 8)
  1331. {
  1332. while(*s == '0') ++s;
  1333. std::size_t bitcount = 3 * std::strlen(s);
  1334. limb_type val;
  1335. std::size_t limb, shift;
  1336. if(bitcount > 3)
  1337. bitcount -= 3;
  1338. else
  1339. bitcount = 0;
  1340. std::size_t newsize = bitcount / (sizeof(limb_type) * CHAR_BIT) + 1;
  1341. result.resize(static_cast<unsigned>(newsize), static_cast<unsigned>(newsize)); // will throw if this is a checked integer that cannot be resized
  1342. std::memset(result.limbs(), 0, result.size() * sizeof(limb_type));
  1343. while(*s)
  1344. {
  1345. if(*s >= '0' && *s <= '7')
  1346. val = *s - '0';
  1347. else
  1348. {
  1349. BOOST_THROW_EXCEPTION(std::runtime_error("Unexpected content found while parsing character string."));
  1350. }
  1351. limb = bitcount / (sizeof(limb_type) * CHAR_BIT);
  1352. shift = bitcount % (sizeof(limb_type) * CHAR_BIT);
  1353. if(result.size() > limb)
  1354. {
  1355. result.limbs()[limb] |= (val << shift);
  1356. if(shift > sizeof(limb_type) * CHAR_BIT - 3)
  1357. {
  1358. // Deal with the bits in val that overflow into the next limb:
  1359. val >>= (sizeof(limb_type) * CHAR_BIT - shift);
  1360. if(val)
  1361. {
  1362. // If this is the most-significant-limb, we may need to allocate an extra one for the overflow:
  1363. if(limb + 1 == newsize)
  1364. result.resize(static_cast<unsigned>(newsize + 1), static_cast<unsigned>(newsize + 1));
  1365. if(result.size() > limb + 1)
  1366. {
  1367. result.limbs()[limb + 1] |= val;
  1368. }
  1369. }
  1370. }
  1371. }
  1372. ++s;
  1373. bitcount -= 3;
  1374. }
  1375. result.normalize();
  1376. }
  1377. else
  1378. {
  1379. // Base 10, we extract blocks of size 10^9 at a time, that way
  1380. // the number of multiplications is kept to a minimum:
  1381. limb_type block_mult = max_block_10;
  1382. while(*s)
  1383. {
  1384. limb_type block = 0;
  1385. for(unsigned i = 0; i < digits_per_block_10; ++i)
  1386. {
  1387. limb_type val;
  1388. if(*s >= '0' && *s <= '9')
  1389. val = *s - '0';
  1390. else
  1391. BOOST_THROW_EXCEPTION(std::runtime_error("Unexpected character encountered in input."));
  1392. block *= 10;
  1393. block += val;
  1394. if(!*++s)
  1395. {
  1396. block_mult = block_multiplier(i);
  1397. break;
  1398. }
  1399. }
  1400. eval_multiply(result, block_mult);
  1401. eval_add(result, block);
  1402. }
  1403. }
  1404. }
  1405. if(isneg)
  1406. result.negate();
  1407. result.swap(*this);
  1408. }
  1409. public:
  1410. cpp_int_backend& operator = (const char* s)
  1411. {
  1412. do_assign_string(s, trivial_tag());
  1413. return *this;
  1414. }
  1415. BOOST_MP_FORCEINLINE void swap(cpp_int_backend& o) BOOST_NOEXCEPT
  1416. {
  1417. this->do_swap(o);
  1418. }
  1419. private:
  1420. std::string do_get_trivial_string(std::ios_base::fmtflags f, const mpl::false_&)const
  1421. {
  1422. typedef typename mpl::if_c<sizeof(typename base_type::local_limb_type) == 1, unsigned, typename base_type::local_limb_type>::type io_type;
  1423. if(this->sign() && (((f & std::ios_base::hex) == std::ios_base::hex) || ((f & std::ios_base::oct) == std::ios_base::oct)))
  1424. BOOST_THROW_EXCEPTION(std::runtime_error("Base 8 or 16 printing of negative numbers is not supported."));
  1425. std::stringstream ss;
  1426. ss.flags(f & ~std::ios_base::showpos);
  1427. ss << static_cast<io_type>(*this->limbs());
  1428. std::string result;
  1429. if(this->sign())
  1430. result += '-';
  1431. else if(f & std::ios_base::showpos)
  1432. result += '+';
  1433. result += ss.str();
  1434. return result;
  1435. }
  1436. std::string do_get_trivial_string(std::ios_base::fmtflags f, const mpl::true_&)const
  1437. {
  1438. // Even though we have only one limb, we can't do IO on it :-(
  1439. int base = 10;
  1440. if((f & std::ios_base::oct) == std::ios_base::oct)
  1441. base = 8;
  1442. else if((f & std::ios_base::hex) == std::ios_base::hex)
  1443. base = 16;
  1444. std::string result;
  1445. unsigned Bits = sizeof(typename base_type::local_limb_type) * CHAR_BIT;
  1446. if(base == 8 || base == 16)
  1447. {
  1448. if(this->sign())
  1449. BOOST_THROW_EXCEPTION(std::runtime_error("Base 8 or 16 printing of negative numbers is not supported."));
  1450. limb_type shift = base == 8 ? 3 : 4;
  1451. limb_type mask = static_cast<limb_type>((1u << shift) - 1);
  1452. typename base_type::local_limb_type v = *this->limbs();
  1453. result.assign(Bits / shift + (Bits % shift ? 1 : 0), '0');
  1454. std::string::difference_type pos = result.size() - 1;
  1455. for(unsigned i = 0; i < Bits / shift; ++i)
  1456. {
  1457. char c = '0' + static_cast<char>(v & mask);
  1458. if(c > '9')
  1459. c += 'A' - '9' - 1;
  1460. result[pos--] = c;
  1461. v >>= shift;
  1462. }
  1463. if(Bits % shift)
  1464. {
  1465. mask = static_cast<limb_type>((1u << (Bits % shift)) - 1);
  1466. char c = '0' + static_cast<char>(v & mask);
  1467. if(c > '9')
  1468. c += 'A' - '9';
  1469. result[pos] = c;
  1470. }
  1471. //
  1472. // Get rid of leading zeros:
  1473. //
  1474. std::string::size_type n = result.find_first_not_of('0');
  1475. if(!result.empty() && (n == std::string::npos))
  1476. n = result.size() - 1;
  1477. result.erase(0, n);
  1478. if(f & std::ios_base::showbase)
  1479. {
  1480. const char* pp = base == 8 ? "0" : "0x";
  1481. result.insert(static_cast<std::string::size_type>(0), pp);
  1482. }
  1483. }
  1484. else
  1485. {
  1486. result.assign(Bits / 3 + 1, '0');
  1487. std::string::difference_type pos = result.size() - 1;
  1488. typename base_type::local_limb_type v(*this->limbs());
  1489. bool neg = false;
  1490. if(this->sign())
  1491. {
  1492. neg = true;
  1493. }
  1494. while(v)
  1495. {
  1496. result[pos] = (v % 10) + '0';
  1497. --pos;
  1498. v /= 10;
  1499. }
  1500. std::string::size_type n = result.find_first_not_of('0');
  1501. result.erase(0, n);
  1502. if(result.empty())
  1503. result = "0";
  1504. if(neg)
  1505. result.insert(static_cast<std::string::size_type>(0), 1, '-');
  1506. else if(f & std::ios_base::showpos)
  1507. result.insert(static_cast<std::string::size_type>(0), 1, '+');
  1508. }
  1509. return result;
  1510. }
  1511. std::string do_get_string(std::ios_base::fmtflags f, const mpl::true_&)const
  1512. {
  1513. #ifdef BOOST_MP_NO_DOUBLE_LIMB_TYPE_IO
  1514. return do_get_trivial_string(f, mpl::bool_<is_same<typename base_type::local_limb_type, double_limb_type>::value>());
  1515. #else
  1516. return do_get_trivial_string(f, mpl::bool_<false>());
  1517. #endif
  1518. }
  1519. std::string do_get_string(std::ios_base::fmtflags f, const mpl::false_&)const
  1520. {
  1521. using default_ops::eval_get_sign;
  1522. int base = 10;
  1523. if((f & std::ios_base::oct) == std::ios_base::oct)
  1524. base = 8;
  1525. else if((f & std::ios_base::hex) == std::ios_base::hex)
  1526. base = 16;
  1527. std::string result;
  1528. unsigned Bits = this->size() * base_type::limb_bits;
  1529. if(base == 8 || base == 16)
  1530. {
  1531. if(this->sign())
  1532. BOOST_THROW_EXCEPTION(std::runtime_error("Base 8 or 16 printing of negative numbers is not supported."));
  1533. limb_type shift = base == 8 ? 3 : 4;
  1534. limb_type mask = static_cast<limb_type>((1u << shift) - 1);
  1535. cpp_int_backend t(*this);
  1536. result.assign(Bits / shift + ((Bits % shift) ? 1 : 0), '0');
  1537. std::string::difference_type pos = result.size() - 1;
  1538. for(unsigned i = 0; i < Bits / shift; ++i)
  1539. {
  1540. char c = '0' + static_cast<char>(t.limbs()[0] & mask);
  1541. if(c > '9')
  1542. c += 'A' - '9' - 1;
  1543. result[pos--] = c;
  1544. eval_right_shift(t, shift);
  1545. }
  1546. if(Bits % shift)
  1547. {
  1548. mask = static_cast<limb_type>((1u << (Bits % shift)) - 1);
  1549. char c = '0' + static_cast<char>(t.limbs()[0] & mask);
  1550. if(c > '9')
  1551. c += 'A' - '9';
  1552. result[pos] = c;
  1553. }
  1554. //
  1555. // Get rid of leading zeros:
  1556. //
  1557. std::string::size_type n = result.find_first_not_of('0');
  1558. if(!result.empty() && (n == std::string::npos))
  1559. n = result.size() - 1;
  1560. result.erase(0, n);
  1561. if(f & std::ios_base::showbase)
  1562. {
  1563. const char* pp = base == 8 ? "0" : "0x";
  1564. result.insert(static_cast<std::string::size_type>(0), pp);
  1565. }
  1566. }
  1567. else
  1568. {
  1569. result.assign(Bits / 3 + 1, '0');
  1570. std::string::difference_type pos = result.size() - 1;
  1571. cpp_int_backend t(*this);
  1572. cpp_int_backend r;
  1573. bool neg = false;
  1574. if(t.sign())
  1575. {
  1576. t.negate();
  1577. neg = true;
  1578. }
  1579. if(this->size() == 1)
  1580. {
  1581. result = boost::lexical_cast<std::string>(t.limbs()[0]);
  1582. }
  1583. else
  1584. {
  1585. cpp_int_backend block10;
  1586. block10 = max_block_10;
  1587. while(eval_get_sign(t) != 0)
  1588. {
  1589. cpp_int_backend t2;
  1590. divide_unsigned_helper(&t2, t, block10, r);
  1591. t = t2;
  1592. limb_type v = r.limbs()[0];
  1593. for(unsigned i = 0; i < digits_per_block_10; ++i)
  1594. {
  1595. char c = '0' + v % 10;
  1596. v /= 10;
  1597. result[pos] = c;
  1598. if(pos-- == 0)
  1599. break;
  1600. }
  1601. }
  1602. }
  1603. std::string::size_type n = result.find_first_not_of('0');
  1604. result.erase(0, n);
  1605. if(result.empty())
  1606. result = "0";
  1607. if(neg)
  1608. result.insert(static_cast<std::string::size_type>(0), 1, '-');
  1609. else if(f & std::ios_base::showpos)
  1610. result.insert(static_cast<std::string::size_type>(0), 1, '+');
  1611. }
  1612. return result;
  1613. }
  1614. public:
  1615. std::string str(std::streamsize /*digits*/, std::ios_base::fmtflags f)const
  1616. {
  1617. return do_get_string(f, trivial_tag());
  1618. }
  1619. private:
  1620. template <class Container>
  1621. void construct_from_container(const Container& c, const mpl::false_&)
  1622. {
  1623. //
  1624. // We assume that c is a sequence of (unsigned) bytes with the most significant byte first:
  1625. //
  1626. unsigned newsize = static_cast<unsigned>(c.size() / sizeof(limb_type));
  1627. if(c.size() % sizeof(limb_type))
  1628. {
  1629. ++newsize;
  1630. }
  1631. if(newsize)
  1632. {
  1633. this->resize(newsize, newsize); // May throw
  1634. std::memset(this->limbs(), 0, this->size());
  1635. typename Container::const_iterator i(c.begin()), j(c.end());
  1636. unsigned byte_location = static_cast<unsigned>(c.size() - 1);
  1637. while(i != j)
  1638. {
  1639. unsigned limb = byte_location / sizeof(limb_type);
  1640. unsigned shift = (byte_location % sizeof(limb_type)) * CHAR_BIT;
  1641. if(this->size() > limb)
  1642. this->limbs()[limb] |= static_cast<limb_type>(static_cast<unsigned char>(*i)) << shift;
  1643. ++i;
  1644. --byte_location;
  1645. }
  1646. }
  1647. }
  1648. template <class Container>
  1649. void construct_from_container(const Container& c, const mpl::true_&)
  1650. {
  1651. //
  1652. // We assume that c is a sequence of (unsigned) bytes with the most significant byte first:
  1653. //
  1654. typedef typename base_type::local_limb_type local_limb_type;
  1655. *this->limbs() = 0;
  1656. if(c.size())
  1657. {
  1658. typename Container::const_iterator i(c.begin()), j(c.end());
  1659. unsigned byte_location = static_cast<unsigned>(c.size() - 1);
  1660. while(i != j)
  1661. {
  1662. unsigned limb = byte_location / sizeof(local_limb_type);
  1663. unsigned shift = (byte_location % sizeof(local_limb_type)) * CHAR_BIT;
  1664. if(limb == 0)
  1665. this->limbs()[0] |= static_cast<limb_type>(static_cast<unsigned char>(*i)) << shift;
  1666. ++i;
  1667. --byte_location;
  1668. }
  1669. }
  1670. }
  1671. public:
  1672. template <class Container>
  1673. cpp_int_backend(const Container& c, typename boost::enable_if_c<boost::multiprecision::detail::is_byte_container<Container>::value>::type const* = 0)
  1674. {
  1675. //
  1676. // We assume that c is a sequence of (unsigned) bytes with the most significant byte first:
  1677. //
  1678. construct_from_container(c, trivial_tag());
  1679. }
  1680. template <unsigned MinBits2, unsigned MaxBits2, cpp_integer_type SignType2, cpp_int_check_type Checked2, class Allocator2>
  1681. int compare_imp(const cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2>& o, const mpl::false_&, const mpl::false_&)const BOOST_NOEXCEPT
  1682. {
  1683. if(this->sign() != o.sign())
  1684. return this->sign() ? -1 : 1;
  1685. // Only do the compare if the same sign:
  1686. int result = compare_unsigned(o);
  1687. if(this->sign())
  1688. result = -result;
  1689. return result;
  1690. }
  1691. template <unsigned MinBits2, unsigned MaxBits2, cpp_integer_type SignType2, cpp_int_check_type Checked2, class Allocator2>
  1692. int compare_imp(const cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2>& o, const mpl::true_&, const mpl::false_&)const
  1693. {
  1694. cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2> t(*this);
  1695. return t.compare(o);
  1696. }
  1697. template <unsigned MinBits2, unsigned MaxBits2, cpp_integer_type SignType2, cpp_int_check_type Checked2, class Allocator2>
  1698. int compare_imp(const cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2>& o, const mpl::false_&, const mpl::true_&)const
  1699. {
  1700. cpp_int_backend<MinBits, MaxBits, SignType, Checked, Allocator> t(o);
  1701. return compare(t);
  1702. }
  1703. template <unsigned MinBits2, unsigned MaxBits2, cpp_integer_type SignType2, cpp_int_check_type Checked2, class Allocator2>
  1704. int compare_imp(const cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2>& o, const mpl::true_&, const mpl::true_&)const BOOST_NOEXCEPT
  1705. {
  1706. if(this->sign())
  1707. {
  1708. if(o.sign())
  1709. {
  1710. return *this->limbs() < *o.limbs() ? 1 : (*this->limbs() > *o.limbs() ? -1 : 0);
  1711. }
  1712. else
  1713. return -1;
  1714. }
  1715. else
  1716. {
  1717. if(o.sign())
  1718. return 1;
  1719. return *this->limbs() < *o.limbs() ? -1 : (*this->limbs() > *o.limbs() ? 1 : 0);
  1720. }
  1721. }
  1722. template <unsigned MinBits2, unsigned MaxBits2, cpp_integer_type SignType2, cpp_int_check_type Checked2, class Allocator2>
  1723. int compare(const cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2>& o)const BOOST_NOEXCEPT
  1724. {
  1725. typedef mpl::bool_<is_trivial_cpp_int<cpp_int_backend<MinBits, MaxBits, SignType, Checked, Allocator> >::value> t1;
  1726. typedef mpl::bool_<is_trivial_cpp_int<cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2> >::value> t2;
  1727. return compare_imp(o, t1(), t2());
  1728. }
  1729. template <unsigned MinBits2, unsigned MaxBits2, cpp_integer_type SignType2, cpp_int_check_type Checked2, class Allocator2>
  1730. int compare_unsigned(const cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2>& o)const BOOST_NOEXCEPT
  1731. {
  1732. if(this->size() != o.size())
  1733. {
  1734. return this->size() > o.size() ? 1 : -1;
  1735. }
  1736. typename base_type::const_limb_pointer pa = this->limbs();
  1737. typename base_type::const_limb_pointer pb = o.limbs();
  1738. for(int i = this->size() - 1; i >= 0; --i)
  1739. {
  1740. if(pa[i] != pb[i])
  1741. return pa[i] > pb[i] ? 1 : -1;
  1742. }
  1743. return 0;
  1744. }
  1745. template <class Arithmetic>
  1746. BOOST_MP_FORCEINLINE typename boost::enable_if<is_arithmetic<Arithmetic>, int>::type compare(Arithmetic i)const
  1747. {
  1748. // braindead version:
  1749. cpp_int_backend t;
  1750. t = i;
  1751. return compare(t);
  1752. }
  1753. };
  1754. } // namespace backends
  1755. namespace default_ops{
  1756. template <class Backend>
  1757. struct double_precision_type;
  1758. template <unsigned MinBits, unsigned MaxBits, cpp_integer_type SignType, cpp_int_check_type Checked, class Allocator>
  1759. struct double_precision_type<backends::cpp_int_backend<MinBits, MaxBits, SignType, Checked, Allocator> >
  1760. {
  1761. typedef typename mpl::if_c<
  1762. backends::is_fixed_precision<backends::cpp_int_backend<MinBits, MaxBits, SignType, Checked, Allocator> >::value,
  1763. backends::cpp_int_backend<
  1764. (is_void<Allocator>::value ?
  1765. 2 * backends::max_precision<backends::cpp_int_backend<MinBits, MaxBits, SignType, Checked, Allocator> >::value
  1766. : MinBits),
  1767. 2 * backends::max_precision<backends::cpp_int_backend<MinBits, MaxBits, SignType, Checked, Allocator> >::value,
  1768. SignType,
  1769. Checked,
  1770. Allocator>,
  1771. backends::cpp_int_backend<MinBits, MaxBits, SignType, Checked, Allocator>
  1772. >::type type;
  1773. };
  1774. }
  1775. template <unsigned MinBits, unsigned MaxBits, cpp_integer_type SignType, cpp_int_check_type Checked>
  1776. struct expression_template_default<backends::cpp_int_backend<MinBits, MaxBits, SignType, Checked, void> >
  1777. {
  1778. static const expression_template_option value = et_off;
  1779. };
  1780. using boost::multiprecision::backends::cpp_int_backend;
  1781. template <unsigned MinBits, unsigned MaxBits, cpp_integer_type SignType, cpp_int_check_type Checked, class Allocator>
  1782. struct number_category<cpp_int_backend<MinBits, MaxBits, SignType, Checked, Allocator> > : public mpl::int_<number_kind_integer>{};
  1783. typedef number<cpp_int_backend<> > cpp_int;
  1784. typedef rational_adaptor<cpp_int_backend<> > cpp_rational_backend;
  1785. typedef number<cpp_rational_backend> cpp_rational;
  1786. // Fixed precision unsigned types:
  1787. typedef number<cpp_int_backend<128, 128, unsigned_magnitude, unchecked, void> > uint128_t;
  1788. typedef number<cpp_int_backend<256, 256, unsigned_magnitude, unchecked, void> > uint256_t;
  1789. typedef number<cpp_int_backend<512, 512, unsigned_magnitude, unchecked, void> > uint512_t;
  1790. typedef number<cpp_int_backend<1024, 1024, unsigned_magnitude, unchecked, void> > uint1024_t;
  1791. // Fixed precision signed types:
  1792. typedef number<cpp_int_backend<128, 128, signed_magnitude, unchecked, void> > int128_t;
  1793. typedef number<cpp_int_backend<256, 256, signed_magnitude, unchecked, void> > int256_t;
  1794. typedef number<cpp_int_backend<512, 512, signed_magnitude, unchecked, void> > int512_t;
  1795. typedef number<cpp_int_backend<1024, 1024, signed_magnitude, unchecked, void> > int1024_t;
  1796. // Over again, but with checking enabled this time:
  1797. typedef number<cpp_int_backend<0, 0, signed_magnitude, checked> > checked_cpp_int;
  1798. typedef rational_adaptor<cpp_int_backend<0, 0, signed_magnitude, checked> > checked_cpp_rational_backend;
  1799. typedef number<checked_cpp_rational_backend> checked_cpp_rational;
  1800. // Fixed precision unsigned types:
  1801. typedef number<cpp_int_backend<128, 128, unsigned_magnitude, checked, void> > checked_uint128_t;
  1802. typedef number<cpp_int_backend<256, 256, unsigned_magnitude, checked, void> > checked_uint256_t;
  1803. typedef number<cpp_int_backend<512, 512, unsigned_magnitude, checked, void> > checked_uint512_t;
  1804. typedef number<cpp_int_backend<1024, 1024, unsigned_magnitude, checked, void> > checked_uint1024_t;
  1805. // Fixed precision signed types:
  1806. typedef number<cpp_int_backend<128, 128, signed_magnitude, checked, void> > checked_int128_t;
  1807. typedef number<cpp_int_backend<256, 256, signed_magnitude, checked, void> > checked_int256_t;
  1808. typedef number<cpp_int_backend<512, 512, signed_magnitude, checked, void> > checked_int512_t;
  1809. typedef number<cpp_int_backend<1024, 1024, signed_magnitude, checked, void> > checked_int1024_t;
  1810. #ifdef BOOST_NO_SFINAE_EXPR
  1811. namespace detail{
  1812. template<unsigned MinBits, unsigned MaxBits, cpp_integer_type SignType, cpp_int_check_type Checked, class Allocator, unsigned MinBits2, unsigned MaxBits2, cpp_integer_type SignType2, cpp_int_check_type Checked2, class Allocator2>
  1813. struct is_explicitly_convertible<cpp_int_backend<MinBits, MaxBits, SignType, Checked, Allocator>, cpp_int_backend<MinBits2, MaxBits2, SignType2, Checked2, Allocator2> > : public mpl::true_ {};
  1814. }
  1815. #endif
  1816. }} // namespaces
  1817. //
  1818. // Last of all we include the implementations of all the eval_* non member functions:
  1819. //
  1820. #include <boost/multiprecision/cpp_int/comparison.hpp>
  1821. #include <boost/multiprecision/cpp_int/add.hpp>
  1822. #include <boost/multiprecision/cpp_int/multiply.hpp>
  1823. #include <boost/multiprecision/cpp_int/divide.hpp>
  1824. #include <boost/multiprecision/cpp_int/bitwise.hpp>
  1825. #include <boost/multiprecision/cpp_int/misc.hpp>
  1826. #include <boost/multiprecision/cpp_int/limits.hpp>
  1827. #ifdef BOOST_MP_USER_DEFINED_LITERALS
  1828. #include <boost/multiprecision/cpp_int/literals.hpp>
  1829. #endif
  1830. #include <boost/multiprecision/cpp_int/serialize.hpp>
  1831. #include <boost/multiprecision/cpp_int/import_export.hpp>
  1832. #endif