147 lines
		
	
	
		
			3.6 KiB
		
	
	
	
		
			C++
		
	
	
	
			
		
		
	
	
			147 lines
		
	
	
		
			3.6 KiB
		
	
	
	
		
			C++
		
	
	
	
| //===----------------------------------------------------------------------===//
 | |
| //
 | |
| //                     The LLVM Compiler Infrastructure
 | |
| //
 | |
| // This file is dual licensed under the MIT and the University of Illinois Open
 | |
| // Source Licenses. See LICENSE.TXT for details.
 | |
| //
 | |
| //===----------------------------------------------------------------------===//
 | |
| 
 | |
| // UNSUPPORTED: c++98, c++03, c++11
 | |
| 
 | |
| // <experimental/tuple>
 | |
| 
 | |
| // template <class F, class T> constexpr decltype(auto) apply(F &&, T &&)
 | |
| 
 | |
| // Stress testing large arities with tuple and array.
 | |
| 
 | |
| #include <experimental/tuple>
 | |
| #include <array>
 | |
| #include <utility>
 | |
| #include <cassert>
 | |
| 
 | |
| ////////////////////////////////////////////////////////////////////////////////
 | |
| template <class T, std::size_t Dummy = 0>
 | |
| struct always_imp
 | |
| {
 | |
|     typedef T type;
 | |
| };
 | |
| 
 | |
| template <class T, std::size_t Dummy = 0>
 | |
| using always_t = typename always_imp<T, Dummy>::type;
 | |
| 
 | |
| ////////////////////////////////////////////////////////////////////////////////
 | |
| template <class Tuple, class Idx>
 | |
| struct make_function;
 | |
| 
 | |
| template <class Tp, std::size_t ...Idx>
 | |
| struct make_function<Tp, std::integer_sequence<std::size_t, Idx...>>
 | |
| {
 | |
|     using type = bool (*)(always_t<Tp, Idx>...);
 | |
| };
 | |
| 
 | |
| template <class Tp, std::size_t Size>
 | |
| using make_function_t = typename make_function<Tp, std::make_index_sequence<Size>>::type;
 | |
| 
 | |
| ////////////////////////////////////////////////////////////////////////////////
 | |
| template <class Tp, class Idx>
 | |
| struct make_tuple_imp;
 | |
| 
 | |
| ////////////////////////////////////////////////////////////////////////////////
 | |
| template <class Tp, std::size_t ...Idx>
 | |
| struct make_tuple_imp<Tp, std::integer_sequence<std::size_t, Idx...>>
 | |
| {
 | |
|     using type = std::tuple<always_t<Tp, Idx>...>;
 | |
| };
 | |
| 
 | |
| template <class Tp, std::size_t Size>
 | |
| using make_tuple_t = typename make_tuple_imp<Tp, std::make_index_sequence<Size>>::type;
 | |
| 
 | |
| template <class ...Types>
 | |
| bool test_apply_fn(Types...) { return true; }
 | |
| 
 | |
| namespace ex = std::experimental;
 | |
| 
 | |
| template <std::size_t Size>
 | |
| void test_all()
 | |
| {
 | |
| 
 | |
|     using A = std::array<int, Size>;
 | |
|     using ConstA = std::array<int const, Size>;
 | |
| 
 | |
|     using Tuple = make_tuple_t<int, Size>;
 | |
|     using CTuple = make_tuple_t<const int, Size>;
 | |
| 
 | |
|     using ValFn  = make_function_t<int, Size>;
 | |
|     ValFn val_fn = &test_apply_fn;
 | |
| 
 | |
|     using RefFn  = make_function_t<int &, Size>;
 | |
|     RefFn ref_fn = &test_apply_fn;
 | |
| 
 | |
|     using CRefFn = make_function_t<int const &, Size>;
 | |
|     CRefFn cref_fn = &test_apply_fn;
 | |
| 
 | |
|     using RRefFn = make_function_t<int &&, Size>;
 | |
|     RRefFn rref_fn = &test_apply_fn;
 | |
| 
 | |
|     {
 | |
|         A a{};
 | |
|         assert(ex::apply(val_fn, a));
 | |
|         assert(ex::apply(ref_fn, a));
 | |
|         assert(ex::apply(cref_fn, a));
 | |
|         assert(ex::apply(rref_fn, std::move(a)));
 | |
|     }
 | |
|     {
 | |
|         ConstA a{};
 | |
|         assert(ex::apply(val_fn, a));
 | |
|         assert(ex::apply(cref_fn, a));
 | |
|     }
 | |
|     {
 | |
|         Tuple a{};
 | |
|         assert(ex::apply(val_fn, a));
 | |
|         assert(ex::apply(ref_fn, a));
 | |
|         assert(ex::apply(cref_fn, a));
 | |
|         assert(ex::apply(rref_fn, std::move(a)));
 | |
|     }
 | |
|     {
 | |
|         CTuple a{};
 | |
|         assert(ex::apply(val_fn, a));
 | |
|         assert(ex::apply(cref_fn, a));
 | |
|     }
 | |
| 
 | |
| }
 | |
| 
 | |
| 
 | |
| template <std::size_t Size>
 | |
| void test_one()
 | |
| {
 | |
|     using A = std::array<int, Size>;
 | |
|     using Tuple = make_tuple_t<int, Size>;
 | |
| 
 | |
|     using ValFn  = make_function_t<int, Size>;
 | |
|     ValFn val_fn = &test_apply_fn;
 | |
| 
 | |
|     {
 | |
|         A a{};
 | |
|         assert(ex::apply(val_fn, a));
 | |
|     }
 | |
|     {
 | |
|         Tuple a{};
 | |
|         assert(ex::apply(val_fn, a));
 | |
|     }
 | |
| }
 | |
| 
 | |
| int main()
 | |
| {
 | |
|     // Instantiate with 1-5 arguments.
 | |
|     test_all<1>();
 | |
|     test_all<2>();
 | |
|     test_all<3>();
 | |
|     test_all<4>();
 | |
|     test_all<5>();
 | |
| 
 | |
|     // Stress test with 128.
 | |
|     test_one<128>();
 | |
|     //test_one<256>();
 | |
| }
 |