| Tom Care | cba9f51 | 2010-07-23 23:04:53 +0000 | [diff] [blame] | 1 | //==- UnreachableCodeChecker.cpp - Generalized dead code checker -*- C++ -*-==// | 
|  | 2 | // | 
|  | 3 | //                     The LLVM Compiler Infrastructure | 
|  | 4 | // | 
|  | 5 | // This file is distributed under the University of Illinois Open Source | 
|  | 6 | // License. See LICENSE.TXT for details. | 
|  | 7 | // | 
|  | 8 | //===----------------------------------------------------------------------===// | 
|  | 9 | // This file implements a generalized unreachable code checker using a | 
|  | 10 | // path-sensitive analysis. We mark any path visited, and then walk the CFG as a | 
|  | 11 | // post-analysis to determine what was never visited. | 
|  | 12 | // | 
| Jordy Rose | 55442ab | 2010-07-27 03:39:53 +0000 | [diff] [blame] | 13 | // A similar flow-sensitive only check exists in Analysis/ReachableCode.cpp | 
| Tom Care | cba9f51 | 2010-07-23 23:04:53 +0000 | [diff] [blame] | 14 | //===----------------------------------------------------------------------===// | 
|  | 15 |  | 
| Argyrios Kyrtzidis | 2d3905f | 2011-02-15 21:25:03 +0000 | [diff] [blame] | 16 | #include "ClangSACheckers.h" | 
| Argyrios Kyrtzidis | bf61d97 | 2011-02-23 07:19:23 +0000 | [diff] [blame] | 17 | #include "clang/AST/ParentMap.h" | 
|  | 18 | #include "clang/Basic/Builtins.h" | 
|  | 19 | #include "clang/Basic/SourceManager.h" | 
| Chandler Carruth | 3a02247 | 2012-12-04 09:13:33 +0000 | [diff] [blame] | 20 | #include "clang/StaticAnalyzer/Core/BugReporter/BugReporter.h" | 
|  | 21 | #include "clang/StaticAnalyzer/Core/Checker.h" | 
|  | 22 | #include "clang/StaticAnalyzer/Core/CheckerManager.h" | 
|  | 23 | #include "clang/StaticAnalyzer/Core/PathSensitive/CheckerContext.h" | 
|  | 24 | #include "clang/StaticAnalyzer/Core/PathSensitive/CheckerHelpers.h" | 
|  | 25 | #include "clang/StaticAnalyzer/Core/PathSensitive/ExplodedGraph.h" | 
|  | 26 | #include "clang/StaticAnalyzer/Core/PathSensitive/SVals.h" | 
| Benjamin Kramer | ac19edd | 2012-04-01 19:30:51 +0000 | [diff] [blame] | 27 | #include "llvm/ADT/SmallSet.h" | 
| Tom Care | cba9f51 | 2010-07-23 23:04:53 +0000 | [diff] [blame] | 28 |  | 
|  | 29 | // The number of CFGBlock pointers we want to reserve memory for. This is used | 
|  | 30 | // once for each function we analyze. | 
|  | 31 | #define DEFAULT_CFGBLOCKS 256 | 
|  | 32 |  | 
|  | 33 | using namespace clang; | 
| Ted Kremenek | 98857c9 | 2010-12-23 07:20:52 +0000 | [diff] [blame] | 34 | using namespace ento; | 
| Tom Care | cba9f51 | 2010-07-23 23:04:53 +0000 | [diff] [blame] | 35 |  | 
|  | 36 | namespace { | 
| Argyrios Kyrtzidis | 6a5674f | 2011-03-01 01:16:21 +0000 | [diff] [blame] | 37 | class UnreachableCodeChecker : public Checker<check::EndAnalysis> { | 
| Tom Care | cba9f51 | 2010-07-23 23:04:53 +0000 | [diff] [blame] | 38 | public: | 
| Argyrios Kyrtzidis | bf61d97 | 2011-02-23 07:19:23 +0000 | [diff] [blame] | 39 | void checkEndAnalysis(ExplodedGraph &G, BugReporter &B, | 
|  | 40 | ExprEngine &Eng) const; | 
| Tom Care | cba9f51 | 2010-07-23 23:04:53 +0000 | [diff] [blame] | 41 | private: | 
| Argyrios Kyrtzidis | bf61d97 | 2011-02-23 07:19:23 +0000 | [diff] [blame] | 42 | typedef llvm::SmallSet<unsigned, DEFAULT_CFGBLOCKS> CFGBlocksSet; | 
|  | 43 |  | 
| Tom Care | be633d9 | 2010-08-03 21:24:13 +0000 | [diff] [blame] | 44 | static inline const Stmt *getUnreachableStmt(const CFGBlock *CB); | 
| Argyrios Kyrtzidis | bf61d97 | 2011-02-23 07:19:23 +0000 | [diff] [blame] | 45 | static void FindUnreachableEntryPoints(const CFGBlock *CB, | 
|  | 46 | CFGBlocksSet &reachable, | 
|  | 47 | CFGBlocksSet &visited); | 
| Tom Care | 29a6250 | 2010-07-27 23:30:21 +0000 | [diff] [blame] | 48 | static bool isInvalidPath(const CFGBlock *CB, const ParentMap &PM); | 
| Tom Care | af9bbad | 2010-08-12 23:01:06 +0000 | [diff] [blame] | 49 | static inline bool isEmptyCFGBlock(const CFGBlock *CB); | 
| Tom Care | cba9f51 | 2010-07-23 23:04:53 +0000 | [diff] [blame] | 50 | }; | 
|  | 51 | } | 
|  | 52 |  | 
| Argyrios Kyrtzidis | bf61d97 | 2011-02-23 07:19:23 +0000 | [diff] [blame] | 53 | void UnreachableCodeChecker::checkEndAnalysis(ExplodedGraph &G, | 
| Tom Care | cba9f51 | 2010-07-23 23:04:53 +0000 | [diff] [blame] | 54 | BugReporter &B, | 
| Argyrios Kyrtzidis | bf61d97 | 2011-02-23 07:19:23 +0000 | [diff] [blame] | 55 | ExprEngine &Eng) const { | 
|  | 56 | CFGBlocksSet reachable, visited; | 
| Ted Kremenek | 85825ae | 2011-12-01 00:59:17 +0000 | [diff] [blame] | 57 |  | 
| Tom Care | 44081fb | 2010-08-03 01:55:07 +0000 | [diff] [blame] | 58 | if (Eng.hasWorkRemaining()) | 
| Tom Care | cba9f51 | 2010-07-23 23:04:53 +0000 | [diff] [blame] | 59 | return; | 
|  | 60 |  | 
| Craig Topper | 0dbb783 | 2014-05-27 02:45:47 +0000 | [diff] [blame] | 61 | const Decl *D = nullptr; | 
|  | 62 | CFG *C = nullptr; | 
|  | 63 | ParentMap *PM = nullptr; | 
|  | 64 | const LocationContext *LC = nullptr; | 
| Tom Care | cba9f51 | 2010-07-23 23:04:53 +0000 | [diff] [blame] | 65 | // Iterate over ExplodedGraph | 
| Tom Care | af9bbad | 2010-08-12 23:01:06 +0000 | [diff] [blame] | 66 | for (ExplodedGraph::node_iterator I = G.nodes_begin(), E = G.nodes_end(); | 
|  | 67 | I != E; ++I) { | 
| Tom Care | cba9f51 | 2010-07-23 23:04:53 +0000 | [diff] [blame] | 68 | const ProgramPoint &P = I->getLocation(); | 
| Anna Zaks | c29bed3 | 2011-09-20 21:38:35 +0000 | [diff] [blame] | 69 | LC = P.getLocationContext(); | 
| Jordan Rose | 95cdf9d | 2013-08-19 17:03:12 +0000 | [diff] [blame] | 70 | if (!LC->inTopFrame()) | 
|  | 71 | continue; | 
| Tom Care | cba9f51 | 2010-07-23 23:04:53 +0000 | [diff] [blame] | 72 |  | 
| Ted Kremenek | 85825ae | 2011-12-01 00:59:17 +0000 | [diff] [blame] | 73 | if (!D) | 
|  | 74 | D = LC->getAnalysisDeclContext()->getDecl(); | 
| Jordan Rose | 95cdf9d | 2013-08-19 17:03:12 +0000 | [diff] [blame] | 75 |  | 
| Tom Care | cba9f51 | 2010-07-23 23:04:53 +0000 | [diff] [blame] | 76 | // Save the CFG if we don't have it already | 
|  | 77 | if (!C) | 
| Ted Kremenek | 81ce1c8 | 2011-10-24 01:32:45 +0000 | [diff] [blame] | 78 | C = LC->getAnalysisDeclContext()->getUnoptimizedCFG(); | 
| Tom Care | 29a6250 | 2010-07-27 23:30:21 +0000 | [diff] [blame] | 79 | if (!PM) | 
|  | 80 | PM = &LC->getParentMap(); | 
| Tom Care | cba9f51 | 2010-07-23 23:04:53 +0000 | [diff] [blame] | 81 |  | 
| David Blaikie | 87396b9 | 2013-02-21 22:23:56 +0000 | [diff] [blame] | 82 | if (Optional<BlockEntrance> BE = P.getAs<BlockEntrance>()) { | 
| Tom Care | cba9f51 | 2010-07-23 23:04:53 +0000 | [diff] [blame] | 83 | const CFGBlock *CB = BE->getBlock(); | 
| Tom Care | be633d9 | 2010-08-03 21:24:13 +0000 | [diff] [blame] | 84 | reachable.insert(CB->getBlockID()); | 
| Tom Care | cba9f51 | 2010-07-23 23:04:53 +0000 | [diff] [blame] | 85 | } | 
|  | 86 | } | 
|  | 87 |  | 
| Tom Care | 29a6250 | 2010-07-27 23:30:21 +0000 | [diff] [blame] | 88 | // Bail out if we didn't get the CFG or the ParentMap. | 
| Ted Kremenek | 85825ae | 2011-12-01 00:59:17 +0000 | [diff] [blame] | 89 | if (!D || !C || !PM) | 
| Tom Care | cba9f51 | 2010-07-23 23:04:53 +0000 | [diff] [blame] | 90 | return; | 
| Ted Kremenek | 85825ae | 2011-12-01 00:59:17 +0000 | [diff] [blame] | 91 |  | 
|  | 92 | // Don't do anything for template instantiations.  Proving that code | 
|  | 93 | // in a template instantiation is unreachable means proving that it is | 
|  | 94 | // unreachable in all instantiations. | 
|  | 95 | if (const FunctionDecl *FD = dyn_cast<FunctionDecl>(D)) | 
|  | 96 | if (FD->isTemplateInstantiation()) | 
|  | 97 | return; | 
| Tom Care | cba9f51 | 2010-07-23 23:04:53 +0000 | [diff] [blame] | 98 |  | 
|  | 99 | // Find CFGBlocks that were not covered by any node | 
| Tom Care | ea53e82 | 2010-10-06 23:02:25 +0000 | [diff] [blame] | 100 | for (CFG::const_iterator I = C->begin(), E = C->end(); I != E; ++I) { | 
| Tom Care | cba9f51 | 2010-07-23 23:04:53 +0000 | [diff] [blame] | 101 | const CFGBlock *CB = *I; | 
|  | 102 | // Check if the block is unreachable | 
| Tom Care | be633d9 | 2010-08-03 21:24:13 +0000 | [diff] [blame] | 103 | if (reachable.count(CB->getBlockID())) | 
| Tom Care | 29a6250 | 2010-07-27 23:30:21 +0000 | [diff] [blame] | 104 | continue; | 
| Tom Care | cba9f51 | 2010-07-23 23:04:53 +0000 | [diff] [blame] | 105 |  | 
| Tom Care | af9bbad | 2010-08-12 23:01:06 +0000 | [diff] [blame] | 106 | // Check if the block is empty (an artificial block) | 
|  | 107 | if (isEmptyCFGBlock(CB)) | 
|  | 108 | continue; | 
|  | 109 |  | 
| Tom Care | 29a6250 | 2010-07-27 23:30:21 +0000 | [diff] [blame] | 110 | // Find the entry points for this block | 
| Tom Care | ea53e82 | 2010-10-06 23:02:25 +0000 | [diff] [blame] | 111 | if (!visited.count(CB->getBlockID())) | 
| Argyrios Kyrtzidis | bf61d97 | 2011-02-23 07:19:23 +0000 | [diff] [blame] | 112 | FindUnreachableEntryPoints(CB, reachable, visited); | 
| Jordy Rose | 55442ab | 2010-07-27 03:39:53 +0000 | [diff] [blame] | 113 |  | 
| Tom Care | 29a6250 | 2010-07-27 23:30:21 +0000 | [diff] [blame] | 114 | // This block may have been pruned; check if we still want to report it | 
| Tom Care | be633d9 | 2010-08-03 21:24:13 +0000 | [diff] [blame] | 115 | if (reachable.count(CB->getBlockID())) | 
| Tom Care | 29a6250 | 2010-07-27 23:30:21 +0000 | [diff] [blame] | 116 | continue; | 
|  | 117 |  | 
|  | 118 | // Check for false positives | 
|  | 119 | if (CB->size() > 0 && isInvalidPath(CB, *PM)) | 
|  | 120 | continue; | 
|  | 121 |  | 
| Ted Kremenek | ef5c554 | 2012-02-29 06:05:28 +0000 | [diff] [blame] | 122 | // It is good practice to always have a "default" label in a "switch", even | 
|  | 123 | // if we should never get there. It can be used to detect errors, for | 
|  | 124 | // instance. Unreachable code directly under a "default" label is therefore | 
|  | 125 | // likely to be a false positive. | 
|  | 126 | if (const Stmt *label = CB->getLabel()) | 
|  | 127 | if (label->getStmtClass() == Stmt::DefaultStmtClass) | 
|  | 128 | continue; | 
|  | 129 |  | 
| Tom Care | 29a6250 | 2010-07-27 23:30:21 +0000 | [diff] [blame] | 130 | // Special case for __builtin_unreachable. | 
|  | 131 | // FIXME: This should be extended to include other unreachable markers, | 
|  | 132 | // such as llvm_unreachable. | 
|  | 133 | if (!CB->empty()) { | 
| Ted Kremenek | e9fda1e | 2011-07-28 23:07:59 +0000 | [diff] [blame] | 134 | bool foundUnreachable = false; | 
|  | 135 | for (CFGBlock::const_iterator ci = CB->begin(), ce = CB->end(); | 
|  | 136 | ci != ce; ++ci) { | 
| David Blaikie | 00be69a | 2013-02-23 00:29:34 +0000 | [diff] [blame] | 137 | if (Optional<CFGStmt> S = (*ci).getAs<CFGStmt>()) | 
|  | 138 | if (const CallExpr *CE = dyn_cast<CallExpr>(S->getStmt())) { | 
| Alp Toker | a724cff | 2013-12-28 21:59:02 +0000 | [diff] [blame] | 139 | if (CE->getBuiltinCallee() == Builtin::BI__builtin_unreachable) { | 
| Ted Kremenek | e9fda1e | 2011-07-28 23:07:59 +0000 | [diff] [blame] | 140 | foundUnreachable = true; | 
|  | 141 | break; | 
|  | 142 | } | 
|  | 143 | } | 
| Jordy Rose | 55442ab | 2010-07-27 03:39:53 +0000 | [diff] [blame] | 144 | } | 
| Ted Kremenek | e9fda1e | 2011-07-28 23:07:59 +0000 | [diff] [blame] | 145 | if (foundUnreachable) | 
|  | 146 | continue; | 
| Tom Care | cba9f51 | 2010-07-23 23:04:53 +0000 | [diff] [blame] | 147 | } | 
| Tom Care | 29a6250 | 2010-07-27 23:30:21 +0000 | [diff] [blame] | 148 |  | 
| Tom Care | be633d9 | 2010-08-03 21:24:13 +0000 | [diff] [blame] | 149 | // We found a block that wasn't covered - find the statement to report | 
|  | 150 | SourceRange SR; | 
| Anna Zaks | c29bed3 | 2011-09-20 21:38:35 +0000 | [diff] [blame] | 151 | PathDiagnosticLocation DL; | 
| Tom Care | be633d9 | 2010-08-03 21:24:13 +0000 | [diff] [blame] | 152 | SourceLocation SL; | 
|  | 153 | if (const Stmt *S = getUnreachableStmt(CB)) { | 
|  | 154 | SR = S->getSourceRange(); | 
| Anna Zaks | c29bed3 | 2011-09-20 21:38:35 +0000 | [diff] [blame] | 155 | DL = PathDiagnosticLocation::createBegin(S, B.getSourceManager(), LC); | 
|  | 156 | SL = DL.asLocation(); | 
|  | 157 | if (SR.isInvalid() || !SL.isValid()) | 
| Tom Care | be633d9 | 2010-08-03 21:24:13 +0000 | [diff] [blame] | 158 | continue; | 
|  | 159 | } | 
|  | 160 | else | 
|  | 161 | continue; | 
|  | 162 |  | 
|  | 163 | // Check if the SourceLocation is in a system header | 
|  | 164 | const SourceManager &SM = B.getSourceManager(); | 
|  | 165 | if (SM.isInSystemHeader(SL) || SM.isInExternCSystemHeader(SL)) | 
|  | 166 | continue; | 
|  | 167 |  | 
| Alexander Kornienko | 4aca9b1 | 2014-02-11 21:49:21 +0000 | [diff] [blame] | 168 | B.EmitBasicReport(D, this, "Unreachable code", "Dead code", | 
| Ted Kremenek | 5a10f08 | 2012-04-04 18:11:35 +0000 | [diff] [blame] | 169 | "This statement is never executed", DL, SR); | 
| Tom Care | cba9f51 | 2010-07-23 23:04:53 +0000 | [diff] [blame] | 170 | } | 
|  | 171 | } | 
|  | 172 |  | 
|  | 173 | // Recursively finds the entry point(s) for this dead CFGBlock. | 
| Argyrios Kyrtzidis | bf61d97 | 2011-02-23 07:19:23 +0000 | [diff] [blame] | 174 | void UnreachableCodeChecker::FindUnreachableEntryPoints(const CFGBlock *CB, | 
|  | 175 | CFGBlocksSet &reachable, | 
|  | 176 | CFGBlocksSet &visited) { | 
| Tom Care | be633d9 | 2010-08-03 21:24:13 +0000 | [diff] [blame] | 177 | visited.insert(CB->getBlockID()); | 
| Tom Care | cba9f51 | 2010-07-23 23:04:53 +0000 | [diff] [blame] | 178 |  | 
| Tom Care | ea53e82 | 2010-10-06 23:02:25 +0000 | [diff] [blame] | 179 | for (CFGBlock::const_pred_iterator I = CB->pred_begin(), E = CB->pred_end(); | 
|  | 180 | I != E; ++I) { | 
| Ted Kremenek | 4b408e7 | 2014-02-27 21:56:41 +0000 | [diff] [blame] | 181 | if (!*I) | 
|  | 182 | continue; | 
|  | 183 |  | 
| Tom Care | 16ba7c6 | 2010-08-05 17:53:44 +0000 | [diff] [blame] | 184 | if (!reachable.count((*I)->getBlockID())) { | 
| Tom Care | ea53e82 | 2010-10-06 23:02:25 +0000 | [diff] [blame] | 185 | // If we find an unreachable predecessor, mark this block as reachable so | 
|  | 186 | // we don't report this block | 
|  | 187 | reachable.insert(CB->getBlockID()); | 
| Tom Care | 16ba7c6 | 2010-08-05 17:53:44 +0000 | [diff] [blame] | 188 | if (!visited.count((*I)->getBlockID())) | 
| Tom Care | ea53e82 | 2010-10-06 23:02:25 +0000 | [diff] [blame] | 189 | // If we haven't previously visited the unreachable predecessor, recurse | 
| Argyrios Kyrtzidis | bf61d97 | 2011-02-23 07:19:23 +0000 | [diff] [blame] | 190 | FindUnreachableEntryPoints(*I, reachable, visited); | 
| Tom Care | cba9f51 | 2010-07-23 23:04:53 +0000 | [diff] [blame] | 191 | } | 
|  | 192 | } | 
| Tom Care | cba9f51 | 2010-07-23 23:04:53 +0000 | [diff] [blame] | 193 | } | 
|  | 194 |  | 
| Tom Care | be633d9 | 2010-08-03 21:24:13 +0000 | [diff] [blame] | 195 | // Find the Stmt* in a CFGBlock for reporting a warning | 
|  | 196 | const Stmt *UnreachableCodeChecker::getUnreachableStmt(const CFGBlock *CB) { | 
| Zhongxing Xu | 2cd7a78 | 2010-09-16 01:25:47 +0000 | [diff] [blame] | 197 | for (CFGBlock::const_iterator I = CB->begin(), E = CB->end(); I != E; ++I) { | 
| David Blaikie | 00be69a | 2013-02-23 00:29:34 +0000 | [diff] [blame] | 198 | if (Optional<CFGStmt> S = I->getAs<CFGStmt>()) | 
|  | 199 | return S->getStmt(); | 
| Zhongxing Xu | 2cd7a78 | 2010-09-16 01:25:47 +0000 | [diff] [blame] | 200 | } | 
|  | 201 | if (const Stmt *S = CB->getTerminator()) | 
| Tom Care | be633d9 | 2010-08-03 21:24:13 +0000 | [diff] [blame] | 202 | return S; | 
| Tom Care | cba9f51 | 2010-07-23 23:04:53 +0000 | [diff] [blame] | 203 | else | 
| Craig Topper | 0dbb783 | 2014-05-27 02:45:47 +0000 | [diff] [blame] | 204 | return nullptr; | 
| Tom Care | cba9f51 | 2010-07-23 23:04:53 +0000 | [diff] [blame] | 205 | } | 
| Tom Care | 29a6250 | 2010-07-27 23:30:21 +0000 | [diff] [blame] | 206 |  | 
| Tom Care | be633d9 | 2010-08-03 21:24:13 +0000 | [diff] [blame] | 207 | // Determines if the path to this CFGBlock contained an element that infers this | 
|  | 208 | // block is a false positive. We assume that FindUnreachableEntryPoints has | 
|  | 209 | // already marked only the entry points to any dead code, so we need only to | 
|  | 210 | // find the condition that led to this block (the predecessor of this block.) | 
|  | 211 | // There will never be more than one predecessor. | 
| Tom Care | 29a6250 | 2010-07-27 23:30:21 +0000 | [diff] [blame] | 212 | bool UnreachableCodeChecker::isInvalidPath(const CFGBlock *CB, | 
|  | 213 | const ParentMap &PM) { | 
| Tom Care | f7b7067 | 2010-08-27 22:37:31 +0000 | [diff] [blame] | 214 | // We only expect a predecessor size of 0 or 1. If it is >1, then an external | 
|  | 215 | // condition has broken our assumption (for example, a sink being placed by | 
|  | 216 | // another check). In these cases, we choose not to report. | 
|  | 217 | if (CB->pred_size() > 1) | 
|  | 218 | return true; | 
| Tom Care | 29a6250 | 2010-07-27 23:30:21 +0000 | [diff] [blame] | 219 |  | 
| Tom Care | be633d9 | 2010-08-03 21:24:13 +0000 | [diff] [blame] | 220 | // If there are no predecessors, then this block is trivially unreachable | 
|  | 221 | if (CB->pred_size() == 0) | 
|  | 222 | return false; | 
|  | 223 |  | 
|  | 224 | const CFGBlock *pred = *CB->pred_begin(); | 
| Ted Kremenek | 4b408e7 | 2014-02-27 21:56:41 +0000 | [diff] [blame] | 225 | if (!pred) | 
|  | 226 | return false; | 
| Tom Care | be633d9 | 2010-08-03 21:24:13 +0000 | [diff] [blame] | 227 |  | 
|  | 228 | // Get the predecessor block's terminator conditon | 
|  | 229 | const Stmt *cond = pred->getTerminatorCondition(); | 
| Tom Care | af9bbad | 2010-08-12 23:01:06 +0000 | [diff] [blame] | 230 |  | 
|  | 231 | //assert(cond && "CFGBlock's predecessor has a terminator condition"); | 
|  | 232 | // The previous assertion is invalid in some cases (eg do/while). Leaving | 
|  | 233 | // reporting of these situations on at the moment to help triage these cases. | 
|  | 234 | if (!cond) | 
|  | 235 | return false; | 
| Tom Care | be633d9 | 2010-08-03 21:24:13 +0000 | [diff] [blame] | 236 |  | 
|  | 237 | // Run each of the checks on the conditions | 
|  | 238 | if (containsMacro(cond) || containsEnum(cond) | 
|  | 239 | || containsStaticLocal(cond) || containsBuiltinOffsetOf(cond) | 
| Peter Collingbourne | e190dee | 2011-03-11 19:24:49 +0000 | [diff] [blame] | 240 | || containsStmt<UnaryExprOrTypeTraitExpr>(cond)) | 
| Tom Care | be633d9 | 2010-08-03 21:24:13 +0000 | [diff] [blame] | 241 | return true; | 
| Tom Care | 29a6250 | 2010-07-27 23:30:21 +0000 | [diff] [blame] | 242 |  | 
|  | 243 | return false; | 
|  | 244 | } | 
| Tom Care | af9bbad | 2010-08-12 23:01:06 +0000 | [diff] [blame] | 245 |  | 
|  | 246 | // Returns true if the given CFGBlock is empty | 
|  | 247 | bool UnreachableCodeChecker::isEmptyCFGBlock(const CFGBlock *CB) { | 
| Craig Topper | 0dbb783 | 2014-05-27 02:45:47 +0000 | [diff] [blame] | 248 | return CB->getLabel() == nullptr // No labels | 
| Tom Care | af9bbad | 2010-08-12 23:01:06 +0000 | [diff] [blame] | 249 | && CB->size() == 0           // No statements | 
| Aaron Ballman | 2ef6a43 | 2013-12-29 18:59:54 +0000 | [diff] [blame] | 250 | && !CB->getTerminator();     // No terminator | 
| Tom Care | af9bbad | 2010-08-12 23:01:06 +0000 | [diff] [blame] | 251 | } | 
| Argyrios Kyrtzidis | bf61d97 | 2011-02-23 07:19:23 +0000 | [diff] [blame] | 252 |  | 
|  | 253 | void ento::registerUnreachableCodeChecker(CheckerManager &mgr) { | 
|  | 254 | mgr.registerChecker<UnreachableCodeChecker>(); | 
|  | 255 | } |