diff options
author | Rafael Espindola <rafael.espindola@gmail.com> | 2012-06-27 18:18:05 +0000 |
---|---|---|
committer | Rafael Espindola <rafael.espindola@gmail.com> | 2012-06-27 18:18:05 +0000 |
commit | 8d852e35adb46e0799538dfc9c80d44f27cd3597 (patch) | |
tree | d20f984c9154db4e664612770705bc793e2a3095 | |
parent | 0713d993cd0b4eb6a1b642c7d51d0f1845c1e986 (diff) |
Implement John McCall's review of r159212 other than the this pointer not
being updated. Will fix that in a second.
git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@159280 91177308-0d34-0410-b5e6-96231b3b80d8
-rw-r--r-- | include/clang/AST/Expr.h | 10 | ||||
-rw-r--r-- | lib/AST/Expr.cpp | 7 | ||||
-rw-r--r-- | lib/CodeGen/CGExprCXX.cpp | 6 | ||||
-rw-r--r-- | lib/Sema/SemaExpr.cpp | 5 |
4 files changed, 12 insertions, 16 deletions
diff --git a/include/clang/AST/Expr.h b/include/clang/AST/Expr.h index 77684eee00..7a9cfaeebe 100644 --- a/include/clang/AST/Expr.h +++ b/include/clang/AST/Expr.h @@ -665,12 +665,14 @@ public: static bool hasAnyTypeDependentArguments(llvm::ArrayRef<Expr *> Exprs); - /// \brief If we have class type (or pointer to class type), return the - /// class decl. Return NULL otherwise. + /// \brief For an expression of class type or pointer to class type, + /// return the most derived class decl the expression is known to refer to. /// /// If this expression is a cast, this method looks through it to find the - /// most derived decl that can be infered from the expression. - const CXXRecordDecl *getMostDerivedClassDeclForType() const; + /// most derived decl that can be inferred from the expression. + /// This is valid because derived-to-base conversions have undefined + /// behavior if the object isn't dynamically of the derived type. + const CXXRecordDecl *getBestDynamicClassType() const; static bool classof(const Stmt *T) { return T->getStmtClass() >= firstExprConstant && diff --git a/lib/AST/Expr.cpp b/lib/AST/Expr.cpp index 22d15be6a8..15cf6602b7 100644 --- a/lib/AST/Expr.cpp +++ b/lib/AST/Expr.cpp @@ -33,7 +33,7 @@ #include <cstring> using namespace clang; -const CXXRecordDecl *Expr::getMostDerivedClassDeclForType() const { +const CXXRecordDecl *Expr::getBestDynamicClassType() const { const Expr *E = this; while (true) { @@ -51,15 +51,10 @@ const CXXRecordDecl *Expr::getMostDerivedClassDeclForType() const { } QualType DerivedType = E->getType(); - if (DerivedType->isDependentType()) - return NULL; if (const PointerType *PTy = DerivedType->getAs<PointerType>()) DerivedType = PTy->getPointeeType(); const RecordType *Ty = DerivedType->castAs<RecordType>(); - if (!Ty) - return NULL; - Decl *D = Ty->getDecl(); return cast<CXXRecordDecl>(D); } diff --git a/lib/CodeGen/CGExprCXX.cpp b/lib/CodeGen/CGExprCXX.cpp index 372eb5407c..30324b97ef 100644 --- a/lib/CodeGen/CGExprCXX.cpp +++ b/lib/CodeGen/CGExprCXX.cpp @@ -102,8 +102,7 @@ static bool canDevirtualizeMemberFunctionCalls(ASTContext &Context, // b->f(); // } // - const CXXRecordDecl *MostDerivedClassDecl = - Base->getMostDerivedClassDeclForType(); + const CXXRecordDecl *MostDerivedClassDecl = Base->getBestDynamicClassType(); if (MostDerivedClassDecl->hasAttr<FinalAttr>()) return true; @@ -228,8 +227,7 @@ RValue CodeGenFunction::EmitCXXMemberCallExpr(const CXXMemberCallExpr *CE, bool UseVirtualCall = MD->isVirtual() && !ME->hasQualifier() && !canDevirtualizeMemberFunctionCalls(getContext(), Base, MD); - const CXXRecordDecl *MostDerivedClassDecl = - Base->getMostDerivedClassDeclForType(); + const CXXRecordDecl *MostDerivedClassDecl = Base->getBestDynamicClassType(); llvm::Value *Callee; if (const CXXDestructorDecl *Dtor = dyn_cast<CXXDestructorDecl>(MD)) { diff --git a/lib/Sema/SemaExpr.cpp b/lib/Sema/SemaExpr.cpp index bc77e6fae7..2a7a8b9f19 100644 --- a/lib/Sema/SemaExpr.cpp +++ b/lib/Sema/SemaExpr.cpp @@ -10854,8 +10854,9 @@ static void MarkExprReferenced(Sema &SemaRef, SourceLocation Loc, if (!MD) return; const Expr *Base = ME->getBase(); - const CXXRecordDecl *MostDerivedClassDecl - = Base->getMostDerivedClassDeclForType(); + if (Base->getType()->isDependentType()) + return; + const CXXRecordDecl *MostDerivedClassDecl = Base->getBestDynamicClassType(); if (!MostDerivedClassDecl) return; CXXMethodDecl *DM = MD->getCorrespondingMethodInClass(MostDerivedClassDecl); |