annotate clang-tools-extra/clang-tidy/ExpandModularHeadersPPCallbacks.cpp @ 160:dbfec6499728

...
author anatofuz
date Wed, 18 Mar 2020 19:11:03 +0900
parents 1d019706d866
children 0572611fdcc8
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
150
anatofuz
parents:
diff changeset
1 //===- ExpandModularHeadersPPCallbacks.h - clang-tidy -----------*- 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 "ExpandModularHeadersPPCallbacks.h"
anatofuz
parents:
diff changeset
10 #include "clang/Frontend/CompilerInstance.h"
anatofuz
parents:
diff changeset
11 #include "clang/Lex/PreprocessorOptions.h"
anatofuz
parents:
diff changeset
12 #include "clang/Serialization/ASTReader.h"
anatofuz
parents:
diff changeset
13
anatofuz
parents:
diff changeset
14 namespace clang {
anatofuz
parents:
diff changeset
15 namespace tooling {
anatofuz
parents:
diff changeset
16
anatofuz
parents:
diff changeset
17 class ExpandModularHeadersPPCallbacks::FileRecorder {
anatofuz
parents:
diff changeset
18 public:
anatofuz
parents:
diff changeset
19 /// Records that a given file entry is needed for replaying callbacks.
anatofuz
parents:
diff changeset
20 void addNecessaryFile(const FileEntry *File) { FilesToRecord.insert(File); }
anatofuz
parents:
diff changeset
21
anatofuz
parents:
diff changeset
22 /// Records content for a file and adds it to the FileSystem.
anatofuz
parents:
diff changeset
23 void recordFileContent(const FileEntry *File,
anatofuz
parents:
diff changeset
24 const SrcMgr::ContentCache &ContentCache,
anatofuz
parents:
diff changeset
25 llvm::vfs::InMemoryFileSystem &InMemoryFs) {
anatofuz
parents:
diff changeset
26 // Return if we are not interested in the contents of this file.
anatofuz
parents:
diff changeset
27 if (!FilesToRecord.count(File))
anatofuz
parents:
diff changeset
28 return;
anatofuz
parents:
diff changeset
29
anatofuz
parents:
diff changeset
30 // FIXME: Why is this happening? We might be losing contents here.
anatofuz
parents:
diff changeset
31 if (!ContentCache.getRawBuffer())
anatofuz
parents:
diff changeset
32 return;
anatofuz
parents:
diff changeset
33
anatofuz
parents:
diff changeset
34 InMemoryFs.addFile(File->getName(), /*ModificationTime=*/0,
anatofuz
parents:
diff changeset
35 llvm::MemoryBuffer::getMemBufferCopy(
anatofuz
parents:
diff changeset
36 ContentCache.getRawBuffer()->getBuffer()));
anatofuz
parents:
diff changeset
37 // Remove the file from the set of necessary files.
anatofuz
parents:
diff changeset
38 FilesToRecord.erase(File);
anatofuz
parents:
diff changeset
39 }
anatofuz
parents:
diff changeset
40
anatofuz
parents:
diff changeset
41 /// Makes sure we have contents for all the files we were interested in. Ideally
anatofuz
parents:
diff changeset
42 /// `FilesToRecord` should be empty.
anatofuz
parents:
diff changeset
43 void checkAllFilesRecorded() {
anatofuz
parents:
diff changeset
44 for (auto FileEntry : FilesToRecord)
anatofuz
parents:
diff changeset
45 llvm::errs() << "Did not record contents for input file: "
anatofuz
parents:
diff changeset
46 << FileEntry->getName() << "\n";
anatofuz
parents:
diff changeset
47 }
anatofuz
parents:
diff changeset
48
anatofuz
parents:
diff changeset
49 private:
anatofuz
parents:
diff changeset
50 /// A set of files whose contents are to be recorded.
anatofuz
parents:
diff changeset
51 llvm::DenseSet<const FileEntry *> FilesToRecord;
anatofuz
parents:
diff changeset
52 };
anatofuz
parents:
diff changeset
53
anatofuz
parents:
diff changeset
54 ExpandModularHeadersPPCallbacks::ExpandModularHeadersPPCallbacks(
anatofuz
parents:
diff changeset
55 CompilerInstance *CI,
anatofuz
parents:
diff changeset
56 IntrusiveRefCntPtr<llvm::vfs::OverlayFileSystem> OverlayFS)
anatofuz
parents:
diff changeset
57 : Recorder(std::make_unique<FileRecorder>()), Compiler(*CI),
anatofuz
parents:
diff changeset
58 InMemoryFs(new llvm::vfs::InMemoryFileSystem),
anatofuz
parents:
diff changeset
59 Sources(Compiler.getSourceManager()),
anatofuz
parents:
diff changeset
60 // Forward the new diagnostics to the original DiagnosticConsumer.
anatofuz
parents:
diff changeset
61 Diags(new DiagnosticIDs, new DiagnosticOptions,
anatofuz
parents:
diff changeset
62 new ForwardingDiagnosticConsumer(Compiler.getDiagnosticClient())),
anatofuz
parents:
diff changeset
63 LangOpts(Compiler.getLangOpts()) {
anatofuz
parents:
diff changeset
64 // Add a FileSystem containing the extra files needed in place of modular
anatofuz
parents:
diff changeset
65 // headers.
anatofuz
parents:
diff changeset
66 OverlayFS->pushOverlay(InMemoryFs);
anatofuz
parents:
diff changeset
67
anatofuz
parents:
diff changeset
68 Diags.setSourceManager(&Sources);
anatofuz
parents:
diff changeset
69
anatofuz
parents:
diff changeset
70 LangOpts.Modules = false;
anatofuz
parents:
diff changeset
71
anatofuz
parents:
diff changeset
72 auto HSO = std::make_shared<HeaderSearchOptions>();
anatofuz
parents:
diff changeset
73 *HSO = Compiler.getHeaderSearchOpts();
anatofuz
parents:
diff changeset
74
anatofuz
parents:
diff changeset
75 HeaderInfo = std::make_unique<HeaderSearch>(HSO, Sources, Diags, LangOpts,
anatofuz
parents:
diff changeset
76 &Compiler.getTarget());
anatofuz
parents:
diff changeset
77
anatofuz
parents:
diff changeset
78 auto PO = std::make_shared<PreprocessorOptions>();
anatofuz
parents:
diff changeset
79 *PO = Compiler.getPreprocessorOpts();
anatofuz
parents:
diff changeset
80
anatofuz
parents:
diff changeset
81 PP = std::make_unique<clang::Preprocessor>(PO, Diags, LangOpts, Sources,
anatofuz
parents:
diff changeset
82 *HeaderInfo, ModuleLoader,
anatofuz
parents:
diff changeset
83 /*IILookup=*/nullptr,
anatofuz
parents:
diff changeset
84 /*OwnsHeaderSearch=*/false);
anatofuz
parents:
diff changeset
85 PP->Initialize(Compiler.getTarget(), Compiler.getAuxTarget());
anatofuz
parents:
diff changeset
86 InitializePreprocessor(*PP, *PO, Compiler.getPCHContainerReader(),
anatofuz
parents:
diff changeset
87 Compiler.getFrontendOpts());
anatofuz
parents:
diff changeset
88 ApplyHeaderSearchOptions(*HeaderInfo, *HSO, LangOpts,
anatofuz
parents:
diff changeset
89 Compiler.getTarget().getTriple());
anatofuz
parents:
diff changeset
90 }
anatofuz
parents:
diff changeset
91
anatofuz
parents:
diff changeset
92 ExpandModularHeadersPPCallbacks::~ExpandModularHeadersPPCallbacks() = default;
anatofuz
parents:
diff changeset
93
anatofuz
parents:
diff changeset
94 Preprocessor *ExpandModularHeadersPPCallbacks::getPreprocessor() const {
anatofuz
parents:
diff changeset
95 return PP.get();
anatofuz
parents:
diff changeset
96 }
anatofuz
parents:
diff changeset
97
anatofuz
parents:
diff changeset
98 void ExpandModularHeadersPPCallbacks::handleModuleFile(
anatofuz
parents:
diff changeset
99 serialization::ModuleFile *MF) {
anatofuz
parents:
diff changeset
100 if (!MF)
anatofuz
parents:
diff changeset
101 return;
anatofuz
parents:
diff changeset
102 // Avoid processing a ModuleFile more than once.
anatofuz
parents:
diff changeset
103 if (VisitedModules.count(MF))
anatofuz
parents:
diff changeset
104 return;
anatofuz
parents:
diff changeset
105 VisitedModules.insert(MF);
anatofuz
parents:
diff changeset
106
anatofuz
parents:
diff changeset
107 // Visit all the input files of this module and mark them to record their
anatofuz
parents:
diff changeset
108 // contents later.
anatofuz
parents:
diff changeset
109 Compiler.getASTReader()->visitInputFiles(
anatofuz
parents:
diff changeset
110 *MF, true, false,
anatofuz
parents:
diff changeset
111 [this](const serialization::InputFile &IF, bool /*IsSystem*/) {
anatofuz
parents:
diff changeset
112 Recorder->addNecessaryFile(IF.getFile());
anatofuz
parents:
diff changeset
113 });
anatofuz
parents:
diff changeset
114 // Recursively handle all transitively imported modules.
anatofuz
parents:
diff changeset
115 for (auto Import : MF->Imports)
anatofuz
parents:
diff changeset
116 handleModuleFile(Import);
anatofuz
parents:
diff changeset
117 }
anatofuz
parents:
diff changeset
118
anatofuz
parents:
diff changeset
119 void ExpandModularHeadersPPCallbacks::parseToLocation(SourceLocation Loc) {
anatofuz
parents:
diff changeset
120 // Load all source locations present in the external sources.
anatofuz
parents:
diff changeset
121 for (unsigned I = 0, N = Sources.loaded_sloc_entry_size(); I != N; ++I) {
anatofuz
parents:
diff changeset
122 Sources.getLoadedSLocEntry(I, nullptr);
anatofuz
parents:
diff changeset
123 }
anatofuz
parents:
diff changeset
124 // Record contents of files we are interested in and add to the FileSystem.
anatofuz
parents:
diff changeset
125 for (auto It = Sources.fileinfo_begin(); It != Sources.fileinfo_end(); ++It) {
anatofuz
parents:
diff changeset
126 Recorder->recordFileContent(It->getFirst(), *It->getSecond(), *InMemoryFs);
anatofuz
parents:
diff changeset
127 }
anatofuz
parents:
diff changeset
128 Recorder->checkAllFilesRecorded();
anatofuz
parents:
diff changeset
129
anatofuz
parents:
diff changeset
130 if (!StartedLexing) {
anatofuz
parents:
diff changeset
131 StartedLexing = true;
anatofuz
parents:
diff changeset
132 PP->Lex(CurrentToken);
anatofuz
parents:
diff changeset
133 }
anatofuz
parents:
diff changeset
134 while (!CurrentToken.is(tok::eof) &&
anatofuz
parents:
diff changeset
135 Sources.isBeforeInTranslationUnit(CurrentToken.getLocation(), Loc)) {
anatofuz
parents:
diff changeset
136 PP->Lex(CurrentToken);
anatofuz
parents:
diff changeset
137 }
anatofuz
parents:
diff changeset
138 }
anatofuz
parents:
diff changeset
139
anatofuz
parents:
diff changeset
140 void ExpandModularHeadersPPCallbacks::FileChanged(
anatofuz
parents:
diff changeset
141 SourceLocation Loc, FileChangeReason Reason,
anatofuz
parents:
diff changeset
142 SrcMgr::CharacteristicKind FileType, FileID PrevFID = FileID()) {
anatofuz
parents:
diff changeset
143 if (!EnteredMainFile) {
anatofuz
parents:
diff changeset
144 EnteredMainFile = true;
anatofuz
parents:
diff changeset
145 PP->EnterMainSourceFile();
anatofuz
parents:
diff changeset
146 }
anatofuz
parents:
diff changeset
147 }
anatofuz
parents:
diff changeset
148
anatofuz
parents:
diff changeset
149 void ExpandModularHeadersPPCallbacks::InclusionDirective(
anatofuz
parents:
diff changeset
150 SourceLocation DirectiveLoc, const Token &IncludeToken,
anatofuz
parents:
diff changeset
151 StringRef IncludedFilename, bool IsAngled, CharSourceRange FilenameRange,
anatofuz
parents:
diff changeset
152 const FileEntry *IncludedFile, StringRef SearchPath, StringRef RelativePath,
anatofuz
parents:
diff changeset
153 const Module *Imported, SrcMgr::CharacteristicKind FileType) {
anatofuz
parents:
diff changeset
154 if (Imported) {
anatofuz
parents:
diff changeset
155 serialization::ModuleFile *MF =
anatofuz
parents:
diff changeset
156 Compiler.getASTReader()->getModuleManager().lookup(
anatofuz
parents:
diff changeset
157 Imported->getASTFile());
anatofuz
parents:
diff changeset
158 handleModuleFile(MF);
anatofuz
parents:
diff changeset
159 }
anatofuz
parents:
diff changeset
160 parseToLocation(DirectiveLoc);
anatofuz
parents:
diff changeset
161 }
anatofuz
parents:
diff changeset
162
anatofuz
parents:
diff changeset
163 void ExpandModularHeadersPPCallbacks::EndOfMainFile() {
anatofuz
parents:
diff changeset
164 while (!CurrentToken.is(tok::eof))
anatofuz
parents:
diff changeset
165 PP->Lex(CurrentToken);
anatofuz
parents:
diff changeset
166 }
anatofuz
parents:
diff changeset
167
anatofuz
parents:
diff changeset
168 // Handle all other callbacks.
anatofuz
parents:
diff changeset
169 // Just parse to the corresponding location to generate the same callback for
anatofuz
parents:
diff changeset
170 // the PPCallbacks registered in our custom preprocessor.
anatofuz
parents:
diff changeset
171 void ExpandModularHeadersPPCallbacks::Ident(SourceLocation Loc, StringRef) {
anatofuz
parents:
diff changeset
172 parseToLocation(Loc);
anatofuz
parents:
diff changeset
173 }
anatofuz
parents:
diff changeset
174 void ExpandModularHeadersPPCallbacks::PragmaDirective(SourceLocation Loc,
anatofuz
parents:
diff changeset
175 PragmaIntroducerKind) {
anatofuz
parents:
diff changeset
176 parseToLocation(Loc);
anatofuz
parents:
diff changeset
177 }
anatofuz
parents:
diff changeset
178 void ExpandModularHeadersPPCallbacks::PragmaComment(SourceLocation Loc,
anatofuz
parents:
diff changeset
179 const IdentifierInfo *,
anatofuz
parents:
diff changeset
180 StringRef) {
anatofuz
parents:
diff changeset
181 parseToLocation(Loc);
anatofuz
parents:
diff changeset
182 }
anatofuz
parents:
diff changeset
183 void ExpandModularHeadersPPCallbacks::PragmaDetectMismatch(SourceLocation Loc,
anatofuz
parents:
diff changeset
184 StringRef,
anatofuz
parents:
diff changeset
185 StringRef) {
anatofuz
parents:
diff changeset
186 parseToLocation(Loc);
anatofuz
parents:
diff changeset
187 }
anatofuz
parents:
diff changeset
188 void ExpandModularHeadersPPCallbacks::PragmaDebug(SourceLocation Loc,
anatofuz
parents:
diff changeset
189 StringRef) {
anatofuz
parents:
diff changeset
190 parseToLocation(Loc);
anatofuz
parents:
diff changeset
191 }
anatofuz
parents:
diff changeset
192 void ExpandModularHeadersPPCallbacks::PragmaMessage(SourceLocation Loc,
anatofuz
parents:
diff changeset
193 StringRef,
anatofuz
parents:
diff changeset
194 PragmaMessageKind,
anatofuz
parents:
diff changeset
195 StringRef) {
anatofuz
parents:
diff changeset
196 parseToLocation(Loc);
anatofuz
parents:
diff changeset
197 }
anatofuz
parents:
diff changeset
198 void ExpandModularHeadersPPCallbacks::PragmaDiagnosticPush(SourceLocation Loc,
anatofuz
parents:
diff changeset
199 StringRef) {
anatofuz
parents:
diff changeset
200 parseToLocation(Loc);
anatofuz
parents:
diff changeset
201 }
anatofuz
parents:
diff changeset
202 void ExpandModularHeadersPPCallbacks::PragmaDiagnosticPop(SourceLocation Loc,
anatofuz
parents:
diff changeset
203 StringRef) {
anatofuz
parents:
diff changeset
204 parseToLocation(Loc);
anatofuz
parents:
diff changeset
205 }
anatofuz
parents:
diff changeset
206 void ExpandModularHeadersPPCallbacks::PragmaDiagnostic(SourceLocation Loc,
anatofuz
parents:
diff changeset
207 StringRef,
anatofuz
parents:
diff changeset
208 diag::Severity,
anatofuz
parents:
diff changeset
209 StringRef) {
anatofuz
parents:
diff changeset
210 parseToLocation(Loc);
anatofuz
parents:
diff changeset
211 }
anatofuz
parents:
diff changeset
212 void ExpandModularHeadersPPCallbacks::HasInclude(SourceLocation Loc, StringRef,
anatofuz
parents:
diff changeset
213 bool, Optional<FileEntryRef>,
anatofuz
parents:
diff changeset
214 SrcMgr::CharacteristicKind) {
anatofuz
parents:
diff changeset
215 parseToLocation(Loc);
anatofuz
parents:
diff changeset
216 }
anatofuz
parents:
diff changeset
217 void ExpandModularHeadersPPCallbacks::PragmaOpenCLExtension(
anatofuz
parents:
diff changeset
218 SourceLocation NameLoc, const IdentifierInfo *, SourceLocation StateLoc,
anatofuz
parents:
diff changeset
219 unsigned) {
anatofuz
parents:
diff changeset
220 // FIME: Figure out whether it's the right location to parse to.
anatofuz
parents:
diff changeset
221 parseToLocation(NameLoc);
anatofuz
parents:
diff changeset
222 }
anatofuz
parents:
diff changeset
223 void ExpandModularHeadersPPCallbacks::PragmaWarning(SourceLocation Loc,
anatofuz
parents:
diff changeset
224 StringRef, ArrayRef<int>) {
anatofuz
parents:
diff changeset
225 parseToLocation(Loc);
anatofuz
parents:
diff changeset
226 }
anatofuz
parents:
diff changeset
227 void ExpandModularHeadersPPCallbacks::PragmaWarningPush(SourceLocation Loc,
anatofuz
parents:
diff changeset
228 int) {
anatofuz
parents:
diff changeset
229 parseToLocation(Loc);
anatofuz
parents:
diff changeset
230 }
anatofuz
parents:
diff changeset
231 void ExpandModularHeadersPPCallbacks::PragmaWarningPop(SourceLocation Loc) {
anatofuz
parents:
diff changeset
232 parseToLocation(Loc);
anatofuz
parents:
diff changeset
233 }
anatofuz
parents:
diff changeset
234 void ExpandModularHeadersPPCallbacks::PragmaAssumeNonNullBegin(
anatofuz
parents:
diff changeset
235 SourceLocation Loc) {
anatofuz
parents:
diff changeset
236 parseToLocation(Loc);
anatofuz
parents:
diff changeset
237 }
anatofuz
parents:
diff changeset
238 void ExpandModularHeadersPPCallbacks::PragmaAssumeNonNullEnd(
anatofuz
parents:
diff changeset
239 SourceLocation Loc) {
anatofuz
parents:
diff changeset
240 parseToLocation(Loc);
anatofuz
parents:
diff changeset
241 }
anatofuz
parents:
diff changeset
242 void ExpandModularHeadersPPCallbacks::MacroExpands(const Token &MacroNameTok,
anatofuz
parents:
diff changeset
243 const MacroDefinition &,
anatofuz
parents:
diff changeset
244 SourceRange Range,
anatofuz
parents:
diff changeset
245 const MacroArgs *) {
anatofuz
parents:
diff changeset
246 // FIME: Figure out whether it's the right location to parse to.
anatofuz
parents:
diff changeset
247 parseToLocation(Range.getBegin());
anatofuz
parents:
diff changeset
248 }
anatofuz
parents:
diff changeset
249 void ExpandModularHeadersPPCallbacks::MacroDefined(const Token &MacroNameTok,
anatofuz
parents:
diff changeset
250 const MacroDirective *MD) {
anatofuz
parents:
diff changeset
251 parseToLocation(MD->getLocation());
anatofuz
parents:
diff changeset
252 }
anatofuz
parents:
diff changeset
253 void ExpandModularHeadersPPCallbacks::MacroUndefined(
anatofuz
parents:
diff changeset
254 const Token &, const MacroDefinition &, const MacroDirective *Undef) {
anatofuz
parents:
diff changeset
255 if (Undef)
anatofuz
parents:
diff changeset
256 parseToLocation(Undef->getLocation());
anatofuz
parents:
diff changeset
257 }
anatofuz
parents:
diff changeset
258 void ExpandModularHeadersPPCallbacks::Defined(const Token &MacroNameTok,
anatofuz
parents:
diff changeset
259 const MacroDefinition &,
anatofuz
parents:
diff changeset
260 SourceRange Range) {
anatofuz
parents:
diff changeset
261 // FIME: Figure out whether it's the right location to parse to.
anatofuz
parents:
diff changeset
262 parseToLocation(Range.getBegin());
anatofuz
parents:
diff changeset
263 }
anatofuz
parents:
diff changeset
264 void ExpandModularHeadersPPCallbacks::SourceRangeSkipped(
anatofuz
parents:
diff changeset
265 SourceRange Range, SourceLocation EndifLoc) {
anatofuz
parents:
diff changeset
266 // FIME: Figure out whether it's the right location to parse to.
anatofuz
parents:
diff changeset
267 parseToLocation(EndifLoc);
anatofuz
parents:
diff changeset
268 }
anatofuz
parents:
diff changeset
269 void ExpandModularHeadersPPCallbacks::If(SourceLocation Loc, SourceRange,
anatofuz
parents:
diff changeset
270 ConditionValueKind) {
anatofuz
parents:
diff changeset
271 parseToLocation(Loc);
anatofuz
parents:
diff changeset
272 }
anatofuz
parents:
diff changeset
273 void ExpandModularHeadersPPCallbacks::Elif(SourceLocation Loc, SourceRange,
anatofuz
parents:
diff changeset
274 ConditionValueKind, SourceLocation) {
anatofuz
parents:
diff changeset
275 parseToLocation(Loc);
anatofuz
parents:
diff changeset
276 }
anatofuz
parents:
diff changeset
277 void ExpandModularHeadersPPCallbacks::Ifdef(SourceLocation Loc, const Token &,
anatofuz
parents:
diff changeset
278 const MacroDefinition &) {
anatofuz
parents:
diff changeset
279 parseToLocation(Loc);
anatofuz
parents:
diff changeset
280 }
anatofuz
parents:
diff changeset
281 void ExpandModularHeadersPPCallbacks::Ifndef(SourceLocation Loc, const Token &,
anatofuz
parents:
diff changeset
282 const MacroDefinition &) {
anatofuz
parents:
diff changeset
283 parseToLocation(Loc);
anatofuz
parents:
diff changeset
284 }
anatofuz
parents:
diff changeset
285 void ExpandModularHeadersPPCallbacks::Else(SourceLocation Loc, SourceLocation) {
anatofuz
parents:
diff changeset
286 parseToLocation(Loc);
anatofuz
parents:
diff changeset
287 }
anatofuz
parents:
diff changeset
288 void ExpandModularHeadersPPCallbacks::Endif(SourceLocation Loc,
anatofuz
parents:
diff changeset
289 SourceLocation) {
anatofuz
parents:
diff changeset
290 parseToLocation(Loc);
anatofuz
parents:
diff changeset
291 }
anatofuz
parents:
diff changeset
292
anatofuz
parents:
diff changeset
293 } // namespace tooling
anatofuz
parents:
diff changeset
294 } // namespace clang