Index: clang/lib/Sema/AnalysisBasedWarnings.cpp =================================================================== --- clang/lib/Sema/AnalysisBasedWarnings.cpp +++ clang/lib/Sema/AnalysisBasedWarnings.cpp @@ -1148,6 +1148,11 @@ continue; } + if (isFollowedByFallThroughComment(LastStmt)) { + ++AnnotatedCnt; + continue; // Fallthrough comment, good. + } + ++UnannotatedCnt; } return !!UnannotatedCnt; @@ -1208,10 +1213,34 @@ return nullptr; } + bool isFollowedByFallThroughComment(const Stmt *stmt) { + // Try to detect whether the fallthough is marked by a comment like + // /*FALLTHOUGH*/ + bool Invalid; + const char *sourceData = + S.getSourceManager().getCharacterData(stmt->getEndLoc(), &Invalid); + if (Invalid) + return false; + const char *lineStart = strchr(sourceData, '\n'); + if (lineStart == nullptr) + return false; + ++lineStart; // start of next line + const char *lineEnd = strchr(lineStart, '\n'); + StringRef line(lineStart, + lineEnd ? lineEnd - lineStart : strlen(lineStart)); + if (!fallthroughRegex) + fallthroughRegex = std::make_unique( + "(/\\*[ \\t]*fall(s | |-)?thr(ough|u)[ \\t]*\\*/)" + "|(//[ \\t]*fall(s | |-)?thr(ough|u)[ \\t]*)", + llvm::Regex::IgnoreCase); + return fallthroughRegex->match(line); + } + bool FoundSwitchStatements; AttrStmts FallthroughStmts; Sema &S; llvm::SmallPtrSet ReachableBlocks; + std::unique_ptr fallthroughRegex; }; } // anonymous namespace Index: clang/test/Sema/fallthrough-comment.c =================================================================== --- /dev/null +++ clang/test/Sema/fallthrough-comment.c @@ -0,0 +1,18 @@ +// RUN: %clang_cc1 -fsyntax-only -std=c11 -verify -Wimplicit-fallthrough %s + +int fallthrough_comment(int n) { + switch (n) { + case 0: + n++; + // FALLTHROUGH + case 1: + n++; + /*fall-through*/ + case 2: + n++; + case 3: // expected-warning{{unannotated fall-through between switch labels}} expected-note{{insert '__attribute__((fallthrough));' to silence this warning}} expected-note{{insert 'break;' to avoid fall-through}} + n++; + break; + } + return n; +}