// RUN: %clang_cc1 -std=c++1z -verify %s // RUN: %clang_cc1 -std=c++1z -verify %s -DUNDEFINED #ifdef UNDEFINED // "used but not defined" errors don't get produced if we have more interesting // errors. namespace std_example { template void g(T &&p, Rest &&... rs) { // use p if constexpr(sizeof...(rs) > 0) g(rs...); } void use_g() { g(1, 2, 3); } static int x(); // no definition of x required int f() { if constexpr (true) return 0; else if (x()) return x(); else return -x(); } } namespace odr_use_in_selected_arm { static int x(); // expected-warning {{is not defined}} int f() { if constexpr (false) return 0; else if (x()) // expected-note {{here}} return x(); else return -x(); } } #else namespace ccce { struct S { }; void f() { if (5) {} if constexpr (5) { } } template void g() { if constexpr (N) { } } template void g<5>(); void h() { if constexpr (4.3) { //expected-warning {{implicit conversion from 'double' to 'bool' changes value}} } constexpr void *p = nullptr; if constexpr (p) { } } void not_constant(int b, S s) { // expected-note 2{{declared here}} if constexpr (bool(b)) { // expected-error {{constexpr if condition is not a constant expression}} expected-note {{cannot be used in a constant expression}} } if constexpr (b) { // expected-error {{constexpr if condition is not a constant expression}} expected-note {{cannot be used in a constant expression}} } if constexpr (s) { // expected-error {{value of type 'S' is not contextually convertible to 'bool'}} } constexpr S constexprS; if constexpr (constexprS) { // expected-error {{value of type 'const S' is not contextually convertible to 'bool'}} } } } namespace generic_lambda { // Substituting for T produces a hard error here, even if substituting for // the type of x would remove the error. template void f() { [](auto x) { if constexpr (sizeof(T) == 1 && sizeof(x) == 1) T::error(); // expected-error 2{{'::'}} // expected-note@-3 2{{while substituting into a lambda expression here}} } (0); } template void g() { [](auto x) { if constexpr (sizeof(T) == 1) if constexpr (sizeof(x) == 1) T::error(); // expected-error {{'::'}} // expected-note@-4 {{while substituting into a lambda expression here}} } (0); } void use() { f(); // expected-note {{instantiation of}} f(); // expected-note {{instantiation of}} g(); // ok g(); // expected-note {{instantiation of}} } } namespace potentially_discarded_branch_target { void in_switch(int n) { switch (n) case 4: if constexpr(sizeof(n) == 4) return; if constexpr(sizeof(n) == 4) switch (n) case 4: return; switch (n) { if constexpr (sizeof(n) == 4) // expected-note 2{{constexpr if}} case 4: return; // expected-error {{cannot jump}} else default: break; // expected-error {{cannot jump}} } } template void in_switch_tmpl(int n) { switch (n) { if constexpr (sizeof(T) == 4) // expected-note 2{{constexpr if}} case 4: return; // expected-error {{cannot jump}} else default: break; // expected-error {{cannot jump}} } } void goto_scope(int n) { goto foo; // expected-error {{cannot jump}} if constexpr(sizeof(n) == 4) // expected-note {{constexpr if}} foo: return; bar: if constexpr(sizeof(n) == 4) goto bar; // ok } template void goto_scope(int n) { goto foo; // expected-error {{cannot jump}} if constexpr(sizeof(n) == 4) // expected-note {{constexpr if}} foo: return; bar: if constexpr(sizeof(n) == 4) goto bar; // ok } void goto_redef(int n) { a: if constexpr(sizeof(n) == 4) // expected-error {{redefinition}} expected-note {{constexpr if}} a: goto a; // expected-note 2{{previous}} else a: goto a; // expected-error {{redefinition}} expected-error {{cannot jump}} } void evil_things() { goto evil_label; // expected-error {{cannot jump}} if constexpr (true || ({evil_label: false;})) {} // expected-note {{constexpr if}} \ // expected-note {{jump enters a statement expression}} if constexpr (true) // expected-note {{constexpr if}} goto surprise; // expected-error {{cannot jump}} else surprise: {} } } namespace deduced_return_type_in_discareded_statement { template auto a(const T &t) { return t; } void f() { if constexpr (false) { a(a(0)); } } } // namespace deduced_return_type_in_discareded_statement #endif