//===----------------------------------------------------------------------===// // // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. // See https://llvm.org/LICENSE.txt for license information. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception // //===----------------------------------------------------------------------===// // // template class tuple; // tuple(tuple&& u); // UNSUPPORTED: c++03 #include #include #include #include "test_macros.h" #include "MoveOnly.h" struct ConstructsWithTupleLeaf { ConstructsWithTupleLeaf() {} ConstructsWithTupleLeaf(ConstructsWithTupleLeaf const &) { assert(false); } ConstructsWithTupleLeaf(ConstructsWithTupleLeaf &&) {} template ConstructsWithTupleLeaf(T) { static_assert(!std::is_same::value, "Constructor instantiated for type other than int"); } }; // move_only type which triggers the empty base optimization struct move_only_ebo { move_only_ebo() = default; move_only_ebo(move_only_ebo&&) = default; }; // a move_only type which does not trigger the empty base optimization struct move_only_large final { move_only_large() : value(42) {} move_only_large(move_only_large&&) = default; int value; }; template void test_sfinae() { using Tup = std::tuple; using Alloc = std::allocator; using Tag = std::allocator_arg_t; // special members { static_assert(std::is_default_constructible::value, ""); static_assert(std::is_move_constructible::value, ""); static_assert(!std::is_copy_constructible::value, ""); static_assert(!std::is_constructible::value, ""); } // args constructors { static_assert(std::is_constructible::value, ""); static_assert(!std::is_constructible::value, ""); static_assert(!std::is_constructible::value, ""); } // uses-allocator special member constructors { static_assert(std::is_constructible::value, ""); static_assert(std::is_constructible::value, ""); static_assert(!std::is_constructible::value, ""); static_assert(!std::is_constructible::value, ""); } // uses-allocator args constructors { static_assert(std::is_constructible::value, ""); static_assert(!std::is_constructible::value, ""); static_assert(!std::is_constructible::value, ""); } } int main(int, char**) { { typedef std::tuple<> T; T t0; T t = std::move(t0); ((void)t); // Prevent unused warning } { typedef std::tuple T; T t0(MoveOnly(0)); T t = std::move(t0); assert(std::get<0>(t) == 0); } { typedef std::tuple T; T t0(MoveOnly(0), MoveOnly(1)); T t = std::move(t0); assert(std::get<0>(t) == 0); assert(std::get<1>(t) == 1); } { typedef std::tuple T; T t0(MoveOnly(0), MoveOnly(1), MoveOnly(2)); T t = std::move(t0); assert(std::get<0>(t) == 0); assert(std::get<1>(t) == 1); assert(std::get<2>(t) == 2); } // A bug in tuple caused __tuple_leaf to use its explicit converting constructor // as its move constructor. This tests that ConstructsWithTupleLeaf is not called // (w/ __tuple_leaf) { typedef std::tuple d_t; d_t d((ConstructsWithTupleLeaf())); d_t d2(static_cast(d)); } { test_sfinae(); test_sfinae(); } return 0; }