clang-tools 22.0.0git
ClangTidyPlugin.cpp
Go to the documentation of this file.
1//===- ClangTidyPlugin.cpp - clang-tidy as a clang plugin -----------------===//
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 "../ClangTidy.h"
11#include "../ClangTidyModule.h"
12#include "clang/Frontend/CompilerInstance.h"
13#include "clang/Frontend/FrontendPluginRegistry.h"
14#include "clang/Frontend/MultiplexConsumer.h"
15
16namespace clang::tidy {
17
18/// The core clang tidy plugin action. This just provides the AST consumer and
19/// command line flag parsing for using clang-tidy as a clang plugin.
21 /// Wrapper to grant the context and diagnostics engine the same lifetime as
22 /// the action.
23 /// We use MultiplexConsumer to avoid writing out all the forwarding methods.
24 class WrapConsumer : public MultiplexConsumer {
25 std::unique_ptr<ClangTidyContext> Context;
26 std::unique_ptr<DiagnosticsEngine> DiagEngine;
27
28 public:
29 WrapConsumer(std::unique_ptr<ClangTidyContext> Context,
30 std::unique_ptr<DiagnosticsEngine> DiagEngine,
31 std::vector<std::unique_ptr<ASTConsumer>> Consumer)
32 : MultiplexConsumer(std::move(Consumer)), Context(std::move(Context)),
33 DiagEngine(std::move(DiagEngine)) {}
34 };
35
36public:
37 std::unique_ptr<ASTConsumer> CreateASTConsumer(CompilerInstance &Compiler,
38 StringRef File) override {
39 // Create and set diagnostics engine
40 auto *DiagConsumer =
41 new ClangTidyDiagnosticConsumer(*Context, &Compiler.getDiagnostics());
42 auto DiagOpts = std::make_unique<DiagnosticOptions>();
43 auto DiagEngine = std::make_unique<DiagnosticsEngine>(
44 DiagnosticIDs::create(), *DiagOpts, DiagConsumer);
45 Context->setDiagnosticsEngine(std::move(DiagOpts), DiagEngine.get());
46
47 // Create the AST consumer.
48 ClangTidyASTConsumerFactory Factory(*Context);
49 std::vector<std::unique_ptr<ASTConsumer>> Vec;
50 Vec.push_back(Factory.createASTConsumer(Compiler, File));
51
52 return std::make_unique<WrapConsumer>(
53 std::move(Context), std::move(DiagEngine), std::move(Vec));
54 }
55
56 bool ParseArgs(const CompilerInstance &,
57 const std::vector<std::string> &Args) override {
58 ClangTidyGlobalOptions GlobalOptions;
59 ClangTidyOptions DefaultOptions;
60 ClangTidyOptions OverrideOptions;
61
62 // Parse the extra command line args.
63 // FIXME: This is very limited at the moment.
64 for (StringRef Arg : Args)
65 if (Arg.starts_with("-checks="))
66 OverrideOptions.Checks = std::string(Arg.substr(strlen("-checks=")));
67
68 auto Options = std::make_unique<FileOptionsProvider>(
69 GlobalOptions, DefaultOptions, OverrideOptions);
70 Context = std::make_unique<ClangTidyContext>(std::move(Options));
71 return true;
72 }
73
74private:
75 std::unique_ptr<ClangTidyContext> Context;
76};
77} // namespace clang::tidy
78
79// This anchor is used to force the linker to link in the generated object file
80// and thus register the clang-tidy plugin.
81// NOLINTNEXTLINE(misc-use-internal-linkage)
83
84static clang::FrontendPluginRegistry::Add<clang::tidy::ClangTidyPluginAction>
85 X("clang-tidy", "clang-tidy");
volatile int ClangTidyPluginAnchorSource
static clang::FrontendPluginRegistry::Add< clang::tidy::ClangTidyPluginAction > X("clang-tidy", "clang-tidy")
std::unique_ptr< clang::ASTConsumer > createASTConsumer(clang::CompilerInstance &Compiler, StringRef File)
Returns an ASTConsumer that runs the specified clang-tidy checks.
A diagnostic consumer that turns each Diagnostic into a SourceManager-independent ClangTidyError.
The core clang tidy plugin action.
std::unique_ptr< ASTConsumer > CreateASTConsumer(CompilerInstance &Compiler, StringRef File) override
bool ParseArgs(const CompilerInstance &, const std::vector< std::string > &Args) override
Contains options for clang-tidy.
std::optional< std::string > Checks
Checks filter.