invoke.cpp 27 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560
  1. /*=============================================================================
  2. Copyright (c) 2005-2006 Joao Abecasis
  3. Copyright (c) 2006-2007 Tobias Schwinger
  4. Use modification and distribution are subject to the Boost Software
  5. License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
  6. http://www.boost.org/LICENSE_1_0.txt).
  7. ==============================================================================*/
  8. #include <boost/config.hpp>
  9. #include <boost/fusion/functional/invocation/invoke.hpp>
  10. #include <boost/detail/lightweight_test.hpp>
  11. #if !defined(BOOST_NO_CXX11_RVALUE_REFERENCES)
  12. #include <functional>
  13. #endif
  14. #include <memory>
  15. #include <boost/noncopyable.hpp>
  16. #include <boost/type_traits/is_same.hpp>
  17. #include <boost/mpl/int.hpp>
  18. #include <boost/fusion/container/vector.hpp>
  19. #include <boost/fusion/container/list.hpp>
  20. #include <boost/fusion/sequence/intrinsic/size.hpp>
  21. #include <boost/fusion/sequence/intrinsic/begin.hpp>
  22. #include <boost/fusion/view/single_view.hpp>
  23. #include <boost/fusion/view/iterator_range.hpp>
  24. #include <boost/fusion/iterator/advance.hpp>
  25. #include <boost/fusion/algorithm/transformation/join.hpp>
  26. #include "../compile_time/sfinae_friendly.hpp"
  27. namespace mpl = boost::mpl;
  28. namespace fusion = boost::fusion;
  29. template <typename T>
  30. inline T const & const_(T const & t)
  31. {
  32. return t;
  33. }
  34. struct object {};
  35. struct object_nc : boost::noncopyable {};
  36. struct fobj
  37. {
  38. // Handle nullary separately to exercise result_of support
  39. template <typename Sig>
  40. struct result;
  41. template <class Self, typename T0>
  42. struct result< Self(T0) >
  43. {
  44. typedef int type;
  45. };
  46. template <class Self, typename T0, typename T1>
  47. struct result< Self(T0, T1) >
  48. {
  49. typedef int type;
  50. };
  51. template <class Self, typename T0, typename T1, typename T2>
  52. struct result< Self(T0, T1, T2) >
  53. {
  54. typedef int type;
  55. };
  56. int operator()(int i) { return 2 + i; }
  57. int operator()(int i) const { return 3 + i; }
  58. int operator()(int i, object &) { return 4 + i; }
  59. int operator()(int i, object &) const { return 5 + i; }
  60. int operator()(int i, object const &) { return 6 + i; }
  61. int operator()(int i, object const &) const { return 7 + i; }
  62. int operator()(int i, object &, object_nc &) { return 10 + i; }
  63. int operator()(int i, object &, object_nc &) const { return 11 + i; }
  64. int operator()(int i, object const &, object_nc &);
  65. int operator()(int i, object const &, object_nc &) const;
  66. };
  67. SFINAE_FRIENDLY_ASSERT((fusion::result_of::invoke<fobj, sfinae_friendly::v0>));
  68. SFINAE_FRIENDLY_ASSERT((fusion::result_of::invoke<fobj, sfinae_friendly::v1>));
  69. SFINAE_FRIENDLY_ASSERT((fusion::result_of::invoke<fobj, sfinae_friendly::v2>));
  70. SFINAE_FRIENDLY_ASSERT((fusion::result_of::invoke<fobj, sfinae_friendly::v3>));
  71. struct nullary_fobj
  72. {
  73. typedef int result_type;
  74. int operator()() { return 0; }
  75. int operator()() const { return 1; }
  76. };
  77. SFINAE_FRIENDLY_ASSERT((fusion::result_of::invoke<nullary_fobj, sfinae_friendly::v1>));
  78. SFINAE_FRIENDLY_ASSERT((fusion::result_of::invoke<nullary_fobj, sfinae_friendly::v2>));
  79. SFINAE_FRIENDLY_ASSERT((fusion::result_of::invoke<nullary_fobj, sfinae_friendly::v3>));
  80. struct fobj_nc
  81. : boost::noncopyable
  82. {
  83. // Handle nullary separately to exercise result_of support
  84. template <typename Sig>
  85. struct result;
  86. template <class Self, typename T0>
  87. struct result< Self(T0)>
  88. {
  89. typedef int type;
  90. };
  91. int operator()(int i) { return 14 + i; }
  92. int operator()(int i) const { return 15 + i; }
  93. };
  94. SFINAE_FRIENDLY_ASSERT((fusion::result_of::invoke<fobj_nc, sfinae_friendly::v0>));
  95. SFINAE_FRIENDLY_ASSERT((fusion::result_of::invoke<fobj_nc, sfinae_friendly::v1>));
  96. SFINAE_FRIENDLY_ASSERT((fusion::result_of::invoke<fobj_nc, sfinae_friendly::v2>));
  97. SFINAE_FRIENDLY_ASSERT((fusion::result_of::invoke<fobj_nc, sfinae_friendly::v3>));
  98. struct nullary_fobj_nc
  99. : boost::noncopyable
  100. {
  101. typedef int result_type;
  102. int operator()() { return 12; }
  103. int operator()() const { return 13; }
  104. };
  105. SFINAE_FRIENDLY_ASSERT((fusion::result_of::invoke<nullary_fobj_nc, sfinae_friendly::v1>));
  106. SFINAE_FRIENDLY_ASSERT((fusion::result_of::invoke<nullary_fobj_nc, sfinae_friendly::v2>));
  107. SFINAE_FRIENDLY_ASSERT((fusion::result_of::invoke<nullary_fobj_nc, sfinae_friendly::v3>));
  108. int nullary() { return 16; }
  109. int unary(int i) { return 17 + i; }
  110. int binary1(int i, object &) { return 18 + i; }
  111. int binary2(int i, object const &) { return 19 + i; }
  112. //FIXME
  113. //SFINAE_FRIENDLY_ASSERT((fusion::result_of::invoke<int(*)(), sfinae_friendly::v1>));
  114. //SFINAE_FRIENDLY_ASSERT((fusion::result_of::invoke<int(*)(), sfinae_friendly::v2>));
  115. //SFINAE_FRIENDLY_ASSERT((fusion::result_of::invoke<int(*)(), sfinae_friendly::v3>));
  116. //
  117. //SFINAE_FRIENDLY_ASSERT((fusion::result_of::invoke<int(*)(int), sfinae_friendly::v0>));
  118. //SFINAE_FRIENDLY_ASSERT((fusion::result_of::invoke<int(*)(int), sfinae_friendly::v1>));
  119. //SFINAE_FRIENDLY_ASSERT((fusion::result_of::invoke<int(*)(int), sfinae_friendly::v2>));
  120. //SFINAE_FRIENDLY_ASSERT((fusion::result_of::invoke<int(*)(int), sfinae_friendly::v3>));
  121. //
  122. //SFINAE_FRIENDLY_ASSERT((fusion::result_of::invoke<int(*)(int, object &), sfinae_friendly::v0>));
  123. //SFINAE_FRIENDLY_ASSERT((fusion::result_of::invoke<int(*)(int, object &), sfinae_friendly::v1>));
  124. //SFINAE_FRIENDLY_ASSERT((fusion::result_of::invoke<int(*)(int, object &), sfinae_friendly::v2>));
  125. //SFINAE_FRIENDLY_ASSERT((fusion::result_of::invoke<int(*)(int, object &), sfinae_friendly::v3>));
  126. //
  127. //SFINAE_FRIENDLY_ASSERT((fusion::result_of::invoke<int(*)(int, object const &), sfinae_friendly::v0>));
  128. //SFINAE_FRIENDLY_ASSERT((fusion::result_of::invoke<int(*)(int, object const &), sfinae_friendly::v1>));
  129. //SFINAE_FRIENDLY_ASSERT((fusion::result_of::invoke<int(*)(int, object const &), sfinae_friendly::v2>));
  130. //SFINAE_FRIENDLY_ASSERT((fusion::result_of::invoke<int(*)(int, object const &), sfinae_friendly::v3>));
  131. typedef int (* func_ptr)(int);
  132. typedef int (* const c_func_ptr)(int);
  133. typedef int (* volatile v_func_ptr)(int);
  134. typedef int (* const volatile cv_func_ptr)(int);
  135. func_ptr func_ptr1 = &unary;
  136. c_func_ptr func_ptr2 = &unary;
  137. v_func_ptr func_ptr3 = &unary;
  138. cv_func_ptr func_ptr4 = &unary;
  139. class members
  140. {
  141. public:
  142. int data;
  143. members()
  144. : data(20)
  145. { }
  146. int nullary() { return data + 1; }
  147. int nullary_c() const { return data + 2; }
  148. int unary(int i) { return data + 3 + i; }
  149. int unary_c(int i) const { return data + 4 + i; }
  150. int binary(int i, object) { return data + 5 + i; }
  151. int binary_c(int i, object) const { return data + 6 + i; }
  152. };
  153. #ifdef BOOST_NO_CXX11_SMART_PTR
  154. typedef std::auto_ptr<members > members_ptr;
  155. typedef std::auto_ptr<members const> const_members_ptr;
  156. #else
  157. typedef std::unique_ptr<members > members_ptr;
  158. typedef std::unique_ptr<members const> const_members_ptr;
  159. #endif
  160. struct derived
  161. : members
  162. {
  163. };
  164. #ifdef BOOST_NO_CXX11_SMART_PTR
  165. typedef std::auto_ptr<derived > derived_ptr;
  166. typedef std::auto_ptr<derived const> const_derived_ptr;
  167. #else
  168. typedef std::unique_ptr<derived > derived_ptr;
  169. typedef std::unique_ptr<derived const> const_derived_ptr;
  170. #endif
  171. typedef int element1_type;
  172. typedef object element2_type;
  173. typedef object_nc & element3_type;
  174. int element1 = 100;
  175. object element2 = object();
  176. object_nc element3;
  177. members that;
  178. members_ptr spt_that(new members);
  179. const_members_ptr spt_that_c(new members);
  180. typedef fusion::single_view<members > sv_obj;
  181. typedef fusion::single_view<members &> sv_ref;
  182. typedef fusion::single_view<members *> sv_ptr;
  183. typedef fusion::single_view<members const > sv_obj_c;
  184. typedef fusion::single_view<members const &> sv_ref_c;
  185. typedef fusion::single_view<members const *> sv_ptr_c;
  186. typedef fusion::single_view<members_ptr const &> sv_spt;
  187. typedef fusion::single_view<const_members_ptr const &> sv_spt_c;
  188. sv_obj sv_obj_ctx( that);
  189. sv_ref sv_ref_ctx( that);
  190. sv_ptr sv_ptr_ctx(& that);
  191. sv_obj_c sv_obj_c_ctx( that);
  192. sv_ref_c sv_ref_c_ctx( that);
  193. sv_ptr_c sv_ptr_c_ctx(& that);
  194. sv_spt sv_spt_ctx(spt_that);
  195. sv_spt_c sv_spt_c_ctx(spt_that_c);
  196. template <typename F, typename S>
  197. struct sv_helper
  198. {
  199. SFINAE_FRIENDLY_ASSERT((fusion::result_of::invoke<F, typename fusion::result_of::join<sv_obj , S>::type>));
  200. SFINAE_FRIENDLY_ASSERT((fusion::result_of::invoke<F, typename fusion::result_of::join<sv_ref , S>::type>));
  201. SFINAE_FRIENDLY_ASSERT((fusion::result_of::invoke<F, typename fusion::result_of::join<sv_ptr , S>::type>));
  202. SFINAE_FRIENDLY_ASSERT((fusion::result_of::invoke<F, typename fusion::result_of::join<sv_obj_c, S>::type>));
  203. SFINAE_FRIENDLY_ASSERT((fusion::result_of::invoke<F, typename fusion::result_of::join<sv_ref_c, S>::type>));
  204. SFINAE_FRIENDLY_ASSERT((fusion::result_of::invoke<F, typename fusion::result_of::join<sv_ptr_c, S>::type>));
  205. SFINAE_FRIENDLY_ASSERT((fusion::result_of::invoke<F, typename fusion::result_of::join<sv_spt , S>::type>));
  206. SFINAE_FRIENDLY_ASSERT((fusion::result_of::invoke<F, typename fusion::result_of::join<sv_spt_c, S>::type>));
  207. };
  208. // FIXME:
  209. //template struct sv_helper<int (members::*)() , sfinae_friendly::v1>;
  210. //template struct sv_helper<int (members::*)() , sfinae_friendly::v2>;
  211. //template struct sv_helper<int (members::*)() , sfinae_friendly::v3>;
  212. //template struct sv_helper<int (members::*)() const, sfinae_friendly::v1>;
  213. //template struct sv_helper<int (members::*)() const, sfinae_friendly::v2>;
  214. //template struct sv_helper<int (members::*)() const, sfinae_friendly::v3>;
  215. //template struct sv_helper<int (members::*)(int) , sfinae_friendly::v0>;
  216. //template struct sv_helper<int (members::*)(int) , sfinae_friendly::v1>;
  217. //template struct sv_helper<int (members::*)(int) , sfinae_friendly::v2>;
  218. //template struct sv_helper<int (members::*)(int) , sfinae_friendly::v3>;
  219. //template struct sv_helper<int (members::*)(int) const, sfinae_friendly::v0>;
  220. //template struct sv_helper<int (members::*)(int) const, sfinae_friendly::v1>;
  221. //template struct sv_helper<int (members::*)(int) const, sfinae_friendly::v2>;
  222. //template struct sv_helper<int (members::*)(int) const, sfinae_friendly::v3>;
  223. //template struct sv_helper<int (members::*)(int, object) , sfinae_friendly::v0>;
  224. //template struct sv_helper<int (members::*)(int, object) , sfinae_friendly::v1>;
  225. //template struct sv_helper<int (members::*)(int, object) , sfinae_friendly::v2>;
  226. //template struct sv_helper<int (members::*)(int, object) , sfinae_friendly::v3>;
  227. //template struct sv_helper<int (members::*)(int, object) const, sfinae_friendly::v0>;
  228. //template struct sv_helper<int (members::*)(int, object) const, sfinae_friendly::v1>;
  229. //template struct sv_helper<int (members::*)(int, object) const, sfinae_friendly::v2>;
  230. //template struct sv_helper<int (members::*)(int, object) const, sfinae_friendly::v3>;
  231. derived derived_that;
  232. derived_ptr spt_derived_that(new derived);
  233. const_derived_ptr spt_derived_that_c(new derived);
  234. typedef fusion::single_view<derived > sv_obj_d;
  235. typedef fusion::single_view<derived &> sv_ref_d;
  236. typedef fusion::single_view<derived *> sv_ptr_d;
  237. typedef fusion::single_view<derived const > sv_obj_c_d;
  238. typedef fusion::single_view<derived const &> sv_ref_c_d;
  239. typedef fusion::single_view<derived const *> sv_ptr_c_d;
  240. typedef fusion::single_view<derived_ptr const &> sv_spt_d;
  241. typedef fusion::single_view<const_derived_ptr const &> sv_spt_c_d;
  242. sv_obj_d sv_obj_d_ctx( derived_that);
  243. sv_ref_d sv_ref_d_ctx( derived_that);
  244. sv_ptr_d sv_ptr_d_ctx(& derived_that);
  245. sv_obj_c_d sv_obj_c_d_ctx( derived_that);
  246. sv_ref_c_d sv_ref_c_d_ctx( derived_that);
  247. sv_ptr_c_d sv_ptr_c_d_ctx(& derived_that);
  248. sv_spt_d sv_spt_d_ctx(spt_derived_that);
  249. sv_spt_c_d sv_spt_c_d_ctx(spt_derived_that_c);
  250. template <typename F, typename S>
  251. struct sv_d_helper
  252. {
  253. SFINAE_FRIENDLY_ASSERT((fusion::result_of::invoke<F, typename fusion::result_of::join<sv_obj_d , S>::type>));
  254. SFINAE_FRIENDLY_ASSERT((fusion::result_of::invoke<F, typename fusion::result_of::join<sv_ref_d , S>::type>));
  255. SFINAE_FRIENDLY_ASSERT((fusion::result_of::invoke<F, typename fusion::result_of::join<sv_ptr_d , S>::type>));
  256. SFINAE_FRIENDLY_ASSERT((fusion::result_of::invoke<F, typename fusion::result_of::join<sv_obj_c_d, S>::type>));
  257. SFINAE_FRIENDLY_ASSERT((fusion::result_of::invoke<F, typename fusion::result_of::join<sv_ref_c_d, S>::type>));
  258. SFINAE_FRIENDLY_ASSERT((fusion::result_of::invoke<F, typename fusion::result_of::join<sv_ptr_c_d, S>::type>));
  259. SFINAE_FRIENDLY_ASSERT((fusion::result_of::invoke<F, typename fusion::result_of::join<sv_spt_d , S>::type>));
  260. SFINAE_FRIENDLY_ASSERT((fusion::result_of::invoke<F, typename fusion::result_of::join<sv_spt_c_d, S>::type>));
  261. };
  262. // FIXME:
  263. //template struct sv_d_helper<int (members::*)() , sfinae_friendly::v1>;
  264. //template struct sv_d_helper<int (members::*)() , sfinae_friendly::v2>;
  265. //template struct sv_d_helper<int (members::*)() , sfinae_friendly::v3>;
  266. //template struct sv_d_helper<int (members::*)() const, sfinae_friendly::v1>;
  267. //template struct sv_d_helper<int (members::*)() const, sfinae_friendly::v2>;
  268. //template struct sv_d_helper<int (members::*)() const, sfinae_friendly::v3>;
  269. //template struct sv_d_helper<int (members::*)(int) , sfinae_friendly::v0>;
  270. //template struct sv_d_helper<int (members::*)(int) , sfinae_friendly::v1>;
  271. //template struct sv_d_helper<int (members::*)(int) , sfinae_friendly::v2>;
  272. //template struct sv_d_helper<int (members::*)(int) , sfinae_friendly::v3>;
  273. //template struct sv_d_helper<int (members::*)(int) const, sfinae_friendly::v0>;
  274. //template struct sv_d_helper<int (members::*)(int) const, sfinae_friendly::v1>;
  275. //template struct sv_d_helper<int (members::*)(int) const, sfinae_friendly::v2>;
  276. //template struct sv_d_helper<int (members::*)(int) const, sfinae_friendly::v3>;
  277. //template struct sv_d_helper<int (members::*)(int, object) , sfinae_friendly::v0>;
  278. //template struct sv_d_helper<int (members::*)(int, object) , sfinae_friendly::v1>;
  279. //template struct sv_d_helper<int (members::*)(int, object) , sfinae_friendly::v2>;
  280. //template struct sv_d_helper<int (members::*)(int, object) , sfinae_friendly::v3>;
  281. //template struct sv_d_helper<int (members::*)(int, object) const, sfinae_friendly::v0>;
  282. //template struct sv_d_helper<int (members::*)(int, object) const, sfinae_friendly::v1>;
  283. //template struct sv_d_helper<int (members::*)(int, object) const, sfinae_friendly::v2>;
  284. //template struct sv_d_helper<int (members::*)(int, object) const, sfinae_friendly::v3>;
  285. template <class Sequence>
  286. void test_sequence_n(Sequence & seq, mpl::int_<0>)
  287. {
  288. // Function Objects
  289. nullary_fobj f;
  290. BOOST_TEST(f () == fusion::invoke(f , seq ));
  291. BOOST_TEST(f () == fusion::invoke(f , const_(seq)));
  292. // Note: The function object is taken by value, so we request the copy
  293. // to be const with an explicit template argument. We can also request
  294. // the function object to be pased by reference...
  295. BOOST_TEST(const_(f)() == fusion::invoke<nullary_fobj const >(const_(f), seq ));
  296. BOOST_TEST(const_(f)() == fusion::invoke<nullary_fobj const &>(const_(f), const_(seq)));
  297. nullary_fobj_nc nc_f;
  298. // ...and we further ensure there is no copying in this case, using a
  299. // noncopyable function object.
  300. BOOST_TEST(nc_f () == fusion::invoke<nullary_fobj_nc &>(nc_f , seq ));
  301. BOOST_TEST(nc_f () == fusion::invoke<nullary_fobj_nc &>(nc_f , const_(seq)));
  302. BOOST_TEST(const_(nc_f)() == fusion::invoke<nullary_fobj_nc const &>(const_(nc_f), seq ));
  303. BOOST_TEST(const_(nc_f)() == fusion::invoke<nullary_fobj_nc const &>(const_(nc_f), const_(seq)));
  304. // Builtin Functions
  305. // Call through ref/ptr to function
  306. BOOST_TEST(nullary() == fusion::invoke<int (&)()>(nullary, seq));
  307. BOOST_TEST(nullary() == fusion::invoke(& nullary, seq));
  308. // Call through ptr to member function
  309. // Note: The non-const function members::nullary can't be invoked with
  310. // fusion::join(sv_obj_ctx,seq)), which is const and so is its first element
  311. BOOST_TEST(that.nullary() == fusion::invoke(& members::nullary, fusion::join(sv_ref_ctx,seq)));
  312. BOOST_TEST(that.nullary() == fusion::invoke(& members::nullary, fusion::join(sv_ptr_ctx,seq)));
  313. BOOST_TEST(that.nullary() == fusion::invoke(& members::nullary, fusion::join(sv_spt_ctx,seq)));
  314. BOOST_TEST(that.nullary_c() == fusion::invoke(& members::nullary_c, fusion::join(sv_obj_ctx,seq)));
  315. BOOST_TEST(that.nullary_c() == fusion::invoke(& members::nullary_c, fusion::join(sv_ref_ctx,seq)));
  316. BOOST_TEST(that.nullary_c() == fusion::invoke(& members::nullary_c, fusion::join(sv_ptr_ctx,seq)));
  317. BOOST_TEST(that.nullary_c() == fusion::invoke(& members::nullary_c, fusion::join(sv_spt_ctx,seq)));
  318. BOOST_TEST(that.nullary_c() == fusion::invoke(& members::nullary_c, fusion::join(sv_obj_c_ctx,seq)));
  319. BOOST_TEST(that.nullary_c() == fusion::invoke(& members::nullary_c, fusion::join(sv_ref_c_ctx,seq)));
  320. BOOST_TEST(that.nullary_c() == fusion::invoke(& members::nullary_c, fusion::join(sv_ptr_c_ctx,seq)));
  321. BOOST_TEST(that.nullary_c() == fusion::invoke(& members::nullary_c, fusion::join(sv_spt_c_ctx,seq)));
  322. // Pointer to data member
  323. // $$$ JDG $$$ disabling this test due to C++11 error: assignment of read-only location
  324. //~ BOOST_TEST(that.data == (fusion::invoke(& members::data, fusion::join(sv_obj_ctx,seq)) = that.data));
  325. BOOST_TEST(that.data == (fusion::invoke(& members::data, fusion::join(sv_ref_ctx,seq)) = that.data));
  326. BOOST_TEST(that.data == (fusion::invoke(& members::data, fusion::join(sv_ptr_ctx,seq)) = that.data));
  327. BOOST_TEST(that.data == (fusion::invoke(& members::data, fusion::join(sv_spt_ctx,seq)) = that.data));
  328. BOOST_TEST(that.data == fusion::invoke(& members::data, fusion::join(sv_obj_c_ctx,seq)));
  329. BOOST_TEST(that.data == fusion::invoke(& members::data, fusion::join(sv_ref_c_ctx,seq)));
  330. BOOST_TEST(that.data == fusion::invoke(& members::data, fusion::join(sv_ptr_c_ctx,seq)));
  331. BOOST_TEST(that.data == fusion::invoke(& members::data, fusion::join(sv_spt_c_ctx,seq)));
  332. // $$$ JDG $$$ disabling this test due to C++11 error: assignment of read-only location
  333. //~ BOOST_TEST(that.data == (fusion::invoke(& members::data, fusion::join(sv_obj_d_ctx,seq)) = that.data));
  334. BOOST_TEST(that.data == (fusion::invoke(& members::data, fusion::join(sv_ref_d_ctx,seq)) = that.data));
  335. BOOST_TEST(that.data == (fusion::invoke(& members::data, fusion::join(sv_ptr_d_ctx,seq)) = that.data));
  336. BOOST_TEST(that.data == (fusion::invoke(& members::data, fusion::join(sv_spt_d_ctx,seq)) = that.data));
  337. BOOST_TEST(that.data == fusion::invoke(& members::data, fusion::join(sv_obj_c_d_ctx,seq)));
  338. BOOST_TEST(that.data == fusion::invoke(& members::data, fusion::join(sv_ref_c_d_ctx,seq)));
  339. BOOST_TEST(that.data == fusion::invoke(& members::data, fusion::join(sv_ptr_c_d_ctx,seq)));
  340. BOOST_TEST(that.data == fusion::invoke(& members::data, fusion::join(sv_spt_c_d_ctx,seq)));
  341. }
  342. template <class Sequence>
  343. void test_sequence_n(Sequence & seq, mpl::int_<1>)
  344. {
  345. fobj f;
  346. BOOST_TEST(f(element1) == fusion::invoke(f , seq ));
  347. BOOST_TEST(f(element1) == fusion::invoke(f , const_(seq)));
  348. BOOST_TEST(const_(f)(element1) == fusion::invoke<fobj const >(const_(f), seq ));
  349. BOOST_TEST(const_(f)(element1) == fusion::invoke<fobj const &>(const_(f), const_(seq)));
  350. fobj_nc nc_f;
  351. BOOST_TEST(nc_f(element1) == fusion::invoke<fobj_nc &>(nc_f, seq ));
  352. BOOST_TEST(nc_f(element1) == fusion::invoke<fobj_nc &>(nc_f, const_(seq)));
  353. BOOST_TEST(const_(nc_f)(element1) == fusion::invoke<fobj_nc const &>(const_(nc_f), seq ));
  354. BOOST_TEST(const_(nc_f)(element1) == fusion::invoke<fobj_nc const &>(const_(nc_f), const_(seq)));
  355. BOOST_TEST(unary(element1) == fusion::invoke<int (&)(int)>(unary, seq));
  356. BOOST_TEST(func_ptr1(element1) == fusion::invoke(func_ptr1, seq));
  357. BOOST_TEST(func_ptr2(element1) == fusion::invoke(func_ptr2, seq));
  358. BOOST_TEST(func_ptr3(element1) == fusion::invoke(func_ptr3, seq));
  359. BOOST_TEST(func_ptr4(element1) == fusion::invoke(func_ptr4, seq));
  360. BOOST_TEST(that.unary(element1) == fusion::invoke(& members::unary, fusion::join(sv_ref_ctx,seq)));
  361. BOOST_TEST(that.unary(element1) == fusion::invoke(& members::unary, fusion::join(sv_ptr_ctx,seq)));
  362. BOOST_TEST(that.unary(element1) == fusion::invoke(& members::unary, fusion::join(sv_spt_ctx,seq)));
  363. BOOST_TEST(that.unary_c(element1) == fusion::invoke(& members::unary_c, fusion::join(sv_obj_ctx,seq)));
  364. BOOST_TEST(that.unary_c(element1) == fusion::invoke(& members::unary_c, fusion::join(sv_ref_ctx,seq)));
  365. BOOST_TEST(that.unary_c(element1) == fusion::invoke(& members::unary_c, fusion::join(sv_ptr_ctx,seq)));
  366. BOOST_TEST(that.unary_c(element1) == fusion::invoke(& members::unary_c, fusion::join(sv_spt_ctx,seq)));
  367. BOOST_TEST(that.unary_c(element1) == fusion::invoke(& members::unary_c, fusion::join(sv_obj_c_ctx,seq)));
  368. BOOST_TEST(that.unary_c(element1) == fusion::invoke(& members::unary_c, fusion::join(sv_ref_c_ctx,seq)));
  369. BOOST_TEST(that.unary_c(element1) == fusion::invoke(& members::unary_c, fusion::join(sv_ptr_c_ctx,seq)));
  370. BOOST_TEST(that.unary_c(element1) == fusion::invoke(& members::unary_c, fusion::join(sv_spt_c_ctx,seq)));
  371. BOOST_TEST(that.unary(element1) == fusion::invoke(& members::unary, fusion::join(sv_ref_d_ctx,seq)));
  372. BOOST_TEST(that.unary(element1) == fusion::invoke(& members::unary, fusion::join(sv_ptr_d_ctx,seq)));
  373. BOOST_TEST(that.unary(element1) == fusion::invoke(& members::unary, fusion::join(sv_spt_d_ctx,seq)));
  374. BOOST_TEST(that.unary_c(element1) == fusion::invoke(& members::unary_c, fusion::join(sv_obj_d_ctx,seq)));
  375. BOOST_TEST(that.unary_c(element1) == fusion::invoke(& members::unary_c, fusion::join(sv_ref_d_ctx,seq)));
  376. BOOST_TEST(that.unary_c(element1) == fusion::invoke(& members::unary_c, fusion::join(sv_ptr_d_ctx,seq)));
  377. BOOST_TEST(that.unary_c(element1) == fusion::invoke(& members::unary_c, fusion::join(sv_spt_d_ctx,seq)));
  378. BOOST_TEST(that.unary_c(element1) == fusion::invoke(& members::unary_c, fusion::join(sv_obj_c_d_ctx,seq)));
  379. BOOST_TEST(that.unary_c(element1) == fusion::invoke(& members::unary_c, fusion::join(sv_ref_c_d_ctx,seq)));
  380. BOOST_TEST(that.unary_c(element1) == fusion::invoke(& members::unary_c, fusion::join(sv_ptr_c_d_ctx,seq)));
  381. BOOST_TEST(that.unary_c(element1) == fusion::invoke(& members::unary_c, fusion::join(sv_spt_c_d_ctx,seq)));
  382. }
  383. template <class Sequence>
  384. void test_sequence_n(Sequence & seq, mpl::int_<2>)
  385. {
  386. fobj f;
  387. BOOST_TEST(f (element1, element2) == fusion::invoke(f , seq));
  388. BOOST_TEST(f (element1, const_(element2)) == fusion::invoke(f , const_(seq)));
  389. BOOST_TEST(const_(f)(element1, element2) == fusion::invoke<fobj const>(const_(f), seq));
  390. BOOST_TEST(const_(f)(element1, const_(element2)) == fusion::invoke<fobj const>(const_(f), const_(seq)));
  391. BOOST_TEST(binary1(element1, element2) == fusion::invoke(binary1, seq));
  392. BOOST_TEST(binary2(element1, element2) == fusion::invoke(binary2, seq));
  393. BOOST_TEST(that.binary(element1,element2) == fusion::invoke(& members::binary, fusion::join(sv_ref_ctx,seq)));
  394. BOOST_TEST(that.binary(element1,element2) == fusion::invoke(& members::binary, fusion::join(sv_ptr_ctx,seq)));
  395. BOOST_TEST(that.binary(element1,element2) == fusion::invoke(& members::binary, fusion::join(sv_spt_ctx,seq)));
  396. BOOST_TEST(that.binary_c(element1,element2) == fusion::invoke(& members::binary_c, fusion::join(sv_obj_ctx,seq)));
  397. BOOST_TEST(that.binary_c(element1,element2) == fusion::invoke(& members::binary_c, fusion::join(sv_ref_ctx,seq)));
  398. BOOST_TEST(that.binary_c(element1,element2) == fusion::invoke(& members::binary_c, fusion::join(sv_ptr_ctx,seq)));
  399. BOOST_TEST(that.binary_c(element1,element2) == fusion::invoke(& members::binary_c, fusion::join(sv_spt_ctx,seq)));
  400. BOOST_TEST(that.binary_c(element1,element2) == fusion::invoke(& members::binary_c, fusion::join(sv_obj_c_ctx,seq)));
  401. BOOST_TEST(that.binary_c(element1,element2) == fusion::invoke(& members::binary_c, fusion::join(sv_ref_c_ctx,seq)));
  402. BOOST_TEST(that.binary_c(element1,element2) == fusion::invoke(& members::binary_c, fusion::join(sv_ptr_c_ctx,seq)));
  403. BOOST_TEST(that.binary_c(element1,element2) == fusion::invoke(& members::binary_c, fusion::join(sv_spt_c_ctx,seq)));
  404. }
  405. template <class Sequence>
  406. void test_sequence_n(Sequence & seq, mpl::int_<3>)
  407. {
  408. fobj f;
  409. BOOST_TEST(f(element1, element2, element3) == fusion::invoke(f, seq));
  410. BOOST_TEST(const_(f)(element1, element2, element3) == fusion::invoke<fobj const>(const_(f), seq));
  411. }
  412. template <class Sequence>
  413. void test_sequence(Sequence & seq)
  414. {
  415. test_sequence_n(seq, mpl::int_<boost::fusion::result_of::size<Sequence>::value>());
  416. }
  417. void result_type_tests()
  418. {
  419. using boost::is_same;
  420. BOOST_TEST(( is_same<
  421. boost::fusion::result_of::invoke<int (*)(), fusion::vector0<> >::type, int
  422. >::value ));
  423. // disabled until boost::result_of supports it
  424. // BOOST_TEST(( is_same<
  425. // boost::fusion::result_of::invoke<int (*)(...), fusion::vector1<int> >::type, int
  426. // >::value ));
  427. BOOST_TEST(( is_same<
  428. boost::fusion::result_of::invoke<int (members::*)(), fusion::vector1<members*> >::type, int
  429. >::value ));
  430. // disabled until boost::result_of supports it
  431. // BOOST_TEST(( is_same<
  432. // boost::fusion::result_of::invoke<int (members::*)(...), fusion::vector2<members*,int> >::type, int
  433. // >::value ));
  434. }
  435. int main()
  436. {
  437. result_type_tests();
  438. typedef fusion::vector<> vector0;
  439. typedef fusion::vector<element1_type> vector1;
  440. typedef fusion::vector<element1_type, element2_type> vector2;
  441. typedef fusion::vector<element1_type, element2_type, element3_type> vector3;
  442. vector0 v0;
  443. vector1 v1(element1);
  444. vector2 v2(element1, element2);
  445. #if !defined(BOOST_NO_CXX11_RVALUE_REFERENCES)
  446. // Note: C++11 will pickup the rvalue overload for the d argument
  447. // since we do not have all permutations (expensive!) for all const&
  448. // and && arguments. We either have all && or all const& arguments only.
  449. // For that matter, use std::ref to disambiguate the call.
  450. vector3 v3(element1, element2, std::ref(element3));
  451. #else
  452. vector3 v3(element1, element2, element3);
  453. #endif
  454. test_sequence(v0);
  455. test_sequence(v1);
  456. test_sequence(v2);
  457. test_sequence(v3);
  458. typedef fusion::list<> list0;
  459. typedef fusion::list<element1_type> list1;
  460. typedef fusion::list<element1_type, element2_type> list2;
  461. typedef fusion::list<element1_type, element2_type, element3_type> list3;
  462. list0 l0;
  463. list1 l1(element1);
  464. list2 l2(element1, element2);
  465. list3 l3(element1, element2, element3);
  466. test_sequence(l0);
  467. test_sequence(l1);
  468. test_sequence(l2);
  469. test_sequence(l3);
  470. return boost::report_errors();
  471. }