blob: 14a23af3f5658839fb0e541281dd0b5b992824e2 [file] [log] [blame]
// RUN: %clang_cc1 -std=c++17 -Wno-c++26-extensions -verify %s
// RUN: %clang_cc1 -std=c++17 -Wno-c++26-extensions -verify %s -fexperimental-new-constant-interpreter
// RUN: %clang_cc1 -std=c++2c -Wpre-c++26-compat -verify=cxx26,expected %s
// RUN: %clang_cc1 -std=c++2c -Wpre-c++26-compat -verify=cxx26,expected %s -fexperimental-new-constant-interpreter
struct X {
bool flag;
int data;
constexpr explicit operator bool() const {
return flag;
}
constexpr operator int() const {
return data;
}
};
namespace CondInIf {
constexpr int f(X x) {
if (auto [ok, d] = x) // cxx26-warning {{structured binding declaration in a condition is incompatible with C++ standards before C++2c}}
return d + int(ok);
else
return d * int(ok);
ok = {}; // expected-error {{use of undeclared identifier 'ok'}}
d = {}; // expected-error {{use of undeclared identifier 'd'}}
}
static_assert(f({true, 2}) == 3);
static_assert(f({false, 2}) == 0);
constexpr char g(char const (&x)[2]) {
if (auto &[a, b] = x) // cxx26-warning {{structured binding declaration in a condition is incompatible with C++ standards before C++2c}}
return a;
else
return b;
if (auto [a, b] = x) // expected-error {{an array type is not allowed here}} \
// cxx26-warning {{structured binding declaration in a condition is incompatible with C++ standards before C++2c}}
;
}
static_assert(g("x") == 'x');
} // namespace CondInIf
namespace CondInSwitch {
constexpr int f(int n) {
switch (X s = {true, n}; auto [ok, d] = s) {
// cxx26-warning@-1 {{structured binding declaration in a condition is incompatible with C++ standards before C++2c}}
s = {};
case 0:
return int(ok);
case 1:
return d * 10;
case 2:
return d * 40;
default:
return 0;
}
ok = {}; // expected-error {{use of undeclared identifier 'ok'}}
d = {}; // expected-error {{use of undeclared identifier 'd'}}
s = {}; // expected-error {{use of undeclared identifier 's'}}
}
static_assert(f(0) == 1);
static_assert(f(1) == 10);
static_assert(f(2) == 80);
} // namespace CondInSwitch
namespace CondInWhile {
constexpr int f(int n) {
int m = 1;
while (auto [ok, d] = X{n > 1, n}) {
// cxx26-warning@-1 {{structured binding declaration in a condition is incompatible with C++ standards before C++2c}}
m *= d;
--n;
}
return m;
return ok; // expected-error {{use of undeclared identifier 'ok'}}
}
static_assert(f(0) == 1);
static_assert(f(1) == 1);
static_assert(f(4) == 24);
} // namespace CondInWhile
namespace CondInFor {
constexpr int f(int n) {
int a = 1, b = 1;
for (X x = {true, n}; auto &[ok, d] = x; --d) {
// cxx26-warning@-1 {{structured binding declaration in a condition is incompatible with C++ standards before C++2c}}
if (d < 2)
ok = false;
else {
int x = b;
b += a;
a = x;
}
}
return b;
return d; // expected-error {{use of undeclared identifier 'd'}}
}
static_assert(f(0) == 1);
static_assert(f(1) == 1);
static_assert(f(2) == 2);
static_assert(f(5) == 8);
} // namespace CondInFor