number_base.hpp 33 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008
  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_BIG_NUM_BASE_HPP
  6. #define BOOST_MATH_BIG_NUM_BASE_HPP
  7. #include <limits>
  8. #include <boost/utility/enable_if.hpp>
  9. #include <boost/type_traits/is_convertible.hpp>
  10. #include <boost/type_traits/is_constructible.hpp>
  11. #include <boost/type_traits/decay.hpp>
  12. #ifdef BOOST_MSVC
  13. # pragma warning(push)
  14. # pragma warning(disable:4307)
  15. #endif
  16. #include <boost/lexical_cast.hpp>
  17. #ifdef BOOST_MSVC
  18. # pragma warning(pop)
  19. #endif
  20. #if defined(NDEBUG) && !defined(_DEBUG)
  21. # define BOOST_MP_FORCEINLINE BOOST_FORCEINLINE
  22. #else
  23. # define BOOST_MP_FORCEINLINE inline
  24. #endif
  25. #if (defined(BOOST_GCC) && (BOOST_GCC <= 40700)) || defined(__SUNPRO_CC)
  26. # define BOOST_MP_NOEXCEPT_IF(x)
  27. #else
  28. # define BOOST_MP_NOEXCEPT_IF(x) BOOST_NOEXCEPT_IF(x)
  29. #endif
  30. #if defined(BOOST_NO_CXX11_EXPLICIT_CONVERSION_OPERATORS) || defined(__SUNPRO_CC)
  31. #define BOOST_MP_NO_CXX11_EXPLICIT_CONVERSION_OPERATORS
  32. #endif
  33. #ifdef BOOST_MSVC
  34. # pragma warning(push)
  35. # pragma warning(disable:6326)
  36. #endif
  37. namespace boost{
  38. namespace multiprecision{
  39. enum expression_template_option
  40. {
  41. et_off = 0,
  42. et_on = 1
  43. };
  44. template <class Backend>
  45. struct expression_template_default
  46. {
  47. static const expression_template_option value = et_on;
  48. };
  49. template <class Backend, expression_template_option ExpressionTemplates = expression_template_default<Backend>::value>
  50. class number;
  51. template <class T>
  52. struct is_number : public mpl::false_ {};
  53. template <class Backend, expression_template_option ExpressionTemplates>
  54. struct is_number<number<Backend, ExpressionTemplates> > : public mpl::true_ {};
  55. namespace detail{
  56. // Forward-declare an expression wrapper
  57. template<class tag, class Arg1 = void, class Arg2 = void, class Arg3 = void, class Arg4 = void>
  58. struct expression;
  59. } // namespace detail
  60. template <class T>
  61. struct is_number_expression : public mpl::false_ {};
  62. template<class tag, class Arg1, class Arg2, class Arg3, class Arg4>
  63. struct is_number_expression<detail::expression<tag, Arg1, Arg2, Arg3, Arg4> > : public mpl::true_ {};
  64. template <class T, class Num>
  65. struct is_compatible_arithmetic_type
  66. : public mpl::bool_<
  67. is_convertible<T, Num>::value
  68. && !is_same<T, Num>::value
  69. && !is_number_expression<T>::value>
  70. {};
  71. namespace detail{
  72. //
  73. // Workaround for missing abs(boost::long_long_type) and abs(__int128) on some compilers:
  74. //
  75. template <class T>
  76. BOOST_CONSTEXPR typename enable_if_c<(is_signed<T>::value || is_floating_point<T>::value), T>::type abs(T t) BOOST_NOEXCEPT
  77. {
  78. // This strange expression avoids a hardware trap in the corner case
  79. // that val is the most negative value permitted in boost::long_long_type.
  80. // See https://svn.boost.org/trac/boost/ticket/9740.
  81. return t < 0 ? T(1u) + T(-(t + 1)) : t;
  82. }
  83. template <class T>
  84. BOOST_CONSTEXPR typename enable_if_c<(is_unsigned<T>::value), T>::type abs(T t) BOOST_NOEXCEPT
  85. {
  86. return t;
  87. }
  88. #define BOOST_MP_USING_ABS using boost::multiprecision::detail::abs;
  89. template <class T>
  90. BOOST_CONSTEXPR typename enable_if_c<(is_signed<T>::value || is_floating_point<T>::value), typename make_unsigned<T>::type>::type unsigned_abs(T t) BOOST_NOEXCEPT
  91. {
  92. // This strange expression avoids a hardware trap in the corner case
  93. // that val is the most negative value permitted in boost::long_long_type.
  94. // See https://svn.boost.org/trac/boost/ticket/9740.
  95. return t < 0 ? static_cast<typename make_unsigned<T>::type>(1u) + static_cast<typename make_unsigned<T>::type>(-(t + 1)) : static_cast<typename make_unsigned<T>::type>(t);
  96. }
  97. template <class T>
  98. BOOST_CONSTEXPR typename enable_if_c<(is_unsigned<T>::value), T>::type unsigned_abs(T t) BOOST_NOEXCEPT
  99. {
  100. return t;
  101. }
  102. //
  103. // Move support:
  104. //
  105. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  106. # define BOOST_MP_MOVE(x) std::move(x)
  107. #else
  108. # define BOOST_MP_MOVE(x) x
  109. #endif
  110. template <class T>
  111. struct bits_of
  112. {
  113. BOOST_STATIC_ASSERT(is_integral<T>::value || is_enum<T>::value || std::numeric_limits<T>::is_specialized);
  114. static const unsigned value =
  115. std::numeric_limits<T>::is_specialized ?
  116. std::numeric_limits<T>::digits
  117. : sizeof(T) * CHAR_BIT - (is_signed<T>::value ? 1 : 0);
  118. };
  119. #if defined(_GLIBCXX_USE_FLOAT128) && defined(BOOST_GCC) && !defined(__STRICT_ANSI__)
  120. template<> struct bits_of<__float128> { static const unsigned value = 113; };
  121. #endif
  122. template <int b>
  123. struct has_enough_bits
  124. {
  125. template <class T>
  126. struct type : public mpl::bool_<bits_of<T>::value>= b>{};
  127. };
  128. template <class Val, class Backend, class Tag>
  129. struct canonical_imp
  130. {
  131. typedef typename remove_cv<typename decay<const Val>::type>::type type;
  132. };
  133. template <class B, class Backend, class Tag>
  134. struct canonical_imp<number<B, et_on>, Backend, Tag>
  135. {
  136. typedef B type;
  137. };
  138. template <class B, class Backend, class Tag>
  139. struct canonical_imp<number<B, et_off>, Backend, Tag>
  140. {
  141. typedef B type;
  142. };
  143. #ifdef __SUNPRO_CC
  144. template <class B, class Backend>
  145. struct canonical_imp<number<B, et_on>, Backend, mpl::int_<3> >
  146. {
  147. typedef B type;
  148. };
  149. template <class B, class Backend>
  150. struct canonical_imp<number<B, et_off>, Backend, mpl::int_<3> >
  151. {
  152. typedef B type;
  153. };
  154. #endif
  155. template <class Val, class Backend>
  156. struct canonical_imp<Val, Backend, mpl::int_<0> >
  157. {
  158. typedef typename has_enough_bits<bits_of<Val>::value>::template type<mpl::_> pred_type;
  159. typedef typename mpl::find_if<
  160. typename Backend::signed_types,
  161. pred_type
  162. >::type iter_type;
  163. typedef typename mpl::end<typename Backend::signed_types>::type end_type;
  164. typedef typename mpl::eval_if<boost::is_same<iter_type, end_type>, mpl::identity<Val>, mpl::deref<iter_type> >::type type;
  165. };
  166. template <class Val, class Backend>
  167. struct canonical_imp<Val, Backend, mpl::int_<1> >
  168. {
  169. typedef typename has_enough_bits<bits_of<Val>::value>::template type<mpl::_> pred_type;
  170. typedef typename mpl::find_if<
  171. typename Backend::unsigned_types,
  172. pred_type
  173. >::type iter_type;
  174. typedef typename mpl::end<typename Backend::unsigned_types>::type end_type;
  175. typedef typename mpl::eval_if<boost::is_same<iter_type, end_type>, mpl::identity<Val>, mpl::deref<iter_type> >::type type;
  176. };
  177. template <class Val, class Backend>
  178. struct canonical_imp<Val, Backend, mpl::int_<2> >
  179. {
  180. typedef typename has_enough_bits<bits_of<Val>::value>::template type<mpl::_> pred_type;
  181. typedef typename mpl::find_if<
  182. typename Backend::float_types,
  183. pred_type
  184. >::type iter_type;
  185. typedef typename mpl::end<typename Backend::float_types>::type end_type;
  186. typedef typename mpl::eval_if<boost::is_same<iter_type, end_type>, mpl::identity<Val>, mpl::deref<iter_type> >::type type;
  187. };
  188. template <class Val, class Backend>
  189. struct canonical_imp<Val, Backend, mpl::int_<3> >
  190. {
  191. typedef const char* type;
  192. };
  193. template <class Val, class Backend>
  194. struct canonical
  195. {
  196. typedef typename mpl::if_<
  197. is_signed<Val>,
  198. mpl::int_<0>,
  199. typename mpl::if_<
  200. is_unsigned<Val>,
  201. mpl::int_<1>,
  202. typename mpl::if_<
  203. is_floating_point<Val>,
  204. mpl::int_<2>,
  205. typename mpl::if_<
  206. mpl::or_<
  207. is_convertible<Val, const char*>,
  208. is_same<Val, std::string>
  209. >,
  210. mpl::int_<3>,
  211. mpl::int_<4>
  212. >::type
  213. >::type
  214. >::type
  215. >::type tag_type;
  216. typedef typename canonical_imp<Val, Backend, tag_type>::type type;
  217. };
  218. struct terminal{};
  219. struct negate{};
  220. struct plus{};
  221. struct minus{};
  222. struct multiplies{};
  223. struct divides{};
  224. struct modulus{};
  225. struct shift_left{};
  226. struct shift_right{};
  227. struct bitwise_and{};
  228. struct bitwise_or{};
  229. struct bitwise_xor{};
  230. struct bitwise_complement{};
  231. struct add_immediates{};
  232. struct subtract_immediates{};
  233. struct multiply_immediates{};
  234. struct divide_immediates{};
  235. struct modulus_immediates{};
  236. struct bitwise_and_immediates{};
  237. struct bitwise_or_immediates{};
  238. struct bitwise_xor_immediates{};
  239. struct complement_immediates{};
  240. struct function{};
  241. struct multiply_add{};
  242. struct multiply_subtract{};
  243. template <class T>
  244. struct backend_type;
  245. template <class T, expression_template_option ExpressionTemplates>
  246. struct backend_type<number<T, ExpressionTemplates> >
  247. {
  248. typedef T type;
  249. };
  250. template <class tag, class A1, class A2, class A3, class A4>
  251. struct backend_type<expression<tag, A1, A2, A3, A4> >
  252. {
  253. typedef typename backend_type<typename expression<tag, A1, A2, A3, A4>::result_type>::type type;
  254. };
  255. template <class T1, class T2>
  256. struct combine_expression
  257. {
  258. #ifdef BOOST_NO_CXX11_DECLTYPE
  259. typedef typename mpl::if_c<(sizeof(T1() + T2()) == sizeof(T1)), T1, T2>::type type;
  260. #else
  261. typedef decltype(T1() + T2()) type;
  262. #endif
  263. };
  264. template <class T1, expression_template_option ExpressionTemplates, class T2>
  265. struct combine_expression<number<T1, ExpressionTemplates>, T2>
  266. {
  267. typedef number<T1, ExpressionTemplates> type;
  268. };
  269. template <class T1, class T2, expression_template_option ExpressionTemplates>
  270. struct combine_expression<T1, number<T2, ExpressionTemplates> >
  271. {
  272. typedef number<T2, ExpressionTemplates> type;
  273. };
  274. template <class T, expression_template_option ExpressionTemplates>
  275. struct combine_expression<number<T, ExpressionTemplates>, number<T, ExpressionTemplates> >
  276. {
  277. typedef number<T, ExpressionTemplates> type;
  278. };
  279. template <class T1, expression_template_option ExpressionTemplates1, class T2, expression_template_option ExpressionTemplates2>
  280. struct combine_expression<number<T1, ExpressionTemplates1>, number<T2, ExpressionTemplates2> >
  281. {
  282. typedef typename mpl::if_c<
  283. is_convertible<number<T2, ExpressionTemplates2>, number<T1, ExpressionTemplates2> >::value,
  284. number<T1, ExpressionTemplates1>,
  285. number<T2, ExpressionTemplates2>
  286. >::type type;
  287. };
  288. template <class T>
  289. struct arg_type
  290. {
  291. typedef expression<terminal, T> type;
  292. };
  293. template <class Tag, class Arg1, class Arg2, class Arg3, class Arg4>
  294. struct arg_type<expression<Tag, Arg1, Arg2, Arg3, Arg4> >
  295. {
  296. typedef expression<Tag, Arg1, Arg2, Arg3, Arg4> type;
  297. };
  298. struct unmentionable
  299. {
  300. unmentionable* proc(){ return 0; }
  301. };
  302. typedef unmentionable* (unmentionable::*unmentionable_type)();
  303. template <class T>
  304. struct expression_storage
  305. {
  306. typedef const T& type;
  307. };
  308. template <class T>
  309. struct expression_storage<T*>
  310. {
  311. typedef T* type;
  312. };
  313. template <class T>
  314. struct expression_storage<const T*>
  315. {
  316. typedef const T* type;
  317. };
  318. template <class tag, class A1, class A2, class A3, class A4>
  319. struct expression_storage<expression<tag, A1, A2, A3, A4> >
  320. {
  321. typedef expression<tag, A1, A2, A3, A4> type;
  322. };
  323. template<class tag, class Arg1>
  324. struct expression<tag, Arg1, void, void, void>
  325. {
  326. typedef mpl::int_<1> arity;
  327. typedef typename arg_type<Arg1>::type left_type;
  328. typedef typename left_type::result_type left_result_type;
  329. typedef typename left_type::result_type result_type;
  330. typedef tag tag_type;
  331. explicit expression(const Arg1& a) : arg(a) {}
  332. left_type left()const { return left_type(arg); }
  333. const Arg1& left_ref()const BOOST_NOEXCEPT { return arg; }
  334. static const unsigned depth = left_type::depth + 1;
  335. #ifndef BOOST_MP_NO_CXX11_EXPLICIT_CONVERSION_OPERATORS
  336. # if (defined(__GNUC__) && (__GNUC__ == 4) && (__GNUC_MINOR__ < 7) && !defined(__clang__)) || (defined(BOOST_INTEL) && (BOOST_INTEL <= 1500))
  337. //
  338. // Horrible workaround for gcc-4.6.x which always prefers the template
  339. // operator bool() rather than the non-template operator when converting to
  340. // an arithmetic type:
  341. //
  342. template <class T, typename boost::enable_if<is_same<T, bool>, int>::type = 0>
  343. explicit operator T ()const
  344. {
  345. result_type r(*this);
  346. return static_cast<bool>(r);
  347. }
  348. template <class T, typename boost::disable_if_c<is_same<T, bool>::value || is_void<T>::value || is_number<T>::value, int>::type = 0>
  349. explicit operator T ()const
  350. {
  351. return static_cast<T>(static_cast<result_type>(*this));
  352. }
  353. # else
  354. template <class T, typename boost::disable_if_c<is_number<T>::value || is_constructible<T const&, result_type>::value, int>::type = 0>
  355. explicit operator T()const
  356. {
  357. return static_cast<T>(static_cast<result_type>(*this));
  358. }
  359. BOOST_MP_FORCEINLINE explicit operator bool()const
  360. {
  361. result_type r(*this);
  362. return static_cast<bool>(r);
  363. }
  364. explicit operator void()const {}
  365. # endif
  366. #else
  367. operator unmentionable_type()const
  368. {
  369. result_type r(*this);
  370. return r ? &unmentionable::proc : 0;
  371. }
  372. #endif
  373. template <class T>
  374. T convert_to()
  375. {
  376. result_type r(*this);
  377. return r.template convert_to<T>();
  378. }
  379. private:
  380. typename expression_storage<Arg1>::type arg;
  381. expression& operator=(const expression&);
  382. };
  383. template<class Arg1>
  384. struct expression<terminal, Arg1, void, void, void>
  385. {
  386. typedef mpl::int_<0> arity;
  387. typedef Arg1 result_type;
  388. typedef terminal tag_type;
  389. explicit expression(const Arg1& a) : arg(a) {}
  390. const Arg1& value()const BOOST_NOEXCEPT { return arg; }
  391. static const unsigned depth = 0;
  392. #ifndef BOOST_MP_NO_CXX11_EXPLICIT_CONVERSION_OPERATORS
  393. # if (defined(__GNUC__) && (__GNUC__ == 4) && (__GNUC_MINOR__ < 7) && !defined(__clang__)) || (defined(BOOST_INTEL) && (BOOST_INTEL <= 1500))
  394. //
  395. // Horrible workaround for gcc-4.6.x which always prefers the template
  396. // operator bool() rather than the non-template operator when converting to
  397. // an arithmetic type:
  398. //
  399. template <class T, typename boost::enable_if<is_same<T, bool>, int>::type = 0>
  400. explicit operator T ()const
  401. {
  402. result_type r(*this);
  403. return static_cast<bool>(r);
  404. }
  405. template <class T, typename boost::disable_if_c<is_same<T, bool>::value || is_void<T>::value || is_number<T>::value, int>::type = 0>
  406. explicit operator T ()const
  407. {
  408. return static_cast<T>(static_cast<result_type>(*this));
  409. }
  410. # else
  411. template <class T, typename boost::disable_if_c<is_number<T>::value || is_constructible<T const&, result_type>::value, int>::type = 0>
  412. explicit operator T()const
  413. {
  414. return static_cast<T>(static_cast<result_type>(*this));
  415. }
  416. BOOST_MP_FORCEINLINE explicit operator bool()const
  417. {
  418. result_type r(*this);
  419. return static_cast<bool>(r);
  420. }
  421. explicit operator void()const {}
  422. # endif
  423. #else
  424. operator unmentionable_type()const
  425. {
  426. return arg ? &unmentionable::proc : 0;
  427. }
  428. #endif
  429. template <class T>
  430. T convert_to()
  431. {
  432. result_type r(*this);
  433. return r.template convert_to<T>();
  434. }
  435. private:
  436. typename expression_storage<Arg1>::type arg;
  437. expression& operator=(const expression&);
  438. };
  439. template <class tag, class Arg1, class Arg2>
  440. struct expression<tag, Arg1, Arg2, void, void>
  441. {
  442. typedef mpl::int_<2> arity;
  443. typedef typename arg_type<Arg1>::type left_type;
  444. typedef typename arg_type<Arg2>::type right_type;
  445. typedef typename left_type::result_type left_result_type;
  446. typedef typename right_type::result_type right_result_type;
  447. typedef typename combine_expression<left_result_type, right_result_type>::type result_type;
  448. typedef tag tag_type;
  449. expression(const Arg1& a1, const Arg2& a2) : arg1(a1), arg2(a2) {}
  450. left_type left()const { return left_type(arg1); }
  451. right_type right()const { return right_type(arg2); }
  452. const Arg1& left_ref()const BOOST_NOEXCEPT { return arg1; }
  453. const Arg2& right_ref()const BOOST_NOEXCEPT { return arg2; }
  454. #ifndef BOOST_MP_NO_CXX11_EXPLICIT_CONVERSION_OPERATORS
  455. # if (defined(__GNUC__) && (__GNUC__ == 4) && (__GNUC_MINOR__ < 7) && !defined(__clang__)) || (defined(BOOST_INTEL) && (BOOST_INTEL <= 1500))
  456. //
  457. // Horrible workaround for gcc-4.6.x which always prefers the template
  458. // operator bool() rather than the non-template operator when converting to
  459. // an arithmetic type:
  460. //
  461. template <class T, typename boost::enable_if<is_same<T, bool>, int>::type = 0>
  462. explicit operator T ()const
  463. {
  464. result_type r(*this);
  465. return static_cast<bool>(r);
  466. }
  467. template <class T, typename boost::disable_if_c<is_same<T, bool>::value || is_void<T>::value || is_number<T>::value, int>::type = 0>
  468. explicit operator T ()const
  469. {
  470. return static_cast<T>(static_cast<result_type>(*this));
  471. }
  472. # else
  473. template <class T, typename boost::disable_if_c<is_number<T>::value || is_constructible<T const&, result_type>::value, int>::type = 0>
  474. explicit operator T()const
  475. {
  476. return static_cast<T>(static_cast<result_type>(*this));
  477. }
  478. BOOST_MP_FORCEINLINE explicit operator bool()const
  479. {
  480. result_type r(*this);
  481. return static_cast<bool>(r);
  482. }
  483. explicit operator void()const {}
  484. # endif
  485. #else
  486. operator unmentionable_type()const
  487. {
  488. result_type r(*this);
  489. return r ? &unmentionable::proc : 0;
  490. }
  491. #endif
  492. template <class T>
  493. T convert_to()
  494. {
  495. result_type r(*this);
  496. return r.template convert_to<T>();
  497. }
  498. static const unsigned left_depth = left_type::depth + 1;
  499. static const unsigned right_depth = right_type::depth + 1;
  500. static const unsigned depth = left_depth > right_depth ? left_depth : right_depth;
  501. private:
  502. typename expression_storage<Arg1>::type arg1;
  503. typename expression_storage<Arg2>::type arg2;
  504. expression& operator=(const expression&);
  505. };
  506. template <class tag, class Arg1, class Arg2, class Arg3>
  507. struct expression<tag, Arg1, Arg2, Arg3, void>
  508. {
  509. typedef mpl::int_<3> arity;
  510. typedef typename arg_type<Arg1>::type left_type;
  511. typedef typename arg_type<Arg2>::type middle_type;
  512. typedef typename arg_type<Arg3>::type right_type;
  513. typedef typename left_type::result_type left_result_type;
  514. typedef typename middle_type::result_type middle_result_type;
  515. typedef typename right_type::result_type right_result_type;
  516. typedef typename combine_expression<
  517. left_result_type,
  518. typename combine_expression<right_result_type, middle_result_type>::type
  519. >::type result_type;
  520. typedef tag tag_type;
  521. expression(const Arg1& a1, const Arg2& a2, const Arg3& a3) : arg1(a1), arg2(a2), arg3(a3) {}
  522. left_type left()const { return left_type(arg1); }
  523. middle_type middle()const { return middle_type(arg2); }
  524. right_type right()const { return right_type(arg3); }
  525. const Arg1& left_ref()const BOOST_NOEXCEPT { return arg1; }
  526. const Arg2& middle_ref()const BOOST_NOEXCEPT { return arg2; }
  527. const Arg3& right_ref()const BOOST_NOEXCEPT { return arg3; }
  528. #ifndef BOOST_MP_NO_CXX11_EXPLICIT_CONVERSION_OPERATORS
  529. # if (defined(__GNUC__) && (__GNUC__ == 4) && (__GNUC_MINOR__ < 7) && !defined(__clang__)) || (defined(BOOST_INTEL) && (BOOST_INTEL <= 1500))
  530. //
  531. // Horrible workaround for gcc-4.6.x which always prefers the template
  532. // operator bool() rather than the non-template operator when converting to
  533. // an arithmetic type:
  534. //
  535. template <class T, typename boost::enable_if<is_same<T, bool>, int>::type = 0>
  536. explicit operator T ()const
  537. {
  538. result_type r(*this);
  539. return static_cast<bool>(r);
  540. }
  541. template <class T, typename boost::disable_if_c<is_same<T, bool>::value || is_void<T>::value || is_number<T>::value, int>::type = 0>
  542. explicit operator T ()const
  543. {
  544. return static_cast<T>(static_cast<result_type>(*this));
  545. }
  546. # else
  547. template <class T, typename boost::disable_if_c<is_number<T>::value || is_constructible<T const&, result_type>::value, int>::type = 0>
  548. explicit operator T()const
  549. {
  550. return static_cast<T>(static_cast<result_type>(*this));
  551. }
  552. BOOST_MP_FORCEINLINE explicit operator bool()const
  553. {
  554. result_type r(*this);
  555. return static_cast<bool>(r);
  556. }
  557. explicit operator void()const {}
  558. # endif
  559. #else
  560. operator unmentionable_type()const
  561. {
  562. result_type r(*this);
  563. return r ? &unmentionable::proc : 0;
  564. }
  565. #endif
  566. template <class T>
  567. T convert_to()
  568. {
  569. result_type r(*this);
  570. return r.template convert_to<T>();
  571. }
  572. static const unsigned left_depth = left_type::depth + 1;
  573. static const unsigned middle_depth = middle_type::depth + 1;
  574. static const unsigned right_depth = right_type::depth + 1;
  575. static const unsigned depth = left_depth > right_depth ? (left_depth > middle_depth ? left_depth : middle_depth) : (right_depth > middle_depth ? right_depth : middle_depth);
  576. private:
  577. typename expression_storage<Arg1>::type arg1;
  578. typename expression_storage<Arg2>::type arg2;
  579. typename expression_storage<Arg3>::type arg3;
  580. expression& operator=(const expression&);
  581. };
  582. template <class tag, class Arg1, class Arg2, class Arg3, class Arg4>
  583. struct expression
  584. {
  585. typedef mpl::int_<4> arity;
  586. typedef typename arg_type<Arg1>::type left_type;
  587. typedef typename arg_type<Arg2>::type left_middle_type;
  588. typedef typename arg_type<Arg3>::type right_middle_type;
  589. typedef typename arg_type<Arg4>::type right_type;
  590. typedef typename left_type::result_type left_result_type;
  591. typedef typename left_middle_type::result_type left_middle_result_type;
  592. typedef typename right_middle_type::result_type right_middle_result_type;
  593. typedef typename right_type::result_type right_result_type;
  594. typedef typename combine_expression<
  595. typename combine_expression<
  596. typename combine_expression<left_result_type, left_middle_result_type>::type,
  597. right_middle_result_type
  598. >::type,
  599. right_result_type
  600. >::type result_type;
  601. typedef tag tag_type;
  602. expression(const Arg1& a1, const Arg2& a2, const Arg3& a3, const Arg4& a4) : arg1(a1), arg2(a2), arg3(a3), arg4(a4) {}
  603. left_type left()const { return left_type(arg1); }
  604. left_middle_type left_middle()const { return left_middle_type(arg2); }
  605. right_middle_type right_middle()const { return right_middle_type(arg3); }
  606. right_type right()const { return right_type(arg4); }
  607. const Arg1& left_ref()const BOOST_NOEXCEPT { return arg1; }
  608. const Arg2& left_middle_ref()const BOOST_NOEXCEPT { return arg2; }
  609. const Arg3& right_middle_ref()const BOOST_NOEXCEPT { return arg3; }
  610. const Arg4& right_ref()const BOOST_NOEXCEPT { return arg4; }
  611. #ifndef BOOST_MP_NO_CXX11_EXPLICIT_CONVERSION_OPERATORS
  612. # if (defined(__GNUC__) && (__GNUC__ == 4) && (__GNUC_MINOR__ < 7) && !defined(__clang__)) || (defined(BOOST_INTEL) && (BOOST_INTEL <= 1500))
  613. //
  614. // Horrible workaround for gcc-4.6.x which always prefers the template
  615. // operator bool() rather than the non-template operator when converting to
  616. // an arithmetic type:
  617. //
  618. template <class T, typename boost::enable_if<is_same<T, bool>, int>::type = 0>
  619. explicit operator T ()const
  620. {
  621. result_type r(*this);
  622. return static_cast<bool>(r);
  623. }
  624. template <class T, typename boost::disable_if_c<is_same<T, bool>::value || is_void<T>::value || is_number<T>::value, int>::type = 0>
  625. explicit operator T ()const
  626. {
  627. return static_cast<T>(static_cast<result_type>(*this));
  628. }
  629. # else
  630. template <class T, typename boost::disable_if_c<is_number<T>::value || is_constructible<T const&, result_type>::value, int>::type = 0>
  631. explicit operator T()const
  632. {
  633. return static_cast<T>(static_cast<result_type>(*this));
  634. }
  635. BOOST_MP_FORCEINLINE explicit operator bool()const
  636. {
  637. result_type r(*this);
  638. return static_cast<bool>(r);
  639. }
  640. explicit operator void()const {}
  641. # endif
  642. #else
  643. operator unmentionable_type()const
  644. {
  645. result_type r(*this);
  646. return r ? &unmentionable::proc : 0;
  647. }
  648. #endif
  649. template <class T>
  650. T convert_to()
  651. {
  652. result_type r(*this);
  653. return r.template convert_to<T>();
  654. }
  655. static const unsigned left_depth = left_type::depth + 1;
  656. static const unsigned left_middle_depth = left_middle_type::depth + 1;
  657. static const unsigned right_middle_depth = right_middle_type::depth + 1;
  658. static const unsigned right_depth = right_type::depth + 1;
  659. static const unsigned left_max_depth = left_depth > left_middle_depth ? left_depth : left_middle_depth;
  660. static const unsigned right_max_depth = right_depth > right_middle_depth ? right_depth : right_middle_depth;
  661. static const unsigned depth = left_max_depth > right_max_depth ? left_max_depth : right_max_depth;
  662. private:
  663. typename expression_storage<Arg1>::type arg1;
  664. typename expression_storage<Arg2>::type arg2;
  665. typename expression_storage<Arg3>::type arg3;
  666. typename expression_storage<Arg4>::type arg4;
  667. expression& operator=(const expression&);
  668. };
  669. template <class T>
  670. struct digits2
  671. {
  672. BOOST_STATIC_ASSERT(std::numeric_limits<T>::is_specialized);
  673. BOOST_STATIC_ASSERT((std::numeric_limits<T>::radix == 2) || (std::numeric_limits<T>::radix == 10));
  674. // If we really have so many digits that this fails, then we're probably going to hit other problems anyway:
  675. BOOST_STATIC_ASSERT(LONG_MAX / 1000 > (std::numeric_limits<T>::digits + 1));
  676. static const long value = std::numeric_limits<T>::radix == 10 ? (((std::numeric_limits<T>::digits + 1) * 1000L) / 301L) : std::numeric_limits<T>::digits;
  677. };
  678. #ifndef BOOST_MP_MIN_EXPONENT_DIGITS
  679. #ifdef _MSC_VER
  680. # define BOOST_MP_MIN_EXPONENT_DIGITS 2
  681. #else
  682. # define BOOST_MP_MIN_EXPONENT_DIGITS 2
  683. #endif
  684. #endif
  685. template <class S>
  686. void format_float_string(S& str, boost::intmax_t my_exp, boost::intmax_t digits, std::ios_base::fmtflags f, bool iszero)
  687. {
  688. typedef typename S::size_type size_type;
  689. bool scientific = (f & std::ios_base::scientific) == std::ios_base::scientific;
  690. bool fixed = (f & std::ios_base::fixed) == std::ios_base::fixed;
  691. bool showpoint = (f & std::ios_base::showpoint) == std::ios_base::showpoint;
  692. bool showpos = (f & std::ios_base::showpos) == std::ios_base::showpos;
  693. bool neg = str.size() && (str[0] == '-');
  694. if(neg)
  695. str.erase(0, 1);
  696. if(digits == 0)
  697. {
  698. digits = (std::max)(str.size(), size_type(16));
  699. }
  700. if(iszero || str.empty() || (str.find_first_not_of('0') == S::npos))
  701. {
  702. // We will be printing zero, even though the value might not
  703. // actually be zero (it just may have been rounded to zero).
  704. str = "0";
  705. if(scientific || fixed)
  706. {
  707. str.append(1, '.');
  708. str.append(size_type(digits), '0');
  709. if(scientific)
  710. str.append("e+00");
  711. }
  712. else
  713. {
  714. if(showpoint)
  715. {
  716. str.append(1, '.');
  717. if(digits > 1)
  718. str.append(size_type(digits - 1), '0');
  719. }
  720. }
  721. if(neg)
  722. str.insert(static_cast<std::string::size_type>(0), 1, '-');
  723. else if(showpos)
  724. str.insert(static_cast<std::string::size_type>(0), 1, '+');
  725. return;
  726. }
  727. if(!fixed && !scientific && !showpoint)
  728. {
  729. //
  730. // Suppress trailing zeros:
  731. //
  732. std::string::iterator pos = str.end();
  733. while(pos != str.begin() && *--pos == '0'){}
  734. if(pos != str.end())
  735. ++pos;
  736. str.erase(pos, str.end());
  737. if(str.empty())
  738. str = '0';
  739. }
  740. else if(!fixed || (my_exp >= 0))
  741. {
  742. //
  743. // Pad out the end with zero's if we need to:
  744. //
  745. boost::intmax_t chars = str.size();
  746. chars = digits - chars;
  747. if(scientific)
  748. ++chars;
  749. if(chars > 0)
  750. {
  751. str.append(static_cast<std::string::size_type>(chars), '0');
  752. }
  753. }
  754. if(fixed || (!scientific && (my_exp >= -4) && (my_exp < digits)))
  755. {
  756. if(1 + my_exp > static_cast<boost::intmax_t>(str.size()))
  757. {
  758. // Just pad out the end with zeros:
  759. str.append(static_cast<std::string::size_type>(1 + my_exp - str.size()), '0');
  760. if(showpoint || fixed)
  761. str.append(".");
  762. }
  763. else if(my_exp + 1 < static_cast<boost::intmax_t>(str.size()))
  764. {
  765. if(my_exp < 0)
  766. {
  767. str.insert(static_cast<std::string::size_type>(0), static_cast<std::string::size_type>(-1 - my_exp), '0');
  768. str.insert(static_cast<std::string::size_type>(0), "0.");
  769. }
  770. else
  771. {
  772. // Insert the decimal point:
  773. str.insert(static_cast<std::string::size_type>(my_exp + 1), 1, '.');
  774. }
  775. }
  776. else if(showpoint || fixed) // we have exactly the digits we require to left of the point
  777. str += ".";
  778. if(fixed)
  779. {
  780. // We may need to add trailing zeros:
  781. boost::intmax_t l = str.find('.') + 1;
  782. l = digits - (str.size() - l);
  783. if(l > 0)
  784. str.append(size_type(l), '0');
  785. }
  786. }
  787. else
  788. {
  789. BOOST_MP_USING_ABS
  790. // Scientific format:
  791. if(showpoint || (str.size() > 1))
  792. str.insert(static_cast<std::string::size_type>(1u), 1, '.');
  793. str.append(static_cast<std::string::size_type>(1u), 'e');
  794. S e = boost::lexical_cast<S>(abs(my_exp));
  795. if(e.size() < BOOST_MP_MIN_EXPONENT_DIGITS)
  796. e.insert(static_cast<std::string::size_type>(0), BOOST_MP_MIN_EXPONENT_DIGITS - e.size(), '0');
  797. if(my_exp < 0)
  798. e.insert(static_cast<std::string::size_type>(0), 1, '-');
  799. else
  800. e.insert(static_cast<std::string::size_type>(0), 1, '+');
  801. str.append(e);
  802. }
  803. if(neg)
  804. str.insert(static_cast<std::string::size_type>(0), 1, '-');
  805. else if(showpos)
  806. str.insert(static_cast<std::string::size_type>(0), 1, '+');
  807. }
  808. template <class V>
  809. void check_shift_range(V val, const mpl::true_&, const mpl::true_&)
  810. {
  811. if(val > (std::numeric_limits<std::size_t>::max)())
  812. BOOST_THROW_EXCEPTION(std::out_of_range("Can not shift by a value greater than std::numeric_limits<std::size_t>::max()."));
  813. if(val < 0)
  814. BOOST_THROW_EXCEPTION(std::out_of_range("Can not shift by a negative value."));
  815. }
  816. template <class V>
  817. void check_shift_range(V val, const mpl::false_&, const mpl::true_&)
  818. {
  819. if(val < 0)
  820. BOOST_THROW_EXCEPTION(std::out_of_range("Can not shift by a negative value."));
  821. }
  822. template <class V>
  823. void check_shift_range(V val, const mpl::true_&, const mpl::false_&)
  824. {
  825. if(val > (std::numeric_limits<std::size_t>::max)())
  826. BOOST_THROW_EXCEPTION(std::out_of_range("Can not shift by a value greater than std::numeric_limits<std::size_t>::max()."));
  827. }
  828. template <class V>
  829. void check_shift_range(V, const mpl::false_&, const mpl::false_&) BOOST_NOEXCEPT{}
  830. } // namespace detail
  831. //
  832. // Traits class, lets us know what kind of number we have, defaults to a floating point type:
  833. //
  834. enum number_category_type
  835. {
  836. number_kind_unknown = -1,
  837. number_kind_integer = 0,
  838. number_kind_floating_point = 1,
  839. number_kind_rational = 2,
  840. number_kind_fixed_point = 3
  841. };
  842. template <class Num>
  843. struct number_category : public mpl::int_<std::numeric_limits<Num>::is_integer ? number_kind_integer : (std::numeric_limits<Num>::max_exponent ? number_kind_floating_point : number_kind_unknown)> {};
  844. template <class Backend, expression_template_option ExpressionTemplates>
  845. struct number_category<number<Backend, ExpressionTemplates> > : public number_category<Backend>{};
  846. template <class tag, class A1, class A2, class A3, class A4>
  847. struct number_category<detail::expression<tag, A1, A2, A3, A4> > : public number_category<typename detail::expression<tag, A1, A2, A3, A4>::result_type>{};
  848. template <class T>
  849. struct component_type;
  850. template <class T, expression_template_option ExpressionTemplates>
  851. struct component_type<number<T, ExpressionTemplates> > : public component_type<T>{};
  852. template <class tag, class A1, class A2, class A3, class A4>
  853. struct component_type<detail::expression<tag, A1, A2, A3, A4> > : public component_type<typename detail::expression<tag, A1, A2, A3, A4>::result_type>{};
  854. template <class T>
  855. struct is_unsigned_number : public mpl::false_{};
  856. template <class Backend, expression_template_option ExpressionTemplates>
  857. struct is_unsigned_number<number<Backend, ExpressionTemplates> > : public is_unsigned_number<Backend> {};
  858. template <class T>
  859. struct is_signed_number : public mpl::bool_<!is_unsigned_number<T>::value> {};
  860. template <class T>
  861. struct is_interval_number : public mpl::false_ {};
  862. template <class Backend, expression_template_option ExpressionTemplates>
  863. struct is_interval_number<number<Backend, ExpressionTemplates> > : public is_interval_number<Backend>{};
  864. }} // namespaces
  865. namespace boost{ namespace math{ namespace tools{
  866. template <class T>
  867. struct promote_arg;
  868. template <class tag, class A1, class A2, class A3, class A4>
  869. struct promote_arg<boost::multiprecision::detail::expression<tag, A1, A2, A3, A4> >
  870. {
  871. typedef typename boost::multiprecision::detail::expression<tag, A1, A2, A3, A4>::result_type type;
  872. };
  873. template <class R, class B, boost::multiprecision::expression_template_option ET>
  874. inline R real_cast(const boost::multiprecision::number<B, ET>& val)
  875. {
  876. return val.template convert_to<R>();
  877. }
  878. template <class R, class tag, class A1, class A2, class A3, class A4>
  879. inline R real_cast(const boost::multiprecision::detail::expression<tag, A1, A2, A3, A4>& val)
  880. {
  881. typedef typename boost::multiprecision::detail::expression<tag, A1, A2, A3, A4>::result_type val_type;
  882. return val_type(val).template convert_to<R>();
  883. }
  884. }
  885. namespace constants{
  886. template <class T>
  887. struct is_explicitly_convertible_from_string;
  888. template <class B, boost::multiprecision::expression_template_option ET>
  889. struct is_explicitly_convertible_from_string<boost::multiprecision::number<B, ET> >
  890. {
  891. static const bool value = true;
  892. };
  893. }
  894. }}
  895. #ifdef BOOST_MSVC
  896. # pragma warning(pop)
  897. #endif
  898. #endif // BOOST_MATH_BIG_NUM_BASE_HPP