From 9a84dc0b360662e352428b2dba85c0351d3d1328 Mon Sep 17 00:00:00 2001 From: Richard Smith Date: Tue, 1 Oct 2019 00:07:14 +0000 Subject: [PATCH] [c++20] Fix crash when constant-evaluating an assignment with a reference member access on its left-hand side. llvm-svn: 373276 --- clang/lib/AST/ExprConstant.cpp | 4 +++- clang/test/SemaCXX/constant-expression-cxx2a.cpp | 23 +++++++++++++++++++++++ 2 files changed, 26 insertions(+), 1 deletion(-) diff --git a/clang/lib/AST/ExprConstant.cpp b/clang/lib/AST/ExprConstant.cpp index f91e078..5659a0e 100644 --- a/clang/lib/AST/ExprConstant.cpp +++ b/clang/lib/AST/ExprConstant.cpp @@ -5258,7 +5258,9 @@ static bool HandleUnionActiveMemberChange(EvalInfo &Info, const Expr *LHSExpr, // -- If E is of the form A.B, S(E) contains the elements of S(A)... if (auto *ME = dyn_cast(E)) { auto *FD = dyn_cast(ME->getMemberDecl()); - if (!FD) + // Note that we can't implicitly start the lifetime of a reference, + // so we don't need to proceed any further if we reach one. + if (!FD || FD->getType()->isReferenceType()) break; // ... and also contains A.B if B names a union member diff --git a/clang/test/SemaCXX/constant-expression-cxx2a.cpp b/clang/test/SemaCXX/constant-expression-cxx2a.cpp index 08fcfdd..2f23975 100644 --- a/clang/test/SemaCXX/constant-expression-cxx2a.cpp +++ b/clang/test/SemaCXX/constant-expression-cxx2a.cpp @@ -561,6 +561,29 @@ namespace Union { S3 s; s.n = 0; } + + union ref_member_1 { + int a; + int b; + }; + struct ref_member_2 { + ref_member_1 &&r; + }; + union ref_member_3 { + ref_member_2 a, b; + }; + constexpr int ref_member_test_1() { + ref_member_3 r = {.a = {.r = {.a = 1}}}; + r.a.r.b = 2; + return r.a.r.b; + } + static_assert(ref_member_test_1() == 2); + constexpr int ref_member_test_2() { // expected-error {{never produces a constant}} + ref_member_3 r = {.a = {.r = {.a = 1}}}; + // FIXME: This note isn't great. The 'read' here is reading the referent of the reference. + r.b.r.b = 2; // expected-note {{read of member 'b' of union with active member 'a'}} + return r.b.r.b; + } } namespace TwosComplementShifts { -- 2.7.4