blob: 46cbaa1f7030132b49ae6b95acb6434de5681511 [file] [log] [blame] [edit]
//===----------------------------------------------------------------------===//
//
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
// See https://llvm.org/LICENSE.txt for license information.
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
//
//===----------------------------------------------------------------------===//
#include "RedundantControlFlowCheck.h"
#include "clang/ASTMatchers/ASTMatchFinder.h"
#include "clang/ASTMatchers/ASTMatchersMacros.h"
#include "clang/Lex/Lexer.h"
using namespace clang::ast_matchers;
namespace clang::tidy::readability {
namespace {
AST_MATCHER_P(CompoundStmt, hasFinalStmt, StatementMatcher, InnerMatcher) {
return !Node.body_empty() &&
InnerMatcher.matches(*Node.body_back(), Finder, Builder);
}
} // namespace
static constexpr StringRef RedundantReturnDiag =
"redundant return statement at the end "
"of a function with a void return type";
static constexpr StringRef RedundantContinueDiag =
"redundant continue statement at the "
"end of loop statement";
void RedundantControlFlowCheck::registerMatchers(MatchFinder *Finder) {
Finder->addMatcher(
functionDecl(returns(voidType()),
hasBody(compoundStmt(hasFinalStmt(
returnStmt(unless(has(expr()))).bind("stmt"))))),
this);
Finder->addMatcher(mapAnyOf(forStmt, cxxForRangeStmt, whileStmt, doStmt)
.with(hasBody(compoundStmt(
hasFinalStmt(continueStmt().bind("stmt"))))),
this);
}
void RedundantControlFlowCheck::check(const MatchFinder::MatchResult &Result) {
const auto &RedundantStmt = *Result.Nodes.getNodeAs<Stmt>("stmt");
const SourceRange StmtRange = RedundantStmt.getSourceRange();
if (StmtRange.getBegin().isMacroID())
return;
const auto RemovedRange = CharSourceRange::getCharRange(
StmtRange.getBegin(),
Lexer::findLocationAfterToken(StmtRange.getEnd(), tok::semi,
*Result.SourceManager, getLangOpts(),
/*SkipTrailingWhitespaceAndNewLine=*/true));
diag(StmtRange.getBegin(), isa<ReturnStmt>(RedundantStmt)
? RedundantReturnDiag
: RedundantContinueDiag)
<< FixItHint::CreateRemoval(RemovedRange);
}
} // namespace clang::tidy::readability