From: Richard Smith Date: Sat, 2 Feb 2013 02:11:36 +0000 (+0000) Subject: Correctly classify T{} as an array temporary if T is an array of class type with... X-Git-Url: https://granicus.if.org/sourcecode?a=commitdiff_plain;h=821b93eec8b58a3e320ef34e7c98906ab61cf8c3;p=clang Correctly classify T{} as an array temporary if T is an array of class type with nontrivial destructor. git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@174261 91177308-0d34-0410-b5e6-96231b3b80d8 --- diff --git a/lib/AST/ExprClassification.cpp b/lib/AST/ExprClassification.cpp index a064541ffc..61bc3e2de5 100644 --- a/lib/AST/ExprClassification.cpp +++ b/lib/AST/ExprClassification.cpp @@ -34,21 +34,6 @@ static Cl::Kinds ClassifyConditional(ASTContext &Ctx, static Cl::ModifiableType IsModifiable(ASTContext &Ctx, const Expr *E, Cl::Kinds Kind, SourceLocation &Loc); -static Cl::Kinds ClassifyExprValueKind(const LangOptions &Lang, - const Expr *E, - ExprValueKind Kind) { - switch (Kind) { - case VK_RValue: - return Lang.CPlusPlus && E->getType()->isRecordType() ? - Cl::CL_ClassTemporary : Cl::CL_PRValue; - case VK_LValue: - return Cl::CL_LValue; - case VK_XValue: - return Cl::CL_XValue; - } - llvm_unreachable("Invalid value category of implicit cast."); -} - Cl Expr::ClassifyImpl(ASTContext &Ctx, SourceLocation *Loc) const { assert(!TR->isReferenceType() && "Expressions can't have reference type."); @@ -100,6 +85,20 @@ static Cl::Kinds ClassifyTemporary(QualType T) { return Cl::CL_PRValue; } +static Cl::Kinds ClassifyExprValueKind(const LangOptions &Lang, + const Expr *E, + ExprValueKind Kind) { + switch (Kind) { + case VK_RValue: + return Lang.CPlusPlus ? ClassifyTemporary(E->getType()) : Cl::CL_PRValue; + case VK_LValue: + return Cl::CL_LValue; + case VK_XValue: + return Cl::CL_XValue; + } + llvm_unreachable("Invalid value category of implicit cast."); +} + static Cl::Kinds ClassifyInternal(ASTContext &Ctx, const Expr *E) { // This function takes the first stab at classifying expressions. const LangOptions &Lang = Ctx.getLangOpts(); diff --git a/test/SemaCXX/address-of-temporary.cpp b/test/SemaCXX/address-of-temporary.cpp index bb6cba3187..5eef1c5521 100644 --- a/test/SemaCXX/address-of-temporary.cpp +++ b/test/SemaCXX/address-of-temporary.cpp @@ -15,8 +15,13 @@ namespace PointerToArrayDecay { struct Y { int a[4]; }; + struct Z { + int n; + ~Z(); + }; typedef int A[4]; + typedef Z AZ[4]; template void consume(T); struct S { int *p; }; @@ -25,11 +30,13 @@ namespace PointerToArrayDecay { void g1() { int *p = Y{}.a; } // expected-warning{{pointer is initialized by a temporary array}} void g2() { int *p = A{}; } // expected-warning{{pointer is initialized by a temporary array}} void g3() { int *p = (A){}; } // expected-warning{{pointer is initialized by a temporary array}} + void g4() { Z *p = AZ{}; } // expected-warning{{pointer is initialized by a temporary array}} void h0() { consume(Y().a); } void h1() { consume(Y{}.a); } void h2() { consume(A{}); } void h3() { consume((A){}); } + void h4() { consume(AZ{}); } void i0() { S s = { Y().a }; } // expected-warning{{pointer is initialized by a temporary array}} void i1() { S s = { Y{}.a }; } // expected-warning{{pointer is initialized by a temporary array}}