et_ops.hpp 99 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515
  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_MP_ET_OPS_HPP
  6. #define BOOST_MP_ET_OPS_HPP
  7. namespace boost { namespace multiprecision {
  8. //
  9. // Non-member operators for number:
  10. //
  11. // Unary operators first.
  12. // Note that these *must* return by value, even though that's somewhat against
  13. // existing practice. The issue is that in C++11 land one could easily and legitimately
  14. // write:
  15. // auto x = +1234_my_user_defined_suffix;
  16. // which would result in a dangling-reference-to-temporary if unary + returned a reference
  17. // to it's argument. While return-by-value is obviously inefficient in other situations
  18. // the reality is that no one ever uses unary operator+ anyway...!
  19. //
  20. template <class B, expression_template_option ExpressionTemplates>
  21. inline BOOST_CONSTEXPR const number<B, ExpressionTemplates> operator+(const number<B, ExpressionTemplates>& v) { return v; }
  22. template <class tag, class Arg1, class Arg2, class Arg3, class Arg4>
  23. inline BOOST_CONSTEXPR const detail::expression<tag, Arg1, Arg2, Arg3, Arg4> operator+(const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& v) { return v; }
  24. template <class B>
  25. inline BOOST_MP_CXX14_CONSTEXPR detail::expression<detail::negate, number<B, et_on> > operator-(const number<B, et_on>& v)
  26. {
  27. BOOST_STATIC_ASSERT_MSG(is_signed_number<B>::value, "Negating an unsigned type results in ill-defined behavior.");
  28. return detail::expression<detail::negate, number<B, et_on> >(v);
  29. }
  30. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  31. template <class B>
  32. inline BOOST_MP_CXX14_CONSTEXPR number<B, et_on> operator-(number<B, et_on>&& v)
  33. {
  34. BOOST_STATIC_ASSERT_MSG(is_signed_number<B>::value, "Negating an unsigned type results in ill-defined behavior.");
  35. return detail::expression<detail::negate, number<B, et_on> >(v);
  36. }
  37. #endif
  38. template <class tag, class Arg1, class Arg2, class Arg3, class Arg4>
  39. inline BOOST_MP_CXX14_CONSTEXPR detail::expression<detail::negate, detail::expression<tag, Arg1, Arg2, Arg3, Arg4> > operator-(const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& v)
  40. {
  41. BOOST_STATIC_ASSERT_MSG((is_signed_number<typename detail::expression<tag, Arg1, Arg2, Arg3, Arg4>::result_type>::value), "Negating an unsigned type results in ill-defined behavior.");
  42. return detail::expression<detail::negate, detail::expression<tag, Arg1, Arg2, Arg3, Arg4> >(v);
  43. }
  44. template <class B>
  45. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<number_category<B>::value == number_kind_integer,
  46. detail::expression<detail::complement_immediates, number<B, et_on> > >::type
  47. operator~(const number<B, et_on>& v) { return detail::expression<detail::complement_immediates, number<B, et_on> >(v); }
  48. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  49. template <class B>
  50. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<number_category<B>::value == number_kind_integer,
  51. number<B, et_on> >::type
  52. operator~(number<B, et_on>&& v) { return detail::expression<detail::complement_immediates, number<B, et_on> >(v); }
  53. #endif
  54. template <class tag, class Arg1, class Arg2, class Arg3, class Arg4>
  55. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<number_category<typename detail::expression<tag, Arg1, Arg2, Arg3, Arg4>::result_type>::value == number_kind_integer,
  56. detail::expression<detail::bitwise_complement, detail::expression<tag, Arg1, Arg2, Arg3, Arg4> > >::type
  57. operator~(const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& v) { return detail::expression<detail::bitwise_complement, detail::expression<tag, Arg1, Arg2, Arg3, Arg4> >(v); }
  58. //
  59. // Then addition:
  60. //
  61. template <class B>
  62. inline BOOST_MP_CXX14_CONSTEXPR detail::expression<detail::add_immediates, number<B, et_on>, number<B, et_on> >
  63. operator+(const number<B, et_on>& a, const number<B, et_on>& b)
  64. {
  65. return detail::expression<detail::add_immediates, number<B, et_on>, number<B, et_on> >(a, b);
  66. }
  67. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  68. template <class B>
  69. inline BOOST_MP_CXX14_CONSTEXPR number<B, et_on>
  70. operator+(number<B, et_on>&& a, const number<B, et_on>& b)
  71. {
  72. return detail::expression<detail::add_immediates, number<B, et_on>, number<B, et_on> >(a, b);
  73. }
  74. template <class B>
  75. inline BOOST_MP_CXX14_CONSTEXPR number<B, et_on>
  76. operator+(const number<B, et_on>& a, number<B, et_on>&& b)
  77. {
  78. return detail::expression<detail::add_immediates, number<B, et_on>, number<B, et_on> >(a, b);
  79. }
  80. template <class B>
  81. inline BOOST_MP_CXX14_CONSTEXPR number<B, et_on>
  82. operator+(number<B, et_on>&& a, number<B, et_on>&& b)
  83. {
  84. return detail::expression<detail::add_immediates, number<B, et_on>, number<B, et_on> >(a, b);
  85. }
  86. #endif
  87. template <class B, class V>
  88. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if<is_compatible_arithmetic_type<V, number<B, et_on> >, detail::expression<detail::add_immediates, number<B, et_on>, V> >::type
  89. operator+(const number<B, et_on>& a, const V& b)
  90. {
  91. return detail::expression<detail::add_immediates, number<B, et_on>, V>(a, b);
  92. }
  93. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  94. template <class B, class V>
  95. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if<is_compatible_arithmetic_type<V, number<B, et_on> >, number<B, et_on> >::type
  96. operator+(number<B, et_on>&& a, const V& b)
  97. {
  98. return detail::expression<detail::add_immediates, number<B, et_on>, V>(a, b);
  99. }
  100. #endif
  101. template <class V, class B>
  102. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if<is_compatible_arithmetic_type<V, number<B, et_on> >, detail::expression<detail::add_immediates, V, number<B, et_on> > >::type
  103. operator+(const V& a, const number<B, et_on>& b)
  104. {
  105. return detail::expression<detail::add_immediates, V, number<B, et_on> >(a, b);
  106. }
  107. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  108. template <class V, class B>
  109. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if<is_compatible_arithmetic_type<V, number<B, et_on> >, number<B, et_on> >::type
  110. operator+(const V& a, number<B, et_on>&& b)
  111. {
  112. return detail::expression<detail::add_immediates, V, number<B, et_on> >(a, b);
  113. }
  114. #endif
  115. template <class B, expression_template_option ET, class tag, class Arg1, class Arg2, class Arg3, class Arg4>
  116. inline BOOST_MP_CXX14_CONSTEXPR detail::expression<detail::plus, number<B, ET>, detail::expression<tag, Arg1, Arg2, Arg3, Arg4> >
  117. operator+(const number<B, ET>& a, const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& b)
  118. {
  119. return detail::expression<detail::plus, number<B, ET>, detail::expression<tag, Arg1, Arg2, Arg3, Arg4> >(a, b);
  120. }
  121. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  122. template <class B, expression_template_option ET, class tag, class Arg1, class Arg2, class Arg3, class Arg4>
  123. inline BOOST_MP_CXX14_CONSTEXPR typename detail::expression<detail::plus, number<B, ET>, detail::expression<tag, Arg1, Arg2, Arg3, Arg4> >::result_type
  124. operator+(number<B, ET>&& a, const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& b)
  125. {
  126. return detail::expression<detail::plus, number<B, ET>, detail::expression<tag, Arg1, Arg2, Arg3, Arg4> >(a, b);
  127. }
  128. #endif
  129. template <class tag, class Arg1, class Arg2, class Arg3, class Arg4, class B, expression_template_option ET>
  130. inline BOOST_MP_CXX14_CONSTEXPR detail::expression<detail::plus, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, number<B, ET> >
  131. operator+(const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& a, const number<B, ET>& b)
  132. {
  133. return detail::expression<detail::plus, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, number<B, ET> >(a, b);
  134. }
  135. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  136. template <class tag, class Arg1, class Arg2, class Arg3, class Arg4, class B, expression_template_option ET>
  137. inline BOOST_MP_CXX14_CONSTEXPR typename detail::expression<detail::plus, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, number<B, ET> >::result_type
  138. operator+(const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& a, number<B, ET>&& b)
  139. {
  140. return detail::expression<detail::plus, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, number<B, ET> >(a, b);
  141. }
  142. #endif
  143. template <class tag, class Arg1, class Arg2, class Arg3, class Arg4, class tag2, class Arg1b, class Arg2b, class Arg3b, class Arg4b>
  144. inline BOOST_MP_CXX14_CONSTEXPR detail::expression<detail::plus, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, detail::expression<tag2, Arg1b, Arg2b, Arg3b, Arg4b> >
  145. operator+(const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& a, const detail::expression<tag2, Arg1b, Arg2b, Arg3b, Arg4b>& b)
  146. {
  147. return detail::expression<detail::plus, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, detail::expression<tag2, Arg1b, Arg2b, Arg3b, Arg4b> >(a, b);
  148. }
  149. template <class tag, class Arg1, class Arg2, class Arg3, class Arg4, class V>
  150. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if<is_compatible_arithmetic_type<V, typename detail::expression<tag, Arg1, Arg2, Arg3, Arg4>::result_type>, detail::expression<detail::plus, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, V> >::type
  151. operator+(const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& a, const V& b)
  152. {
  153. return detail::expression<detail::plus, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, V>(a, b);
  154. }
  155. template <class V, class tag, class Arg1, class Arg2, class Arg3, class Arg4>
  156. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if<is_compatible_arithmetic_type<V, typename detail::expression<tag, Arg1, Arg2, Arg3, Arg4>::result_type>, detail::expression<detail::plus, V, detail::expression<tag, Arg1, Arg2, Arg3, Arg4> > >::type
  157. operator+(const V& a, const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& b)
  158. {
  159. return detail::expression<detail::plus, V, detail::expression<tag, Arg1, Arg2, Arg3, Arg4> >(a, b);
  160. }
  161. //
  162. // Fused multiply add:
  163. //
  164. template <class V, class Arg1, class Arg2, class Arg3, class Arg4>
  165. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if<is_compatible_arithmetic_type<V, typename detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>::result_type>,
  166. detail::expression<detail::multiply_add, typename detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>::left_type, typename detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>::right_type, V> >::type
  167. operator+(const V& a, const detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>& b)
  168. {
  169. return detail::expression<detail::multiply_add, typename detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>::left_type, typename detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>::right_type, V>(b.left(), b.right(), a);
  170. }
  171. template <class Arg1, class Arg2, class Arg3, class Arg4, class V>
  172. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if<is_compatible_arithmetic_type<V, typename detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>::result_type>,
  173. detail::expression<detail::multiply_add, typename detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>::left_type, typename detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>::right_type, V> >::type
  174. operator+(const detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>& a, const V& b)
  175. {
  176. return detail::expression<detail::multiply_add, typename detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>::left_type, typename detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>::right_type, V>(a.left(), a.right(), b);
  177. }
  178. template <class B, expression_template_option ET, class Arg1, class Arg2, class Arg3, class Arg4>
  179. inline BOOST_MP_CXX14_CONSTEXPR detail::expression<detail::multiply_add, typename detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>::left_type, typename detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>::right_type, number<B, ET> >
  180. operator+(const number<B, ET>& a, const detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>& b)
  181. {
  182. return detail::expression<detail::multiply_add, typename detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>::left_type, typename detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>::right_type, number<B, ET> >(b.left(), b.right(), a);
  183. }
  184. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  185. template <class B, expression_template_option ET, class Arg1, class Arg2, class Arg3, class Arg4>
  186. inline BOOST_MP_CXX14_CONSTEXPR typename detail::expression<detail::multiply_add, typename detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>::left_type, typename detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>::right_type, number<B, ET> >::result_type
  187. operator+(number<B, ET>&& a, const detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>& b)
  188. {
  189. return detail::expression<detail::multiply_add, typename detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>::left_type, typename detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>::right_type, number<B, ET> >(b.left(), b.right(), a);
  190. }
  191. #endif
  192. template <class Arg1, class Arg2, class Arg3, class Arg4, class B, expression_template_option ET>
  193. inline BOOST_MP_CXX14_CONSTEXPR detail::expression<detail::multiply_add, typename detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>::left_type, typename detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>::right_type, number<B, ET> >
  194. operator+(const detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>& a, const number<B, ET>& b)
  195. {
  196. return detail::expression<detail::multiply_add, typename detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>::left_type, typename detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>::right_type, number<B, ET> >(a.left(), a.right(), b);
  197. }
  198. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  199. template <class Arg1, class Arg2, class Arg3, class Arg4, class B, expression_template_option ET>
  200. inline BOOST_MP_CXX14_CONSTEXPR typename detail::expression<detail::multiply_add, typename detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>::left_type, typename detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>::right_type, number<B, ET> >::result_type
  201. operator+(const detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>& a, number<B, ET>&& b)
  202. {
  203. return detail::expression<detail::multiply_add, typename detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>::left_type, typename detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>::right_type, number<B, ET> >(a.left(), a.right(), b);
  204. }
  205. #endif
  206. //
  207. // Fused multiply subtract:
  208. //
  209. template <class V, class Arg1, class Arg2, class Arg3, class Arg4>
  210. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if<is_compatible_arithmetic_type<V, typename detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>::result_type>,
  211. detail::expression<detail::negate, detail::expression<detail::multiply_subtract, typename detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>::left_type, typename detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>::right_type, V> > >::type
  212. operator-(const V& a, const detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>& b)
  213. {
  214. return detail::expression<detail::negate, detail::expression<detail::multiply_subtract, typename detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>::left_type, typename detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>::right_type, V> >(detail::expression<detail::multiply_subtract, typename detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>::left_type, typename detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>::right_type, V>(b.left(), b.right(), a));
  215. }
  216. template <class Arg1, class Arg2, class Arg3, class Arg4, class V>
  217. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if<is_compatible_arithmetic_type<V, typename detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>::result_type>,
  218. detail::expression<detail::multiply_subtract, typename detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>::left_type, typename detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>::right_type, V> >::type
  219. operator-(const detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>& a, const V& b)
  220. {
  221. return detail::expression<detail::multiply_subtract, typename detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>::left_type, typename detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>::right_type, V>(a.left(), a.right(), b);
  222. }
  223. template <class B, expression_template_option ET, class Arg1, class Arg2, class Arg3, class Arg4>
  224. inline BOOST_MP_CXX14_CONSTEXPR detail::expression<detail::negate, detail::expression<detail::multiply_subtract, typename detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>::left_type, typename detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>::right_type, number<B, ET> > >
  225. operator-(const number<B, ET>& a, const detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>& b)
  226. {
  227. return detail::expression<detail::negate, detail::expression<detail::multiply_subtract, typename detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>::left_type, typename detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>::right_type, number<B, ET> > >(detail::expression<detail::multiply_subtract, typename detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>::left_type, typename detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>::right_type, number<B, ET> >(b.left(), b.right(), a));
  228. }
  229. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  230. template <class B, expression_template_option ET, class Arg1, class Arg2, class Arg3, class Arg4>
  231. inline BOOST_MP_CXX14_CONSTEXPR typename detail::expression<detail::negate, detail::expression<detail::multiply_subtract, typename detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>::left_type, typename detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>::right_type, number<B, ET> > >::result_type
  232. operator-(number<B, ET>&& a, const detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>& b)
  233. {
  234. return detail::expression<detail::negate, detail::expression<detail::multiply_subtract, typename detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>::left_type, typename detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>::right_type, number<B, ET> > >(detail::expression<detail::multiply_subtract, typename detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>::left_type, typename detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>::right_type, number<B, ET> >(b.left(), b.right(), a));
  235. }
  236. #endif
  237. template <class Arg1, class Arg2, class Arg3, class Arg4, class B, expression_template_option ET>
  238. inline BOOST_MP_CXX14_CONSTEXPR detail::expression<detail::multiply_subtract, typename detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>::left_type, typename detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>::right_type, number<B, ET> >
  239. operator-(const detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>& a, const number<B, ET>& b)
  240. {
  241. return detail::expression<detail::multiply_subtract, typename detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>::left_type, typename detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>::right_type, number<B, ET> >(a.left(), a.right(), b);
  242. }
  243. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  244. template <class Arg1, class Arg2, class Arg3, class Arg4, class B, expression_template_option ET>
  245. inline BOOST_MP_CXX14_CONSTEXPR typename detail::expression<detail::multiply_subtract, typename detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>::left_type, typename detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>::right_type, number<B, ET> >::result_type
  246. operator-(const detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>& a, number<B, ET>&& b)
  247. {
  248. return detail::expression<detail::multiply_subtract, typename detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>::left_type, typename detail::expression<detail::multiply_immediates, Arg1, Arg2, Arg3, Arg4>::right_type, number<B, ET> >(a.left(), a.right(), b);
  249. }
  250. #endif
  251. //
  252. // Repeat operator for negated arguments: propagate the negation to the top level to avoid temporaries:
  253. //
  254. template <class B, expression_template_option ET, class Arg1, class Arg2, class Arg3, class Arg4>
  255. inline BOOST_MP_CXX14_CONSTEXPR detail::expression<detail::minus, number<B, ET>, Arg1>
  256. operator+(const number<B, ET>& a, const detail::expression<detail::negate, Arg1, Arg2, Arg3, Arg4>& b)
  257. {
  258. return detail::expression<detail::minus, number<B, ET>, Arg1>(a, b.left_ref());
  259. }
  260. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  261. template <class B, expression_template_option ET, class Arg1, class Arg2, class Arg3, class Arg4>
  262. inline BOOST_MP_CXX14_CONSTEXPR typename detail::expression<detail::minus, number<B, ET>, Arg1>::result_type
  263. operator+(number<B, ET>&& a, const detail::expression<detail::negate, Arg1, Arg2, Arg3, Arg4>& b)
  264. {
  265. return detail::expression<detail::minus, number<B, ET>, Arg1>(a, b.left_ref());
  266. }
  267. #endif
  268. template <class Arg1, class Arg2, class Arg3, class Arg4, class B, expression_template_option ET>
  269. inline BOOST_MP_CXX14_CONSTEXPR detail::expression<detail::minus, number<B, ET>, Arg1>
  270. operator+(const detail::expression<detail::negate, Arg1, Arg2, Arg3, Arg4>& a, const number<B, ET>& b)
  271. {
  272. return detail::expression<detail::minus, number<B, ET>, Arg1>(b, a.left_ref());
  273. }
  274. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  275. template <class Arg1, class Arg2, class Arg3, class Arg4, class B, expression_template_option ET>
  276. inline BOOST_MP_CXX14_CONSTEXPR typename detail::expression<detail::minus, number<B, ET>, Arg1>::result_type
  277. operator+(const detail::expression<detail::negate, Arg1, Arg2, Arg3, Arg4>& a, number<B, ET>&& b)
  278. {
  279. return detail::expression<detail::minus, number<B, ET>, Arg1>(b, a.left_ref());
  280. }
  281. #endif
  282. template <class B>
  283. inline BOOST_MP_CXX14_CONSTEXPR detail::expression<detail::subtract_immediates, number<B, et_on>, number<B, et_on> >
  284. operator+(const number<B, et_on>& a, const detail::expression<detail::negate, number<B, et_on> >& b)
  285. {
  286. return detail::expression<detail::subtract_immediates, number<B, et_on>, number<B, et_on> >(a, b.left_ref());
  287. }
  288. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  289. template <class B>
  290. inline BOOST_MP_CXX14_CONSTEXPR typename detail::expression<detail::subtract_immediates, number<B, et_on>, number<B, et_on> >::result_type
  291. operator+(number<B, et_on>&& a, const detail::expression<detail::negate, number<B, et_on> >& b)
  292. {
  293. return detail::expression<detail::subtract_immediates, number<B, et_on>, number<B, et_on> >(a, b.left_ref());
  294. }
  295. #endif
  296. template <class B>
  297. inline BOOST_MP_CXX14_CONSTEXPR detail::expression<detail::subtract_immediates, number<B, et_on>, number<B, et_on> >
  298. operator+(const detail::expression<detail::negate, number<B, et_on> >& a, const number<B, et_on>& b)
  299. {
  300. return detail::expression<detail::subtract_immediates, number<B, et_on>, number<B, et_on> >(b, a.left_ref());
  301. }
  302. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  303. template <class B>
  304. inline BOOST_MP_CXX14_CONSTEXPR typename detail::expression<detail::subtract_immediates, number<B, et_on>, number<B, et_on> >::result_type
  305. operator+(const detail::expression<detail::negate, number<B, et_on> >& a, number<B, et_on>&& b)
  306. {
  307. return detail::expression<detail::subtract_immediates, number<B, et_on>, number<B, et_on> >(b, a.left_ref());
  308. }
  309. #endif
  310. template <class B, class V>
  311. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if<is_compatible_arithmetic_type<V, number<B, et_on> >, detail::expression<detail::subtract_immediates, V, number<B, et_on> > >::type
  312. operator+(const detail::expression<detail::negate, number<B, et_on> >& a, const V& b)
  313. {
  314. return detail::expression<detail::subtract_immediates, V, number<B, et_on> >(b, a.left_ref());
  315. }
  316. template <class B, class B2, expression_template_option ET>
  317. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if<is_compatible_arithmetic_type<number<B2, ET>, number<B, et_on> >, detail::expression<detail::subtract_immediates, number<B2, ET>, number<B, et_on> > >::type
  318. operator+(const detail::expression<detail::negate, number<B, et_on> >& a, const number<B2, ET>& b)
  319. {
  320. return detail::expression<detail::subtract_immediates, number<B2, ET>, number<B, et_on> >(b, a.left_ref());
  321. }
  322. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  323. template <class B, class B2, expression_template_option ET>
  324. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if<is_compatible_arithmetic_type<number<B2, ET>, number<B, et_on> >, typename detail::expression<detail::subtract_immediates, number<B2, ET>, number<B, et_on> >::result_type>::type
  325. operator+(const detail::expression<detail::negate, number<B, et_on> >& a, number<B2, ET>&& b)
  326. {
  327. return detail::expression<detail::subtract_immediates, number<B2, ET>, number<B, et_on> >(b, a.left_ref());
  328. }
  329. #endif
  330. template <class B2, expression_template_option ET, class B>
  331. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if<is_compatible_arithmetic_type<number<B2, ET>, number<B, et_on> >, detail::expression<detail::subtract_immediates, number<B2, ET>, number<B, et_on> > >::type
  332. operator+(const number<B2, ET>& a, const detail::expression<detail::negate, number<B, et_on> >& b)
  333. {
  334. return detail::expression<detail::subtract_immediates, number<B2, ET>, number<B, et_on> >(a, b.left_ref());
  335. }
  336. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  337. template <class B2, expression_template_option ET, class B>
  338. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if<is_compatible_arithmetic_type<number<B2, ET>, number<B, et_on> >, typename detail::expression<detail::subtract_immediates, number<B2, ET>, number<B, et_on> >::result_type>::type
  339. operator+(number<B2, ET>&& a, const detail::expression<detail::negate, number<B, et_on> >& b)
  340. {
  341. return detail::expression<detail::subtract_immediates, number<B2, ET>, number<B, et_on> >(a, b.left_ref());
  342. }
  343. #endif
  344. template <class B>
  345. inline BOOST_MP_CXX14_CONSTEXPR detail::expression<detail::negate, detail::expression<detail::add_immediates, number<B, et_on>, number<B, et_on> > >
  346. operator+(const detail::expression<detail::negate, number<B, et_on> >& a, const detail::expression<detail::negate, number<B, et_on> >& b)
  347. {
  348. return detail::expression<detail::negate, detail::expression<detail::add_immediates, number<B, et_on>, number<B, et_on> > >(detail::expression<detail::add_immediates, number<B, et_on>, number<B, et_on> >(a.left_ref(), b.left_ref()));
  349. }
  350. //
  351. // Subtraction:
  352. //
  353. template <class B>
  354. inline BOOST_MP_CXX14_CONSTEXPR detail::expression<detail::subtract_immediates, number<B, et_on>, number<B, et_on> >
  355. operator-(const number<B, et_on>& a, const number<B, et_on>& b)
  356. {
  357. return detail::expression<detail::subtract_immediates, number<B, et_on>, number<B, et_on> >(a, b);
  358. }
  359. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  360. template <class B>
  361. inline BOOST_MP_CXX14_CONSTEXPR number<B, et_on>
  362. operator-(number<B, et_on>&& a, const number<B, et_on>& b)
  363. {
  364. return detail::expression<detail::subtract_immediates, number<B, et_on>, number<B, et_on> >(a, b);
  365. }
  366. template <class B>
  367. inline BOOST_MP_CXX14_CONSTEXPR number<B, et_on>
  368. operator-(const number<B, et_on>& a, number<B, et_on>&& b)
  369. {
  370. return detail::expression<detail::subtract_immediates, number<B, et_on>, number<B, et_on> >(a, b);
  371. }
  372. template <class B>
  373. inline BOOST_MP_CXX14_CONSTEXPR number<B, et_on>
  374. operator-(number<B, et_on>&& a, number<B, et_on>&& b)
  375. {
  376. return detail::expression<detail::subtract_immediates, number<B, et_on>, number<B, et_on> >(a, b);
  377. }
  378. #endif
  379. template <class B, class V>
  380. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if<is_compatible_arithmetic_type<V, number<B, et_on> >, detail::expression<detail::subtract_immediates, number<B, et_on>, V> >::type
  381. operator-(const number<B, et_on>& a, const V& b)
  382. {
  383. return detail::expression<detail::subtract_immediates, number<B, et_on>, V>(a, b);
  384. }
  385. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  386. template <class B, class V>
  387. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if<is_compatible_arithmetic_type<V, number<B, et_on> >, number<B, et_on> >::type
  388. operator-(number<B, et_on>&& a, const V& b)
  389. {
  390. return detail::expression<detail::subtract_immediates, number<B, et_on>, V>(a, b);
  391. }
  392. #endif
  393. template <class V, class B>
  394. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if<is_compatible_arithmetic_type<V, number<B, et_on> >, detail::expression<detail::subtract_immediates, V, number<B, et_on> > >::type
  395. operator-(const V& a, const number<B, et_on>& b)
  396. {
  397. return detail::expression<detail::subtract_immediates, V, number<B, et_on> >(a, b);
  398. }
  399. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  400. template <class V, class B>
  401. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if<is_compatible_arithmetic_type<V, number<B, et_on> >, number<B, et_on> >::type
  402. operator-(const V& a, number<B, et_on>&& b)
  403. {
  404. return detail::expression<detail::subtract_immediates, V, number<B, et_on> >(a, b);
  405. }
  406. #endif
  407. template <class B, expression_template_option ET, class tag, class Arg1, class Arg2, class Arg3, class Arg4>
  408. inline BOOST_MP_CXX14_CONSTEXPR detail::expression<detail::minus, number<B, ET>, detail::expression<tag, Arg1, Arg2, Arg3, Arg4> >
  409. operator-(const number<B, ET>& a, const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& b)
  410. {
  411. return detail::expression<detail::minus, number<B, ET>, detail::expression<tag, Arg1, Arg2, Arg3, Arg4> >(a, b);
  412. }
  413. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  414. template <class B, expression_template_option ET, class tag, class Arg1, class Arg2, class Arg3, class Arg4>
  415. inline BOOST_MP_CXX14_CONSTEXPR typename detail::expression<detail::minus, number<B, ET>, detail::expression<tag, Arg1, Arg2, Arg3, Arg4> >::result_type
  416. operator-(number<B, ET>&& a, const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& b)
  417. {
  418. return detail::expression<detail::minus, number<B, ET>, detail::expression<tag, Arg1, Arg2, Arg3, Arg4> >(a, b);
  419. }
  420. #endif
  421. template <class tag, class Arg1, class Arg2, class Arg3, class Arg4, class B, expression_template_option ET>
  422. inline BOOST_MP_CXX14_CONSTEXPR detail::expression<detail::minus, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, number<B, ET> >
  423. operator-(const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& a, const number<B, ET>& b)
  424. {
  425. return detail::expression<detail::minus, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, number<B, ET> >(a, b);
  426. }
  427. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  428. template <class tag, class Arg1, class Arg2, class Arg3, class Arg4, class B, expression_template_option ET>
  429. inline BOOST_MP_CXX14_CONSTEXPR typename detail::expression<detail::minus, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, number<B, ET> >::result_type
  430. operator-(const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& a, number<B, ET>&& b)
  431. {
  432. return detail::expression<detail::minus, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, number<B, ET> >(a, b);
  433. }
  434. #endif
  435. template <class tag, class Arg1, class Arg2, class Arg3, class Arg4, class tag2, class Arg1b, class Arg2b, class Arg3b, class Arg4b>
  436. inline BOOST_MP_CXX14_CONSTEXPR detail::expression<detail::minus, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, detail::expression<tag2, Arg1b, Arg2b, Arg3b, Arg4b> >
  437. operator-(const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& a, const detail::expression<tag2, Arg1b, Arg2b, Arg3b, Arg4b>& b)
  438. {
  439. return detail::expression<detail::minus, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, detail::expression<tag2, Arg1b, Arg2b, Arg3b, Arg4b> >(a, b);
  440. }
  441. template <class tag, class Arg1, class Arg2, class Arg3, class Arg4, class V>
  442. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if<is_compatible_arithmetic_type<V, typename detail::expression<tag, Arg1, Arg2, Arg3, Arg4>::result_type>, detail::expression<detail::minus, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, V> >::type
  443. operator-(const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& a, const V& b)
  444. {
  445. return detail::expression<detail::minus, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, V>(a, b);
  446. }
  447. template <class V, class tag, class Arg1, class Arg2, class Arg3, class Arg4>
  448. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if<is_compatible_arithmetic_type<V, typename detail::expression<tag, Arg1, Arg2, Arg3, Arg4>::result_type>, detail::expression<detail::minus, V, detail::expression<tag, Arg1, Arg2, Arg3, Arg4> > >::type
  449. operator-(const V& a, const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& b)
  450. {
  451. return detail::expression<detail::minus, V, detail::expression<tag, Arg1, Arg2, Arg3, Arg4> >(a, b);
  452. }
  453. //
  454. // Repeat operator for negated arguments: propagate the negation to the top level to avoid temporaries:
  455. //
  456. template <class B, expression_template_option ET, class Arg1, class Arg2, class Arg3, class Arg4>
  457. inline BOOST_MP_CXX14_CONSTEXPR detail::expression<detail::plus, number<B, ET>, Arg1>
  458. operator-(const number<B, ET>& a, const detail::expression<detail::negate, Arg1, Arg2, Arg3, Arg4>& b)
  459. {
  460. return detail::expression<detail::plus, number<B, ET>, Arg1>(a, b.left_ref());
  461. }
  462. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  463. template <class B, expression_template_option ET, class Arg1, class Arg2, class Arg3, class Arg4>
  464. inline BOOST_MP_CXX14_CONSTEXPR typename detail::expression<detail::plus, number<B, ET>, Arg1>::result_type
  465. operator-(number<B, ET>&& a, const detail::expression<detail::negate, Arg1, Arg2, Arg3, Arg4>& b)
  466. {
  467. return detail::expression<detail::plus, number<B, ET>, Arg1>(a, b.left_ref());
  468. }
  469. #endif
  470. template <class Arg1, class Arg2, class Arg3, class Arg4, class B, expression_template_option ET>
  471. inline BOOST_MP_CXX14_CONSTEXPR detail::expression<detail::negate, detail::expression<detail::plus, number<B, ET>, Arg1> >
  472. operator-(const detail::expression<detail::negate, Arg1, Arg2, Arg3, Arg4>& a, const number<B, ET>& b)
  473. {
  474. return detail::expression<detail::negate, detail::expression<detail::plus, number<B, ET>, Arg1> >(
  475. detail::expression<detail::plus, number<B, ET>, Arg1>(b, a.left_ref()));
  476. }
  477. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  478. template <class Arg1, class Arg2, class Arg3, class Arg4, class B, expression_template_option ET>
  479. inline BOOST_MP_CXX14_CONSTEXPR typename detail::expression<detail::negate, detail::expression<detail::plus, number<B, ET>, Arg1> >::result_type
  480. operator-(const detail::expression<detail::negate, Arg1, Arg2, Arg3, Arg4>& a, number<B, ET>&& b)
  481. {
  482. return detail::expression<detail::negate, detail::expression<detail::plus, number<B, ET>, Arg1> >(
  483. detail::expression<detail::plus, number<B, ET>, Arg1>(b, a.left_ref()));
  484. }
  485. #endif
  486. template <class B>
  487. inline BOOST_MP_CXX14_CONSTEXPR detail::expression<detail::add_immediates, number<B, et_on>, number<B, et_on> >
  488. operator-(const number<B, et_on>& a, const detail::expression<detail::negate, number<B, et_on> >& b)
  489. {
  490. return detail::expression<detail::add_immediates, number<B, et_on>, number<B, et_on> >(a, b.left_ref());
  491. }
  492. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  493. template <class B>
  494. inline BOOST_MP_CXX14_CONSTEXPR typename detail::expression<detail::add_immediates, number<B, et_on>, number<B, et_on> >::result_type
  495. operator-(number<B, et_on>&& a, const detail::expression<detail::negate, number<B, et_on> >& b)
  496. {
  497. return detail::expression<detail::add_immediates, number<B, et_on>, number<B, et_on> >(a, b.left_ref());
  498. }
  499. #endif
  500. template <class B>
  501. inline BOOST_MP_CXX14_CONSTEXPR detail::expression<detail::negate, detail::expression<detail::add_immediates, number<B, et_on>, number<B, et_on> > >
  502. operator-(const detail::expression<detail::negate, number<B, et_on> >& a, const number<B, et_on>& b)
  503. {
  504. return detail::expression<detail::negate, detail::expression<detail::add_immediates, number<B, et_on>, number<B, et_on> > >(
  505. detail::expression<detail::add_immediates, number<B, et_on>, number<B, et_on> >(b, a.left_ref()));
  506. }
  507. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  508. template <class B>
  509. inline BOOST_MP_CXX14_CONSTEXPR typename detail::expression<detail::negate, detail::expression<detail::add_immediates, number<B, et_on>, number<B, et_on> > >::result_type
  510. operator-(const detail::expression<detail::negate, number<B, et_on> >& a, number<B, et_on>&& b)
  511. {
  512. return detail::expression<detail::negate, detail::expression<detail::add_immediates, number<B, et_on>, number<B, et_on> > >(
  513. detail::expression<detail::add_immediates, number<B, et_on>, number<B, et_on> >(b, a.left_ref()));
  514. }
  515. #endif
  516. template <class B, class V>
  517. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if<is_compatible_arithmetic_type<V, number<B, et_on> >, detail::expression<detail::negate, detail::expression<detail::add_immediates, number<B, et_on>, V> > >::type
  518. operator-(const detail::expression<detail::negate, number<B, et_on> >& a, const V& b)
  519. {
  520. return detail::expression<detail::negate, detail::expression<detail::add_immediates, number<B, et_on>, V> >(detail::expression<detail::add_immediates, number<B, et_on>, V>(a.left_ref(), b));
  521. }
  522. template <class B, class B2, expression_template_option ET>
  523. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if<is_compatible_arithmetic_type<number<B2, ET>, number<B, et_on> >, detail::expression<detail::negate, detail::expression<detail::add_immediates, number<B, et_on>, number<B2, ET> > > >::type
  524. operator-(const detail::expression<detail::negate, number<B, et_on> >& a, const number<B2, ET>& b)
  525. {
  526. return detail::expression<detail::negate, detail::expression<detail::add_immediates, number<B, et_on>, number<B2, ET> > >(detail::expression<detail::add_immediates, number<B, et_on>, number<B2, ET> >(a.left_ref(), b));
  527. }
  528. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  529. template <class B, class B2, expression_template_option ET>
  530. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if<is_compatible_arithmetic_type<number<B2, ET>, number<B, et_on> >, typename detail::expression<detail::negate, detail::expression<detail::add_immediates, number<B, et_on>, number<B2, ET> > >::result_type>::type
  531. operator-(const detail::expression<detail::negate, number<B, et_on> >& a, number<B2, ET>&& b)
  532. {
  533. return detail::expression<detail::negate, detail::expression<detail::add_immediates, number<B, et_on>, number<B2, ET> > >(detail::expression<detail::add_immediates, number<B, et_on>, number<B2, ET> >(a.left_ref(), b));
  534. }
  535. #endif
  536. template <class V, class B>
  537. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if<is_compatible_arithmetic_type<V, number<B, et_on> >, detail::expression<detail::add_immediates, V, number<B, et_on> > >::type
  538. operator-(const V& a, const detail::expression<detail::negate, number<B, et_on> >& b)
  539. {
  540. return detail::expression<detail::add_immediates, V, number<B, et_on> >(a, b.left_ref());
  541. }
  542. template <class B2, expression_template_option ET, class B>
  543. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if<is_compatible_arithmetic_type<number<B2, ET>, number<B, et_on> >, detail::expression<detail::add_immediates, number<B2, ET>, number<B, et_on> > >::type
  544. operator-(const number<B2, ET>& a, const detail::expression<detail::negate, number<B, et_on> >& b)
  545. {
  546. return detail::expression<detail::add_immediates, number<B2, ET>, number<B, et_on> >(a, b.left_ref());
  547. }
  548. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  549. template <class B2, expression_template_option ET, class B>
  550. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if<is_compatible_arithmetic_type<number<B2, ET>, number<B, et_on> >, typename detail::expression<detail::add_immediates, number<B2, ET>, number<B, et_on> >::result_type>::type
  551. operator-(number<B2, ET>&& a, const detail::expression<detail::negate, number<B, et_on> >& b)
  552. {
  553. return detail::expression<detail::add_immediates, number<B2, ET>, number<B, et_on> >(a, b.left_ref());
  554. }
  555. #endif
  556. //
  557. // Multiplication:
  558. //
  559. template <class B>
  560. inline BOOST_MP_CXX14_CONSTEXPR detail::expression<detail::multiply_immediates, number<B, et_on>, number<B, et_on> >
  561. operator*(const number<B, et_on>& a, const number<B, et_on>& b)
  562. {
  563. return detail::expression<detail::multiply_immediates, number<B, et_on>, number<B, et_on> >(a, b);
  564. }
  565. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  566. template <class B>
  567. inline BOOST_MP_CXX14_CONSTEXPR number<B, et_on>
  568. operator*(number<B, et_on>&& a, const number<B, et_on>& b)
  569. {
  570. return detail::expression<detail::multiply_immediates, number<B, et_on>, number<B, et_on> >(a, b);
  571. }
  572. template <class B>
  573. inline BOOST_MP_CXX14_CONSTEXPR number<B, et_on>
  574. operator*(const number<B, et_on>& a, number<B, et_on>&& b)
  575. {
  576. return detail::expression<detail::multiply_immediates, number<B, et_on>, number<B, et_on> >(a, b);
  577. }
  578. template <class B>
  579. inline BOOST_MP_CXX14_CONSTEXPR number<B, et_on>
  580. operator*(number<B, et_on>&& a, number<B, et_on>&& b)
  581. {
  582. return detail::expression<detail::multiply_immediates, number<B, et_on>, number<B, et_on> >(a, b);
  583. }
  584. #endif
  585. template <class B, class V>
  586. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if<is_compatible_arithmetic_type<V, number<B, et_on> >, detail::expression<detail::multiply_immediates, number<B, et_on>, V> >::type
  587. operator*(const number<B, et_on>& a, const V& b)
  588. {
  589. return detail::expression<detail::multiply_immediates, number<B, et_on>, V>(a, b);
  590. }
  591. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  592. template <class B, class V>
  593. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if<is_compatible_arithmetic_type<V, number<B, et_on> >, number<B, et_on> >::type
  594. operator*(number<B, et_on>&& a, const V& b)
  595. {
  596. return detail::expression<detail::multiply_immediates, number<B, et_on>, V>(a, b);
  597. }
  598. #endif
  599. template <class V, class B>
  600. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if<is_compatible_arithmetic_type<V, number<B, et_on> >, detail::expression<detail::multiply_immediates, V, number<B, et_on> > >::type
  601. operator*(const V& a, const number<B, et_on>& b)
  602. {
  603. return detail::expression<detail::multiply_immediates, V, number<B, et_on> >(a, b);
  604. }
  605. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  606. template <class V, class B>
  607. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if<is_compatible_arithmetic_type<V, number<B, et_on> >, number<B, et_on> >::type
  608. operator*(const V& a, number<B, et_on>&& b)
  609. {
  610. return detail::expression<detail::multiply_immediates, V, number<B, et_on> >(a, b);
  611. }
  612. #endif
  613. template <class B, expression_template_option ET, class tag, class Arg1, class Arg2, class Arg3, class Arg4>
  614. inline BOOST_MP_CXX14_CONSTEXPR detail::expression<detail::multiplies, number<B, ET>, detail::expression<tag, Arg1, Arg2, Arg3, Arg4> >
  615. operator*(const number<B, ET>& a, const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& b)
  616. {
  617. return detail::expression<detail::multiplies, number<B, ET>, detail::expression<tag, Arg1, Arg2, Arg3, Arg4> >(a, b);
  618. }
  619. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  620. template <class B, expression_template_option ET, class tag, class Arg1, class Arg2, class Arg3, class Arg4>
  621. inline BOOST_MP_CXX14_CONSTEXPR typename detail::expression<detail::multiplies, number<B, ET>, detail::expression<tag, Arg1, Arg2, Arg3, Arg4> >::result_type
  622. operator*(number<B, ET>&& a, const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& b)
  623. {
  624. return detail::expression<detail::multiplies, number<B, ET>, detail::expression<tag, Arg1, Arg2, Arg3, Arg4> >(a, b);
  625. }
  626. #endif
  627. template <class tag, class Arg1, class Arg2, class Arg3, class Arg4, class B, expression_template_option ET>
  628. inline BOOST_MP_CXX14_CONSTEXPR detail::expression<detail::multiplies, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, number<B, ET> >
  629. operator*(const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& a, const number<B, ET>& b)
  630. {
  631. return detail::expression<detail::multiplies, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, number<B, ET> >(a, b);
  632. }
  633. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  634. template <class tag, class Arg1, class Arg2, class Arg3, class Arg4, class B, expression_template_option ET>
  635. inline BOOST_MP_CXX14_CONSTEXPR typename detail::expression<detail::multiplies, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, number<B, ET> >::result_type
  636. operator*(const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& a, number<B, ET>&& b)
  637. {
  638. return detail::expression<detail::multiplies, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, number<B, ET> >(a, b);
  639. }
  640. #endif
  641. template <class tag, class Arg1, class Arg2, class Arg3, class Arg4, class tag2, class Arg1b, class Arg2b, class Arg3b, class Arg4b>
  642. inline BOOST_MP_CXX14_CONSTEXPR detail::expression<detail::multiplies, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, detail::expression<tag2, Arg1b, Arg2b, Arg3b, Arg4b> >
  643. operator*(const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& a, const detail::expression<tag2, Arg1b, Arg2b, Arg3b, Arg4b>& b)
  644. {
  645. return detail::expression<detail::multiplies, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, detail::expression<tag2, Arg1b, Arg2b, Arg3b, Arg4b> >(a, b);
  646. }
  647. template <class tag, class Arg1, class Arg2, class Arg3, class Arg4, class V>
  648. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if<is_compatible_arithmetic_type<V, typename detail::expression<tag, Arg1, Arg2, Arg3, Arg4>::result_type>, detail::expression<detail::multiplies, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, V> >::type
  649. operator*(const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& a, const V& b)
  650. {
  651. return detail::expression<detail::multiplies, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, V>(a, b);
  652. }
  653. template <class V, class tag, class Arg1, class Arg2, class Arg3, class Arg4>
  654. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if<is_compatible_arithmetic_type<V, typename detail::expression<tag, Arg1, Arg2, Arg3, Arg4>::result_type>, detail::expression<detail::multiplies, V, detail::expression<tag, Arg1, Arg2, Arg3, Arg4> > >::type
  655. operator*(const V& a, const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& b)
  656. {
  657. return detail::expression<detail::multiplies, V, detail::expression<tag, Arg1, Arg2, Arg3, Arg4> >(a, b);
  658. }
  659. //
  660. // Repeat operator for negated arguments: propagate the negation to the top level to avoid temporaries:
  661. //
  662. template <class B, expression_template_option ET, class Arg1, class Arg2, class Arg3, class Arg4>
  663. inline BOOST_MP_CXX14_CONSTEXPR detail::expression<detail::negate, detail::expression<detail::multiplies, number<B, ET>, Arg1> >
  664. operator*(const number<B, ET>& a, const detail::expression<detail::negate, Arg1, Arg2, Arg3, Arg4>& b)
  665. {
  666. return detail::expression<detail::negate, detail::expression<detail::multiplies, number<B, ET>, Arg1> >(
  667. detail::expression<detail::multiplies, number<B, ET>, Arg1>(a, b.left_ref()));
  668. }
  669. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  670. template <class B, expression_template_option ET, class Arg1, class Arg2, class Arg3, class Arg4>
  671. inline BOOST_MP_CXX14_CONSTEXPR typename detail::expression<detail::negate, detail::expression<detail::multiplies, number<B, ET>, Arg1> >::result_type
  672. operator*(number<B, ET>&& a, const detail::expression<detail::negate, Arg1, Arg2, Arg3, Arg4>& b)
  673. {
  674. return detail::expression<detail::negate, detail::expression<detail::multiplies, number<B, ET>, Arg1> >(
  675. detail::expression<detail::multiplies, number<B, ET>, Arg1>(a, b.left_ref()));
  676. }
  677. #endif
  678. template <class Arg1, class Arg2, class Arg3, class Arg4, class B, expression_template_option ET>
  679. inline BOOST_MP_CXX14_CONSTEXPR detail::expression<detail::negate, detail::expression<detail::multiplies, number<B, ET>, Arg1> >
  680. operator*(const detail::expression<detail::negate, Arg1, Arg2, Arg3, Arg4>& a, const number<B, ET>& b)
  681. {
  682. return detail::expression<detail::negate, detail::expression<detail::multiplies, number<B, ET>, Arg1> >(
  683. detail::expression<detail::multiplies, number<B, ET>, Arg1>(b, a.left_ref()));
  684. }
  685. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  686. template <class Arg1, class Arg2, class Arg3, class Arg4, class B, expression_template_option ET>
  687. inline BOOST_MP_CXX14_CONSTEXPR typename detail::expression<detail::negate, detail::expression<detail::multiplies, number<B, ET>, Arg1> >::result_type
  688. operator*(const detail::expression<detail::negate, Arg1, Arg2, Arg3, Arg4>& a, number<B, ET>&& b)
  689. {
  690. return detail::expression<detail::negate, detail::expression<detail::multiplies, number<B, ET>, Arg1> >(
  691. detail::expression<detail::multiplies, number<B, ET>, Arg1>(b, a.left_ref()));
  692. }
  693. #endif
  694. template <class B>
  695. inline BOOST_MP_CXX14_CONSTEXPR detail::expression<detail::negate, detail::expression<detail::multiply_immediates, number<B, et_on>, number<B, et_on> > >
  696. operator*(const number<B, et_on>& a, const detail::expression<detail::negate, number<B, et_on> >& b)
  697. {
  698. return detail::expression<detail::negate, detail::expression<detail::multiply_immediates, number<B, et_on>, number<B, et_on> > >(
  699. detail::expression<detail::multiply_immediates, number<B, et_on>, number<B, et_on> >(a, b.left_ref()));
  700. }
  701. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  702. template <class B>
  703. inline BOOST_MP_CXX14_CONSTEXPR typename detail::expression<detail::negate, detail::expression<detail::multiply_immediates, number<B, et_on>, number<B, et_on> > >::result_type
  704. operator*(number<B, et_on>&& a, const detail::expression<detail::negate, number<B, et_on> >& b)
  705. {
  706. return detail::expression<detail::negate, detail::expression<detail::multiply_immediates, number<B, et_on>, number<B, et_on> > >(
  707. detail::expression<detail::multiply_immediates, number<B, et_on>, number<B, et_on> >(a, b.left_ref()));
  708. }
  709. #endif
  710. template <class B>
  711. inline BOOST_MP_CXX14_CONSTEXPR detail::expression<detail::negate, detail::expression<detail::multiply_immediates, number<B, et_on>, number<B, et_on> > >
  712. operator*(const detail::expression<detail::negate, number<B, et_on> >& a, const number<B, et_on>& b)
  713. {
  714. return detail::expression<detail::negate, detail::expression<detail::multiply_immediates, number<B, et_on>, number<B, et_on> > >(
  715. detail::expression<detail::multiply_immediates, number<B, et_on>, number<B, et_on> >(b, a.left_ref()));
  716. }
  717. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  718. template <class B>
  719. inline BOOST_MP_CXX14_CONSTEXPR typename detail::expression<detail::negate, detail::expression<detail::multiply_immediates, number<B, et_on>, number<B, et_on> > >::result_type
  720. operator*(const detail::expression<detail::negate, number<B, et_on> >& a, number<B, et_on>&& b)
  721. {
  722. return detail::expression<detail::negate, detail::expression<detail::multiply_immediates, number<B, et_on>, number<B, et_on> > >(
  723. detail::expression<detail::multiply_immediates, number<B, et_on>, number<B, et_on> >(b, a.left_ref()));
  724. }
  725. #endif
  726. template <class B, class V>
  727. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if<is_compatible_arithmetic_type<V, number<B, et_on> >, detail::expression<detail::negate, detail::expression<detail::multiply_immediates, number<B, et_on>, V> > >::type
  728. operator*(const detail::expression<detail::negate, number<B, et_on> >& a, const V& b)
  729. {
  730. return detail::expression<detail::negate, detail::expression<detail::multiply_immediates, number<B, et_on>, V> >(
  731. detail::expression<detail::multiply_immediates, number<B, et_on>, V>(a.left_ref(), b));
  732. }
  733. template <class B, class B2, expression_template_option ET>
  734. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if<is_compatible_arithmetic_type<number<B2, ET>, number<B, et_on> >, detail::expression<detail::negate, detail::expression<detail::multiply_immediates, number<B, et_on>, number<B2, ET> > > >::type
  735. operator*(const detail::expression<detail::negate, number<B, et_on> >& a, const number<B2, ET>& b)
  736. {
  737. return detail::expression<detail::negate, detail::expression<detail::multiply_immediates, number<B, et_on>, number<B2, ET> > >(
  738. detail::expression<detail::multiply_immediates, number<B, et_on>, number<B2, ET> >(a.left_ref(), b));
  739. }
  740. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  741. template <class B, class B2, expression_template_option ET>
  742. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if<is_compatible_arithmetic_type<number<B2, ET>, number<B, et_on> >, typename detail::expression<detail::negate, detail::expression<detail::multiply_immediates, number<B, et_on>, number<B2, ET> > >::result_type >::type
  743. operator*(const detail::expression<detail::negate, number<B, et_on> >& a, number<B2, ET>&& b)
  744. {
  745. return detail::expression<detail::negate, detail::expression<detail::multiply_immediates, number<B, et_on>, number<B2, ET> > >(
  746. detail::expression<detail::multiply_immediates, number<B, et_on>, number<B2, ET> >(a.left_ref(), b));
  747. }
  748. #endif
  749. template <class V, class B>
  750. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if<is_compatible_arithmetic_type<V, number<B, et_on> >, detail::expression<detail::negate, detail::expression<detail::multiply_immediates, number<B, et_on>, V> > >::type
  751. operator*(const V& a, const detail::expression<detail::negate, number<B, et_on> >& b)
  752. {
  753. return detail::expression<detail::negate, detail::expression<detail::multiply_immediates, number<B, et_on>, V> >(
  754. detail::expression<detail::multiply_immediates, number<B, et_on>, V>(b.left_ref(), a));
  755. }
  756. template <class B2, expression_template_option ET, class B>
  757. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if<is_compatible_arithmetic_type<number<B2, ET>, number<B, et_on> >, detail::expression<detail::negate, detail::expression<detail::multiply_immediates, number<B, et_on>, number<B2, ET> > > >::type
  758. operator*(const number<B2, ET>& a, const detail::expression<detail::negate, number<B, et_on> >& b)
  759. {
  760. return detail::expression<detail::negate, detail::expression<detail::multiply_immediates, number<B, et_on>, number<B2, ET> > >(
  761. detail::expression<detail::multiply_immediates, number<B, et_on>, number<B2, ET> >(b.left_ref(), a));
  762. }
  763. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  764. template <class B2, expression_template_option ET, class B>
  765. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if<is_compatible_arithmetic_type<number<B2, ET>, number<B, et_on> >, typename detail::expression<detail::negate, detail::expression<detail::multiply_immediates, number<B, et_on>, number<B2, ET> > >::result_type>::type
  766. operator*(number<B2, ET>&& a, const detail::expression<detail::negate, number<B, et_on> >& b)
  767. {
  768. return detail::expression<detail::negate, detail::expression<detail::multiply_immediates, number<B, et_on>, number<B2, ET> > >(
  769. detail::expression<detail::multiply_immediates, number<B, et_on>, number<B2, ET> >(b.left_ref(), a));
  770. }
  771. #endif
  772. //
  773. // Division:
  774. //
  775. template <class B>
  776. inline BOOST_MP_CXX14_CONSTEXPR detail::expression<detail::divide_immediates, number<B, et_on>, number<B, et_on> >
  777. operator/(const number<B, et_on>& a, const number<B, et_on>& b)
  778. {
  779. return detail::expression<detail::divide_immediates, number<B, et_on>, number<B, et_on> >(a, b);
  780. }
  781. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  782. template <class B>
  783. inline BOOST_MP_CXX14_CONSTEXPR number<B, et_on>
  784. operator/(number<B, et_on>&& a, const number<B, et_on>& b)
  785. {
  786. return detail::expression<detail::divide_immediates, number<B, et_on>, number<B, et_on> >(a, b);
  787. }
  788. template <class B>
  789. inline BOOST_MP_CXX14_CONSTEXPR number<B, et_on>
  790. operator/(const number<B, et_on>& a, number<B, et_on>&& b)
  791. {
  792. return detail::expression<detail::divide_immediates, number<B, et_on>, number<B, et_on> >(a, b);
  793. }
  794. template <class B>
  795. inline BOOST_MP_CXX14_CONSTEXPR number<B, et_on>
  796. operator/(number<B, et_on>&& a, number<B, et_on>&& b)
  797. {
  798. return detail::expression<detail::divide_immediates, number<B, et_on>, number<B, et_on> >(a, b);
  799. }
  800. #endif
  801. template <class B, class V>
  802. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if<is_compatible_arithmetic_type<V, number<B, et_on> >, detail::expression<detail::divide_immediates, number<B, et_on>, V> >::type
  803. operator/(const number<B, et_on>& a, const V& b)
  804. {
  805. return detail::expression<detail::divide_immediates, number<B, et_on>, V>(a, b);
  806. }
  807. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  808. template <class B, class V>
  809. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if<is_compatible_arithmetic_type<V, number<B, et_on> >, number<B, et_on> >::type
  810. operator/(number<B, et_on>&& a, const V& b)
  811. {
  812. return detail::expression<detail::divide_immediates, number<B, et_on>, V>(a, b);
  813. }
  814. #endif
  815. template <class V, class B>
  816. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if<is_compatible_arithmetic_type<V, number<B, et_on> >, detail::expression<detail::divide_immediates, V, number<B, et_on> > >::type
  817. operator/(const V& a, const number<B, et_on>& b)
  818. {
  819. return detail::expression<detail::divide_immediates, V, number<B, et_on> >(a, b);
  820. }
  821. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  822. template <class V, class B>
  823. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if<is_compatible_arithmetic_type<V, number<B, et_on> >, number<B, et_on> >::type
  824. operator/(const V& a, number<B, et_on>&& b)
  825. {
  826. return detail::expression<detail::divide_immediates, V, number<B, et_on> >(a, b);
  827. }
  828. #endif
  829. template <class B, expression_template_option ET, class tag, class Arg1, class Arg2, class Arg3, class Arg4>
  830. inline BOOST_MP_CXX14_CONSTEXPR detail::expression<detail::divides, number<B, ET>, detail::expression<tag, Arg1, Arg2, Arg3, Arg4> >
  831. operator/(const number<B, ET>& a, const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& b)
  832. {
  833. return detail::expression<detail::divides, number<B, ET>, detail::expression<tag, Arg1, Arg2, Arg3, Arg4> >(a, b);
  834. }
  835. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  836. template <class B, expression_template_option ET, class tag, class Arg1, class Arg2, class Arg3, class Arg4>
  837. inline BOOST_MP_CXX14_CONSTEXPR typename detail::expression<detail::divides, number<B, ET>, detail::expression<tag, Arg1, Arg2, Arg3, Arg4> >::result_type
  838. operator/(number<B, ET>&& a, const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& b)
  839. {
  840. return detail::expression<detail::divides, number<B, ET>, detail::expression<tag, Arg1, Arg2, Arg3, Arg4> >(a, b);
  841. }
  842. #endif
  843. template <class tag, class Arg1, class Arg2, class Arg3, class Arg4, class B, expression_template_option ET>
  844. inline BOOST_MP_CXX14_CONSTEXPR detail::expression<detail::divides, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, number<B, ET> >
  845. operator/(const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& a, const number<B, ET>& b)
  846. {
  847. return detail::expression<detail::divides, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, number<B, ET> >(a, b);
  848. }
  849. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  850. template <class tag, class Arg1, class Arg2, class Arg3, class Arg4, class B, expression_template_option ET>
  851. inline BOOST_MP_CXX14_CONSTEXPR typename detail::expression<detail::divides, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, number<B, ET> >::result_type
  852. operator/(const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& a, number<B, ET>&& b)
  853. {
  854. return detail::expression<detail::divides, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, number<B, ET> >(a, b);
  855. }
  856. #endif
  857. template <class tag, class Arg1, class Arg2, class Arg3, class Arg4, class tag2, class Arg1b, class Arg2b, class Arg3b, class Arg4b>
  858. inline BOOST_MP_CXX14_CONSTEXPR detail::expression<detail::divides, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, detail::expression<tag2, Arg1b, Arg2b, Arg3b, Arg4b> >
  859. operator/(const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& a, const detail::expression<tag2, Arg1b, Arg2b, Arg3b, Arg4b>& b)
  860. {
  861. return detail::expression<detail::divides, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, detail::expression<tag2, Arg1b, Arg2b, Arg3b, Arg4b> >(a, b);
  862. }
  863. template <class tag, class Arg1, class Arg2, class Arg3, class Arg4, class V>
  864. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if<is_compatible_arithmetic_type<V, typename detail::expression<tag, Arg1, Arg2, Arg3, Arg4>::result_type>, detail::expression<detail::divides, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, V> >::type
  865. operator/(const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& a, const V& b)
  866. {
  867. return detail::expression<detail::divides, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, V>(a, b);
  868. }
  869. template <class V, class tag, class Arg1, class Arg2, class Arg3, class Arg4>
  870. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if<is_compatible_arithmetic_type<V, typename detail::expression<tag, Arg1, Arg2, Arg3, Arg4>::result_type>, detail::expression<detail::divides, V, detail::expression<tag, Arg1, Arg2, Arg3, Arg4> > >::type
  871. operator/(const V& a, const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& b)
  872. {
  873. return detail::expression<detail::divides, V, detail::expression<tag, Arg1, Arg2, Arg3, Arg4> >(a, b);
  874. }
  875. //
  876. // Repeat operator for negated arguments: propagate the negation to the top level to avoid temporaries:
  877. //
  878. template <class B, expression_template_option ET, class Arg1, class Arg2, class Arg3, class Arg4>
  879. inline BOOST_MP_CXX14_CONSTEXPR detail::expression<detail::negate, detail::expression<detail::divides, number<B, ET>, Arg1> >
  880. operator/(const number<B, ET>& a, const detail::expression<detail::negate, Arg1, Arg2, Arg3, Arg4>& b)
  881. {
  882. return detail::expression<detail::negate, detail::expression<detail::divides, number<B, ET>, Arg1> >(
  883. detail::expression<detail::divides, number<B, ET>, Arg1>(a, b.left_ref()));
  884. }
  885. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  886. template <class B, expression_template_option ET, class Arg1, class Arg2, class Arg3, class Arg4>
  887. inline typename detail::expression<detail::negate, detail::expression<detail::divides, number<B, ET>, Arg1> >::result_type
  888. operator/(number<B, ET>&& a, const detail::expression<detail::negate, Arg1, Arg2, Arg3, Arg4>& b)
  889. {
  890. return detail::expression<detail::negate, detail::expression<detail::divides, number<B, ET>, Arg1> >(
  891. detail::expression<detail::divides, number<B, ET>, Arg1>(a, b.left_ref()));
  892. }
  893. #endif
  894. template <class Arg1, class Arg2, class Arg3, class Arg4, class B, expression_template_option ET>
  895. inline BOOST_MP_CXX14_CONSTEXPR detail::expression<detail::negate, detail::expression<detail::divides, Arg1, number<B, ET> > >
  896. operator/(const detail::expression<detail::negate, Arg1, Arg2, Arg3, Arg4>& a, const number<B, ET>& b)
  897. {
  898. return detail::expression<detail::negate, detail::expression<detail::divides, Arg1, number<B, ET> > >(
  899. detail::expression<detail::divides, Arg1, number<B, ET> >(a.left_ref(), b));
  900. }
  901. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  902. template <class Arg1, class Arg2, class Arg3, class Arg4, class B, expression_template_option ET>
  903. inline BOOST_MP_CXX14_CONSTEXPR typename detail::expression<detail::negate, detail::expression<detail::divides, Arg1, number<B, ET> > >::result_type
  904. operator/(const detail::expression<detail::negate, Arg1, Arg2, Arg3, Arg4>& a, number<B, ET>&& b)
  905. {
  906. return detail::expression<detail::negate, detail::expression<detail::divides, Arg1, number<B, ET> > >(
  907. detail::expression<detail::divides, Arg1, number<B, ET> >(a.left_ref(), b));
  908. }
  909. #endif
  910. template <class B>
  911. inline BOOST_MP_CXX14_CONSTEXPR detail::expression<detail::negate, detail::expression<detail::divide_immediates, number<B, et_on>, number<B, et_on> > >
  912. operator/(const number<B, et_on>& a, const detail::expression<detail::negate, number<B, et_on> >& b)
  913. {
  914. return detail::expression<detail::negate, detail::expression<detail::divide_immediates, number<B, et_on>, number<B, et_on> > >(
  915. detail::expression<detail::divide_immediates, number<B, et_on>, number<B, et_on> >(a, b.left_ref()));
  916. }
  917. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  918. template <class B>
  919. inline BOOST_MP_CXX14_CONSTEXPR typename detail::expression<detail::negate, detail::expression<detail::divide_immediates, number<B, et_on>, number<B, et_on> > >::result_type
  920. operator/(number<B, et_on>&& a, const detail::expression<detail::negate, number<B, et_on> >& b)
  921. {
  922. return detail::expression<detail::negate, detail::expression<detail::divide_immediates, number<B, et_on>, number<B, et_on> > >(
  923. detail::expression<detail::divide_immediates, number<B, et_on>, number<B, et_on> >(a, b.left_ref()));
  924. }
  925. #endif
  926. template <class B>
  927. inline BOOST_MP_CXX14_CONSTEXPR detail::expression<detail::negate, detail::expression<detail::divide_immediates, number<B, et_on>, number<B, et_on> > >
  928. operator/(const detail::expression<detail::negate, number<B, et_on> >& a, const number<B, et_on>& b)
  929. {
  930. return detail::expression<detail::negate, detail::expression<detail::divide_immediates, number<B, et_on>, number<B, et_on> > >(
  931. detail::expression<detail::divide_immediates, number<B, et_on>, number<B, et_on> >(a.left_ref(), b));
  932. }
  933. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  934. template <class B>
  935. inline BOOST_MP_CXX14_CONSTEXPR typename detail::expression<detail::negate, detail::expression<detail::divide_immediates, number<B, et_on>, number<B, et_on> > >::result_type
  936. operator/(const detail::expression<detail::negate, number<B, et_on> >& a, number<B, et_on>&& b)
  937. {
  938. return detail::expression<detail::negate, detail::expression<detail::divide_immediates, number<B, et_on>, number<B, et_on> > >(
  939. detail::expression<detail::divide_immediates, number<B, et_on>, number<B, et_on> >(a.left_ref(), b));
  940. }
  941. #endif
  942. template <class B, class V>
  943. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if<is_compatible_arithmetic_type<V, number<B, et_on> >, detail::expression<detail::negate, detail::expression<detail::divide_immediates, number<B, et_on>, V> > >::type
  944. operator/(const detail::expression<detail::negate, number<B, et_on> >& a, const V& b)
  945. {
  946. return detail::expression<detail::negate, detail::expression<detail::divide_immediates, number<B, et_on>, V> >(
  947. detail::expression<detail::divide_immediates, number<B, et_on>, V>(a.left_ref(), b));
  948. }
  949. template <class B, class B2, expression_template_option ET>
  950. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if<is_compatible_arithmetic_type<number<B2, ET>, number<B, et_on> >, detail::expression<detail::negate, detail::expression<detail::divide_immediates, number<B, et_on>, number<B2, ET> > > >::type
  951. operator/(const detail::expression<detail::negate, number<B, et_on> >& a, const number<B2, ET>& b)
  952. {
  953. return detail::expression<detail::negate, detail::expression<detail::divide_immediates, number<B, et_on>, number<B2, ET> > >(
  954. detail::expression<detail::divide_immediates, number<B, et_on>, number<B2, ET> >(a.left_ref(), b));
  955. }
  956. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  957. template <class B, class B2, expression_template_option ET>
  958. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if<is_compatible_arithmetic_type<number<B2, ET>, number<B, et_on> >, typename detail::expression<detail::negate, detail::expression<detail::divide_immediates, number<B, et_on>, number<B2, ET> > >::result_type>::type
  959. operator/(const detail::expression<detail::negate, number<B, et_on> >& a, number<B2, ET>&& b)
  960. {
  961. return detail::expression<detail::negate, detail::expression<detail::divide_immediates, number<B, et_on>, number<B2, ET> > >(
  962. detail::expression<detail::divide_immediates, number<B, et_on>, number<B2, ET> >(a.left_ref(), b));
  963. }
  964. #endif
  965. template <class V, class B>
  966. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if<is_compatible_arithmetic_type<V, number<B, et_on> >, detail::expression<detail::negate, detail::expression<detail::divide_immediates, V, number<B, et_on> > > >::type
  967. operator/(const V& a, const detail::expression<detail::negate, number<B, et_on> >& b)
  968. {
  969. return detail::expression<detail::negate, detail::expression<detail::divide_immediates, V, number<B, et_on> > >(
  970. detail::expression<detail::divide_immediates, V, number<B, et_on> >(a, b.left_ref()));
  971. }
  972. template <class B2, expression_template_option ET, class B>
  973. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if<is_compatible_arithmetic_type<number<B2, ET>, number<B, et_on> >, detail::expression<detail::negate, detail::expression<detail::divide_immediates, number<B2, ET>, number<B, et_on> > > >::type
  974. operator/(const number<B2, ET>& a, const detail::expression<detail::negate, number<B, et_on> >& b)
  975. {
  976. return detail::expression<detail::negate, detail::expression<detail::divide_immediates, number<B2, ET>, number<B, et_on> > >(
  977. detail::expression<detail::divide_immediates, number<B2, ET>, number<B, et_on> >(a, b.left_ref()));
  978. }
  979. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  980. template <class B2, expression_template_option ET, class B>
  981. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if<is_compatible_arithmetic_type<number<B2, ET>, typename detail::expression<detail::negate, detail::expression<detail::divide_immediates, number<B2, ET>, number<B, et_on> > >::result_type >, number<B, et_on> >::type
  982. operator/(number<B2, ET>&& a, const detail::expression<detail::negate, number<B, et_on> >& b)
  983. {
  984. return detail::expression<detail::negate, detail::expression<detail::divide_immediates, number<B2, ET>, number<B, et_on> > >(
  985. detail::expression<detail::divide_immediates, number<B2, ET>, number<B, et_on> >(a, b.left_ref()));
  986. }
  987. #endif
  988. //
  989. // Modulus:
  990. //
  991. template <class B>
  992. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<number_category<B>::value == number_kind_integer,
  993. detail::expression<detail::modulus_immediates, number<B, et_on>, number<B, et_on> > >::type
  994. operator%(const number<B, et_on>& a, const number<B, et_on>& b)
  995. {
  996. return detail::expression<detail::modulus_immediates, number<B, et_on>, number<B, et_on> >(a, b);
  997. }
  998. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  999. template <class B>
  1000. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<number_category<B>::value == number_kind_integer,
  1001. number<B, et_on> >::type
  1002. operator%(number<B, et_on>&& a, const number<B, et_on>& b)
  1003. {
  1004. return detail::expression<detail::modulus_immediates, number<B, et_on>, number<B, et_on> >(a, b);
  1005. }
  1006. template <class B>
  1007. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<number_category<B>::value == number_kind_integer,
  1008. number<B, et_on> >::type
  1009. operator%(const number<B, et_on>& a, number<B, et_on>&& b)
  1010. {
  1011. return detail::expression<detail::modulus_immediates, number<B, et_on>, number<B, et_on> >(a, b);
  1012. }
  1013. template <class B>
  1014. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<number_category<B>::value == number_kind_integer,
  1015. number<B, et_on> >::type
  1016. operator%(number<B, et_on>&& a, number<B, et_on>&& b)
  1017. {
  1018. return detail::expression<detail::modulus_immediates, number<B, et_on>, number<B, et_on> >(a, b);
  1019. }
  1020. #endif
  1021. template <class B, class V>
  1022. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<is_compatible_arithmetic_type<V, number<B, et_on> >::value && (number_category<B>::value == number_kind_integer),
  1023. detail::expression<detail::modulus_immediates, number<B, et_on>, V> >::type
  1024. operator%(const number<B, et_on>& a, const V& b)
  1025. {
  1026. return detail::expression<detail::modulus_immediates, number<B, et_on>, V>(a, b);
  1027. }
  1028. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  1029. template <class B, class V>
  1030. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<is_compatible_arithmetic_type<V, number<B, et_on> >::value && (number_category<B>::value == number_kind_integer),
  1031. number<B, et_on> >::type
  1032. operator%(number<B, et_on>&& a, const V& b)
  1033. {
  1034. return detail::expression<detail::modulus_immediates, number<B, et_on>, V>(a, b);
  1035. }
  1036. #endif
  1037. template <class V, class B>
  1038. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<is_compatible_arithmetic_type<V, number<B, et_on> >::value && (number_category<B>::value == number_kind_integer),
  1039. detail::expression<detail::modulus_immediates, V, number<B, et_on> > >::type
  1040. operator%(const V& a, const number<B, et_on>& b)
  1041. {
  1042. return detail::expression<detail::modulus_immediates, V, number<B, et_on> >(a, b);
  1043. }
  1044. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  1045. template <class V, class B>
  1046. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<is_compatible_arithmetic_type<V, number<B, et_on> >::value && (number_category<B>::value == number_kind_integer),
  1047. number<B, et_on> >::type
  1048. operator%(const V& a, number<B, et_on>&& b)
  1049. {
  1050. return detail::expression<detail::modulus_immediates, V, number<B, et_on> >(a, b);
  1051. }
  1052. #endif
  1053. template <class B, class tag, class Arg1, class Arg2, class Arg3, class Arg4>
  1054. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<number_category<B>::value == number_kind_integer,
  1055. detail::expression<detail::modulus, number<B, et_on>, detail::expression<tag, Arg1, Arg2, Arg3, Arg4> > >::type
  1056. operator%(const number<B, et_on>& a, const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& b)
  1057. {
  1058. return detail::expression<detail::modulus, number<B, et_on>, detail::expression<tag, Arg1, Arg2, Arg3, Arg4> >(a, b);
  1059. }
  1060. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  1061. template <class B, class tag, class Arg1, class Arg2, class Arg3, class Arg4>
  1062. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<number_category<B>::value == number_kind_integer,
  1063. typename detail::expression<detail::modulus, number<B, et_on>, detail::expression<tag, Arg1, Arg2, Arg3, Arg4> >::result_type >::type
  1064. operator%(number<B, et_on>&& a, const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& b)
  1065. {
  1066. return detail::expression<detail::modulus, number<B, et_on>, detail::expression<tag, Arg1, Arg2, Arg3, Arg4> >(a, b);
  1067. }
  1068. #endif
  1069. template <class tag, class Arg1, class Arg2, class Arg3, class Arg4, class B>
  1070. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<number_category<B>::value == number_kind_integer,
  1071. detail::expression<detail::modulus, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, number<B, et_on> > >::type
  1072. operator%(const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& a, const number<B, et_on>& b)
  1073. {
  1074. return detail::expression<detail::modulus, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, number<B, et_on> >(a, b);
  1075. }
  1076. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  1077. template <class tag, class Arg1, class Arg2, class Arg3, class Arg4, class B>
  1078. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<number_category<B>::value == number_kind_integer,
  1079. typename detail::expression<detail::modulus, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, number<B, et_on> >::result_type >::type
  1080. operator%(const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& a, number<B, et_on>&& b)
  1081. {
  1082. return detail::expression<detail::modulus, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, number<B, et_on> >(a, b);
  1083. }
  1084. #endif
  1085. template <class tag, class Arg1, class Arg2, class Arg3, class Arg4, class tag2, class Arg1b, class Arg2b, class Arg3b, class Arg4b>
  1086. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<number_category<typename detail::expression<tag, Arg1, Arg2, Arg3, Arg4>::result_type>::value == number_kind_integer,
  1087. detail::expression<detail::modulus, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, detail::expression<tag2, Arg1b, Arg2b, Arg3b, Arg4b> > >::type
  1088. operator%(const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& a, const detail::expression<tag2, Arg1b, Arg2b, Arg3b, Arg4b>& b)
  1089. {
  1090. return detail::expression<detail::modulus, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, detail::expression<tag2, Arg1b, Arg2b, Arg3b, Arg4b> >(a, b);
  1091. }
  1092. template <class tag, class Arg1, class Arg2, class Arg3, class Arg4, class V>
  1093. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<is_compatible_arithmetic_type<V, typename detail::expression<tag, Arg1, Arg2, Arg3, Arg4>::result_type>::value && (number_category<typename detail::expression<tag, Arg1, Arg2, Arg3, Arg4>::result_type>::value == number_kind_integer),
  1094. detail::expression<detail::modulus, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, V> >::type
  1095. operator%(const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& a, const V& b)
  1096. {
  1097. return detail::expression<detail::modulus, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, V>(a, b);
  1098. }
  1099. template <class V, class tag, class Arg1, class Arg2, class Arg3, class Arg4>
  1100. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<is_compatible_arithmetic_type<V, typename detail::expression<tag, Arg1, Arg2, Arg3, Arg4>::result_type>::value && (number_category<typename detail::expression<tag, Arg1, Arg2, Arg3, Arg4>::result_type>::value == number_kind_integer),
  1101. detail::expression<detail::modulus, V, detail::expression<tag, Arg1, Arg2, Arg3, Arg4> > >::type
  1102. operator%(const V& a, const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& b)
  1103. {
  1104. return detail::expression<detail::modulus, V, detail::expression<tag, Arg1, Arg2, Arg3, Arg4> >(a, b);
  1105. }
  1106. //
  1107. // Left shift:
  1108. //
  1109. template <class B, class I>
  1110. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<is_integral<I>::value && (number_category<B>::value == number_kind_integer), detail::expression<detail::shift_left, number<B, et_on>, I> >::type
  1111. operator<<(const number<B, et_on>& a, const I& b)
  1112. {
  1113. return detail::expression<detail::shift_left, number<B, et_on>, I>(a, b);
  1114. }
  1115. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  1116. template <class B, class I>
  1117. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<is_integral<I>::value && (number_category<B>::value == number_kind_integer), number<B, et_on> >::type
  1118. operator<<(number<B, et_on>&& a, const I& b)
  1119. {
  1120. return detail::expression<detail::shift_left, number<B, et_on>, I>(a, b);
  1121. }
  1122. #endif
  1123. template <class tag, class Arg1, class Arg2, class Arg3, class Arg4, class I>
  1124. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<is_integral<I>::value && (number_category<typename detail::expression<tag, Arg1, Arg2, Arg3, Arg4>::result_type>::value == number_kind_integer),
  1125. detail::expression<detail::shift_left, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, I> >::type
  1126. operator<<(const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& a, const I& b)
  1127. {
  1128. return detail::expression<detail::shift_left, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, I>(a, b);
  1129. }
  1130. //
  1131. // Right shift:
  1132. //
  1133. template <class B, class I>
  1134. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<is_integral<I>::value && (number_category<B>::value == number_kind_integer),
  1135. detail::expression<detail::shift_right, number<B, et_on>, I> >::type
  1136. operator>>(const number<B, et_on>& a, const I& b)
  1137. {
  1138. return detail::expression<detail::shift_right, number<B, et_on>, I>(a, b);
  1139. }
  1140. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  1141. template <class B, class I>
  1142. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<is_integral<I>::value && (number_category<B>::value == number_kind_integer),
  1143. number<B, et_on> >::type
  1144. operator>>(number<B, et_on>&& a, const I& b)
  1145. {
  1146. return detail::expression<detail::shift_right, number<B, et_on>, I>(a, b);
  1147. }
  1148. #endif
  1149. template <class tag, class Arg1, class Arg2, class Arg3, class Arg4, class I>
  1150. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<is_integral<I>::value && (number_category<typename detail::expression<tag, Arg1, Arg2, Arg3, Arg4>::result_type>::value == number_kind_integer),
  1151. detail::expression<detail::shift_right, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, I> >::type
  1152. operator>>(const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& a, const I& b)
  1153. {
  1154. return detail::expression<detail::shift_right, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, I>(a, b);
  1155. }
  1156. //
  1157. // Bitwise AND:
  1158. //
  1159. template <class B>
  1160. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<number_category<B>::value == number_kind_integer,
  1161. detail::expression<detail::bitwise_and_immediates, number<B, et_on>, number<B, et_on> > >::type
  1162. operator&(const number<B, et_on>& a, const number<B, et_on>& b)
  1163. {
  1164. return detail::expression<detail::bitwise_and_immediates, number<B, et_on>, number<B, et_on> >(a, b);
  1165. }
  1166. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  1167. template <class B>
  1168. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<number_category<B>::value == number_kind_integer,
  1169. number<B, et_on> >::type
  1170. operator&(number<B, et_on>&& a, const number<B, et_on>& b)
  1171. {
  1172. return detail::expression<detail::bitwise_and_immediates, number<B, et_on>, number<B, et_on> >(a, b);
  1173. }
  1174. template <class B>
  1175. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<number_category<B>::value == number_kind_integer,
  1176. number<B, et_on> >::type
  1177. operator&(const number<B, et_on>& a, number<B, et_on>&& b)
  1178. {
  1179. return detail::expression<detail::bitwise_and_immediates, number<B, et_on>, number<B, et_on> >(a, b);
  1180. }
  1181. template <class B>
  1182. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<number_category<B>::value == number_kind_integer,
  1183. number<B, et_on> >::type
  1184. operator&(number<B, et_on>&& a, number<B, et_on>&& b)
  1185. {
  1186. return detail::expression<detail::bitwise_and_immediates, number<B, et_on>, number<B, et_on> >(a, b);
  1187. }
  1188. #endif
  1189. template <class B, class V>
  1190. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<is_compatible_arithmetic_type<V, number<B, et_on> >::value && (number_category<B>::value == number_kind_integer),
  1191. detail::expression<detail::bitwise_and_immediates, number<B, et_on>, V> >::type
  1192. operator&(const number<B, et_on>& a, const V& b)
  1193. {
  1194. return detail::expression<detail::bitwise_and_immediates, number<B, et_on>, V>(a, b);
  1195. }
  1196. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  1197. template <class B, class V>
  1198. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<is_compatible_arithmetic_type<V, number<B, et_on> >::value && (number_category<B>::value == number_kind_integer),
  1199. number<B, et_on> >::type
  1200. operator&(number<B, et_on>&& a, const V& b)
  1201. {
  1202. return detail::expression<detail::bitwise_and_immediates, number<B, et_on>, V>(a, b);
  1203. }
  1204. #endif
  1205. template <class V, class B>
  1206. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<is_compatible_arithmetic_type<V, number<B, et_on> >::value && (number_category<B>::value == number_kind_integer),
  1207. detail::expression<detail::bitwise_and_immediates, V, number<B, et_on> > >::type
  1208. operator&(const V& a, const number<B, et_on>& b)
  1209. {
  1210. return detail::expression<detail::bitwise_and_immediates, V, number<B, et_on> >(a, b);
  1211. }
  1212. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  1213. template <class V, class B>
  1214. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<is_compatible_arithmetic_type<V, number<B, et_on> >::value && (number_category<B>::value == number_kind_integer),
  1215. number<B, et_on> >::type
  1216. operator&(const V& a, number<B, et_on>&& b)
  1217. {
  1218. return detail::expression<detail::bitwise_and_immediates, V, number<B, et_on> >(a, b);
  1219. }
  1220. #endif
  1221. template <class B, class tag, class Arg1, class Arg2, class Arg3, class Arg4>
  1222. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<number_category<B>::value == number_kind_integer,
  1223. detail::expression<detail::bitwise_and, number<B, et_on>, detail::expression<tag, Arg1, Arg2, Arg3, Arg4> > >::type
  1224. operator&(const number<B, et_on>& a, const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& b)
  1225. {
  1226. return detail::expression<detail::bitwise_and, number<B, et_on>, detail::expression<tag, Arg1, Arg2, Arg3, Arg4> >(a, b);
  1227. }
  1228. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  1229. template <class B, class tag, class Arg1, class Arg2, class Arg3, class Arg4>
  1230. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<number_category<B>::value == number_kind_integer,
  1231. typename detail::expression<detail::bitwise_and, number<B, et_on>, detail::expression<tag, Arg1, Arg2, Arg3, Arg4> >::result_type >::type
  1232. operator&(number<B, et_on>&& a, const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& b)
  1233. {
  1234. return detail::expression<detail::bitwise_and, number<B, et_on>, detail::expression<tag, Arg1, Arg2, Arg3, Arg4> >(a, b);
  1235. }
  1236. #endif
  1237. template <class tag, class Arg1, class Arg2, class Arg3, class Arg4, class B>
  1238. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<number_category<B>::value == number_kind_integer,
  1239. detail::expression<detail::bitwise_and, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, number<B, et_on> > >::type
  1240. operator&(const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& a, const number<B, et_on>& b)
  1241. {
  1242. return detail::expression<detail::bitwise_and, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, number<B, et_on> >(a, b);
  1243. }
  1244. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  1245. template <class tag, class Arg1, class Arg2, class Arg3, class Arg4, class B>
  1246. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<number_category<B>::value == number_kind_integer,
  1247. typename detail::expression<detail::bitwise_and, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, number<B, et_on> >::result_type >::type
  1248. operator&(const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& a, number<B, et_on>&& b)
  1249. {
  1250. return detail::expression<detail::bitwise_and, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, number<B, et_on> >(a, b);
  1251. }
  1252. #endif
  1253. template <class tag, class Arg1, class Arg2, class Arg3, class Arg4, class tag2, class Arg1b, class Arg2b, class Arg3b, class Arg4b>
  1254. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<number_category<typename detail::expression<tag, Arg1, Arg2, Arg3, Arg4>::result_type>::value == number_kind_integer,
  1255. detail::expression<detail::bitwise_and, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, detail::expression<tag2, Arg1b, Arg2b, Arg3b, Arg4b> > >::type
  1256. operator&(const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& a, const detail::expression<tag2, Arg1b, Arg2b, Arg3b, Arg4b>& b)
  1257. {
  1258. return detail::expression<detail::bitwise_and, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, detail::expression<tag2, Arg1b, Arg2b, Arg3b, Arg4b> >(a, b);
  1259. }
  1260. template <class tag, class Arg1, class Arg2, class Arg3, class Arg4, class V>
  1261. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<is_compatible_arithmetic_type<V, typename detail::expression<tag, Arg1, Arg2, Arg3, Arg4>::result_type>::value && (number_category<typename detail::expression<tag, Arg1, Arg2, Arg3, Arg4>::result_type>::value == number_kind_integer),
  1262. detail::expression<detail::bitwise_and, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, V> >::type
  1263. operator&(const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& a, const V& b)
  1264. {
  1265. return detail::expression<detail::bitwise_and, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, V>(a, b);
  1266. }
  1267. template <class V, class tag, class Arg1, class Arg2, class Arg3, class Arg4>
  1268. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<is_compatible_arithmetic_type<V, typename detail::expression<tag, Arg1, Arg2, Arg3, Arg4>::result_type>::value && (number_category<typename detail::expression<tag, Arg1, Arg2, Arg3, Arg4>::result_type>::value == number_kind_integer),
  1269. detail::expression<detail::bitwise_and, V, detail::expression<tag, Arg1, Arg2, Arg3, Arg4> > >::type
  1270. operator&(const V& a, const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& b)
  1271. {
  1272. return detail::expression<detail::bitwise_and, V, detail::expression<tag, Arg1, Arg2, Arg3, Arg4> >(a, b);
  1273. }
  1274. //
  1275. // Bitwise OR:
  1276. //
  1277. template <class B>
  1278. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<number_category<B>::value == number_kind_integer,
  1279. detail::expression<detail::bitwise_or_immediates, number<B, et_on>, number<B, et_on> > >::type
  1280. operator|(const number<B, et_on>& a, const number<B, et_on>& b)
  1281. {
  1282. return detail::expression<detail::bitwise_or_immediates, number<B, et_on>, number<B, et_on> >(a, b);
  1283. }
  1284. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  1285. template <class B>
  1286. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<number_category<B>::value == number_kind_integer,
  1287. number<B, et_on> >::type
  1288. operator|(number<B, et_on>&& a, const number<B, et_on>& b)
  1289. {
  1290. return detail::expression<detail::bitwise_or_immediates, number<B, et_on>, number<B, et_on> >(a, b);
  1291. }
  1292. template <class B>
  1293. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<number_category<B>::value == number_kind_integer,
  1294. number<B, et_on> >::type
  1295. operator|(const number<B, et_on>& a, number<B, et_on>&& b)
  1296. {
  1297. return detail::expression<detail::bitwise_or_immediates, number<B, et_on>, number<B, et_on> >(a, b);
  1298. }
  1299. template <class B>
  1300. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<number_category<B>::value == number_kind_integer,
  1301. number<B, et_on> >::type
  1302. operator|(number<B, et_on>&& a, number<B, et_on>&& b)
  1303. {
  1304. return detail::expression<detail::bitwise_or_immediates, number<B, et_on>, number<B, et_on> >(a, b);
  1305. }
  1306. #endif
  1307. template <class B, class V>
  1308. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<is_compatible_arithmetic_type<V, number<B, et_on> >::value && (number_category<B>::value == number_kind_integer),
  1309. detail::expression<detail::bitwise_or_immediates, number<B, et_on>, V> >::type
  1310. operator|(const number<B, et_on>& a, const V& b)
  1311. {
  1312. return detail::expression<detail::bitwise_or_immediates, number<B, et_on>, V>(a, b);
  1313. }
  1314. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  1315. template <class B, class V>
  1316. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<is_compatible_arithmetic_type<V, number<B, et_on> >::value && (number_category<B>::value == number_kind_integer),
  1317. number<B, et_on> >::type
  1318. operator|(number<B, et_on>&& a, const V& b)
  1319. {
  1320. return detail::expression<detail::bitwise_or_immediates, number<B, et_on>, V>(a, b);
  1321. }
  1322. #endif
  1323. template <class V, class B>
  1324. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<is_compatible_arithmetic_type<V, number<B, et_on> >::value && (number_category<B>::value == number_kind_integer),
  1325. detail::expression<detail::bitwise_or_immediates, V, number<B, et_on> > >::type
  1326. operator|(const V& a, const number<B, et_on>& b)
  1327. {
  1328. return detail::expression<detail::bitwise_or_immediates, V, number<B, et_on> >(a, b);
  1329. }
  1330. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  1331. template <class V, class B>
  1332. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<is_compatible_arithmetic_type<V, number<B, et_on> >::value && (number_category<B>::value == number_kind_integer),
  1333. number<B, et_on> >::type
  1334. operator|(const V& a, number<B, et_on>&& b)
  1335. {
  1336. return detail::expression<detail::bitwise_or_immediates, V, number<B, et_on> >(a, b);
  1337. }
  1338. #endif
  1339. template <class B, class tag, class Arg1, class Arg2, class Arg3, class Arg4>
  1340. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<number_category<B>::value == number_kind_integer,
  1341. detail::expression<detail::bitwise_or, number<B, et_on>, detail::expression<tag, Arg1, Arg2, Arg3, Arg4> > >::type
  1342. operator|(const number<B, et_on>& a, const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& b)
  1343. {
  1344. return detail::expression<detail::bitwise_or, number<B, et_on>, detail::expression<tag, Arg1, Arg2, Arg3, Arg4> >(a, b);
  1345. }
  1346. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  1347. template <class B, class tag, class Arg1, class Arg2, class Arg3, class Arg4>
  1348. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<number_category<B>::value == number_kind_integer,
  1349. typename detail::expression<detail::bitwise_or, number<B, et_on>, detail::expression<tag, Arg1, Arg2, Arg3, Arg4> >::result_type>::type
  1350. operator|(number<B, et_on>&& a, const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& b)
  1351. {
  1352. return detail::expression<detail::bitwise_or, number<B, et_on>, detail::expression<tag, Arg1, Arg2, Arg3, Arg4> >(a, b);
  1353. }
  1354. #endif
  1355. template <class tag, class Arg1, class Arg2, class Arg3, class Arg4, class B>
  1356. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<number_category<B>::value == number_kind_integer,
  1357. detail::expression<detail::bitwise_or, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, number<B, et_on> > >::type
  1358. operator|(const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& a, const number<B, et_on>& b)
  1359. {
  1360. return detail::expression<detail::bitwise_or, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, number<B, et_on> >(a, b);
  1361. }
  1362. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  1363. template <class tag, class Arg1, class Arg2, class Arg3, class Arg4, class B>
  1364. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<number_category<B>::value == number_kind_integer,
  1365. typename detail::expression<detail::bitwise_or, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, number<B, et_on> >::result_type>::type
  1366. operator|(const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& a, number<B, et_on>&& b)
  1367. {
  1368. return detail::expression<detail::bitwise_or, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, number<B, et_on> >(a, b);
  1369. }
  1370. #endif
  1371. template <class tag, class Arg1, class Arg2, class Arg3, class Arg4, class tag2, class Arg1b, class Arg2b, class Arg3b, class Arg4b>
  1372. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<number_category<typename detail::expression<tag, Arg1, Arg2, Arg3, Arg4>::result_type>::value == number_kind_integer,
  1373. detail::expression<detail::bitwise_or, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, detail::expression<tag2, Arg1b, Arg2b, Arg3b, Arg4b> > >::type
  1374. operator|(const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& a, const detail::expression<tag2, Arg1b, Arg2b, Arg3b, Arg4b>& b)
  1375. {
  1376. return detail::expression<detail::bitwise_or, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, detail::expression<tag2, Arg1b, Arg2b, Arg3b, Arg4b> >(a, b);
  1377. }
  1378. template <class tag, class Arg1, class Arg2, class Arg3, class Arg4, class V>
  1379. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<is_compatible_arithmetic_type<V, typename detail::expression<tag, Arg1, Arg2, Arg3, Arg4>::result_type>::value && (number_category<typename detail::expression<tag, Arg1, Arg2, Arg3, Arg4>::result_type>::value == number_kind_integer),
  1380. detail::expression<detail::bitwise_or, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, V> >::type
  1381. operator|(const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& a, const V& b)
  1382. {
  1383. return detail::expression<detail::bitwise_or, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, V>(a, b);
  1384. }
  1385. template <class V, class tag, class Arg1, class Arg2, class Arg3, class Arg4>
  1386. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<is_compatible_arithmetic_type<V, typename detail::expression<tag, Arg1, Arg2, Arg3, Arg4>::result_type>::value && (number_category<typename detail::expression<tag, Arg1, Arg2, Arg3, Arg4>::result_type>::value == number_kind_integer),
  1387. detail::expression<detail::bitwise_or, V, detail::expression<tag, Arg1, Arg2, Arg3, Arg4> > >::type
  1388. operator|(const V& a, const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& b)
  1389. {
  1390. return detail::expression<detail::bitwise_or, V, detail::expression<tag, Arg1, Arg2, Arg3, Arg4> >(a, b);
  1391. }
  1392. //
  1393. // Bitwise XOR:
  1394. //
  1395. template <class B>
  1396. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<number_category<B>::value == number_kind_integer,
  1397. detail::expression<detail::bitwise_xor_immediates, number<B, et_on>, number<B, et_on> > >::type
  1398. operator^(const number<B, et_on>& a, const number<B, et_on>& b)
  1399. {
  1400. return detail::expression<detail::bitwise_xor_immediates, number<B, et_on>, number<B, et_on> >(a, b);
  1401. }
  1402. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  1403. template <class B>
  1404. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<number_category<B>::value == number_kind_integer,
  1405. number<B, et_on> >::type
  1406. operator^(number<B, et_on>&& a, const number<B, et_on>& b)
  1407. {
  1408. return detail::expression<detail::bitwise_xor_immediates, number<B, et_on>, number<B, et_on> >(a, b);
  1409. }
  1410. template <class B>
  1411. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<number_category<B>::value == number_kind_integer,
  1412. number<B, et_on> >::type
  1413. operator^(const number<B, et_on>& a, number<B, et_on>&& b)
  1414. {
  1415. return detail::expression<detail::bitwise_xor_immediates, number<B, et_on>, number<B, et_on> >(a, b);
  1416. }
  1417. template <class B>
  1418. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<number_category<B>::value == number_kind_integer,
  1419. number<B, et_on> >::type
  1420. operator^(number<B, et_on>&& a, number<B, et_on>&& b)
  1421. {
  1422. return detail::expression<detail::bitwise_xor_immediates, number<B, et_on>, number<B, et_on> >(a, b);
  1423. }
  1424. #endif
  1425. template <class B, class V>
  1426. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<is_compatible_arithmetic_type<V, number<B, et_on> >::value && (number_category<B>::value == number_kind_integer),
  1427. detail::expression<detail::bitwise_xor_immediates, number<B, et_on>, V> >::type
  1428. operator^(const number<B, et_on>& a, const V& b)
  1429. {
  1430. return detail::expression<detail::bitwise_xor_immediates, number<B, et_on>, V>(a, b);
  1431. }
  1432. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  1433. template <class B, class V>
  1434. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<is_compatible_arithmetic_type<V, number<B, et_on> >::value && (number_category<B>::value == number_kind_integer),
  1435. number<B, et_on> >::type
  1436. operator^(number<B, et_on>&& a, const V& b)
  1437. {
  1438. return detail::expression<detail::bitwise_xor_immediates, number<B, et_on>, V>(a, b);
  1439. }
  1440. #endif
  1441. template <class V, class B>
  1442. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<is_compatible_arithmetic_type<V, number<B, et_on> >::value && (number_category<B>::value == number_kind_integer),
  1443. detail::expression<detail::bitwise_xor_immediates, V, number<B, et_on> > >::type
  1444. operator^(const V& a, const number<B, et_on>& b)
  1445. {
  1446. return detail::expression<detail::bitwise_xor_immediates, V, number<B, et_on> >(a, b);
  1447. }
  1448. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  1449. template <class V, class B>
  1450. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<is_compatible_arithmetic_type<V, number<B, et_on> >::value && (number_category<B>::value == number_kind_integer),
  1451. number<B, et_on> >::type
  1452. operator^(const V& a, number<B, et_on>&& b)
  1453. {
  1454. return detail::expression<detail::bitwise_xor_immediates, V, number<B, et_on> >(a, b);
  1455. }
  1456. #endif
  1457. template <class B, class tag, class Arg1, class Arg2, class Arg3, class Arg4>
  1458. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<number_category<B>::value == number_kind_integer,
  1459. detail::expression<detail::bitwise_xor, number<B, et_on>, detail::expression<tag, Arg1, Arg2, Arg3, Arg4> > >::type
  1460. operator^(const number<B, et_on>& a, const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& b)
  1461. {
  1462. return detail::expression<detail::bitwise_xor, number<B, et_on>, detail::expression<tag, Arg1, Arg2, Arg3, Arg4> >(a, b);
  1463. }
  1464. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  1465. template <class B, class tag, class Arg1, class Arg2, class Arg3, class Arg4>
  1466. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<number_category<B>::value == number_kind_integer,
  1467. typename detail::expression<detail::bitwise_xor, number<B, et_on>, detail::expression<tag, Arg1, Arg2, Arg3, Arg4> >::result_type>::type
  1468. operator^(number<B, et_on>&& a, const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& b)
  1469. {
  1470. return detail::expression<detail::bitwise_xor, number<B, et_on>, detail::expression<tag, Arg1, Arg2, Arg3, Arg4> >(a, b);
  1471. }
  1472. #endif
  1473. template <class tag, class Arg1, class Arg2, class Arg3, class Arg4, class B>
  1474. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<number_category<B>::value == number_kind_integer,
  1475. detail::expression<detail::bitwise_xor, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, number<B, et_on> > >::type
  1476. operator^(const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& a, const number<B, et_on>& b)
  1477. {
  1478. return detail::expression<detail::bitwise_xor, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, number<B, et_on> >(a, b);
  1479. }
  1480. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  1481. template <class tag, class Arg1, class Arg2, class Arg3, class Arg4, class B>
  1482. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<number_category<B>::value == number_kind_integer,
  1483. typename detail::expression<detail::bitwise_xor, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, number<B, et_on> >::result_type>::type
  1484. operator^(const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& a, number<B, et_on>&& b)
  1485. {
  1486. return detail::expression<detail::bitwise_xor, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, number<B, et_on> >(a, b);
  1487. }
  1488. #endif
  1489. template <class tag, class Arg1, class Arg2, class Arg3, class Arg4, class tag2, class Arg1b, class Arg2b, class Arg3b, class Arg4b>
  1490. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<number_category<typename detail::expression<tag, Arg1, Arg2, Arg3, Arg4>::result_type>::value == number_kind_integer,
  1491. detail::expression<detail::bitwise_xor, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, detail::expression<tag2, Arg1b, Arg2b, Arg3b, Arg4b> > >::type
  1492. operator^(const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& a, const detail::expression<tag2, Arg1b, Arg2b, Arg3b, Arg4b>& b)
  1493. {
  1494. return detail::expression<detail::bitwise_xor, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, detail::expression<tag2, Arg1b, Arg2b, Arg3b, Arg4b> >(a, b);
  1495. }
  1496. template <class tag, class Arg1, class Arg2, class Arg3, class Arg4, class V>
  1497. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<is_compatible_arithmetic_type<V, typename detail::expression<tag, Arg1, Arg2, Arg3, Arg4>::result_type>::value && (number_category<typename detail::expression<tag, Arg1, Arg2, Arg3, Arg4>::result_type>::value == number_kind_integer),
  1498. detail::expression<detail::bitwise_xor, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, V> >::type
  1499. operator^(const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& a, const V& b)
  1500. {
  1501. return detail::expression<detail::bitwise_xor, detail::expression<tag, Arg1, Arg2, Arg3, Arg4>, V>(a, b);
  1502. }
  1503. template <class V, class tag, class Arg1, class Arg2, class Arg3, class Arg4>
  1504. inline BOOST_MP_CXX14_CONSTEXPR typename enable_if_c<is_compatible_arithmetic_type<V, typename detail::expression<tag, Arg1, Arg2, Arg3, Arg4>::result_type>::value && (number_category<typename detail::expression<tag, Arg1, Arg2, Arg3, Arg4>::result_type>::value == number_kind_integer), detail::expression<detail::bitwise_xor, V, detail::expression<tag, Arg1, Arg2, Arg3, Arg4> > >::type
  1505. operator^(const V& a, const detail::expression<tag, Arg1, Arg2, Arg3, Arg4>& b)
  1506. {
  1507. return detail::expression<detail::bitwise_xor, V, detail::expression<tag, Arg1, Arg2, Arg3, Arg4> >(a, b);
  1508. }
  1509. }} // namespace boost::multiprecision
  1510. #endif