Clang Project

clang_source_code/test/CXX/temp/temp.decls/temp.variadic/p4.cpp
1// RUN: %clang_cc1 -std=c++11 -fsyntax-only -fexceptions -fcxx-exceptions -verify %s
2// RUN: %clang_cc1 -std=c++2a -fsyntax-only -fexceptions -fcxx-exceptions -verify %s
3
4template<typename... Types> struct tuple;
5template<int I> struct int_c;
6
7template<typename T>
8struct identity {
9  typedef T type;
10};
11
12template<typename T, typename U>
13struct is_same {
14  static const bool value = false;
15};
16
17template<typename T>
18struct is_same<T, T> {
19  static const bool value = true;
20};
21
22// FIXME: Several more bullets to go
23
24// In a function parameter pack, the pattern is the parameter-declaration
25// without the ellipsis.
26namespace PR11850 {
27  template<typename ...T> struct S {
28    int f(T...a, int b) { return b; }
29  };
30  S<> s;
31  S<int*, char, const double&> t;
32  int k = s.f(0);
33  int l = t.f(&k, 'x', 5.9, 4);
34
35  template<typename ...As> struct A {
36    template<typename ...Bs> struct B {
37      template<typename ...Cs> struct C {
38        C(As..., Bs..., int &k, Cs...);
39      };
40    };
41  };
42  A<>::B<>::C<> c000(k);
43  A<int>::B<>::C<int> c101(1, k, 3);
44  A<>::B<int>::C<int> c011(1, k, 3);
45  A<int>::B<int>::C<> c110(1, 2, k);
46  A<int, int>::B<int, int>::C<int, int> c222(1, 2, 3, 4, k, 5, 6);
47  A<int, int, int>::B<>::C<> c300(1, 2, 3, k);
48
49  int &f();
50  char &f(void*);
51  template<typename ...A> struct U {
52    template<typename ...B> struct V {
53      auto g(A...a, B...b) -> decltype(f(a...));
54    };
55  };
56  U<>::V<int*> v0;
57  U<int*>::V<> v1;
58  int &v0f = v0.g(0);
59  char &v1f = v1.g(0);
60}
61namespace PR12096 {
62  void Foo(int) {}
63  void Foo(int, int) = delete;
64  template<typename ...Args> struct Var {
65    Var(const Args &...args, int *) { Foo(args...); }
66  };
67  Var<int> var(1, 0);
68}
69
70// In an initializer-list (8.5); the pattern is an initializer-clause.
71// Note: this also covers expression-lists, since expression-list is
72// just defined as initializer-list.
73void five_args(int, int, int, int, int); // expected-note{{candidate function not viable: requires 5 arguments, but 6 were provided}}
74
75template<int ...Values>
76void initializer_list_expansion() {
77  int values[5] = { Values... }; // expected-error{{excess elements in array initializer}}
78  five_args(Values...); // expected-error{{no matching function for call to 'five_args'}}
79}
80
81template void initializer_list_expansion<1, 2, 3, 4, 5>();
82template void initializer_list_expansion<1, 2, 3, 4, 5, 6>(); // expected-note{{in instantiation of function template specialization 'initializer_list_expansion<1, 2, 3, 4, 5, 6>' requested here}}
83
84namespace PR8977 {
85  struct A { };
86  template<typename T, typename... Args> void f(Args... args) {
87    // An empty expression-list performs value initialization.
88    constexpr T t(args...);
89  };
90
91  template void f<A>();
92}
93
94// In a base-specifier-list (Clause 10); the pattern is a base-specifier.
95template<typename ...Mixins>
96struct HasMixins : public Mixins... { 
97  HasMixins();
98  HasMixins(const HasMixins&);
99  HasMixins(int i);
100};
101
102struct A { }; // expected-note{{candidate constructor (the implicit copy constructor) not viable: no known conversion from 'int' to 'const A' for 1st argument}} \
103// expected-note{{candidate constructor (the implicit move constructor) not viable: no known conversion from 'int' to 'A' for 1st argument}} \
104// expected-note{{candidate constructor (the implicit default constructor) not viable: requires 0 arguments, but 1 was provided}}
105struct B { };
106struct C { };
107struct D { };
108
109A *checkA = new HasMixins<A, B, C, D>;
110B *checkB = new HasMixins<A, B, C, D>;
111D *checkD = new HasMixins<A, B, C, D>;
112C *checkC = new HasMixins<A, B, D>; // expected-error{{cannot initialize a variable of type 'C *' with an rvalue of type 'HasMixins<A, B, D> *'}}
113HasMixins<> *checkNone = new HasMixins<>;
114
115template<typename Mixins>
116struct BrokenMixins : public Mixins... { }; // expected-error{{pack expansion does not contain any unexpanded parameter packs}}
117
118// In a mem-initializer-list (12.6.2); the pattern is a mem-initializer.
119template<typename ...Mixins>
120HasMixins<Mixins...>::HasMixins(): Mixins()... { }
121
122template<typename ...Mixins>
123HasMixins<Mixins...>::HasMixins(const HasMixins &other): Mixins(other)... { }
124
125template<typename ...Mixins>
126HasMixins<Mixins...>::HasMixins(int i): Mixins(i)... { } // expected-error{{no matching constructor for initialization of 'A'}}
127
128void test_has_mixins() {
129  HasMixins<A, B> ab;
130  HasMixins<A, B> ab2 = ab;
131  HasMixins<A, B> ab3(17); // expected-note{{in instantiation of member function 'HasMixins<A, B>::HasMixins' requested here}}
132}
133
134template<typename T>
135struct X {
136  T member;
137
138  X() : member()... { } // expected-error{{pack expansion for initialization of member 'member'}}
139};
140
141// There was a bug in the delayed parsing code for the
142// following case.
143template<typename ...T>
144struct DelayedParseTest : T...
145{
146  int a;
147  DelayedParseTest(T... i) : T{i}..., a{10} {}
148};
149
150
151// In a template-argument-list (14.3); the pattern is a template-argument.
152template<typename ...Types>
153struct tuple_of_refs {
154  typedef tuple<Types& ...> types;
155};
156
157tuple<int&, float&> *t_int_ref_float_ref;
158tuple_of_refs<int&, float&>::types *t_int_ref_float_ref_2 =  t_int_ref_float_ref;
159
160template<typename ...Types>
161struct extract_nested_types {
162  typedef tuple<typename Types::type...> types;
163};
164
165tuple<int, float> *t_int_float;
166extract_nested_types<identity<int>, identity<float> >::types *t_int_float_2 
167  = t_int_float;
168
169template<int ...N>
170struct tuple_of_ints {
171  typedef tuple<int_c<N>...> type;
172};
173
174int check_temp_arg_1[is_same<tuple_of_ints<1, 2, 3, 4, 5>::type, 
175                             tuple<int_c<1>, int_c<2>, int_c<3>, int_c<4>, 
176                                   int_c<5>>>::value? 1 : -1];
177
178#if __cplusplus < 201703L
179// In a dynamic-exception-specification (15.4); the pattern is a type-id.
180template<typename ...Types>
181struct f_with_except {
182  virtual void f() throw(Types...); // expected-note{{overridden virtual function is here}}
183};
184
185struct check_f_with_except_1 : f_with_except<int, float> {
186  virtual void f() throw(int, float);
187};
188
189struct check_f_with_except_2 : f_with_except<int, float> {
190  virtual void f() throw(int);
191};
192
193struct check_f_with_except_3 : f_with_except<int, float> {
194  virtual void f() throw(int, float, double); // expected-error{{exception specification of overriding function is more lax than base version}}
195};
196#endif
197
198namespace PackExpansionWithinLambda {
199  void swallow(...);
200  template<typename ...T, typename ...U> void f(U ...u) {
201    swallow([=] {
202      // C++17 [temp.variadic]p4:
203      //   Pack expansions can occur in the following contexts:
204
205      //    - in a function parameter pack
206      void g(T...);
207
208#if __cplusplus >= 201703L
209      struct A : T... {
210        //  - in a using-declaration
211        using T::x...;
212        using typename T::U...;
213      };
214#endif
215
216      //    - in a template parameter pack that is a pack expansion
217      // FIXME: We do not support any way to reach this case yet.
218
219      //    - in an initializer-list
220      int arr[] = {T().x...};
221
222      //    - in a base-specifier-list
223      struct B : T... {
224        //  - in a mem-initializer-list
225        B() : T{0}... {}
226      };
227
228      //    - in a template-argument-list
229      f<T...>();
230
231      //    - in an attribute-list
232      // FIXME: We do not support any such attributes yet.
233      
234      //    - in an alignment-specifier
235      alignas(T...) int y;
236
237      //    - in a capture-list
238      [](T ...t) { [t...]{}(); } (T()...);
239
240      //    - in a sizeof... expression
241      const int k1 = sizeof...(T);
242
243#if __cplusplus >= 201703L
244      //    - in a fold-expression
245      const int k2 = ((sizeof(T)/sizeof(T)) + ...);
246
247      static_assert(k1 == k2);
248#endif
249
250      // Trigger clang to look in here for unexpanded packs.
251      U u;
252    } ...);
253  }
254
255  template<typename ...T> void nested() {
256    swallow([=] {
257      [](T ...t) { [t]{}(); } (T()...); // expected-error {{unexpanded parameter pack 't'}}
258    }...); // expected-error {{does not contain any unexpanded}}
259  }
260
261  template <typename ...T> void g() {
262    // Check that we do detect the above cases when the pack is not expanded.
263    swallow([=] { void h(T); }); // expected-error {{unexpanded parameter pack 'T'}}
264    swallow([=] { struct A : T {}; }); // expected-error {{unexpanded parameter pack 'T'}}
265#if __cplusplus >= 201703L
266    swallow([=] { struct A : T... { using T::x; }; }); // expected-error {{unexpanded parameter pack 'T'}}
267    swallow([=] { struct A : T... { using typename T::U; }; }); // expected-error {{unexpanded parameter pack 'T'}}
268#endif
269
270    swallow([=] { int arr[] = {T().x}; }); // expected-error {{unexpanded parameter pack 'T'}}
271    swallow([=] { struct B : T... { B() : T{0} {} }; }); // expected-error {{unexpanded parameter pack 'T'}}
272    swallow([=] { f<T>(); }); // expected-error {{unexpanded parameter pack 'T'}}
273    swallow([=] { alignas(T) int y; }); // expected-error {{unexpanded parameter pack 'T'}}
274    swallow([=] { [](T ...t) {
275          [t]{}(); // expected-error {{unexpanded parameter pack 't'}}
276        } (T()...); });
277  }
278
279  struct T { int x; using U = int; };
280  void g() { f<T>(1, 2, 3); }
281
282  template<typename ...T, typename ...U> void pack_in_lambda(U ...u) { // expected-note {{here}}
283    // FIXME: Move this test into 'f' above once we support this syntax.
284    []<T *...v, template<T *> typename ...U>(U<v> ...uv) {}; // expected-error {{expected body of lambda}} expected-error {{does not refer to a value}}
285  }
286
287  template<typename ...T> void pack_expand_attr() {
288    // FIXME: Move this test into 'f' above once we support this.
289    [[gnu::aligned(alignof(T))...]] int x; // expected-error {{cannot be used as an attribute pack}} expected-error {{unexpanded}}
290  }
291}
292