basic_tuple.hpp 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299
  1. /*!
  2. @file
  3. Defines `boost::hana::basic_tuple`.
  4. @copyright Louis Dionne 2013-2016
  5. Distributed under the Boost Software License, Version 1.0.
  6. (See accompanying file LICENSE.md or copy at http://boost.org/LICENSE_1_0.txt)
  7. */
  8. #ifndef BOOST_HANA_BASIC_TUPLE_HPP
  9. #define BOOST_HANA_BASIC_TUPLE_HPP
  10. #include <boost/hana/fwd/basic_tuple.hpp>
  11. #include <boost/hana/config.hpp>
  12. #include <boost/hana/detail/decay.hpp>
  13. #include <boost/hana/detail/intrinsics.hpp>
  14. #include <boost/hana/fwd/at.hpp>
  15. #include <boost/hana/fwd/bool.hpp>
  16. #include <boost/hana/fwd/concept/sequence.hpp>
  17. #include <boost/hana/fwd/core/make.hpp>
  18. #include <boost/hana/fwd/core/tag_of.hpp>
  19. #include <boost/hana/fwd/drop_front.hpp>
  20. #include <boost/hana/fwd/is_empty.hpp>
  21. #include <boost/hana/fwd/transform.hpp>
  22. #include <boost/hana/fwd/unpack.hpp>
  23. #if 0 //! @todo Until we strip down headers, this includes too much
  24. #include <boost/hana/fwd/integral_constant.hpp>
  25. #include <boost/hana/fwd/length.hpp>
  26. #endif
  27. #include <cstddef>
  28. #include <type_traits>
  29. #include <utility>
  30. BOOST_HANA_NAMESPACE_BEGIN
  31. namespace detail {
  32. //////////////////////////////////////////////////////////////////////
  33. // elt<n, Xn>
  34. //
  35. // `elt` stands for `tuple_element`; the name is compressed to reduce
  36. // symbol lengths.
  37. //
  38. // Wrapper holding the actual elements of a tuple. It takes care of
  39. // optimizing the storage for empty types.
  40. //
  41. // When available, we use compiler intrinsics to reduce the number
  42. // of instantiations.
  43. //////////////////////////////////////////////////////////////////////
  44. template <std::size_t n, typename Xn, bool =
  45. BOOST_HANA_TT_IS_EMPTY(Xn) && !BOOST_HANA_TT_IS_FINAL(Xn)
  46. >
  47. struct elt;
  48. // Specialize storage for empty types
  49. template <std::size_t n, typename Xn>
  50. struct elt<n, Xn, true> : Xn {
  51. constexpr elt() = default;
  52. template <typename Yn>
  53. explicit constexpr elt(Yn&& yn)
  54. : Xn(static_cast<Yn&&>(yn))
  55. { }
  56. };
  57. // Specialize storage for non-empty types
  58. template <std::size_t n, typename Xn>
  59. struct elt<n, Xn, false> {
  60. constexpr elt() = default;
  61. template <typename Yn>
  62. explicit constexpr elt(Yn&& yn)
  63. : data_(static_cast<Yn&&>(yn))
  64. { }
  65. Xn data_;
  66. };
  67. }
  68. //////////////////////////////////////////////////////////////////////////
  69. // get_impl
  70. //////////////////////////////////////////////////////////////////////////
  71. template <std::size_t n, typename Xn>
  72. constexpr Xn const& get_impl(detail::elt<n, Xn, true> const& xn)
  73. { return xn; }
  74. template <std::size_t n, typename Xn>
  75. constexpr Xn& get_impl(detail::elt<n, Xn, true>& xn)
  76. { return xn; }
  77. template <std::size_t n, typename Xn>
  78. constexpr Xn&& get_impl(detail::elt<n, Xn, true>&& xn)
  79. { return static_cast<Xn&&>(xn); }
  80. template <std::size_t n, typename Xn>
  81. constexpr Xn const& get_impl(detail::elt<n, Xn, false> const& xn)
  82. { return xn.data_; }
  83. template <std::size_t n, typename Xn>
  84. constexpr Xn& get_impl(detail::elt<n, Xn, false>& xn)
  85. { return xn.data_; }
  86. template <std::size_t n, typename Xn>
  87. constexpr Xn&& get_impl(detail::elt<n, Xn, false>&& xn)
  88. { return static_cast<Xn&&>(xn.data_); }
  89. namespace detail {
  90. //////////////////////////////////////////////////////////////////////
  91. // basic_tuple_impl<n, Xn>
  92. //////////////////////////////////////////////////////////////////////
  93. struct from_other { };
  94. template <typename Indices, typename ...Xn>
  95. struct basic_tuple_impl;
  96. template <std::size_t ...n, typename ...Xn>
  97. struct basic_tuple_impl<std::index_sequence<n...>, Xn...>
  98. : detail::elt<n, Xn>...
  99. {
  100. static constexpr std::size_t size_ = sizeof...(Xn);
  101. constexpr basic_tuple_impl() = default;
  102. template <typename Other>
  103. explicit constexpr basic_tuple_impl(detail::from_other, Other&& other)
  104. : detail::elt<n, Xn>(get_impl<n>(static_cast<Other&&>(other)))...
  105. { }
  106. template <typename ...Yn>
  107. explicit constexpr basic_tuple_impl(Yn&& ...yn)
  108. : detail::elt<n, Xn>(static_cast<Yn&&>(yn))...
  109. { }
  110. };
  111. }
  112. //////////////////////////////////////////////////////////////////////////
  113. // basic_tuple
  114. //////////////////////////////////////////////////////////////////////////
  115. //! @cond
  116. template <typename ...Xn>
  117. struct basic_tuple final
  118. : detail::basic_tuple_impl<std::make_index_sequence<sizeof...(Xn)>, Xn...>
  119. {
  120. using Base = detail::basic_tuple_impl<std::make_index_sequence<sizeof...(Xn)>, Xn...>;
  121. constexpr basic_tuple() = default;
  122. // copy constructor
  123. template <typename Other, typename = typename std::enable_if<
  124. std::is_same<typename std::decay<Other>::type, basic_tuple>::value
  125. >::type>
  126. constexpr basic_tuple(Other&& other)
  127. : Base(detail::from_other{}, static_cast<Other&&>(other))
  128. { }
  129. template <typename ...Yn>
  130. explicit constexpr basic_tuple(Yn&& ...yn)
  131. : Base(static_cast<Yn&&>(yn)...)
  132. { }
  133. };
  134. //! @endcond
  135. template <typename ...Xn>
  136. struct tag_of<basic_tuple<Xn...>> {
  137. using type = basic_tuple_tag;
  138. };
  139. //////////////////////////////////////////////////////////////////////////
  140. // Foldable
  141. //////////////////////////////////////////////////////////////////////////
  142. template <>
  143. struct unpack_impl<basic_tuple_tag> {
  144. template <std::size_t ...i, typename ...Xn, typename F>
  145. static constexpr decltype(auto)
  146. apply(detail::basic_tuple_impl<std::index_sequence<i...>, Xn...> const& xs, F&& f) {
  147. return static_cast<F&&>(f)(
  148. get_impl<i>(static_cast<detail::elt<i, Xn> const&>(xs))...
  149. );
  150. }
  151. template <std::size_t ...i, typename ...Xn, typename F>
  152. static constexpr decltype(auto)
  153. apply(detail::basic_tuple_impl<std::index_sequence<i...>, Xn...>& xs, F&& f) {
  154. return static_cast<F&&>(f)(
  155. get_impl<i>(static_cast<detail::elt<i, Xn>&>(xs))...
  156. );
  157. }
  158. template <std::size_t ...i, typename ...Xn, typename F>
  159. static constexpr decltype(auto)
  160. apply(detail::basic_tuple_impl<std::index_sequence<i...>, Xn...>&& xs, F&& f) {
  161. return static_cast<F&&>(f)(
  162. get_impl<i>(static_cast<detail::elt<i, Xn>&&>(xs))...
  163. );
  164. }
  165. };
  166. //////////////////////////////////////////////////////////////////////////
  167. // Functor
  168. //////////////////////////////////////////////////////////////////////////
  169. template <>
  170. struct transform_impl<basic_tuple_tag> {
  171. template <std::size_t ...i, typename ...Xn, typename F>
  172. static constexpr auto
  173. apply(detail::basic_tuple_impl<std::index_sequence<i...>, Xn...> const& xs, F const& f) {
  174. return hana::make_basic_tuple(
  175. f(get_impl<i>(static_cast<detail::elt<i, Xn> const&>(xs)))...
  176. );
  177. }
  178. template <std::size_t ...i, typename ...Xn, typename F>
  179. static constexpr auto
  180. apply(detail::basic_tuple_impl<std::index_sequence<i...>, Xn...>& xs, F const& f) {
  181. return hana::make_basic_tuple(
  182. f(get_impl<i>(static_cast<detail::elt<i, Xn>&>(xs)))...
  183. );
  184. }
  185. template <std::size_t ...i, typename ...Xn, typename F>
  186. static constexpr auto
  187. apply(detail::basic_tuple_impl<std::index_sequence<i...>, Xn...>&& xs, F const& f) {
  188. return hana::make_basic_tuple(
  189. f(get_impl<i>(static_cast<detail::elt<i, Xn>&&>(xs)))...
  190. );
  191. }
  192. };
  193. //////////////////////////////////////////////////////////////////////////
  194. // Iterable
  195. //////////////////////////////////////////////////////////////////////////
  196. template <>
  197. struct at_impl<basic_tuple_tag> {
  198. template <typename Xs, typename N>
  199. static constexpr decltype(auto) apply(Xs&& xs, N const&) {
  200. constexpr std::size_t index = N::value;
  201. return hana::get_impl<index>(static_cast<Xs&&>(xs));
  202. }
  203. };
  204. template <>
  205. struct drop_front_impl<basic_tuple_tag> {
  206. template <std::size_t N, typename Xs, std::size_t ...i>
  207. static constexpr auto drop_front_helper(Xs&& xs, std::index_sequence<i...>) {
  208. return hana::make_basic_tuple(hana::get_impl<i+N>(static_cast<Xs&&>(xs))...);
  209. }
  210. template <typename Xs, typename N>
  211. static constexpr auto apply(Xs&& xs, N const&) {
  212. constexpr std::size_t len = detail::decay<Xs>::type::size_;
  213. return drop_front_helper<N::value>(static_cast<Xs&&>(xs), std::make_index_sequence<
  214. N::value < len ? len - N::value : 0
  215. >{});
  216. }
  217. };
  218. template <>
  219. struct is_empty_impl<basic_tuple_tag> {
  220. template <typename ...Xs>
  221. static constexpr auto apply(basic_tuple<Xs...> const&)
  222. { return hana::bool_c<sizeof...(Xs) == 0>; }
  223. };
  224. //////////////////////////////////////////////////////////////////////////
  225. // Sequence
  226. //////////////////////////////////////////////////////////////////////////
  227. template <>
  228. struct Sequence<basic_tuple_tag> {
  229. static constexpr bool value = true;
  230. };
  231. template <>
  232. struct make_impl<basic_tuple_tag> {
  233. template <typename ...Xn>
  234. static constexpr basic_tuple<typename detail::decay<Xn>::type...>
  235. apply(Xn&& ...xn) {
  236. return basic_tuple<typename detail::decay<Xn>::type...>{
  237. static_cast<Xn&&>(xn)...
  238. };
  239. }
  240. };
  241. #if 0
  242. //////////////////////////////////////////////////////////////////////////
  243. // length
  244. //////////////////////////////////////////////////////////////////////////
  245. template <>
  246. struct length_impl<basic_tuple_tag> {
  247. template <typename ...Xn>
  248. static constexpr auto apply(basic_tuple<Xn...> const&) {
  249. return hana::size_c<sizeof...(Xn)>;
  250. }
  251. };
  252. #endif
  253. BOOST_HANA_NAMESPACE_END
  254. #endif // !BOOST_HANA_BASIC_TUPLE_HPP