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 &&
#include <cstring>
using namespace clang;
-const CXXRecordDecl *Expr::getMostDerivedClassDeclForType() const {
+const CXXRecordDecl *Expr::getBestDynamicClassType() const {
const Expr *E = this;
while (true) {
}
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);
}
// b->f();
// }
//
- const CXXRecordDecl *MostDerivedClassDecl =
- Base->getMostDerivedClassDeclForType();
+ const CXXRecordDecl *MostDerivedClassDecl = Base->getBestDynamicClassType();
if (MostDerivedClassDecl->hasAttr<FinalAttr>())
return true;
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)) {
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);