clang-tools 22.0.0git
AvoidUnderscoreInGoogletestNameCheck.cpp
Go to the documentation of this file.
1//===----------------------------------------------------------------------===//
2//
3// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4// See https://llvm.org/LICENSE.txt for license information.
5// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6//
7//===----------------------------------------------------------------------===//
8
9#include <string>
10
12#include "clang/AST/ASTContext.h"
13#include "clang/Frontend/CompilerInstance.h"
14#include "clang/Lex/MacroArgs.h"
15#include "clang/Lex/PPCallbacks.h"
16#include "clang/Lex/Preprocessor.h"
17
19
20constexpr llvm::StringLiteral KDisabledTestPrefix = "DISABLED_";
21
22// Determines whether the macro is a Googletest test macro.
23static bool isGoogletestTestMacro(StringRef MacroName) {
24 static const llvm::StringSet<> MacroNames = {"TEST", "TEST_F", "TEST_P",
25 "TYPED_TEST", "TYPED_TEST_P"};
26 return MacroNames.contains(MacroName);
27}
28
29namespace {
30
31class AvoidUnderscoreInGoogletestNameCallback : public PPCallbacks {
32public:
33 AvoidUnderscoreInGoogletestNameCallback(
34 Preprocessor *PP, AvoidUnderscoreInGoogletestNameCheck *Check)
35 : PP(PP), Check(Check) {}
36
37 // Detects expansions of the TEST, TEST_F, TEST_P, TYPED_TEST, TYPED_TEST_P
38 // macros and checks that their arguments do not have any underscores.
39 void MacroExpands(const Token &MacroNameToken,
40 const MacroDefinition &MacroDefinition, SourceRange Range,
41 const MacroArgs *Args) override {
42 IdentifierInfo *NameIdentifierInfo = MacroNameToken.getIdentifierInfo();
43 if (!NameIdentifierInfo)
44 return;
45 StringRef MacroName = NameIdentifierInfo->getName();
46 if (!isGoogletestTestMacro(MacroName) || !Args ||
47 Args->getNumMacroArguments() < 2)
48 return;
49 const Token *TestSuiteNameToken = Args->getUnexpArgument(0);
50 const Token *TestNameToken = Args->getUnexpArgument(1);
51 if (!TestSuiteNameToken || !TestNameToken)
52 return;
53 std::string TestSuiteNameMaybeDisabled =
54 PP->getSpelling(*TestSuiteNameToken);
55 StringRef TestSuiteName = TestSuiteNameMaybeDisabled;
56 TestSuiteName.consume_front(KDisabledTestPrefix);
57 if (TestSuiteName.contains('_'))
58 Check->diag(TestSuiteNameToken->getLocation(),
59 "avoid using \"_\" in test suite name \"%0\" according to "
60 "Googletest FAQ")
61 << TestSuiteName;
62
63 std::string TestNameMaybeDisabled = PP->getSpelling(*TestNameToken);
64 StringRef TestName = TestNameMaybeDisabled;
65 TestName.consume_front(KDisabledTestPrefix);
66 if (TestName.contains('_'))
67 Check->diag(TestNameToken->getLocation(),
68 "avoid using \"_\" in test name \"%0\" according to "
69 "Googletest FAQ")
70 << TestName;
71 }
72
73private:
74 Preprocessor *PP;
75 AvoidUnderscoreInGoogletestNameCheck *Check;
76};
77
78} // namespace
79
81 const SourceManager &SM, Preprocessor *PP, Preprocessor *ModuleExpanderPP) {
82 PP->addPPCallbacks(
83 std::make_unique<AvoidUnderscoreInGoogletestNameCallback>(PP, this));
84}
85
86} // namespace clang::tidy::google::readability
void registerPPCallbacks(const SourceManager &SM, Preprocessor *PP, Preprocessor *ModuleExpanderPP) override