Baranov Victor 975d56272e
[clang-tidy] Add new check readability-trailing-comma (#173669)
clang-format has a couple of similar options:

https://clang.llvm.org/docs/ClangFormatStyleOptions.html#enumtrailingcomma
- add trailing commas for enum

https://clang.llvm.org/docs/ClangFormatStyleOptions.html#inserttrailingcommas
- add trailing commas for C++
but generally they are marked with such warning:

> Warning
>
> Setting this option to any value other than Leave could lead to
incorrect code formatting due to clang-format’s lack of complete
semantic information. As such, extra care should be taken to review code
changes made by this option.

clang-tidy on the other hand has all semantic information, thus can
(hopefully) provide 0 false-positives.

Note that we have already overlapping checks in clang-format/clang-tidy
like:
https://clang.llvm.org/docs/ClangFormatStyleOptions.html#insertbraces vs
https://clang.llvm.org/extra/clang-tidy/checks/readability/braces-around-statements.html
2026-01-31 19:43:45 +00:00

180 lines
6.5 KiB
C++

//===----------------------------------------------------------------------===//
//
// 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 "TrailingCommaCheck.h"
#include "../utils/LexerUtils.h"
#include "clang/AST/ASTContext.h"
#include "clang/ASTMatchers/ASTMatchFinder.h"
#include "clang/Lex/Lexer.h"
using namespace clang::ast_matchers;
namespace clang::tidy {
template <>
struct OptionEnumMapping<readability::TrailingCommaCheck::CommaPolicyKind> {
static llvm::ArrayRef<
std::pair<readability::TrailingCommaCheck::CommaPolicyKind, StringRef>>
getEnumMapping() {
static constexpr std::pair<readability::TrailingCommaCheck::CommaPolicyKind,
StringRef>
Mapping[] = {
{readability::TrailingCommaCheck::CommaPolicyKind::Append,
"Append"},
{readability::TrailingCommaCheck::CommaPolicyKind::Remove,
"Remove"},
{readability::TrailingCommaCheck::CommaPolicyKind::Ignore,
"Ignore"},
};
return {Mapping};
}
};
} // namespace clang::tidy
namespace clang::tidy::readability {
static bool isSingleLine(SourceRange Range, const SourceManager &SM) {
return SM.getExpansionLineNumber(Range.getBegin()) ==
SM.getExpansionLineNumber(Range.getEnd());
}
namespace {
AST_POLYMORPHIC_MATCHER(isMacro,
AST_POLYMORPHIC_SUPPORTED_TYPES(EnumDecl,
InitListExpr)) {
return Node.getBeginLoc().isMacroID() || Node.getEndLoc().isMacroID();
}
AST_MATCHER(EnumDecl, isEmptyEnum) { return Node.enumerators().empty(); }
AST_MATCHER(InitListExpr, isEmptyInitList) { return Node.getNumInits() == 0; }
} // namespace
TrailingCommaCheck::TrailingCommaCheck(StringRef Name,
ClangTidyContext *Context)
: ClangTidyCheck(Name, Context),
SingleLineCommaPolicy(
Options.get("SingleLineCommaPolicy", CommaPolicyKind::Remove)),
MultiLineCommaPolicy(
Options.get("MultiLineCommaPolicy", CommaPolicyKind::Append)) {
if (SingleLineCommaPolicy == CommaPolicyKind::Ignore &&
MultiLineCommaPolicy == CommaPolicyKind::Ignore)
configurationDiag("The check '%0' will not perform any analysis because "
"'SingleLineCommaPolicy' and 'MultiLineCommaPolicy' are "
"both set to 'Ignore'.")
<< Name;
}
void TrailingCommaCheck::storeOptions(ClangTidyOptions::OptionMap &Opts) {
Options.store(Opts, "SingleLineCommaPolicy", SingleLineCommaPolicy);
Options.store(Opts, "MultiLineCommaPolicy", MultiLineCommaPolicy);
}
void TrailingCommaCheck::registerMatchers(MatchFinder *Finder) {
Finder->addMatcher(
enumDecl(isDefinition(), unless(isEmptyEnum()), unless(isMacro()))
.bind("enum"),
this);
Finder->addMatcher(initListExpr(unless(isEmptyInitList()), unless(isMacro()))
.bind("initlist"),
this);
}
void TrailingCommaCheck::check(const MatchFinder::MatchResult &Result) {
if (const auto *Enum = Result.Nodes.getNodeAs<EnumDecl>("enum"))
checkEnumDecl(Enum, Result);
else if (const auto *InitList =
Result.Nodes.getNodeAs<InitListExpr>("initlist"))
checkInitListExpr(InitList, Result);
else
llvm_unreachable("No matches found");
}
void TrailingCommaCheck::checkEnumDecl(const EnumDecl *Enum,
const MatchFinder::MatchResult &Result) {
const bool IsSingleLine = isSingleLine(
{Enum->getBeginLoc(), Enum->getEndLoc()}, *Result.SourceManager);
const CommaPolicyKind Policy =
IsSingleLine ? SingleLineCommaPolicy : MultiLineCommaPolicy;
if (Policy == CommaPolicyKind::Ignore)
return;
const std::optional<Token> LastTok =
Lexer::findPreviousToken(Enum->getBraceRange().getEnd(),
*Result.SourceManager, getLangOpts(), false);
if (!LastTok)
return;
emitDiag(LastTok->getLocation(), LastTok, DiagKind::Enum, Result, Policy);
}
void TrailingCommaCheck::checkInitListExpr(
const InitListExpr *InitList, const MatchFinder::MatchResult &Result) {
// We need to use non-empty syntactic form for correct source locations.
if (const InitListExpr *SynInitInitList = InitList->getSyntacticForm();
SynInitInitList && SynInitInitList->getNumInits() > 0)
InitList = SynInitInitList;
const bool IsSingleLine = isSingleLine(
{InitList->getBeginLoc(), InitList->getEndLoc()}, *Result.SourceManager);
const CommaPolicyKind Policy =
IsSingleLine ? SingleLineCommaPolicy : MultiLineCommaPolicy;
if (Policy == CommaPolicyKind::Ignore)
return;
const Expr *LastInit = InitList->inits().back();
assert(LastInit);
// Skip pack expansions - they already have special syntax with '...'
if (isa<PackExpansionExpr>(LastInit))
return;
const std::optional<Token> NextTok =
utils::lexer::findNextTokenSkippingComments(
LastInit->getEndLoc(), *Result.SourceManager, getLangOpts());
// If the next token is neither a comma nor closing brace, there might be
// a macro (e.g., #define COMMA ,) that we can't safely analyze.
if (NextTok && !NextTok->isOneOf(tok::comma, tok::r_brace))
return;
emitDiag(LastInit->getEndLoc(), NextTok, DiagKind::InitList, Result, Policy);
}
void TrailingCommaCheck::emitDiag(
SourceLocation LastLoc, std::optional<Token> Token, DiagKind Kind,
const ast_matchers::MatchFinder::MatchResult &Result,
CommaPolicyKind Policy) {
if (LastLoc.isInvalid() || !Token)
return;
const bool HasTrailingComma = Token->is(tok::comma);
if (Policy == CommaPolicyKind::Append && !HasTrailingComma) {
const SourceLocation InsertLoc = Lexer::getLocForEndOfToken(
LastLoc, 0, *Result.SourceManager, getLangOpts());
diag(InsertLoc, "%select{initializer list|enum}0 should have "
"a trailing comma")
<< Kind << FixItHint::CreateInsertion(InsertLoc, ",");
} else if (Policy == CommaPolicyKind::Remove && HasTrailingComma) {
const SourceLocation CommaLoc = Token->getLocation();
if (CommaLoc.isInvalid())
return;
diag(CommaLoc, "%select{initializer list|enum}0 should not have "
"a trailing comma")
<< Kind << FixItHint::CreateRemoval(CommaLoc);
}
}
} // namespace clang::tidy::readability