1 //===----------------------------------------------------------------------===//
2 //
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6 //
7 //===----------------------------------------------------------------------===//
8 
9 // <tuple>
10 
11 // template <class... Types> class tuple;
12 
13 // explicit(see-below) constexpr tuple();
14 
15 // UNSUPPORTED: c++03
16 
17 #include <tuple>
18 #include <string>
19 #include <cassert>
20 #include <type_traits>
21 
22 #include "test_macros.h"
23 #include "DefaultOnly.h"
24 
25 struct NoDefault {
26     NoDefault() = delete;
NoDefaultNoDefault27     explicit NoDefault(int) { }
28 };
29 
30 struct NoExceptDefault {
31     NoExceptDefault() noexcept = default;
32 };
33 
34 struct ThrowingDefault {
ThrowingDefaultThrowingDefault35     ThrowingDefault() { }
36 };
37 
38 struct IllFormedDefault {
IllFormedDefaultIllFormedDefault39     IllFormedDefault(int x) : value(x) {}
40     template <bool Pred = false>
IllFormedDefaultIllFormedDefault41     constexpr IllFormedDefault() {
42         static_assert(Pred,
43             "The default constructor should not be instantiated");
44     }
45     int value;
46 };
47 
main(int,char **)48 int main(int, char**)
49 {
50     {
51         std::tuple<> t;
52         (void)t;
53     }
54     {
55         std::tuple<int> t;
56         assert(std::get<0>(t) == 0);
57     }
58     {
59         std::tuple<int, char*> t;
60         assert(std::get<0>(t) == 0);
61         assert(std::get<1>(t) == nullptr);
62     }
63     {
64         std::tuple<int, char*, std::string> t;
65         assert(std::get<0>(t) == 0);
66         assert(std::get<1>(t) == nullptr);
67         assert(std::get<2>(t) == "");
68     }
69     {
70         std::tuple<int, char*, std::string, DefaultOnly> t;
71         assert(std::get<0>(t) == 0);
72         assert(std::get<1>(t) == nullptr);
73         assert(std::get<2>(t) == "");
74         assert(std::get<3>(t) == DefaultOnly());
75     }
76     {
77         // See bug #21157.
78         static_assert(!std::is_default_constructible<std::tuple<NoDefault>>(), "");
79         static_assert(!std::is_default_constructible<std::tuple<DefaultOnly, NoDefault>>(), "");
80         static_assert(!std::is_default_constructible<std::tuple<NoDefault, DefaultOnly, NoDefault>>(), "");
81     }
82     {
83         static_assert(noexcept(std::tuple<NoExceptDefault>()), "");
84         static_assert(noexcept(std::tuple<NoExceptDefault, NoExceptDefault>()), "");
85 
86         static_assert(!noexcept(std::tuple<ThrowingDefault, NoExceptDefault>()), "");
87         static_assert(!noexcept(std::tuple<NoExceptDefault, ThrowingDefault>()), "");
88         static_assert(!noexcept(std::tuple<ThrowingDefault, ThrowingDefault>()), "");
89     }
90     {
91         constexpr std::tuple<> t;
92         (void)t;
93     }
94     {
95         constexpr std::tuple<int> t;
96         assert(std::get<0>(t) == 0);
97     }
98     {
99         constexpr std::tuple<int, char*> t;
100         assert(std::get<0>(t) == 0);
101         assert(std::get<1>(t) == nullptr);
102     }
103     {
104     // Check that the SFINAE on the default constructor is not evaluated when
105     // it isn't needed. If the default constructor is evaluated then this test
106     // should fail to compile.
107         IllFormedDefault v(0);
108         std::tuple<IllFormedDefault> t(v);
109     }
110     {
111         struct Base { };
112         struct Derived : Base { protected: Derived() = default; };
113         static_assert(!std::is_default_constructible<std::tuple<Derived, int> >::value, "");
114     }
115 
116     return 0;
117 }
118