aboutsummaryrefslogtreecommitdiff
path: root/clang
diff options
context:
space:
mode:
authorTed Kremenek <kremenek@apple.com>2014-03-20 06:07:35 +0000
committerTed Kremenek <kremenek@apple.com>2014-03-20 06:07:35 +0000
commit2766ad27e867c40378f2e5c18094594aa112b7b4 (patch)
treebda98c5b5110b2c195cea12cd10945618777303b /clang
parentf3c93bb61b486d3b01de8d0b071688a35388474d (diff)
downloadllvm-2766ad27e867c40378f2e5c18094594aa112b7b4.zip
llvm-2766ad27e867c40378f2e5c18094594aa112b7b4.tar.gz
llvm-2766ad27e867c40378f2e5c18094594aa112b7b4.tar.bz2
[-Wunreachable-code] constexpr functions can be used as configuration values.
llvm-svn: 204308
Diffstat (limited to 'clang')
-rw-r--r--clang/lib/Analysis/ReachableCode.cpp5
-rw-r--r--clang/test/SemaCXX/warn-unreachable.cpp23
2 files changed, 27 insertions, 1 deletions
diff --git a/clang/lib/Analysis/ReachableCode.cpp b/clang/lib/Analysis/ReachableCode.cpp
index c79a94b82..ffe576c 100644
--- a/clang/lib/Analysis/ReachableCode.cpp
+++ b/clang/lib/Analysis/ReachableCode.cpp
@@ -139,6 +139,11 @@ static bool isConfigurationValue(const Stmt *S,
S = Ex->IgnoreParenCasts();
switch (S->getStmtClass()) {
+ case Stmt::CallExprClass: {
+ const FunctionDecl *Callee =
+ dyn_cast_or_null<FunctionDecl>(cast<CallExpr>(S)->getCalleeDecl());
+ return Callee ? Callee->isConstexpr() : false;
+ }
case Stmt::DeclRefExprClass: {
const DeclRefExpr *DR = cast<DeclRefExpr>(S);
const ValueDecl *D = DR->getDecl();
diff --git a/clang/test/SemaCXX/warn-unreachable.cpp b/clang/test/SemaCXX/warn-unreachable.cpp
index abab966..9db7592 100644
--- a/clang/test/SemaCXX/warn-unreachable.cpp
+++ b/clang/test/SemaCXX/warn-unreachable.cpp
@@ -1,4 +1,4 @@
-// RUN: %clang_cc1 %s -fcxx-exceptions -fexceptions -fsyntax-only -verify -fblocks -Wunreachable-code-aggressive -Wno-unused-value
+// RUN: %clang_cc1 %s -fcxx-exceptions -fexceptions -fsyntax-only -verify -fblocks -std=c++11 -Wunreachable-code-aggressive -Wno-unused-value
int &halt() __attribute__((noreturn));
int &live();
@@ -234,3 +234,24 @@ Frobozz test_return_object_control_flow(int flag) {
return Frobozz(flag ? 42 : 24); // expected-warning {{code will never be executed}}
}
+void somethingToCall();
+
+ static constexpr bool isConstExprConfigValue() { return true; }
+
+ int test_const_expr_config_value() {
+ if (isConstExprConfigValue()) {
+ somethingToCall();
+ return 0;
+ }
+ somethingToCall(); // no-warning
+ return 1;
+ }
+ int test_const_expr_config_value_2() {
+ if (!isConstExprConfigValue()) {
+ somethingToCall(); // no-warning
+ return 0;
+ }
+ somethingToCall();
+ return 1;
+ }
+