diff options
-rw-r--r-- | lib/Analysis/ThreadSafety.cpp | 14 | ||||
-rw-r--r-- | test/SemaCXX/warn-thread-safety-analysis.cpp | 34 |
2 files changed, 48 insertions, 0 deletions
diff --git a/lib/Analysis/ThreadSafety.cpp b/lib/Analysis/ThreadSafety.cpp index e7d9a2d642..036d0b8888 100644 --- a/lib/Analysis/ThreadSafety.cpp +++ b/lib/Analysis/ThreadSafety.cpp @@ -2374,6 +2374,20 @@ void ThreadSafetyAnalyzer::runAnalysis(AnalysisDeclContext &AC) { } } + + // Check to make sure that the exit block is reachable + bool ExitUnreachable = true; + for (CFGBlock::const_pred_iterator PI = CFGraph->getExit().pred_begin(), + PE = CFGraph->getExit().pred_end(); PI != PE; ++PI) { + if (!(*PI)->hasNoReturnElement()) { + ExitUnreachable = false; + break; + } + } + // Skip the final check if the exit block is unreachable. + if (ExitUnreachable) + return; + CFGBlockInfo *Initial = &BlockInfo[CFGraph->getEntry().getBlockID()]; CFGBlockInfo *Final = &BlockInfo[CFGraph->getExit().getBlockID()]; diff --git a/test/SemaCXX/warn-thread-safety-analysis.cpp b/test/SemaCXX/warn-thread-safety-analysis.cpp index 4e8893d3ca..0d6aa4c79b 100644 --- a/test/SemaCXX/warn-thread-safety-analysis.cpp +++ b/test/SemaCXX/warn-thread-safety-analysis.cpp @@ -3419,3 +3419,37 @@ void test2() { } }; // end namespace ComplexNameTest + + +namespace UnreachableExitTest { + +class FemmeFatale { +public: + FemmeFatale(); + ~FemmeFatale() __attribute__((noreturn)); +}; + +void exitNow() __attribute__((noreturn)); + +Mutex fatalmu_; + +void test1() EXCLUSIVE_LOCKS_REQUIRED(fatalmu_) { + exitNow(); +} + +void test2() EXCLUSIVE_LOCKS_REQUIRED(fatalmu_) { + FemmeFatale femme; +} + +bool c; + +void test3() EXCLUSIVE_LOCKS_REQUIRED(fatalmu_) { + if (c) { + exitNow(); + } + else { + FemmeFatale femme; + } +} + +} // end namespace UnreachableExitTest |