2020-02-19 08:13:23 +08:00
|
|
|
// RUN: %clang_cc1 -std=c++17 -verify %s
|
2016-07-23 07:36:59 +08:00
|
|
|
|
|
|
|
void use_from_own_init() {
|
|
|
|
auto [a] = a; // expected-error {{binding 'a' cannot appear in the initializer of its own decomposition declaration}}
|
|
|
|
}
|
|
|
|
|
2016-08-12 06:25:46 +08:00
|
|
|
// As a Clang extension, _Complex can be decomposed.
|
|
|
|
float decompose_complex(_Complex float cf) {
|
2016-08-12 08:39:32 +08:00
|
|
|
static _Complex float scf;
|
|
|
|
auto &[sre, sim] = scf;
|
|
|
|
// ok, this is references initialized by constant expressions all the way down
|
|
|
|
static_assert(&sre == &__real scf);
|
|
|
|
static_assert(&sim == &__imag scf);
|
|
|
|
|
2016-08-12 06:25:46 +08:00
|
|
|
auto [re, im] = cf;
|
|
|
|
return re*re + im*im;
|
|
|
|
}
|
|
|
|
|
|
|
|
// As a Clang extension, vector types can be decomposed.
|
|
|
|
typedef float vf3 __attribute__((ext_vector_type(3)));
|
|
|
|
float decompose_vector(vf3 v) {
|
|
|
|
auto [x, y, z] = v;
|
|
|
|
auto *p = &x; // expected-error {{address of vector element requested}}
|
|
|
|
return x + y + z;
|
|
|
|
}
|
|
|
|
|
2016-08-12 08:39:32 +08:00
|
|
|
struct S { int a, b; };
|
|
|
|
constexpr int f(S s) {
|
|
|
|
auto &[a, b] = s;
|
|
|
|
return a * 10 + b;
|
|
|
|
}
|
|
|
|
static_assert(f({1, 2}) == 12);
|
|
|
|
|
|
|
|
constexpr bool g(S &&s) {
|
|
|
|
auto &[a, b] = s;
|
|
|
|
return &a == &s.a && &b == &s.b && &a != &b;
|
|
|
|
}
|
|
|
|
static_assert(g({1, 2}));
|
|
|
|
|
2019-05-25 09:04:17 +08:00
|
|
|
auto [outer1, outer2] = S{1, 2};
|
2016-08-15 10:34:23 +08:00
|
|
|
void enclosing() {
|
2019-05-25 09:04:17 +08:00
|
|
|
struct S { int a = outer1; };
|
2016-08-15 10:34:23 +08:00
|
|
|
auto [n] = S(); // expected-note 2{{'n' declared here}}
|
|
|
|
|
|
|
|
struct Q { int f() { return n; } }; // expected-error {{reference to local binding 'n' declared in enclosing function}}
|
|
|
|
(void) [&] { return n; }; // expected-error {{reference to local binding 'n' declared in enclosing function}}
|
|
|
|
(void) [n] {}; // expected-error {{'n' in capture list does not name a variable}}
|
2019-05-25 09:04:17 +08:00
|
|
|
|
|
|
|
static auto [m] = S(); // expected-warning {{extension}}
|
|
|
|
struct R { int f() { return m; } };
|
|
|
|
(void) [&] { return m; };
|
|
|
|
(void) [m] {}; // expected-error {{'m' in capture list does not name a variable}}
|
2016-08-15 10:34:23 +08:00
|
|
|
}
|
|
|
|
|
2016-10-21 02:29:25 +08:00
|
|
|
void bitfield() {
|
|
|
|
struct { int a : 3, : 4, b : 5; } a;
|
|
|
|
auto &[x, y] = a;
|
|
|
|
auto &[p, q, r] = a; // expected-error {{decomposes into 2 elements, but 3 names were provided}}
|
|
|
|
}
|
|
|
|
|
2016-11-12 04:51:04 +08:00
|
|
|
void for_range() {
|
|
|
|
int x = 1;
|
|
|
|
for (auto[a, b] : x) { // expected-error {{invalid range expression of type 'int'; no viable 'begin' function available}}
|
|
|
|
a++;
|
|
|
|
}
|
|
|
|
|
|
|
|
int y[5];
|
|
|
|
for (auto[c] : y) { // expected-error {{cannot decompose non-class, non-array type 'int'}}
|
|
|
|
c++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-01-05 07:14:16 +08:00
|
|
|
int error_recovery() {
|
|
|
|
auto [foobar]; // expected-error {{requires an initializer}}
|
|
|
|
return foobar_; // expected-error {{undeclared identifier 'foobar_'}}
|
|
|
|
}
|
|
|
|
|
2017-06-13 00:11:06 +08:00
|
|
|
// PR32172
|
|
|
|
template <class T> void dependent_foreach(T t) {
|
|
|
|
for (auto [a,b,c] : t)
|
|
|
|
a,b,c;
|
|
|
|
}
|
|
|
|
|
2018-05-08 06:23:38 +08:00
|
|
|
struct PR37352 {
|
|
|
|
int n;
|
2020-02-19 08:13:23 +08:00
|
|
|
void f() { static auto [a] = *this; } // expected-warning {{C++20 extension}}
|
2018-05-08 06:23:38 +08:00
|
|
|
};
|
|
|
|
|
2019-01-28 03:19:59 +08:00
|
|
|
namespace instantiate_template {
|
|
|
|
|
|
|
|
template <typename T1, typename T2>
|
|
|
|
struct pair {
|
|
|
|
T1 a;
|
|
|
|
T2 b;
|
|
|
|
};
|
|
|
|
|
|
|
|
const pair<int, int> &f1();
|
|
|
|
|
|
|
|
int f2() {
|
|
|
|
const auto &[a, b] = f1();
|
|
|
|
return a + b;
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace instantiate_template
|
|
|
|
|
2016-08-12 06:25:46 +08:00
|
|
|
// FIXME: by-value array copies
|