clang 22.0.0git
FrontendAction.cpp
Go to the documentation of this file.
1//===--- FrontendAction.cpp -----------------------------------------------===//
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
12#include "clang/AST/Decl.h"
13#include "clang/AST/DeclGroup.h"
19#include "clang/Basic/Sarif.h"
22#include "clang/Basic/Stack.h"
42#include "llvm/ADT/ScopeExit.h"
43#include "llvm/ADT/SmallPtrSet.h"
44#include "llvm/ADT/StringRef.h"
45#include "llvm/Support/BuryPointer.h"
46#include "llvm/Support/ErrorHandling.h"
47#include "llvm/Support/FileSystem.h"
48#include "llvm/Support/Path.h"
49#include "llvm/Support/Timer.h"
50#include "llvm/Support/raw_ostream.h"
51#include <memory>
52#include <system_error>
53using namespace clang;
54
55LLVM_INSTANTIATE_REGISTRY(FrontendPluginRegistry)
56
57namespace {
58
59/// DeserializedDeclsLineRangePrinter dumps ranges of deserialized declarations
60/// to aid debugging and bug minimization. It implements ASTConsumer and
61/// ASTDeserializationListener, so that an object of
62/// DeserializedDeclsLineRangePrinter registers as its own listener. The
63/// ASTDeserializationListener interface provides the DeclRead callback that we
64/// use to collect the deserialized Decls. Note that printing or otherwise
65/// processing them as this point is dangerous, since that could trigger
66/// additional deserialization and crash compilation. Therefore, we process the
67/// collected Decls in HandleTranslationUnit method of ASTConsumer. This is a
68/// safe point, since we know that by this point all the Decls needed by the
69/// compiler frontend have been deserialized. In case our processing causes
70/// further deserialization, DeclRead from the listener might be called again.
71/// However, at that point we don't accept any more Decls for processing.
72class DeserializedDeclsSourceRangePrinter : public ASTConsumer,
74public:
75 explicit DeserializedDeclsSourceRangePrinter(
76 SourceManager &SM, std::unique_ptr<llvm::raw_fd_ostream> OS)
77 : ASTDeserializationListener(), SM(SM), OS(std::move(OS)) {}
78
79 ASTDeserializationListener *GetASTDeserializationListener() override {
80 return this;
81 }
82
83 void DeclRead(GlobalDeclID ID, const Decl *D) override {
84 if (!IsCollectingDecls)
85 return;
88 // These decls cover a lot of nested declarations that might not be used,
89 // reducing the granularity and making the output less useful.
90 return;
91 }
92 if (isa<ParmVarDecl>(D)) {
93 // Parameters are covered by their functions.
94 return;
95 }
96 auto *DC = D->getLexicalDeclContext();
97 if (!DC || !shouldIncludeDeclsIn(DC))
98 return;
99
100 PendingDecls.push_back(D);
101 for (; (isa<ExportDecl>(DC) || isa<NamespaceDecl>(DC)) &&
102 ProcessedDeclContexts.insert(DC).second;
103 DC = DC->getLexicalParent()) {
104 // Add any interesting decl contexts that we have not seen before.
105 // Note that we filter them out from DeclRead as that would include all
106 // redeclarations of namespaces, potentially those that do not have any
107 // imported declarations.
108 PendingDecls.push_back(cast<Decl>(DC));
109 }
110 }
111
112 struct Position {
113 unsigned Line;
114 unsigned Column;
115
116 bool operator<(const Position &other) const {
117 return std::tie(Line, Column) < std::tie(other.Line, other.Column);
118 }
119
120 static Position GetBeginSpelling(const SourceManager &SM,
121 const CharSourceRange &R) {
122 SourceLocation Begin = R.getBegin();
123 return {SM.getSpellingLineNumber(Begin),
124 SM.getSpellingColumnNumber(Begin)};
125 }
126
127 static Position GetEndSpelling(const SourceManager &SM,
128 const CharSourceRange &Range,
129 const LangOptions &LangOpts) {
130 // For token ranges, compute end location for end character of the range.
131 CharSourceRange R = Lexer::getAsCharRange(Range, SM, LangOpts);
132 SourceLocation End = R.getEnd();
133 // Relex the token past the end location of the last token in the source
134 // range. If it's a semicolon, advance the location by one token.
135 Token PossiblySemi;
136 Lexer::getRawToken(End, PossiblySemi, SM, LangOpts, true);
137 if (PossiblySemi.is(tok::semi))
138 End = End.getLocWithOffset(1);
139 // Column number of the returned end position is exclusive.
140 return {SM.getSpellingLineNumber(End), SM.getSpellingColumnNumber(End)};
141 }
142 };
143
144 struct RequiredRanges {
145 StringRef Filename;
146 std::vector<std::pair<Position, Position>> FromTo;
147 };
148 void HandleTranslationUnit(ASTContext &Context) override {
149 assert(IsCollectingDecls && "HandleTranslationUnit called twice?");
150 IsCollectingDecls = false;
151
152 // Merge ranges in each of the files.
153 struct FileData {
154 std::vector<std::pair<Position, Position>> FromTo;
156 };
157 llvm::DenseMap<const FileEntry *, FileData> FileToRanges;
158
159 for (const Decl *D : PendingDecls) {
160 for (CharSourceRange R : getRangesToMark(D)) {
161 if (!R.isValid())
162 continue;
163
164 auto *F = SM.getFileEntryForID(SM.getFileID(R.getBegin()));
165 if (F != SM.getFileEntryForID(SM.getFileID(R.getEnd()))) {
166 // Such cases are rare and difficult to handle.
167 continue;
168 }
169
170 auto &Data = FileToRanges[F];
171 if (!Data.Ref)
172 Data.Ref = SM.getFileEntryRefForID(SM.getFileID(R.getBegin()));
173 Data.FromTo.push_back(
174 {Position::GetBeginSpelling(SM, R),
175 Position::GetEndSpelling(SM, R, D->getLangOpts())});
176 }
177 }
178
179 // To simplify output, merge consecutive and intersecting ranges.
180 std::vector<RequiredRanges> Result;
181 for (auto &[F, Data] : FileToRanges) {
182 auto &FromTo = Data.FromTo;
183 assert(!FromTo.empty());
184
185 if (!Data.Ref)
186 continue;
187
188 llvm::sort(FromTo);
189
190 std::vector<std::pair<Position, Position>> MergedRanges;
191 MergedRanges.push_back(FromTo.front());
192 for (auto It = FromTo.begin() + 1; It < FromTo.end(); ++It) {
193 if (MergedRanges.back().second < It->first) {
194 MergedRanges.push_back(*It);
195 continue;
196 }
197 if (MergedRanges.back().second < It->second)
198 MergedRanges.back().second = It->second;
199 }
200 Result.push_back({Data.Ref->getName(), std::move(MergedRanges)});
201 }
202 printJson(Result);
203 }
204
205private:
206 std::vector<const Decl *> PendingDecls;
207 llvm::SmallPtrSet<const DeclContext *, 0> ProcessedDeclContexts;
208 bool IsCollectingDecls = true;
209 const SourceManager &SM;
210 std::unique_ptr<llvm::raw_ostream> OS;
211
212 static bool shouldIncludeDeclsIn(const DeclContext *DC) {
213 assert(DC && "DC is null");
214 // We choose to work at namespace level to reduce complexity and the number
215 // of cases we care about.
216 // We still need to carefully handle composite declarations like
217 // `ExportDecl`.
218 for (; DC; DC = DC->getLexicalParent()) {
219 if (DC->isFileContext())
220 return true;
221 if (isa<ExportDecl>(DC))
222 continue; // Depends on the parent.
223 return false;
224 }
225 llvm_unreachable("DeclContext chain must end with a translation unit");
226 }
227
228 llvm::SmallVector<CharSourceRange, 2> getRangesToMark(const Decl *D) {
229 if (auto *ED = dyn_cast<ExportDecl>(D)) {
230 if (!ED->hasBraces())
231 return {SM.getExpansionRange(ED->getExportLoc())};
232
233 return {SM.getExpansionRange(SourceRange(
234 ED->getExportLoc(),
235 lexForLBrace(ED->getExportLoc(), D->getLangOpts()))),
236 SM.getExpansionRange(ED->getRBraceLoc())};
237 }
238
239 auto *NS = dyn_cast<NamespaceDecl>(D);
240 if (!NS)
241 return {SM.getExpansionRange(D->getSourceRange())};
242
243 SourceLocation LBraceLoc;
244 if (NS->isAnonymousNamespace()) {
245 LBraceLoc = NS->getLocation();
246 } else {
247 // Start with the location of the identifier.
248 SourceLocation TokenBeforeLBrace = NS->getLocation();
249 if (NS->hasAttrs()) {
250 for (auto *A : NS->getAttrs()) {
251 // But attributes may go after it.
252 if (SM.isBeforeInTranslationUnit(TokenBeforeLBrace,
253 A->getRange().getEnd())) {
254 // Give up, the attributes are often coming from macros and we
255 // cannot skip them reliably.
256 return {};
257 }
258 }
259 }
260 LBraceLoc = lexForLBrace(TokenBeforeLBrace, D->getLangOpts());
261 }
262 return {SM.getExpansionRange(SourceRange(NS->getBeginLoc(), LBraceLoc)),
263 SM.getExpansionRange(NS->getRBraceLoc())};
264 }
265
266 void printJson(llvm::ArrayRef<RequiredRanges> Result) {
267 *OS << "{\n";
268 *OS << R"( "required_ranges": [)" << "\n";
269 for (size_t I = 0; I < Result.size(); ++I) {
270 auto &F = Result[I].Filename;
271 auto &MergedRanges = Result[I].FromTo;
272 *OS << R"( {)" << "\n";
273 *OS << R"( "file": ")" << F << "\"," << "\n";
274 *OS << R"( "range": [)" << "\n";
275 for (size_t J = 0; J < MergedRanges.size(); ++J) {
276 auto &From = MergedRanges[J].first;
277 auto &To = MergedRanges[J].second;
278 *OS << R"( {)" << "\n";
279 *OS << R"( "from": {)" << "\n";
280 *OS << R"( "line": )" << From.Line << ",\n";
281 *OS << R"( "column": )" << From.Column << "\n"
282 << R"( },)" << "\n";
283 *OS << R"( "to": {)" << "\n";
284 *OS << R"( "line": )" << To.Line << ",\n";
285 *OS << R"( "column": )" << To.Column << "\n"
286 << R"( })" << "\n";
287 *OS << R"( })";
288 if (J < MergedRanges.size() - 1) {
289 *OS << ",";
290 }
291 *OS << "\n";
292 }
293 *OS << " ]" << "\n" << " }";
294 if (I < Result.size() - 1)
295 *OS << ",";
296 *OS << "\n";
297 }
298 *OS << " ]\n";
299 *OS << "}\n";
300
301 OS->flush();
302 }
303
304 SourceLocation lexForLBrace(SourceLocation TokenBeforeLBrace,
305 const LangOptions &LangOpts) {
306 // Now skip one token, the next should be the lbrace.
307 Token Tok;
308 if (Lexer::getRawToken(TokenBeforeLBrace, Tok, SM, LangOpts, true) ||
309 Lexer::getRawToken(Tok.getEndLoc(), Tok, SM, LangOpts, true) ||
310 Tok.getKind() != tok::l_brace) {
311 // On error or if we did not find the token we expected, avoid marking
312 // everything inside the namespace as used.
313 return SourceLocation();
314 }
315 return Tok.getLocation();
316 }
317};
318
319/// Dumps deserialized declarations.
320class DeserializedDeclsDumper : public DelegatingDeserializationListener {
321public:
322 explicit DeserializedDeclsDumper(ASTDeserializationListener *Previous,
323 bool DeletePrevious)
324 : DelegatingDeserializationListener(Previous, DeletePrevious) {}
325
326 void DeclRead(GlobalDeclID ID, const Decl *D) override {
327 llvm::outs() << "PCH DECL: " << D->getDeclKindName();
328 if (const NamedDecl *ND = dyn_cast<NamedDecl>(D)) {
329 llvm::outs() << " - ";
330 ND->printQualifiedName(llvm::outs());
331 }
332 llvm::outs() << "\n";
333
335 }
336};
337
338/// Checks deserialized declarations and emits error if a name
339/// matches one given in command-line using -error-on-deserialized-decl.
340class DeserializedDeclsChecker : public DelegatingDeserializationListener {
341 ASTContext &Ctx;
342 std::set<std::string> NamesToCheck;
343
344public:
345 DeserializedDeclsChecker(ASTContext &Ctx,
346 const std::set<std::string> &NamesToCheck,
347 ASTDeserializationListener *Previous,
348 bool DeletePrevious)
349 : DelegatingDeserializationListener(Previous, DeletePrevious), Ctx(Ctx),
350 NamesToCheck(NamesToCheck) {}
351
352 void DeclRead(GlobalDeclID ID, const Decl *D) override {
353 if (const NamedDecl *ND = dyn_cast<NamedDecl>(D))
354 if (NamesToCheck.find(ND->getNameAsString()) != NamesToCheck.end()) {
355 unsigned DiagID
356 = Ctx.getDiagnostics().getCustomDiagID(DiagnosticsEngine::Error,
357 "%0 was deserialized");
358 Ctx.getDiagnostics().Report(Ctx.getFullLoc(D->getLocation()), DiagID)
359 << ND;
360 }
361
363 }
364};
365
366} // end anonymous namespace
367
369
371
373 std::unique_ptr<ASTUnit> AST) {
374 this->CurrentInput = CurrentInput;
375 CurrentASTUnit = std::move(AST);
376}
377
383
384std::unique_ptr<ASTConsumer>
385FrontendAction::CreateWrappedASTConsumer(CompilerInstance &CI,
386 StringRef InFile) {
387 std::unique_ptr<ASTConsumer> Consumer = CreateASTConsumer(CI, InFile);
388 if (!Consumer)
389 return nullptr;
390
391 std::vector<std::unique_ptr<ASTConsumer>> Consumers;
392 llvm::StringRef DumpDeserializedDeclarationRangesPath =
394 if (!DumpDeserializedDeclarationRangesPath.empty()) {
395 std::error_code ErrorCode;
396 auto FileStream = std::make_unique<llvm::raw_fd_ostream>(
397 DumpDeserializedDeclarationRangesPath, ErrorCode,
398 llvm::sys::fs::OF_TextWithCRLF);
399 if (!ErrorCode) {
400 Consumers.push_back(std::make_unique<DeserializedDeclsSourceRangePrinter>(
401 CI.getSourceManager(), std::move(FileStream)));
402 } else {
403 llvm::errs() << "Failed to create output file for "
404 "-dump-minimization-hints flag, file path: "
405 << DumpDeserializedDeclarationRangesPath
406 << ", error: " << ErrorCode.message() << "\n";
407 }
408 }
409
410 // Validate -add-plugin args.
411 bool FoundAllPlugins = true;
412 for (const std::string &Arg : CI.getFrontendOpts().AddPluginActions) {
413 bool Found = false;
414 for (const FrontendPluginRegistry::entry &Plugin :
415 FrontendPluginRegistry::entries()) {
416 if (Plugin.getName() == Arg)
417 Found = true;
418 }
419 if (!Found) {
420 CI.getDiagnostics().Report(diag::err_fe_invalid_plugin_name) << Arg;
421 FoundAllPlugins = false;
422 }
423 }
424 if (!FoundAllPlugins)
425 return nullptr;
426
427 // If this is a code completion run, avoid invoking the plugin consumers
429 return Consumer;
430
431 // Collect the list of plugins that go before the main action (in Consumers)
432 // or after it (in AfterConsumers)
433 std::vector<std::unique_ptr<ASTConsumer>> AfterConsumers;
434 for (const FrontendPluginRegistry::entry &Plugin :
435 FrontendPluginRegistry::entries()) {
436 std::unique_ptr<PluginASTAction> P = Plugin.instantiate();
437 PluginASTAction::ActionType ActionType = P->getActionType();
438 if (ActionType == PluginASTAction::CmdlineAfterMainAction ||
440 // This is O(|plugins| * |add_plugins|), but since both numbers are
441 // way below 50 in practice, that's ok.
442 if (llvm::is_contained(CI.getFrontendOpts().AddPluginActions,
443 Plugin.getName())) {
446 else
448 }
449 }
450 if ((ActionType == PluginASTAction::AddBeforeMainAction ||
452 P->ParseArgs(
453 CI,
454 CI.getFrontendOpts().PluginArgs[std::string(Plugin.getName())])) {
455 std::unique_ptr<ASTConsumer> PluginConsumer = P->CreateASTConsumer(CI, InFile);
456 if (ActionType == PluginASTAction::AddBeforeMainAction) {
457 Consumers.push_back(std::move(PluginConsumer));
458 } else {
459 AfterConsumers.push_back(std::move(PluginConsumer));
460 }
461 }
462 }
463
464 // Add to Consumers the main consumer, then all the plugins that go after it
465 Consumers.push_back(std::move(Consumer));
466 if (!AfterConsumers.empty()) {
467 // If we have plugins after the main consumer, which may be the codegen
468 // action, they likely will need the ASTContext, so don't clear it in the
469 // codegen action.
470 CI.getCodeGenOpts().ClearASTBeforeBackend = false;
471 for (auto &C : AfterConsumers)
472 Consumers.push_back(std::move(C));
473 }
474
475 assert(Consumers.size() >= 1 && "should have added the main consumer");
476 if (Consumers.size() == 1)
477 return std::move(Consumers.front());
478 return std::make_unique<MultiplexConsumer>(std::move(Consumers));
479}
480
481/// For preprocessed files, if the first line is the linemarker and specifies
482/// the original source file name, use that name as the input file name.
483/// Returns the location of the first token after the line marker directive.
484///
485/// \param CI The compiler instance.
486/// \param InputFile Populated with the filename from the line marker.
487/// \param IsModuleMap If \c true, add a line note corresponding to this line
488/// directive. (We need to do this because the directive will not be
489/// visited by the preprocessor.)
491 std::string &InputFile,
492 bool IsModuleMap = false) {
493 auto &SourceMgr = CI.getSourceManager();
494 auto MainFileID = SourceMgr.getMainFileID();
495
496 auto MainFileBuf = SourceMgr.getBufferOrNone(MainFileID);
497 if (!MainFileBuf)
498 return SourceLocation();
499
500 std::unique_ptr<Lexer> RawLexer(
501 new Lexer(MainFileID, *MainFileBuf, SourceMgr, CI.getLangOpts()));
502
503 // If the first line has the syntax of
504 //
505 // # NUM "FILENAME"
506 //
507 // we use FILENAME as the input file name.
508 Token T;
509 if (RawLexer->LexFromRawLexer(T) || T.getKind() != tok::hash)
510 return SourceLocation();
511 if (RawLexer->LexFromRawLexer(T) || T.isAtStartOfLine() ||
512 T.getKind() != tok::numeric_constant)
513 return SourceLocation();
514
515 unsigned LineNo;
516 SourceLocation LineNoLoc = T.getLocation();
517 if (IsModuleMap) {
519 if (Lexer::getSpelling(LineNoLoc, Buffer, SourceMgr, CI.getLangOpts())
520 .getAsInteger(10, LineNo))
521 return SourceLocation();
522 }
523
524 RawLexer->LexFromRawLexer(T);
525 if (T.isAtStartOfLine() || T.getKind() != tok::string_literal)
526 return SourceLocation();
527
529 if (Literal.hadError)
530 return SourceLocation();
531 RawLexer->LexFromRawLexer(T);
532 if (T.isNot(tok::eof) && !T.isAtStartOfLine())
533 return SourceLocation();
534 InputFile = Literal.GetString().str();
535
536 if (IsModuleMap)
538 LineNoLoc, LineNo, SourceMgr.getLineTableFilenameID(InputFile), false,
540
541 return T.getLocation();
542}
543
545operator+=(SmallVectorImpl<char> &Includes, StringRef RHS) {
546 Includes.append(RHS.begin(), RHS.end());
547 return Includes;
548}
549
550static void addHeaderInclude(StringRef HeaderName,
551 SmallVectorImpl<char> &Includes,
552 const LangOptions &LangOpts,
553 bool IsExternC) {
554 if (IsExternC && LangOpts.CPlusPlus)
555 Includes += "extern \"C\" {\n";
556 if (LangOpts.ObjC)
557 Includes += "#import \"";
558 else
559 Includes += "#include \"";
560
561 Includes += HeaderName;
562
563 Includes += "\"\n";
564 if (IsExternC && LangOpts.CPlusPlus)
565 Includes += "}\n";
566}
567
568/// Collect the set of header includes needed to construct the given
569/// module and update the TopHeaders file set of the module.
570///
571/// \param Module The module we're collecting includes from.
572///
573/// \param Includes Will be augmented with the set of \#includes or \#imports
574/// needed to load all of the named headers.
575static std::error_code collectModuleHeaderIncludes(
578 // Don't collect any headers for unavailable modules.
579 if (!Module->isAvailable())
580 return std::error_code();
581
582 // Resolve all lazy header directives to header files.
583 ModMap.resolveHeaderDirectives(Module, /*File=*/std::nullopt);
584
585 // If any headers are missing, we can't build this module. In most cases,
586 // diagnostics for this should have already been produced; we only get here
587 // if explicit stat information was provided.
588 // FIXME: If the name resolves to a file with different stat information,
589 // produce a better diagnostic.
590 if (!Module->MissingHeaders.empty()) {
591 auto &MissingHeader = Module->MissingHeaders.front();
592 Diag.Report(MissingHeader.FileNameLoc, diag::err_module_header_missing)
593 << MissingHeader.IsUmbrella << MissingHeader.FileName;
594 return std::error_code();
595 }
596
597 // Add includes for each of these headers.
598 for (auto HK : {Module::HK_Normal, Module::HK_Private}) {
599 for (const Module::Header &H : Module->getHeaders(HK)) {
600 Module->addTopHeader(H.Entry);
601 // Use the path as specified in the module map file. We'll look for this
602 // file relative to the module build directory (the directory containing
603 // the module map file) so this will find the same file that we found
604 // while parsing the module map.
605 addHeaderInclude(H.PathRelativeToRootModuleDirectory, Includes, LangOpts,
607 }
608 }
609 // Note that Module->PrivateHeaders will not be a TopHeader.
610
611 if (std::optional<Module::Header> UmbrellaHeader =
613 Module->addTopHeader(UmbrellaHeader->Entry);
614 if (Module->Parent)
615 // Include the umbrella header for submodules.
616 addHeaderInclude(UmbrellaHeader->PathRelativeToRootModuleDirectory,
617 Includes, LangOpts, Module->IsExternC);
618 } else if (std::optional<Module::DirectoryName> UmbrellaDir =
620 // Add all of the headers we find in this subdirectory.
621 std::error_code EC;
622 SmallString<128> DirNative;
623 llvm::sys::path::native(UmbrellaDir->Entry.getName(), DirNative);
624
625 llvm::vfs::FileSystem &FS = FileMgr.getVirtualFileSystem();
627 for (llvm::vfs::recursive_directory_iterator Dir(FS, DirNative, EC), End;
628 Dir != End && !EC; Dir.increment(EC)) {
629 // Check whether this entry has an extension typically associated with
630 // headers.
631 if (!llvm::StringSwitch<bool>(llvm::sys::path::extension(Dir->path()))
632 .Cases(".h", ".H", ".hh", ".hpp", true)
633 .Default(false))
634 continue;
635
636 // Compute the relative path from the directory to this file.
638 auto PathIt = llvm::sys::path::rbegin(Dir->path());
639 for (int I = 0; I != Dir.level() + 1; ++I, ++PathIt)
640 Components.push_back(*PathIt);
641 SmallString<128> RelativeHeader(
642 UmbrellaDir->PathRelativeToRootModuleDirectory);
643 for (auto It = Components.rbegin(), End = Components.rend(); It != End;
644 ++It)
645 llvm::sys::path::append(RelativeHeader, *It);
646
647 HeaderPaths.push_back(
648 std::make_pair(Dir->path().str(), RelativeHeader.c_str()));
649 }
650
651 if (EC)
652 return EC;
653
654 // Sort header paths and make the header inclusion order deterministic
655 // across different OSs and filesystems. As the header search table
656 // serialization order depends on the file reference UID, we need to create
657 // file references in deterministic order too.
658 llvm::sort(HeaderPaths, llvm::less_first());
659 for (auto &[Path, RelPath] : HeaderPaths) {
660 auto Header = FileMgr.getOptionalFileRef(Path);
661 // FIXME: This shouldn't happen unless there is a file system race. Is
662 // that worth diagnosing?
663 if (!Header)
664 continue;
665
666 // If this header is marked 'unavailable' in this module, don't include
667 // it.
668 if (ModMap.isHeaderUnavailableInModule(*Header, Module))
669 continue;
670
671 // Include this header as part of the umbrella directory.
672 Module->addTopHeader(*Header);
673 addHeaderInclude(RelPath, Includes, LangOpts, Module->IsExternC);
674 }
675 }
676
677 // Recurse into submodules.
678 for (auto *Submodule : Module->submodules())
679 if (std::error_code Err = collectModuleHeaderIncludes(
680 LangOpts, FileMgr, Diag, ModMap, Submodule, Includes))
681 return Err;
682
683 return std::error_code();
684}
685
686static bool loadModuleMapForModuleBuild(CompilerInstance &CI, bool IsSystem,
687 bool IsPreprocessed,
688 std::string &PresumedModuleMapFile,
689 unsigned &Offset) {
690 auto &SrcMgr = CI.getSourceManager();
692
693 // Map the current input to a file.
694 FileID ModuleMapID = SrcMgr.getMainFileID();
695 OptionalFileEntryRef ModuleMap = SrcMgr.getFileEntryRefForID(ModuleMapID);
696 assert(ModuleMap && "MainFileID without FileEntry");
697
698 // If the module map is preprocessed, handle the initial line marker;
699 // line directives are not part of the module map syntax in general.
700 Offset = 0;
701 if (IsPreprocessed) {
702 SourceLocation EndOfLineMarker =
703 ReadOriginalFileName(CI, PresumedModuleMapFile, /*IsModuleMap*/ true);
704 if (EndOfLineMarker.isValid())
705 Offset = CI.getSourceManager().getDecomposedLoc(EndOfLineMarker).second;
706 }
707
708 // Load the module map file.
709 if (HS.parseAndLoadModuleMapFile(*ModuleMap, IsSystem, ModuleMapID, &Offset,
710 PresumedModuleMapFile))
711 return true;
712
713 if (SrcMgr.getBufferOrFake(ModuleMapID).getBufferSize() == Offset)
714 Offset = 0;
715
716 // Infer framework module if possible.
717 if (HS.getModuleMap().canInferFrameworkModule(ModuleMap->getDir())) {
718 SmallString<128> InferredFrameworkPath = ModuleMap->getDir().getName();
719 llvm::sys::path::append(InferredFrameworkPath,
720 CI.getLangOpts().ModuleName + ".framework");
721 if (auto Dir =
722 CI.getFileManager().getOptionalDirectoryRef(InferredFrameworkPath))
723 (void)HS.getModuleMap().inferFrameworkModule(*Dir, IsSystem, nullptr);
724 }
725
726 return false;
727}
728
730 StringRef ModuleMapFilename) {
731 if (CI.getLangOpts().CurrentModule.empty()) {
732 CI.getDiagnostics().Report(diag::err_missing_module_name);
733
734 // FIXME: Eventually, we could consider asking whether there was just
735 // a single module described in the module map, and use that as a
736 // default. Then it would be fairly trivial to just "compile" a module
737 // map with a single module (the common case).
738 return nullptr;
739 }
740
741 // Dig out the module definition.
744 /*AllowSearch=*/true);
745 if (!M) {
746 CI.getDiagnostics().Report(diag::err_missing_module)
747 << CI.getLangOpts().CurrentModule << ModuleMapFilename;
748
749 return nullptr;
750 }
751
752 // Check whether we can build this module at all.
754 CI.getDiagnostics()))
755 return nullptr;
756
757 // Inform the preprocessor that includes from within the input buffer should
758 // be resolved relative to the build directory of the module map file.
760
761 // If the module was inferred from a different module map (via an expanded
762 // umbrella module definition), track that fact.
763 // FIXME: It would be preferable to fill this in as part of processing
764 // the module map, rather than adding it after the fact.
765 StringRef OriginalModuleMapName = CI.getFrontendOpts().OriginalModuleMap;
766 if (!OriginalModuleMapName.empty()) {
767 auto OriginalModuleMap =
768 CI.getFileManager().getOptionalFileRef(OriginalModuleMapName,
769 /*openFile*/ true);
770 if (!OriginalModuleMap) {
771 CI.getDiagnostics().Report(diag::err_module_map_not_found)
772 << OriginalModuleMapName;
773 return nullptr;
774 }
775 if (*OriginalModuleMap != CI.getSourceManager().getFileEntryRefForID(
777 auto FileCharacter =
779 FileID OriginalModuleMapFID = CI.getSourceManager().getOrCreateFileID(
780 *OriginalModuleMap, FileCharacter);
781 CI.getPreprocessor()
783 .getModuleMap()
784 .setInferredModuleAllowedBy(M, OriginalModuleMapFID);
785 }
786 }
787
788 // If we're being run from the command-line, the module build stack will not
789 // have been filled in yet, so complete it now in order to allow us to detect
790 // module cycles.
791 SourceManager &SourceMgr = CI.getSourceManager();
792 if (SourceMgr.getModuleBuildStack().empty())
794 FullSourceLoc(SourceLocation(), SourceMgr));
795 return M;
796}
797
798/// Compute the input buffer that should be used to build the specified module.
799static std::unique_ptr<llvm::MemoryBuffer>
802
803 // Collect the set of #includes we need to build the module.
804 SmallString<256> HeaderContents;
805 std::error_code Err = std::error_code();
806 if (std::optional<Module::Header> UmbrellaHeader =
808 addHeaderInclude(UmbrellaHeader->PathRelativeToRootModuleDirectory,
809 HeaderContents, CI.getLangOpts(), M->IsExternC);
813 HeaderContents);
814
815 if (Err) {
816 CI.getDiagnostics().Report(diag::err_module_cannot_create_includes)
817 << M->getFullModuleName() << Err.message();
818 return nullptr;
819 }
820
821 return llvm::MemoryBuffer::getMemBufferCopy(
822 HeaderContents, Module::getModuleInputBufferName());
823}
824
826 const FrontendInputFile &RealInput) {
827 FrontendInputFile Input(RealInput);
828 assert(!Instance && "Already processing a source file!");
829 assert(!Input.isEmpty() && "Unexpected empty filename!");
830 setCurrentInput(Input);
832
833 bool HasBegunSourceFile = false;
834 bool ReplayASTFile = Input.getKind().getFormat() == InputKind::Precompiled &&
836
837 // If we fail, reset state since the client will not end up calling the
838 // matching EndSourceFile(). All paths that return true should release this.
839 auto FailureCleanup = llvm::make_scope_exit([&]() {
840 if (HasBegunSourceFile)
842 CI.setASTConsumer(nullptr);
843 CI.clearOutputFiles(/*EraseFiles=*/true);
844 CI.getLangOpts().setCompilingModule(LangOptions::CMK_None);
846 setCompilerInstance(nullptr);
847 });
848
849 if (!BeginInvocation(CI))
850 return false;
851
852 // If we're replaying the build of an AST file, import it and set up
853 // the initial state from its build.
854 if (ReplayASTFile) {
856
857 // The AST unit populates its own diagnostics engine rather than ours.
858 auto ASTDiags = llvm::makeIntrusiveRefCnt<DiagnosticsEngine>(
859 Diags->getDiagnosticIDs(), Diags->getDiagnosticOptions());
860 ASTDiags->setClient(Diags->getClient(), /*OwnsClient*/false);
861
862 // FIXME: What if the input is a memory buffer?
863 StringRef InputFile = Input.getFile();
864
865 std::unique_ptr<ASTUnit> AST = ASTUnit::LoadFromASTFile(
867 nullptr, ASTDiags, CI.getFileSystemOpts(), CI.getHeaderSearchOpts());
868 if (!AST)
869 return false;
870
871 // Options relating to how we treat the input (but not what we do with it)
872 // are inherited from the AST unit.
873 CI.getHeaderSearchOpts() = AST->getHeaderSearchOpts();
874 CI.getPreprocessorOpts() = AST->getPreprocessorOpts();
875 CI.getLangOpts() = AST->getLangOpts();
876
877 // Set the shared objects, these are reset when we finish processing the
878 // file, otherwise the CompilerInstance will happily destroy them.
879 CI.setVirtualFileSystem(AST->getFileManager().getVirtualFileSystemPtr());
880 CI.setFileManager(AST->getFileManagerPtr());
882 CI.getSourceManager().initializeForReplay(AST->getSourceManager());
883
884 // Preload all the module files loaded transitively by the AST unit. Also
885 // load all module map files that were parsed as part of building the AST
886 // unit.
887 if (auto ASTReader = AST->getASTReader()) {
888 auto &MM = ASTReader->getModuleManager();
889 auto &PrimaryModule = MM.getPrimaryModule();
890
891 for (serialization::ModuleFile &MF : MM)
892 if (&MF != &PrimaryModule)
893 CI.getFrontendOpts().ModuleFiles.push_back(MF.FileName);
894
895 ASTReader->visitTopLevelModuleMaps(PrimaryModule, [&](FileEntryRef FE) {
896 CI.getFrontendOpts().ModuleMapFiles.push_back(
897 std::string(FE.getName()));
898 });
899 }
900
901 // Set up the input file for replay purposes.
902 auto Kind = AST->getInputKind();
903 if (Kind.getFormat() == InputKind::ModuleMap) {
904 Module *ASTModule =
905 AST->getPreprocessor().getHeaderSearchInfo().lookupModule(
906 AST->getLangOpts().CurrentModule, SourceLocation(),
907 /*AllowSearch*/ false);
908 assert(ASTModule && "module file does not define its own module");
909 Input = FrontendInputFile(ASTModule->PresumedModuleMapFile, Kind);
910 } else {
911 auto &OldSM = AST->getSourceManager();
912 FileID ID = OldSM.getMainFileID();
913 if (auto File = OldSM.getFileEntryRefForID(ID))
914 Input = FrontendInputFile(File->getName(), Kind);
915 else
916 Input = FrontendInputFile(OldSM.getBufferOrFake(ID), Kind);
917 }
918 setCurrentInput(Input, std::move(AST));
919 }
920
921 // AST files follow a very different path, since they share objects via the
922 // AST unit.
923 if (Input.getKind().getFormat() == InputKind::Precompiled) {
924 assert(!usesPreprocessorOnly() && "this case was handled above");
925 assert(hasASTFileSupport() &&
926 "This action does not have AST file support!");
927
929
930 // FIXME: What if the input is a memory buffer?
931 StringRef InputFile = Input.getFile();
932
933 std::unique_ptr<ASTUnit> AST = ASTUnit::LoadFromASTFile(
934 InputFile, CI.getPCHContainerReader(), ASTUnit::LoadEverything, nullptr,
935 Diags, CI.getFileSystemOpts(), CI.getHeaderSearchOpts(),
936 &CI.getLangOpts());
937
938 if (!AST)
939 return false;
940
941 // Inform the diagnostic client we are processing a source file.
943 HasBegunSourceFile = true;
944
945 // Set the shared objects, these are reset when we finish processing the
946 // file, otherwise the CompilerInstance will happily destroy them.
947 CI.setFileManager(AST->getFileManagerPtr());
948 CI.setSourceManager(AST->getSourceManagerPtr());
949 CI.setPreprocessor(AST->getPreprocessorPtr());
950 Preprocessor &PP = CI.getPreprocessor();
952 PP.getLangOpts());
953 CI.setASTContext(AST->getASTContextPtr());
954
955 setCurrentInput(Input, std::move(AST));
956
957 // Initialize the action.
958 if (!BeginSourceFileAction(CI))
959 return false;
960
961 // Create the AST consumer.
962 CI.setASTConsumer(CreateWrappedASTConsumer(CI, InputFile));
963 if (!CI.hasASTConsumer())
964 return false;
965
966 FailureCleanup.release();
967 return true;
968 }
969
970 // Set up the file system, file and source managers, if needed.
971 if (!CI.hasVirtualFileSystem())
973 if (!CI.hasFileManager()) {
974 if (!CI.createFileManager()) {
975 return false;
976 }
977 }
978 if (!CI.hasSourceManager()) {
980 if (CI.getDiagnosticOpts().getFormat() == DiagnosticOptions::SARIF) {
981 static_cast<SARIFDiagnosticPrinter *>(&CI.getDiagnosticClient())
983 std::make_unique<SarifDocumentWriter>(CI.getSourceManager()));
984 }
985 }
986
987 // Set up embedding for any specified files. Do this before we load any
988 // source files, including the primary module map for the compilation.
989 for (const auto &F : CI.getFrontendOpts().ModulesEmbedFiles) {
990 if (auto FE = CI.getFileManager().getOptionalFileRef(F, /*openFile*/true))
992 else
993 CI.getDiagnostics().Report(diag::err_modules_embed_file_not_found) << F;
994 }
997
998 // IR files bypass the rest of initialization.
999 if (Input.getKind().getLanguage() == Language::LLVM_IR) {
1000 if (!hasIRSupport()) {
1001 CI.getDiagnostics().Report(diag::err_ast_action_on_llvm_ir)
1002 << Input.getFile();
1003 return false;
1004 }
1005
1006 // Inform the diagnostic client we are processing a source file.
1008 HasBegunSourceFile = true;
1009
1010 // Initialize the action.
1011 if (!BeginSourceFileAction(CI))
1012 return false;
1013
1014 // Initialize the main file entry.
1015 if (!CI.InitializeSourceManager(CurrentInput))
1016 return false;
1017
1018 FailureCleanup.release();
1019 return true;
1020 }
1021
1022 // If the implicit PCH include is actually a directory, rather than
1023 // a single file, search for a suitable PCH file in that directory.
1024 if (!CI.getPreprocessorOpts().ImplicitPCHInclude.empty()) {
1027 StringRef PCHInclude = PPOpts.ImplicitPCHInclude;
1028 std::string SpecificModuleCachePath = CI.getSpecificModuleCachePath();
1029 if (auto PCHDir = FileMgr.getOptionalDirectoryRef(PCHInclude)) {
1030 std::error_code EC;
1031 SmallString<128> DirNative;
1032 llvm::sys::path::native(PCHDir->getName(), DirNative);
1033 bool Found = false;
1034 llvm::vfs::FileSystem &FS = FileMgr.getVirtualFileSystem();
1035 for (llvm::vfs::directory_iterator Dir = FS.dir_begin(DirNative, EC),
1036 DirEnd;
1037 Dir != DirEnd && !EC; Dir.increment(EC)) {
1038 // Check whether this is an acceptable AST file.
1040 Dir->path(), FileMgr, CI.getModuleCache(),
1042 CI.getCodeGenOpts(), CI.getTargetOpts(),
1043 CI.getPreprocessorOpts(), SpecificModuleCachePath,
1044 /*RequireStrictOptionMatches=*/true)) {
1045 PPOpts.ImplicitPCHInclude = std::string(Dir->path());
1046 Found = true;
1047 break;
1048 }
1049 }
1050
1051 if (!Found) {
1052 CI.getDiagnostics().Report(diag::err_fe_no_pch_in_dir) << PCHInclude;
1053 return false;
1054 }
1055 }
1056 }
1057
1058 // Set up the preprocessor if needed. When parsing model files the
1059 // preprocessor of the original source is reused.
1060 if (!isModelParsingAction())
1062
1063 // Inform the diagnostic client we are processing a source file.
1065 &CI.getPreprocessor());
1066 HasBegunSourceFile = true;
1067
1068 // Handle C++20 header units.
1069 // Here, the user has the option to specify that the header name should be
1070 // looked up in the pre-processor search paths (and the main filename as
1071 // passed by the driver might therefore be incomplete until that look-up).
1072 if (CI.getLangOpts().CPlusPlusModules && Input.getKind().isHeaderUnit() &&
1073 !Input.getKind().isPreprocessed()) {
1074 StringRef FileName = Input.getFile();
1075 InputKind Kind = Input.getKind();
1076 if (Kind.getHeaderUnitKind() != InputKind::HeaderUnit_Abs) {
1077 assert(CI.hasPreprocessor() &&
1078 "trying to build a header unit without a Pre-processor?");
1080 // Relative searches begin from CWD.
1081 auto Dir = CI.getFileManager().getOptionalDirectoryRef(".");
1083 CWD.push_back({std::nullopt, *Dir});
1086 /*Angled*/ Input.getKind().getHeaderUnitKind() ==
1088 nullptr, nullptr, CWD, nullptr, nullptr, nullptr,
1089 nullptr, nullptr, nullptr);
1090 if (!FE) {
1091 CI.getDiagnostics().Report(diag::err_module_header_file_not_found)
1092 << FileName;
1093 return false;
1094 }
1095 // We now have the filename...
1096 FileName = FE->getName();
1097 // ... still a header unit, but now use the path as written.
1099 Input = FrontendInputFile(FileName, Kind, Input.isSystem());
1100 }
1101 // Unless the user has overridden the name, the header unit module name is
1102 // the pathname for the file.
1103 if (CI.getLangOpts().ModuleName.empty())
1104 CI.getLangOpts().ModuleName = std::string(FileName);
1106 }
1107
1108 if (!CI.InitializeSourceManager(Input))
1109 return false;
1110
1111 if (CI.getLangOpts().CPlusPlusModules && Input.getKind().isHeaderUnit() &&
1113 // We have an input filename like foo.iih, but we want to find the right
1114 // module name (and original file, to build the map entry).
1115 // Check if the first line specifies the original source file name with a
1116 // linemarker.
1117 std::string PresumedInputFile = std::string(getCurrentFileOrBufferName());
1118 ReadOriginalFileName(CI, PresumedInputFile);
1119 // Unless the user overrides this, the module name is the name by which the
1120 // original file was known.
1121 if (CI.getLangOpts().ModuleName.empty())
1122 CI.getLangOpts().ModuleName = std::string(PresumedInputFile);
1124 }
1125
1126 // For module map files, we first parse the module map and synthesize a
1127 // "<module-includes>" buffer before more conventional processing.
1128 if (Input.getKind().getFormat() == InputKind::ModuleMap) {
1129 CI.getLangOpts().setCompilingModule(LangOptions::CMK_ModuleMap);
1130
1131 std::string PresumedModuleMapFile;
1132 unsigned OffsetToContents;
1133 if (loadModuleMapForModuleBuild(CI, Input.isSystem(),
1134 Input.isPreprocessed(),
1135 PresumedModuleMapFile, OffsetToContents))
1136 return false;
1137
1138 auto *CurrentModule = prepareToBuildModule(CI, Input.getFile());
1139 if (!CurrentModule)
1140 return false;
1141
1142 CurrentModule->PresumedModuleMapFile = PresumedModuleMapFile;
1143
1144 if (OffsetToContents)
1145 // If the module contents are in the same file, skip to them.
1146 CI.getPreprocessor().setSkipMainFilePreamble(OffsetToContents, true);
1147 else {
1148 // Otherwise, convert the module description to a suitable input buffer.
1149 auto Buffer = getInputBufferForModule(CI, CurrentModule);
1150 if (!Buffer)
1151 return false;
1152
1153 // Reinitialize the main file entry to refer to the new input.
1154 auto Kind = CurrentModule->IsSystem ? SrcMgr::C_System : SrcMgr::C_User;
1155 auto &SourceMgr = CI.getSourceManager();
1156 auto BufferID = SourceMgr.createFileID(std::move(Buffer), Kind);
1157 assert(BufferID.isValid() && "couldn't create module buffer ID");
1158 SourceMgr.setMainFileID(BufferID);
1159 }
1160 }
1161
1162 // Initialize the action.
1163 if (!BeginSourceFileAction(CI))
1164 return false;
1165
1166 // If we were asked to load any module map files, do so now.
1167 for (const auto &Filename : CI.getFrontendOpts().ModuleMapFiles) {
1168 if (auto File = CI.getFileManager().getOptionalFileRef(Filename))
1170 *File, /*IsSystem*/ false);
1171 else
1172 CI.getDiagnostics().Report(diag::err_module_map_not_found) << Filename;
1173 }
1174
1175 // If compiling implementation of a module, load its module map file now.
1177
1178 // Add a module declaration scope so that modules from -fmodule-map-file
1179 // arguments may shadow modules found implicitly in search paths.
1180 CI.getPreprocessor()
1182 .getModuleMap()
1184
1185 // Create the AST context and consumer unless this is a preprocessor only
1186 // action.
1187 if (!usesPreprocessorOnly()) {
1188 // Parsing a model file should reuse the existing ASTContext.
1189 if (!isModelParsingAction())
1190 CI.createASTContext();
1191
1192 // For preprocessed files, check if the first line specifies the original
1193 // source file name with a linemarker.
1194 std::string PresumedInputFile = std::string(getCurrentFileOrBufferName());
1195 if (Input.isPreprocessed())
1196 ReadOriginalFileName(CI, PresumedInputFile);
1197
1198 std::unique_ptr<ASTConsumer> Consumer =
1199 CreateWrappedASTConsumer(CI, PresumedInputFile);
1200 if (!Consumer)
1201 return false;
1202
1203 // FIXME: should not overwrite ASTMutationListener when parsing model files?
1204 if (!isModelParsingAction())
1205 CI.getASTContext().setASTMutationListener(Consumer->GetASTMutationListener());
1206
1207 if (!CI.getPreprocessorOpts().ChainedIncludes.empty()) {
1208 // Convert headers to PCH and chain them.
1211 source = createChainedIncludesSource(CI, FinalReader);
1212 if (!source)
1213 return false;
1214 CI.setASTReader(FinalReader);
1215 CI.getASTContext().setExternalSource(source);
1216 } else if (CI.getLangOpts().Modules ||
1218 // Use PCM or PCH.
1219 assert(hasPCHSupport() && "This action does not have PCH support!");
1220 ASTDeserializationListener *DeserialListener =
1221 Consumer->GetASTDeserializationListener();
1222 bool DeleteDeserialListener = false;
1224 DeserialListener = new DeserializedDeclsDumper(DeserialListener,
1225 DeleteDeserialListener);
1226 DeleteDeserialListener = true;
1227 }
1229 DeserialListener = new DeserializedDeclsChecker(
1230 CI.getASTContext(),
1232 DeserialListener, DeleteDeserialListener);
1233 DeleteDeserialListener = true;
1234 }
1235 if (!CI.getPreprocessorOpts().ImplicitPCHInclude.empty()) {
1240 DeserialListener, DeleteDeserialListener);
1241 if (!CI.getASTContext().getExternalSource())
1242 return false;
1243 }
1244 // If modules are enabled, create the AST reader before creating
1245 // any builtins, so that all declarations know that they might be
1246 // extended by an external source.
1247 if (CI.getLangOpts().Modules || !CI.hasASTContext() ||
1249 CI.createASTReader();
1250 CI.getASTReader()->setDeserializationListener(DeserialListener,
1251 DeleteDeserialListener);
1252 }
1253 }
1254
1255 CI.setASTConsumer(std::move(Consumer));
1256 if (!CI.hasASTConsumer())
1257 return false;
1258 }
1259
1260 // Initialize built-in info as long as we aren't using an external AST
1261 // source.
1262 if (CI.getLangOpts().Modules || !CI.hasASTContext() ||
1264 Preprocessor &PP = CI.getPreprocessor();
1266 PP.getLangOpts());
1267 } else {
1268 // FIXME: If this is a problem, recover from it by creating a multiplex
1269 // source.
1270 assert((!CI.getLangOpts().Modules || CI.getASTReader()) &&
1271 "modules enabled but created an external source that "
1272 "doesn't support modules");
1273 }
1274
1275 // If we were asked to load any module files, do so now.
1276 for (const auto &ModuleFile : CI.getFrontendOpts().ModuleFiles) {
1277 serialization::ModuleFile *Loaded = nullptr;
1278 if (!CI.loadModuleFile(ModuleFile, Loaded))
1279 return false;
1280
1281 if (Loaded && Loaded->StandardCXXModule)
1283 diag::warn_eagerly_load_for_standard_cplusplus_modules);
1284 }
1285
1286 // If there is a layout overrides file, attach an external AST source that
1287 // provides the layouts from that file.
1288 if (!CI.getFrontendOpts().OverrideRecordLayoutsFile.empty() &&
1290 auto Override = llvm::makeIntrusiveRefCnt<LayoutOverrideSource>(
1293 }
1294
1295 // Setup HLSL External Sema Source
1296 if (CI.getLangOpts().HLSL && CI.hasASTContext()) {
1297 auto HLSLSema = llvm::makeIntrusiveRefCnt<HLSLExternalSemaSource>();
1298 if (auto SemaSource = dyn_cast_if_present<ExternalSemaSource>(
1300 auto MultiSema = llvm::makeIntrusiveRefCnt<MultiplexExternalSemaSource>(
1301 std::move(SemaSource), std::move(HLSLSema));
1302 CI.getASTContext().setExternalSource(std::move(MultiSema));
1303 } else
1304 CI.getASTContext().setExternalSource(std::move(HLSLSema));
1305 }
1306
1307 FailureCleanup.release();
1308 return true;
1309}
1310
1313 ExecuteAction();
1314
1315 // If we are supposed to rebuild the global module index, do so now unless
1316 // there were any module-build failures.
1318 CI.hasPreprocessor()) {
1319 StringRef Cache =
1321 if (!Cache.empty()) {
1322 if (llvm::Error Err = GlobalModuleIndex::writeIndex(
1324 // FIXME this drops the error on the floor, but
1325 // Index/pch-from-libclang.c seems to rely on dropping at least some of
1326 // the error conditions!
1327 consumeError(std::move(Err));
1328 }
1329 }
1330 }
1331
1332 return llvm::Error::success();
1333}
1334
1337
1338 // Inform the preprocessor we are done.
1339 if (CI.hasPreprocessor())
1341
1342 // Inform the diagnostic client we are done with this source file.
1343 // Do this after notifying the preprocessor, so that end-of-file preprocessor
1344 // callbacks can report diagnostics.
1346
1347 // Finalize the action.
1349
1350 // Sema references the ast consumer, so reset sema first.
1351 //
1352 // FIXME: There is more per-file stuff we could just drop here?
1353 bool DisableFree = CI.getFrontendOpts().DisableFree;
1354 if (DisableFree) {
1355 CI.resetAndLeakSema();
1357 llvm::BuryPointer(CI.takeASTConsumer().get());
1358 } else {
1359 CI.setSema(nullptr);
1360 CI.setASTContext(nullptr);
1361 CI.setASTConsumer(nullptr);
1362 }
1363
1364 if (CI.getFrontendOpts().ShowStats) {
1365 llvm::errs() << "\nSTATISTICS FOR '" << getCurrentFileOrBufferName() << "':\n";
1366 if (CI.hasPreprocessor()) {
1370 }
1371 if (CI.hasSourceManager()) {
1373 }
1374 llvm::errs() << "\n";
1375 }
1376
1377 // Cleanup the output streams, and erase the output files if instructed by the
1378 // FrontendAction.
1379 CI.clearOutputFiles(/*EraseFiles=*/shouldEraseOutputFiles());
1380
1381 // The resources are owned by AST when the current file is AST.
1382 // So we reset the resources here to avoid users accessing it
1383 // accidently.
1384 if (isCurrentFileAST()) {
1385 if (DisableFree) {
1389 llvm::BuryPointer(std::move(CurrentASTUnit));
1390 } else {
1391 CI.setPreprocessor(nullptr);
1392 CI.setSourceManager(nullptr);
1393 CI.setFileManager(nullptr);
1394 }
1395 }
1396
1397 setCompilerInstance(nullptr);
1399 CI.getLangOpts().setCompilingModule(LangOptions::CMK_None);
1400}
1401
1405
1406//===----------------------------------------------------------------------===//
1407// Utility Actions
1408//===----------------------------------------------------------------------===//
1409
1412 if (!CI.hasPreprocessor())
1413 return;
1414 // This is a fallback: If the client forgets to invoke this, we mark the
1415 // current stack as the bottom. Though not optimal, this could help prevent
1416 // stack overflow during deep recursion.
1418
1419 // FIXME: Move the truncation aspect of this into Sema, we delayed this till
1420 // here so the source manager would be initialized.
1424
1425 // Use a code completion consumer?
1426 CodeCompleteConsumer *CompletionConsumer = nullptr;
1428 CompletionConsumer = &CI.getCodeCompletionConsumer();
1429
1430 if (!CI.hasSema())
1431 CI.createSema(getTranslationUnitKind(), CompletionConsumer);
1432
1435}
1436
1437void PluginASTAction::anchor() { }
1438
1439std::unique_ptr<ASTConsumer>
1441 StringRef InFile) {
1442 llvm_unreachable("Invalid CreateASTConsumer on preprocessor action!");
1443}
1444
1446 return WrappedAction->PrepareToExecuteAction(CI);
1447}
1448std::unique_ptr<ASTConsumer>
1450 StringRef InFile) {
1451 return WrappedAction->CreateASTConsumer(CI, InFile);
1452}
1454 return WrappedAction->BeginInvocation(CI);
1455}
1457 WrappedAction->setCurrentInput(getCurrentInput());
1458 WrappedAction->setCompilerInstance(&CI);
1459 auto Ret = WrappedAction->BeginSourceFileAction(CI);
1460 // BeginSourceFileAction may change CurrentInput, e.g. during module builds.
1461 setCurrentInput(WrappedAction->getCurrentInput());
1462 return Ret;
1463}
1465 WrappedAction->ExecuteAction();
1466}
1469 WrappedAction->EndSourceFileAction();
1470}
1472 return WrappedAction->shouldEraseOutputFiles();
1473}
1474
1476 return WrappedAction->usesPreprocessorOnly();
1477}
1482 return WrappedAction->hasPCHSupport();
1483}
1485 return WrappedAction->hasASTFileSupport();
1486}
1488 return WrappedAction->hasIRSupport();
1489}
1491 return WrappedAction->hasCodeCompletionSupport();
1492}
1493
1495 std::unique_ptr<FrontendAction> WrappedAction)
1496 : WrappedAction(std::move(WrappedAction)) {}
Defines the clang::ASTContext interface.
Defines enum values for all the target-independent builtin functions.
Defines interfaces for clang::FileEntry and clang::FileEntryRef.
Token Tok
The Token.
FormatToken * Previous
The previous token in the unwrapped line.
static std::error_code collectModuleHeaderIncludes(const LangOptions &LangOpts, FileManager &FileMgr, DiagnosticsEngine &Diag, ModuleMap &ModMap, clang::Module *Module, SmallVectorImpl< char > &Includes)
Collect the set of header includes needed to construct the given module and update the TopHeaders fil...
static Module * prepareToBuildModule(CompilerInstance &CI, StringRef ModuleMapFilename)
static void addHeaderInclude(StringRef HeaderName, SmallVectorImpl< char > &Includes, const LangOptions &LangOpts, bool IsExternC)
static bool loadModuleMapForModuleBuild(CompilerInstance &CI, bool IsSystem, bool IsPreprocessed, std::string &PresumedModuleMapFile, unsigned &Offset)
static SourceLocation ReadOriginalFileName(CompilerInstance &CI, std::string &InputFile, bool IsModuleMap=false)
For preprocessed files, if the first line is the linemarker and specifies the original source file na...
static SmallVectorImpl< char > & operator+=(SmallVectorImpl< char > &Includes, StringRef RHS)
static std::unique_ptr< llvm::MemoryBuffer > getInputBufferForModule(CompilerInstance &CI, Module *M)
Compute the input buffer that should be used to build the specified module.
Defines the clang::FrontendAction interface and various convenience abstract classes (clang::ASTFront...
Defines the clang::LangOptions interface.
static DiagnosticBuilder Diag(DiagnosticsEngine *Diags, const LangOptions &Features, FullSourceLoc TokLoc, const char *TokBegin, const char *TokRangeBegin, const char *TokRangeEnd, unsigned DiagID)
Produce a diagnostic highlighting some portion of a literal.
#define SM(sm)
Defines the clang::Preprocessor interface.
Defines clang::SarifDocumentWriter, clang::SarifRule, clang::SarifResult.
Defines the clang::SourceLocation class and associated facilities.
Defines the SourceManager interface.
Defines utilities for dealing with stack allocation and stack space.
Defines the clang::TokenKind enum and support functions.
ASTConsumer - This is an abstract interface that should be implemented by clients that read ASTs.
Definition ASTConsumer.h:34
void setASTMutationListener(ASTMutationListener *Listener)
Attach an AST mutation listener to the AST context.
void setExternalSource(IntrusiveRefCntPtr< ExternalASTSource > Source)
Attach an external AST source to the AST context.
IntrusiveRefCntPtr< ExternalASTSource > getExternalSourcePtr() const
Retrieve a pointer to the external AST source associated with this AST context, if any.
ExternalASTSource * getExternalSource() const
Retrieve a pointer to the external AST source associated with this AST context, if any.
void ExecuteAction() override
Implement the ExecuteAction interface by running Sema on the already-initialized AST consumer.
Reads an AST files chain containing the contents of a translation unit.
Definition ASTReader.h:430
void visitTopLevelModuleMaps(serialization::ModuleFile &MF, llvm::function_ref< void(FileEntryRef)> Visitor)
Visit all the top-level module maps loaded when building the given module file.
ModuleManager & getModuleManager()
Retrieve the module manager.
Definition ASTReader.h:1991
static bool isAcceptableASTFile(StringRef Filename, FileManager &FileMgr, const ModuleCache &ModCache, const PCHContainerReader &PCHContainerRdr, const LangOptions &LangOpts, const CodeGenOptions &CGOpts, const TargetOptions &TargetOpts, const PreprocessorOptions &PPOpts, StringRef ExistingModuleCachePath, bool RequireStrictOptionMatches=false)
Determine whether the given AST file is acceptable to load into a translation unit with the given lan...
static std::unique_ptr< ASTUnit > LoadFromASTFile(StringRef Filename, const PCHContainerReader &PCHContainerRdr, WhatToLoad ToLoad, std::shared_ptr< DiagnosticOptions > DiagOpts, IntrusiveRefCntPtr< DiagnosticsEngine > Diags, const FileSystemOptions &FileSystemOpts, const HeaderSearchOptions &HSOpts, const LangOptions *LangOpts=nullptr, bool OnlyLocalDecls=false, CaptureDiagsKind CaptureDiagnostics=CaptureDiagsKind::None, bool AllowASTWithCompilerErrors=false, bool UserFilesAreVolatile=false, IntrusiveRefCntPtr< llvm::vfs::FileSystem > VFS=llvm::vfs::getRealFileSystem())
Create a ASTUnit from an AST file.
Definition ASTUnit.cpp:809
@ LoadPreprocessorOnly
Load options and the preprocessor state.
Definition ASTUnit.h:711
@ LoadEverything
Load everything, including Sema.
Definition ASTUnit.h:717
void initializeBuiltins(IdentifierTable &Table, const LangOptions &LangOpts)
Mark the identifiers for all the builtins with their appropriate builtin ID # and mark any non-portab...
Definition Builtins.cpp:203
SourceLocation getEnd() const
SourceLocation getBegin() const
Abstract interface for a consumer of code-completion information.
CompilerInstance - Helper class for managing a single instance of the Clang compiler.
void createPCHExternalASTSource(StringRef Path, DisableValidationForModuleKind DisableValidation, bool AllowPCHWithCompilerErrors, void *DeserializationListener, bool OwnDeserializationListener)
Create an external AST source to read a PCH file and attach it to the AST context.
DiagnosticConsumer & getDiagnosticClient() const
void createPreprocessor(TranslationUnitKind TUKind)
Create the preprocessor, using the invocation, file, and source managers, and replace any existing on...
void createSourceManager(FileManager &FileMgr)
Create the source manager and replace any existing one with it.
const PCHContainerReader & getPCHContainerReader() const
Return the appropriate PCHContainerReader depending on the current CodeGenOptions.
DiagnosticsEngine & getDiagnostics() const
Get the current diagnostics engine.
FileSystemOptions & getFileSystemOpts()
bool InitializeSourceManager(const FrontendInputFile &Input)
InitializeSourceManager - Initialize the source manager to set InputFile as the main file.
void setVirtualFileSystem(IntrusiveRefCntPtr< llvm::vfs::FileSystem > FS)
Use the given file system.
FileManager & getFileManager() const
Return the current file manager to the caller.
IntrusiveRefCntPtr< DiagnosticsEngine > getDiagnosticsPtr() const
void setASTConsumer(std::unique_ptr< ASTConsumer > Value)
setASTConsumer - Replace the current AST consumer; the compiler instance takes ownership of Value.
ModuleCache & getModuleCache() const
IntrusiveRefCntPtr< ASTReader > getASTReader() const
void createASTContext()
Create the AST context.
Preprocessor & getPreprocessor() const
Return the current preprocessor.
ASTContext & getASTContext() const
TargetOptions & getTargetOpts()
void createVirtualFileSystem(IntrusiveRefCntPtr< llvm::vfs::FileSystem > BaseFS=llvm::vfs::getRealFileSystem(), DiagnosticConsumer *DC=nullptr)
Create a virtual file system instance based on the invocation.
void setASTReader(IntrusiveRefCntPtr< ASTReader > Reader)
FrontendOptions & getFrontendOpts()
void setSema(Sema *S)
Replace the current Sema; the compiler instance takes ownership of S.
void setSourceManager(llvm::IntrusiveRefCntPtr< SourceManager > Value)
setSourceManager - Replace the current source manager.
void setASTContext(llvm::IntrusiveRefCntPtr< ASTContext > Value)
setASTContext - Replace the current AST context.
HeaderSearchOptions & getHeaderSearchOpts()
bool hasCodeCompletionConsumer() const
std::unique_ptr< ASTConsumer > takeASTConsumer()
takeASTConsumer - Remove the current AST consumer and give ownership to the caller.
PreprocessorOptions & getPreprocessorOpts()
std::string getSpecificModuleCachePath(StringRef ModuleHash)
FileManager * createFileManager()
Create the file manager and replace any existing one with it.
void setFileManager(IntrusiveRefCntPtr< FileManager > Value)
Replace the current file manager and virtual file system.
TargetInfo & getTarget() const
void createCodeCompletionConsumer()
Create a code completion consumer using the invocation; note that this will cause the source manager ...
void clearOutputFiles(bool EraseFiles)
clearOutputFiles - Clear the output file list.
DiagnosticOptions & getDiagnosticOpts()
CodeGenOptions & getCodeGenOpts()
SourceManager & getSourceManager() const
Return the current source manager.
CodeCompleteConsumer & getCodeCompletionConsumer() const
bool shouldBuildGlobalModuleIndex() const
Indicates whether we should (re)build the global module index.
bool loadModuleFile(StringRef FileName, serialization::ModuleFile *&LoadedModuleFile)
void setPreprocessor(std::shared_ptr< Preprocessor > Value)
Replace the current preprocessor.
void createSema(TranslationUnitKind TUKind, CodeCompleteConsumer *CompletionConsumer)
Create the Sema object to be used for parsing.
bool isFileContext() const
Definition DeclBase.h:2180
DeclContext * getLexicalParent()
getLexicalParent - Returns the containing lexical DeclContext.
Definition DeclBase.h:2125
SourceLocation getLocation() const
Definition DeclBase.h:439
const char * getDeclKindName() const
Definition DeclBase.cpp:147
DeclContext * getLexicalDeclContext()
getLexicalDeclContext - The declaration context where this Decl was lexically declared (LexicalDC).
Definition DeclBase.h:918
const LangOptions & getLangOpts() const LLVM_READONLY
Helper to get the language options from the ASTContext.
Definition DeclBase.cpp:530
virtual SourceRange getSourceRange() const LLVM_READONLY
Source range that this declaration covers.
Definition DeclBase.h:427
void DeclRead(GlobalDeclID ID, const Decl *D) override
A decl was deserialized from the AST file.
virtual void EndSourceFile()
Callback to inform the diagnostic client that processing of a source file has ended.
virtual void BeginSourceFile(const LangOptions &LangOpts, const Preprocessor *PP=nullptr)
Callback to inform the diagnostic client that processing of a source file is beginning.
Concrete class used by the front-end to report problems and issues.
Definition Diagnostic.h:231
DiagnosticBuilder Report(SourceLocation Loc, unsigned DiagID)
Issue the message to the client.
bool hasErrorOccurred() const
Definition Diagnostic.h:871
A reference to a FileEntry that includes the name of the file as it was accessed by the FileManager's...
Definition FileEntry.h:57
StringRef getName() const
The name of this FileEntry.
Definition FileEntry.h:61
An opaque identifier used by SourceManager which refers to a source file (MemoryBuffer) along with it...
Implements support for file system lookup, file system caching, and directory search management.
Definition FileManager.h:53
OptionalFileEntryRef getOptionalFileRef(StringRef Filename, bool OpenFile=false, bool CacheFailure=true)
Get a FileEntryRef if it exists, without doing anything on error.
OptionalDirectoryEntryRef getOptionalDirectoryRef(StringRef DirName, bool CacheFailure=true)
Get a DirectoryEntryRef if it exists, without doing anything on error.
const FrontendInputFile & getCurrentInput() const
virtual void EndSourceFile()
Perform any per-file post processing, deallocate per-file objects, and run statistics and output file...
virtual bool shouldEraseOutputFiles()
Callback at the end of processing a single input, to determine if the output files should be erased o...
virtual bool hasIRSupport() const
Does this action support use with IR files?
virtual void EndSourceFileAction()
Callback at the end of processing a single input.
virtual std::unique_ptr< ASTConsumer > CreateASTConsumer(CompilerInstance &CI, StringRef InFile)=0
Create the AST consumer object for this action, if supported.
CompilerInstance & getCompilerInstance() const
llvm::Error Execute()
Set the source manager's main input file, and run the action.
virtual bool hasASTFileSupport() const
Does this action support use with AST files?
Module * getCurrentModule() const
void setCurrentInput(const FrontendInputFile &CurrentInput, std::unique_ptr< ASTUnit > AST=nullptr)
virtual bool BeginSourceFileAction(CompilerInstance &CI)
Callback at the start of processing a single input.
virtual void ExecuteAction()=0
Callback to run the program action, using the initialized compiler instance.
void setCompilerInstance(CompilerInstance *Value)
virtual TranslationUnitKind getTranslationUnitKind()
For AST-based actions, the kind of translation unit we're handling.
bool BeginSourceFile(CompilerInstance &CI, const FrontendInputFile &Input)
Prepare the action for processing the input file Input.
virtual bool hasCodeCompletionSupport() const
Does this action support use with code completion?
StringRef getCurrentFileOrBufferName() const
virtual bool isModelParsingAction() const
Is this action invoked on a model file?
bool isCurrentFileAST() const
virtual bool usesPreprocessorOnly() const =0
Does this action only use the preprocessor?
friend class WrapperFrontendAction
virtual bool BeginInvocation(CompilerInstance &CI)
Callback before starting processing a single input, giving the opportunity to modify the CompilerInvo...
virtual bool hasPCHSupport() const
Does this action support use with PCH?
An input file for the front end.
InputKind getKind() const
StringRef getFile() const
std::vector< std::string > ModuleFiles
The list of additional prebuilt module files to load before processing the input.
unsigned SkipFunctionBodies
Skip over function bodies to speed up parsing in cases you do not need them (e.g.
std::map< std::string, std::vector< std::string > > PluginArgs
Args to pass to the plugins.
unsigned ShowStats
Show frontend performance metrics and statistics.
ParsedSourceLocation CodeCompletionAt
If given, enable code completion at the provided location.
std::string OriginalModuleMap
When the input is a module map, the original module map file from which that map was inferred,...
std::vector< std::string > ModulesEmbedFiles
The list of files to embed into the compiled module file.
unsigned ModulesEmbedAllFiles
Whether we should embed all used files into the PCM file.
std::vector< std::string > AddPluginActions
The list of plugin actions to run in addition to the normal action.
std::string DumpMinimizationHintsPath
Output path to dump ranges of deserialized declarations to use as minimization hints.
unsigned DisableFree
Disable memory freeing on exit.
std::string OverrideRecordLayoutsFile
File name of the file that will provide record layouts (in the format produced by -fdump-record-layou...
std::vector< std::string > ModuleMapFiles
The list of module map files to load before processing the input.
A SourceLocation and its associated SourceManager.
static llvm::Error writeIndex(FileManager &FileMgr, const PCHContainerReader &PCHContainerRdr, llvm::StringRef Path)
Write a global index into the given.
Encapsulates the information needed to find the file referenced by a #include or #include_next,...
Module * lookupModule(StringRef ModuleName, SourceLocation ImportLoc=SourceLocation(), bool AllowSearch=true, bool AllowExtraModuleMapSearch=false)
Lookup a module Search for a module with the given name.
OptionalFileEntryRef LookupFile(StringRef Filename, SourceLocation IncludeLoc, bool isAngled, ConstSearchDirIterator FromDir, ConstSearchDirIterator *CurDir, ArrayRef< std::pair< OptionalFileEntryRef, DirectoryEntryRef > > Includers, SmallVectorImpl< char > *SearchPath, SmallVectorImpl< char > *RelativePath, Module *RequestingModule, ModuleMap::KnownHeader *SuggestedModule, bool *IsMapped, bool *IsFrameworkFound, bool SkipCache=false, bool BuildSystemModule=false, bool OpenFile=true, bool CacheFailures=true)
Given a "foo" or <foo> reference, look up the indicated file, return null on failure.
StringRef getModuleCachePath() const
Retrieve the path to the module cache.
bool parseAndLoadModuleMapFile(FileEntryRef File, bool IsSystem, FileID ID=FileID(), unsigned *Offset=nullptr, StringRef OriginalModuleMapFile=StringRef())
Read the contents of the given module map file.
ModuleMap & getModuleMap()
Retrieve the module map.
void PrintStats() const
Print some statistics to stderr that indicate how well the hashing is doing.
The kind of a file that we've been handed as an input.
bool isPreprocessed() const
InputKind withHeaderUnit(HeaderUnitKind HU) const
bool isHeaderUnit() const
Format getFormat() const
HeaderUnitKind getHeaderUnitKind() const
Language getLanguage() const
@ CMK_None
Not compiling a module interface at all.
@ CMK_ModuleMap
Compiling a module from a module map.
Keeps track of the various options that can be enabled, which controls the dialect of C or C++ that i...
std::string ModuleName
The module currently being compiled as specified by -fmodule-name.
std::string CurrentModule
The name of the current module, of which the main source file is a part.
Lexer - This provides a simple interface that turns a text buffer into a stream of tokens.
Definition Lexer.h:78
static CharSourceRange getAsCharRange(SourceRange Range, const SourceManager &SM, const LangOptions &LangOpts)
Given a token range, produce a corresponding CharSourceRange that is not a token range.
Definition Lexer.h:430
static unsigned getSpelling(const Token &Tok, const char *&Buffer, const SourceManager &SourceMgr, const LangOptions &LangOpts, bool *Invalid=nullptr)
getSpelling - This method is used to get the spelling of a token into a preallocated buffer,...
Definition Lexer.cpp:451
static bool getRawToken(SourceLocation Loc, Token &Result, const SourceManager &SM, const LangOptions &LangOpts, bool IgnoreWhiteSpace=false)
Relex the token at the specified location.
Definition Lexer.cpp:509
void finishModuleDeclarationScope()
Creates a new declaration scope for module names, allowing previously defined modules to shadow defin...
Definition ModuleMap.h:605
bool canInferFrameworkModule(const DirectoryEntry *Dir) const
Check whether a framework module can be inferred in the given directory.
Definition ModuleMap.h:614
bool isHeaderUnavailableInModule(FileEntryRef Header, const Module *RequestingModule) const
Determine whether the given header is unavailable as part of the specified module.
void resolveHeaderDirectives(const FileEntry *File) const
Resolve all lazy header directives for the specified file.
void setInferredModuleAllowedBy(Module *M, FileID ModMapFID)
Describes a module or submodule.
Definition Module.h:144
SmallVector< UnresolvedHeaderDirective, 1 > MissingHeaders
Headers that are mentioned in the module map file but could not be found on the file system.
Definition Module.h:341
Module * Parent
The parent of this module.
Definition Module.h:193
unsigned IsSystem
Whether this is a "system" module (which assumes that all headers in it are system headers).
Definition Module.h:389
static StringRef getModuleInputBufferName()
Definition Module.h:851
llvm::iterator_range< submodule_iterator > submodules()
Definition Module.h:838
unsigned IsExternC
Whether this is an 'extern "C"' module (which implicitly puts all headers in it within an 'extern "C"...
Definition Module.h:395
std::optional< Header > getUmbrellaHeaderAsWritten() const
Retrieve the umbrella header as written.
Definition Module.h:756
OptionalDirectoryEntryRef Directory
The build directory of this module.
Definition Module.h:198
std::string PresumedModuleMapFile
The presumed file name for the module map defining this module.
Definition Module.h:202
ArrayRef< Header > getHeaders(HeaderKind HK) const
Definition Module.h:302
bool isAvailable() const
Determine whether this module is available for use within the current translation unit.
Definition Module.h:586
std::optional< DirectoryName > getUmbrellaDirAsWritten() const
Retrieve the umbrella directory as written.
Definition Module.h:748
std::string getFullModuleName(bool AllowStringLiterals=false) const
Retrieve the full name of this module, including the path from its top-level module.
Definition Module.cpp:239
void addTopHeader(FileEntryRef File)
Add a top-level header associated with this module.
Definition Module.cpp:271
@ AddAfterMainAction
Execute the action after the main action.
@ AddBeforeMainAction
Execute the action before the main action.
@ CmdlineBeforeMainAction
Execute the action before the main action if on the command line.
@ CmdlineAfterMainAction
Execute the action after the main action if on the command line.
std::unique_ptr< ASTConsumer > CreateASTConsumer(CompilerInstance &CI, StringRef InFile) override
Provide a default implementation which returns aborts; this method should never be called by Frontend...
PreprocessorOptions - This class is used for passing the various options used in preprocessor initial...
std::set< std::string > DeserializedPCHDeclsToErrorOn
This is a set of names for decls that we do not want to be deserialized, and we emit an error if they...
std::vector< std::string > ChainedIncludes
Headers that will be converted to chained PCHs in memory.
std::string ImplicitPCHInclude
The implicit PCH included at the start of the translation unit, or empty.
DisableValidationForModuleKind DisablePCHOrModuleValidation
Whether to disable most of the normal validation performed on precompiled headers and module files.
bool DumpDeserializedPCHDecls
Dump declarations that are deserialized from PCH, for testing.
bool AllowPCHWithCompilerErrors
When true, a PCH with compiler errors will not be rejected.
Engages in a tight little dance with the lexer to efficiently preprocess tokens.
void setMainFileDir(DirectoryEntryRef Dir)
Set the directory in which the main file should be considered to have been found, if it is not a real...
static bool checkModuleIsAvailable(const LangOptions &LangOpts, const TargetInfo &TargetInfo, const Module &M, DiagnosticsEngine &Diags)
Check that the given module is available, producing a diagnostic if not.
Module * getCurrentModuleImplementation()
Retrieves the module whose implementation we're current compiling, if any.
HeaderSearch & getHeaderSearchInfo() const
IdentifierTable & getIdentifierTable()
Builtin::Context & getBuiltinInfo()
void setSkipMainFilePreamble(unsigned Bytes, bool StartOfLine)
Instruct the preprocessor to skip part of the main source file.
const LangOptions & getLangOpts() const
void EndSourceFile()
Inform the preprocessor callbacks that processing is complete.
void setSarifWriter(std::unique_ptr< SarifDocumentWriter > SarifWriter)
Encodes a location in the source.
bool isValid() const
Return true if this is a valid SourceLocation object.
SourceLocation getLocWithOffset(IntTy Offset) const
Return a source location with the specified offset from this SourceLocation.
This class handles loading and caching of source files into memory.
FileIDAndOffset getDecomposedLoc(SourceLocation Loc) const
Decompose the specified location into a raw FileID + Offset pair.
void setAllFilesAreTransient(bool Transient)
Specify that all files that are read during this compilation are transient.
void AddLineNote(SourceLocation Loc, unsigned LineNo, int FilenameID, bool IsFileEntry, bool IsFileExit, SrcMgr::CharacteristicKind FileKind)
Add a line note to the line table for the FileID and offset specified by Loc.
void setFileIsTransient(FileEntryRef SourceFile)
Specify that a file is transient.
OptionalFileEntryRef getFileEntryRefForID(FileID FID) const
Returns the FileEntryRef for the provided FileID.
FileID createFileID(FileEntryRef SourceFile, SourceLocation IncludePos, SrcMgr::CharacteristicKind FileCharacter, int LoadedID=0, SourceLocation::UIntTy LoadedOffset=0)
Create a new FileID that represents the specified file being #included from the specified IncludePosi...
void PrintStats() const
Print statistics to stderr.
FileID getMainFileID() const
Returns the FileID of the main source file.
void pushModuleBuildStack(StringRef moduleName, FullSourceLoc importLoc)
Push an entry to the module build stack.
void initializeForReplay(const SourceManager &Old)
Initialize this source manager suitably to replay the compilation described by Old.
FileID getOrCreateFileID(FileEntryRef SourceFile, SrcMgr::CharacteristicKind FileCharacter)
Get the FileID for SourceFile if it exists.
StringLiteralParser - This decodes string escape characters and performs wide string analysis and Tra...
Token - This structure provides full information about a lexed token.
Definition Token.h:36
SourceLocation getEndLoc() const
Definition Token.h:161
SourceLocation getLocation() const
Return a source location identifier for the specified offset in the current file.
Definition Token.h:134
bool is(tok::TokenKind K) const
is/isNot - Predicates to check if this token is a specific kind, as in "if (Tok.is(tok::l_brace)) {....
Definition Token.h:102
tok::TokenKind getKind() const
Definition Token.h:97
bool BeginSourceFileAction(CompilerInstance &CI) override
Callback at the start of processing a single input.
bool hasIRSupport() const override
Does this action support use with IR files?
bool PrepareToExecuteAction(CompilerInstance &CI) override
Prepare to execute the action on the given CompilerInstance.
bool hasASTFileSupport() const override
Does this action support use with AST files?
bool usesPreprocessorOnly() const override
Does this action only use the preprocessor?
void ExecuteAction() override
Callback to run the program action, using the initialized compiler instance.
bool hasCodeCompletionSupport() const override
Does this action support use with code completion?
bool shouldEraseOutputFiles() override
Callback at the end of processing a single input, to determine if the output files should be erased o...
TranslationUnitKind getTranslationUnitKind() override
For AST-based actions, the kind of translation unit we're handling.
void EndSourceFileAction() override
Callback at the end of processing a single input.
bool BeginInvocation(CompilerInstance &CI) override
Callback before starting processing a single input, giving the opportunity to modify the CompilerInvo...
bool hasPCHSupport() const override
Does this action support use with PCH?
std::unique_ptr< ASTConsumer > CreateASTConsumer(CompilerInstance &CI, StringRef InFile) override
Create the AST consumer object for this action, if supported.
std::unique_ptr< FrontendAction > WrappedAction
void EndSourceFile() override
Perform any per-file post processing, deallocate per-file objects, and run statistics and output file...
Information about a module that has been loaded by the ASTReader.
Definition ModuleFile.h:130
bool StandardCXXModule
Whether this module file is a standard C++ module.
Definition ModuleFile.h:176
ModuleFile & getPrimaryModule()
Returns the primary module associated with the manager, that is, the first module loaded.
Public enums and private classes that are part of the SourceManager implementation.
The JSON file list parser is used to communicate input to InstallAPI.
bool isa(CodeGen::Address addr)
Definition Address.h:330
CustomizableOptional< FileEntryRef > OptionalFileEntryRef
Definition FileEntry.h:208
void ParseAST(Preprocessor &pp, ASTConsumer *C, ASTContext &Ctx, bool PrintStats=false, TranslationUnitKind TUKind=TU_Complete, CodeCompleteConsumer *CompletionConsumer=nullptr, bool SkipFunctionBodies=false)
Parse the entire file specified, notifying the ASTConsumer as the file is parsed.
Definition ParseAST.cpp:99
@ Override
Merge availability attributes for an override, which requires an exact match or a weakening of constr...
Definition Sema.h:634
nullptr
This class represents a compute construct, representing a 'Kind' of ‘parallel’, 'serial',...
bool operator<(DeclarationName LHS, DeclarationName RHS)
Ordering on two declaration names.
@ Result
The result type of a method or function.
Definition TypeBase.h:905
const FunctionProtoType * T
IntrusiveRefCntPtr< ExternalSemaSource > createChainedIncludesSource(CompilerInstance &CI, IntrusiveRefCntPtr< ASTReader > &OutReader)
The ChainedIncludesSource class converts headers to chained PCHs in memory, mainly for testing.
llvm::Registry< PluginASTAction > FrontendPluginRegistry
The frontend plugin registry.
void noteBottomOfStack(bool ForceSet=false)
Call this once on each thread, as soon after starting the thread as feasible, to note the approximate...
Definition Stack.cpp:20
TranslationUnitKind
Describes the kind of translation unit being processed.
U cast(CodeGen::Address addr)
Definition Address.h:327
Information about a header directive as found in the module map file.
Definition Module.h:287