number.hpp 77 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950
  1. ///////////////////////////////////////////////////////////////////////////////
  2. // Copyright 2011 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_0.txt)
  5. #ifndef BOOST_MATH_EXTENDED_REAL_HPP
  6. #define BOOST_MATH_EXTENDED_REAL_HPP
  7. #include <boost/cstdint.hpp>
  8. #include <boost/mpl/max.hpp>
  9. #include <boost/mpl/plus.hpp>
  10. #include <boost/mpl/or.hpp>
  11. #include <boost/mpl/find_if.hpp>
  12. #include <boost/assert.hpp>
  13. #include <boost/type_traits/remove_pointer.hpp>
  14. #include <boost/type_traits/is_signed.hpp>
  15. #include <boost/type_traits/is_unsigned.hpp>
  16. #include <boost/type_traits/is_floating_point.hpp>
  17. #include <boost/type_traits/is_integral.hpp>
  18. #include <boost/type_traits/make_unsigned.hpp>
  19. #include <boost/throw_exception.hpp>
  20. #include <boost/multiprecision/detail/generic_interconvert.hpp>
  21. #include <boost/multiprecision/detail/number_compare.hpp>
  22. #include <boost/multiprecision/traits/is_restricted_conversion.hpp>
  23. #include <istream> // stream operators
  24. #include <cstdio> // EOF
  25. #include <cctype> // isspace
  26. namespace boost{ namespace multiprecision{
  27. #ifdef BOOST_MSVC
  28. // warning C4127: conditional expression is constant
  29. // warning C4714: function marked as __forceinline not inlined
  30. #pragma warning(push)
  31. #pragma warning(disable:4127 4714 6326)
  32. #endif
  33. template <class Backend, expression_template_option ExpressionTemplates>
  34. class number
  35. {
  36. typedef number<Backend, ExpressionTemplates> self_type;
  37. public:
  38. typedef Backend backend_type;
  39. BOOST_MP_FORCEINLINE BOOST_CONSTEXPR number() BOOST_MP_NOEXCEPT_IF(noexcept(Backend())) {}
  40. BOOST_MP_FORCEINLINE BOOST_CONSTEXPR number(const number& e) BOOST_MP_NOEXCEPT_IF(noexcept(Backend(std::declval<Backend const&>()))) : m_backend(e.m_backend){}
  41. template <class V>
  42. BOOST_MP_FORCEINLINE number(const V& v, typename boost::enable_if_c<
  43. (boost::is_arithmetic<V>::value || is_same<std::string, V>::value || is_convertible<V, const char*>::value)
  44. && !is_convertible<typename detail::canonical<V, Backend>::type, Backend>::value
  45. && !detail::is_restricted_conversion<typename detail::canonical<V, Backend>::type, Backend>::value
  46. >::type* = 0)
  47. {
  48. m_backend = canonical_value(v);
  49. }
  50. template <class V>
  51. BOOST_MP_FORCEINLINE BOOST_CONSTEXPR number(const V& v, typename boost::enable_if_c<
  52. is_convertible<typename detail::canonical<V, Backend>::type, Backend>::value
  53. && !detail::is_restricted_conversion<typename detail::canonical<V, Backend>::type, Backend>::value
  54. >::type* = 0)
  55. #ifndef BOOST_INTEL
  56. BOOST_MP_NOEXCEPT_IF(noexcept(Backend(std::declval<typename detail::canonical<V, Backend>::type const&>())))
  57. #endif
  58. : m_backend(canonical_value(v)) {}
  59. BOOST_MP_FORCEINLINE BOOST_CONSTEXPR number(const number& e, unsigned digits10)
  60. BOOST_MP_NOEXCEPT_IF(noexcept(Backend(std::declval<Backend const&>(), std::declval<unsigned>())))
  61. : m_backend(e.m_backend, digits10){}
  62. template <class V>
  63. explicit BOOST_MP_FORCEINLINE number(const V& v, typename boost::enable_if_c<
  64. (boost::is_arithmetic<V>::value || is_same<std::string, V>::value || is_convertible<V, const char*>::value)
  65. && !detail::is_explicitly_convertible<typename detail::canonical<V, Backend>::type, Backend>::value
  66. && detail::is_restricted_conversion<typename detail::canonical<V, Backend>::type, Backend>::value
  67. >::type* = 0)
  68. BOOST_MP_NOEXCEPT_IF(noexcept(std::declval<Backend&>() = std::declval<typename detail::canonical<V, Backend>::type const&>()))
  69. {
  70. m_backend = canonical_value(v);
  71. }
  72. template <class V>
  73. explicit BOOST_MP_FORCEINLINE BOOST_CONSTEXPR number(const V& v, typename boost::enable_if_c<
  74. detail::is_explicitly_convertible<typename detail::canonical<V, Backend>::type, Backend>::value
  75. && (detail::is_restricted_conversion<typename detail::canonical<V, Backend>::type, Backend>::value
  76. || !is_convertible<typename detail::canonical<V, Backend>::type, Backend>::value)
  77. >::type* = 0)
  78. BOOST_MP_NOEXCEPT_IF(noexcept(Backend(std::declval<typename detail::canonical<V, Backend>::type const&>())))
  79. : m_backend(canonical_value(v)) {}
  80. /*
  81. //
  82. // This conflicts with component based initialization (for rational and complex types)
  83. // which is arguably more useful. Disabled for now.
  84. //
  85. template <class V>
  86. number(V v, unsigned digits10, typename boost::enable_if<mpl::or_<boost::is_arithmetic<V>, is_same<std::string, V>, is_convertible<V, const char*> > >::type* dummy1 = 0)
  87. {
  88. m_backend.precision(digits10);
  89. m_backend = canonical_value(v);
  90. }
  91. */
  92. template<expression_template_option ET>
  93. BOOST_MP_FORCEINLINE BOOST_CONSTEXPR number(const number<Backend, ET>& val)
  94. BOOST_MP_NOEXCEPT_IF(noexcept(Backend(std::declval<Backend const&>()))) : m_backend(val.backend()) {}
  95. template <class Other, expression_template_option ET>
  96. BOOST_MP_FORCEINLINE number(const number<Other, ET>& val,
  97. typename boost::enable_if_c<(boost::is_convertible<Other, Backend>::value && !detail::is_restricted_conversion<Other, Backend>::value)>::type* = 0)
  98. BOOST_MP_NOEXCEPT_IF(noexcept(Backend(std::declval<Other const&>())))
  99. : m_backend(val.backend()) {}
  100. template <class Other, expression_template_option ET>
  101. explicit number(const number<Other, ET>& val, typename boost::enable_if_c<
  102. (!detail::is_explicitly_convertible<Other, Backend>::value)
  103. >::type* = 0)
  104. {
  105. //
  106. // Attempt a generic interconvertion:
  107. //
  108. detail::generic_interconvert(backend(), val.backend(), number_category<Backend>(), number_category<Other>());
  109. }
  110. template <class Other, expression_template_option ET>
  111. explicit BOOST_MP_FORCEINLINE number(const number<Other, ET>& val, typename boost::enable_if_c<
  112. (detail::is_explicitly_convertible<Other, Backend>::value
  113. && (detail::is_restricted_conversion<Other, Backend>::value || !boost::is_convertible<Other, Backend>::value))
  114. >::type* = 0) BOOST_MP_NOEXCEPT_IF(noexcept(Backend(std::declval<Other const&>())))
  115. : m_backend(val.backend()) {}
  116. template <class V>
  117. BOOST_MP_FORCEINLINE number(V v1, V v2, typename boost::enable_if<mpl::or_<boost::is_arithmetic<V>, is_same<std::string, V>, is_convertible<V, const char*> > >::type* = 0)
  118. {
  119. using default_ops::assign_components;
  120. assign_components(m_backend, canonical_value(v1), canonical_value(v2));
  121. }
  122. template <class Other, expression_template_option ET>
  123. BOOST_MP_FORCEINLINE number(const number<Other, ET>& v1, const number<Other, ET>& v2, typename boost::enable_if<boost::is_convertible<Other, Backend> >::type* = 0)
  124. {
  125. using default_ops::assign_components;
  126. assign_components(m_backend, v1.backend(), v2.backend());
  127. }
  128. template <class tag, class Arg1, class Arg2, class Arg3, class Arg4>
  129. typename boost::enable_if<is_convertible<typename detail::expression<tag, Arg1, Arg2, Arg3, Arg4>::result_type, self_type>, number&>::type operator=(const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& e)
  130. {
  131. typedef typename is_same<number, typename detail::expression<tag, Arg1, Arg2, Arg3, Arg4>::result_type>::type tag_type;
  132. do_assign(e, tag_type());
  133. return *this;
  134. }
  135. template <class tag, class Arg1, class Arg2, class Arg3, class Arg4>
  136. number& assign(const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& e)
  137. {
  138. typedef typename is_same<number, typename detail::expression<tag, Arg1, Arg2, Arg3, Arg4>::result_type>::type tag_type;
  139. do_assign(e, tag_type());
  140. return *this;
  141. }
  142. BOOST_MP_FORCEINLINE number& operator=(const number& e)
  143. BOOST_MP_NOEXCEPT_IF(noexcept(std::declval<Backend&>() = std::declval<Backend const&>()))
  144. {
  145. m_backend = e.m_backend;
  146. return *this;
  147. }
  148. template <class V>
  149. BOOST_MP_FORCEINLINE typename boost::enable_if<is_convertible<V, self_type>, number<Backend, ExpressionTemplates>& >::type
  150. operator=(const V& v)
  151. BOOST_MP_NOEXCEPT_IF(noexcept(std::declval<Backend&>() = std::declval<const typename detail::canonical<V, Backend>::type&>()))
  152. {
  153. m_backend = canonical_value(v);
  154. return *this;
  155. }
  156. template <class V>
  157. BOOST_MP_FORCEINLINE number<Backend, ExpressionTemplates>& assign(const V& v)
  158. BOOST_MP_NOEXCEPT_IF(noexcept(std::declval<Backend&>() = std::declval<const typename detail::canonical<V, Backend>::type&>()))
  159. {
  160. m_backend = canonical_value(v);
  161. return *this;
  162. }
  163. template <class Other, expression_template_option ET>
  164. typename boost::disable_if<boost::multiprecision::detail::is_explicitly_convertible<Other, Backend>, number<Backend, ExpressionTemplates>& >::type
  165. assign(const number<Other, ET>& v)
  166. {
  167. //
  168. // Attempt a generic interconvertion:
  169. //
  170. detail::generic_interconvert(backend(), v.backend(), number_category<Backend>(), number_category<Other>());
  171. return *this;
  172. }
  173. template <class tag, class Arg1, class Arg2, class Arg3, class Arg4>
  174. number(const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& e, typename boost::enable_if_c<is_convertible<typename detail::expression<tag, Arg1, Arg2, Arg3, Arg4>::result_type, self_type>::value>::type* = 0)
  175. {
  176. *this = e;
  177. }
  178. template <class tag, class Arg1, class Arg2, class Arg3, class Arg4>
  179. explicit number(const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& e,
  180. typename boost::enable_if_c<!is_convertible<typename detail::expression<tag, Arg1, Arg2, Arg3, Arg4>::result_type, self_type>::value
  181. && boost::multiprecision::detail::is_explicitly_convertible<typename detail::expression<tag, Arg1, Arg2, Arg3, Arg4>::result_type, self_type>::value>::type* = 0)
  182. {
  183. assign(e);
  184. }
  185. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  186. BOOST_MP_FORCEINLINE BOOST_CONSTEXPR number(number&& r)
  187. BOOST_MP_NOEXCEPT_IF(noexcept(Backend(std::declval<Backend>())))
  188. : m_backend(static_cast<Backend&&>(r.m_backend)){}
  189. BOOST_MP_FORCEINLINE number& operator=(number&& r) BOOST_MP_NOEXCEPT_IF(noexcept(std::declval<Backend&>() = std::declval<Backend>()))
  190. {
  191. m_backend = static_cast<Backend&&>(r.m_backend);
  192. return *this;
  193. }
  194. #endif
  195. number& operator+=(const self_type& val)
  196. {
  197. do_add(detail::expression<detail::terminal, self_type>(val), detail::terminal());
  198. return *this;
  199. }
  200. template <class tag, class Arg1, class Arg2, class Arg3, class Arg4>
  201. number& operator+=(const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& e)
  202. {
  203. // Create a copy if e contains this, but not if we're just doing a
  204. // x += x
  205. if(contains_self(e) && !is_self(e))
  206. {
  207. self_type temp(e);
  208. do_add(detail::expression<detail::terminal, self_type>(temp), detail::terminal());
  209. }
  210. else
  211. {
  212. do_add(e, tag());
  213. }
  214. return *this;
  215. }
  216. template <class Arg1, class Arg2, class Arg3, class Arg4>
  217. number& operator+=(const detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>& e)
  218. {
  219. //
  220. // Fused multiply-add:
  221. //
  222. using default_ops::eval_multiply_add;
  223. eval_multiply_add(m_backend, canonical_value(e.left_ref()), canonical_value(e.right_ref()));
  224. return *this;
  225. }
  226. template <class V>
  227. typename boost::enable_if<boost::is_convertible<V, self_type>, number<Backend, ExpressionTemplates>& >::type
  228. operator+=(const V& v)
  229. {
  230. using default_ops::eval_add;
  231. eval_add(m_backend, canonical_value(v));
  232. return *this;
  233. }
  234. number& operator-=(const self_type& val)
  235. {
  236. do_subtract(detail::expression<detail::terminal, self_type>(val), detail::terminal());
  237. return *this;
  238. }
  239. template <class tag, class Arg1, class Arg2, class Arg3, class Arg4>
  240. number& operator-=(const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& e)
  241. {
  242. // Create a copy if e contains this:
  243. if(contains_self(e))
  244. {
  245. self_type temp(e);
  246. do_subtract(detail::expression<detail::terminal, self_type>(temp), detail::terminal());
  247. }
  248. else
  249. {
  250. do_subtract(e, typename detail::expression<tag, Arg1, Arg2, Arg3, Arg4>::tag_type());
  251. }
  252. return *this;
  253. }
  254. template <class V>
  255. typename boost::enable_if<boost::is_convertible<V, self_type>, number<Backend, ExpressionTemplates>& >::type
  256. operator-=(const V& v)
  257. {
  258. using default_ops::eval_subtract;
  259. eval_subtract(m_backend, canonical_value(v));
  260. return *this;
  261. }
  262. template <class Arg1, class Arg2, class Arg3, class Arg4>
  263. number& operator-=(const detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>& e)
  264. {
  265. //
  266. // Fused multiply-subtract:
  267. //
  268. using default_ops::eval_multiply_subtract;
  269. eval_multiply_subtract(m_backend, canonical_value(e.left_ref()), canonical_value(e.right_ref()));
  270. return *this;
  271. }
  272. number& operator *= (const self_type& e)
  273. {
  274. do_multiplies(detail::expression<detail::terminal, self_type>(e), detail::terminal());
  275. return *this;
  276. }
  277. template <class tag, class Arg1, class Arg2, class Arg3, class Arg4>
  278. number& operator*=(const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& e)
  279. {
  280. // Create a temporary if the RHS references *this, but not
  281. // if we're just doing an x *= x;
  282. if(contains_self(e) && !is_self(e))
  283. {
  284. self_type temp(e);
  285. do_multiplies(detail::expression<detail::terminal, self_type>(temp), detail::terminal());
  286. }
  287. else
  288. {
  289. do_multiplies(e, typename detail::expression<tag, Arg1, Arg2, Arg3, Arg4>::tag_type());
  290. }
  291. return *this;
  292. }
  293. template <class V>
  294. typename boost::enable_if<boost::is_convertible<V, self_type>, number<Backend, ExpressionTemplates>& >::type
  295. operator*=(const V& v)
  296. {
  297. using default_ops::eval_multiply;
  298. eval_multiply(m_backend, canonical_value(v));
  299. return *this;
  300. }
  301. number& operator%=(const self_type& e)
  302. {
  303. BOOST_STATIC_ASSERT_MSG(number_category<Backend>::value == number_kind_integer, "The modulus operation is only valid for integer types");
  304. do_modulus(detail::expression<detail::terminal, self_type>(e), detail::terminal());
  305. return *this;
  306. }
  307. template <class tag, class Arg1, class Arg2, class Arg3, class Arg4>
  308. number& operator%=(const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& e)
  309. {
  310. BOOST_STATIC_ASSERT_MSG(number_category<Backend>::value == number_kind_integer, "The modulus operation is only valid for integer types");
  311. // Create a temporary if the RHS references *this:
  312. if(contains_self(e))
  313. {
  314. self_type temp(e);
  315. do_modulus(detail::expression<detail::terminal, self_type>(temp), detail::terminal());
  316. }
  317. else
  318. {
  319. do_modulus(e, typename detail::expression<tag, Arg1, Arg2, Arg3, Arg4>::tag_type());
  320. }
  321. return *this;
  322. }
  323. template <class V>
  324. typename boost::enable_if<boost::is_convertible<V, self_type>, number<Backend, ExpressionTemplates>& >::type
  325. operator%=(const V& v)
  326. {
  327. BOOST_STATIC_ASSERT_MSG(number_category<Backend>::value == number_kind_integer, "The modulus operation is only valid for integer types");
  328. using default_ops::eval_modulus;
  329. eval_modulus(m_backend, canonical_value(v));
  330. return *this;
  331. }
  332. //
  333. // These operators are *not* proto-ized.
  334. // The issue is that the increment/decrement must happen
  335. // even if the result of the operator *is never used*.
  336. // Possibly we could modify our expression wrapper to
  337. // execute the increment/decrement on destruction, but
  338. // correct implementation will be tricky, so defered for now...
  339. //
  340. BOOST_MP_FORCEINLINE number& operator++()
  341. {
  342. using default_ops::eval_increment;
  343. eval_increment(m_backend);
  344. return *this;
  345. }
  346. BOOST_MP_FORCEINLINE number& operator--()
  347. {
  348. using default_ops::eval_decrement;
  349. eval_decrement(m_backend);
  350. return *this;
  351. }
  352. inline number operator++(int)
  353. {
  354. using default_ops::eval_increment;
  355. self_type temp(*this);
  356. eval_increment(m_backend);
  357. return BOOST_MP_MOVE(temp);
  358. }
  359. inline number operator--(int)
  360. {
  361. using default_ops::eval_decrement;
  362. self_type temp(*this);
  363. eval_decrement(m_backend);
  364. return BOOST_MP_MOVE(temp);
  365. }
  366. template <class V>
  367. BOOST_MP_FORCEINLINE typename boost::enable_if<is_integral<V>, number&>::type operator <<= (V val)
  368. {
  369. BOOST_STATIC_ASSERT_MSG(number_category<Backend>::value == number_kind_integer, "The left-shift operation is only valid for integer types");
  370. detail::check_shift_range(val, mpl::bool_<(sizeof(V) > sizeof(std::size_t))>(), is_signed<V>());
  371. eval_left_shift(m_backend, static_cast<std::size_t>(canonical_value(val)));
  372. return *this;
  373. }
  374. template <class V>
  375. BOOST_MP_FORCEINLINE typename boost::enable_if<is_integral<V>, number&>::type operator >>= (V val)
  376. {
  377. BOOST_STATIC_ASSERT_MSG(number_category<Backend>::value == number_kind_integer, "The right-shift operation is only valid for integer types");
  378. detail::check_shift_range(val, mpl::bool_<(sizeof(V) > sizeof(std::size_t))>(), is_signed<V>());
  379. eval_right_shift(m_backend, static_cast<std::size_t>(canonical_value(val)));
  380. return *this;
  381. }
  382. BOOST_MP_FORCEINLINE number& operator /= (const self_type& e)
  383. {
  384. do_divide(detail::expression<detail::terminal, self_type>(e), detail::terminal());
  385. return *this;
  386. }
  387. template <class tag, class Arg1, class Arg2, class Arg3, class Arg4>
  388. number& operator/=(const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& e)
  389. {
  390. // Create a temporary if the RHS references *this:
  391. if(contains_self(e))
  392. {
  393. self_type temp(e);
  394. do_divide(detail::expression<detail::terminal, self_type>(temp), detail::terminal());
  395. }
  396. else
  397. {
  398. do_divide(e, typename detail::expression<tag, Arg1, Arg2, Arg3, Arg4>::tag_type());
  399. }
  400. return *this;
  401. }
  402. template <class V>
  403. BOOST_MP_FORCEINLINE typename boost::enable_if<boost::is_convertible<V, self_type>, number<Backend, ExpressionTemplates>& >::type
  404. operator/=(const V& v)
  405. {
  406. using default_ops::eval_divide;
  407. eval_divide(m_backend, canonical_value(v));
  408. return *this;
  409. }
  410. BOOST_MP_FORCEINLINE number& operator&=(const self_type& e)
  411. {
  412. BOOST_STATIC_ASSERT_MSG(number_category<Backend>::value == number_kind_integer, "The bitwise & operation is only valid for integer types");
  413. do_bitwise_and(detail::expression<detail::terminal, self_type>(e), detail::terminal());
  414. return *this;
  415. }
  416. template <class tag, class Arg1, class Arg2, class Arg3, class Arg4>
  417. number& operator&=(const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& e)
  418. {
  419. BOOST_STATIC_ASSERT_MSG(number_category<Backend>::value == number_kind_integer, "The bitwise & operation is only valid for integer types");
  420. // Create a temporary if the RHS references *this, but not
  421. // if we're just doing an x &= x;
  422. if(contains_self(e) && !is_self(e))
  423. {
  424. self_type temp(e);
  425. do_bitwise_and(detail::expression<detail::terminal, self_type>(temp), detail::terminal());
  426. }
  427. else
  428. {
  429. do_bitwise_and(e, typename detail::expression<tag, Arg1, Arg2, Arg3, Arg4>::tag_type());
  430. }
  431. return *this;
  432. }
  433. template <class V>
  434. BOOST_MP_FORCEINLINE typename boost::enable_if<boost::is_convertible<V, self_type>, number<Backend, ExpressionTemplates>& >::type
  435. operator&=(const V& v)
  436. {
  437. BOOST_STATIC_ASSERT_MSG(number_category<Backend>::value == number_kind_integer, "The bitwise & operation is only valid for integer types");
  438. using default_ops::eval_bitwise_and;
  439. eval_bitwise_and(m_backend, canonical_value(v));
  440. return *this;
  441. }
  442. BOOST_MP_FORCEINLINE number& operator|=(const self_type& e)
  443. {
  444. BOOST_STATIC_ASSERT_MSG(number_category<Backend>::value == number_kind_integer, "The bitwise | operation is only valid for integer types");
  445. do_bitwise_or(detail::expression<detail::terminal, self_type>(e), detail::terminal());
  446. return *this;
  447. }
  448. template <class tag, class Arg1, class Arg2, class Arg3, class Arg4>
  449. number& operator|=(const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& e)
  450. {
  451. BOOST_STATIC_ASSERT_MSG(number_category<Backend>::value == number_kind_integer, "The bitwise | operation is only valid for integer types");
  452. // Create a temporary if the RHS references *this, but not
  453. // if we're just doing an x |= x;
  454. if(contains_self(e) && !is_self(e))
  455. {
  456. self_type temp(e);
  457. do_bitwise_or(detail::expression<detail::terminal, self_type>(temp), detail::terminal());
  458. }
  459. else
  460. {
  461. do_bitwise_or(e, typename detail::expression<tag, Arg1, Arg2, Arg3, Arg4>::tag_type());
  462. }
  463. return *this;
  464. }
  465. template <class V>
  466. BOOST_MP_FORCEINLINE typename boost::enable_if<boost::is_convertible<V, self_type>, number<Backend, ExpressionTemplates>& >::type
  467. operator|=(const V& v)
  468. {
  469. BOOST_STATIC_ASSERT_MSG(number_category<Backend>::value == number_kind_integer, "The bitwise | operation is only valid for integer types");
  470. using default_ops::eval_bitwise_or;
  471. eval_bitwise_or(m_backend, canonical_value(v));
  472. return *this;
  473. }
  474. BOOST_MP_FORCEINLINE number& operator^=(const self_type& e)
  475. {
  476. BOOST_STATIC_ASSERT_MSG(number_category<Backend>::value == number_kind_integer, "The bitwise ^ operation is only valid for integer types");
  477. do_bitwise_xor(detail::expression<detail::terminal, self_type>(e), detail::terminal());
  478. return *this;
  479. }
  480. template <class tag, class Arg1, class Arg2, class Arg3, class Arg4>
  481. number& operator^=(const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& e)
  482. {
  483. BOOST_STATIC_ASSERT_MSG(number_category<Backend>::value == number_kind_integer, "The bitwise ^ operation is only valid for integer types");
  484. if(contains_self(e))
  485. {
  486. self_type temp(e);
  487. do_bitwise_xor(detail::expression<detail::terminal, self_type>(temp), detail::terminal());
  488. }
  489. else
  490. {
  491. do_bitwise_xor(e, typename detail::expression<tag, Arg1, Arg2, Arg3, Arg4>::tag_type());
  492. }
  493. return *this;
  494. }
  495. template <class V>
  496. BOOST_MP_FORCEINLINE typename boost::enable_if<boost::is_convertible<V, self_type>, number<Backend, ExpressionTemplates>& >::type
  497. operator^=(const V& v)
  498. {
  499. BOOST_STATIC_ASSERT_MSG(number_category<Backend>::value == number_kind_integer, "The bitwise ^ operation is only valid for integer types");
  500. using default_ops::eval_bitwise_xor;
  501. eval_bitwise_xor(m_backend, canonical_value(v));
  502. return *this;
  503. }
  504. //
  505. // swap:
  506. //
  507. BOOST_MP_FORCEINLINE void swap(self_type& other) BOOST_MP_NOEXCEPT_IF(noexcept(std::declval<Backend>().swap(std::declval<Backend&>())))
  508. {
  509. m_backend.swap(other.backend());
  510. }
  511. //
  512. // Zero and sign:
  513. //
  514. BOOST_MP_FORCEINLINE bool is_zero()const
  515. {
  516. using default_ops::eval_is_zero;
  517. return eval_is_zero(m_backend);
  518. }
  519. BOOST_MP_FORCEINLINE int sign()const
  520. {
  521. using default_ops::eval_get_sign;
  522. return eval_get_sign(m_backend);
  523. }
  524. //
  525. // String conversion functions:
  526. //
  527. std::string str(std::streamsize digits = 0, std::ios_base::fmtflags f = std::ios_base::fmtflags(0))const
  528. {
  529. return m_backend.str(digits, f);
  530. }
  531. template<class Archive>
  532. void serialize(Archive & ar, const unsigned int /*version*/)
  533. {
  534. ar & m_backend;
  535. }
  536. private:
  537. template <class T>
  538. void convert_to_imp(T* result)const
  539. {
  540. using default_ops::eval_convert_to;
  541. eval_convert_to(result, m_backend);
  542. }
  543. template <class B2, expression_template_option ET>
  544. typename enable_if_c<detail::is_explicitly_convertible<Backend, B2>::value>::type convert_to_imp(number<B2, ET>* result)const
  545. {
  546. result->assign(*this);
  547. }
  548. void convert_to_imp(std::string* result)const
  549. {
  550. *result = this->str();
  551. }
  552. public:
  553. template <class T>
  554. T convert_to()const
  555. {
  556. T result;
  557. convert_to_imp(&result);
  558. return result;
  559. }
  560. //
  561. // Use in boolean context, and explicit conversion operators:
  562. //
  563. #ifndef BOOST_MP_NO_CXX11_EXPLICIT_CONVERSION_OPERATORS
  564. # if (defined(__GNUC__) && (__GNUC__ == 4) && (__GNUC_MINOR__ < 7)) || (defined(BOOST_INTEL) && (BOOST_INTEL <= 1500))
  565. //
  566. // Horrible workaround for gcc-4.6.x which always prefers the template
  567. // operator bool() rather than the non-template operator when converting to
  568. // an arithmetic type:
  569. //
  570. template <class T, typename boost::enable_if<is_same<T, bool>, int>::type = 0>
  571. explicit operator T ()const
  572. {
  573. using default_ops::eval_is_zero;
  574. return !eval_is_zero(backend());
  575. }
  576. template <class T, typename boost::disable_if_c<is_same<T, bool>::value || is_void<T>::value, int>::type = 0>
  577. explicit operator T ()const
  578. {
  579. return this->template convert_to<T>();
  580. }
  581. # else
  582. template <class T>
  583. explicit operator T()const
  584. {
  585. return this->template convert_to<T>();
  586. }
  587. BOOST_MP_FORCEINLINE explicit operator bool()const
  588. {
  589. return !is_zero();
  590. }
  591. explicit operator void()const {}
  592. # endif
  593. #else
  594. typedef bool (self_type::*unmentionable_type)()const;
  595. BOOST_MP_FORCEINLINE operator unmentionable_type()const
  596. {
  597. return is_zero() ? 0 : &self_type::is_zero;
  598. }
  599. #endif
  600. //
  601. // Default precision:
  602. //
  603. static unsigned default_precision() BOOST_NOEXCEPT
  604. {
  605. return Backend::default_precision();
  606. }
  607. static void default_precision(unsigned digits10)
  608. {
  609. Backend::default_precision(digits10);
  610. }
  611. unsigned precision()const BOOST_NOEXCEPT
  612. {
  613. return m_backend.precision();
  614. }
  615. void precision(unsigned digits10)
  616. {
  617. m_backend.precision(digits10);
  618. }
  619. //
  620. // Comparison:
  621. //
  622. BOOST_MP_FORCEINLINE int compare(const number<Backend, ExpressionTemplates>& o)const
  623. BOOST_MP_NOEXCEPT_IF(noexcept(std::declval<Backend>().compare(std::declval<Backend>())))
  624. {
  625. return m_backend.compare(o.m_backend);
  626. }
  627. template <class V>
  628. BOOST_MP_FORCEINLINE typename boost::enable_if<is_arithmetic<V>, int>::type compare(const V& o)const
  629. {
  630. using default_ops::eval_get_sign;
  631. if(o == 0)
  632. return eval_get_sign(m_backend);
  633. return m_backend.compare(canonical_value(o));
  634. }
  635. BOOST_MP_FORCEINLINE Backend& backend() BOOST_NOEXCEPT
  636. {
  637. return m_backend;
  638. }
  639. BOOST_MP_FORCEINLINE BOOST_CONSTEXPR const Backend& backend()const BOOST_NOEXCEPT
  640. {
  641. return m_backend;
  642. }
  643. private:
  644. template <class tag, class Arg1, class Arg2, class Arg3, class Arg4>
  645. void do_assign(const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& e, const mpl::true_&)
  646. {
  647. do_assign(e, tag());
  648. }
  649. template <class tag, class Arg1, class Arg2, class Arg3, class Arg4>
  650. void do_assign(const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& e, const mpl::false_&)
  651. {
  652. // The result of the expression isn't the same type as this -
  653. // create a temporary result and assign it to *this:
  654. typedef typename detail::expression<tag, Arg1, Arg2, Arg3, Arg4>::result_type temp_type;
  655. temp_type t(e);
  656. this->assign(t);
  657. }
  658. template <class Exp>
  659. void do_assign(const Exp& e, const detail::add_immediates&)
  660. {
  661. using default_ops::eval_add;
  662. eval_add(m_backend, canonical_value(e.left().value()), canonical_value(e.right().value()));
  663. }
  664. template <class Exp>
  665. void do_assign(const Exp& e, const detail::subtract_immediates&)
  666. {
  667. using default_ops::eval_subtract;
  668. eval_subtract(m_backend, canonical_value(e.left().value()), canonical_value(e.right().value()));
  669. }
  670. template <class Exp>
  671. void do_assign(const Exp& e, const detail::multiply_immediates&)
  672. {
  673. using default_ops::eval_multiply;
  674. eval_multiply(m_backend, canonical_value(e.left().value()), canonical_value(e.right().value()));
  675. }
  676. template <class Exp>
  677. void do_assign(const Exp& e, const detail::multiply_add&)
  678. {
  679. using default_ops::eval_multiply_add;
  680. eval_multiply_add(m_backend, canonical_value(e.left().value()), canonical_value(e.middle().value()), canonical_value(e.right().value()));
  681. }
  682. template <class Exp>
  683. void do_assign(const Exp& e, const detail::multiply_subtract&)
  684. {
  685. using default_ops::eval_multiply_subtract;
  686. eval_multiply_subtract(m_backend, canonical_value(e.left().value()), canonical_value(e.middle().value()), canonical_value(e.right().value()));
  687. }
  688. template <class Exp>
  689. void do_assign(const Exp& e, const detail::divide_immediates&)
  690. {
  691. using default_ops::eval_divide;
  692. eval_divide(m_backend, canonical_value(e.left().value()), canonical_value(e.right().value()));
  693. }
  694. template <class Exp>
  695. void do_assign(const Exp& e, const detail::negate&)
  696. {
  697. typedef typename Exp::left_type left_type;
  698. do_assign(e.left(), typename left_type::tag_type());
  699. m_backend.negate();
  700. }
  701. template <class Exp>
  702. void do_assign(const Exp& e, const detail::plus&)
  703. {
  704. typedef typename Exp::left_type left_type;
  705. typedef typename Exp::right_type right_type;
  706. static int const left_depth = left_type::depth;
  707. static int const right_depth = right_type::depth;
  708. bool bl = contains_self(e.left());
  709. bool br = contains_self(e.right());
  710. if(bl && is_self(e.left()))
  711. {
  712. // Ignore the left node, it's *this, just add the right:
  713. do_add(e.right(), typename right_type::tag_type());
  714. }
  715. else if(br && is_self(e.right()))
  716. {
  717. // Ignore the right node, it's *this, just add the left:
  718. do_add(e.left(), typename left_type::tag_type());
  719. }
  720. else if(bl && br)
  721. {
  722. self_type temp(e);
  723. temp.m_backend.swap(this->m_backend);
  724. }
  725. else if(!br && (bl || (left_depth >= right_depth)))
  726. { // br is always false, but if bl is true we must take the this branch:
  727. do_assign(e.left(), typename left_type::tag_type());
  728. do_add(e.right(), typename right_type::tag_type());
  729. }
  730. else
  731. {
  732. do_assign(e.right(), typename right_type::tag_type());
  733. do_add(e.left(), typename left_type::tag_type());
  734. }
  735. }
  736. template <class Exp>
  737. void do_assign(const Exp& e, const detail::minus&)
  738. {
  739. typedef typename Exp::left_type left_type;
  740. typedef typename Exp::right_type right_type;
  741. static int const left_depth = left_type::depth;
  742. static int const right_depth = right_type::depth;
  743. bool bl = contains_self(e.left());
  744. bool br = contains_self(e.right());
  745. if(bl && is_self(e.left()))
  746. {
  747. // Ignore the left node, it's *this, just subtract the right:
  748. do_subtract(e.right(), typename right_type::tag_type());
  749. }
  750. else if(br && is_self(e.right()))
  751. {
  752. // Ignore the right node, it's *this, just subtract the left and negate the result:
  753. do_subtract(e.left(), typename left_type::tag_type());
  754. m_backend.negate();
  755. }
  756. else if(bl && br)
  757. {
  758. self_type temp(e);
  759. temp.m_backend.swap(this->m_backend);
  760. }
  761. else if(!br && (bl || (left_depth >= right_depth)))
  762. { // br is always false, but if bl is true we must take the this branch:
  763. do_assign(e.left(), typename left_type::tag_type());
  764. do_subtract(e.right(), typename right_type::tag_type());
  765. }
  766. else
  767. {
  768. do_assign(e.right(), typename right_type::tag_type());
  769. do_subtract(e.left(), typename left_type::tag_type());
  770. m_backend.negate();
  771. }
  772. }
  773. template <class Exp>
  774. void do_assign(const Exp& e, const detail::multiplies&)
  775. {
  776. typedef typename Exp::left_type left_type;
  777. typedef typename Exp::right_type right_type;
  778. static int const left_depth = left_type::depth;
  779. static int const right_depth = right_type::depth;
  780. bool bl = contains_self(e.left());
  781. bool br = contains_self(e.right());
  782. if(bl && is_self(e.left()))
  783. {
  784. // Ignore the left node, it's *this, just add the right:
  785. do_multiplies(e.right(), typename right_type::tag_type());
  786. }
  787. else if(br && is_self(e.right()))
  788. {
  789. // Ignore the right node, it's *this, just add the left:
  790. do_multiplies(e.left(), typename left_type::tag_type());
  791. }
  792. else if(bl && br)
  793. {
  794. self_type temp(e);
  795. temp.m_backend.swap(this->m_backend);
  796. }
  797. else if(!br && (bl || (left_depth >= right_depth)))
  798. { // br is always false, but if bl is true we must take the this branch:
  799. do_assign(e.left(), typename left_type::tag_type());
  800. do_multiplies(e.right(), typename right_type::tag_type());
  801. }
  802. else
  803. {
  804. do_assign(e.right(), typename right_type::tag_type());
  805. do_multiplies(e.left(), typename left_type::tag_type());
  806. }
  807. }
  808. template <class Exp>
  809. void do_assign(const Exp& e, const detail::divides&)
  810. {
  811. typedef typename Exp::left_type left_type;
  812. typedef typename Exp::right_type right_type;
  813. bool bl = contains_self(e.left());
  814. bool br = contains_self(e.right());
  815. if(bl && is_self(e.left()))
  816. {
  817. // Ignore the left node, it's *this, just add the right:
  818. do_divide(e.right(), typename right_type::tag_type());
  819. }
  820. else if(br)
  821. {
  822. self_type temp(e);
  823. temp.m_backend.swap(this->m_backend);
  824. }
  825. else
  826. {
  827. do_assign(e.left(), typename left_type::tag_type());
  828. do_divide(e.right(), typename right_type::tag_type());
  829. }
  830. }
  831. template <class Exp>
  832. void do_assign(const Exp& e, const detail::modulus&)
  833. {
  834. //
  835. // This operation is only valid for integer backends:
  836. //
  837. BOOST_STATIC_ASSERT_MSG(number_category<Backend>::value == number_kind_integer, "The modulus operation is only valid for integer types");
  838. typedef typename Exp::left_type left_type;
  839. typedef typename Exp::right_type right_type;
  840. bool bl = contains_self(e.left());
  841. bool br = contains_self(e.right());
  842. if(bl && is_self(e.left()))
  843. {
  844. // Ignore the left node, it's *this, just add the right:
  845. do_modulus(e.right(), typename right_type::tag_type());
  846. }
  847. else if(br)
  848. {
  849. self_type temp(e);
  850. temp.m_backend.swap(this->m_backend);
  851. }
  852. else
  853. {
  854. do_assign(e.left(), typename left_type::tag_type());
  855. do_modulus(e.right(), typename right_type::tag_type());
  856. }
  857. }
  858. template <class Exp>
  859. void do_assign(const Exp& e, const detail::modulus_immediates&)
  860. {
  861. BOOST_STATIC_ASSERT_MSG(number_category<Backend>::value == number_kind_integer, "The modulus operation is only valid for integer types");
  862. using default_ops::eval_modulus;
  863. eval_modulus(m_backend, canonical_value(e.left().value()), canonical_value(e.right().value()));
  864. }
  865. template <class Exp>
  866. void do_assign(const Exp& e, const detail::bitwise_and&)
  867. {
  868. //
  869. // This operation is only valid for integer backends:
  870. //
  871. BOOST_STATIC_ASSERT_MSG(number_category<Backend>::value == number_kind_integer, "Bitwise operations are only valid for integer types");
  872. typedef typename Exp::left_type left_type;
  873. typedef typename Exp::right_type right_type;
  874. static int const left_depth = left_type::depth;
  875. static int const right_depth = right_type::depth;
  876. bool bl = contains_self(e.left());
  877. bool br = contains_self(e.right());
  878. if(bl && is_self(e.left()))
  879. {
  880. // Ignore the left node, it's *this, just add the right:
  881. do_bitwise_and(e.right(), typename right_type::tag_type());
  882. }
  883. else if(br && is_self(e.right()))
  884. {
  885. do_bitwise_and(e.left(), typename left_type::tag_type());
  886. }
  887. else if(!br && (bl || (left_depth >= right_depth)))
  888. {
  889. do_assign(e.left(), typename left_type::tag_type());
  890. do_bitwise_and(e.right(), typename right_type::tag_type());
  891. }
  892. else
  893. {
  894. do_assign(e.right(), typename right_type::tag_type());
  895. do_bitwise_and(e.left(), typename left_type::tag_type());
  896. }
  897. }
  898. template <class Exp>
  899. void do_assign(const Exp& e, const detail::bitwise_and_immediates&)
  900. {
  901. BOOST_STATIC_ASSERT_MSG(number_category<Backend>::value == number_kind_integer, "Bitwise operations are only valid for integer types");
  902. using default_ops::eval_bitwise_and;
  903. eval_bitwise_and(m_backend, canonical_value(e.left().value()), canonical_value(e.right().value()));
  904. }
  905. template <class Exp>
  906. void do_assign(const Exp& e, const detail::bitwise_or&)
  907. {
  908. //
  909. // This operation is only valid for integer backends:
  910. //
  911. BOOST_STATIC_ASSERT_MSG(number_category<Backend>::value == number_kind_integer, "Bitwise operations are only valid for integer types");
  912. typedef typename Exp::left_type left_type;
  913. typedef typename Exp::right_type right_type;
  914. static int const left_depth = left_type::depth;
  915. static int const right_depth = right_type::depth;
  916. bool bl = contains_self(e.left());
  917. bool br = contains_self(e.right());
  918. if(bl && is_self(e.left()))
  919. {
  920. // Ignore the left node, it's *this, just add the right:
  921. do_bitwise_or(e.right(), typename right_type::tag_type());
  922. }
  923. else if(br && is_self(e.right()))
  924. {
  925. do_bitwise_or(e.left(), typename left_type::tag_type());
  926. }
  927. else if(!br && (bl || (left_depth >= right_depth)))
  928. {
  929. do_assign(e.left(), typename left_type::tag_type());
  930. do_bitwise_or(e.right(), typename right_type::tag_type());
  931. }
  932. else
  933. {
  934. do_assign(e.right(), typename right_type::tag_type());
  935. do_bitwise_or(e.left(), typename left_type::tag_type());
  936. }
  937. }
  938. template <class Exp>
  939. void do_assign(const Exp& e, const detail::bitwise_or_immediates&)
  940. {
  941. BOOST_STATIC_ASSERT_MSG(number_category<Backend>::value == number_kind_integer, "Bitwise operations are only valid for integer types");
  942. using default_ops::eval_bitwise_or;
  943. eval_bitwise_or(m_backend, canonical_value(e.left().value()), canonical_value(e.right().value()));
  944. }
  945. template <class Exp>
  946. void do_assign(const Exp& e, const detail::bitwise_xor&)
  947. {
  948. //
  949. // This operation is only valid for integer backends:
  950. //
  951. BOOST_STATIC_ASSERT_MSG(number_category<Backend>::value == number_kind_integer, "Bitwise operations are only valid for integer types");
  952. typedef typename Exp::left_type left_type;
  953. typedef typename Exp::right_type right_type;
  954. static int const left_depth = left_type::depth;
  955. static int const right_depth = right_type::depth;
  956. bool bl = contains_self(e.left());
  957. bool br = contains_self(e.right());
  958. if(bl && is_self(e.left()))
  959. {
  960. // Ignore the left node, it's *this, just add the right:
  961. do_bitwise_xor(e.right(), typename right_type::tag_type());
  962. }
  963. else if(br && is_self(e.right()))
  964. {
  965. do_bitwise_xor(e.left(), typename left_type::tag_type());
  966. }
  967. else if(!br && (bl || (left_depth >= right_depth)))
  968. {
  969. do_assign(e.left(), typename left_type::tag_type());
  970. do_bitwise_xor(e.right(), typename right_type::tag_type());
  971. }
  972. else
  973. {
  974. do_assign(e.right(), typename right_type::tag_type());
  975. do_bitwise_xor(e.left(), typename left_type::tag_type());
  976. }
  977. }
  978. template <class Exp>
  979. void do_assign(const Exp& e, const detail::bitwise_xor_immediates&)
  980. {
  981. BOOST_STATIC_ASSERT_MSG(number_category<Backend>::value == number_kind_integer, "Bitwise operations are only valid for integer types");
  982. using default_ops::eval_bitwise_xor;
  983. eval_bitwise_xor(m_backend, canonical_value(e.left().value()), canonical_value(e.right().value()));
  984. }
  985. template <class Exp>
  986. void do_assign(const Exp& e, const detail::terminal&)
  987. {
  988. if(!is_self(e))
  989. {
  990. m_backend = canonical_value(e.value());
  991. }
  992. }
  993. template <class Exp>
  994. void do_assign(const Exp& e, const detail::function&)
  995. {
  996. typedef typename Exp::arity tag_type;
  997. do_assign_function(e, tag_type());
  998. }
  999. template <class Exp>
  1000. void do_assign(const Exp& e, const detail::shift_left&)
  1001. {
  1002. // We can only shift by an integer value, not an arbitrary expression:
  1003. typedef typename Exp::left_type left_type;
  1004. typedef typename Exp::right_type right_type;
  1005. typedef typename right_type::arity right_arity;
  1006. BOOST_STATIC_ASSERT_MSG(right_arity::value == 0, "The left shift operator requires an integer value for the shift operand.");
  1007. typedef typename right_type::result_type right_value_type;
  1008. BOOST_STATIC_ASSERT_MSG(is_integral<right_value_type>::value, "The left shift operator requires an integer value for the shift operand.");
  1009. typedef typename left_type::tag_type tag_type;
  1010. do_assign_left_shift(e.left(), canonical_value(e.right().value()), tag_type());
  1011. }
  1012. template <class Exp>
  1013. void do_assign(const Exp& e, const detail::shift_right&)
  1014. {
  1015. // We can only shift by an integer value, not an arbitrary expression:
  1016. typedef typename Exp::left_type left_type;
  1017. typedef typename Exp::right_type right_type;
  1018. typedef typename right_type::arity right_arity;
  1019. BOOST_STATIC_ASSERT_MSG(right_arity::value == 0, "The left shift operator requires an integer value for the shift operand.");
  1020. typedef typename right_type::result_type right_value_type;
  1021. BOOST_STATIC_ASSERT_MSG(is_integral<right_value_type>::value, "The left shift operator requires an integer value for the shift operand.");
  1022. typedef typename left_type::tag_type tag_type;
  1023. do_assign_right_shift(e.left(), canonical_value(e.right().value()), tag_type());
  1024. }
  1025. template <class Exp>
  1026. void do_assign(const Exp& e, const detail::bitwise_complement&)
  1027. {
  1028. BOOST_STATIC_ASSERT_MSG(number_category<Backend>::value == number_kind_integer, "The bitwise ~ operation is only valid for integer types");
  1029. using default_ops::eval_complement;
  1030. self_type temp(e.left());
  1031. eval_complement(m_backend, temp.backend());
  1032. }
  1033. template <class Exp>
  1034. void do_assign(const Exp& e, const detail::complement_immediates&)
  1035. {
  1036. BOOST_STATIC_ASSERT_MSG(number_category<Backend>::value == number_kind_integer, "The bitwise ~ operation is only valid for integer types");
  1037. using default_ops::eval_complement;
  1038. eval_complement(m_backend, canonical_value(e.left().value()));
  1039. }
  1040. template <class Exp, class Val>
  1041. void do_assign_right_shift(const Exp& e, const Val& val, const detail::terminal&)
  1042. {
  1043. BOOST_STATIC_ASSERT_MSG(number_category<Backend>::value == number_kind_integer, "The right shift operation is only valid for integer types");
  1044. using default_ops::eval_right_shift;
  1045. detail::check_shift_range(val, mpl::bool_<(sizeof(Val) > sizeof(std::size_t))>(), is_signed<Val>());
  1046. eval_right_shift(m_backend, canonical_value(e.value()), static_cast<std::size_t>(val));
  1047. }
  1048. template <class Exp, class Val>
  1049. void do_assign_left_shift(const Exp& e, const Val& val, const detail::terminal&)
  1050. {
  1051. BOOST_STATIC_ASSERT_MSG(number_category<Backend>::value == number_kind_integer, "The left shift operation is only valid for integer types");
  1052. using default_ops::eval_left_shift;
  1053. detail::check_shift_range(val, mpl::bool_<(sizeof(Val) > sizeof(std::size_t))>(), is_signed<Val>());
  1054. eval_left_shift(m_backend, canonical_value(e.value()), static_cast<std::size_t>(val));
  1055. }
  1056. template <class Exp, class Val, class Tag>
  1057. void do_assign_right_shift(const Exp& e, const Val& val, const Tag&)
  1058. {
  1059. BOOST_STATIC_ASSERT_MSG(number_category<Backend>::value == number_kind_integer, "The right shift operation is only valid for integer types");
  1060. using default_ops::eval_right_shift;
  1061. self_type temp(e);
  1062. detail::check_shift_range(val, mpl::bool_<(sizeof(Val) > sizeof(std::size_t))>(), is_signed<Val>());
  1063. eval_right_shift(m_backend, temp.backend(), static_cast<std::size_t>(val));
  1064. }
  1065. template <class Exp, class Val, class Tag>
  1066. void do_assign_left_shift(const Exp& e, const Val& val, const Tag&)
  1067. {
  1068. BOOST_STATIC_ASSERT_MSG(number_category<Backend>::value == number_kind_integer, "The left shift operation is only valid for integer types");
  1069. using default_ops::eval_left_shift;
  1070. self_type temp(e);
  1071. detail::check_shift_range(val, mpl::bool_<(sizeof(Val) > sizeof(std::size_t))>(), is_signed<Val>());
  1072. eval_left_shift(m_backend, temp.backend(), static_cast<std::size_t>(val));
  1073. }
  1074. template <class Exp>
  1075. void do_assign_function(const Exp& e, const mpl::int_<1>&)
  1076. {
  1077. e.left().value()(&m_backend);
  1078. }
  1079. template <class Exp>
  1080. void do_assign_function(const Exp& e, const mpl::int_<2>&)
  1081. {
  1082. typedef typename Exp::right_type right_type;
  1083. typedef typename right_type::tag_type tag_type;
  1084. do_assign_function_1(e.left().value(), e.right_ref(), tag_type());
  1085. }
  1086. template <class F, class Exp>
  1087. void do_assign_function_1(const F& f, const Exp& val, const detail::terminal&)
  1088. {
  1089. f(m_backend, function_arg_value(val));
  1090. }
  1091. template <class F, class Exp, class Tag>
  1092. void do_assign_function_1(const F& f, const Exp& val, const Tag&)
  1093. {
  1094. number t(val);
  1095. f(m_backend, t.backend());
  1096. }
  1097. template <class Exp>
  1098. void do_assign_function(const Exp& e, const mpl::int_<3>&)
  1099. {
  1100. typedef typename Exp::middle_type middle_type;
  1101. typedef typename middle_type::tag_type tag_type;
  1102. typedef typename Exp::right_type end_type;
  1103. typedef typename end_type::tag_type end_tag;
  1104. do_assign_function_2(e.left().value(), e.middle_ref(), e.right_ref(), tag_type(), end_tag());
  1105. }
  1106. template <class F, class Exp1, class Exp2>
  1107. void do_assign_function_2(const F& f, const Exp1& val1, const Exp2& val2, const detail::terminal&, const detail::terminal&)
  1108. {
  1109. f(m_backend, function_arg_value(val1), function_arg_value(val2));
  1110. }
  1111. template <class F, class Exp1, class Exp2, class Tag1>
  1112. void do_assign_function_2(const F& f, const Exp1& val1, const Exp2& val2, const Tag1&, const detail::terminal&)
  1113. {
  1114. self_type temp1(val1);
  1115. f(m_backend, BOOST_MP_MOVE(temp1.backend()), function_arg_value(val2));
  1116. }
  1117. template <class F, class Exp1, class Exp2, class Tag2>
  1118. void do_assign_function_2(const F& f, const Exp1& val1, const Exp2& val2, const detail::terminal&, const Tag2&)
  1119. {
  1120. self_type temp2(val2);
  1121. f(m_backend, function_arg_value(val1), BOOST_MP_MOVE(temp2.backend()));
  1122. }
  1123. template <class F, class Exp1, class Exp2, class Tag1, class Tag2>
  1124. void do_assign_function_2(const F& f, const Exp1& val1, const Exp2& val2, const Tag1&, const Tag2&)
  1125. {
  1126. self_type temp1(val1);
  1127. self_type temp2(val2);
  1128. f(m_backend, BOOST_MP_MOVE(temp1.backend()), BOOST_MP_MOVE(temp2.backend()));
  1129. }
  1130. template <class Exp>
  1131. void do_assign_function(const Exp& e, const mpl::int_<4>&)
  1132. {
  1133. typedef typename Exp::left_middle_type left_type;
  1134. typedef typename left_type::tag_type left_tag_type;
  1135. typedef typename Exp::right_middle_type middle_type;
  1136. typedef typename middle_type::tag_type middle_tag_type;
  1137. typedef typename Exp::right_type right_type;
  1138. typedef typename right_type::tag_type right_tag_type;
  1139. do_assign_function_3a(e.left().value(), e.left_middle_ref(), e.right_middle_ref(), e.right_ref(), left_tag_type(), middle_tag_type(), right_tag_type());
  1140. }
  1141. template <class F, class Exp1, class Exp2, class Exp3, class Tag2, class Tag3>
  1142. void do_assign_function_3a(const F& f, const Exp1& val1, const Exp2& val2, const Exp3& val3, const detail::terminal&, const Tag2& t2, const Tag3& t3)
  1143. {
  1144. do_assign_function_3b(f, val1, val2, val3, t2, t3);
  1145. }
  1146. template <class F, class Exp1, class Exp2, class Exp3, class Tag1, class Tag2, class Tag3>
  1147. void do_assign_function_3a(const F& f, const Exp1& val1, const Exp2& val2, const Exp3& val3, const Tag1&, const Tag2& t2, const Tag3& t3)
  1148. {
  1149. number t(val1);
  1150. do_assign_function_3b(f, BOOST_MP_MOVE(t), val2, val3, t2, t3);
  1151. }
  1152. template <class F, class Exp1, class Exp2, class Exp3, class Tag3>
  1153. void do_assign_function_3b(const F& f, const Exp1& val1, const Exp2& val2, const Exp3& val3, const detail::terminal&, const Tag3& t3)
  1154. {
  1155. do_assign_function_3c(f, val1, val2, val3, t3);
  1156. }
  1157. template <class F, class Exp1, class Exp2, class Exp3, class Tag2, class Tag3>
  1158. void do_assign_function_3b(const F& f, const Exp1& val1, const Exp2& val2, const Exp3& val3, const Tag2& /*t2*/, const Tag3& t3)
  1159. {
  1160. number t(val2);
  1161. do_assign_function_3c(f, val1, BOOST_MP_MOVE(t), val3, t3);
  1162. }
  1163. template <class F, class Exp1, class Exp2, class Exp3>
  1164. void do_assign_function_3c(const F& f, const Exp1& val1, const Exp2& val2, const Exp3& val3, const detail::terminal&)
  1165. {
  1166. f(m_backend, function_arg_value(val1), function_arg_value(val2), function_arg_value(val3));
  1167. }
  1168. template <class F, class Exp1, class Exp2, class Exp3, class Tag3>
  1169. void do_assign_function_3c(const F& f, const Exp1& val1, const Exp2& val2, const Exp3& val3, const Tag3& /*t3*/)
  1170. {
  1171. number t(val3);
  1172. do_assign_function_3c(f, val1, val2, BOOST_MP_MOVE(t), detail::terminal());
  1173. }
  1174. template <class Exp>
  1175. void do_add(const Exp& e, const detail::terminal&)
  1176. {
  1177. using default_ops::eval_add;
  1178. eval_add(m_backend, canonical_value(e.value()));
  1179. }
  1180. template <class Exp>
  1181. void do_add(const Exp& e, const detail::negate&)
  1182. {
  1183. typedef typename Exp::left_type left_type;
  1184. do_subtract(e.left(), typename left_type::tag_type());
  1185. }
  1186. template <class Exp>
  1187. void do_add(const Exp& e, const detail::plus&)
  1188. {
  1189. typedef typename Exp::left_type left_type;
  1190. typedef typename Exp::right_type right_type;
  1191. do_add(e.left(), typename left_type::tag_type());
  1192. do_add(e.right(), typename right_type::tag_type());
  1193. }
  1194. template <class Exp>
  1195. void do_add(const Exp& e, const detail::minus&)
  1196. {
  1197. typedef typename Exp::left_type left_type;
  1198. typedef typename Exp::right_type right_type;
  1199. do_add(e.left(), typename left_type::tag_type());
  1200. do_subtract(e.right(), typename right_type::tag_type());
  1201. }
  1202. template <class Exp, class unknown>
  1203. void do_add(const Exp& e, const unknown&)
  1204. {
  1205. self_type temp(e);
  1206. do_add(detail::expression<detail::terminal, self_type>(temp), detail::terminal());
  1207. }
  1208. template <class Exp>
  1209. void do_add(const Exp& e, const detail::add_immediates&)
  1210. {
  1211. using default_ops::eval_add;
  1212. eval_add(m_backend, canonical_value(e.left().value()));
  1213. eval_add(m_backend, canonical_value(e.right().value()));
  1214. }
  1215. template <class Exp>
  1216. void do_add(const Exp& e, const detail::subtract_immediates&)
  1217. {
  1218. using default_ops::eval_add;
  1219. using default_ops::eval_subtract;
  1220. eval_add(m_backend, canonical_value(e.left().value()));
  1221. eval_subtract(m_backend, canonical_value(e.right().value()));
  1222. }
  1223. template <class Exp>
  1224. void do_subtract(const Exp& e, const detail::terminal&)
  1225. {
  1226. using default_ops::eval_subtract;
  1227. eval_subtract(m_backend, canonical_value(e.value()));
  1228. }
  1229. template <class Exp>
  1230. void do_subtract(const Exp& e, const detail::negate&)
  1231. {
  1232. typedef typename Exp::left_type left_type;
  1233. do_add(e.left(), typename left_type::tag_type());
  1234. }
  1235. template <class Exp>
  1236. void do_subtract(const Exp& e, const detail::plus&)
  1237. {
  1238. typedef typename Exp::left_type left_type;
  1239. typedef typename Exp::right_type right_type;
  1240. do_subtract(e.left(), typename left_type::tag_type());
  1241. do_subtract(e.right(), typename right_type::tag_type());
  1242. }
  1243. template <class Exp>
  1244. void do_subtract(const Exp& e, const detail::minus&)
  1245. {
  1246. typedef typename Exp::left_type left_type;
  1247. typedef typename Exp::right_type right_type;
  1248. do_subtract(e.left(), typename left_type::tag_type());
  1249. do_add(e.right(), typename right_type::tag_type());
  1250. }
  1251. template <class Exp>
  1252. void do_subtract(const Exp& e, const detail::add_immediates&)
  1253. {
  1254. using default_ops::eval_subtract;
  1255. eval_subtract(m_backend, canonical_value(e.left().value()));
  1256. eval_subtract(m_backend, canonical_value(e.right().value()));
  1257. }
  1258. template <class Exp>
  1259. void do_subtract(const Exp& e, const detail::subtract_immediates&)
  1260. {
  1261. using default_ops::eval_add;
  1262. using default_ops::eval_subtract;
  1263. eval_subtract(m_backend, canonical_value(e.left().value()));
  1264. eval_add(m_backend, canonical_value(e.right().value()));
  1265. }
  1266. template <class Exp, class unknown>
  1267. void do_subtract(const Exp& e, const unknown&)
  1268. {
  1269. self_type temp(e);
  1270. do_subtract(detail::expression<detail::terminal, self_type>(temp), detail::terminal());
  1271. }
  1272. template <class Exp>
  1273. void do_multiplies(const Exp& e, const detail::terminal&)
  1274. {
  1275. using default_ops::eval_multiply;
  1276. eval_multiply(m_backend, canonical_value(e.value()));
  1277. }
  1278. template <class Exp>
  1279. void do_multiplies(const Exp& e, const detail::negate&)
  1280. {
  1281. typedef typename Exp::left_type left_type;
  1282. do_multiplies(e.left(), typename left_type::tag_type());
  1283. m_backend.negate();
  1284. }
  1285. template <class Exp>
  1286. void do_multiplies(const Exp& e, const detail::multiplies&)
  1287. {
  1288. typedef typename Exp::left_type left_type;
  1289. typedef typename Exp::right_type right_type;
  1290. do_multiplies(e.left(), typename left_type::tag_type());
  1291. do_multiplies(e.right(), typename right_type::tag_type());
  1292. }
  1293. //
  1294. // This rearrangement is disabled for integer types, the test on sizeof(Exp) is simply to make
  1295. // the disable_if dependent on the template argument (the size of 1 can never occur in practice).
  1296. //
  1297. template <class Exp>
  1298. typename boost::disable_if_c<boost::multiprecision::number_category<self_type>::value == boost::multiprecision::number_kind_integer || sizeof(Exp) == 1>::type
  1299. do_multiplies(const Exp& e, const detail::divides&)
  1300. {
  1301. typedef typename Exp::left_type left_type;
  1302. typedef typename Exp::right_type right_type;
  1303. do_multiplies(e.left(), typename left_type::tag_type());
  1304. do_divide(e.right(), typename right_type::tag_type());
  1305. }
  1306. template <class Exp>
  1307. void do_multiplies(const Exp& e, const detail::multiply_immediates&)
  1308. {
  1309. using default_ops::eval_multiply;
  1310. eval_multiply(m_backend, canonical_value(e.left().value()));
  1311. eval_multiply(m_backend, canonical_value(e.right().value()));
  1312. }
  1313. //
  1314. // This rearrangement is disabled for integer types, the test on sizeof(Exp) is simply to make
  1315. // the disable_if dependent on the template argument (the size of 1 can never occur in practice).
  1316. //
  1317. template <class Exp>
  1318. typename boost::disable_if_c<boost::multiprecision::number_category<self_type>::value == boost::multiprecision::number_kind_integer || sizeof(Exp) == 1>::type
  1319. do_multiplies(const Exp& e, const detail::divide_immediates&)
  1320. {
  1321. using default_ops::eval_multiply;
  1322. using default_ops::eval_divide;
  1323. eval_multiply(m_backend, canonical_value(e.left().value()));
  1324. eval_divide(m_backend, canonical_value(e.right().value()));
  1325. }
  1326. template <class Exp, class unknown>
  1327. void do_multiplies(const Exp& e, const unknown&)
  1328. {
  1329. using default_ops::eval_multiply;
  1330. self_type temp(e);
  1331. eval_multiply(m_backend, temp.m_backend);
  1332. }
  1333. template <class Exp>
  1334. void do_divide(const Exp& e, const detail::terminal&)
  1335. {
  1336. using default_ops::eval_divide;
  1337. eval_divide(m_backend, canonical_value(e.value()));
  1338. }
  1339. template <class Exp>
  1340. void do_divide(const Exp& e, const detail::negate&)
  1341. {
  1342. typedef typename Exp::left_type left_type;
  1343. do_divide(e.left(), typename left_type::tag_type());
  1344. m_backend.negate();
  1345. }
  1346. //
  1347. // This rearrangement is disabled for integer types, the test on sizeof(Exp) is simply to make
  1348. // the disable_if dependent on the template argument (the size of 1 can never occur in practice).
  1349. //
  1350. template <class Exp>
  1351. typename boost::disable_if_c<boost::multiprecision::number_category<self_type>::value == boost::multiprecision::number_kind_integer || sizeof(Exp) == 1>::type
  1352. do_divide(const Exp& e, const detail::multiplies&)
  1353. {
  1354. typedef typename Exp::left_type left_type;
  1355. typedef typename Exp::right_type right_type;
  1356. do_divide(e.left(), typename left_type::tag_type());
  1357. do_divide(e.right(), typename right_type::tag_type());
  1358. }
  1359. //
  1360. // This rearrangement is disabled for integer types, the test on sizeof(Exp) is simply to make
  1361. // the disable_if dependent on the template argument (the size of 1 can never occur in practice).
  1362. //
  1363. template <class Exp>
  1364. typename boost::disable_if_c<boost::multiprecision::number_category<self_type>::value == boost::multiprecision::number_kind_integer || sizeof(Exp) == 1>::type
  1365. do_divide(const Exp& e, const detail::divides&)
  1366. {
  1367. typedef typename Exp::left_type left_type;
  1368. typedef typename Exp::right_type right_type;
  1369. do_divide(e.left(), typename left_type::tag_type());
  1370. do_multiplies(e.right(), typename right_type::tag_type());
  1371. }
  1372. //
  1373. // This rearrangement is disabled for integer types, the test on sizeof(Exp) is simply to make
  1374. // the disable_if dependent on the template argument (the size of 1 can never occur in practice).
  1375. //
  1376. template <class Exp>
  1377. typename boost::disable_if_c<boost::multiprecision::number_category<self_type>::value == boost::multiprecision::number_kind_integer || sizeof(Exp) == 1>::type
  1378. do_divides(const Exp& e, const detail::multiply_immediates&)
  1379. {
  1380. using default_ops::eval_divide;
  1381. eval_divide(m_backend, canonical_value(e.left().value()));
  1382. eval_divide(m_backend, canonical_value(e.right().value()));
  1383. }
  1384. //
  1385. // This rearrangement is disabled for integer types, the test on sizeof(Exp) is simply to make
  1386. // the disable_if dependent on the template argument (the size of 1 can never occur in practice).
  1387. //
  1388. template <class Exp>
  1389. typename boost::disable_if_c<boost::multiprecision::number_category<self_type>::value == boost::multiprecision::number_kind_integer || sizeof(Exp) == 1>::type
  1390. do_divides(const Exp& e, const detail::divide_immediates&)
  1391. {
  1392. using default_ops::eval_multiply;
  1393. using default_ops::eval_divide;
  1394. eval_divide(m_backend, canonical_value(e.left().value()));
  1395. mutiply(m_backend, canonical_value(e.right().value()));
  1396. }
  1397. template <class Exp, class unknown>
  1398. void do_divide(const Exp& e, const unknown&)
  1399. {
  1400. using default_ops::eval_multiply;
  1401. self_type temp(e);
  1402. eval_divide(m_backend, temp.m_backend);
  1403. }
  1404. template <class Exp>
  1405. void do_modulus(const Exp& e, const detail::terminal&)
  1406. {
  1407. BOOST_STATIC_ASSERT_MSG(number_category<Backend>::value == number_kind_integer, "The modulus operation is only valid for integer types");
  1408. using default_ops::eval_modulus;
  1409. eval_modulus(m_backend, canonical_value(e.value()));
  1410. }
  1411. template <class Exp, class Unknown>
  1412. void do_modulus(const Exp& e, const Unknown&)
  1413. {
  1414. BOOST_STATIC_ASSERT_MSG(number_category<Backend>::value == number_kind_integer, "The modulus operation is only valid for integer types");
  1415. using default_ops::eval_modulus;
  1416. self_type temp(e);
  1417. eval_modulus(m_backend, canonical_value(temp));
  1418. }
  1419. template <class Exp>
  1420. void do_bitwise_and(const Exp& e, const detail::terminal&)
  1421. {
  1422. BOOST_STATIC_ASSERT_MSG(number_category<Backend>::value == number_kind_integer, "The bitwise & operation is only valid for integer types");
  1423. using default_ops::eval_bitwise_and;
  1424. eval_bitwise_and(m_backend, canonical_value(e.value()));
  1425. }
  1426. template <class Exp>
  1427. void do_bitwise_and(const Exp& e, const detail::bitwise_and&)
  1428. {
  1429. BOOST_STATIC_ASSERT_MSG(number_category<Backend>::value == number_kind_integer, "The bitwise & operation is only valid for integer types");
  1430. typedef typename Exp::left_type left_type;
  1431. typedef typename Exp::right_type right_type;
  1432. do_bitwise_and(e.left(), typename left_type::tag_type());
  1433. do_bitwise_and(e.right(), typename right_type::tag_type());
  1434. }
  1435. template <class Exp, class unknown>
  1436. void do_bitwise_and(const Exp& e, const unknown&)
  1437. {
  1438. BOOST_STATIC_ASSERT_MSG(number_category<Backend>::value == number_kind_integer, "The bitwise & operation is only valid for integer types");
  1439. using default_ops::eval_bitwise_and;
  1440. self_type temp(e);
  1441. eval_bitwise_and(m_backend, temp.m_backend);
  1442. }
  1443. template <class Exp>
  1444. void do_bitwise_or(const Exp& e, const detail::terminal&)
  1445. {
  1446. BOOST_STATIC_ASSERT_MSG(number_category<Backend>::value == number_kind_integer, "The bitwise | operation is only valid for integer types");
  1447. using default_ops::eval_bitwise_or;
  1448. eval_bitwise_or(m_backend, canonical_value(e.value()));
  1449. }
  1450. template <class Exp>
  1451. void do_bitwise_or(const Exp& e, const detail::bitwise_or&)
  1452. {
  1453. BOOST_STATIC_ASSERT_MSG(number_category<Backend>::value == number_kind_integer, "The bitwise | operation is only valid for integer types");
  1454. typedef typename Exp::left_type left_type;
  1455. typedef typename Exp::right_type right_type;
  1456. do_bitwise_or(e.left(), typename left_type::tag_type());
  1457. do_bitwise_or(e.right(), typename right_type::tag_type());
  1458. }
  1459. template <class Exp, class unknown>
  1460. void do_bitwise_or(const Exp& e, const unknown&)
  1461. {
  1462. BOOST_STATIC_ASSERT_MSG(number_category<Backend>::value == number_kind_integer, "The bitwise | operation is only valid for integer types");
  1463. using default_ops::eval_bitwise_or;
  1464. self_type temp(e);
  1465. eval_bitwise_or(m_backend, temp.m_backend);
  1466. }
  1467. template <class Exp>
  1468. void do_bitwise_xor(const Exp& e, const detail::terminal&)
  1469. {
  1470. BOOST_STATIC_ASSERT_MSG(number_category<Backend>::value == number_kind_integer, "The bitwise ^ operation is only valid for integer types");
  1471. using default_ops::eval_bitwise_xor;
  1472. eval_bitwise_xor(m_backend, canonical_value(e.value()));
  1473. }
  1474. template <class Exp>
  1475. void do_bitwise_xor(const Exp& e, const detail::bitwise_xor&)
  1476. {
  1477. BOOST_STATIC_ASSERT_MSG(number_category<Backend>::value == number_kind_integer, "The bitwise ^ operation is only valid for integer types");
  1478. typedef typename Exp::left_type left_type;
  1479. typedef typename Exp::right_type right_type;
  1480. do_bitwise_xor(e.left(), typename left_type::tag_type());
  1481. do_bitwise_xor(e.right(), typename right_type::tag_type());
  1482. }
  1483. template <class Exp, class unknown>
  1484. void do_bitwise_xor(const Exp& e, const unknown&)
  1485. {
  1486. BOOST_STATIC_ASSERT_MSG(number_category<Backend>::value == number_kind_integer, "The bitwise ^ operation is only valid for integer types");
  1487. using default_ops::eval_bitwise_xor;
  1488. self_type temp(e);
  1489. eval_bitwise_xor(m_backend, temp.m_backend);
  1490. }
  1491. // Tests if the expression contains a reference to *this:
  1492. template <class Exp>
  1493. BOOST_MP_FORCEINLINE bool contains_self(const Exp& e)const BOOST_NOEXCEPT
  1494. {
  1495. return contains_self(e, typename Exp::arity());
  1496. }
  1497. template <class Exp>
  1498. BOOST_MP_FORCEINLINE bool contains_self(const Exp& e, mpl::int_<0> const&)const BOOST_NOEXCEPT
  1499. {
  1500. return is_realy_self(e.value());
  1501. }
  1502. template <class Exp>
  1503. BOOST_MP_FORCEINLINE bool contains_self(const Exp& e, mpl::int_<1> const&)const BOOST_NOEXCEPT
  1504. {
  1505. typedef typename Exp::left_type child_type;
  1506. return contains_self(e.left(), typename child_type::arity());
  1507. }
  1508. template <class Exp>
  1509. BOOST_MP_FORCEINLINE bool contains_self(const Exp& e, mpl::int_<2> const&)const BOOST_NOEXCEPT
  1510. {
  1511. typedef typename Exp::left_type child0_type;
  1512. typedef typename Exp::right_type child1_type;
  1513. return contains_self(e.left(), typename child0_type::arity())
  1514. || contains_self(e.right(), typename child1_type::arity());
  1515. }
  1516. template <class Exp>
  1517. BOOST_MP_FORCEINLINE bool contains_self(const Exp& e, mpl::int_<3> const&)const BOOST_NOEXCEPT
  1518. {
  1519. typedef typename Exp::left_type child0_type;
  1520. typedef typename Exp::middle_type child1_type;
  1521. typedef typename Exp::right_type child2_type;
  1522. return contains_self(e.left(), typename child0_type::arity())
  1523. || contains_self(e.middle(), typename child1_type::arity())
  1524. || contains_self(e.right(), typename child2_type::arity());
  1525. }
  1526. // Test if the expression is a reference to *this:
  1527. template <class Exp>
  1528. BOOST_MP_FORCEINLINE BOOST_CONSTEXPR bool is_self(const Exp& e)const BOOST_NOEXCEPT
  1529. {
  1530. return is_self(e, typename Exp::arity());
  1531. }
  1532. template <class Exp>
  1533. BOOST_MP_FORCEINLINE BOOST_CONSTEXPR bool is_self(const Exp& e, mpl::int_<0> const&)const BOOST_NOEXCEPT
  1534. {
  1535. return is_realy_self(e.value());
  1536. }
  1537. template <class Exp, int v>
  1538. BOOST_MP_FORCEINLINE BOOST_CONSTEXPR bool is_self(const Exp&, mpl::int_<v> const&)const BOOST_NOEXCEPT
  1539. {
  1540. return false;
  1541. }
  1542. template <class Val>
  1543. BOOST_MP_FORCEINLINE BOOST_CONSTEXPR bool is_realy_self(const Val&)const BOOST_NOEXCEPT{ return false; }
  1544. BOOST_MP_FORCEINLINE BOOST_CONSTEXPR bool is_realy_self(const self_type& v)const BOOST_NOEXCEPT{ return &v == this; }
  1545. static BOOST_MP_FORCEINLINE BOOST_CONSTEXPR const Backend& function_arg_value(const self_type& v) BOOST_NOEXCEPT { return v.backend(); }
  1546. template <class V>
  1547. static BOOST_MP_FORCEINLINE BOOST_CONSTEXPR const V& function_arg_value(const V& v) BOOST_NOEXCEPT { return v; }
  1548. template <class A1, class A2, class A3, class A4>
  1549. static BOOST_MP_FORCEINLINE const A1& function_arg_value(const detail::expression<detail::terminal, A1, A2, A3, A4>& exp) BOOST_NOEXCEPT { return exp.value(); }
  1550. template <class A2, class A3, class A4>
  1551. static BOOST_MP_FORCEINLINE BOOST_CONSTEXPR const Backend& function_arg_value(const detail::expression<detail::terminal, number<Backend>, A2, A3, A4>& exp) BOOST_NOEXCEPT { return exp.value().backend(); }
  1552. Backend m_backend;
  1553. public:
  1554. //
  1555. // These shouldn't really need to be public, or even member functions, but it makes implementing
  1556. // the non-member operators way easier if they are:
  1557. //
  1558. static BOOST_MP_FORCEINLINE BOOST_CONSTEXPR const Backend& canonical_value(const self_type& v) BOOST_NOEXCEPT { return v.m_backend; }
  1559. template <class B2, expression_template_option ET>
  1560. static BOOST_MP_FORCEINLINE BOOST_CONSTEXPR const B2& canonical_value(const number<B2, ET>& v) BOOST_NOEXCEPT { return v.backend(); }
  1561. template <class V>
  1562. static BOOST_MP_FORCEINLINE BOOST_CONSTEXPR typename boost::disable_if<is_same<typename detail::canonical<V, Backend>::type, V>, typename detail::canonical<V, Backend>::type>::type
  1563. canonical_value(const V& v) BOOST_NOEXCEPT { return static_cast<typename detail::canonical<V, Backend>::type>(v); }
  1564. template <class V>
  1565. static BOOST_MP_FORCEINLINE BOOST_CONSTEXPR typename boost::enable_if<is_same<typename detail::canonical<V, Backend>::type, V>, const V&>::type
  1566. canonical_value(const V& v) BOOST_NOEXCEPT { return v; }
  1567. static BOOST_MP_FORCEINLINE typename detail::canonical<std::string, Backend>::type canonical_value(const std::string& v) BOOST_NOEXCEPT { return v.c_str(); }
  1568. };
  1569. template <class Backend, expression_template_option ExpressionTemplates>
  1570. inline std::ostream& operator << (std::ostream& os, const number<Backend, ExpressionTemplates>& r)
  1571. {
  1572. std::streamsize d = os.precision();
  1573. std::string s = r.str(d, os.flags());
  1574. std::streamsize ss = os.width();
  1575. if(ss > static_cast<std::streamsize>(s.size()))
  1576. {
  1577. char fill = os.fill();
  1578. if((os.flags() & std::ios_base::left) == std::ios_base::left)
  1579. s.append(static_cast<std::string::size_type>(ss - s.size()), fill);
  1580. else
  1581. s.insert(static_cast<std::string::size_type>(0), static_cast<std::string::size_type>(ss - s.size()), fill);
  1582. }
  1583. return os << s;
  1584. }
  1585. namespace detail{
  1586. template <class tag, class A1, class A2, class A3, class A4>
  1587. inline std::ostream& operator << (std::ostream& os, const expression<tag, A1, A2, A3, A4>& r)
  1588. {
  1589. typedef typename expression<tag, A1, A2, A3, A4>::result_type value_type;
  1590. value_type temp(r);
  1591. return os << temp;
  1592. }
  1593. //
  1594. // What follows is the input streaming code: this is not "proper" iostream code at all
  1595. // but that's fiendishly hard to write when dealing with multiple backends all
  1596. // with different requirements... yes we could deligate this to the backend author...
  1597. // but we really want backends to be EASY to write!
  1598. // For now just pull in all the characters that could possibly form the number
  1599. // and let the backend's string parser make use of it. This fixes most use cases
  1600. // including CSV type formats such as those used by the Random lib.
  1601. //
  1602. inline std::string read_string_while(std::istream& is, std::string const& permitted_chars)
  1603. {
  1604. std::ios_base::iostate state = std::ios_base::goodbit;
  1605. const std::istream::sentry sentry_check(is);
  1606. std::string result;
  1607. if(sentry_check)
  1608. {
  1609. int c = is.rdbuf()->sgetc();
  1610. for(;; c = is.rdbuf()->snextc())
  1611. if(std::istream::traits_type::eq_int_type(std::istream::traits_type::eof(), c))
  1612. { // end of file:
  1613. state |= std::ios_base::eofbit;
  1614. break;
  1615. }
  1616. else if(permitted_chars.find_first_of(std::istream::traits_type::to_char_type(c)) == std::string::npos)
  1617. {
  1618. // Invalid numeric character, stop reading:
  1619. is.rdbuf()->sputbackc(static_cast<char>(c));
  1620. break;
  1621. }
  1622. else
  1623. {
  1624. result.append(1, std::istream::traits_type::to_char_type(c));
  1625. }
  1626. }
  1627. if(!result.size())
  1628. state |= std::ios_base::failbit;
  1629. is.setstate(state);
  1630. return result;
  1631. }
  1632. } // namespace detail
  1633. template <class Backend, expression_template_option ExpressionTemplates>
  1634. inline std::istream& operator >> (std::istream& is, number<Backend, ExpressionTemplates>& r)
  1635. {
  1636. bool hex_format = (is.flags() & std::ios_base::hex) == std::ios_base::hex;
  1637. bool oct_format = (is.flags() & std::ios_base::oct) == std::ios_base::oct;
  1638. std::string s;
  1639. switch(boost::multiprecision::number_category<number<Backend, ExpressionTemplates> >::value)
  1640. {
  1641. case boost::multiprecision::number_kind_integer:
  1642. if(oct_format)
  1643. s = detail::read_string_while(is, "+-01234567");
  1644. else if(hex_format)
  1645. s = detail::read_string_while(is, "+-xXabcdefABCDEF0123456789");
  1646. else
  1647. s = detail::read_string_while(is, "+-0123456789");
  1648. break;
  1649. case boost::multiprecision::number_kind_floating_point:
  1650. s = detail::read_string_while(is, "+-eE.0123456789infINFnanNANinfinityINFINITY");
  1651. break;
  1652. default:
  1653. is >> s;
  1654. }
  1655. if(s.size())
  1656. {
  1657. if(hex_format && (number_category<Backend>::value == number_kind_integer) && ((s[0] != '0') || (s[1] != 'x')))
  1658. s.insert(s.find_first_not_of("+-"), "0x");
  1659. if(oct_format && (number_category<Backend>::value == number_kind_integer) && (s[0] != '0'))
  1660. s.insert(s.find_first_not_of("+-"), "0");
  1661. r.assign(s);
  1662. }
  1663. else if(!is.fail())
  1664. is.setstate(std::istream::failbit);
  1665. return is;
  1666. }
  1667. template <class Backend, expression_template_option ExpressionTemplates>
  1668. BOOST_MP_FORCEINLINE void swap(number<Backend, ExpressionTemplates>& a, number<Backend, ExpressionTemplates>& b)
  1669. BOOST_MP_NOEXCEPT_IF(noexcept(std::declval<number<Backend, ExpressionTemplates>&>() = std::declval<number<Backend, ExpressionTemplates>&>()))
  1670. {
  1671. a.swap(b);
  1672. }
  1673. } // namespace multiprecision
  1674. template <class T>
  1675. class rational;
  1676. template <class Backend, multiprecision::expression_template_option ExpressionTemplates>
  1677. inline std::istream& operator >> (std::istream& is, rational<multiprecision::number<Backend, ExpressionTemplates> >& r)
  1678. {
  1679. std::string s1;
  1680. multiprecision::number<Backend, ExpressionTemplates> v1, v2;
  1681. char c;
  1682. bool have_hex = false;
  1683. bool hex_format = (is.flags() & std::ios_base::hex) == std::ios_base::hex;
  1684. bool oct_format = (is.flags() & std::ios_base::oct) == std::ios_base::oct;
  1685. while((EOF != (c = static_cast<char>(is.peek()))) && (c == 'x' || c == 'X' || c == '-' || c == '+' || (c >= '0' && c <= '9') || (have_hex && (c >= 'a' && c <= 'f')) || (have_hex && (c >= 'A' && c <= 'F'))))
  1686. {
  1687. if(c == 'x' || c == 'X')
  1688. have_hex = true;
  1689. s1.append(1, c);
  1690. is.get();
  1691. }
  1692. if(hex_format && ((s1[0] != '0') || (s1[1] != 'x')))
  1693. s1.insert(static_cast<std::string::size_type>(0), "0x");
  1694. if(oct_format && (s1[0] != '0'))
  1695. s1.insert(static_cast<std::string::size_type>(0), "0");
  1696. v1.assign(s1);
  1697. s1.erase();
  1698. if(c == '/')
  1699. {
  1700. is.get();
  1701. while((EOF != (c = static_cast<char>(is.peek()))) && (c == 'x' || c == 'X' || c == '-' || c == '+' || (c >= '0' && c <= '9') || (have_hex && (c >= 'a' && c <= 'f')) || (have_hex && (c >= 'A' && c <= 'F'))))
  1702. {
  1703. if(c == 'x' || c == 'X')
  1704. have_hex = true;
  1705. s1.append(1, c);
  1706. is.get();
  1707. }
  1708. if(hex_format && ((s1[0] != '0') || (s1[1] != 'x')))
  1709. s1.insert(static_cast<std::string::size_type>(0), "0x");
  1710. if(oct_format && (s1[0] != '0'))
  1711. s1.insert(static_cast<std::string::size_type>(0), "0");
  1712. v2.assign(s1);
  1713. }
  1714. else
  1715. v2 = 1;
  1716. r.assign(v1, v2);
  1717. return is;
  1718. }
  1719. template <class T, multiprecision::expression_template_option ExpressionTemplates, class Arithmetic>
  1720. typename boost::enable_if<boost::is_arithmetic<Arithmetic>, bool>::type operator == (const rational<multiprecision::number<T, ExpressionTemplates> >& a, const Arithmetic& b)
  1721. {
  1722. return a == multiprecision::number<T, ExpressionTemplates>(b);
  1723. }
  1724. template <class T, multiprecision::expression_template_option ExpressionTemplates, class Arithmetic>
  1725. typename boost::enable_if<boost::is_arithmetic<Arithmetic>, bool>::type operator == (const Arithmetic& b, const rational<multiprecision::number<T, ExpressionTemplates> >& a)
  1726. {
  1727. return a == multiprecision::number<T, ExpressionTemplates>(b);
  1728. }
  1729. template <class T, multiprecision::expression_template_option ExpressionTemplates, class Arithmetic>
  1730. typename boost::enable_if<boost::is_arithmetic<Arithmetic>, bool>::type operator != (const rational<multiprecision::number<T, ExpressionTemplates> >& a, const Arithmetic& b)
  1731. {
  1732. return a != multiprecision::number<T, ExpressionTemplates>(b);
  1733. }
  1734. template <class T, multiprecision::expression_template_option ExpressionTemplates, class Arithmetic>
  1735. typename boost::enable_if<boost::is_arithmetic<Arithmetic>, bool>::type operator != (const Arithmetic& b, const rational<multiprecision::number<T, ExpressionTemplates> >& a)
  1736. {
  1737. return a != multiprecision::number<T, ExpressionTemplates>(b);
  1738. }
  1739. template <class T, multiprecision::expression_template_option ExpressionTemplates, class Arithmetic>
  1740. typename boost::enable_if<boost::is_arithmetic<Arithmetic>, bool>::type operator < (const rational<multiprecision::number<T, ExpressionTemplates> >& a, const Arithmetic& b)
  1741. {
  1742. return a < multiprecision::number<T, ExpressionTemplates>(b);
  1743. }
  1744. template <class T, multiprecision::expression_template_option ExpressionTemplates, class Arithmetic>
  1745. typename boost::enable_if<boost::is_arithmetic<Arithmetic>, bool>::type operator < (const Arithmetic& b, const rational<multiprecision::number<T, ExpressionTemplates> >& a)
  1746. {
  1747. return a > multiprecision::number<T, ExpressionTemplates>(b);
  1748. }
  1749. template <class T, multiprecision::expression_template_option ExpressionTemplates, class Arithmetic>
  1750. typename boost::enable_if<boost::is_arithmetic<Arithmetic>, bool>::type operator <= (const rational<multiprecision::number<T, ExpressionTemplates> >& a, const Arithmetic& b)
  1751. {
  1752. return a <= multiprecision::number<T, ExpressionTemplates>(b);
  1753. }
  1754. template <class T, multiprecision::expression_template_option ExpressionTemplates, class Arithmetic>
  1755. typename boost::enable_if<boost::is_arithmetic<Arithmetic>, bool>::type operator <= (const Arithmetic& b, const rational<multiprecision::number<T, ExpressionTemplates> >& a)
  1756. {
  1757. return a >= multiprecision::number<T, ExpressionTemplates>(b);
  1758. }
  1759. template <class T, multiprecision::expression_template_option ExpressionTemplates, class Arithmetic>
  1760. typename boost::enable_if<boost::is_arithmetic<Arithmetic>, bool>::type operator > (const rational<multiprecision::number<T, ExpressionTemplates> >& a, const Arithmetic& b)
  1761. {
  1762. return a > multiprecision::number<T, ExpressionTemplates>(b);
  1763. }
  1764. template <class T, multiprecision::expression_template_option ExpressionTemplates, class Arithmetic>
  1765. typename boost::enable_if<boost::is_arithmetic<Arithmetic>, bool>::type operator > (const Arithmetic& b, const rational<multiprecision::number<T, ExpressionTemplates> >& a)
  1766. {
  1767. return a < multiprecision::number<T, ExpressionTemplates>(b);
  1768. }
  1769. template <class T, multiprecision::expression_template_option ExpressionTemplates, class Arithmetic>
  1770. typename boost::enable_if<boost::is_arithmetic<Arithmetic>, bool>::type operator >= (const rational<multiprecision::number<T, ExpressionTemplates> >& a, const Arithmetic& b)
  1771. {
  1772. return a >= multiprecision::number<T, ExpressionTemplates>(b);
  1773. }
  1774. template <class T, multiprecision::expression_template_option ExpressionTemplates, class Arithmetic>
  1775. typename boost::enable_if<boost::is_arithmetic<Arithmetic>, bool>::type operator >= (const Arithmetic& b, const rational<multiprecision::number<T, ExpressionTemplates> >& a)
  1776. {
  1777. return a <= multiprecision::number<T, ExpressionTemplates>(b);
  1778. }
  1779. template <class T, multiprecision::expression_template_option ExpressionTemplates>
  1780. inline multiprecision::number<T, ExpressionTemplates> numerator(const rational<multiprecision::number<T, ExpressionTemplates> >& a)
  1781. {
  1782. return a.numerator();
  1783. }
  1784. template <class T, multiprecision::expression_template_option ExpressionTemplates>
  1785. inline multiprecision::number<T, ExpressionTemplates> denominator(const rational<multiprecision::number<T, ExpressionTemplates> >& a)
  1786. {
  1787. return a.denominator();
  1788. }
  1789. namespace multiprecision
  1790. {
  1791. template <class I>
  1792. struct component_type<boost::rational<I> >
  1793. {
  1794. typedef I type;
  1795. };
  1796. }
  1797. #ifdef BOOST_MSVC
  1798. #pragma warning(pop)
  1799. #endif
  1800. } // namespaces
  1801. #include <boost/multiprecision/detail/ublas_interop.hpp>
  1802. #endif