//===----------------------------------------------------------------------===//
//
// 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.
//
//===----------------------------------------------------------------------===//
// <tuple>
// template <class... Types> class tuple;
// template <class... Tuples> tuple<CTypes...> tuple_cat(Tuples&&... tpls);
// UNSUPPORTED: c++98, c++03
#include <tuple>
#include <utility>
#include <array>
#include <string>
#include <cassert>
#include "MoveOnly.h"
int main()
{
{
std::tuple<> t = std::tuple_cat();
((void)t); // Prevent unused warning
}
{
std::tuple<> t1;
std::tuple<> t2 = std::tuple_cat(t1);
((void)t2); // Prevent unused warning
}
{
std::tuple<> t = std::tuple_cat(std::tuple<>());
((void)t); // Prevent unused warning
}
{
std::tuple<> t = std::tuple_cat(std::array<int, 0>());
((void)t); // Prevent unused warning
}
{
std::tuple<int> t1(1);
std::tuple<int> t = std::tuple_cat(t1);
assert(std::get<0>(t) == 1);
}
#if _LIBCPP_STD_VER > 11
{
constexpr std::tuple<> t = std::tuple_cat();
((void)t); // Prevent unused warning
}
{
constexpr std::tuple<> t1;
constexpr std::tuple<> t2 = std::tuple_cat(t1);
((void)t2); // Prevent unused warning
}
{
constexpr std::tuple<> t = std::tuple_cat(std::tuple<>());
((void)t); // Prevent unused warning
}
{
constexpr std::tuple<> t = std::tuple_cat(std::array<int, 0>());
((void)t); // Prevent unused warning
}
{
constexpr std::tuple<int> t1(1);
constexpr std::tuple<int> t = std::tuple_cat(t1);
static_assert(std::get<0>(t) == 1, "");
}
{
constexpr std::tuple<int> t1(1);
constexpr std::tuple<int, int> t = std::tuple_cat(t1, t1);
static_assert(std::get<0>(t) == 1, "");
static_assert(std::get<1>(t) == 1, "");
}
#endif
{
std::tuple<int, MoveOnly> t =
std::tuple_cat(std::tuple<int, MoveOnly>(1, 2));
assert(std::get<0>(t) == 1);
assert(std::get<1>(t) == 2);
}
{
std::tuple<int, int, int> t = std::tuple_cat(std::array<int, 3>());
assert(std::get<0>(t) == 0);
assert(std::get<1>(t) == 0);
assert(std::get<2>(t) == 0);
}
{
std::tuple<int, MoveOnly> t = std::tuple_cat(std::pair<int, MoveOnly>(2, 1));
assert(std::get<0>(t) == 2);
assert(std::get<1>(t) == 1);
}
{
std::tuple<> t1;
std::tuple<> t2;
std::tuple<> t3 = std::tuple_cat(t1, t2);
((void)t3); // Prevent unused warning
}
{
std::tuple<> t1;
std::tuple<int> t2(2);
std::tuple<int> t3 = std::tuple_cat(t1, t2);
assert(std::get<0>(t3) == 2);
}
{
std::tuple<> t1;
std::tuple<int> t2(2);
std::tuple<int> t3 = std::tuple_cat(t2, t1);
assert(std::get<0>(t3) == 2);
}
{
std::tuple<int*> t1;
std::tuple<int> t2(2);
std::tuple<int*, int> t3 = std::tuple_cat(t1, t2);
assert(std::get<0>(t3) == nullptr);
assert(std::get<1>(t3) == 2);
}
{
std::tuple<int*> t1;
std::tuple<int> t2(2);
std::tuple<int, int*> t3 = std::tuple_cat(t2, t1);
assert(std::get<0>(t3) == 2);
assert(std::get<1>(t3) == nullptr);
}
{
std::tuple<int*> t1;
std::tuple<int, double> t2(2, 3.5);
std::tuple<int*, int, double> t3 = std::tuple_cat(t1, t2);
assert(std::get<0>(t3) == nullptr);
assert(std::get<1>(t3) == 2);
assert(std::get<2>(t3) == 3.5);
}
{
std::tuple<int*> t1;
std::tuple<int, double> t2(2, 3.5);
std::tuple<int, double, int*> t3 = std::tuple_cat(t2, t1);
assert(std::get<0>(t3) == 2);
assert(std::get<1>(t3) == 3.5);
assert(std::get<2>(t3) == nullptr);
}
{
std::tuple<int*, MoveOnly> t1(nullptr, 1);
std::tuple<int, double> t2(2, 3.5);
std::tuple<int*, MoveOnly, int, double> t3 =
std::tuple_cat(std::move(t1), t2);
assert(std::get<0>(t3) == nullptr);
assert(std::get<1>(t3) == 1);
assert(std::get<2>(t3) == 2);
assert(std::get<3>(t3) == 3.5);
}
{
std::tuple<int*, MoveOnly> t1(nullptr, 1);
std::tuple<int, double> t2(2, 3.5);
std::tuple<int, double, int*, MoveOnly> t3 =
std::tuple_cat(t2, std::move(t1));
assert(std::get<0>(t3) == 2);
assert(std::get<1>(t3) == 3.5);
assert(std::get<2>(t3) == nullptr);
assert(std::get<3>(t3) == 1);
}
{
std::tuple<MoveOnly, MoveOnly> t1(1, 2);
std::tuple<int*, MoveOnly> t2(nullptr, 4);
std::tuple<MoveOnly, MoveOnly, int*, MoveOnly> t3 =
std::tuple_cat(std::move(t1), std::move(t2));
assert(std::get<0>(t3) == 1);
assert(std::get<1>(t3) == 2);
assert(std::get<2>(t3) == nullptr);
assert(std::get<3>(t3) == 4);
}
{
std::tuple<MoveOnly, MoveOnly> t1(1, 2);
std::tuple<int*, MoveOnly> t2(nullptr, 4);
std::tuple<MoveOnly, MoveOnly, int*, MoveOnly> t3 =
std::tuple_cat(std::tuple<>(),
std::move(t1),
std::move(t2));
assert(std::get<0>(t3) == 1);
assert(std::get<1>(t3) == 2);
assert(std::get<2>(t3) == nullptr);
assert(std::get<3>(t3) == 4);
}
{
std::tuple<MoveOnly, MoveOnly> t1(1, 2);
std::tuple<int*, MoveOnly> t2(nullptr, 4);
std::tuple<MoveOnly, MoveOnly, int*, MoveOnly> t3 =
std::tuple_cat(std::move(t1),
std::tuple<>(),
std::move(t2));
assert(std::get<0>(t3) == 1);
assert(std::get<1>(t3) == 2);
assert(std::get<2>(t3) == nullptr);
assert(std::get<3>(t3) == 4);
}
{
std::tuple<MoveOnly, MoveOnly> t1(1, 2);
std::tuple<int*, MoveOnly> t2(nullptr, 4);
std::tuple<MoveOnly, MoveOnly, int*, MoveOnly> t3 =
std::tuple_cat(std::move(t1),
std::move(t2),
std::tuple<>());
assert(std::get<0>(t3) == 1);
assert(std::get<1>(t3) == 2);
assert(std::get<2>(t3) == nullptr);
assert(std::get<3>(t3) == 4);
}
{
std::tuple<MoveOnly, MoveOnly> t1(1, 2);
std::tuple<int*, MoveOnly> t2(nullptr, 4);
std::tuple<MoveOnly, MoveOnly, int*, MoveOnly, int> t3 =
std::tuple_cat(std::move(t1),
std::move(t2),
std::tuple<int>(5));
assert(std::get<0>(t3) == 1);
assert(std::get<1>(t3) == 2);
assert(std::get<2>(t3) == nullptr);
assert(std::get<3>(t3) == 4);
assert(std::get<4>(t3) == 5);
}
{
// See bug #19616.
auto t1 = std::tuple_cat(
std::make_tuple(std::make_tuple(1)),
std::make_tuple()
);
assert(t1 == std::make_tuple(std::make_tuple(1)));
auto t2 = std::tuple_cat(
std::make_tuple(std::make_tuple(1)),
std::make_tuple(std::make_tuple(2))
);
assert(t2 == std::make_tuple(std::make_tuple(1), std::make_tuple(2)));
}
}