annotate clang-tools-extra/clangd/ParsedAST.cpp @ 221:79ff65ed7e25

LLVM12 Original
author Shinji KONO <kono@ie.u-ryukyu.ac.jp>
date Tue, 15 Jun 2021 19:15:29 +0900
parents 0572611fdcc8
children 5f17cb93ff66
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
150
anatofuz
parents:
diff changeset
1 //===--- ParsedAST.cpp -------------------------------------------*- C++-*-===//
anatofuz
parents:
diff changeset
2 //
anatofuz
parents:
diff changeset
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
anatofuz
parents:
diff changeset
4 // See https://llvm.org/LICENSE.txt for license information.
anatofuz
parents:
diff changeset
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
anatofuz
parents:
diff changeset
6 //
anatofuz
parents:
diff changeset
7 //===----------------------------------------------------------------------===//
anatofuz
parents:
diff changeset
8
anatofuz
parents:
diff changeset
9 #include "ParsedAST.h"
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
10 #include "../clang-tidy/ClangTidyCheck.h"
150
anatofuz
parents:
diff changeset
11 #include "../clang-tidy/ClangTidyDiagnosticConsumer.h"
anatofuz
parents:
diff changeset
12 #include "../clang-tidy/ClangTidyModuleRegistry.h"
anatofuz
parents:
diff changeset
13 #include "AST.h"
anatofuz
parents:
diff changeset
14 #include "Compiler.h"
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
15 #include "Config.h"
150
anatofuz
parents:
diff changeset
16 #include "Diagnostics.h"
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
17 #include "FeatureModule.h"
150
anatofuz
parents:
diff changeset
18 #include "Headers.h"
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
19 #include "HeuristicResolver.h"
150
anatofuz
parents:
diff changeset
20 #include "IncludeFixer.h"
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
21 #include "Preamble.h"
150
anatofuz
parents:
diff changeset
22 #include "SourceCode.h"
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
23 #include "TidyProvider.h"
150
anatofuz
parents:
diff changeset
24 #include "index/CanonicalIncludes.h"
anatofuz
parents:
diff changeset
25 #include "index/Index.h"
173
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
26 #include "support/Logger.h"
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
27 #include "support/Trace.h"
150
anatofuz
parents:
diff changeset
28 #include "clang/AST/ASTContext.h"
anatofuz
parents:
diff changeset
29 #include "clang/AST/Decl.h"
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
30 #include "clang/Basic/Diagnostic.h"
150
anatofuz
parents:
diff changeset
31 #include "clang/Basic/LangOptions.h"
anatofuz
parents:
diff changeset
32 #include "clang/Basic/SourceLocation.h"
anatofuz
parents:
diff changeset
33 #include "clang/Basic/SourceManager.h"
anatofuz
parents:
diff changeset
34 #include "clang/Basic/TokenKinds.h"
anatofuz
parents:
diff changeset
35 #include "clang/Frontend/CompilerInstance.h"
anatofuz
parents:
diff changeset
36 #include "clang/Frontend/CompilerInvocation.h"
anatofuz
parents:
diff changeset
37 #include "clang/Frontend/FrontendActions.h"
anatofuz
parents:
diff changeset
38 #include "clang/Frontend/Utils.h"
anatofuz
parents:
diff changeset
39 #include "clang/Index/IndexDataConsumer.h"
anatofuz
parents:
diff changeset
40 #include "clang/Index/IndexingAction.h"
anatofuz
parents:
diff changeset
41 #include "clang/Lex/Lexer.h"
anatofuz
parents:
diff changeset
42 #include "clang/Lex/MacroInfo.h"
anatofuz
parents:
diff changeset
43 #include "clang/Lex/PPCallbacks.h"
anatofuz
parents:
diff changeset
44 #include "clang/Lex/Preprocessor.h"
anatofuz
parents:
diff changeset
45 #include "clang/Lex/PreprocessorOptions.h"
anatofuz
parents:
diff changeset
46 #include "clang/Sema/Sema.h"
anatofuz
parents:
diff changeset
47 #include "clang/Serialization/ASTWriter.h"
anatofuz
parents:
diff changeset
48 #include "clang/Serialization/PCHContainerOperations.h"
anatofuz
parents:
diff changeset
49 #include "clang/Tooling/CompilationDatabase.h"
anatofuz
parents:
diff changeset
50 #include "clang/Tooling/Syntax/Tokens.h"
anatofuz
parents:
diff changeset
51 #include "llvm/ADT/ArrayRef.h"
anatofuz
parents:
diff changeset
52 #include "llvm/ADT/STLExtras.h"
anatofuz
parents:
diff changeset
53 #include "llvm/ADT/SmallString.h"
anatofuz
parents:
diff changeset
54 #include "llvm/ADT/SmallVector.h"
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
55 #include "llvm/ADT/StringRef.h"
150
anatofuz
parents:
diff changeset
56 #include "llvm/Support/raw_ostream.h"
anatofuz
parents:
diff changeset
57 #include <algorithm>
anatofuz
parents:
diff changeset
58 #include <memory>
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
59 #include <vector>
150
anatofuz
parents:
diff changeset
60
anatofuz
parents:
diff changeset
61 // Force the linker to link in Clang-tidy modules.
anatofuz
parents:
diff changeset
62 // clangd doesn't support the static analyzer.
anatofuz
parents:
diff changeset
63 #define CLANG_TIDY_DISABLE_STATIC_ANALYZER_CHECKS
anatofuz
parents:
diff changeset
64 #include "../clang-tidy/ClangTidyForceLinker.h"
anatofuz
parents:
diff changeset
65
anatofuz
parents:
diff changeset
66 namespace clang {
anatofuz
parents:
diff changeset
67 namespace clangd {
anatofuz
parents:
diff changeset
68 namespace {
anatofuz
parents:
diff changeset
69
anatofuz
parents:
diff changeset
70 template <class T> std::size_t getUsedBytes(const std::vector<T> &Vec) {
anatofuz
parents:
diff changeset
71 return Vec.capacity() * sizeof(T);
anatofuz
parents:
diff changeset
72 }
anatofuz
parents:
diff changeset
73
anatofuz
parents:
diff changeset
74 class DeclTrackingASTConsumer : public ASTConsumer {
anatofuz
parents:
diff changeset
75 public:
anatofuz
parents:
diff changeset
76 DeclTrackingASTConsumer(std::vector<Decl *> &TopLevelDecls)
anatofuz
parents:
diff changeset
77 : TopLevelDecls(TopLevelDecls) {}
anatofuz
parents:
diff changeset
78
anatofuz
parents:
diff changeset
79 bool HandleTopLevelDecl(DeclGroupRef DG) override {
anatofuz
parents:
diff changeset
80 for (Decl *D : DG) {
anatofuz
parents:
diff changeset
81 auto &SM = D->getASTContext().getSourceManager();
anatofuz
parents:
diff changeset
82 if (!isInsideMainFile(D->getLocation(), SM))
anatofuz
parents:
diff changeset
83 continue;
anatofuz
parents:
diff changeset
84 if (const NamedDecl *ND = dyn_cast<NamedDecl>(D))
anatofuz
parents:
diff changeset
85 if (isImplicitTemplateInstantiation(ND))
anatofuz
parents:
diff changeset
86 continue;
anatofuz
parents:
diff changeset
87
anatofuz
parents:
diff changeset
88 // ObjCMethodDecl are not actually top-level decls.
anatofuz
parents:
diff changeset
89 if (isa<ObjCMethodDecl>(D))
anatofuz
parents:
diff changeset
90 continue;
anatofuz
parents:
diff changeset
91
anatofuz
parents:
diff changeset
92 TopLevelDecls.push_back(D);
anatofuz
parents:
diff changeset
93 }
anatofuz
parents:
diff changeset
94 return true;
anatofuz
parents:
diff changeset
95 }
anatofuz
parents:
diff changeset
96
anatofuz
parents:
diff changeset
97 private:
anatofuz
parents:
diff changeset
98 std::vector<Decl *> &TopLevelDecls;
anatofuz
parents:
diff changeset
99 };
anatofuz
parents:
diff changeset
100
anatofuz
parents:
diff changeset
101 class ClangdFrontendAction : public SyntaxOnlyAction {
anatofuz
parents:
diff changeset
102 public:
anatofuz
parents:
diff changeset
103 std::vector<Decl *> takeTopLevelDecls() { return std::move(TopLevelDecls); }
anatofuz
parents:
diff changeset
104
anatofuz
parents:
diff changeset
105 protected:
anatofuz
parents:
diff changeset
106 std::unique_ptr<ASTConsumer>
anatofuz
parents:
diff changeset
107 CreateASTConsumer(CompilerInstance &CI, llvm::StringRef InFile) override {
anatofuz
parents:
diff changeset
108 return std::make_unique<DeclTrackingASTConsumer>(/*ref*/ TopLevelDecls);
anatofuz
parents:
diff changeset
109 }
anatofuz
parents:
diff changeset
110
anatofuz
parents:
diff changeset
111 private:
anatofuz
parents:
diff changeset
112 std::vector<Decl *> TopLevelDecls;
anatofuz
parents:
diff changeset
113 };
anatofuz
parents:
diff changeset
114
anatofuz
parents:
diff changeset
115 // When using a preamble, only preprocessor events outside its bounds are seen.
anatofuz
parents:
diff changeset
116 // This is almost what we want: replaying transitive preprocessing wastes time.
anatofuz
parents:
diff changeset
117 // However this confuses clang-tidy checks: they don't see any #includes!
anatofuz
parents:
diff changeset
118 // So we replay the *non-transitive* #includes that appear in the main-file.
anatofuz
parents:
diff changeset
119 // It would be nice to replay other events (macro definitions, ifdefs etc) but
anatofuz
parents:
diff changeset
120 // this addresses the most common cases fairly cheaply.
anatofuz
parents:
diff changeset
121 class ReplayPreamble : private PPCallbacks {
anatofuz
parents:
diff changeset
122 public:
anatofuz
parents:
diff changeset
123 // Attach preprocessor hooks such that preamble events will be injected at
anatofuz
parents:
diff changeset
124 // the appropriate time.
anatofuz
parents:
diff changeset
125 // Events will be delivered to the *currently registered* PP callbacks.
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
126 static void attach(std::vector<Inclusion> Includes, CompilerInstance &Clang,
173
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
127 const PreambleBounds &PB) {
150
anatofuz
parents:
diff changeset
128 auto &PP = Clang.getPreprocessor();
anatofuz
parents:
diff changeset
129 auto *ExistingCallbacks = PP.getPPCallbacks();
anatofuz
parents:
diff changeset
130 // No need to replay events if nobody is listening.
anatofuz
parents:
diff changeset
131 if (!ExistingCallbacks)
anatofuz
parents:
diff changeset
132 return;
173
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
133 PP.addPPCallbacks(std::unique_ptr<PPCallbacks>(new ReplayPreamble(
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
134 std::move(Includes), ExistingCallbacks, Clang.getSourceManager(), PP,
173
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
135 Clang.getLangOpts(), PB)));
150
anatofuz
parents:
diff changeset
136 // We're relying on the fact that addPPCallbacks keeps the old PPCallbacks
anatofuz
parents:
diff changeset
137 // around, creating a chaining wrapper. Guard against other implementations.
anatofuz
parents:
diff changeset
138 assert(PP.getPPCallbacks() != ExistingCallbacks &&
anatofuz
parents:
diff changeset
139 "Expected chaining implementation");
anatofuz
parents:
diff changeset
140 }
anatofuz
parents:
diff changeset
141
anatofuz
parents:
diff changeset
142 private:
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
143 ReplayPreamble(std::vector<Inclusion> Includes, PPCallbacks *Delegate,
150
anatofuz
parents:
diff changeset
144 const SourceManager &SM, Preprocessor &PP,
173
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
145 const LangOptions &LangOpts, const PreambleBounds &PB)
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
146 : Includes(std::move(Includes)), Delegate(Delegate), SM(SM), PP(PP) {
173
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
147 // Only tokenize the preamble section of the main file, as we are not
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
148 // interested in the rest of the tokens.
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
149 MainFileTokens = syntax::tokenize(
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
150 syntax::FileRange(SM.getMainFileID(), 0, PB.Size), SM, LangOpts);
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
151 }
150
anatofuz
parents:
diff changeset
152
anatofuz
parents:
diff changeset
153 // In a normal compile, the preamble traverses the following structure:
anatofuz
parents:
diff changeset
154 //
anatofuz
parents:
diff changeset
155 // mainfile.cpp
anatofuz
parents:
diff changeset
156 // <built-in>
anatofuz
parents:
diff changeset
157 // ... macro definitions like __cplusplus ...
anatofuz
parents:
diff changeset
158 // <command-line>
anatofuz
parents:
diff changeset
159 // ... macro definitions for args like -Dfoo=bar ...
anatofuz
parents:
diff changeset
160 // "header1.h"
anatofuz
parents:
diff changeset
161 // ... header file contents ...
anatofuz
parents:
diff changeset
162 // "header2.h"
anatofuz
parents:
diff changeset
163 // ... header file contents ...
anatofuz
parents:
diff changeset
164 // ... main file contents ...
anatofuz
parents:
diff changeset
165 //
anatofuz
parents:
diff changeset
166 // When using a preamble, the "header1" and "header2" subtrees get skipped.
anatofuz
parents:
diff changeset
167 // We insert them right after the built-in header, which still appears.
anatofuz
parents:
diff changeset
168 void FileChanged(SourceLocation Loc, FileChangeReason Reason,
anatofuz
parents:
diff changeset
169 SrcMgr::CharacteristicKind Kind, FileID PrevFID) override {
anatofuz
parents:
diff changeset
170 // It'd be nice if there was a better way to identify built-in headers...
anatofuz
parents:
diff changeset
171 if (Reason == FileChangeReason::ExitFile &&
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
172 SM.getBufferOrFake(PrevFID).getBufferIdentifier() == "<built-in>")
150
anatofuz
parents:
diff changeset
173 replay();
anatofuz
parents:
diff changeset
174 }
anatofuz
parents:
diff changeset
175
anatofuz
parents:
diff changeset
176 void replay() {
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
177 for (const auto &Inc : Includes) {
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
178 llvm::Optional<FileEntryRef> File;
150
anatofuz
parents:
diff changeset
179 if (Inc.Resolved != "")
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
180 File = expectedToOptional(SM.getFileManager().getFileRef(Inc.Resolved));
150
anatofuz
parents:
diff changeset
181
173
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
182 // Re-lex the #include directive to find its interesting parts.
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
183 auto HashLoc = SM.getComposedLoc(SM.getMainFileID(), Inc.HashOffset);
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
184 auto HashTok = llvm::partition_point(MainFileTokens,
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
185 [&HashLoc](const syntax::Token &T) {
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
186 return T.location() < HashLoc;
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
187 });
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
188 assert(HashTok != MainFileTokens.end() && HashTok->kind() == tok::hash);
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
189
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
190 auto IncludeTok = std::next(HashTok);
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
191 assert(IncludeTok != MainFileTokens.end());
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
192
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
193 auto FileTok = std::next(IncludeTok);
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
194 assert(FileTok != MainFileTokens.end());
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
195
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
196 // Create a fake import/include token, none of the callers seem to care
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
197 // about clang::Token::Flags.
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
198 Token SynthesizedIncludeTok;
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
199 SynthesizedIncludeTok.startToken();
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
200 SynthesizedIncludeTok.setLocation(IncludeTok->location());
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
201 SynthesizedIncludeTok.setLength(IncludeTok->length());
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
202 SynthesizedIncludeTok.setKind(tok::raw_identifier);
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
203 SynthesizedIncludeTok.setRawIdentifierData(IncludeTok->text(SM).data());
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
204 PP.LookUpIdentifierInfo(SynthesizedIncludeTok);
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
205
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
206 // Same here, create a fake one for Filename, including angles or quotes.
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
207 Token SynthesizedFilenameTok;
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
208 SynthesizedFilenameTok.startToken();
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
209 SynthesizedFilenameTok.setLocation(FileTok->location());
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
210 // Note that we can't make use of FileTok->length/text in here as in the
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
211 // case of angled includes this will contain tok::less instead of
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
212 // filename. Whereas Inc.Written contains the full header name including
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
213 // quotes/angles.
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
214 SynthesizedFilenameTok.setLength(Inc.Written.length());
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
215 SynthesizedFilenameTok.setKind(tok::header_name);
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
216 SynthesizedFilenameTok.setLiteralData(Inc.Written.data());
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
217
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
218 const FileEntry *FE = File ? &File->getFileEntry() : nullptr;
150
anatofuz
parents:
diff changeset
219 llvm::StringRef WrittenFilename =
anatofuz
parents:
diff changeset
220 llvm::StringRef(Inc.Written).drop_front().drop_back();
173
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
221 Delegate->InclusionDirective(HashTok->location(), SynthesizedIncludeTok,
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
222 WrittenFilename, Inc.Written.front() == '<',
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
223 FileTok->range(SM).toCharRange(SM), FE,
173
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
224 "SearchPath", "RelPath",
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
225 /*Imported=*/nullptr, Inc.FileKind);
150
anatofuz
parents:
diff changeset
226 if (File)
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
227 Delegate->FileSkipped(*File, SynthesizedFilenameTok, Inc.FileKind);
150
anatofuz
parents:
diff changeset
228 else {
anatofuz
parents:
diff changeset
229 llvm::SmallString<1> UnusedRecovery;
anatofuz
parents:
diff changeset
230 Delegate->FileNotFound(WrittenFilename, UnusedRecovery);
anatofuz
parents:
diff changeset
231 }
anatofuz
parents:
diff changeset
232 }
anatofuz
parents:
diff changeset
233 }
anatofuz
parents:
diff changeset
234
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
235 const std::vector<Inclusion> Includes;
150
anatofuz
parents:
diff changeset
236 PPCallbacks *Delegate;
anatofuz
parents:
diff changeset
237 const SourceManager &SM;
anatofuz
parents:
diff changeset
238 Preprocessor &PP;
173
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
239 std::vector<syntax::Token> MainFileTokens;
150
anatofuz
parents:
diff changeset
240 };
anatofuz
parents:
diff changeset
241
anatofuz
parents:
diff changeset
242 } // namespace
anatofuz
parents:
diff changeset
243
anatofuz
parents:
diff changeset
244 llvm::Optional<ParsedAST>
173
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
245 ParsedAST::build(llvm::StringRef Filename, const ParseInputs &Inputs,
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
246 std::unique_ptr<clang::CompilerInvocation> CI,
150
anatofuz
parents:
diff changeset
247 llvm::ArrayRef<Diag> CompilerInvocationDiags,
173
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
248 std::shared_ptr<const PreambleData> Preamble) {
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
249 trace::Span Tracer("BuildAST");
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
250 SPAN_ATTACH(Tracer, "File", Filename);
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
251
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
252 auto VFS = Inputs.TFS->view(Inputs.CompileCommand.Directory);
173
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
253 if (Preamble && Preamble->StatCache)
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
254 VFS = Preamble->StatCache->getConsumingFS(std::move(VFS));
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
255
150
anatofuz
parents:
diff changeset
256 assert(CI);
anatofuz
parents:
diff changeset
257 // Command-line parsing sets DisableFree to true by default, but we don't want
anatofuz
parents:
diff changeset
258 // to leak memory in clangd.
anatofuz
parents:
diff changeset
259 CI->getFrontendOpts().DisableFree = false;
anatofuz
parents:
diff changeset
260 const PrecompiledPreamble *PreamblePCH =
anatofuz
parents:
diff changeset
261 Preamble ? &Preamble->Preamble : nullptr;
anatofuz
parents:
diff changeset
262
173
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
263 // This is on-by-default in windows to allow parsing SDK headers, but it
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
264 // breaks many features. Disable it for the main-file (not preamble).
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
265 CI->getLangOpts()->DelayedTemplateParsing = false;
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
266
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
267 std::vector<std::unique_ptr<FeatureModule::ASTListener>> ASTListeners;
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
268 if (Inputs.FeatureModules) {
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
269 for (auto &M : *Inputs.FeatureModules) {
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
270 if (auto Listener = M.astListeners())
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
271 ASTListeners.emplace_back(std::move(Listener));
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
272 }
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
273 }
150
anatofuz
parents:
diff changeset
274 StoreDiags ASTDiags;
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
275 ASTDiags.setDiagCallback(
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
276 [&ASTListeners](const clang::Diagnostic &D, clangd::Diag &Diag) {
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
277 llvm::for_each(ASTListeners,
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
278 [&](const auto &L) { L->sawDiagnostic(D, Diag); });
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
279 });
150
anatofuz
parents:
diff changeset
280
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
281 llvm::Optional<PreamblePatch> Patch;
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
282 bool PreserveDiags = true;
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
283 if (Preamble) {
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
284 Patch = PreamblePatch::create(Filename, Inputs, *Preamble);
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
285 Patch->apply(*CI);
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
286 PreserveDiags = Patch->preserveDiagnostics();
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
287 }
173
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
288 auto Clang = prepareCompilerInstance(
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
289 std::move(CI), PreamblePCH,
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
290 llvm::MemoryBuffer::getMemBufferCopy(Inputs.Contents, Filename), VFS,
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
291 ASTDiags);
150
anatofuz
parents:
diff changeset
292 if (!Clang)
anatofuz
parents:
diff changeset
293 return None;
anatofuz
parents:
diff changeset
294
anatofuz
parents:
diff changeset
295 auto Action = std::make_unique<ClangdFrontendAction>();
anatofuz
parents:
diff changeset
296 const FrontendInputFile &MainInput = Clang->getFrontendOpts().Inputs[0];
anatofuz
parents:
diff changeset
297 if (!Action->BeginSourceFile(*Clang, MainInput)) {
anatofuz
parents:
diff changeset
298 log("BeginSourceFile() failed when building AST for {0}",
anatofuz
parents:
diff changeset
299 MainInput.getFile());
anatofuz
parents:
diff changeset
300 return None;
anatofuz
parents:
diff changeset
301 }
anatofuz
parents:
diff changeset
302
anatofuz
parents:
diff changeset
303 // Set up ClangTidy. Must happen after BeginSourceFile() so ASTContext exists.
173
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
304 // Clang-tidy has some limitations to ensure reasonable performance:
150
anatofuz
parents:
diff changeset
305 // - checks don't see all preprocessor events in the preamble
anatofuz
parents:
diff changeset
306 // - matchers run only over the main-file top-level decls (and can't see
anatofuz
parents:
diff changeset
307 // ancestors outside this scope).
anatofuz
parents:
diff changeset
308 // In practice almost all checks work well without modifications.
anatofuz
parents:
diff changeset
309 std::vector<std::unique_ptr<tidy::ClangTidyCheck>> CTChecks;
anatofuz
parents:
diff changeset
310 ast_matchers::MatchFinder CTFinder;
anatofuz
parents:
diff changeset
311 llvm::Optional<tidy::ClangTidyContext> CTContext;
anatofuz
parents:
diff changeset
312 {
anatofuz
parents:
diff changeset
313 trace::Span Tracer("ClangTidyInit");
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
314 tidy::ClangTidyOptions ClangTidyOpts =
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
315 getTidyOptionsForFile(Inputs.ClangTidyProvider, Filename);
150
anatofuz
parents:
diff changeset
316 dlog("ClangTidy configuration for file {0}: {1}", Filename,
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
317 tidy::configurationAsText(ClangTidyOpts));
150
anatofuz
parents:
diff changeset
318 tidy::ClangTidyCheckFactories CTFactories;
anatofuz
parents:
diff changeset
319 for (const auto &E : tidy::ClangTidyModuleRegistry::entries())
anatofuz
parents:
diff changeset
320 E.instantiate()->addCheckFactories(CTFactories);
anatofuz
parents:
diff changeset
321 CTContext.emplace(std::make_unique<tidy::DefaultOptionsProvider>(
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
322 tidy::ClangTidyGlobalOptions(), ClangTidyOpts));
150
anatofuz
parents:
diff changeset
323 CTContext->setDiagnosticsEngine(&Clang->getDiagnostics());
anatofuz
parents:
diff changeset
324 CTContext->setASTContext(&Clang->getASTContext());
anatofuz
parents:
diff changeset
325 CTContext->setCurrentFile(Filename);
anatofuz
parents:
diff changeset
326 CTChecks = CTFactories.createChecks(CTContext.getPointer());
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
327 llvm::erase_if(CTChecks, [&](const auto &Check) {
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
328 return !Check->isLanguageVersionSupported(CTContext->getLangOpts());
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
329 });
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
330 Preprocessor *PP = &Clang->getPreprocessor();
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
331 for (const auto &Check : CTChecks) {
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
332 Check->registerPPCallbacks(Clang->getSourceManager(), PP, PP);
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
333 Check->registerMatchers(&CTFinder);
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
334 }
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
335
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
336 const Config &Cfg = Config::current();
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
337 ASTDiags.setLevelAdjuster([&](DiagnosticsEngine::Level DiagLevel,
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
338 const clang::Diagnostic &Info) {
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
339 if (Cfg.Diagnostics.SuppressAll ||
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
340 isBuiltinDiagnosticSuppressed(Info.getID(), Cfg.Diagnostics.Suppress))
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
341 return DiagnosticsEngine::Ignored;
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
342 if (!CTChecks.empty()) {
150
anatofuz
parents:
diff changeset
343 std::string CheckName = CTContext->getCheckName(Info.getID());
anatofuz
parents:
diff changeset
344 bool IsClangTidyDiag = !CheckName.empty();
anatofuz
parents:
diff changeset
345 if (IsClangTidyDiag) {
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
346 if (Cfg.Diagnostics.Suppress.contains(CheckName))
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
347 return DiagnosticsEngine::Ignored;
173
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
348 // Check for suppression comment. Skip the check for diagnostics not
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
349 // in the main file, because we don't want that function to query the
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
350 // source buffer for preamble files. For the same reason, we ask
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
351 // shouldSuppressDiagnostic to avoid I/O.
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
352 // We let suppression comments take precedence over warning-as-error
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
353 // to match clang-tidy's behaviour.
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
354 bool IsInsideMainFile =
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
355 Info.hasSourceManager() &&
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
356 isInsideMainFile(Info.getLocation(), Info.getSourceManager());
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
357 if (IsInsideMainFile &&
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
358 tidy::shouldSuppressDiagnostic(DiagLevel, Info, *CTContext,
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
359 /*AllowIO=*/false)) {
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
360 return DiagnosticsEngine::Ignored;
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
361 }
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
362
150
anatofuz
parents:
diff changeset
363 // Check for warning-as-error.
anatofuz
parents:
diff changeset
364 if (DiagLevel == DiagnosticsEngine::Warning &&
anatofuz
parents:
diff changeset
365 CTContext->treatAsError(CheckName)) {
anatofuz
parents:
diff changeset
366 return DiagnosticsEngine::Error;
anatofuz
parents:
diff changeset
367 }
anatofuz
parents:
diff changeset
368 }
anatofuz
parents:
diff changeset
369 }
anatofuz
parents:
diff changeset
370 return DiagLevel;
anatofuz
parents:
diff changeset
371 });
anatofuz
parents:
diff changeset
372 }
anatofuz
parents:
diff changeset
373
anatofuz
parents:
diff changeset
374 // Add IncludeFixer which can recover diagnostics caused by missing includes
anatofuz
parents:
diff changeset
375 // (e.g. incomplete type) and attach include insertion fixes to diagnostics.
anatofuz
parents:
diff changeset
376 llvm::Optional<IncludeFixer> FixIncludes;
anatofuz
parents:
diff changeset
377 auto BuildDir = VFS->getCurrentWorkingDirectory();
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
378 if (Inputs.Index && !BuildDir.getError()) {
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
379 auto Style = getFormatStyleForFile(Filename, Inputs.Contents, *Inputs.TFS);
150
anatofuz
parents:
diff changeset
380 auto Inserter = std::make_shared<IncludeInserter>(
173
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
381 Filename, Inputs.Contents, Style, BuildDir.get(),
150
anatofuz
parents:
diff changeset
382 &Clang->getPreprocessor().getHeaderSearchInfo());
anatofuz
parents:
diff changeset
383 if (Preamble) {
anatofuz
parents:
diff changeset
384 for (const auto &Inc : Preamble->Includes.MainFileIncludes)
anatofuz
parents:
diff changeset
385 Inserter->addExisting(Inc);
anatofuz
parents:
diff changeset
386 }
173
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
387 FixIncludes.emplace(Filename, Inserter, *Inputs.Index,
150
anatofuz
parents:
diff changeset
388 /*IndexRequestLimit=*/5);
anatofuz
parents:
diff changeset
389 ASTDiags.contributeFixes([&FixIncludes](DiagnosticsEngine::Level DiagLevl,
anatofuz
parents:
diff changeset
390 const clang::Diagnostic &Info) {
anatofuz
parents:
diff changeset
391 return FixIncludes->fix(DiagLevl, Info);
anatofuz
parents:
diff changeset
392 });
anatofuz
parents:
diff changeset
393 Clang->setExternalSemaSource(FixIncludes->unresolvedNameRecorder());
anatofuz
parents:
diff changeset
394 }
anatofuz
parents:
diff changeset
395
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
396 IncludeStructure Includes;
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
397 // If we are using a preamble, copy existing includes.
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
398 if (Preamble) {
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
399 Includes = Preamble->Includes;
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
400 Includes.MainFileIncludes = Patch->preambleIncludes();
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
401 // Replay the preamble includes so that clang-tidy checks can see them.
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
402 ReplayPreamble::attach(Patch->preambleIncludes(), *Clang,
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
403 Patch->modifiedBounds());
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
404 }
150
anatofuz
parents:
diff changeset
405 // Important: collectIncludeStructure is registered *after* ReplayPreamble!
anatofuz
parents:
diff changeset
406 // Otherwise we would collect the replayed includes again...
anatofuz
parents:
diff changeset
407 // (We can't *just* use the replayed includes, they don't have Resolved path).
anatofuz
parents:
diff changeset
408 Clang->getPreprocessor().addPPCallbacks(
anatofuz
parents:
diff changeset
409 collectIncludeStructureCallback(Clang->getSourceManager(), &Includes));
anatofuz
parents:
diff changeset
410 // Copy over the macros in the preamble region of the main file, and combine
anatofuz
parents:
diff changeset
411 // with non-preamble macros below.
anatofuz
parents:
diff changeset
412 MainFileMacros Macros;
anatofuz
parents:
diff changeset
413 if (Preamble)
anatofuz
parents:
diff changeset
414 Macros = Preamble->Macros;
anatofuz
parents:
diff changeset
415 Clang->getPreprocessor().addPPCallbacks(
anatofuz
parents:
diff changeset
416 std::make_unique<CollectMainFileMacros>(Clang->getSourceManager(),
173
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
417 Macros));
150
anatofuz
parents:
diff changeset
418
anatofuz
parents:
diff changeset
419 // Copy over the includes from the preamble, then combine with the
anatofuz
parents:
diff changeset
420 // non-preamble includes below.
anatofuz
parents:
diff changeset
421 CanonicalIncludes CanonIncludes;
anatofuz
parents:
diff changeset
422 if (Preamble)
anatofuz
parents:
diff changeset
423 CanonIncludes = Preamble->CanonIncludes;
anatofuz
parents:
diff changeset
424 else
anatofuz
parents:
diff changeset
425 CanonIncludes.addSystemHeadersMapping(Clang->getLangOpts());
anatofuz
parents:
diff changeset
426 std::unique_ptr<CommentHandler> IWYUHandler =
anatofuz
parents:
diff changeset
427 collectIWYUHeaderMaps(&CanonIncludes);
anatofuz
parents:
diff changeset
428 Clang->getPreprocessor().addCommentHandler(IWYUHandler.get());
anatofuz
parents:
diff changeset
429
anatofuz
parents:
diff changeset
430 // Collect tokens of the main file.
anatofuz
parents:
diff changeset
431 syntax::TokenCollector CollectTokens(Clang->getPreprocessor());
anatofuz
parents:
diff changeset
432
anatofuz
parents:
diff changeset
433 if (llvm::Error Err = Action->Execute())
anatofuz
parents:
diff changeset
434 log("Execute() failed when building AST for {0}: {1}", MainInput.getFile(),
anatofuz
parents:
diff changeset
435 toString(std::move(Err)));
anatofuz
parents:
diff changeset
436
anatofuz
parents:
diff changeset
437 // We have to consume the tokens before running clang-tidy to avoid collecting
anatofuz
parents:
diff changeset
438 // tokens from running the preprocessor inside the checks (only
anatofuz
parents:
diff changeset
439 // modernize-use-trailing-return-type does that today).
anatofuz
parents:
diff changeset
440 syntax::TokenBuffer Tokens = std::move(CollectTokens).consume();
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
441 // Makes SelectionTree build much faster.
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
442 Tokens.indexExpandedTokens();
150
anatofuz
parents:
diff changeset
443 std::vector<Decl *> ParsedDecls = Action->takeTopLevelDecls();
anatofuz
parents:
diff changeset
444 // AST traversals should exclude the preamble, to avoid performance cliffs.
anatofuz
parents:
diff changeset
445 Clang->getASTContext().setTraversalScope(ParsedDecls);
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
446 if (!CTChecks.empty()) {
150
anatofuz
parents:
diff changeset
447 // Run the AST-dependent part of the clang-tidy checks.
anatofuz
parents:
diff changeset
448 // (The preprocessor part ran already, via PPCallbacks).
anatofuz
parents:
diff changeset
449 trace::Span Tracer("ClangTidyMatch");
anatofuz
parents:
diff changeset
450 CTFinder.matchAST(Clang->getASTContext());
anatofuz
parents:
diff changeset
451 }
anatofuz
parents:
diff changeset
452
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
453 // XXX: This is messy: clang-tidy checks flush some diagnostics at EOF.
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
454 // However Action->EndSourceFile() would destroy the ASTContext!
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
455 // So just inform the preprocessor of EOF, while keeping everything alive.
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
456 Clang->getPreprocessor().EndSourceFile();
150
anatofuz
parents:
diff changeset
457 // UnitDiagsConsumer is local, we can not store it in CompilerInstance that
anatofuz
parents:
diff changeset
458 // has a longer lifetime.
anatofuz
parents:
diff changeset
459 Clang->getDiagnostics().setClient(new IgnoreDiagnostics);
anatofuz
parents:
diff changeset
460 // CompilerInstance won't run this callback, do it directly.
anatofuz
parents:
diff changeset
461 ASTDiags.EndSourceFile();
anatofuz
parents:
diff changeset
462
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
463 llvm::Optional<std::vector<Diag>> Diags;
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
464 // FIXME: Also skip generation of diagnostics alltogether to speed up ast
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
465 // builds when we are patching a stale preamble.
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
466 if (PreserveDiags) {
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
467 Diags = CompilerInvocationDiags;
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
468 // Add diagnostics from the preamble, if any.
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
469 if (Preamble)
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
470 Diags->insert(Diags->end(), Preamble->Diags.begin(),
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
471 Preamble->Diags.end());
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
472 // Finally, add diagnostics coming from the AST.
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
473 {
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
474 std::vector<Diag> D = ASTDiags.take(CTContext.getPointer());
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
475 Diags->insert(Diags->end(), D.begin(), D.end());
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
476 }
150
anatofuz
parents:
diff changeset
477 }
173
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
478 return ParsedAST(Inputs.Version, std::move(Preamble), std::move(Clang),
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
479 std::move(Action), std::move(Tokens), std::move(Macros),
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
480 std::move(ParsedDecls), std::move(Diags),
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
481 std::move(Includes), std::move(CanonIncludes));
150
anatofuz
parents:
diff changeset
482 }
anatofuz
parents:
diff changeset
483
anatofuz
parents:
diff changeset
484 ParsedAST::ParsedAST(ParsedAST &&Other) = default;
anatofuz
parents:
diff changeset
485
anatofuz
parents:
diff changeset
486 ParsedAST &ParsedAST::operator=(ParsedAST &&Other) = default;
anatofuz
parents:
diff changeset
487
anatofuz
parents:
diff changeset
488 ParsedAST::~ParsedAST() {
anatofuz
parents:
diff changeset
489 if (Action) {
anatofuz
parents:
diff changeset
490 // We already notified the PP of end-of-file earlier, so detach it first.
anatofuz
parents:
diff changeset
491 // We must keep it alive until after EndSourceFile(), Sema relies on this.
anatofuz
parents:
diff changeset
492 auto PP = Clang->getPreprocessorPtr(); // Keep PP alive for now.
anatofuz
parents:
diff changeset
493 Clang->setPreprocessor(nullptr); // Detach so we don't send EOF again.
anatofuz
parents:
diff changeset
494 Action->EndSourceFile(); // Destroy ASTContext and Sema.
anatofuz
parents:
diff changeset
495 // Now Sema is gone, it's safe for PP to go out of scope.
anatofuz
parents:
diff changeset
496 }
anatofuz
parents:
diff changeset
497 }
anatofuz
parents:
diff changeset
498
anatofuz
parents:
diff changeset
499 ASTContext &ParsedAST::getASTContext() { return Clang->getASTContext(); }
anatofuz
parents:
diff changeset
500
anatofuz
parents:
diff changeset
501 const ASTContext &ParsedAST::getASTContext() const {
anatofuz
parents:
diff changeset
502 return Clang->getASTContext();
anatofuz
parents:
diff changeset
503 }
anatofuz
parents:
diff changeset
504
anatofuz
parents:
diff changeset
505 Preprocessor &ParsedAST::getPreprocessor() { return Clang->getPreprocessor(); }
anatofuz
parents:
diff changeset
506
anatofuz
parents:
diff changeset
507 std::shared_ptr<Preprocessor> ParsedAST::getPreprocessorPtr() {
anatofuz
parents:
diff changeset
508 return Clang->getPreprocessorPtr();
anatofuz
parents:
diff changeset
509 }
anatofuz
parents:
diff changeset
510
anatofuz
parents:
diff changeset
511 const Preprocessor &ParsedAST::getPreprocessor() const {
anatofuz
parents:
diff changeset
512 return Clang->getPreprocessor();
anatofuz
parents:
diff changeset
513 }
anatofuz
parents:
diff changeset
514
anatofuz
parents:
diff changeset
515 llvm::ArrayRef<Decl *> ParsedAST::getLocalTopLevelDecls() {
anatofuz
parents:
diff changeset
516 return LocalTopLevelDecls;
anatofuz
parents:
diff changeset
517 }
anatofuz
parents:
diff changeset
518
anatofuz
parents:
diff changeset
519 const MainFileMacros &ParsedAST::getMacros() const { return Macros; }
anatofuz
parents:
diff changeset
520
anatofuz
parents:
diff changeset
521 std::size_t ParsedAST::getUsedBytes() const {
anatofuz
parents:
diff changeset
522 auto &AST = getASTContext();
anatofuz
parents:
diff changeset
523 // FIXME(ibiryukov): we do not account for the dynamically allocated part of
anatofuz
parents:
diff changeset
524 // Message and Fixes inside each diagnostic.
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
525 std::size_t Total = clangd::getUsedBytes(LocalTopLevelDecls) +
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
526 (Diags ? clangd::getUsedBytes(*Diags) : 0);
150
anatofuz
parents:
diff changeset
527
anatofuz
parents:
diff changeset
528 // FIXME: the rest of the function is almost a direct copy-paste from
anatofuz
parents:
diff changeset
529 // libclang's clang_getCXTUResourceUsage. We could share the implementation.
anatofuz
parents:
diff changeset
530
173
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
531 // Sum up various allocators inside the ast context and the preprocessor.
150
anatofuz
parents:
diff changeset
532 Total += AST.getASTAllocatedMemory();
anatofuz
parents:
diff changeset
533 Total += AST.getSideTableAllocatedMemory();
anatofuz
parents:
diff changeset
534 Total += AST.Idents.getAllocator().getTotalMemory();
anatofuz
parents:
diff changeset
535 Total += AST.Selectors.getTotalMemory();
anatofuz
parents:
diff changeset
536
anatofuz
parents:
diff changeset
537 Total += AST.getSourceManager().getContentCacheSize();
anatofuz
parents:
diff changeset
538 Total += AST.getSourceManager().getDataStructureSizes();
anatofuz
parents:
diff changeset
539 Total += AST.getSourceManager().getMemoryBufferSizes().malloc_bytes;
anatofuz
parents:
diff changeset
540
anatofuz
parents:
diff changeset
541 if (ExternalASTSource *Ext = AST.getExternalSource())
anatofuz
parents:
diff changeset
542 Total += Ext->getMemoryBufferSizes().malloc_bytes;
anatofuz
parents:
diff changeset
543
anatofuz
parents:
diff changeset
544 const Preprocessor &PP = getPreprocessor();
anatofuz
parents:
diff changeset
545 Total += PP.getTotalMemory();
anatofuz
parents:
diff changeset
546 if (PreprocessingRecord *PRec = PP.getPreprocessingRecord())
anatofuz
parents:
diff changeset
547 Total += PRec->getTotalMemory();
anatofuz
parents:
diff changeset
548 Total += PP.getHeaderSearchInfo().getTotalMemory();
anatofuz
parents:
diff changeset
549
anatofuz
parents:
diff changeset
550 return Total;
anatofuz
parents:
diff changeset
551 }
anatofuz
parents:
diff changeset
552
anatofuz
parents:
diff changeset
553 const IncludeStructure &ParsedAST::getIncludeStructure() const {
anatofuz
parents:
diff changeset
554 return Includes;
anatofuz
parents:
diff changeset
555 }
anatofuz
parents:
diff changeset
556
anatofuz
parents:
diff changeset
557 const CanonicalIncludes &ParsedAST::getCanonicalIncludes() const {
anatofuz
parents:
diff changeset
558 return CanonIncludes;
anatofuz
parents:
diff changeset
559 }
anatofuz
parents:
diff changeset
560
173
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
561 ParsedAST::ParsedAST(llvm::StringRef Version,
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
562 std::shared_ptr<const PreambleData> Preamble,
150
anatofuz
parents:
diff changeset
563 std::unique_ptr<CompilerInstance> Clang,
anatofuz
parents:
diff changeset
564 std::unique_ptr<FrontendAction> Action,
anatofuz
parents:
diff changeset
565 syntax::TokenBuffer Tokens, MainFileMacros Macros,
anatofuz
parents:
diff changeset
566 std::vector<Decl *> LocalTopLevelDecls,
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
567 llvm::Optional<std::vector<Diag>> Diags,
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
568 IncludeStructure Includes, CanonicalIncludes CanonIncludes)
173
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
569 : Version(Version), Preamble(std::move(Preamble)), Clang(std::move(Clang)),
150
anatofuz
parents:
diff changeset
570 Action(std::move(Action)), Tokens(std::move(Tokens)),
anatofuz
parents:
diff changeset
571 Macros(std::move(Macros)), Diags(std::move(Diags)),
anatofuz
parents:
diff changeset
572 LocalTopLevelDecls(std::move(LocalTopLevelDecls)),
anatofuz
parents:
diff changeset
573 Includes(std::move(Includes)), CanonIncludes(std::move(CanonIncludes)) {
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
574 Resolver = std::make_unique<HeuristicResolver>(getASTContext());
150
anatofuz
parents:
diff changeset
575 assert(this->Clang);
anatofuz
parents:
diff changeset
576 assert(this->Action);
anatofuz
parents:
diff changeset
577 }
anatofuz
parents:
diff changeset
578
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
579 llvm::Optional<llvm::StringRef> ParsedAST::preambleVersion() const {
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
580 if (!Preamble)
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
581 return llvm::None;
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
582 return llvm::StringRef(Preamble->Version);
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
583 }
150
anatofuz
parents:
diff changeset
584 } // namespace clangd
anatofuz
parents:
diff changeset
585 } // namespace clang