// RUN: %clang_cc1 -verify=expected,both -std=c++20 %s -fexperimental-new-constant-interpreter // RUN: %clang_cc1 -verify=ref,both -std=c++20 %s /// FIXME: Slight difference in diagnostic output here. struct Foo { int a; }; constexpr int dead1() { Foo *F2 = nullptr; { Foo F{12}; // expected-note {{declared here}} F2 = &F; } // Ends lifetime of F. return F2->a; // expected-note {{read of variable whose lifetime has ended}} \ // ref-note {{read of object outside its lifetime is not allowed in a constant expression}} } static_assert(dead1() == 1, ""); // both-error {{not an integral constant expression}} \ // both-note {{in call to}} struct S { int &&r; // both-note {{reference member declared here}} int t; constexpr S() : r(0), t(r) {} // both-error {{reference member 'r' binds to a temporary object whose lifetime would be shorter than the lifetime of the constructed object}} \ // ref-note {{read of object outside its lifetime is not allowed in a constant expression}} \ // expected-note {{temporary created here}} \ // expected-note {{read of temporary whose lifetime has ended}} }; constexpr int k1 = S().t; // both-error {{must be initialized by a constant expression}} \ // ref-note {{in call to}} \ // expected-note {{in call to}} namespace MoveFnWorks { template constexpr T &&ref(T &&t) { return (T&&)t; } struct Buf {}; struct A { constexpr A(Buf &buf) : buf(buf) { } Buf &buf; }; constexpr bool dtor_calls_dtor() { struct B { A &&d; constexpr B(Buf &buf) : d(ref(A(buf))) {} }; Buf buf; { B b(buf); } return true; } static_assert(dtor_calls_dtor(), ""); } namespace PrimitiveMoveFn { /// This used to crash. void test() { const float y = 100; const float &x = y; } } /// FIXME: /// 1) This doesn't work for parameters /// 2) We need to do this for all fields in composite scenarios namespace PseudoDtor { typedef int I; constexpr bool foo() { // both-error {{never produces a constant expression}} { int a; // both-note {{destroying object 'a' whose lifetime has already ended}} a.~I(); } return true; } int k; struct T { int n : (k.~I(), 1); // both-error {{constant expression}} \ // both-note {{visible outside that expression}} }; } /// Diagnostic differences namespace CallScope { struct Q { int n = 0; constexpr int f() const { return 0; } }; constexpr Q *out_of_lifetime(Q q) { return &q; } // both-warning {{address of stack}} constexpr int k3 = out_of_lifetime({})->n; // both-error {{must be initialized by a constant expression}} \ // expected-note {{read of temporary whose lifetime has ended}} \ // expected-note {{temporary created here}} \ // ref-note {{read of object outside its lifetime}} constexpr int k4 = out_of_lifetime({})->f(); // both-error {{must be initialized by a constant expression}} \ // expected-note {{member call on temporary whose lifetime has ended}} \ // expected-note {{temporary created here}} \ // ref-note {{member call on object outside its lifetime}} }