extends.hpp 41 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647
  1. ///////////////////////////////////////////////////////////////////////////////
  2. /// \file extends.hpp
  3. /// Macros and a base class for defining end-user expression types
  4. //
  5. // Copyright 2008 Eric Niebler. Distributed under the Boost
  6. // Software License, Version 1.0. (See accompanying file
  7. // LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
  8. #ifndef BOOST_PROTO_EXTENDS_HPP_EAN_11_1_2006
  9. #define BOOST_PROTO_EXTENDS_HPP_EAN_11_1_2006
  10. #include <cstddef> // for offsetof
  11. #include <boost/config.hpp>
  12. #include <boost/detail/workaround.hpp>
  13. #include <boost/preprocessor/facilities/empty.hpp>
  14. #include <boost/preprocessor/tuple/elem.hpp>
  15. #include <boost/preprocessor/control/if.hpp>
  16. #include <boost/preprocessor/arithmetic/inc.hpp>
  17. #include <boost/preprocessor/arithmetic/dec.hpp>
  18. #include <boost/preprocessor/iteration/local.hpp>
  19. #include <boost/preprocessor/repetition/enum_params.hpp>
  20. #include <boost/preprocessor/repetition/repeat_from_to.hpp>
  21. #include <boost/preprocessor/repetition/enum_binary_params.hpp>
  22. #include <boost/preprocessor/repetition/enum_trailing_params.hpp>
  23. #include <boost/preprocessor/repetition/enum_trailing_binary_params.hpp>
  24. #include <boost/preprocessor/seq/for_each.hpp>
  25. #include <boost/utility/addressof.hpp>
  26. #include <boost/utility/result_of.hpp>
  27. #include <boost/proto/proto_fwd.hpp>
  28. #include <boost/proto/traits.hpp>
  29. #include <boost/proto/expr.hpp>
  30. #include <boost/proto/args.hpp>
  31. #include <boost/proto/traits.hpp>
  32. #include <boost/proto/generate.hpp>
  33. #include <boost/proto/detail/remove_typename.hpp>
  34. #if defined(_MSC_VER)
  35. # pragma warning(push)
  36. # pragma warning(disable : 4714) // function 'xxx' marked as __forceinline not inlined
  37. #endif
  38. namespace boost { namespace proto
  39. {
  40. #ifdef __GNUC__
  41. /// INTERNAL ONLY
  42. ///
  43. # define BOOST_PROTO_ADDROF(x) ((char const volatile*)boost::addressof(x))
  44. /// INTERNAL ONLY
  45. ///
  46. # define BOOST_PROTO_OFFSETOF(s,m) (BOOST_PROTO_ADDROF((((s *)this)->m)) - BOOST_PROTO_ADDROF(*((s *)this)))
  47. #else
  48. /// INTERNAL ONLY
  49. ///
  50. # define BOOST_PROTO_OFFSETOF offsetof
  51. #endif
  52. /// INTERNAL ONLY
  53. ///
  54. #define BOOST_PROTO_CONST() const
  55. /// INTERNAL ONLY
  56. ///
  57. #define BOOST_PROTO_TYPENAME() typename
  58. /// INTERNAL ONLY
  59. ///
  60. #define BOOST_PROTO_TEMPLATE_YES_(Z, N) template<BOOST_PP_ENUM_PARAMS_Z(Z, N, typename A)>
  61. /// INTERNAL ONLY
  62. ///
  63. #define BOOST_PROTO_TEMPLATE_NO_(Z, N)
  64. /// INTERNAL ONLY
  65. ///
  66. #define BOOST_PROTO_DEFINE_FUN_OP_IMPL_(Z, N, DATA, Const) \
  67. BOOST_PP_IF(N, BOOST_PROTO_TEMPLATE_YES_, BOOST_PROTO_TEMPLATE_NO_)(Z, N) \
  68. BOOST_PROTO_DISABLE_MSVC_C4714 BOOST_FORCEINLINE \
  69. typename BOOST_PROTO_RESULT_OF< \
  70. proto_generator( \
  71. typename boost::proto::result_of::BOOST_PP_CAT(funop, N)< \
  72. proto_derived_expr Const() \
  73. , proto_domain \
  74. BOOST_PP_ENUM_TRAILING_PARAMS_Z(Z, N, const A) \
  75. >::type \
  76. ) \
  77. >::type const \
  78. operator ()(BOOST_PP_ENUM_BINARY_PARAMS_Z(Z, N, A, const &a)) Const() \
  79. { \
  80. typedef boost::proto::result_of::BOOST_PP_CAT(funop, N)< \
  81. proto_derived_expr Const() \
  82. , proto_domain \
  83. BOOST_PP_ENUM_TRAILING_PARAMS_Z(Z, N, const A) \
  84. > funop; \
  85. return proto_generator()( \
  86. funop::call( \
  87. *static_cast<proto_derived_expr Const() *>(this) \
  88. BOOST_PP_ENUM_TRAILING_PARAMS_Z(Z, N, a) \
  89. ) \
  90. ); \
  91. } \
  92. /**/
  93. /// INTERNAL ONLY
  94. ///
  95. #define BOOST_PROTO_DEFINE_FUN_OP_VARIADIC_IMPL_(Const) \
  96. template<typename... A> \
  97. BOOST_PROTO_DISABLE_MSVC_C4714 BOOST_FORCEINLINE \
  98. typename BOOST_PROTO_RESULT_OF< \
  99. proto_generator( \
  100. typename boost::proto::result_of::funop< \
  101. proto_derived_expr Const()(A const &...) \
  102. , proto_derived_expr \
  103. , proto_domain \
  104. >::type \
  105. ) \
  106. >::type const \
  107. operator ()(A const &...a) Const() \
  108. { \
  109. typedef boost::proto::result_of::funop< \
  110. proto_derived_expr Const()(A const &...) \
  111. , proto_derived_expr \
  112. , proto_domain \
  113. > funop; \
  114. return proto_generator()( \
  115. funop::call( \
  116. *static_cast<proto_derived_expr Const() *>(this) \
  117. , a... \
  118. ) \
  119. ); \
  120. } \
  121. /**/
  122. /// INTERNAL ONLY
  123. ///
  124. #define BOOST_PROTO_DEFINE_FUN_OP_CONST(Z, N, DATA) \
  125. BOOST_PROTO_DEFINE_FUN_OP_IMPL_(Z, N, DATA, BOOST_PROTO_CONST) \
  126. /**/
  127. /// INTERNAL ONLY
  128. ///
  129. #define BOOST_PROTO_DEFINE_FUN_OP_NON_CONST(Z, N, DATA) \
  130. BOOST_PROTO_DEFINE_FUN_OP_IMPL_(Z, N, DATA, BOOST_PP_EMPTY) \
  131. /**/
  132. /// INTERNAL ONLY
  133. ///
  134. #define BOOST_PROTO_DEFINE_FUN_OP(Z, N, DATA) \
  135. BOOST_PROTO_DEFINE_FUN_OP_CONST(Z, N, DATA) \
  136. BOOST_PROTO_DEFINE_FUN_OP_NON_CONST(Z, N, DATA) \
  137. /**/
  138. /// INTERNAL ONLY
  139. ///
  140. #define BOOST_PROTO_EXTENDS_CHILD(Z, N, DATA) \
  141. typedef \
  142. typename proto_base_expr::BOOST_PP_CAT(proto_child, N) \
  143. BOOST_PP_CAT(proto_child, N); \
  144. /**/
  145. #define BOOST_PROTO_BASIC_EXTENDS_(Expr, Derived, Domain) \
  146. Expr proto_expr_; \
  147. \
  148. typedef Expr proto_base_expr_; /**< INTERNAL ONLY */ \
  149. typedef typename proto_base_expr_::proto_base_expr proto_base_expr; \
  150. typedef BOOST_PROTO_REMOVE_TYPENAME(Domain) proto_domain; \
  151. typedef Derived proto_derived_expr; \
  152. typedef Domain::proto_generator proto_generator; \
  153. typedef typename proto_base_expr::proto_tag proto_tag; \
  154. typedef typename proto_base_expr::proto_args proto_args; \
  155. typedef typename proto_base_expr::proto_arity proto_arity; \
  156. typedef typename proto_base_expr::proto_grammar proto_grammar; \
  157. typedef typename proto_base_expr::address_of_hack_type_ proto_address_of_hack_type_; \
  158. typedef void proto_is_expr_; /**< INTERNAL ONLY */ \
  159. static const long proto_arity_c = proto_base_expr::proto_arity_c; \
  160. typedef boost::proto::tag::proto_expr<proto_tag, proto_domain> fusion_tag; \
  161. BOOST_PP_REPEAT(BOOST_PROTO_MAX_ARITY, BOOST_PROTO_EXTENDS_CHILD, ~) \
  162. \
  163. BOOST_PROTO_DISABLE_MSVC_C4714 BOOST_FORCEINLINE \
  164. static proto_derived_expr const make(Expr const &e) \
  165. { \
  166. proto_derived_expr that = {e}; \
  167. return that; \
  168. } \
  169. \
  170. BOOST_PROTO_DISABLE_MSVC_C4714 BOOST_FORCEINLINE \
  171. proto_base_expr &proto_base() \
  172. { \
  173. return this->proto_expr_.proto_base(); \
  174. } \
  175. \
  176. BOOST_PROTO_DISABLE_MSVC_C4714 BOOST_FORCEINLINE \
  177. proto_base_expr const &proto_base() const \
  178. { \
  179. return this->proto_expr_.proto_base(); \
  180. } \
  181. \
  182. BOOST_PROTO_DISABLE_MSVC_C4714 BOOST_FORCEINLINE \
  183. operator proto_address_of_hack_type_() const \
  184. { \
  185. return boost::addressof(this->proto_base().child0); \
  186. } \
  187. /**/
  188. #define BOOST_PROTO_BASIC_EXTENDS(Expr, Derived, Domain) \
  189. BOOST_PROTO_BASIC_EXTENDS_(Expr, Derived, Domain) \
  190. typedef void proto_is_aggregate_; \
  191. /**< INTERNAL ONLY */
  192. #define BOOST_PROTO_EXTENDS_COPY_ASSIGN_IMPL_(This, Const, Typename) \
  193. BOOST_PROTO_DISABLE_MSVC_C4522 \
  194. BOOST_PROTO_DISABLE_MSVC_C4714 BOOST_FORCEINLINE \
  195. Typename() BOOST_PROTO_RESULT_OF< \
  196. Typename() This::proto_generator( \
  197. Typename() boost::proto::base_expr< \
  198. Typename() This::proto_domain \
  199. , boost::proto::tag::assign \
  200. , boost::proto::list2< \
  201. This & \
  202. , This Const() & \
  203. > \
  204. >::type \
  205. ) \
  206. >::type const \
  207. operator =(This Const() &a) \
  208. { \
  209. typedef \
  210. Typename() boost::proto::base_expr< \
  211. Typename() This::proto_domain \
  212. , boost::proto::tag::assign \
  213. , boost::proto::list2< \
  214. This & \
  215. , This Const() & \
  216. > \
  217. >::type \
  218. that_type; \
  219. that_type const that = { \
  220. *this \
  221. , a \
  222. }; \
  223. return Typename() This::proto_generator()(that); \
  224. } \
  225. /**/
  226. // MSVC 8.0 and higher seem to need copy-assignment operator to be overloaded on *both*
  227. // const and non-const rhs arguments.
  228. #if BOOST_WORKAROUND(BOOST_MSVC, BOOST_TESTED_AT(1600)) && (BOOST_MSVC > 1310)
  229. #define BOOST_PROTO_EXTENDS_COPY_ASSIGN_(This, Typename) \
  230. BOOST_PROTO_EXTENDS_COPY_ASSIGN_IMPL_(This, BOOST_PP_EMPTY, Typename) \
  231. BOOST_PROTO_EXTENDS_COPY_ASSIGN_IMPL_(This, BOOST_PROTO_CONST, Typename) \
  232. /**/
  233. #else
  234. #define BOOST_PROTO_EXTENDS_COPY_ASSIGN_(This, Typename) \
  235. BOOST_PROTO_EXTENDS_COPY_ASSIGN_IMPL_(This, BOOST_PROTO_CONST, Typename) \
  236. /**/
  237. #endif
  238. /// INTERNAL ONLY
  239. ///
  240. #define BOOST_PROTO_EXTENDS_ASSIGN_IMPL_(ThisConst, ThatConst) \
  241. template<typename A> \
  242. BOOST_PROTO_DISABLE_MSVC_C4714 BOOST_FORCEINLINE \
  243. typename BOOST_PROTO_RESULT_OF< \
  244. proto_generator( \
  245. typename boost::proto::base_expr< \
  246. proto_domain \
  247. , boost::proto::tag::assign \
  248. , boost::proto::list2< \
  249. proto_derived_expr ThisConst() & \
  250. , typename boost::proto::result_of::as_child<A ThatConst(), proto_domain>::type \
  251. > \
  252. >::type \
  253. ) \
  254. >::type const \
  255. operator =(A ThatConst() &a) ThisConst() \
  256. { \
  257. typedef \
  258. typename boost::proto::base_expr< \
  259. proto_domain \
  260. , boost::proto::tag::assign \
  261. , boost::proto::list2< \
  262. proto_derived_expr ThisConst() & \
  263. , typename boost::proto::result_of::as_child<A ThatConst(), proto_domain>::type \
  264. > \
  265. >::type \
  266. that_type; \
  267. that_type const that = { \
  268. *static_cast<proto_derived_expr ThisConst() *>(this) \
  269. , boost::proto::as_child<proto_domain>(a) \
  270. }; \
  271. return proto_generator()(that); \
  272. } \
  273. /**/
  274. #define BOOST_PROTO_EXTENDS_ASSIGN_CONST_() \
  275. BOOST_PROTO_EXTENDS_ASSIGN_IMPL_(BOOST_PROTO_CONST, BOOST_PP_EMPTY) \
  276. BOOST_PROTO_EXTENDS_ASSIGN_IMPL_(BOOST_PROTO_CONST, BOOST_PROTO_CONST) \
  277. /**/
  278. #define BOOST_PROTO_EXTENDS_ASSIGN_NON_CONST_() \
  279. BOOST_PROTO_EXTENDS_ASSIGN_IMPL_(BOOST_PP_EMPTY, BOOST_PP_EMPTY) \
  280. BOOST_PROTO_EXTENDS_ASSIGN_IMPL_(BOOST_PP_EMPTY, BOOST_PROTO_CONST) \
  281. /**/
  282. #define BOOST_PROTO_EXTENDS_ASSIGN_() \
  283. BOOST_PROTO_EXTENDS_ASSIGN_CONST_() \
  284. BOOST_PROTO_EXTENDS_ASSIGN_NON_CONST_() \
  285. /**/
  286. #define BOOST_PROTO_EXTENDS_ASSIGN_CONST() \
  287. BOOST_PROTO_EXTENDS_COPY_ASSIGN_(proto_derived_expr, BOOST_PROTO_TYPENAME) \
  288. BOOST_PROTO_EXTENDS_ASSIGN_CONST_() \
  289. /**/
  290. #define BOOST_PROTO_EXTENDS_ASSIGN_NON_CONST() \
  291. BOOST_PROTO_EXTENDS_COPY_ASSIGN_(proto_derived_expr, BOOST_PROTO_TYPENAME) \
  292. BOOST_PROTO_EXTENDS_ASSIGN_NON_CONST_() \
  293. /**/
  294. #define BOOST_PROTO_EXTENDS_ASSIGN() \
  295. BOOST_PROTO_EXTENDS_COPY_ASSIGN_(proto_derived_expr, BOOST_PROTO_TYPENAME) \
  296. BOOST_PROTO_EXTENDS_ASSIGN_() \
  297. /**/
  298. /// INTERNAL ONLY
  299. ///
  300. #define BOOST_PROTO_EXTENDS_SUBSCRIPT_IMPL_(ThisConst, ThatConst) \
  301. template<typename A> \
  302. BOOST_PROTO_DISABLE_MSVC_C4714 BOOST_FORCEINLINE \
  303. typename BOOST_PROTO_RESULT_OF< \
  304. proto_generator( \
  305. typename boost::proto::base_expr< \
  306. proto_domain \
  307. , boost::proto::tag::subscript \
  308. , boost::proto::list2< \
  309. proto_derived_expr ThisConst() & \
  310. , typename boost::proto::result_of::as_child<A ThatConst(), proto_domain>::type \
  311. > \
  312. >::type \
  313. ) \
  314. >::type const \
  315. operator [](A ThatConst() &a) ThisConst() \
  316. { \
  317. typedef \
  318. typename boost::proto::base_expr< \
  319. proto_domain \
  320. , boost::proto::tag::subscript \
  321. , boost::proto::list2< \
  322. proto_derived_expr ThisConst() & \
  323. , typename boost::proto::result_of::as_child<A ThatConst(), proto_domain>::type \
  324. > \
  325. >::type \
  326. that_type; \
  327. that_type const that = { \
  328. *static_cast<proto_derived_expr ThisConst() *>(this) \
  329. , boost::proto::as_child<proto_domain>(a) \
  330. }; \
  331. return proto_generator()(that); \
  332. } \
  333. /**/
  334. #define BOOST_PROTO_EXTENDS_SUBSCRIPT_CONST() \
  335. BOOST_PROTO_EXTENDS_SUBSCRIPT_IMPL_(BOOST_PROTO_CONST, BOOST_PP_EMPTY) \
  336. BOOST_PROTO_EXTENDS_SUBSCRIPT_IMPL_(BOOST_PROTO_CONST, BOOST_PROTO_CONST) \
  337. /**/
  338. #define BOOST_PROTO_EXTENDS_SUBSCRIPT_NON_CONST() \
  339. BOOST_PROTO_EXTENDS_SUBSCRIPT_IMPL_(BOOST_PP_EMPTY, BOOST_PP_EMPTY) \
  340. BOOST_PROTO_EXTENDS_SUBSCRIPT_IMPL_(BOOST_PP_EMPTY, BOOST_PROTO_CONST) \
  341. /**/
  342. #define BOOST_PROTO_EXTENDS_SUBSCRIPT() \
  343. BOOST_PROTO_EXTENDS_SUBSCRIPT_CONST() \
  344. BOOST_PROTO_EXTENDS_SUBSCRIPT_NON_CONST() \
  345. /**/
  346. /// INTERNAL ONLY
  347. ///
  348. #define BOOST_PROTO_EXTENDS_FUNCTION_() \
  349. template<typename Sig> \
  350. struct result \
  351. { \
  352. typedef \
  353. typename BOOST_PROTO_RESULT_OF< \
  354. proto_generator( \
  355. typename boost::proto::result_of::funop< \
  356. Sig \
  357. , proto_derived_expr \
  358. , proto_domain \
  359. >::type \
  360. ) \
  361. >::type const \
  362. type; \
  363. }; \
  364. /**/
  365. #ifndef BOOST_NO_CXX11_VARIADIC_TEMPLATES
  366. #define BOOST_PROTO_EXTENDS_FUNCTION_CONST() \
  367. BOOST_PROTO_EXTENDS_FUNCTION_() \
  368. BOOST_PROTO_DEFINE_FUN_OP_VARIADIC_IMPL_(BOOST_PROTO_CONST) \
  369. /**/
  370. #define BOOST_PROTO_EXTENDS_FUNCTION_NON_CONST() \
  371. BOOST_PROTO_EXTENDS_FUNCTION_() \
  372. BOOST_PROTO_DEFINE_FUN_OP_VARIADIC_IMPL_(BOOST_PP_EMPTY) \
  373. /**/
  374. #define BOOST_PROTO_EXTENDS_FUNCTION() \
  375. BOOST_PROTO_EXTENDS_FUNCTION_() \
  376. BOOST_PROTO_DEFINE_FUN_OP_VARIADIC_IMPL_(BOOST_PP_EMPTY) \
  377. BOOST_PROTO_DEFINE_FUN_OP_VARIADIC_IMPL_(BOOST_PROTO_CONST) \
  378. /**/
  379. #else
  380. #define BOOST_PROTO_EXTENDS_FUNCTION_CONST() \
  381. BOOST_PROTO_EXTENDS_FUNCTION_() \
  382. BOOST_PP_REPEAT_FROM_TO( \
  383. 0 \
  384. , BOOST_PROTO_MAX_FUNCTION_CALL_ARITY \
  385. , BOOST_PROTO_DEFINE_FUN_OP_CONST \
  386. , ~ \
  387. ) \
  388. /**/
  389. #define BOOST_PROTO_EXTENDS_FUNCTION_NON_CONST() \
  390. BOOST_PROTO_EXTENDS_FUNCTION_() \
  391. BOOST_PP_REPEAT_FROM_TO( \
  392. 0 \
  393. , BOOST_PROTO_MAX_FUNCTION_CALL_ARITY \
  394. , BOOST_PROTO_DEFINE_FUN_OP_NON_CONST \
  395. , ~ \
  396. ) \
  397. /**/
  398. #define BOOST_PROTO_EXTENDS_FUNCTION() \
  399. BOOST_PROTO_EXTENDS_FUNCTION_() \
  400. BOOST_PP_REPEAT_FROM_TO( \
  401. 0 \
  402. , BOOST_PROTO_MAX_FUNCTION_CALL_ARITY \
  403. , BOOST_PROTO_DEFINE_FUN_OP \
  404. , ~ \
  405. ) \
  406. /**/
  407. #endif
  408. #define BOOST_PROTO_EXTENDS(Expr, Derived, Domain) \
  409. BOOST_PROTO_BASIC_EXTENDS(Expr, Derived, Domain) \
  410. BOOST_PROTO_EXTENDS_ASSIGN() \
  411. BOOST_PROTO_EXTENDS_SUBSCRIPT() \
  412. BOOST_PROTO_EXTENDS_FUNCTION() \
  413. /**/
  414. #define BOOST_PROTO_EXTENDS_USING_ASSIGN(Derived) \
  415. typedef typename Derived::proto_extends proto_extends; \
  416. using proto_extends::operator =; \
  417. BOOST_PROTO_EXTENDS_COPY_ASSIGN_(Derived, BOOST_PROTO_TYPENAME) \
  418. /**/
  419. #define BOOST_PROTO_EXTENDS_USING_ASSIGN_NON_DEPENDENT(Derived) \
  420. typedef Derived::proto_extends proto_extends; \
  421. using proto_extends::operator =; \
  422. BOOST_PROTO_EXTENDS_COPY_ASSIGN_(Derived, BOOST_PP_EMPTY) \
  423. /**/
  424. namespace exprns_
  425. {
  426. /// \brief Empty type to be used as a dummy template parameter of
  427. /// POD expression wrappers. It allows argument-dependent lookup
  428. /// to find Proto's operator overloads.
  429. ///
  430. /// \c proto::is_proto_expr allows argument-dependent lookup
  431. /// to find Proto's operator overloads. For example:
  432. ///
  433. /// \code
  434. /// template<typename T, typename Dummy = proto::is_proto_expr>
  435. /// struct my_terminal
  436. /// {
  437. /// BOOST_PROTO_BASIC_EXTENDS(
  438. /// typename proto::terminal<T>::type
  439. /// , my_terminal<T>
  440. /// , default_domain
  441. /// )
  442. /// };
  443. ///
  444. /// // ...
  445. /// my_terminal<int> _1, _2;
  446. /// _1 + _2; // OK, uses proto::operator+
  447. /// \endcode
  448. ///
  449. /// Without the second \c Dummy template parameter, Proto's operator
  450. /// overloads would not be considered by name lookup.
  451. struct is_proto_expr
  452. {};
  453. /// \brief extends\<\> class template for adding behaviors to a Proto expression template
  454. ///
  455. template<
  456. typename Expr
  457. , typename Derived
  458. , typename Domain // = proto::default_domain
  459. , long Arity // = Expr::proto_arity_c
  460. >
  461. struct extends
  462. {
  463. BOOST_FORCEINLINE
  464. extends()
  465. : proto_expr_()
  466. {}
  467. BOOST_FORCEINLINE
  468. extends(extends const &that)
  469. : proto_expr_(that.proto_expr_)
  470. {}
  471. BOOST_FORCEINLINE
  472. extends(Expr const &expr_)
  473. : proto_expr_(expr_)
  474. {}
  475. typedef extends proto_extends;
  476. BOOST_PROTO_BASIC_EXTENDS_(Expr, Derived, typename Domain)
  477. BOOST_PROTO_EXTENDS_ASSIGN_CONST_()
  478. BOOST_PROTO_EXTENDS_SUBSCRIPT_CONST()
  479. // Instead of using BOOST_PROTO_EXTENDS_FUNCTION, which uses
  480. // nested preprocessor loops, use file iteration here to generate
  481. // the operator() overloads, which is more efficient.
  482. #include <boost/proto/detail/extends_funop_const.hpp>
  483. };
  484. /// \brief extends\<\> class template for adding behaviors to a Proto expression template
  485. ///
  486. template<typename Expr, typename Derived, typename Domain>
  487. struct extends<Expr, Derived, Domain, 0>
  488. {
  489. BOOST_FORCEINLINE
  490. extends()
  491. : proto_expr_()
  492. {}
  493. BOOST_FORCEINLINE
  494. extends(extends const &that)
  495. : proto_expr_(that.proto_expr_)
  496. {}
  497. BOOST_FORCEINLINE
  498. extends(Expr const &expr_)
  499. : proto_expr_(expr_)
  500. {}
  501. typedef extends proto_extends;
  502. BOOST_PROTO_BASIC_EXTENDS_(Expr, Derived, typename Domain)
  503. BOOST_PROTO_EXTENDS_ASSIGN_()
  504. BOOST_PROTO_EXTENDS_SUBSCRIPT()
  505. // Instead of using BOOST_PROTO_EXTENDS_FUNCTION, which uses
  506. // nested preprocessor loops, use file iteration here to generate
  507. // the operator() overloads, which is more efficient.
  508. #include <boost/proto/detail/extends_funop.hpp>
  509. };
  510. /// INTERNAL ONLY
  511. ///
  512. template<typename This, typename Fun, typename Domain>
  513. struct virtual_member
  514. {
  515. typedef Domain proto_domain;
  516. typedef typename Domain::proto_generator proto_generator;
  517. typedef virtual_member<This, Fun, Domain> proto_derived_expr;
  518. typedef tag::member proto_tag;
  519. typedef list2<This &, expr<tag::terminal, term<Fun> > const &> proto_args;
  520. typedef mpl::long_<2> proto_arity;
  521. typedef detail::not_a_valid_type proto_address_of_hack_type_;
  522. typedef void proto_is_expr_; /**< INTERNAL ONLY */
  523. static const long proto_arity_c = 2;
  524. typedef boost::proto::tag::proto_expr<proto_tag, Domain> fusion_tag;
  525. typedef This &proto_child0;
  526. typedef expr<tag::terminal, term<Fun> > const &proto_child1;
  527. typedef expr<proto_tag, proto_args, proto_arity_c> proto_base_expr;
  528. typedef basic_expr<proto_tag, proto_args, proto_arity_c> proto_grammar;
  529. typedef void proto_is_aggregate_; /**< INTERNAL ONLY */
  530. BOOST_PROTO_EXTENDS_ASSIGN_()
  531. BOOST_PROTO_EXTENDS_SUBSCRIPT()
  532. // Instead of using BOOST_PROTO_EXTENDS_FUNCTION, which uses
  533. // nested preprocessor loops, use file iteration here to generate
  534. // the operator() overloads, which is more efficient.
  535. #define BOOST_PROTO_NO_WAVE_OUTPUT
  536. #include <boost/proto/detail/extends_funop.hpp>
  537. #undef BOOST_PROTO_NO_WAVE_OUTPUT
  538. BOOST_FORCEINLINE
  539. proto_base_expr const proto_base() const
  540. {
  541. proto_base_expr that = {this->child0(), this->child1()};
  542. return that;
  543. }
  544. BOOST_FORCEINLINE
  545. proto_child0 child0() const
  546. {
  547. using std::size_t;
  548. return *(This *)((char *)this - BOOST_PROTO_OFFSETOF(This, proto_member_union_start_));
  549. }
  550. BOOST_FORCEINLINE
  551. proto_child1 child1() const
  552. {
  553. static expr<tag::terminal, term<Fun>, 0> const that = {Fun()};
  554. return that;
  555. }
  556. };
  557. /// INTERNAL ONLY
  558. ///
  559. #define BOOST_PROTO_EXTENDS_MEMBER_(R, DOMAIN, ELEM) \
  560. boost::proto::exprns_::virtual_member< \
  561. proto_derived_expr \
  562. , BOOST_PP_TUPLE_ELEM(2, 0, ELEM) \
  563. , DOMAIN \
  564. > BOOST_PP_TUPLE_ELEM(2, 1, ELEM); \
  565. /**/
  566. /// \brief For declaring virtual data members in an extension class.
  567. ///
  568. #define BOOST_PROTO_EXTENDS_MEMBERS_WITH_DOMAIN(SEQ, DOMAIN) \
  569. union \
  570. { \
  571. char proto_member_union_start_; \
  572. BOOST_PP_SEQ_FOR_EACH(BOOST_PROTO_EXTENDS_MEMBER_, DOMAIN, SEQ) \
  573. }; \
  574. /**/
  575. /// \brief For declaring virtual data members in an extension class.
  576. ///
  577. #define BOOST_PROTO_EXTENDS_MEMBERS(SEQ) \
  578. BOOST_PROTO_EXTENDS_MEMBERS_WITH_DOMAIN(SEQ, proto_domain) \
  579. /**/
  580. }
  581. }}
  582. #if defined(_MSC_VER)
  583. # pragma warning(pop)
  584. #endif
  585. #endif