From 1de2e14f2f3a0f703298f111146f61fbaec1b752 Mon Sep 17 00:00:00 2001 From: Ted Kremenek Date: Thu, 6 Mar 2014 00:17:44 +0000 Subject: [-Wunreachable-code] Handle idiomatic do...while() with an uninteresting condition. Sometimes do..while() is used to create a scope that can be left early. In such cases, the unreachable 'while()' test is not usually interesting unless it actually does something that is observable. llvm-svn: 203036 --- clang/lib/Analysis/ReachableCode.cpp | 24 +++++++++++++++++------- 1 file changed, 17 insertions(+), 7 deletions(-) (limited to 'clang/lib/Analysis') diff --git a/clang/lib/Analysis/ReachableCode.cpp b/clang/lib/Analysis/ReachableCode.cpp index 7958840b89d..9f6792e2cc7 100644 --- a/clang/lib/Analysis/ReachableCode.cpp +++ b/clang/lib/Analysis/ReachableCode.cpp @@ -296,10 +296,7 @@ static bool isTrivialExpression(const Expr *Ex) { isEnumConstant(Ex); } -static bool isTrivialReturn(const CFGBlock *B, const Stmt *S) { - if (B->pred_empty()) - return false; - +static bool isTrivialReturnOrDoWhile(const CFGBlock *B, const Stmt *S) { const Expr *Ex = dyn_cast(S); if (!Ex) return false; @@ -307,6 +304,16 @@ static bool isTrivialReturn(const CFGBlock *B, const Stmt *S) { if (!isTrivialExpression(Ex)) return false; + // Check if the block ends with a do...while() and see if 'S' is the + // condition. + if (const Stmt *Term = B->getTerminator()) { + if (const DoStmt *DS = dyn_cast(Term)) + if (DS->getCond() == S) + return true; + } + + + // Look to see if the block ends with a 'return', and see if 'S' // is a substatement. The 'return' may not be the last element in // the block because of destructors. @@ -317,12 +324,15 @@ static bool isTrivialReturn(const CFGBlock *B, const Stmt *S) { if (const ReturnStmt *RS = dyn_cast(CS->getStmt())) { const Expr *RE = RS->getRetValue(); if (RE && RE->IgnoreParenCasts() == Ex) - return true; + break; } - break; + return false; } } + if (B->pred_size() == 1) + return bodyEndsWithNoReturn(*B->pred_begin()); + return false; } @@ -336,7 +346,7 @@ void DeadCodeScan::reportDeadCode(const CFGBlock *B, return; // Suppress trivial 'return' statements that are dead. - if (isTrivialReturn(B, S)) + if (isTrivialReturnOrDoWhile(B, S)) return; SourceRange R1, R2; -- cgit v1.2.3