unpack.cpp 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255
  1. /*=============================================================================
  2. Copyright (c) 2017 Paul Fultz II
  3. unpack.cpp
  4. Distributed under the Boost Software License, Version 1.0. (See accompanying
  5. file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
  6. ==============================================================================*/
  7. #include <boost/hof/unpack.hpp>
  8. #include <boost/hof/static.hpp>
  9. #include <boost/hof/lambda.hpp>
  10. #include "test.hpp"
  11. #include <memory>
  12. static constexpr boost::hof::static_<boost::hof::unpack_adaptor<unary_class> > unary_unpack = {};
  13. static constexpr boost::hof::static_<boost::hof::unpack_adaptor<binary_class> > binary_unpack = {};
  14. BOOST_HOF_STATIC_AUTO unary_unpack_constexpr = boost::hof::unpack_adaptor<unary_class>();
  15. #if BOOST_HOF_HAS_CONSTEXPR_TUPLE
  16. BOOST_HOF_STATIC_AUTO binary_unpack_constexpr = boost::hof::unpack_adaptor<binary_class>();
  17. #endif
  18. BOOST_HOF_STATIC_AUTO unary_unpack_reveal = boost::hof::reveal_adaptor<boost::hof::unpack_adaptor<unary_class>>();
  19. BOOST_HOF_STATIC_AUTO binary_unpack_reveal = boost::hof::reveal_adaptor<boost::hof::unpack_adaptor<binary_class>>();
  20. #if BOOST_HOF_HAS_NOEXCEPT_DEDUCTION
  21. BOOST_HOF_TEST_CASE()
  22. {
  23. static_assert(noexcept(boost::hof::unpack(unary_class())(boost::hof::pack(3))), "noexcept unpack");
  24. static_assert(noexcept(unary_unpack(boost::hof::pack(3))), "noexcept unpack");
  25. static_assert(noexcept(binary_unpack(boost::hof::pack(3), boost::hof::pack(2))), "noexcept unpack");
  26. }
  27. #endif
  28. BOOST_HOF_TEST_CASE()
  29. {
  30. BOOST_HOF_TEST_CHECK(3 == boost::hof::unpack(unary_class())(std::make_tuple(3)));
  31. BOOST_HOF_TEST_CHECK(3 == unary_unpack(std::make_tuple(3)));
  32. BOOST_HOF_TEST_CHECK(3 == unary_unpack_reveal(std::make_tuple(3)));
  33. int ifu = 3;
  34. BOOST_HOF_TEST_CHECK(3 == unary_unpack(std::tuple<int&>(ifu)));
  35. #if BOOST_HOF_HAS_CONSTEXPR_TUPLE
  36. BOOST_HOF_STATIC_TEST_CHECK(3 == boost::hof::unpack(unary_class())(std::make_tuple(3)));
  37. BOOST_HOF_STATIC_TEST_CHECK(3 == unary_unpack_constexpr(std::make_tuple(3)));
  38. BOOST_HOF_STATIC_TEST_CHECK(3 == unary_unpack_reveal(std::make_tuple(3)));
  39. #endif
  40. }
  41. BOOST_HOF_TEST_CASE()
  42. {
  43. BOOST_HOF_TEST_CHECK(3 == boost::hof::unpack(unary_class())(boost::hof::pack(3)));
  44. BOOST_HOF_TEST_CHECK(3 == unary_unpack(boost::hof::pack(3)));
  45. BOOST_HOF_TEST_CHECK(3 == unary_unpack_reveal(boost::hof::pack(3)));
  46. int ifu = 3;
  47. BOOST_HOF_TEST_CHECK(3 == unary_unpack(boost::hof::pack_forward(ifu)));
  48. BOOST_HOF_STATIC_TEST_CHECK(3 == boost::hof::unpack(unary_class())(boost::hof::pack(3)));
  49. BOOST_HOF_STATIC_TEST_CHECK(3 == unary_unpack_constexpr(boost::hof::pack(3)));
  50. BOOST_HOF_STATIC_TEST_CHECK(3 == unary_unpack_reveal(boost::hof::pack(3)));
  51. }
  52. BOOST_HOF_TEST_CASE()
  53. {
  54. BOOST_HOF_TEST_CHECK(3 == boost::hof::unpack(binary_class())(std::make_tuple(1, 2)));
  55. BOOST_HOF_TEST_CHECK(3 == binary_unpack(std::make_tuple(1, 2)));
  56. BOOST_HOF_TEST_CHECK(3 == binary_unpack_reveal(std::make_tuple(1, 2)));
  57. BOOST_HOF_TEST_CHECK(3 == boost::hof::unpack(binary_class())(std::make_tuple(1), std::make_tuple(2)));
  58. BOOST_HOF_TEST_CHECK(3 == binary_unpack(std::make_tuple(1), std::make_tuple(2)));
  59. BOOST_HOF_TEST_CHECK(3 == binary_unpack_reveal(std::make_tuple(1), std::make_tuple(2)));
  60. BOOST_HOF_TEST_CHECK(3 == boost::hof::unpack(binary_class())(std::make_tuple(1), std::make_tuple(), std::make_tuple(2)));
  61. BOOST_HOF_TEST_CHECK(3 == binary_unpack(std::make_tuple(1), std::make_tuple(), std::make_tuple(2)));
  62. BOOST_HOF_TEST_CHECK(3 == binary_unpack_reveal(std::make_tuple(1), std::make_tuple(), std::make_tuple(2)));
  63. BOOST_HOF_TEST_CHECK(3 == boost::hof::unpack(binary_class())(std::make_tuple(), std::make_tuple(1), std::make_tuple(), std::make_tuple(2)));
  64. BOOST_HOF_TEST_CHECK(3 == binary_unpack(std::make_tuple(), std::make_tuple(1), std::make_tuple(), std::make_tuple(2)));
  65. BOOST_HOF_TEST_CHECK(3 == binary_unpack_reveal(std::make_tuple(), std::make_tuple(1), std::make_tuple(), std::make_tuple(2)));
  66. BOOST_HOF_TEST_CHECK(3 == boost::hof::unpack(binary_class())(std::make_tuple(1), std::make_tuple(), std::make_tuple(2), std::make_tuple()));
  67. BOOST_HOF_TEST_CHECK(3 == binary_unpack(std::make_tuple(1), std::make_tuple(), std::make_tuple(2), std::make_tuple()));
  68. BOOST_HOF_TEST_CHECK(3 == binary_unpack_reveal(std::make_tuple(1), std::make_tuple(), std::make_tuple(2), std::make_tuple()));
  69. #if BOOST_HOF_HAS_CONSTEXPR_TUPLE
  70. BOOST_HOF_STATIC_TEST_CHECK(3 == boost::hof::unpack(binary_class())(std::make_tuple(1, 2)));
  71. BOOST_HOF_STATIC_TEST_CHECK(3 == binary_unpack_constexpr(std::make_tuple(1, 2)));
  72. BOOST_HOF_STATIC_TEST_CHECK(3 == binary_unpack_reveal(std::make_tuple(1, 2)));
  73. BOOST_HOF_STATIC_TEST_CHECK(3 == boost::hof::unpack(binary_class())(std::make_tuple(1), std::make_tuple(2)));
  74. BOOST_HOF_STATIC_TEST_CHECK(3 == binary_unpack_constexpr(std::make_tuple(1), std::make_tuple(2)));
  75. BOOST_HOF_STATIC_TEST_CHECK(3 == binary_unpack_reveal(std::make_tuple(1), std::make_tuple(2)));
  76. BOOST_HOF_STATIC_TEST_CHECK(3 == boost::hof::unpack(binary_class())(std::make_tuple(1), std::make_tuple(), std::make_tuple(2)));
  77. BOOST_HOF_STATIC_TEST_CHECK(3 == binary_unpack_constexpr(std::make_tuple(1), std::make_tuple(), std::make_tuple(2)));
  78. BOOST_HOF_STATIC_TEST_CHECK(3 == binary_unpack_reveal(std::make_tuple(1), std::make_tuple(), std::make_tuple(2)));
  79. BOOST_HOF_STATIC_TEST_CHECK(3 == boost::hof::unpack(binary_class())(std::make_tuple(), std::make_tuple(1), std::make_tuple(), std::make_tuple(2)));
  80. BOOST_HOF_STATIC_TEST_CHECK(3 == binary_unpack_constexpr(std::make_tuple(), std::make_tuple(1), std::make_tuple(), std::make_tuple(2)));
  81. BOOST_HOF_STATIC_TEST_CHECK(3 == binary_unpack_reveal(std::make_tuple(), std::make_tuple(1), std::make_tuple(), std::make_tuple(2)));
  82. BOOST_HOF_STATIC_TEST_CHECK(3 == boost::hof::unpack(binary_class())(std::make_tuple(1), std::make_tuple(), std::make_tuple(2), std::make_tuple()));
  83. BOOST_HOF_STATIC_TEST_CHECK(3 == binary_unpack_constexpr(std::make_tuple(1), std::make_tuple(), std::make_tuple(2), std::make_tuple()));
  84. BOOST_HOF_STATIC_TEST_CHECK(3 == binary_unpack_reveal(std::make_tuple(1), std::make_tuple(), std::make_tuple(2), std::make_tuple()));
  85. #endif
  86. }
  87. BOOST_HOF_TEST_CASE()
  88. {
  89. auto p1 = boost::hof::pack_basic(1, 2);
  90. static_assert(boost::hof::is_unpackable<decltype(p1)>::value, "Not unpackable");
  91. static_assert(boost::hof::is_unpackable<decltype((p1))>::value, "Not unpackable");
  92. auto p2 = boost::hof::pack_forward(1, 2);
  93. static_assert(boost::hof::is_unpackable<decltype(p2)>::value, "Not unpackable");
  94. static_assert(boost::hof::is_unpackable<decltype((p2))>::value, "Not unpackable");
  95. auto p3 = boost::hof::pack(1, 2);
  96. static_assert(boost::hof::is_unpackable<decltype(p3)>::value, "Not unpackable");
  97. static_assert(boost::hof::is_unpackable<decltype((p3))>::value, "Not unpackable");
  98. static_assert(boost::hof::is_unpackable<std::tuple<int>>::value, "Not unpackable");
  99. static_assert(!boost::hof::is_unpackable<int>::value, "Unpackable");
  100. static_assert(!boost::hof::is_unpackable<void>::value, "Unpackable");
  101. }
  102. BOOST_HOF_TEST_CASE()
  103. {
  104. typedef std::tuple<int, int> tuple_type;
  105. static_assert(boost::hof::is_unpackable<tuple_type>::value, "Not unpackable");
  106. static_assert(boost::hof::is_unpackable<tuple_type&>::value, "Not unpackable");
  107. static_assert(boost::hof::is_unpackable<const tuple_type&>::value, "Not unpackable");
  108. static_assert(boost::hof::is_unpackable<tuple_type&&>::value, "Not unpackable");
  109. }
  110. BOOST_HOF_STATIC_AUTO lambda_unary_unpack = boost::hof::unpack(BOOST_HOF_STATIC_LAMBDA(int x)
  111. {
  112. return x;
  113. });
  114. BOOST_HOF_TEST_CASE()
  115. {
  116. BOOST_HOF_TEST_CHECK(3 == lambda_unary_unpack(std::make_tuple(3)));
  117. }
  118. BOOST_HOF_TEST_CASE()
  119. {
  120. BOOST_HOF_TEST_CHECK(3 == lambda_unary_unpack(boost::hof::pack(3)));
  121. }
  122. struct unary_move
  123. {
  124. std::unique_ptr<int> i;
  125. unary_move()
  126. : i(new int(2))
  127. {}
  128. template<class T>
  129. T operator()(T x) const
  130. {
  131. return x + *i;
  132. }
  133. };
  134. static constexpr boost::hof::static_<boost::hof::unpack_adaptor<unary_move> > unary_move_unpack = {};
  135. BOOST_HOF_TEST_CASE()
  136. {
  137. BOOST_HOF_TEST_CHECK(3 == boost::hof::unpack(unary_move())(std::make_tuple(1)));
  138. BOOST_HOF_TEST_CHECK(3 == unary_move_unpack(std::make_tuple(1)));
  139. }
  140. BOOST_HOF_TEST_CASE()
  141. {
  142. BOOST_HOF_TEST_CHECK(3 == boost::hof::unpack(unary_move())(boost::hof::pack(1)));
  143. BOOST_HOF_TEST_CHECK(3 == unary_move_unpack(boost::hof::pack(1)));
  144. }
  145. struct indirect_sum_f
  146. {
  147. template<class T, class U>
  148. auto operator()(T x, U y) const
  149. BOOST_HOF_RETURNS(*x + *y);
  150. };
  151. #define MAKE_UNIQUE_PTR(x) std::unique_ptr<int>(new int(x))
  152. BOOST_HOF_TEST_CASE()
  153. {
  154. BOOST_HOF_TEST_CHECK(3 == boost::hof::unpack(indirect_sum_f())(boost::hof::pack_basic(MAKE_UNIQUE_PTR(1), MAKE_UNIQUE_PTR(2))));
  155. BOOST_HOF_TEST_CHECK(3 == boost::hof::unpack(indirect_sum_f())(boost::hof::pack_forward(MAKE_UNIQUE_PTR(1), MAKE_UNIQUE_PTR(2))));
  156. BOOST_HOF_TEST_CHECK(3 == boost::hof::unpack(indirect_sum_f())(boost::hof::pack(MAKE_UNIQUE_PTR(1), MAKE_UNIQUE_PTR(2))));
  157. BOOST_HOF_TEST_CHECK(3 == boost::hof::unpack(indirect_sum_f())(std::make_tuple(MAKE_UNIQUE_PTR(1), MAKE_UNIQUE_PTR(2))));
  158. }
  159. template<class...>
  160. struct deduce_types
  161. {};
  162. struct deducer
  163. {
  164. template<class... Ts>
  165. deduce_types<Ts...> operator()(Ts&&...) const;
  166. };
  167. static constexpr boost::hof::unpack_adaptor<deducer> deduce = {};
  168. BOOST_HOF_TEST_CASE()
  169. {
  170. STATIC_ASSERT_SAME(deduce_types<int, int>, decltype(deduce(std::make_tuple(1, 2))));
  171. STATIC_ASSERT_SAME(deduce_types<int, int>, decltype(deduce(std::make_tuple(1), std::make_tuple(2))));
  172. STATIC_ASSERT_SAME(deduce_types<int, int, int>, decltype(deduce(std::make_tuple(1), std::make_tuple(2), std::make_tuple(3))));
  173. STATIC_ASSERT_SAME(std::tuple<int&&, int&&>, decltype(std::forward_as_tuple(1, 2)));
  174. // Disable this test, it seems that rvalue references get swalllowed by type deduction
  175. // STATIC_ASSERT_SAME(deduce_types<int&&, int&&>, decltype(deduce(std::forward_as_tuple(1, 2))));
  176. STATIC_ASSERT_SAME(deduce_types<int, int>, decltype(deduce(boost::hof::pack_basic(1, 2))));
  177. STATIC_ASSERT_SAME(deduce_types<int, int>, decltype(deduce(boost::hof::pack_basic(1), boost::hof::pack_basic(2))));
  178. STATIC_ASSERT_SAME(deduce_types<int, int, int>, decltype(deduce(boost::hof::pack_basic(1), boost::hof::pack_basic(2), boost::hof::pack_basic(3))));
  179. // STATIC_ASSERT_SAME(deduce_types<int&&, int&&>, decltype(deduce(boost::hof::pack_forward(1, 2))));
  180. }
  181. struct not_unpackable
  182. {};
  183. BOOST_HOF_TEST_CASE()
  184. {
  185. auto f = boost::hof::unpack(boost::hof::always(1));
  186. static_assert(!boost::hof::is_invocable<decltype(f), not_unpackable>::value, "SFINAE for unpack failed");
  187. }
  188. struct simple_unpackable
  189. {};
  190. namespace boost { namespace hof {
  191. template<>
  192. struct unpack_sequence<simple_unpackable>
  193. {
  194. template<class F, class S>
  195. constexpr static auto apply(F&& f, S&&) BOOST_HOF_RETURNS
  196. (
  197. f(1)
  198. );
  199. };
  200. }} // namespace boost::hof
  201. BOOST_HOF_TEST_CASE()
  202. {
  203. BOOST_HOF_TEST_CHECK(boost::hof::unpack(boost::hof::identity)(simple_unpackable{}) == 1);
  204. BOOST_HOF_STATIC_TEST_CHECK(boost::hof::unpack(boost::hof::identity)(simple_unpackable{}) == 1);
  205. }