diff options
Diffstat (limited to 'clang')
-rw-r--r-- | clang/lib/Sema/AnalysisBasedWarnings.cpp | 3 | ||||
-rw-r--r-- | clang/test/SemaCXX/switch-implicit-fallthrough.cpp | 60 |
2 files changed, 44 insertions, 19 deletions
diff --git a/clang/lib/Sema/AnalysisBasedWarnings.cpp b/clang/lib/Sema/AnalysisBasedWarnings.cpp index 0e522dc4157..c4524cde93a 100644 --- a/clang/lib/Sema/AnalysisBasedWarnings.cpp +++ b/clang/lib/Sema/AnalysisBasedWarnings.cpp @@ -1025,6 +1025,9 @@ namespace { // methods separately. bool TraverseDecl(Decl *D) { return true; } + // We analyze lambda bodies separately. Skip them here. + bool TraverseLambdaBody(LambdaExpr *LE) { return true; } + private: static const AttributedStmt *asFallThroughAttr(const Stmt *S) { diff --git a/clang/test/SemaCXX/switch-implicit-fallthrough.cpp b/clang/test/SemaCXX/switch-implicit-fallthrough.cpp index 831324a64e9..0bc43cdbd45 100644 --- a/clang/test/SemaCXX/switch-implicit-fallthrough.cpp +++ b/clang/test/SemaCXX/switch-implicit-fallthrough.cpp @@ -229,25 +229,6 @@ int fallthrough_covered_enums(Enum e) { return n; } -int fallthrough_targets(int n) { - [[clang::fallthrough]]; // expected-error{{fallthrough annotation is outside switch statement}} - - [[clang::fallthrough]] // expected-error{{fallthrough attribute is only allowed on empty statements}} - switch (n) { - case 121: - n += 400; - [[clang::fallthrough]]; // no warning here, correct target - case 123: - [[clang::fallthrough]] // expected-error{{fallthrough attribute is only allowed on empty statements}} - n += 800; - break; - [[clang::fallthrough]] // expected-error{{fallthrough attribute is only allowed on empty statements}} expected-note{{did you forget ';'?}} - case 125: - n += 1600; - } - return n; -} - // Fallthrough annotations in local classes used to generate "fallthrough // annotation does not directly precede switch label" warning. void fallthrough_in_local_class() { @@ -259,12 +240,34 @@ void fallthrough_in_local_class() { [[clang::fallthrough]]; // no diagnostics case 1: x++; + default: // \ + expected-warning{{unannotated fall-through between switch labels}} \ + expected-note{{insert 'break;' to avoid fall-through}} break; } } }; } +// Fallthrough annotations in lambdas used to generate "fallthrough +// annotation does not directly precede switch label" warning. +void fallthrough_in_lambda() { + (void)[] { + int x = 0; + switch (x) { + case 0: + x++; + [[clang::fallthrough]]; // no diagnostics + case 1: + x++; + default: // \ + expected-warning{{unannotated fall-through between switch labels}} \ + expected-note{{insert 'break;' to avoid fall-through}} + break; + } + }; +} + namespace PR18983 { void fatal() __attribute__((noreturn)); int num(); @@ -278,3 +281,22 @@ namespace PR18983 { } } } + +int fallthrough_targets(int n) { + [[clang::fallthrough]]; // expected-error{{fallthrough annotation is outside switch statement}} + + [[clang::fallthrough]] // expected-error{{fallthrough attribute is only allowed on empty statements}} + switch (n) { + case 121: + n += 400; + [[clang::fallthrough]]; // no warning here, correct target + case 123: + [[clang::fallthrough]] // expected-error{{fallthrough attribute is only allowed on empty statements}} + n += 800; + break; + [[clang::fallthrough]] // expected-error{{fallthrough attribute is only allowed on empty statements}} expected-note{{did you forget ';'?}} + case 125: + n += 1600; + } + return n; +} |