// 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 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}} }