diff options
author | Ted Kremenek <kremenek@apple.com> | 2011-02-12 00:17:19 +0000 |
---|---|---|
committer | Ted Kremenek <kremenek@apple.com> | 2011-02-12 00:17:19 +0000 |
commit | b1c392aa56225c5b8dc82ee5af502f58c7fbafce (patch) | |
tree | e1915a1e55eba3a8cebf4a434656527fa8954f61 | |
parent | a222c0458897b5b54a3cb5a59b6cbbb2a45aca2c (diff) | |
download | bcm5719-llvm-b1c392aa56225c5b8dc82ee5af502f58c7fbafce.tar.gz bcm5719-llvm-b1c392aa56225c5b8dc82ee5af502f58c7fbafce.zip |
Don't emit a dead store for '++' operations unless it occurs with a return statement. We've never seen any other cases that were real bugs.
Fixes <rdar://problem/6962292>.
llvm-svn: 125419
-rw-r--r-- | clang/include/clang/AST/ParentMap.h | 5 | ||||
-rw-r--r-- | clang/lib/AST/ParentMap.cpp | 9 | ||||
-rw-r--r-- | clang/lib/StaticAnalyzer/Checkers/DeadStoresChecker.cpp | 11 | ||||
-rw-r--r-- | clang/test/Analysis/dead-stores.c | 7 | ||||
-rw-r--r-- | clang/test/Analysis/dead-stores.cpp | 6 |
5 files changed, 26 insertions, 12 deletions
diff --git a/clang/include/clang/AST/ParentMap.h b/clang/include/clang/AST/ParentMap.h index 600321268e5..9ea5a0930d3 100644 --- a/clang/include/clang/AST/ParentMap.h +++ b/clang/include/clang/AST/ParentMap.h @@ -31,6 +31,7 @@ public: Stmt *getParent(Stmt*) const; Stmt *getParentIgnoreParens(Stmt *) const; + Stmt *getParentIgnoreParenCasts(Stmt *) const; const Stmt *getParent(const Stmt* S) const { return getParent(const_cast<Stmt*>(S)); @@ -40,6 +41,10 @@ public: return getParentIgnoreParens(const_cast<Stmt*>(S)); } + const Stmt *getParentIgnoreParenCasts(const Stmt *S) const { + return getParentIgnoreParenCasts(const_cast<Stmt*>(S)); + } + bool hasParent(Stmt* S) const { return getParent(S) != 0; } diff --git a/clang/lib/AST/ParentMap.cpp b/clang/lib/AST/ParentMap.cpp index 21847f28217..87f8f36e6e2 100644 --- a/clang/lib/AST/ParentMap.cpp +++ b/clang/lib/AST/ParentMap.cpp @@ -57,6 +57,15 @@ Stmt *ParentMap::getParentIgnoreParens(Stmt *S) const { return S; } +Stmt *ParentMap::getParentIgnoreParenCasts(Stmt *S) const { + do { + S = getParent(S); + } + while (S && (isa<ParenExpr>(S) || isa<CastExpr>(S))); + + return S; +} + bool ParentMap::isConsumedExpr(Expr* E) const { Stmt *P = getParent(E); Stmt *DirectChild = E; diff --git a/clang/lib/StaticAnalyzer/Checkers/DeadStoresChecker.cpp b/clang/lib/StaticAnalyzer/Checkers/DeadStoresChecker.cpp index 442e1b3af78..a6c0ea3154e 100644 --- a/clang/lib/StaticAnalyzer/Checkers/DeadStoresChecker.cpp +++ b/clang/lib/StaticAnalyzer/Checkers/DeadStoresChecker.cpp @@ -191,6 +191,8 @@ public: if (S->getLocStart().isMacroID()) return; + // Only cover dead stores from regular assignments. ++/-- dead stores + // have never flagged a real bug. if (BinaryOperator* B = dyn_cast<BinaryOperator>(S)) { if (!B->isAssignmentOp()) return; // Skip non-assignments. @@ -221,14 +223,11 @@ public: } } else if (UnaryOperator* U = dyn_cast<UnaryOperator>(S)) { - if (!U->isIncrementOp()) + if (!U->isIncrementOp() || U->isPrefix()) return; - // Handle: ++x within a subexpression. The solution is not warn - // about preincrements to dead variables when the preincrement occurs - // as a subexpression. This can lead to false negatives, e.g. "(++x);" - // A generalized dead code checker should find such issues. - if (U->isPrefix() && Parents.isConsumedExpr(U)) + Stmt *parent = Parents.getParentIgnoreParenCasts(U); + if (!parent || !isa<ReturnStmt>(parent)) return; Expr *Ex = U->getSubExpr()->IgnoreParenCasts(); diff --git a/clang/test/Analysis/dead-stores.c b/clang/test/Analysis/dead-stores.c index 9e3f736d770..942b80e70e1 100644 --- a/clang/test/Analysis/dead-stores.c +++ b/clang/test/Analysis/dead-stores.c @@ -44,10 +44,11 @@ void f5() { } +// int f6() { int x = 4; - ++x; // expected-warning{{never read}} + ++x; // no-warning return 1; } @@ -231,7 +232,7 @@ void halt() __attribute__((noreturn)); int f21() { int x = 4; - ++x; // expected-warning{{never read}} + x = x + 1; // expected-warning{{never read}} if (1) { halt(); (void)x; @@ -263,7 +264,7 @@ void f22() { int y19 = 4; int y20 = 4; - ++x; // expected-warning{{never read}} + x = x + 1; // expected-warning{{never read}} ++y1; ++y2; ++y3; diff --git a/clang/test/Analysis/dead-stores.cpp b/clang/test/Analysis/dead-stores.cpp index b5fc05028f7..9648baceb1b 100644 --- a/clang/test/Analysis/dead-stores.cpp +++ b/clang/test/Analysis/dead-stores.cpp @@ -12,7 +12,7 @@ int j; void test1() { int x = 4; - ++x; // expected-warning{{never read}} + x = x + 1; // expected-warning{{never read}} switch (j) { case 1: @@ -69,11 +69,11 @@ void test2_b() { //===----------------------------------------------------------------------===// void test3_a(int x) { - ++x; // expected-warning{{never read}} + x = x + 1; // expected-warning{{never read}} } void test3_b(int &x) { - ++x; // no-warninge + x = x + 1; // no-warninge } void test3_c(int x) { |