192 lines
4.3 KiB
C++
192 lines
4.3 KiB
C++
// RUN: %clang_cc1 -std=c++17 -fexperimental-new-constant-interpreter -verify=expected,both %s
|
|
// RUN: %clang_cc1 -std=c++17 -verify=ref,both %s
|
|
|
|
constexpr bool isEven(int a) {
|
|
bool v = false;
|
|
switch(a) {
|
|
case 2: return true;
|
|
case 4: return true;
|
|
case 6: return true;
|
|
|
|
case 8:
|
|
case 10:
|
|
case 12:
|
|
case 14:
|
|
case 16:
|
|
return true;
|
|
case 18:
|
|
v = true;
|
|
break;
|
|
|
|
default:
|
|
switch(a) {
|
|
case 1:
|
|
break;
|
|
case 3:
|
|
return false;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
return v;
|
|
}
|
|
static_assert(isEven(2), "");
|
|
static_assert(isEven(8), "");
|
|
static_assert(isEven(10), "");
|
|
static_assert(isEven(18), "");
|
|
static_assert(!isEven(1), "");
|
|
static_assert(!isEven(3), "");
|
|
|
|
|
|
constexpr int withInit() {
|
|
switch(int a = 2; a) {
|
|
case 1: return -1;
|
|
case 2: return 2;
|
|
}
|
|
return -1;
|
|
}
|
|
static_assert(withInit() == 2, "");
|
|
|
|
constexpr int FT(int a) {
|
|
int m = 0;
|
|
switch(a) {
|
|
case 4: m++;
|
|
case 3: m++;
|
|
case 2: m++;
|
|
case 1: m++;
|
|
return m;
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
static_assert(FT(1) == 1, "");
|
|
static_assert(FT(4) == 4, "");
|
|
static_assert(FT(5) == -1, "");
|
|
|
|
|
|
constexpr int good() { return 1; }
|
|
constexpr int test(int val) {
|
|
switch (val) {
|
|
case good(): return 100;
|
|
default: return -1;
|
|
}
|
|
return 0;
|
|
}
|
|
static_assert(test(1) == 100, "");
|
|
|
|
constexpr int bad(int val) { return val / 0; } // both-warning {{division by zero}}
|
|
constexpr int another_test(int val) { // both-note {{declared here}}
|
|
switch (val) {
|
|
case bad(val): return 100; // both-error {{case value is not a constant expression}} \
|
|
// both-note {{cannot be used in a constant expression}}
|
|
default: return -1;
|
|
}
|
|
return 0;
|
|
}
|
|
static_assert(another_test(1) == 100, ""); // both-error {{static assertion failed}} \
|
|
// both-note {{evaluates to}}
|
|
|
|
namespace gnurange {
|
|
constexpr int l(int n) {
|
|
return n + 1;
|
|
}
|
|
constexpr int h(int n) {
|
|
return 2 * n + 1;
|
|
}
|
|
constexpr int f(int x) {
|
|
const int n = 2;
|
|
constexpr struct {
|
|
char lo {'a'};
|
|
char hi {'z'};
|
|
} s;
|
|
|
|
switch (x) {
|
|
case l(n) ... h(n):
|
|
return 1;
|
|
case -1 ... 1:
|
|
return 2;
|
|
case 9 ... 14:
|
|
return 3;
|
|
case 15:
|
|
return 4;
|
|
case 16 ... 20:
|
|
return 5;
|
|
case s.lo ... s.hi:
|
|
return 6;
|
|
default:
|
|
return -1;
|
|
}
|
|
}
|
|
static_assert(f(0) == 2);
|
|
static_assert(f(2) == -1);
|
|
static_assert(f(3) == 1);
|
|
static_assert(f(4) == 1);
|
|
static_assert(f(5) == 1);
|
|
static_assert(f(6) == -1);
|
|
static_assert(f(14) == 3);
|
|
static_assert(f(15) == 4);
|
|
static_assert(f(16) == 5);
|
|
static_assert(f(20) == 5);
|
|
static_assert(f('d') == 6);
|
|
|
|
template <int Lo, int Hi>
|
|
constexpr bool g(int x) {
|
|
switch (x) {
|
|
case Lo ... Hi:
|
|
break;
|
|
default:
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
static_assert(g<100, 200>(132));
|
|
|
|
constexpr bool m(int x) {
|
|
switch (x) {
|
|
case 10 ... 1: // both-warning {{empty case range specified}}
|
|
return true;
|
|
default:
|
|
return false;
|
|
}
|
|
}
|
|
static_assert(m(3)); // both-error {{static assertion failed due to requirement 'm(3)'}}
|
|
static_assert(!m(3));
|
|
|
|
constexpr bool j(int x) { // both-note {{declared here}}
|
|
switch (x) {
|
|
case bad(x) ... 100: // both-error {{case value is not a constant expression}} \
|
|
// both-note {{cannot be used in a constant expression}}
|
|
return true;
|
|
default:
|
|
break;
|
|
}
|
|
return false;
|
|
}
|
|
static_assert(j(1)); // both-error {{static assertion failed}}
|
|
|
|
constexpr bool d(int x) { // both-note {{declared here}}
|
|
switch (x) {
|
|
case -100 ... bad(x): // both-error {{case value is not a constant expression}} \
|
|
// both-note {{cannot be used in a constant expression}}
|
|
return true;
|
|
default:
|
|
break;
|
|
}
|
|
return false;
|
|
}
|
|
static_assert(d(1)); // both-error {{static assertion failed}}
|
|
|
|
constexpr bool s(int x) { // both-note {{declared here}}
|
|
switch (x) {
|
|
case bad(x) - 100 ... bad(x) + 100: // both-error {{case value is not a constant expression}} \
|
|
// both-note {{cannot be used in a constant expression}}
|
|
return true;
|
|
default:
|
|
break;
|
|
}
|
|
return false;
|
|
}
|
|
static_assert(s(1)); // both-error {{static assertion failed}}
|
|
}
|