diff options
author | Ted Kremenek <kremenek@apple.com> | 2011-03-03 01:21:32 +0000 |
---|---|---|
committer | Ted Kremenek <kremenek@apple.com> | 2011-03-03 01:21:32 +0000 |
commit | 8cfe207861b4f7c529861bc309e8aa90776e20f6 (patch) | |
tree | e24637f65537cc697d041e21f42532d29d0911e6 | |
parent | 1676a042e30b62c09f5493929e8130170e6a3fcd (diff) | |
download | bcm5719-llvm-8cfe207861b4f7c529861bc309e8aa90776e20f6.tar.gz bcm5719-llvm-8cfe207861b4f7c529861bc309e8aa90776e20f6.zip |
Teach CFGImplicitDtor::getDestructorDecl() about arrays of objects with destructors.
llvm-svn: 126910
-rw-r--r-- | clang/include/clang/Analysis/CFG.h | 4 | ||||
-rw-r--r-- | clang/lib/Analysis/CFG.cpp | 10 | ||||
-rw-r--r-- | clang/lib/Sema/AnalysisBasedWarnings.cpp | 2 | ||||
-rw-r--r-- | clang/test/SemaCXX/return-noreturn.cpp | 4 |
4 files changed, 14 insertions, 6 deletions
diff --git a/clang/include/clang/Analysis/CFG.h b/clang/include/clang/Analysis/CFG.h index 992b32a6d74..03beba59dae 100644 --- a/clang/include/clang/Analysis/CFG.h +++ b/clang/include/clang/Analysis/CFG.h @@ -129,8 +129,8 @@ protected: } public: - const CXXDestructorDecl *getDestructorDecl() const; - bool isNoReturn() const; + const CXXDestructorDecl *getDestructorDecl(ASTContext &astContext) const; + bool isNoReturn(ASTContext &astContext) const; static bool classof(const CFGElement *E) { Kind kind = E->getKind(); diff --git a/clang/lib/Analysis/CFG.cpp b/clang/lib/Analysis/CFG.cpp index fc50071a32c..6d8d5c53f60 100644 --- a/clang/lib/Analysis/CFG.cpp +++ b/clang/lib/Analysis/CFG.cpp @@ -2772,7 +2772,8 @@ CFG* CFG::buildCFG(const Decl *D, Stmt* Statement, ASTContext *C, return Builder.buildCFG(D, Statement, C, BO); } -const CXXDestructorDecl *CFGImplicitDtor::getDestructorDecl() const { +const CXXDestructorDecl * +CFGImplicitDtor::getDestructorDecl(ASTContext &astContext) const { switch (getKind()) { case CFGElement::Invalid: case CFGElement::Statement: @@ -2783,6 +2784,9 @@ const CXXDestructorDecl *CFGImplicitDtor::getDestructorDecl() const { const VarDecl *var = cast<CFGAutomaticObjDtor>(this)->getVarDecl(); QualType ty = var->getType(); ty = ty.getNonReferenceType(); + if (const ArrayType *arrayType = astContext.getAsArrayType(ty)) { + ty = arrayType->getElementType(); + } const RecordType *recordType = ty->getAs<RecordType>(); const CXXRecordDecl *classDecl = cast<CXXRecordDecl>(recordType->getDecl()); @@ -2804,8 +2808,8 @@ const CXXDestructorDecl *CFGImplicitDtor::getDestructorDecl() const { return 0; } -bool CFGImplicitDtor::isNoReturn() const { - if (const CXXDestructorDecl *cdecl = getDestructorDecl()) { +bool CFGImplicitDtor::isNoReturn(ASTContext &astContext) const { + if (const CXXDestructorDecl *cdecl = getDestructorDecl(astContext)) { QualType ty = cdecl->getType(); return cast<FunctionType>(ty)->getNoReturnAttr(); } diff --git a/clang/lib/Sema/AnalysisBasedWarnings.cpp b/clang/lib/Sema/AnalysisBasedWarnings.cpp index 84efbd50d1a..710f5ef70fc 100644 --- a/clang/lib/Sema/AnalysisBasedWarnings.cpp +++ b/clang/lib/Sema/AnalysisBasedWarnings.cpp @@ -137,7 +137,7 @@ static ControlFlowKind CheckFallThrough(AnalysisContext &AC) { // FIXME: The right solution is to just sever the edges in the // CFG itself. if (const CFGImplicitDtor *iDtor = ri->getAs<CFGImplicitDtor>()) - if (iDtor->isNoReturn()) { + if (iDtor->isNoReturn(AC.getASTContext())) { hasNoReturnDtor = true; HasFakeEdge = true; break; diff --git a/clang/test/SemaCXX/return-noreturn.cpp b/clang/test/SemaCXX/return-noreturn.cpp index 4fb3732201c..5f044ca1794 100644 --- a/clang/test/SemaCXX/return-noreturn.cpp +++ b/clang/test/SemaCXX/return-noreturn.cpp @@ -39,3 +39,7 @@ void test_PR9380(const PR9380& aKey) { const PR9380& flatKey = PR9380_B(aKey); } +// Array of objects with destructors. This is purely a coverage test case. +void test_array() { + PR9380 a[2]; +} |