cpp_bin_float.hpp 78 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647
  1. ///////////////////////////////////////////////////////////////
  2. // Copyright 2013 John Maddock. Distributed under the Boost
  3. // Software License, Version 1.0. (See accompanying file
  4. // LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_
  5. #ifndef BOOST_MATH_CPP_BIN_FLOAT_HPP
  6. #define BOOST_MATH_CPP_BIN_FLOAT_HPP
  7. #include <boost/multiprecision/cpp_int.hpp>
  8. #include <boost/multiprecision/integer.hpp>
  9. #include <boost/math/special_functions/trunc.hpp>
  10. #include <boost/multiprecision/detail/float_string_cvt.hpp>
  11. namespace boost{ namespace multiprecision{ namespace backends{
  12. enum digit_base_type
  13. {
  14. digit_base_2 = 2,
  15. digit_base_10 = 10
  16. };
  17. #ifdef BOOST_MSVC
  18. #pragma warning(push)
  19. #pragma warning(disable:4522 6326) // multiple assignment operators specified, comparison of two constants
  20. #endif
  21. namespace detail{
  22. template <class U>
  23. inline typename enable_if_c<is_unsigned<U>::value, bool>::type is_negative(U) { return false; }
  24. template <class S>
  25. inline typename disable_if_c<is_unsigned<S>::value, bool>::type is_negative(S s) { return s < 0; }
  26. }
  27. template <unsigned Digits, digit_base_type DigitBase = digit_base_10, class Allocator = void, class Exponent = int, Exponent MinExponent = 0, Exponent MaxExponent = 0>
  28. class cpp_bin_float
  29. {
  30. public:
  31. static const unsigned bit_count = DigitBase == digit_base_2 ? Digits : (Digits * 1000uL) / 301uL + (((Digits * 1000uL) % 301) ? 2u : 1u);
  32. typedef cpp_int_backend<is_void<Allocator>::value ? bit_count : 0, bit_count, is_void<Allocator>::value ? unsigned_magnitude : signed_magnitude, unchecked, Allocator> rep_type;
  33. typedef cpp_int_backend<is_void<Allocator>::value ? 2 * bit_count : 0, 2 * bit_count, is_void<Allocator>::value ? unsigned_magnitude : signed_magnitude, unchecked, Allocator> double_rep_type;
  34. typedef typename rep_type::signed_types signed_types;
  35. typedef typename rep_type::unsigned_types unsigned_types;
  36. typedef boost::mpl::list<float, double, long double> float_types;
  37. typedef Exponent exponent_type;
  38. static const exponent_type max_exponent_limit = boost::integer_traits<exponent_type>::const_max - 2 * static_cast<exponent_type>(bit_count);
  39. static const exponent_type min_exponent_limit = boost::integer_traits<exponent_type>::const_min + 2 * static_cast<exponent_type>(bit_count);
  40. BOOST_STATIC_ASSERT_MSG(MinExponent >= min_exponent_limit, "Template parameter MinExponent is too negative for our internal logic to function correctly, sorry!");
  41. BOOST_STATIC_ASSERT_MSG(MaxExponent <= max_exponent_limit, "Template parameter MaxExponent is too large for our internal logic to function correctly, sorry!");
  42. BOOST_STATIC_ASSERT_MSG(MinExponent <= 0, "Template parameter MinExponent can not be positive!");
  43. BOOST_STATIC_ASSERT_MSG(MaxExponent >= 0, "Template parameter MaxExponent can not be negative!");
  44. static const exponent_type max_exponent = MaxExponent == 0 ? max_exponent_limit : MaxExponent;
  45. static const exponent_type min_exponent = MinExponent == 0 ? min_exponent_limit : MinExponent;
  46. static const exponent_type exponent_zero = max_exponent + 1;
  47. static const exponent_type exponent_infinity = max_exponent + 2;
  48. static const exponent_type exponent_nan = max_exponent + 3;
  49. private:
  50. rep_type m_data;
  51. exponent_type m_exponent;
  52. bool m_sign;
  53. public:
  54. cpp_bin_float() BOOST_MP_NOEXCEPT_IF(noexcept(rep_type())) : m_data(), m_exponent(exponent_nan), m_sign(false) {}
  55. cpp_bin_float(const cpp_bin_float &o) BOOST_MP_NOEXCEPT_IF(noexcept(rep_type(std::declval<const rep_type&>())))
  56. : m_data(o.m_data), m_exponent(o.m_exponent), m_sign(o.m_sign) {}
  57. template <unsigned D, digit_base_type B, class A, class E, E MinE, E MaxE>
  58. cpp_bin_float(const cpp_bin_float<D, B, A, E, MinE, MaxE> &o, typename boost::enable_if_c<(bit_count >= cpp_bin_float<D, B, A, E, MinE, MaxE>::bit_count)>::type const* = 0)
  59. {
  60. *this = o;
  61. }
  62. template <unsigned D, digit_base_type B, class A, class E, E MinE, E MaxE>
  63. explicit cpp_bin_float(const cpp_bin_float<D, B, A, E, MinE, MaxE> &o, typename boost::disable_if_c<(bit_count >= cpp_bin_float<D, B, A, E, MinE, MaxE>::bit_count)>::type const* = 0)
  64. : m_exponent(o.exponent()), m_sign(o.sign())
  65. {
  66. *this = o;
  67. }
  68. template <class Float>
  69. cpp_bin_float(const Float& f,
  70. typename boost::enable_if_c<
  71. (number_category<Float>::value == number_kind_floating_point)
  72. && (std::numeric_limits<Float>::digits <= (int)bit_count)
  73. && (std::numeric_limits<Float>::radix == 2)
  74. >::type const* = 0)
  75. : m_data(), m_exponent(0), m_sign(false)
  76. {
  77. this->assign_float(f);
  78. }
  79. template <class Float>
  80. explicit cpp_bin_float(const Float& f,
  81. typename boost::enable_if_c<
  82. (number_category<Float>::value == number_kind_floating_point)
  83. && (std::numeric_limits<Float>::digits > (int)bit_count)
  84. && (std::numeric_limits<Float>::radix == 2)
  85. >::type const* = 0)
  86. : m_data(), m_exponent(0), m_sign(false)
  87. {
  88. this->assign_float(f);
  89. }
  90. cpp_bin_float& operator=(const cpp_bin_float &o) BOOST_MP_NOEXCEPT_IF(noexcept(std::declval<rep_type&>() = std::declval<const rep_type&>()))
  91. {
  92. m_data = o.m_data;
  93. m_exponent = o.m_exponent;
  94. m_sign = o.m_sign;
  95. return *this;
  96. }
  97. template <unsigned D, digit_base_type B, class A, class E, E MinE, E MaxE>
  98. cpp_bin_float& operator=(const cpp_bin_float<D, B, A, E, MinE, MaxE> &f)
  99. {
  100. switch(eval_fpclassify(f))
  101. {
  102. case FP_ZERO:
  103. m_data = limb_type(0);
  104. m_sign = false;
  105. m_exponent = exponent_zero;
  106. break;
  107. case FP_NAN:
  108. m_data = limb_type(0);
  109. m_sign = false;
  110. m_exponent = exponent_nan;
  111. break;;
  112. case FP_INFINITE:
  113. m_data = limb_type(0);
  114. m_sign = f.sign();
  115. m_exponent = exponent_infinity;
  116. break;
  117. default:
  118. typename cpp_bin_float<D, B, A, E, MinE, MaxE>::rep_type b(f.bits());
  119. this->exponent() = f.exponent() + (int)bit_count - (int)cpp_bin_float<D, B, A, E, MinE, MaxE>::bit_count;
  120. this->sign() = f.sign();
  121. copy_and_round(*this, b);
  122. }
  123. return *this;
  124. }
  125. template <class Float>
  126. typename boost::enable_if_c<
  127. (number_category<Float>::value == number_kind_floating_point)
  128. //&& (std::numeric_limits<Float>::digits <= (int)bit_count)
  129. && (std::numeric_limits<Float>::radix == 2), cpp_bin_float&>::type operator=(const Float& f)
  130. {
  131. return assign_float(f);
  132. }
  133. template <class Float>
  134. typename boost::enable_if_c<is_floating_point<Float>::value, cpp_bin_float&>::type assign_float(Float f)
  135. {
  136. BOOST_MATH_STD_USING
  137. using default_ops::eval_add;
  138. typedef typename boost::multiprecision::detail::canonical<int, cpp_bin_float>::type bf_int_type;
  139. switch((boost::math::fpclassify)(f))
  140. {
  141. case FP_ZERO:
  142. m_data = limb_type(0);
  143. m_sign = false;
  144. m_exponent = exponent_zero;
  145. return *this;
  146. case FP_NAN:
  147. m_data = limb_type(0);
  148. m_sign = false;
  149. m_exponent = exponent_nan;
  150. return *this;
  151. case FP_INFINITE:
  152. m_data = limb_type(0);
  153. m_sign = (f < 0);
  154. m_exponent = exponent_infinity;
  155. return *this;
  156. }
  157. if(f < 0)
  158. {
  159. *this = -f;
  160. this->negate();
  161. return *this;
  162. }
  163. typedef typename mpl::front<unsigned_types>::type ui_type;
  164. m_data = static_cast<ui_type>(0u);
  165. m_sign = false;
  166. m_exponent = 0;
  167. static const int bits = sizeof(int) * CHAR_BIT - 1;
  168. int e;
  169. f = frexp(f, &e);
  170. while(f)
  171. {
  172. f = ldexp(f, bits);
  173. e -= bits;
  174. #ifndef BOOST_MATH_NO_LONG_DOUBLE_MATH_FUNCTIONS
  175. int ipart = itrunc(f);
  176. #else
  177. int ipart = static_cast<int>(f);
  178. #endif
  179. f -= ipart;
  180. m_exponent += bits;
  181. cpp_bin_float t;
  182. t = static_cast<bf_int_type>(ipart);
  183. eval_add(*this, t);
  184. }
  185. m_exponent += static_cast<Exponent>(e);
  186. return *this;
  187. }
  188. template <class Float>
  189. typename boost::enable_if_c<
  190. (number_category<Float>::value == number_kind_floating_point)
  191. && !boost::is_floating_point<Float>::value
  192. /*&& (std::numeric_limits<number<Float> >::radix == 2)*/,
  193. cpp_bin_float&>::type assign_float(Float f)
  194. {
  195. BOOST_MATH_STD_USING
  196. using default_ops::eval_add;
  197. using default_ops::eval_get_sign;
  198. using default_ops::eval_convert_to;
  199. using default_ops::eval_subtract;
  200. typedef typename boost::multiprecision::detail::canonical<int, Float>::type f_int_type;
  201. typedef typename boost::multiprecision::detail::canonical<int, cpp_bin_float>::type bf_int_type;
  202. switch(eval_fpclassify(f))
  203. {
  204. case FP_ZERO:
  205. m_data = limb_type(0);
  206. m_sign = false;
  207. m_exponent = exponent_zero;
  208. return *this;
  209. case FP_NAN:
  210. m_data = limb_type(0);
  211. m_sign = false;
  212. m_exponent = exponent_nan;
  213. return *this;
  214. case FP_INFINITE:
  215. m_data = limb_type(0);
  216. m_sign = (f < 0);
  217. m_exponent = exponent_infinity;
  218. return *this;
  219. }
  220. if(eval_get_sign(f) < 0)
  221. {
  222. f.negate();
  223. *this = f;
  224. this->negate();
  225. return *this;
  226. }
  227. typedef typename mpl::front<unsigned_types>::type ui_type;
  228. m_data = static_cast<ui_type>(0u);
  229. m_sign = false;
  230. m_exponent = 0;
  231. static const int bits = sizeof(int) * CHAR_BIT - 1;
  232. int e;
  233. eval_frexp(f, f, &e);
  234. while(eval_get_sign(f) != 0)
  235. {
  236. eval_ldexp(f, f, bits);
  237. e -= bits;
  238. int ipart;
  239. eval_convert_to(&ipart, f);
  240. eval_subtract(f, static_cast<f_int_type>(ipart));
  241. m_exponent += bits;
  242. eval_add(*this, static_cast<bf_int_type>(ipart));
  243. }
  244. m_exponent += e;
  245. if(m_exponent > max_exponent)
  246. m_exponent = exponent_infinity;
  247. if(m_exponent < min_exponent)
  248. {
  249. m_data = limb_type(0u);
  250. m_exponent = exponent_zero;
  251. m_sign = false;
  252. }
  253. else if(eval_get_sign(m_data) == 0)
  254. {
  255. m_exponent = exponent_zero;
  256. m_sign = false;
  257. }
  258. return *this;
  259. }
  260. template <class I>
  261. typename boost::enable_if<is_integral<I>, cpp_bin_float&>::type operator=(const I& i)
  262. {
  263. using default_ops::eval_bit_test;
  264. if(!i)
  265. {
  266. m_data = static_cast<limb_type>(0);
  267. m_exponent = exponent_zero;
  268. m_sign = false;
  269. }
  270. else
  271. {
  272. typedef typename make_unsigned<I>::type ui_type;
  273. ui_type fi = static_cast<ui_type>(boost::multiprecision::detail::unsigned_abs(i));
  274. typedef typename boost::multiprecision::detail::canonical<ui_type, rep_type>::type ar_type;
  275. m_data = static_cast<ar_type>(fi);
  276. unsigned shift = msb(fi);
  277. if(shift >= bit_count)
  278. {
  279. m_exponent = static_cast<Exponent>(shift);
  280. m_data = static_cast<ar_type>(fi >> (shift + 1 - bit_count));
  281. }
  282. else
  283. {
  284. m_exponent = static_cast<Exponent>(shift);
  285. eval_left_shift(m_data, bit_count - shift - 1);
  286. }
  287. BOOST_ASSERT(eval_bit_test(m_data, bit_count-1));
  288. m_sign = detail::is_negative(i);
  289. }
  290. return *this;
  291. }
  292. cpp_bin_float& operator=(const char *s);
  293. void swap(cpp_bin_float &o) BOOST_NOEXCEPT
  294. {
  295. m_data.swap(o.m_data);
  296. std::swap(m_exponent, o.m_exponent);
  297. std::swap(m_sign, o.m_sign);
  298. }
  299. std::string str(std::streamsize dig, std::ios_base::fmtflags f) const;
  300. void negate()
  301. {
  302. if((m_exponent != exponent_zero) && (m_exponent != exponent_nan))
  303. m_sign = !m_sign;
  304. }
  305. int compare(const cpp_bin_float &o) const BOOST_NOEXCEPT
  306. {
  307. if(m_sign != o.m_sign)
  308. return m_sign ? -1 : 1;
  309. int result;
  310. if(m_exponent == exponent_nan)
  311. return -1;
  312. else if(m_exponent != o.m_exponent)
  313. {
  314. if(m_exponent == exponent_zero)
  315. result = -1;
  316. else if(o.m_exponent == exponent_zero)
  317. result = 1;
  318. else
  319. result = m_exponent > o.m_exponent ? 1 : -1;
  320. }
  321. else
  322. result = m_data.compare(o.m_data);
  323. if(m_sign)
  324. result = -result;
  325. return result;
  326. }
  327. template <class A>
  328. int compare(const A& o) const BOOST_NOEXCEPT
  329. {
  330. cpp_bin_float b;
  331. b = o;
  332. return compare(b);
  333. }
  334. rep_type& bits() { return m_data; }
  335. const rep_type& bits()const { return m_data; }
  336. exponent_type& exponent() { return m_exponent; }
  337. const exponent_type& exponent()const { return m_exponent; }
  338. bool& sign() { return m_sign; }
  339. const bool& sign()const { return m_sign; }
  340. void check_invariants()
  341. {
  342. using default_ops::eval_bit_test;
  343. using default_ops::eval_is_zero;
  344. if((m_exponent <= max_exponent) && (m_exponent >= min_exponent))
  345. {
  346. BOOST_ASSERT(eval_bit_test(m_data, bit_count - 1));
  347. }
  348. else
  349. {
  350. BOOST_ASSERT(m_exponent > max_exponent);
  351. BOOST_ASSERT(m_exponent <= exponent_nan);
  352. BOOST_ASSERT(eval_is_zero(m_data));
  353. }
  354. }
  355. template<class Archive>
  356. void serialize(Archive & ar, const unsigned int /*version*/)
  357. {
  358. ar & m_data;
  359. ar & m_exponent;
  360. ar & m_sign;
  361. }
  362. };
  363. #ifdef BOOST_MSVC
  364. #pragma warning(pop)
  365. #endif
  366. template <unsigned Digits, digit_base_type DigitBase, class Allocator, class Exponent, Exponent MinE, Exponent MaxE, class Int>
  367. inline void copy_and_round(cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> &res, Int &arg)
  368. {
  369. // Precondition: exponent of res must have been set before this function is called
  370. // as we may need to adjust it based on how many cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::bit_count in arg are set.
  371. using default_ops::eval_msb;
  372. using default_ops::eval_lsb;
  373. using default_ops::eval_left_shift;
  374. using default_ops::eval_bit_test;
  375. using default_ops::eval_right_shift;
  376. using default_ops::eval_increment;
  377. using default_ops::eval_get_sign;
  378. // cancellation may have resulted in arg being all zeros:
  379. if(eval_get_sign(arg) == 0)
  380. {
  381. res.exponent() = cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_zero;
  382. res.sign() = false;
  383. res.bits() = static_cast<limb_type>(0u);
  384. return;
  385. }
  386. int msb = eval_msb(arg);
  387. if(static_cast<int>(cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::bit_count) > msb + 1)
  388. {
  389. // Must have had cancellation in subtraction, shift left and copy:
  390. res.bits() = arg;
  391. eval_left_shift(res.bits(), cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::bit_count - msb - 1);
  392. res.exponent() -= static_cast<Exponent>(cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::bit_count - msb - 1);
  393. }
  394. else if(static_cast<int>(cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::bit_count) < msb + 1)
  395. {
  396. // We have more cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::bit_count than we need, so round as required,
  397. // first get the rounding bit:
  398. bool roundup = eval_bit_test(arg, msb - cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::bit_count);
  399. // Then check for a tie:
  400. if(roundup && (msb - cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::bit_count == eval_lsb(arg)))
  401. {
  402. // Ties round towards even:
  403. if(!eval_bit_test(arg, msb - cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::bit_count + 1))
  404. roundup = false;
  405. }
  406. // Shift off the cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::bit_count we don't need:
  407. eval_right_shift(arg, msb - cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::bit_count + 1);
  408. res.exponent() += static_cast<Exponent>(msb - (int)cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::bit_count + 1);
  409. if(roundup)
  410. {
  411. eval_increment(arg);
  412. if(eval_bit_test(arg, cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::bit_count))
  413. {
  414. // This happens very very rairly:
  415. eval_right_shift(arg, 1u);
  416. ++res.exponent();
  417. }
  418. }
  419. res.bits() = arg;
  420. }
  421. else
  422. {
  423. res.bits() = arg;
  424. }
  425. BOOST_ASSERT((eval_msb(res.bits()) == cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::bit_count - 1));
  426. if(res.exponent() > cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::max_exponent)
  427. {
  428. // Overflow:
  429. res.exponent() = cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_infinity;
  430. res.bits() = static_cast<limb_type>(0u);
  431. }
  432. else if(res.exponent() < cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::min_exponent)
  433. {
  434. // Underflow:
  435. res.exponent() = cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_zero;
  436. res.bits() = static_cast<limb_type>(0u);
  437. res.sign() = false;
  438. }
  439. }
  440. template <unsigned Digits, digit_base_type DigitBase, class Allocator, class Exponent, Exponent MinE, Exponent MaxE>
  441. inline void do_eval_add(cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> &res, const cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> &a, const cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> &b)
  442. {
  443. using default_ops::eval_add;
  444. using default_ops::eval_bit_test;
  445. typename cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::double_rep_type dt;
  446. // Special cases first:
  447. switch(a.exponent())
  448. {
  449. case cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_zero:
  450. res = b;
  451. if(res.sign())
  452. res.negate();
  453. return;
  454. case cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_infinity:
  455. if(b.exponent() == cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_nan)
  456. res = b;
  457. else
  458. res = a;
  459. return; // result is still infinite.
  460. case cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_nan:
  461. res = a;
  462. return; // result is still a NaN.
  463. }
  464. switch(b.exponent())
  465. {
  466. case cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_zero:
  467. res = a;
  468. return;
  469. case cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_infinity:
  470. res = b;
  471. if(res.sign())
  472. res.negate();
  473. return; // result is infinite.
  474. case cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_nan:
  475. res = b;
  476. return; // result is a NaN.
  477. }
  478. typename cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_type e_diff = a.exponent() - b.exponent();
  479. bool s = a.sign();
  480. if(e_diff >= 0)
  481. {
  482. dt = a.bits();
  483. if(e_diff < (int)cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::bit_count)
  484. {
  485. eval_left_shift(dt, e_diff);
  486. res.exponent() = a.exponent() - e_diff;
  487. eval_add(dt, b.bits());
  488. }
  489. else
  490. res.exponent() = a.exponent();
  491. }
  492. else
  493. {
  494. dt= b.bits();
  495. if(-e_diff < (int)cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::bit_count)
  496. {
  497. eval_left_shift(dt, -e_diff);
  498. res.exponent() = b.exponent() + e_diff;
  499. eval_add(dt, a.bits());
  500. }
  501. else
  502. res.exponent() = b.exponent();
  503. }
  504. copy_and_round(res, dt);
  505. res.check_invariants();
  506. if(res.sign() != s)
  507. res.negate();
  508. }
  509. template <unsigned Digits, digit_base_type DigitBase, class Allocator, class Exponent, Exponent MinE, Exponent MaxE>
  510. inline void do_eval_subtract(cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> &res, const cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> &a, const cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> &b)
  511. {
  512. using default_ops::eval_subtract;
  513. using default_ops::eval_bit_test;
  514. typename cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::double_rep_type dt;
  515. // Special cases first:
  516. switch(a.exponent())
  517. {
  518. case cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_zero:
  519. if(b.exponent() == cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_nan)
  520. res = std::numeric_limits<number<cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> > >::quiet_NaN().backend();
  521. else
  522. {
  523. res = b;
  524. if(!res.sign())
  525. res.negate();
  526. }
  527. return;
  528. case cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_infinity:
  529. if((b.exponent() == cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_nan) || (b.exponent() == cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_infinity))
  530. res = std::numeric_limits<number<cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> > >::quiet_NaN().backend();
  531. else
  532. res = a;
  533. return;
  534. case cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_nan:
  535. res = a;
  536. return; // result is still a NaN.
  537. }
  538. switch(b.exponent())
  539. {
  540. case cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_zero:
  541. res = a;
  542. return;
  543. case cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_infinity:
  544. res.exponent() = cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_infinity;
  545. res.sign() = true;
  546. res.bits() = static_cast<limb_type>(0u);
  547. return; // result is a NaN.
  548. case cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_nan:
  549. res = b;
  550. return; // result is still a NaN.
  551. }
  552. typename cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_type e_diff = a.exponent() - b.exponent();
  553. bool s = a.sign();
  554. if((e_diff > 0) || ((e_diff == 0) && a.bits().compare(b.bits()) >= 0))
  555. {
  556. dt = a.bits();
  557. if(e_diff <= (int)cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::bit_count)
  558. {
  559. eval_left_shift(dt, e_diff);
  560. res.exponent() = a.exponent() - e_diff;
  561. eval_subtract(dt, b.bits());
  562. }
  563. else
  564. res.exponent() = a.exponent();
  565. }
  566. else
  567. {
  568. dt = b.bits();
  569. if(-e_diff <= (int)cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::bit_count)
  570. {
  571. eval_left_shift(dt, -e_diff);
  572. res.exponent() = b.exponent() + e_diff;
  573. eval_subtract(dt, a.bits());
  574. }
  575. else
  576. res.exponent() = b.exponent();
  577. s = !s;
  578. }
  579. copy_and_round(res, dt);
  580. if(res.sign() != s)
  581. res.negate();
  582. res.check_invariants();
  583. }
  584. template <unsigned Digits, digit_base_type DigitBase, class Allocator, class Exponent, Exponent MinE, Exponent MaxE>
  585. inline void eval_add(cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> &res, const cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> &a, const cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> &b)
  586. {
  587. if(a.sign() == b.sign())
  588. do_eval_add(res, a, b);
  589. else
  590. do_eval_subtract(res, a, b);
  591. }
  592. template <unsigned Digits, digit_base_type DigitBase, class Allocator, class Exponent, Exponent MinE, Exponent MaxE>
  593. inline void eval_add(cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> &res, const cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> &a)
  594. {
  595. return eval_add(res, res, a);
  596. }
  597. template <unsigned Digits, digit_base_type DigitBase, class Allocator, class Exponent, Exponent MinE, Exponent MaxE>
  598. inline void eval_subtract(cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> &res, const cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> &a, const cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> &b)
  599. {
  600. if(a.sign() != b.sign())
  601. do_eval_add(res, a, b);
  602. else
  603. do_eval_subtract(res, a, b);
  604. }
  605. template <unsigned Digits, digit_base_type DigitBase, class Allocator, class Exponent, Exponent MinE, Exponent MaxE>
  606. inline void eval_subtract(cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> &res, const cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> &a)
  607. {
  608. return eval_subtract(res, res, a);
  609. }
  610. template <unsigned Digits, digit_base_type DigitBase, class Allocator, class Exponent, Exponent MinE, Exponent MaxE>
  611. inline void eval_multiply(cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> &res, const cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> &a, const cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> &b)
  612. {
  613. using default_ops::eval_bit_test;
  614. using default_ops::eval_multiply;
  615. // Special cases first:
  616. switch(a.exponent())
  617. {
  618. case cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_zero:
  619. if(b.exponent() == cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_nan)
  620. res = b;
  621. else
  622. res = a;
  623. return;
  624. case cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_infinity:
  625. switch(b.exponent())
  626. {
  627. case cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_zero:
  628. res = std::numeric_limits<number<cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> > >::quiet_NaN().backend();
  629. break;
  630. case cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_nan:
  631. res = b;
  632. break;
  633. default:
  634. res = a;
  635. break;
  636. }
  637. return;
  638. case cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_nan:
  639. res = a;
  640. return;
  641. }
  642. if(b.exponent() > cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::max_exponent)
  643. {
  644. res = b;
  645. return;
  646. }
  647. if((a.exponent() > 0) && (b.exponent() > 0))
  648. {
  649. if(cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::max_exponent + 2 - a.exponent() < b.exponent())
  650. {
  651. // We will certainly overflow:
  652. res.exponent() = cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_infinity;
  653. res.sign() = a.sign() != b.sign();
  654. res.bits() = static_cast<limb_type>(0u);
  655. return;
  656. }
  657. }
  658. if((a.exponent() < 0) && (b.exponent() < 0))
  659. {
  660. if(cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::min_exponent - 2 - a.exponent() > b.exponent())
  661. {
  662. // We will certainly underflow:
  663. res.exponent() = cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_zero;
  664. res.sign() = false;
  665. res.bits() = static_cast<limb_type>(0u);
  666. return;
  667. }
  668. }
  669. typename cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::double_rep_type dt;
  670. eval_multiply(dt, a.bits(), b.bits());
  671. res.exponent() = a.exponent() + b.exponent() - cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::bit_count + 1;
  672. copy_and_round(res, dt);
  673. res.check_invariants();
  674. res.sign() = a.sign() != b.sign();
  675. }
  676. template <unsigned Digits, digit_base_type DigitBase, class Allocator, class Exponent, Exponent MinE, Exponent MaxE>
  677. inline void eval_multiply(cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> &res, const cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> &a)
  678. {
  679. eval_multiply(res, res, a);
  680. }
  681. template <unsigned Digits, digit_base_type DigitBase, class Allocator, class Exponent, Exponent MinE, Exponent MaxE, class U>
  682. inline typename enable_if_c<is_unsigned<U>::value>::type eval_multiply(cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> &res, const cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> &a, const U &b)
  683. {
  684. using default_ops::eval_bit_test;
  685. using default_ops::eval_multiply;
  686. // Special cases first:
  687. switch(a.exponent())
  688. {
  689. case cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_zero:
  690. res = a;
  691. return;
  692. case cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_infinity:
  693. if(b == 0)
  694. res = std::numeric_limits<number<cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> > >::quiet_NaN().backend();
  695. else
  696. res = a;
  697. return;
  698. case cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_nan:
  699. res = a;
  700. return;
  701. }
  702. typename cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::double_rep_type dt;
  703. typedef typename boost::multiprecision::detail::canonical<U, typename cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::double_rep_type>::type canon_ui_type;
  704. eval_multiply(dt, a.bits(), static_cast<canon_ui_type>(b));
  705. res.exponent() = a.exponent();
  706. copy_and_round(res, dt);
  707. res.check_invariants();
  708. res.sign() = a.sign();
  709. }
  710. template <unsigned Digits, digit_base_type DigitBase, class Allocator, class Exponent, Exponent MinE, Exponent MaxE, class U>
  711. inline typename enable_if_c<is_unsigned<U>::value>::type eval_multiply(cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> &res, const U &b)
  712. {
  713. eval_multiply(res, res, b);
  714. }
  715. template <unsigned Digits, digit_base_type DigitBase, class Allocator, class Exponent, Exponent MinE, Exponent MaxE, class S>
  716. inline typename enable_if_c<is_signed<S>::value>::type eval_multiply(cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> &res, const cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> &a, const S &b)
  717. {
  718. typedef typename make_unsigned<S>::type ui_type;
  719. eval_multiply(res, a, static_cast<ui_type>(boost::multiprecision::detail::unsigned_abs(b)));
  720. if(b < 0)
  721. res.negate();
  722. }
  723. template <unsigned Digits, digit_base_type DigitBase, class Allocator, class Exponent, Exponent MinE, Exponent MaxE, class S>
  724. inline typename enable_if_c<is_signed<S>::value>::type eval_multiply(cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> &res, const S &b)
  725. {
  726. eval_multiply(res, res, b);
  727. }
  728. template <unsigned Digits, digit_base_type DigitBase, class Allocator, class Exponent, Exponent MinE, Exponent MaxE>
  729. inline void eval_divide(cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> &res, const cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> &u, const cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> &v)
  730. {
  731. #ifdef BOOST_MSVC
  732. #pragma warning(push)
  733. #pragma warning(disable:6326) // comparison of two constants
  734. #endif
  735. using default_ops::eval_subtract;
  736. using default_ops::eval_qr;
  737. using default_ops::eval_bit_test;
  738. using default_ops::eval_get_sign;
  739. using default_ops::eval_increment;
  740. //
  741. // Special cases first:
  742. //
  743. switch(u.exponent())
  744. {
  745. case cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_zero:
  746. switch(v.exponent())
  747. {
  748. case cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_zero:
  749. case cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_nan:
  750. res = std::numeric_limits<number<cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> > >::quiet_NaN().backend();
  751. return;
  752. }
  753. res = u;
  754. return;
  755. case cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_infinity:
  756. switch(v.exponent())
  757. {
  758. case cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_infinity:
  759. case cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_nan:
  760. res = std::numeric_limits<number<cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> > >::quiet_NaN().backend();
  761. return;
  762. }
  763. res = u;
  764. return;
  765. case cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_nan:
  766. res = std::numeric_limits<number<cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> > >::quiet_NaN().backend();
  767. return;
  768. }
  769. switch(v.exponent())
  770. {
  771. case cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_zero:
  772. {
  773. bool s = u.sign() != v.sign();
  774. res = std::numeric_limits<number<cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> > >::infinity().backend();
  775. res.sign() = s;
  776. return;
  777. }
  778. case cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_infinity:
  779. res.exponent() = cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_zero;
  780. res.bits() = limb_type(0);
  781. res.sign() = false;
  782. return;
  783. case cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_nan:
  784. res = std::numeric_limits<number<cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> > >::quiet_NaN().backend();
  785. return;
  786. }
  787. // We can scale u and v so that both are integers, then perform integer
  788. // division to obtain quotient q and remainder r, such that:
  789. //
  790. // q * v + r = u
  791. //
  792. // and hense:
  793. //
  794. // q + r/v = u/v
  795. //
  796. // From this, assuming q has cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::bit_count
  797. // bits we only need to determine whether
  798. // r/v is less than, equal to, or greater than 0.5 to determine rounding -
  799. // this we can do with a shift and comparison.
  800. //
  801. // We can set the exponent and sign of the result up front:
  802. //
  803. res.exponent() = u.exponent() - v.exponent() - 1;
  804. res.sign() = u.sign() != v.sign();
  805. //
  806. // Now get the quotient and remainder:
  807. //
  808. typename cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::double_rep_type t(u.bits()), t2(v.bits()), q, r;
  809. eval_left_shift(t, cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::bit_count);
  810. eval_qr(t, t2, q, r);
  811. //
  812. // We now have either "cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::bit_count"
  813. // or "cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::bit_count+1" significant
  814. // bits in q.
  815. //
  816. static const unsigned limb_bits = sizeof(limb_type) * CHAR_BIT;
  817. if(eval_bit_test(q, cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::bit_count))
  818. {
  819. //
  820. // OK we have cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::bit_count+1 bits,
  821. // so we already have rounding info,
  822. // we just need to changes things if the last bit is 1 and either the
  823. // remainder is non-zero (ie we do not have a tie) or the quotient would
  824. // be odd if it were shifted to the correct number of bits (ie a tiebreak).
  825. //
  826. BOOST_ASSERT((eval_msb(q) == cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::bit_count));
  827. if((q.limbs()[0] & 1u) && (eval_get_sign(r) || (q.limbs()[0] & 2u)))
  828. {
  829. eval_increment(q);
  830. }
  831. }
  832. else
  833. {
  834. //
  835. // We have exactly "cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::bit_count" bits in q.
  836. // Get rounding info, which we can get by comparing 2r with v.
  837. // We want to call copy_and_round to handle rounding and general cleanup,
  838. // so we'll left shift q and add some fake digits on the end to represent
  839. // how we'll be rounding.
  840. //
  841. BOOST_ASSERT((eval_msb(q) == cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::bit_count - 1));
  842. static const unsigned lshift = (cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::bit_count < limb_bits) ? 2 : limb_bits;
  843. eval_left_shift(q, lshift);
  844. res.exponent() -= lshift;
  845. eval_left_shift(r, 1u);
  846. int c = r.compare(v.bits());
  847. if(c == 0)
  848. q.limbs()[0] |= static_cast<limb_type>(1u) << (lshift - 1);
  849. else if(c > 0)
  850. q.limbs()[0] |= (static_cast<limb_type>(1u) << (lshift - 1)) + static_cast<limb_type>(1u);
  851. }
  852. copy_and_round(res, q);
  853. #ifdef BOOST_MSVC
  854. #pragma warning(pop)
  855. #endif
  856. }
  857. template <unsigned Digits, digit_base_type DigitBase, class Allocator, class Exponent, Exponent MinE, Exponent MaxE>
  858. inline void eval_divide(cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> &res, const cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> &arg)
  859. {
  860. eval_divide(res, res, arg);
  861. }
  862. template <unsigned Digits, digit_base_type DigitBase, class Allocator, class Exponent, Exponent MinE, Exponent MaxE, class U>
  863. inline typename enable_if_c<is_unsigned<U>::value>::type eval_divide(cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> &res, const cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> &u, const U &v)
  864. {
  865. #ifdef BOOST_MSVC
  866. #pragma warning(push)
  867. #pragma warning(disable:6326) // comparison of two constants
  868. #endif
  869. using default_ops::eval_subtract;
  870. using default_ops::eval_qr;
  871. using default_ops::eval_bit_test;
  872. using default_ops::eval_get_sign;
  873. using default_ops::eval_increment;
  874. //
  875. // Special cases first:
  876. //
  877. switch(u.exponent())
  878. {
  879. case cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_zero:
  880. if(v == 0)
  881. {
  882. res = std::numeric_limits<number<cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> > >::quiet_NaN().backend();
  883. return;
  884. }
  885. res = u;
  886. return;
  887. case cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_infinity:
  888. res = u;
  889. return;
  890. case cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_nan:
  891. res = std::numeric_limits<number<cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> > >::quiet_NaN().backend();
  892. return;
  893. }
  894. if(v == 0)
  895. {
  896. bool s = u.sign();
  897. res = std::numeric_limits<number<cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> > >::infinity().backend();
  898. res.sign() = s;
  899. return;
  900. }
  901. // We can scale u and v so that both are integers, then perform integer
  902. // division to obtain quotient q and remainder r, such that:
  903. //
  904. // q * v + r = u
  905. //
  906. // and hense:
  907. //
  908. // q + r/v = u/v
  909. //
  910. // From this, assuming q has "cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::bit_count" cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::bit_count, we only need to determine whether
  911. // r/v is less than, equal to, or greater than 0.5 to determine rounding -
  912. // this we can do with a shift and comparison.
  913. //
  914. // We can set the exponent and sign of the result up front:
  915. //
  916. int gb = msb(v);
  917. res.exponent() = u.exponent() - static_cast<Exponent>(gb) - static_cast<Exponent>(1);
  918. res.sign() = u.sign();
  919. //
  920. // Now get the quotient and remainder:
  921. //
  922. typename cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::double_rep_type t(u.bits()), q, r;
  923. eval_left_shift(t, gb + 1);
  924. eval_qr(t, number<typename cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::double_rep_type>::canonical_value(v), q, r);
  925. //
  926. // We now have either "cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::bit_count" or "cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::bit_count+1" significant cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::bit_count in q.
  927. //
  928. static const unsigned limb_bits = sizeof(limb_type) * CHAR_BIT;
  929. if(eval_bit_test(q, cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::bit_count))
  930. {
  931. //
  932. // OK we have cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::bit_count+1 cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::bit_count, so we already have rounding info,
  933. // we just need to changes things if the last bit is 1 and the
  934. // remainder is non-zero (ie we do not have a tie).
  935. //
  936. BOOST_ASSERT((eval_msb(q) == cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::bit_count));
  937. if((q.limbs()[0] & 1u) && eval_get_sign(r))
  938. {
  939. eval_increment(q);
  940. }
  941. }
  942. else
  943. {
  944. //
  945. // We have exactly "cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::bit_count" cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::bit_count in q.
  946. // Get rounding info, which we can get by comparing 2r with v.
  947. // We want to call copy_and_round to handle rounding and general cleanup,
  948. // so we'll left shift q and add some fake cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::bit_count on the end to represent
  949. // how we'll be rounding.
  950. //
  951. BOOST_ASSERT((eval_msb(q) == cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::bit_count - 1));
  952. static const unsigned lshift = cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::bit_count < limb_bits ? 2 : limb_bits;
  953. eval_left_shift(q, lshift);
  954. res.exponent() -= lshift;
  955. eval_left_shift(r, 1u);
  956. int c = r.compare(number<typename cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::double_rep_type>::canonical_value(v));
  957. if(c == 0)
  958. q.limbs()[0] |= static_cast<limb_type>(1u) << (lshift - 1);
  959. else if(c > 0)
  960. q.limbs()[0] |= (static_cast<limb_type>(1u) << (lshift - 1)) + static_cast<limb_type>(1u);
  961. }
  962. copy_and_round(res, q);
  963. #ifdef BOOST_MSVC
  964. #pragma warning(pop)
  965. #endif
  966. }
  967. template <unsigned Digits, digit_base_type DigitBase, class Allocator, class Exponent, Exponent MinE, Exponent MaxE, class U>
  968. inline typename enable_if_c<is_unsigned<U>::value>::type eval_divide(cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> &res, const U &v)
  969. {
  970. eval_divide(res, res, v);
  971. }
  972. template <unsigned Digits, digit_base_type DigitBase, class Allocator, class Exponent, Exponent MinE, Exponent MaxE, class S>
  973. inline typename enable_if_c<is_signed<S>::value>::type eval_divide(cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> &res, const cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> &u, const S &v)
  974. {
  975. typedef typename make_unsigned<S>::type ui_type;
  976. eval_divide(res, u, static_cast<ui_type>(boost::multiprecision::detail::unsigned_abs(v)));
  977. if(v < 0)
  978. res.negate();
  979. }
  980. template <unsigned Digits, digit_base_type DigitBase, class Allocator, class Exponent, Exponent MinE, Exponent MaxE, class S>
  981. inline typename enable_if_c<is_signed<S>::value>::type eval_divide(cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> &res, const S &v)
  982. {
  983. eval_divide(res, res, v);
  984. }
  985. template <unsigned Digits, digit_base_type DigitBase, class Allocator, class Exponent, Exponent MinE, Exponent MaxE>
  986. inline int eval_get_sign(const cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> &arg)
  987. {
  988. return arg.exponent() == cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_zero ? 0 : arg.sign() ? -1 : 1;
  989. }
  990. template <unsigned Digits, digit_base_type DigitBase, class Allocator, class Exponent, Exponent MinE, Exponent MaxE>
  991. inline bool eval_is_zero(const cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> &arg)
  992. {
  993. return arg.exponent() == cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_zero;
  994. }
  995. template <unsigned Digits, digit_base_type DigitBase, class Allocator, class Exponent, Exponent MinE, Exponent MaxE>
  996. inline bool eval_eq(const cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> &a, cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> &b)
  997. {
  998. return (a.exponent() == b.exponent())
  999. && (a.sign() == b.sign())
  1000. && (a.bits().compare(b.bits()) == 0)
  1001. && (a.exponent() != cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_nan);
  1002. }
  1003. template <unsigned Digits, digit_base_type DigitBase, class Allocator, class Exponent, Exponent MinE, Exponent MaxE>
  1004. inline void eval_convert_to(boost::long_long_type *res, const cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> &arg)
  1005. {
  1006. switch(arg.exponent())
  1007. {
  1008. case cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_zero:
  1009. *res = 0;
  1010. return;
  1011. case cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_nan:
  1012. BOOST_THROW_EXCEPTION(std::runtime_error("Could not convert NaN to integer."));
  1013. case cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_infinity:
  1014. *res = (std::numeric_limits<boost::long_long_type>::max)();
  1015. if(arg.sign())
  1016. *res = -*res;
  1017. return;
  1018. }
  1019. typename cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::rep_type man(arg.bits());
  1020. typename mpl::if_c<sizeof(typename cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_type) < sizeof(int), int, typename cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_type>::type shift
  1021. = (int)cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::bit_count - 1 - arg.exponent();
  1022. if(shift > (int)cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::bit_count - 1)
  1023. {
  1024. *res = 0;
  1025. return;
  1026. }
  1027. if(arg.sign() && (arg.compare((std::numeric_limits<boost::long_long_type>::min)()) <= 0))
  1028. {
  1029. *res = (std::numeric_limits<boost::long_long_type>::min)();
  1030. return;
  1031. }
  1032. else if(!arg.sign() && (arg.compare((std::numeric_limits<boost::long_long_type>::max)()) >= 0))
  1033. {
  1034. *res = (std::numeric_limits<boost::long_long_type>::max)();
  1035. return;
  1036. }
  1037. eval_right_shift(man, shift);
  1038. eval_convert_to(res, man);
  1039. if(arg.sign())
  1040. {
  1041. *res = -*res;
  1042. }
  1043. }
  1044. template <unsigned Digits, digit_base_type DigitBase, class Allocator, class Exponent, Exponent MinE, Exponent MaxE>
  1045. inline void eval_convert_to(boost::ulong_long_type *res, const cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> &arg)
  1046. {
  1047. switch(arg.exponent())
  1048. {
  1049. case cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_zero:
  1050. *res = 0;
  1051. return;
  1052. case cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_nan:
  1053. BOOST_THROW_EXCEPTION(std::runtime_error("Could not convert NaN to integer."));
  1054. case cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_infinity:
  1055. *res = (std::numeric_limits<boost::ulong_long_type>::max)();
  1056. return;
  1057. }
  1058. typename cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::rep_type man(arg.bits());
  1059. typename mpl::if_c<sizeof(typename cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_type) < sizeof(int), int, typename cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_type>::type shift
  1060. = (int)cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::bit_count - 1 - arg.exponent();
  1061. if(shift > (int)cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::bit_count - 1)
  1062. {
  1063. *res = 0;
  1064. return;
  1065. }
  1066. else if(shift < 0)
  1067. {
  1068. // TODO: what if we have fewer cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::bit_count than a boost::long_long_type?
  1069. *res = (std::numeric_limits<boost::long_long_type>::max)();
  1070. return;
  1071. }
  1072. eval_right_shift(man, shift);
  1073. eval_convert_to(res, man);
  1074. }
  1075. template <class Float, unsigned Digits, digit_base_type DigitBase, class Allocator, class Exponent, Exponent MinE, Exponent MaxE>
  1076. inline typename boost::enable_if_c<boost::is_float<Float>::value>::type eval_convert_to(Float *res, const cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> &original_arg)
  1077. {
  1078. //
  1079. // Perform rounding first, then afterwards extract the digits:
  1080. //
  1081. cpp_bin_float<std::numeric_limits<Float>::digits, digit_base_2, void, Exponent, MinE, MaxE> arg(original_arg);
  1082. switch(arg.exponent())
  1083. {
  1084. case cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_zero:
  1085. *res = 0;
  1086. return;
  1087. case cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_nan:
  1088. *res = std::numeric_limits<Float>::quiet_NaN();
  1089. return;
  1090. case cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_infinity:
  1091. *res = (std::numeric_limits<Float>::infinity)();
  1092. if(arg.sign())
  1093. *res = -*res;
  1094. return;
  1095. }
  1096. typename cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_type e = arg.exponent();
  1097. e -= cpp_bin_float<std::numeric_limits<Float>::digits, digit_base_2, void, Exponent, MinE, MaxE>::bit_count - 1;
  1098. *res = std::ldexp(static_cast<Float>(*arg.bits().limbs()), e);
  1099. for(unsigned i = 1; i < arg.bits().size(); ++i)
  1100. {
  1101. e += sizeof(*arg.bits().limbs()) * CHAR_BIT;
  1102. *res += std::ldexp(static_cast<Float>(arg.bits().limbs()[i]), e);
  1103. }
  1104. if(arg.sign())
  1105. *res = -*res;
  1106. }
  1107. template <unsigned Digits, digit_base_type DigitBase, class Allocator, class Exponent, Exponent MinE, Exponent MaxE>
  1108. inline void eval_frexp(cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> &res, const cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> &arg, Exponent *e)
  1109. {
  1110. switch(arg.exponent())
  1111. {
  1112. case cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_zero:
  1113. case cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_nan:
  1114. case cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_infinity:
  1115. *e = 0;
  1116. res = arg;
  1117. return;
  1118. }
  1119. res = arg;
  1120. *e = arg.exponent() + 1;
  1121. res.exponent() = -1;
  1122. }
  1123. template <unsigned Digits, digit_base_type DigitBase, class Allocator, class Exponent, Exponent MinE, Exponent MaxE, class I>
  1124. inline void eval_frexp(cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> &res, const cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> &arg, I *pe)
  1125. {
  1126. Exponent e;
  1127. eval_frexp(res, arg, &e);
  1128. if((e > (std::numeric_limits<I>::max)()) || (e < (std::numeric_limits<I>::min)()))
  1129. {
  1130. BOOST_THROW_EXCEPTION(std::runtime_error("Exponent was outside of the range of the argument type to frexp."));
  1131. }
  1132. *pe = static_cast<I>(e);
  1133. }
  1134. template <unsigned Digits, digit_base_type DigitBase, class Allocator, class Exponent, Exponent MinE, Exponent MaxE>
  1135. inline void eval_ldexp(cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> &res, const cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> &arg, Exponent e)
  1136. {
  1137. switch(arg.exponent())
  1138. {
  1139. case cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_zero:
  1140. case cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_nan:
  1141. case cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_infinity:
  1142. res = arg;
  1143. return;
  1144. }
  1145. if((e > 0) && (cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::max_exponent - e < arg.exponent()))
  1146. {
  1147. // Overflow:
  1148. res = std::numeric_limits<number<cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> > >::infinity().backend();
  1149. res.sign() = arg.sign();
  1150. }
  1151. else if((e < 0) && (cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::min_exponent - e > arg.exponent()))
  1152. {
  1153. // Underflow:
  1154. res = limb_type(0);
  1155. }
  1156. else
  1157. {
  1158. res = arg;
  1159. res.exponent() += e;
  1160. }
  1161. }
  1162. template <unsigned Digits, digit_base_type DigitBase, class Allocator, class Exponent, Exponent MinE, Exponent MaxE, class I>
  1163. inline typename enable_if_c<is_unsigned<I>::value>::type eval_ldexp(cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> &res, const cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> &arg, I e)
  1164. {
  1165. typedef typename make_signed<I>::type si_type;
  1166. if(e > static_cast<I>((std::numeric_limits<si_type>::max)()))
  1167. res = std::numeric_limits<number<cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> > >::infinity().backend();
  1168. else
  1169. eval_ldexp(res, arg, static_cast<si_type>(e));
  1170. }
  1171. template <unsigned Digits, digit_base_type DigitBase, class Allocator, class Exponent, Exponent MinE, Exponent MaxE, class I>
  1172. inline typename enable_if_c<is_signed<I>::value>::type eval_ldexp(cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> &res, const cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> &arg, I e)
  1173. {
  1174. if((e > (std::numeric_limits<Exponent>::max)()) || (e < (std::numeric_limits<Exponent>::min)()))
  1175. {
  1176. res = std::numeric_limits<number<cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> > >::infinity().backend();
  1177. if(e < 0)
  1178. res.negate();
  1179. }
  1180. else
  1181. eval_ldexp(res, arg, static_cast<Exponent>(e));
  1182. }
  1183. /*
  1184. * Sign manipulation
  1185. */
  1186. template <unsigned Digits, digit_base_type DigitBase, class Allocator, class Exponent, Exponent MinE, Exponent MaxE>
  1187. inline void eval_abs(cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> &res, const cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> &arg)
  1188. {
  1189. res = arg;
  1190. res.sign() = false;
  1191. }
  1192. template <unsigned Digits, digit_base_type DigitBase, class Allocator, class Exponent, Exponent MinE, Exponent MaxE>
  1193. inline void eval_fabs(cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> &res, const cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> &arg)
  1194. {
  1195. res = arg;
  1196. res.sign() = false;
  1197. }
  1198. template <unsigned Digits, digit_base_type DigitBase, class Allocator, class Exponent, Exponent MinE, Exponent MaxE>
  1199. inline int eval_fpclassify(const cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> &arg)
  1200. {
  1201. switch(arg.exponent())
  1202. {
  1203. case cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_zero:
  1204. return FP_ZERO;
  1205. case cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_infinity:
  1206. return FP_INFINITE;
  1207. case cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_nan:
  1208. return FP_NAN;
  1209. }
  1210. return FP_NORMAL;
  1211. }
  1212. template <unsigned Digits, digit_base_type DigitBase, class Allocator, class Exponent, Exponent MinE, Exponent MaxE>
  1213. inline void eval_sqrt(cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> &res, const cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> &arg)
  1214. {
  1215. using default_ops::eval_integer_sqrt;
  1216. using default_ops::eval_bit_test;
  1217. using default_ops::eval_increment;
  1218. switch(arg.exponent())
  1219. {
  1220. case cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_zero:
  1221. case cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_nan:
  1222. res = arg;
  1223. return;
  1224. case cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_infinity:
  1225. res = std::numeric_limits<number<cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> > >::quiet_NaN().backend();
  1226. return;
  1227. }
  1228. if(arg.sign())
  1229. {
  1230. res = std::numeric_limits<number<cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> > >::quiet_NaN().backend();
  1231. return;
  1232. }
  1233. typename cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::double_rep_type t(arg.bits()), r, s;
  1234. eval_left_shift(t, arg.exponent() & 1 ? cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::bit_count : cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::bit_count - 1);
  1235. eval_integer_sqrt(s, r, t);
  1236. if(!eval_bit_test(s, cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::bit_count))
  1237. {
  1238. // We have exactly the right number of cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::bit_count in the result, round as required:
  1239. if(s.compare(r) < 0)
  1240. {
  1241. eval_increment(s);
  1242. }
  1243. }
  1244. typename cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_type ae = arg.exponent();
  1245. res.exponent() = ae / 2;
  1246. if((ae & 1) && (ae < 0))
  1247. --res.exponent();
  1248. copy_and_round(res, s);
  1249. }
  1250. template <unsigned Digits, digit_base_type DigitBase, class Allocator, class Exponent, Exponent MinE, Exponent MaxE>
  1251. inline void eval_floor(cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> &res, const cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> &arg)
  1252. {
  1253. using default_ops::eval_increment;
  1254. switch(arg.exponent())
  1255. {
  1256. case cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_zero:
  1257. case cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_nan:
  1258. case cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_infinity:
  1259. res = arg;
  1260. return;
  1261. }
  1262. typename mpl::if_c<sizeof(typename cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_type) < sizeof(int), int, typename cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_type>::type shift =
  1263. (int)cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::bit_count - arg.exponent() - 1;
  1264. if((arg.exponent() > (int)cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::max_exponent) || (shift <= 0))
  1265. {
  1266. // Either arg is already an integer, or a special value:
  1267. res = arg;
  1268. return;
  1269. }
  1270. if(shift >= (int)cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::bit_count)
  1271. {
  1272. res = static_cast<signed_limb_type>(arg.sign() ? -1 : 0);
  1273. return;
  1274. }
  1275. bool fractional = (int)eval_lsb(arg.bits()) < shift;
  1276. res = arg;
  1277. eval_right_shift(res.bits(), shift);
  1278. if(fractional && res.sign())
  1279. {
  1280. eval_increment(res.bits());
  1281. if(eval_msb(res.bits()) != cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::bit_count - 1 - shift)
  1282. {
  1283. // Must have extended result by one bit in the increment:
  1284. --shift;
  1285. ++res.exponent();
  1286. }
  1287. }
  1288. eval_left_shift(res.bits(), shift);
  1289. }
  1290. template <unsigned Digits, digit_base_type DigitBase, class Allocator, class Exponent, Exponent MinE, Exponent MaxE>
  1291. inline void eval_ceil(cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> &res, const cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> &arg)
  1292. {
  1293. using default_ops::eval_increment;
  1294. switch(arg.exponent())
  1295. {
  1296. case cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_zero:
  1297. case cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_nan:
  1298. case cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_infinity:
  1299. res = arg;
  1300. return;
  1301. }
  1302. typename mpl::if_c<sizeof(typename cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_type) < sizeof(int), int, typename cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_type>::type shift = (int)cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::bit_count - arg.exponent() - 1;
  1303. if((arg.exponent() > (int)cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::max_exponent) || (shift <= 0))
  1304. {
  1305. // Either arg is already an integer, or a special value:
  1306. res = arg;
  1307. return;
  1308. }
  1309. if(shift >= (int)cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::bit_count)
  1310. {
  1311. res = static_cast<signed_limb_type>(arg.sign() ? 0 : 1);
  1312. return;
  1313. }
  1314. bool fractional = (int)eval_lsb(arg.bits()) < shift;
  1315. res = arg;
  1316. eval_right_shift(res.bits(), shift);
  1317. if(fractional && !res.sign())
  1318. {
  1319. eval_increment(res.bits());
  1320. if(eval_msb(res.bits()) != cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::bit_count - 1 - shift)
  1321. {
  1322. // Must have extended result by one bit in the increment:
  1323. --shift;
  1324. ++res.exponent();
  1325. }
  1326. }
  1327. eval_left_shift(res.bits(), shift);
  1328. }
  1329. } // namespace backends
  1330. #ifdef BOOST_NO_SFINAE_EXPR
  1331. namespace detail{
  1332. template<unsigned D1, backends::digit_base_type B1, class A1, class E1, E1 M1, E1 M2, unsigned D2, backends::digit_base_type B2, class A2, class E2, E2 M3, E2 M4>
  1333. struct is_explicitly_convertible<backends::cpp_bin_float<D1, B1, A1, E1, M1, M2>, backends::cpp_bin_float<D2, B2, A2, E2, M3, M4> > : public mpl::true_ {};
  1334. template<class FloatT, unsigned D2, backends::digit_base_type B2, class A2, class E2, E2 M3, E2 M4>
  1335. struct is_explicitly_convertible<FloatT, backends::cpp_bin_float<D2, B2, A2, E2, M3, M4> > : public boost::is_floating_point<FloatT> {};
  1336. }
  1337. #endif
  1338. using backends::cpp_bin_float;
  1339. using backends::digit_base_2;
  1340. using backends::digit_base_10;
  1341. template<unsigned Digits, backends::digit_base_type DigitBase, class Exponent, Exponent MinE, Exponent MaxE, class Allocator>
  1342. struct number_category<cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> > : public boost::mpl::int_<boost::multiprecision::number_kind_floating_point>{};
  1343. template<unsigned Digits, backends::digit_base_type DigitBase, class Allocator, class Exponent, Exponent MinE, Exponent MaxE>
  1344. struct expression_template_default<cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> >
  1345. {
  1346. static const expression_template_option value = is_void<Allocator>::value ? et_off : et_on;
  1347. };
  1348. typedef number<backends::cpp_bin_float<50> > cpp_bin_float_50;
  1349. typedef number<backends::cpp_bin_float<100> > cpp_bin_float_100;
  1350. typedef number<backends::cpp_bin_float<24, backends::digit_base_2, void, boost::int16_t, -126, 127>, et_off> cpp_bin_float_single;
  1351. typedef number<backends::cpp_bin_float<53, backends::digit_base_2, void, boost::int16_t, -1022, 1023>, et_off> cpp_bin_float_double;
  1352. typedef number<backends::cpp_bin_float<64, backends::digit_base_2, void, boost::int16_t, -16382, 16383>, et_off> cpp_bin_float_double_extended;
  1353. typedef number<backends::cpp_bin_float<113, backends::digit_base_2, void, boost::int16_t, -16382, 16383>, et_off> cpp_bin_float_quad;
  1354. }} // namespaces
  1355. #include <boost/multiprecision/cpp_bin_float/io.hpp>
  1356. #include <boost/multiprecision/cpp_bin_float/transcendental.hpp>
  1357. namespace std{
  1358. //
  1359. // numeric_limits [partial] specializations for the types declared in this header:
  1360. //
  1361. template<unsigned Digits, boost::multiprecision::backends::digit_base_type DigitBase, class Allocator, class Exponent, Exponent MinE, Exponent MaxE, boost::multiprecision::expression_template_option ExpressionTemplates>
  1362. class numeric_limits<boost::multiprecision::number<boost::multiprecision::cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>, ExpressionTemplates> >
  1363. {
  1364. typedef boost::multiprecision::number<boost::multiprecision::cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>, ExpressionTemplates> number_type;
  1365. public:
  1366. BOOST_STATIC_CONSTEXPR bool is_specialized = true;
  1367. static number_type (min)()
  1368. {
  1369. initializer.do_nothing();
  1370. static std::pair<bool, number_type> value;
  1371. if(!value.first)
  1372. {
  1373. value.first = true;
  1374. value.second = 1u;
  1375. value.second.backend().exponent() = boost::multiprecision::cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::min_exponent;
  1376. }
  1377. return value.second;
  1378. }
  1379. static number_type (max)()
  1380. {
  1381. initializer.do_nothing();
  1382. static std::pair<bool, number_type> value;
  1383. if(!value.first)
  1384. {
  1385. value.first = true;
  1386. eval_complement(value.second.backend().bits(), value.second.backend().bits());
  1387. value.second.backend().exponent() = boost::multiprecision::cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::max_exponent;
  1388. }
  1389. return value.second;
  1390. }
  1391. BOOST_STATIC_CONSTEXPR number_type lowest()
  1392. {
  1393. return -(max)();
  1394. }
  1395. BOOST_STATIC_CONSTEXPR int digits = boost::multiprecision::cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::bit_count;
  1396. BOOST_STATIC_CONSTEXPR int digits10 = (digits - 1) * 301 / 1000;
  1397. // Is this really correct???
  1398. BOOST_STATIC_CONSTEXPR int max_digits10 = (digits * 301 / 1000) + 3;
  1399. BOOST_STATIC_CONSTEXPR bool is_signed = true;
  1400. BOOST_STATIC_CONSTEXPR bool is_integer = false;
  1401. BOOST_STATIC_CONSTEXPR bool is_exact = false;
  1402. BOOST_STATIC_CONSTEXPR int radix = 2;
  1403. static number_type epsilon()
  1404. {
  1405. initializer.do_nothing();
  1406. static std::pair<bool, number_type> value;
  1407. if(!value.first)
  1408. {
  1409. value.first = true;
  1410. value.second = 1;
  1411. value.second = ldexp(value.second, 1 - (int)digits);
  1412. }
  1413. return value.second;
  1414. }
  1415. // What value should this be????
  1416. static number_type round_error()
  1417. {
  1418. // returns 0.5
  1419. initializer.do_nothing();
  1420. static std::pair<bool, number_type> value;
  1421. if(!value.first)
  1422. {
  1423. value.first = true;
  1424. value.second = 1;
  1425. value.second = ldexp(value.second, -1);
  1426. }
  1427. return value.second;
  1428. }
  1429. BOOST_STATIC_CONSTEXPR typename boost::multiprecision::cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_type min_exponent = boost::multiprecision::cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::min_exponent;
  1430. BOOST_STATIC_CONSTEXPR typename boost::multiprecision::cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_type min_exponent10 = (min_exponent / 1000) * 301L;
  1431. BOOST_STATIC_CONSTEXPR typename boost::multiprecision::cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_type max_exponent = boost::multiprecision::cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::max_exponent;
  1432. BOOST_STATIC_CONSTEXPR typename boost::multiprecision::cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_type max_exponent10 = (max_exponent / 1000) * 301L;
  1433. BOOST_STATIC_CONSTEXPR bool has_infinity = true;
  1434. BOOST_STATIC_CONSTEXPR bool has_quiet_NaN = true;
  1435. BOOST_STATIC_CONSTEXPR bool has_signaling_NaN = false;
  1436. BOOST_STATIC_CONSTEXPR float_denorm_style has_denorm = denorm_absent;
  1437. BOOST_STATIC_CONSTEXPR bool has_denorm_loss = false;
  1438. static number_type infinity()
  1439. {
  1440. // returns epsilon/2
  1441. initializer.do_nothing();
  1442. static std::pair<bool, number_type> value;
  1443. if(!value.first)
  1444. {
  1445. value.first = true;
  1446. value.second.backend().exponent() = boost::multiprecision::cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_infinity;
  1447. }
  1448. return value.second;
  1449. }
  1450. static number_type quiet_NaN()
  1451. {
  1452. return number_type();
  1453. }
  1454. BOOST_STATIC_CONSTEXPR number_type signaling_NaN()
  1455. {
  1456. return number_type(0);
  1457. }
  1458. BOOST_STATIC_CONSTEXPR number_type denorm_min() { return number_type(0); }
  1459. BOOST_STATIC_CONSTEXPR bool is_iec559 = false;
  1460. BOOST_STATIC_CONSTEXPR bool is_bounded = true;
  1461. BOOST_STATIC_CONSTEXPR bool is_modulo = false;
  1462. BOOST_STATIC_CONSTEXPR bool traps = true;
  1463. BOOST_STATIC_CONSTEXPR bool tinyness_before = false;
  1464. BOOST_STATIC_CONSTEXPR float_round_style round_style = round_to_nearest;
  1465. private:
  1466. struct data_initializer
  1467. {
  1468. data_initializer()
  1469. {
  1470. std::numeric_limits<boost::multiprecision::number<boost::multiprecision::cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> > >::epsilon();
  1471. std::numeric_limits<boost::multiprecision::number<boost::multiprecision::cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> > >::round_error();
  1472. (std::numeric_limits<boost::multiprecision::number<boost::multiprecision::cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> > >::min)();
  1473. (std::numeric_limits<boost::multiprecision::number<boost::multiprecision::cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> > >::max)();
  1474. std::numeric_limits<boost::multiprecision::number<boost::multiprecision::cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> > >::infinity();
  1475. std::numeric_limits<boost::multiprecision::number<boost::multiprecision::cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE> > >::quiet_NaN();
  1476. }
  1477. void do_nothing()const{}
  1478. };
  1479. static const data_initializer initializer;
  1480. };
  1481. template<unsigned Digits, boost::multiprecision::backends::digit_base_type DigitBase, class Allocator, class Exponent, Exponent MinE, Exponent MaxE, boost::multiprecision::expression_template_option ExpressionTemplates>
  1482. const typename numeric_limits<boost::multiprecision::number<boost::multiprecision::cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>, ExpressionTemplates> >::data_initializer numeric_limits<boost::multiprecision::number<boost::multiprecision::cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>, ExpressionTemplates> >::initializer;
  1483. #ifndef BOOST_NO_INCLASS_MEMBER_INITIALIZATION
  1484. template <unsigned Digits, boost::multiprecision::backends::digit_base_type DigitBase, class Allocator, class Exponent, Exponent MinE, Exponent MaxE, boost::multiprecision::expression_template_option ExpressionTemplates>
  1485. BOOST_CONSTEXPR_OR_CONST int numeric_limits<boost::multiprecision::number<boost::multiprecision::cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>, ExpressionTemplates> >::digits;
  1486. template <unsigned Digits, boost::multiprecision::backends::digit_base_type DigitBase, class Allocator, class Exponent, Exponent MinE, Exponent MaxE, boost::multiprecision::expression_template_option ExpressionTemplates>
  1487. BOOST_CONSTEXPR_OR_CONST int numeric_limits<boost::multiprecision::number<boost::multiprecision::cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>, ExpressionTemplates> >::digits10;
  1488. template <unsigned Digits, boost::multiprecision::backends::digit_base_type DigitBase, class Allocator, class Exponent, Exponent MinE, Exponent MaxE, boost::multiprecision::expression_template_option ExpressionTemplates>
  1489. BOOST_CONSTEXPR_OR_CONST int numeric_limits<boost::multiprecision::number<boost::multiprecision::cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>, ExpressionTemplates> >::max_digits10;
  1490. template <unsigned Digits, boost::multiprecision::backends::digit_base_type DigitBase, class Allocator, class Exponent, Exponent MinE, Exponent MaxE, boost::multiprecision::expression_template_option ExpressionTemplates>
  1491. BOOST_CONSTEXPR_OR_CONST bool numeric_limits<boost::multiprecision::number<boost::multiprecision::cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>, ExpressionTemplates> >::is_signed;
  1492. template <unsigned Digits, boost::multiprecision::backends::digit_base_type DigitBase, class Allocator, class Exponent, Exponent MinE, Exponent MaxE, boost::multiprecision::expression_template_option ExpressionTemplates>
  1493. BOOST_CONSTEXPR_OR_CONST bool numeric_limits<boost::multiprecision::number<boost::multiprecision::cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>, ExpressionTemplates> >::is_integer;
  1494. template <unsigned Digits, boost::multiprecision::backends::digit_base_type DigitBase, class Allocator, class Exponent, Exponent MinE, Exponent MaxE, boost::multiprecision::expression_template_option ExpressionTemplates>
  1495. BOOST_CONSTEXPR_OR_CONST bool numeric_limits<boost::multiprecision::number<boost::multiprecision::cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>, ExpressionTemplates> >::is_exact;
  1496. template <unsigned Digits, boost::multiprecision::backends::digit_base_type DigitBase, class Allocator, class Exponent, Exponent MinE, Exponent MaxE, boost::multiprecision::expression_template_option ExpressionTemplates>
  1497. BOOST_CONSTEXPR_OR_CONST int numeric_limits<boost::multiprecision::number<boost::multiprecision::cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>, ExpressionTemplates> >::radix;
  1498. template <unsigned Digits, boost::multiprecision::backends::digit_base_type DigitBase, class Allocator, class Exponent, Exponent MinE, Exponent MaxE, boost::multiprecision::expression_template_option ExpressionTemplates>
  1499. BOOST_CONSTEXPR_OR_CONST typename boost::multiprecision::cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_type numeric_limits<boost::multiprecision::number<boost::multiprecision::cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>, ExpressionTemplates> >::min_exponent;
  1500. template <unsigned Digits, boost::multiprecision::backends::digit_base_type DigitBase, class Allocator, class Exponent, Exponent MinE, Exponent MaxE, boost::multiprecision::expression_template_option ExpressionTemplates>
  1501. BOOST_CONSTEXPR_OR_CONST typename boost::multiprecision::cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_type numeric_limits<boost::multiprecision::number<boost::multiprecision::cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>, ExpressionTemplates> >::min_exponent10;
  1502. template <unsigned Digits, boost::multiprecision::backends::digit_base_type DigitBase, class Allocator, class Exponent, Exponent MinE, Exponent MaxE, boost::multiprecision::expression_template_option ExpressionTemplates>
  1503. BOOST_CONSTEXPR_OR_CONST typename boost::multiprecision::cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_type numeric_limits<boost::multiprecision::number<boost::multiprecision::cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>, ExpressionTemplates> >::max_exponent;
  1504. template <unsigned Digits, boost::multiprecision::backends::digit_base_type DigitBase, class Allocator, class Exponent, Exponent MinE, Exponent MaxE, boost::multiprecision::expression_template_option ExpressionTemplates>
  1505. BOOST_CONSTEXPR_OR_CONST typename boost::multiprecision::cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>::exponent_type numeric_limits<boost::multiprecision::number<boost::multiprecision::cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>, ExpressionTemplates> >::max_exponent10;
  1506. template <unsigned Digits, boost::multiprecision::backends::digit_base_type DigitBase, class Allocator, class Exponent, Exponent MinE, Exponent MaxE, boost::multiprecision::expression_template_option ExpressionTemplates>
  1507. BOOST_CONSTEXPR_OR_CONST bool numeric_limits<boost::multiprecision::number<boost::multiprecision::cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>, ExpressionTemplates> >::has_infinity;
  1508. template <unsigned Digits, boost::multiprecision::backends::digit_base_type DigitBase, class Allocator, class Exponent, Exponent MinE, Exponent MaxE, boost::multiprecision::expression_template_option ExpressionTemplates>
  1509. BOOST_CONSTEXPR_OR_CONST bool numeric_limits<boost::multiprecision::number<boost::multiprecision::cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>, ExpressionTemplates> >::has_quiet_NaN;
  1510. template <unsigned Digits, boost::multiprecision::backends::digit_base_type DigitBase, class Allocator, class Exponent, Exponent MinE, Exponent MaxE, boost::multiprecision::expression_template_option ExpressionTemplates>
  1511. BOOST_CONSTEXPR_OR_CONST bool numeric_limits<boost::multiprecision::number<boost::multiprecision::cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>, ExpressionTemplates> >::has_signaling_NaN;
  1512. template <unsigned Digits, boost::multiprecision::backends::digit_base_type DigitBase, class Allocator, class Exponent, Exponent MinE, Exponent MaxE, boost::multiprecision::expression_template_option ExpressionTemplates>
  1513. BOOST_CONSTEXPR_OR_CONST float_denorm_style numeric_limits<boost::multiprecision::number<boost::multiprecision::cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>, ExpressionTemplates> >::has_denorm;
  1514. template <unsigned Digits, boost::multiprecision::backends::digit_base_type DigitBase, class Allocator, class Exponent, Exponent MinE, Exponent MaxE, boost::multiprecision::expression_template_option ExpressionTemplates>
  1515. BOOST_CONSTEXPR_OR_CONST bool numeric_limits<boost::multiprecision::number<boost::multiprecision::cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>, ExpressionTemplates> >::has_denorm_loss;
  1516. template <unsigned Digits, boost::multiprecision::backends::digit_base_type DigitBase, class Allocator, class Exponent, Exponent MinE, Exponent MaxE, boost::multiprecision::expression_template_option ExpressionTemplates>
  1517. BOOST_CONSTEXPR_OR_CONST bool numeric_limits<boost::multiprecision::number<boost::multiprecision::cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>, ExpressionTemplates> >::is_iec559;
  1518. template <unsigned Digits, boost::multiprecision::backends::digit_base_type DigitBase, class Allocator, class Exponent, Exponent MinE, Exponent MaxE, boost::multiprecision::expression_template_option ExpressionTemplates>
  1519. BOOST_CONSTEXPR_OR_CONST bool numeric_limits<boost::multiprecision::number<boost::multiprecision::cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>, ExpressionTemplates> >::is_bounded;
  1520. template <unsigned Digits, boost::multiprecision::backends::digit_base_type DigitBase, class Allocator, class Exponent, Exponent MinE, Exponent MaxE, boost::multiprecision::expression_template_option ExpressionTemplates>
  1521. BOOST_CONSTEXPR_OR_CONST bool numeric_limits<boost::multiprecision::number<boost::multiprecision::cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>, ExpressionTemplates> >::is_modulo;
  1522. template <unsigned Digits, boost::multiprecision::backends::digit_base_type DigitBase, class Allocator, class Exponent, Exponent MinE, Exponent MaxE, boost::multiprecision::expression_template_option ExpressionTemplates>
  1523. BOOST_CONSTEXPR_OR_CONST bool numeric_limits<boost::multiprecision::number<boost::multiprecision::cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>, ExpressionTemplates> >::traps;
  1524. template <unsigned Digits, boost::multiprecision::backends::digit_base_type DigitBase, class Allocator, class Exponent, Exponent MinE, Exponent MaxE, boost::multiprecision::expression_template_option ExpressionTemplates>
  1525. BOOST_CONSTEXPR_OR_CONST bool numeric_limits<boost::multiprecision::number<boost::multiprecision::cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>, ExpressionTemplates> >::tinyness_before;
  1526. template <unsigned Digits, boost::multiprecision::backends::digit_base_type DigitBase, class Allocator, class Exponent, Exponent MinE, Exponent MaxE, boost::multiprecision::expression_template_option ExpressionTemplates>
  1527. BOOST_CONSTEXPR_OR_CONST float_round_style numeric_limits<boost::multiprecision::number<boost::multiprecision::cpp_bin_float<Digits, DigitBase, Allocator, Exponent, MinE, MaxE>, ExpressionTemplates> >::round_style;
  1528. #endif
  1529. } // namespace std
  1530. #endif