clang 23.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"
20#include "clang/Basic/Sarif.h"
23#include "clang/Basic/Stack.h"
43#include "llvm/ADT/ScopeExit.h"
44#include "llvm/ADT/SmallPtrSet.h"
45#include "llvm/ADT/StringRef.h"
46#include "llvm/Support/BuryPointer.h"
47#include "llvm/Support/ErrorHandling.h"
48#include "llvm/Support/FileSystem.h"
49#include "llvm/Support/Path.h"
50#include "llvm/Support/Timer.h"
51#include "llvm/Support/raw_ostream.h"
52#include <memory>
53#include <system_error>
54using namespace clang;
55
57
58namespace {
59
60/// DeserializedDeclsLineRangePrinter dumps ranges of deserialized declarations
61/// to aid debugging and bug minimization. It implements ASTConsumer and
62/// ASTDeserializationListener, so that an object of
63/// DeserializedDeclsLineRangePrinter registers as its own listener. The
64/// ASTDeserializationListener interface provides the DeclRead callback that we
65/// use to collect the deserialized Decls. Note that printing or otherwise
66/// processing them as this point is dangerous, since that could trigger
67/// additional deserialization and crash compilation. Therefore, we process the
68/// collected Decls in HandleTranslationUnit method of ASTConsumer. This is a
69/// safe point, since we know that by this point all the Decls needed by the
70/// compiler frontend have been deserialized. In case our processing causes
71/// further deserialization, DeclRead from the listener might be called again.
72/// However, at that point we don't accept any more Decls for processing.
73class DeserializedDeclsSourceRangePrinter : public ASTConsumer,
75public:
76 explicit DeserializedDeclsSourceRangePrinter(
77 SourceManager &SM, std::unique_ptr<llvm::raw_fd_ostream> OS)
78 : ASTDeserializationListener(), SM(SM), OS(std::move(OS)) {}
79
80 ASTDeserializationListener *GetASTDeserializationListener() override {
81 return this;
82 }
83
84 void DeclRead(GlobalDeclID ID, const Decl *D) override {
85 if (!IsCollectingDecls)
86 return;
89 // These decls cover a lot of nested declarations that might not be used,
90 // reducing the granularity and making the output less useful.
91 return;
92 }
93 if (isa<ParmVarDecl>(D)) {
94 // Parameters are covered by their functions.
95 return;
96 }
97 auto *DC = D->getLexicalDeclContext();
98 if (!DC || !shouldIncludeDeclsIn(DC))
99 return;
100
101 PendingDecls.push_back(D);
102 for (; (isa<ExportDecl>(DC) || isa<NamespaceDecl>(DC)) &&
103 ProcessedDeclContexts.insert(DC).second;
104 DC = DC->getLexicalParent()) {
105 // Add any interesting decl contexts that we have not seen before.
106 // Note that we filter them out from DeclRead as that would include all
107 // redeclarations of namespaces, potentially those that do not have any
108 // imported declarations.
109 PendingDecls.push_back(cast<Decl>(DC));
110 }
111 }
112
113 struct Position {
114 unsigned Line;
115 unsigned Column;
116
117 bool operator<(const Position &other) const {
118 return std::tie(Line, Column) < std::tie(other.Line, other.Column);
119 }
120
121 static Position GetBeginSpelling(const SourceManager &SM,
122 const CharSourceRange &R) {
123 SourceLocation Begin = R.getBegin();
124 return {SM.getSpellingLineNumber(Begin),
125 SM.getSpellingColumnNumber(Begin)};
126 }
127
128 static Position GetEndSpelling(const SourceManager &SM,
129 const CharSourceRange &Range,
130 const LangOptions &LangOpts) {
131 // For token ranges, compute end location for end character of the range.
132 CharSourceRange R = Lexer::getAsCharRange(Range, SM, LangOpts);
133 SourceLocation End = R.getEnd();
134 // Relex the token past the end location of the last token in the source
135 // range. If it's a semicolon, advance the location by one token.
136 Token PossiblySemi;
137 Lexer::getRawToken(End, PossiblySemi, SM, LangOpts, true);
138 if (PossiblySemi.is(tok::semi))
139 End = End.getLocWithOffset(1);
140 // Column number of the returned end position is exclusive.
141 return {SM.getSpellingLineNumber(End), SM.getSpellingColumnNumber(End)};
142 }
143 };
144
145 struct RequiredRanges {
146 StringRef Filename;
147 std::vector<std::pair<Position, Position>> FromTo;
148 };
149 void HandleTranslationUnit(ASTContext &Context) override {
150 assert(IsCollectingDecls && "HandleTranslationUnit called twice?");
151 IsCollectingDecls = false;
152
153 // Merge ranges in each of the files.
154 struct FileData {
155 std::vector<std::pair<Position, Position>> FromTo;
157 };
158 llvm::DenseMap<const FileEntry *, FileData> FileToRanges;
159
160 for (const Decl *D : PendingDecls) {
161 for (CharSourceRange R : getRangesToMark(D)) {
162 if (!R.isValid())
163 continue;
164
165 auto *F = SM.getFileEntryForID(SM.getFileID(R.getBegin()));
166 if (F != SM.getFileEntryForID(SM.getFileID(R.getEnd()))) {
167 // Such cases are rare and difficult to handle.
168 continue;
169 }
170
171 auto &Data = FileToRanges[F];
172 if (!Data.Ref)
173 Data.Ref = SM.getFileEntryRefForID(SM.getFileID(R.getBegin()));
174 Data.FromTo.push_back(
175 {Position::GetBeginSpelling(SM, R),
176 Position::GetEndSpelling(SM, R, D->getLangOpts())});
177 }
178 }
179
180 // To simplify output, merge consecutive and intersecting ranges.
181 std::vector<RequiredRanges> Result;
182 for (auto &[F, Data] : FileToRanges) {
183 auto &FromTo = Data.FromTo;
184 assert(!FromTo.empty());
185
186 if (!Data.Ref)
187 continue;
188
189 llvm::sort(FromTo);
190
191 std::vector<std::pair<Position, Position>> MergedRanges;
192 MergedRanges.push_back(FromTo.front());
193 for (auto It = FromTo.begin() + 1; It < FromTo.end(); ++It) {
194 if (MergedRanges.back().second < It->first) {
195 MergedRanges.push_back(*It);
196 continue;
197 }
198 if (MergedRanges.back().second < It->second)
199 MergedRanges.back().second = It->second;
200 }
201 Result.push_back({Data.Ref->getName(), std::move(MergedRanges)});
202 }
203 printJson(Result);
204 }
205
206private:
207 std::vector<const Decl *> PendingDecls;
208 llvm::SmallPtrSet<const DeclContext *, 0> ProcessedDeclContexts;
209 bool IsCollectingDecls = true;
210 const SourceManager &SM;
211 std::unique_ptr<llvm::raw_ostream> OS;
212
213 static bool shouldIncludeDeclsIn(const DeclContext *DC) {
214 assert(DC && "DC is null");
215 // We choose to work at namespace level to reduce complexity and the number
216 // of cases we care about.
217 // We still need to carefully handle composite declarations like
218 // `ExportDecl`.
219 for (; DC; DC = DC->getLexicalParent()) {
220 if (DC->isFileContext())
221 return true;
222 if (isa<ExportDecl>(DC))
223 continue; // Depends on the parent.
224 return false;
225 }
226 llvm_unreachable("DeclContext chain must end with a translation unit");
227 }
228
229 llvm::SmallVector<CharSourceRange, 2> getRangesToMark(const Decl *D) {
230 if (auto *ED = dyn_cast<ExportDecl>(D)) {
231 if (!ED->hasBraces())
232 return {SM.getExpansionRange(ED->getExportLoc())};
233
234 return {SM.getExpansionRange(SourceRange(
235 ED->getExportLoc(),
236 lexForLBrace(ED->getExportLoc(), D->getLangOpts()))),
237 SM.getExpansionRange(ED->getRBraceLoc())};
238 }
239
240 auto *NS = dyn_cast<NamespaceDecl>(D);
241 if (!NS)
242 return {SM.getExpansionRange(D->getSourceRange())};
243
244 SourceLocation LBraceLoc;
245 if (NS->isAnonymousNamespace()) {
246 LBraceLoc = NS->getLocation();
247 } else {
248 // Start with the location of the identifier.
249 SourceLocation TokenBeforeLBrace = NS->getLocation();
250 if (NS->hasAttrs()) {
251 for (auto *A : NS->getAttrs()) {
252 // But attributes may go after it.
253 if (SM.isBeforeInTranslationUnit(TokenBeforeLBrace,
254 A->getRange().getEnd())) {
255 // Give up, the attributes are often coming from macros and we
256 // cannot skip them reliably.
257 return {};
258 }
259 }
260 }
261 LBraceLoc = lexForLBrace(TokenBeforeLBrace, D->getLangOpts());
262 }
263 return {SM.getExpansionRange(SourceRange(NS->getBeginLoc(), LBraceLoc)),
264 SM.getExpansionRange(NS->getRBraceLoc())};
265 }
266
267 void printJson(llvm::ArrayRef<RequiredRanges> Result) {
268 *OS << "{\n";
269 *OS << R"( "required_ranges": [)" << "\n";
270 for (size_t I = 0; I < Result.size(); ++I) {
271 auto &F = Result[I].Filename;
272 auto &MergedRanges = Result[I].FromTo;
273 *OS << R"( {)" << "\n";
274 *OS << R"( "file": ")" << F << "\"," << "\n";
275 *OS << R"( "range": [)" << "\n";
276 for (size_t J = 0; J < MergedRanges.size(); ++J) {
277 auto &From = MergedRanges[J].first;
278 auto &To = MergedRanges[J].second;
279 *OS << R"( {)" << "\n";
280 *OS << R"( "from": {)" << "\n";
281 *OS << R"( "line": )" << From.Line << ",\n";
282 *OS << R"( "column": )" << From.Column << "\n"
283 << R"( },)" << "\n";
284 *OS << R"( "to": {)" << "\n";
285 *OS << R"( "line": )" << To.Line << ",\n";
286 *OS << R"( "column": )" << To.Column << "\n"
287 << R"( })" << "\n";
288 *OS << R"( })";
289 if (J < MergedRanges.size() - 1) {
290 *OS << ",";
291 }
292 *OS << "\n";
293 }
294 *OS << " ]" << "\n" << " }";
295 if (I < Result.size() - 1)
296 *OS << ",";
297 *OS << "\n";
298 }
299 *OS << " ]\n";
300 *OS << "}\n";
301
302 OS->flush();
303 }
304
305 SourceLocation lexForLBrace(SourceLocation TokenBeforeLBrace,
306 const LangOptions &LangOpts) {
307 // Now skip one token, the next should be the lbrace.
308 Token Tok;
309 if (Lexer::getRawToken(TokenBeforeLBrace, Tok, SM, LangOpts, true) ||
310 Lexer::getRawToken(Tok.getEndLoc(), Tok, SM, LangOpts, true) ||
311 Tok.getKind() != tok::l_brace) {
312 // On error or if we did not find the token we expected, avoid marking
313 // everything inside the namespace as used.
314 return SourceLocation();
315 }
316 return Tok.getLocation();
317 }
318};
319
320/// Dumps deserialized declarations.
321class DeserializedDeclsDumper : public DelegatingDeserializationListener {
322public:
323 explicit DeserializedDeclsDumper(ASTDeserializationListener *Previous,
324 bool DeletePrevious)
325 : DelegatingDeserializationListener(Previous, DeletePrevious) {}
326
327 void DeclRead(GlobalDeclID ID, const Decl *D) override {
328 llvm::outs() << "PCH DECL: " << D->getDeclKindName();
329 if (const NamedDecl *ND = dyn_cast<NamedDecl>(D)) {
330 llvm::outs() << " - ";
331 ND->printQualifiedName(llvm::outs());
332 }
333 llvm::outs() << "\n";
334
336 }
337};
338
339/// Checks deserialized declarations and emits error if a name
340/// matches one given in command-line using -error-on-deserialized-decl.
341class DeserializedDeclsChecker : public DelegatingDeserializationListener {
342 ASTContext &Ctx;
343 std::set<std::string> NamesToCheck;
344
345public:
346 DeserializedDeclsChecker(ASTContext &Ctx,
347 const std::set<std::string> &NamesToCheck,
348 ASTDeserializationListener *Previous,
349 bool DeletePrevious)
350 : DelegatingDeserializationListener(Previous, DeletePrevious), Ctx(Ctx),
351 NamesToCheck(NamesToCheck) {}
352
353 void DeclRead(GlobalDeclID ID, const Decl *D) override {
354 if (const NamedDecl *ND = dyn_cast<NamedDecl>(D))
355 if (NamesToCheck.find(ND->getNameAsString()) != NamesToCheck.end()) {
356 unsigned DiagID
357 = Ctx.getDiagnostics().getCustomDiagID(DiagnosticsEngine::Error,
358 "%0 was deserialized");
359 Ctx.getDiagnostics().Report(Ctx.getFullLoc(D->getLocation()), DiagID)
360 << ND;
361 }
362
364 }
365};
366
367} // end anonymous namespace
368
370
372
374 std::unique_ptr<ASTUnit> AST) {
375 this->CurrentInput = CurrentInput;
376 CurrentASTUnit = std::move(AST);
377}
378
384
385std::unique_ptr<ASTConsumer>
386FrontendAction::CreateWrappedASTConsumer(CompilerInstance &CI,
387 StringRef InFile) {
388 std::unique_ptr<ASTConsumer> Consumer = CreateASTConsumer(CI, InFile);
389 if (!Consumer)
390 return nullptr;
391
392 std::vector<std::unique_ptr<ASTConsumer>> Consumers;
393 llvm::StringRef DumpDeserializedDeclarationRangesPath =
395 if (!DumpDeserializedDeclarationRangesPath.empty()) {
396 std::error_code ErrorCode;
397 auto FileStream = std::make_unique<llvm::raw_fd_ostream>(
398 DumpDeserializedDeclarationRangesPath, ErrorCode,
399 llvm::sys::fs::OF_TextWithCRLF);
400 if (!ErrorCode) {
401 Consumers.push_back(std::make_unique<DeserializedDeclsSourceRangePrinter>(
402 CI.getSourceManager(), std::move(FileStream)));
403 } else {
404 llvm::errs() << "Failed to create output file for "
405 "-dump-minimization-hints flag, file path: "
406 << DumpDeserializedDeclarationRangesPath
407 << ", error: " << ErrorCode.message() << "\n";
408 }
409 }
410
411 // Validate -add-plugin args.
412 bool FoundAllPlugins = true;
413 for (const std::string &Arg : CI.getFrontendOpts().AddPluginActions) {
414 bool Found = false;
415 for (const FrontendPluginRegistry::entry &Plugin :
416 FrontendPluginRegistry::entries()) {
417 if (Plugin.getName() == Arg)
418 Found = true;
419 }
420 if (!Found) {
421 CI.getDiagnostics().Report(diag::err_fe_invalid_plugin_name) << Arg;
422 FoundAllPlugins = false;
423 }
424 }
425 if (!FoundAllPlugins)
426 return nullptr;
427
428 // If this is a code completion run, avoid invoking the plugin consumers
430 return Consumer;
431
432 // Collect the list of plugins that go before the main action (in Consumers)
433 // or after it (in AfterConsumers)
434 std::vector<std::unique_ptr<ASTConsumer>> AfterConsumers;
435 for (const FrontendPluginRegistry::entry &Plugin :
436 FrontendPluginRegistry::entries()) {
437 std::unique_ptr<PluginASTAction> P = Plugin.instantiate();
438 PluginASTAction::ActionType ActionType = P->getActionType();
439 if (ActionType == PluginASTAction::CmdlineAfterMainAction ||
441 // This is O(|plugins| * |add_plugins|), but since both numbers are
442 // way below 50 in practice, that's ok.
443 if (llvm::is_contained(CI.getFrontendOpts().AddPluginActions,
444 Plugin.getName())) {
447 else
449 }
450 }
451 if ((ActionType == PluginASTAction::AddBeforeMainAction ||
453 P->ParseArgs(
454 CI,
455 CI.getFrontendOpts().PluginArgs[std::string(Plugin.getName())])) {
456 std::unique_ptr<ASTConsumer> PluginConsumer = P->CreateASTConsumer(CI, InFile);
457 if (ActionType == PluginASTAction::AddBeforeMainAction) {
458 Consumers.push_back(std::move(PluginConsumer));
459 } else {
460 AfterConsumers.push_back(std::move(PluginConsumer));
461 }
462 }
463 }
464
465 // Add to Consumers the main consumer, then all the plugins that go after it
466 Consumers.push_back(std::move(Consumer));
467 if (!AfterConsumers.empty()) {
468 // If we have plugins after the main consumer, which may be the codegen
469 // action, they likely will need the ASTContext, so don't clear it in the
470 // codegen action.
471 CI.getCodeGenOpts().ClearASTBeforeBackend = false;
472 for (auto &C : AfterConsumers)
473 Consumers.push_back(std::move(C));
474 }
475
476 assert(Consumers.size() >= 1 && "should have added the main consumer");
477 if (Consumers.size() == 1)
478 return std::move(Consumers.front());
479 return std::make_unique<MultiplexConsumer>(std::move(Consumers));
480}
481
482/// For preprocessed files, if the first line is the linemarker and specifies
483/// the original source file name, use that name as the input file name.
484/// Returns the location of the first token after the line marker directive.
485///
486/// \param CI The compiler instance.
487/// \param InputFile Populated with the filename from the line marker.
488/// \param IsModuleMap If \c true, add a line note corresponding to this line
489/// directive. (We need to do this because the directive will not be
490/// visited by the preprocessor.)
492 std::string &InputFile,
493 bool IsModuleMap = false) {
494 auto &SourceMgr = CI.getSourceManager();
495 auto MainFileID = SourceMgr.getMainFileID();
496
497 auto MainFileBuf = SourceMgr.getBufferOrNone(MainFileID);
498 if (!MainFileBuf)
499 return SourceLocation();
500
501 auto RawLexer = std::make_unique<Lexer>(MainFileID, *MainFileBuf, SourceMgr,
502 CI.getLangOpts());
503
504 // If the first line has the syntax of
505 //
506 // # NUM "FILENAME"
507 //
508 // we use FILENAME as the input file name.
509 Token T;
510 if (RawLexer->LexFromRawLexer(T) || T.getKind() != tok::hash)
511 return SourceLocation();
512 if (RawLexer->LexFromRawLexer(T) || T.isAtStartOfLine() ||
513 T.getKind() != tok::numeric_constant)
514 return SourceLocation();
515
516 unsigned LineNo;
517 SourceLocation LineNoLoc = T.getLocation();
518 if (IsModuleMap) {
520 if (Lexer::getSpelling(LineNoLoc, Buffer, SourceMgr, CI.getLangOpts())
521 .getAsInteger(10, LineNo))
522 return SourceLocation();
523 }
524
525 RawLexer->LexFromRawLexer(T);
526 if (T.isAtStartOfLine() || T.getKind() != tok::string_literal)
527 return SourceLocation();
528
529 StringLiteralParser Literal(T, CI.getPreprocessor(),
531 if (Literal.hadError)
532 return SourceLocation();
533 RawLexer->LexFromRawLexer(T);
534 if (T.isNot(tok::eof) && !T.isAtStartOfLine())
535 return SourceLocation();
536 InputFile = Literal.GetString().str();
537
538 if (IsModuleMap)
540 LineNoLoc, LineNo, SourceMgr.getLineTableFilenameID(InputFile), false,
542
543 return T.getLocation();
544}
545
547operator+=(SmallVectorImpl<char> &Includes, StringRef RHS) {
548 Includes.append(RHS.begin(), RHS.end());
549 return Includes;
550}
551
552static void addHeaderInclude(StringRef HeaderName,
553 SmallVectorImpl<char> &Includes,
554 const LangOptions &LangOpts,
555 bool IsExternC) {
556 if (IsExternC && LangOpts.CPlusPlus)
557 Includes += "extern \"C\" {\n";
558 if (LangOpts.ObjC)
559 Includes += "#import \"";
560 else
561 Includes += "#include \"";
562
563 Includes += HeaderName;
564
565 Includes += "\"\n";
566 if (IsExternC && LangOpts.CPlusPlus)
567 Includes += "}\n";
568}
569
570/// Collect the set of header includes needed to construct the given
571/// module and update the TopHeaders file set of the module.
572///
573/// \param Module The module we're collecting includes from.
574///
575/// \param Includes Will be augmented with the set of \#includes or \#imports
576/// needed to load all of the named headers.
577static std::error_code collectModuleHeaderIncludes(
580 // Don't collect any headers for unavailable modules.
581 if (!Module->isAvailable())
582 return std::error_code();
583
584 // Resolve all lazy header directives to header files.
585 ModMap.resolveHeaderDirectives(Module, /*File=*/std::nullopt);
586
587 // If any headers are missing, we can't build this module. In most cases,
588 // diagnostics for this should have already been produced; we only get here
589 // if explicit stat information was provided.
590 // FIXME: If the name resolves to a file with different stat information,
591 // produce a better diagnostic.
592 if (!Module->MissingHeaders.empty()) {
593 auto &MissingHeader = Module->MissingHeaders.front();
594 Diag.Report(MissingHeader.FileNameLoc, diag::err_module_header_missing)
595 << MissingHeader.IsUmbrella << MissingHeader.FileName;
596 return std::error_code();
597 }
598
599 // Add includes for each of these headers.
600 for (auto HK : {Module::HK_Normal, Module::HK_Private}) {
601 for (const Module::Header &H : Module->getHeaders(HK)) {
602 Module->addTopHeader(H.Entry);
603 // Use the path as specified in the module map file. We'll look for this
604 // file relative to the module build directory (the directory containing
605 // the module map file) so this will find the same file that we found
606 // while parsing the module map.
607 addHeaderInclude(H.PathRelativeToRootModuleDirectory, Includes, LangOpts,
609 }
610 }
611 // Note that Module->PrivateHeaders will not be a TopHeader.
612
613 if (std::optional<Module::Header> UmbrellaHeader =
615 Module->addTopHeader(UmbrellaHeader->Entry);
616 if (Module->Parent)
617 // Include the umbrella header for submodules.
618 addHeaderInclude(UmbrellaHeader->PathRelativeToRootModuleDirectory,
619 Includes, LangOpts, Module->IsExternC);
620 } else if (std::optional<Module::DirectoryName> UmbrellaDir =
622 // Add all of the headers we find in this subdirectory.
623 std::error_code EC;
624 SmallString<128> DirNative;
625 llvm::sys::path::native(UmbrellaDir->Entry.getName(), DirNative);
626
627 llvm::vfs::FileSystem &FS = FileMgr.getVirtualFileSystem();
629 for (llvm::vfs::recursive_directory_iterator Dir(FS, DirNative, EC), End;
630 Dir != End && !EC; Dir.increment(EC)) {
631 // Check whether this entry has an extension typically associated with
632 // headers.
633 if (!llvm::StringSwitch<bool>(llvm::sys::path::extension(Dir->path()))
634 .Cases({".h", ".H", ".hh", ".hpp"}, true)
635 .Default(false))
636 continue;
637
638 // Compute the relative path from the directory to this file.
640 auto PathIt = llvm::sys::path::rbegin(Dir->path());
641 for (int I = 0; I != Dir.level() + 1; ++I, ++PathIt)
642 Components.push_back(*PathIt);
643 SmallString<128> RelativeHeader(
644 UmbrellaDir->PathRelativeToRootModuleDirectory);
645 for (auto It = Components.rbegin(), End = Components.rend(); It != End;
646 ++It)
647 llvm::sys::path::append(RelativeHeader, *It);
648
649 HeaderPaths.push_back(
650 std::make_pair(Dir->path().str(), RelativeHeader.c_str()));
651 }
652
653 if (EC)
654 return EC;
655
656 // Sort header paths and make the header inclusion order deterministic
657 // across different OSs and filesystems. As the header search table
658 // serialization order depends on the file reference UID, we need to create
659 // file references in deterministic order too.
660 llvm::sort(HeaderPaths, llvm::less_first());
661 for (auto &[Path, RelPath] : HeaderPaths) {
662 auto Header = FileMgr.getOptionalFileRef(Path);
663 // FIXME: This shouldn't happen unless there is a file system race. Is
664 // that worth diagnosing?
665 if (!Header)
666 continue;
667
668 // If this header is marked 'unavailable' in this module, don't include
669 // it.
670 if (ModMap.isHeaderUnavailableInModule(*Header, Module))
671 continue;
672
673 // Include this header as part of the umbrella directory.
674 Module->addTopHeader(*Header);
675 addHeaderInclude(RelPath, Includes, LangOpts, Module->IsExternC);
676 }
677 }
678
679 // Recurse into submodules.
680 for (clang::Module *Submodule : Module->submodules())
681 if (std::error_code Err = collectModuleHeaderIncludes(
682 LangOpts, FileMgr, Diag, ModMap, Submodule, Includes))
683 return Err;
684
685 return std::error_code();
686}
687
688static bool loadModuleMapForModuleBuild(CompilerInstance &CI, bool IsSystem,
689 bool IsPreprocessed,
690 std::string &PresumedModuleMapFile,
691 unsigned &Offset) {
692 auto &SrcMgr = CI.getSourceManager();
694
695 // Map the current input to a file.
696 FileID ModuleMapID = SrcMgr.getMainFileID();
697 OptionalFileEntryRef ModuleMap = SrcMgr.getFileEntryRefForID(ModuleMapID);
698 assert(ModuleMap && "MainFileID without FileEntry");
699
700 // If the module map is preprocessed, handle the initial line marker;
701 // line directives are not part of the module map syntax in general.
702 Offset = 0;
703 if (IsPreprocessed) {
704 SourceLocation EndOfLineMarker =
705 ReadOriginalFileName(CI, PresumedModuleMapFile, /*IsModuleMap*/ true);
706 if (EndOfLineMarker.isValid())
707 Offset = CI.getSourceManager().getDecomposedLoc(EndOfLineMarker).second;
708 }
709
710 // Load the module map file.
711 if (HS.parseAndLoadModuleMapFile(*ModuleMap, IsSystem,
712 /*ImplicitlyDiscovered=*/false, ModuleMapID,
713 &Offset, PresumedModuleMapFile))
714 return true;
715
716 if (SrcMgr.getBufferOrFake(ModuleMapID).getBufferSize() == Offset)
717 Offset = 0;
718
719 // Infer framework module if possible.
720 if (HS.getModuleMap().canInferFrameworkModule(ModuleMap->getDir())) {
721 SmallString<128> InferredFrameworkPath = ModuleMap->getDir().getName();
722 llvm::sys::path::append(InferredFrameworkPath,
723 CI.getLangOpts().ModuleName + ".framework");
724 if (auto Dir =
725 CI.getFileManager().getOptionalDirectoryRef(InferredFrameworkPath))
726 (void)HS.getModuleMap().inferFrameworkModule(*Dir, IsSystem, nullptr);
727 }
728
729 return false;
730}
731
733 StringRef ModuleMapFilename) {
734 if (CI.getLangOpts().CurrentModule.empty()) {
735 CI.getDiagnostics().Report(diag::err_missing_module_name);
736
737 // FIXME: Eventually, we could consider asking whether there was just
738 // a single module described in the module map, and use that as a
739 // default. Then it would be fairly trivial to just "compile" a module
740 // map with a single module (the common case).
741 return nullptr;
742 }
743
744 // Dig out the module definition.
747 /*AllowSearch=*/true);
748 if (!M) {
749 CI.getDiagnostics().Report(diag::err_missing_module)
750 << CI.getLangOpts().CurrentModule << ModuleMapFilename;
751
752 return nullptr;
753 }
754
755 // Check whether we can build this module at all.
757 CI.getDiagnostics()))
758 return nullptr;
759
760 // Inform the preprocessor that includes from within the input buffer should
761 // be resolved relative to the build directory of the module map file.
763
764 // If the module was inferred from a different module map (via an expanded
765 // umbrella module definition), track that fact.
766 // FIXME: It would be preferable to fill this in as part of processing
767 // the module map, rather than adding it after the fact.
768 StringRef OriginalModuleMapName = CI.getFrontendOpts().OriginalModuleMap;
769 if (!OriginalModuleMapName.empty()) {
770 auto OriginalModuleMap =
771 CI.getFileManager().getOptionalFileRef(OriginalModuleMapName,
772 /*openFile*/ true);
773 if (!OriginalModuleMap) {
774 CI.getDiagnostics().Report(diag::err_module_map_not_found)
775 << OriginalModuleMapName;
776 return nullptr;
777 }
778 if (*OriginalModuleMap != CI.getSourceManager().getFileEntryRefForID(
780 auto FileCharacter =
782 FileID OriginalModuleMapFID = CI.getSourceManager().getOrCreateFileID(
783 *OriginalModuleMap, FileCharacter);
784 CI.getPreprocessor()
786 .getModuleMap()
787 .setInferredModuleAllowedBy(M, OriginalModuleMapFID);
788 }
789 }
790
791 // If we're being run from the command-line, the module build stack will not
792 // have been filled in yet, so complete it now in order to allow us to detect
793 // module cycles.
794 SourceManager &SourceMgr = CI.getSourceManager();
795 if (SourceMgr.getModuleBuildStack().empty())
797 FullSourceLoc(SourceLocation(), SourceMgr));
798 return M;
799}
800
801/// Compute the input buffer that should be used to build the specified module.
802static std::unique_ptr<llvm::MemoryBuffer>
805
806 // Collect the set of #includes we need to build the module.
807 SmallString<256> HeaderContents;
808 std::error_code Err = std::error_code();
809 if (std::optional<Module::Header> UmbrellaHeader =
811 addHeaderInclude(UmbrellaHeader->PathRelativeToRootModuleDirectory,
812 HeaderContents, CI.getLangOpts(), M->IsExternC);
816 HeaderContents);
817
818 if (Err) {
819 CI.getDiagnostics().Report(diag::err_module_cannot_create_includes)
820 << M->getFullModuleName() << Err.message();
821 return nullptr;
822 }
823
824 return llvm::MemoryBuffer::getMemBufferCopy(
825 HeaderContents, Module::getModuleInputBufferName());
826}
827
829 const FrontendInputFile &RealInput) {
830 FrontendInputFile Input(RealInput);
831 assert(!Instance && "Already processing a source file!");
832 assert(!Input.isEmpty() && "Unexpected empty filename!");
833 setCurrentInput(Input);
835
836 bool HasBegunSourceFile = false;
837 bool ReplayASTFile = Input.getKind().getFormat() == InputKind::Precompiled &&
839
840 // If we fail, reset state since the client will not end up calling the
841 // matching EndSourceFile(). All paths that return true should release this.
842 llvm::scope_exit FailureCleanup([&]() {
843 if (HasBegunSourceFile)
845 CI.setASTConsumer(nullptr);
846 CI.clearOutputFiles(/*EraseFiles=*/true);
847 CI.getLangOpts().setCompilingModule(LangOptions::CMK_None);
849 setCompilerInstance(nullptr);
850 });
851
852 if (!BeginInvocation(CI))
853 return false;
854
855 // The list of module files the input AST file depends on. This is separate
856 // from FrontendOptions::ModuleFiles, because those only represent explicit
857 // modules, while this is capable of representing implicit ones too.
858 SmallVector<ModuleFileName> ModuleFiles;
859
860 // If we're replaying the build of an AST file, import it and set up
861 // the initial state from its build.
862 if (ReplayASTFile) {
864
865 // The AST unit populates its own diagnostics engine rather than ours.
866 auto ASTDiags = llvm::makeIntrusiveRefCnt<DiagnosticsEngine>(
867 Diags->getDiagnosticIDs(), Diags->getDiagnosticOptions());
868 ASTDiags->setClient(Diags->getClient(), /*OwnsClient*/false);
869
870 // FIXME: What if the input is a memory buffer?
871 StringRef InputFile = Input.getFile();
872
873 std::unique_ptr<ASTUnit> AST = ASTUnit::LoadFromASTFile(
875 CI.getVirtualFileSystemPtr(), nullptr, ASTDiags, CI.getFileSystemOpts(),
877 if (!AST)
878 return false;
879
880 // Options relating to how we treat the input (but not what we do with it)
881 // are inherited from the AST unit.
882 CI.getHeaderSearchOpts() = AST->getHeaderSearchOpts();
883 CI.getPreprocessorOpts() = AST->getPreprocessorOpts();
884 CI.getLangOpts() = AST->getLangOpts();
885
886 // Set the shared objects, these are reset when we finish processing the
887 // file, otherwise the CompilerInstance will happily destroy them.
888 CI.setVirtualFileSystem(AST->getFileManager().getVirtualFileSystemPtr());
889 CI.setFileManager(AST->getFileManagerPtr());
891 CI.getSourceManager().initializeForReplay(AST->getSourceManager());
892
893 // Preload all the module files loaded transitively by the AST unit. Also
894 // load all module map files that were parsed as part of building the AST
895 // unit.
896 if (auto ASTReader = AST->getASTReader()) {
897 auto &MM = ASTReader->getModuleManager();
898 auto &PrimaryModule = MM.getPrimaryModule();
899
900 for (serialization::ModuleFile &MF : MM)
901 if (&MF != &PrimaryModule)
902 ModuleFiles.emplace_back(MF.FileName);
903
904 ASTReader->visitTopLevelModuleMaps(PrimaryModule, [&](FileEntryRef FE) {
905 CI.getFrontendOpts().ModuleMapFiles.push_back(
906 std::string(FE.getName()));
907 });
908 }
909
910 // Set up the input file for replay purposes.
911 auto Kind = AST->getInputKind();
912 if (Kind.getFormat() == InputKind::ModuleMap) {
913 Module *ASTModule =
914 AST->getPreprocessor().getHeaderSearchInfo().lookupModule(
915 AST->getLangOpts().CurrentModule, SourceLocation(),
916 /*AllowSearch*/ false);
917 assert(ASTModule && "module file does not define its own module");
918 Input = FrontendInputFile(ASTModule->PresumedModuleMapFile, Kind);
919 } else {
920 auto &OldSM = AST->getSourceManager();
921 FileID ID = OldSM.getMainFileID();
922 if (auto File = OldSM.getFileEntryRefForID(ID))
923 Input = FrontendInputFile(File->getName(), Kind);
924 else
925 Input = FrontendInputFile(OldSM.getBufferOrFake(ID), Kind);
926 }
927 setCurrentInput(Input, std::move(AST));
928 }
929
930 // AST files follow a very different path, since they share objects via the
931 // AST unit.
932 if (Input.getKind().getFormat() == InputKind::Precompiled) {
933 assert(!usesPreprocessorOnly() && "this case was handled above");
934 assert(hasASTFileSupport() &&
935 "This action does not have AST file support!");
936
938
939 // FIXME: What if the input is a memory buffer?
940 StringRef InputFile = Input.getFile();
941
942 std::unique_ptr<ASTUnit> AST = ASTUnit::LoadFromASTFile(
944 CI.getVirtualFileSystemPtr(), nullptr, Diags, CI.getFileSystemOpts(),
945 CI.getHeaderSearchOpts(), &CI.getLangOpts());
946
947 if (!AST)
948 return false;
949
950 // Inform the diagnostic client we are processing a source file.
952 HasBegunSourceFile = true;
953
954 // Set the shared objects, these are reset when we finish processing the
955 // file, otherwise the CompilerInstance will happily destroy them.
956 CI.setVirtualFileSystem(AST->getVirtualFileSystemPtr());
957 CI.setFileManager(AST->getFileManagerPtr());
958 CI.setSourceManager(AST->getSourceManagerPtr());
959 CI.setPreprocessor(AST->getPreprocessorPtr());
960 Preprocessor &PP = CI.getPreprocessor();
962 PP.getLangOpts());
963 CI.setASTContext(AST->getASTContextPtr());
964
965 setCurrentInput(Input, std::move(AST));
966
967 // Initialize the action.
968 if (!BeginSourceFileAction(CI))
969 return false;
970
971 // Create the AST consumer.
972 CI.setASTConsumer(CreateWrappedASTConsumer(CI, InputFile));
973 if (!CI.hasASTConsumer())
974 return false;
975
976 FailureCleanup.release();
977 return true;
978 }
979
980 // Set up the file system, file and source managers, if needed.
981 if (!CI.hasVirtualFileSystem())
983 if (!CI.hasFileManager())
985 if (!CI.hasSourceManager()) {
987 if (CI.getDiagnosticOpts().getFormat() == DiagnosticOptions::SARIF) {
988 static_cast<SARIFDiagnosticPrinter *>(&CI.getDiagnosticClient())
990 std::make_unique<SarifDocumentWriter>(CI.getSourceManager()));
991 }
992 }
993
994 // Set up embedding for any specified files. Do this before we load any
995 // source files, including the primary module map for the compilation.
996 for (const auto &F : CI.getFrontendOpts().ModulesEmbedFiles) {
997 if (auto FE = CI.getFileManager().getOptionalFileRef(F, /*openFile*/true))
999 else
1000 CI.getDiagnostics().Report(diag::err_modules_embed_file_not_found) << F;
1001 }
1004
1005 // IR files bypass the rest of initialization.
1006 if (Input.getKind().getLanguage() == Language::LLVM_IR) {
1007 if (!hasIRSupport()) {
1008 CI.getDiagnostics().Report(diag::err_ast_action_on_llvm_ir)
1009 << Input.getFile();
1010 return false;
1011 }
1012
1013 // Inform the diagnostic client we are processing a source file.
1015 HasBegunSourceFile = true;
1016
1017 // Initialize the action.
1018 if (!BeginSourceFileAction(CI))
1019 return false;
1020
1021 // Initialize the main file entry.
1022 if (!CI.InitializeSourceManager(CurrentInput))
1023 return false;
1024
1025 FailureCleanup.release();
1026 return true;
1027 }
1028
1029 if (!CI.getPreprocessorOpts().ImplicitPCHInclude.empty()) {
1032
1033 // Canonicalize ImplicitPCHInclude. This way, all the downstream code,
1034 // including the ASTWriter, will receive the absolute path to the included
1035 // PCH.
1036 SmallString<128> PCHIncludePath(PPOpts.ImplicitPCHInclude);
1037 FileMgr.makeAbsolutePath(PCHIncludePath);
1038 llvm::sys::path::remove_dots(PCHIncludePath, true);
1039 PPOpts.ImplicitPCHInclude = PCHIncludePath.str();
1040 StringRef PCHInclude = PPOpts.ImplicitPCHInclude;
1041
1042 // If the implicit PCH include is actually a directory, rather than
1043 // a single file, search for a suitable PCH file in that directory.
1044 if (auto PCHDir = FileMgr.getOptionalDirectoryRef(PCHInclude)) {
1045 std::error_code EC;
1046 SmallString<128> DirNative;
1047 llvm::sys::path::native(PCHDir->getName(), DirNative);
1048 bool Found = false;
1049 llvm::vfs::FileSystem &FS = FileMgr.getVirtualFileSystem();
1050 std::string SpecificModuleCachePath = createSpecificModuleCachePath(
1054 for (llvm::vfs::directory_iterator Dir = FS.dir_begin(DirNative, EC),
1055 DirEnd;
1056 Dir != DirEnd && !EC; Dir.increment(EC)) {
1057 // Check whether this is an acceptable AST file.
1059 Dir->path(), FileMgr, CI.getModuleCache(),
1061 CI.getCodeGenOpts(), CI.getTargetOpts(),
1063 SpecificModuleCachePath,
1064 /*RequireStrictOptionMatches=*/true)) {
1065 PPOpts.ImplicitPCHInclude = std::string(Dir->path());
1066 Found = true;
1067 break;
1068 }
1069 }
1070
1071 if (!Found) {
1072 CI.getDiagnostics().Report(diag::err_fe_no_pch_in_dir) << PCHInclude;
1073 return false;
1074 }
1075 }
1076 }
1077
1078 // Set up the preprocessor if needed. When parsing model files the
1079 // preprocessor of the original source is reused.
1080 if (!isModelParsingAction())
1082
1083 // Inform the diagnostic client we are processing a source file.
1085 &CI.getPreprocessor());
1086 HasBegunSourceFile = true;
1087
1088 // Handle C++20 header units.
1089 // Here, the user has the option to specify that the header name should be
1090 // looked up in the pre-processor search paths (and the main filename as
1091 // passed by the driver might therefore be incomplete until that look-up).
1092 if (CI.getLangOpts().CPlusPlusModules && Input.getKind().isHeaderUnit() &&
1093 !Input.getKind().isPreprocessed()) {
1094 StringRef FileName = Input.getFile();
1095 InputKind Kind = Input.getKind();
1096 if (Kind.getHeaderUnitKind() != InputKind::HeaderUnit_Abs) {
1097 assert(CI.hasPreprocessor() &&
1098 "trying to build a header unit without a Pre-processor?");
1100 // Relative searches begin from CWD.
1101 auto Dir = CI.getFileManager().getOptionalDirectoryRef(".");
1103 CWD.push_back({std::nullopt, *Dir});
1106 /*Angled*/ Input.getKind().getHeaderUnitKind() ==
1108 nullptr, nullptr, CWD, nullptr, nullptr, nullptr,
1109 nullptr, nullptr, nullptr);
1110 if (!FE) {
1111 CI.getDiagnostics().Report(diag::err_module_header_file_not_found)
1112 << FileName;
1113 return false;
1114 }
1115 // We now have the filename...
1116 FileName = FE->getName();
1117 // ... still a header unit, but now use the path as written.
1119 Input = FrontendInputFile(FileName, Kind, Input.isSystem());
1120 }
1121 // Unless the user has overridden the name, the header unit module name is
1122 // the pathname for the file.
1123 if (CI.getLangOpts().ModuleName.empty())
1124 CI.getLangOpts().ModuleName = std::string(FileName);
1126 }
1127
1128 if (!CI.InitializeSourceManager(Input))
1129 return false;
1130
1131 if (CI.getLangOpts().CPlusPlusModules && Input.getKind().isHeaderUnit() &&
1133 // We have an input filename like foo.iih, but we want to find the right
1134 // module name (and original file, to build the map entry).
1135 // Check if the first line specifies the original source file name with a
1136 // linemarker.
1137 std::string PresumedInputFile = std::string(getCurrentFileOrBufferName());
1138 ReadOriginalFileName(CI, PresumedInputFile);
1139 // Unless the user overrides this, the module name is the name by which the
1140 // original file was known.
1141 if (CI.getLangOpts().ModuleName.empty())
1142 CI.getLangOpts().ModuleName = std::string(PresumedInputFile);
1144 }
1145
1146 // For module map files, we first parse the module map and synthesize a
1147 // "<module-includes>" buffer before more conventional processing.
1148 if (Input.getKind().getFormat() == InputKind::ModuleMap) {
1149 CI.getLangOpts().setCompilingModule(LangOptions::CMK_ModuleMap);
1150
1151 std::string PresumedModuleMapFile;
1152 unsigned OffsetToContents;
1153 if (loadModuleMapForModuleBuild(CI, Input.isSystem(),
1154 Input.isPreprocessed(),
1155 PresumedModuleMapFile, OffsetToContents))
1156 return false;
1157
1158 auto *CurrentModule = prepareToBuildModule(CI, Input.getFile());
1159 if (!CurrentModule)
1160 return false;
1161
1162 CurrentModule->PresumedModuleMapFile = PresumedModuleMapFile;
1163
1164 if (OffsetToContents)
1165 // If the module contents are in the same file, skip to them.
1166 CI.getPreprocessor().setSkipMainFilePreamble(OffsetToContents, true);
1167 else {
1168 // Otherwise, convert the module description to a suitable input buffer.
1169 auto Buffer = getInputBufferForModule(CI, CurrentModule);
1170 if (!Buffer)
1171 return false;
1172
1173 // Reinitialize the main file entry to refer to the new input.
1174 auto Kind = CurrentModule->IsSystem ? SrcMgr::C_System : SrcMgr::C_User;
1175 auto &SourceMgr = CI.getSourceManager();
1176 auto BufferID = SourceMgr.createFileID(std::move(Buffer), Kind);
1177 assert(BufferID.isValid() && "couldn't create module buffer ID");
1178 SourceMgr.setMainFileID(BufferID);
1179 }
1180 }
1181
1182 // Initialize the action.
1183 if (!BeginSourceFileAction(CI))
1184 return false;
1185
1186 // If we were asked to load any module map files, do so now.
1187 for (const auto &Filename : CI.getFrontendOpts().ModuleMapFiles) {
1188 if (auto File = CI.getFileManager().getOptionalFileRef(Filename))
1190 *File, /*IsSystem*/ false, /*ImplicitlyDiscovered=*/false);
1191 else
1192 CI.getDiagnostics().Report(diag::err_module_map_not_found) << Filename;
1193 }
1194
1195 // If compiling implementation of a module, load its module map file now.
1197
1198 // Add a module declaration scope so that modules from -fmodule-map-file
1199 // arguments may shadow modules found implicitly in search paths.
1200 CI.getPreprocessor()
1202 .getModuleMap()
1204
1205 // Create the AST context and consumer unless this is a preprocessor only
1206 // action.
1207 if (!usesPreprocessorOnly()) {
1208 // Parsing a model file should reuse the existing ASTContext.
1209 if (!isModelParsingAction())
1210 CI.createASTContext();
1211
1212 // For preprocessed files, check if the first line specifies the original
1213 // source file name with a linemarker.
1214 std::string PresumedInputFile = std::string(getCurrentFileOrBufferName());
1215 if (Input.isPreprocessed())
1216 ReadOriginalFileName(CI, PresumedInputFile);
1217
1218 std::unique_ptr<ASTConsumer> Consumer =
1219 CreateWrappedASTConsumer(CI, PresumedInputFile);
1220 if (!Consumer)
1221 return false;
1222
1223 // FIXME: should not overwrite ASTMutationListener when parsing model files?
1224 if (!isModelParsingAction())
1225 CI.getASTContext().setASTMutationListener(Consumer->GetASTMutationListener());
1226
1227 if (!CI.getPreprocessorOpts().ChainedIncludes.empty()) {
1228 // Convert headers to PCH and chain them.
1231 source = createChainedIncludesSource(CI, FinalReader);
1232 if (!source)
1233 return false;
1234 CI.setASTReader(FinalReader);
1235 CI.getASTContext().setExternalSource(source);
1236 } else if (CI.getLangOpts().Modules ||
1238 // Use PCM or PCH.
1239 assert(hasPCHSupport() && "This action does not have PCH support!");
1240 ASTDeserializationListener *DeserialListener =
1241 Consumer->GetASTDeserializationListener();
1242 bool DeleteDeserialListener = false;
1244 DeserialListener = new DeserializedDeclsDumper(DeserialListener,
1245 DeleteDeserialListener);
1246 DeleteDeserialListener = true;
1247 }
1249 DeserialListener = new DeserializedDeclsChecker(
1250 CI.getASTContext(),
1252 DeserialListener, DeleteDeserialListener);
1253 DeleteDeserialListener = true;
1254 }
1255 if (!CI.getPreprocessorOpts().ImplicitPCHInclude.empty()) {
1260 DeserialListener, DeleteDeserialListener);
1261 if (!CI.getASTContext().getExternalSource())
1262 return false;
1263 }
1264 // If modules are enabled, create the AST reader before creating
1265 // any builtins, so that all declarations know that they might be
1266 // extended by an external source.
1267 if (CI.getLangOpts().Modules || !CI.hasASTContext() ||
1269 CI.createASTReader();
1270 CI.getASTReader()->setDeserializationListener(DeserialListener,
1271 DeleteDeserialListener);
1272 }
1273 }
1274
1275 CI.setASTConsumer(std::move(Consumer));
1276 if (!CI.hasASTConsumer())
1277 return false;
1278 }
1279
1280 // Initialize built-in info as long as we aren't using an external AST
1281 // source.
1282 if (CI.getLangOpts().Modules || !CI.hasASTContext() ||
1284 Preprocessor &PP = CI.getPreprocessor();
1286 PP.getLangOpts());
1287 } else {
1288 // FIXME: If this is a problem, recover from it by creating a multiplex
1289 // source.
1290 assert((!CI.getLangOpts().Modules || CI.getASTReader()) &&
1291 "modules enabled but created an external source that "
1292 "doesn't support modules");
1293 }
1294
1295 // If we were asked to load any module files, do so now.
1296 for (const auto &ModuleFile : CI.getFrontendOpts().ModuleFiles) {
1297 serialization::ModuleFile *Loaded = nullptr;
1298 if (!CI.loadModuleFile(ModuleFileName::makeExplicit(ModuleFile), Loaded))
1299 return false;
1300
1301 if (Loaded && Loaded->StandardCXXModule)
1303 diag::warn_eagerly_load_for_standard_cplusplus_modules);
1304 }
1305
1306 // If we were asked to load any module files by the ASTUnit, do so now.
1307 for (const auto &ModuleFile : ModuleFiles) {
1308 serialization::ModuleFile *Loaded = nullptr;
1309 if (!CI.loadModuleFile(ModuleFile, Loaded))
1310 return false;
1311
1312 if (Loaded && Loaded->StandardCXXModule)
1314 diag::warn_eagerly_load_for_standard_cplusplus_modules);
1315 }
1316
1317 // If there is a layout overrides file, attach an external AST source that
1318 // provides the layouts from that file.
1319 if (!CI.getFrontendOpts().OverrideRecordLayoutsFile.empty() &&
1321 auto Override = llvm::makeIntrusiveRefCnt<LayoutOverrideSource>(
1324 }
1325
1326 // Setup HLSL External Sema Source
1327 if (CI.getLangOpts().HLSL && CI.hasASTContext()) {
1328 auto HLSLSema = llvm::makeIntrusiveRefCnt<HLSLExternalSemaSource>();
1329 if (auto SemaSource = dyn_cast_if_present<ExternalSemaSource>(
1331 auto MultiSema = llvm::makeIntrusiveRefCnt<MultiplexExternalSemaSource>(
1332 std::move(SemaSource), std::move(HLSLSema));
1333 CI.getASTContext().setExternalSource(std::move(MultiSema));
1334 } else
1335 CI.getASTContext().setExternalSource(std::move(HLSLSema));
1336 }
1337
1338 FailureCleanup.release();
1339 return true;
1340}
1341
1344 ExecuteAction();
1345
1346 // If we are supposed to rebuild the global module index, do so now unless
1347 // there were any module-build failures.
1349 CI.hasPreprocessor()) {
1350 StringRef Cache =
1352 if (!Cache.empty()) {
1353 if (llvm::Error Err = GlobalModuleIndex::writeIndex(
1355 // FIXME this drops the error on the floor, but
1356 // Index/pch-from-libclang.c seems to rely on dropping at least some of
1357 // the error conditions!
1358 consumeError(std::move(Err));
1359 }
1360 }
1361 }
1362
1363 return llvm::Error::success();
1364}
1365
1368
1369 // Inform the preprocessor we are done.
1370 if (CI.hasPreprocessor())
1372
1373 // Inform the diagnostic client we are done with this source file.
1374 // Do this after notifying the preprocessor, so that end-of-file preprocessor
1375 // callbacks can report diagnostics.
1377
1378 // Finalize the action.
1380
1381 // Sema references the ast consumer, so reset sema first.
1382 //
1383 // FIXME: There is more per-file stuff we could just drop here?
1384 bool DisableFree = CI.getFrontendOpts().DisableFree;
1385 if (DisableFree) {
1386 CI.resetAndLeakSema();
1388 llvm::BuryPointer(CI.takeASTConsumer().get());
1389 } else {
1390 CI.setSema(nullptr);
1391 CI.setASTContext(nullptr);
1392 CI.setASTConsumer(nullptr);
1393 }
1394
1395 if (CI.getFrontendOpts().ShowStats) {
1396 llvm::errs() << "\nSTATISTICS FOR '" << getCurrentFileOrBufferName() << "':\n";
1397 if (CI.hasPreprocessor()) {
1401 }
1402 if (CI.hasSourceManager()) {
1404 }
1405 llvm::errs() << "\n";
1406 }
1407
1408 // Cleanup the output streams, and erase the output files if instructed by the
1409 // FrontendAction.
1410 CI.clearOutputFiles(/*EraseFiles=*/shouldEraseOutputFiles());
1411
1412 // The resources are owned by AST when the current file is AST.
1413 // So we reset the resources here to avoid users accessing it
1414 // accidently.
1415 if (isCurrentFileAST()) {
1416 if (DisableFree) {
1420 llvm::BuryPointer(std::move(CurrentASTUnit));
1421 } else {
1422 CI.setPreprocessor(nullptr);
1423 CI.setSourceManager(nullptr);
1424 CI.setFileManager(nullptr);
1425 }
1426 }
1427
1428 setCompilerInstance(nullptr);
1430 CI.getLangOpts().setCompilingModule(LangOptions::CMK_None);
1431}
1432
1436
1437//===----------------------------------------------------------------------===//
1438// Utility Actions
1439//===----------------------------------------------------------------------===//
1440
1443 if (!CI.hasPreprocessor())
1444 return;
1445 // This is a fallback: If the client forgets to invoke this, we mark the
1446 // current stack as the bottom. Though not optimal, this could help prevent
1447 // stack overflow during deep recursion.
1449
1450 // FIXME: Move the truncation aspect of this into Sema, we delayed this till
1451 // here so the source manager would be initialized.
1455
1456 // Use a code completion consumer?
1457 CodeCompleteConsumer *CompletionConsumer = nullptr;
1459 CompletionConsumer = &CI.getCodeCompletionConsumer();
1460
1461 if (!CI.hasSema())
1462 CI.createSema(getTranslationUnitKind(), CompletionConsumer);
1463
1466}
1467
1468void PluginASTAction::anchor() { }
1469
1470std::unique_ptr<ASTConsumer>
1472 StringRef InFile) {
1473 llvm_unreachable("Invalid CreateASTConsumer on preprocessor action!");
1474}
1475
1477 return WrappedAction->PrepareToExecuteAction(CI);
1478}
1479std::unique_ptr<ASTConsumer>
1481 StringRef InFile) {
1482 return WrappedAction->CreateASTConsumer(CI, InFile);
1483}
1485 return WrappedAction->BeginInvocation(CI);
1486}
1488 WrappedAction->setCurrentInput(getCurrentInput());
1489 WrappedAction->setCompilerInstance(&CI);
1490 auto Ret = WrappedAction->BeginSourceFileAction(CI);
1491 // BeginSourceFileAction may change CurrentInput, e.g. during module builds.
1492 setCurrentInput(WrappedAction->getCurrentInput());
1493 return Ret;
1494}
1496 WrappedAction->ExecuteAction();
1497}
1500 WrappedAction->EndSourceFileAction();
1501}
1503 return WrappedAction->shouldEraseOutputFiles();
1504}
1505
1507 return WrappedAction->usesPreprocessorOnly();
1508}
1513 return WrappedAction->hasPCHSupport();
1514}
1516 return WrappedAction->hasASTFileSupport();
1517}
1519 return WrappedAction->hasIRSupport();
1520}
1522 return WrappedAction->hasCodeCompletionSupport();
1523}
1524
1526 std::unique_ptr<FrontendAction> WrappedAction)
1527 : WrappedAction(std::move(WrappedAction)) {}
Defines the clang::ASTContext interface.
Defines enum values for all the target-independent builtin functions.
LLVM_INSTANTIATE_REGISTRY_EX(CLANG_ABI_EXPORT, clang::tooling::ToolExecutorPluginRegistry) namespace clang
Definition Execution.cpp:14
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:35
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:427
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:1977
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, const HeaderSearchOptions &HSOpts, StringRef SpecificModuleCachePath, 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, IntrusiveRefCntPtr< llvm::vfs::FileSystem > VFS, 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)
Create a ASTUnit from an AST file.
Definition ASTUnit.cpp:685
@ LoadPreprocessorOnly
Load options and the preprocessor state.
Definition ASTUnit.h:710
@ LoadEverything
Load everything, including Sema.
Definition ASTUnit.h:716
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:310
Abstract interface for a consumer of code-completion information.
CompilerInstance - Helper class for managing a single instance of the Clang compiler.
bool loadModuleFile(ModuleFileName FileName, serialization::ModuleFile *&LoadedModuleFile)
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...
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 createFileManager()
Create the file manager and replace any existing one with it.
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
IntrusiveRefCntPtr< llvm::vfs::FileSystem > getVirtualFileSystemPtr() 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
void createSourceManager()
Create the source manager and replace any existing one with it.
CompilerInvocation & getInvocation()
std::unique_ptr< ASTConsumer > takeASTConsumer()
takeASTConsumer - Remove the current AST consumer and give ownership to the caller.
PreprocessorOptions & getPreprocessorOpts()
void setFileManager(IntrusiveRefCntPtr< FileManager > Value)
Replace the current file manager.
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.
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.
std::string computeContextHash() const
Compute the context hash - a string that uniquely identifies compiler settings.
bool isFileContext() const
Definition DeclBase.h:2197
DeclContext * getLexicalParent()
getLexicalParent - Returns the containing lexical DeclContext.
Definition DeclBase.h:2142
SourceLocation getLocation() const
Definition DeclBase.h:447
const char * getDeclKindName() const
Definition DeclBase.cpp:169
DeclContext * getLexicalDeclContext()
getLexicalDeclContext - The declaration context where this Decl was lexically declared (LexicalDC).
Definition DeclBase.h:935
const LangOptions & getLangOpts() const LLVM_READONLY
Helper to get the language options from the ASTContext.
Definition DeclBase.cpp:553
virtual SourceRange getSourceRange() const LLVM_READONLY
Source range that this declaration covers.
Definition DeclBase.h:435
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:234
DiagnosticBuilder Report(SourceLocation Loc, unsigned DiagID)
Issue the message to the client.
bool hasErrorOccurred() const
Definition Diagnostic.h:882
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:52
OptionalFileEntryRef getOptionalFileRef(StringRef Filename, bool OpenFile=false, bool CacheFailure=true, bool IsText=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.
std::string ModuleCachePath
The directory used for the module cache.
unsigned DisableModuleHash
Whether we should disable the use of the hash string within the module cache.
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 getSpecificModuleCachePath() const
Retrieve the specific module cache path.
ModuleMap & getModuleMap()
Retrieve the module map.
bool parseAndLoadModuleMapFile(FileEntryRef File, bool IsSystem, bool ImplicitlyDiscovered, FileID ID=FileID(), unsigned *Offset=nullptr, StringRef OriginalModuleMapFile=StringRef())
Read the contents of the given module map file.
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.
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:438
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:461
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:542
static ModuleFileName makeExplicit(std::string Name)
Creates a file name for an explicit module.
Definition Module.h:142
void finishModuleDeclarationScope()
Creates a new declaration scope for module names, allowing previously defined modules to shadow defin...
Definition ModuleMap.h:638
bool canInferFrameworkModule(const DirectoryEntry *Dir) const
Check whether a framework module can be inferred in the given directory.
Definition ModuleMap.h:647
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:340
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:541
Module * Parent
The parent of this module.
Definition Module.h:389
unsigned IsSystem
Whether this is a "system" module (which assumes that all headers in it are system headers).
Definition Module.h:589
static StringRef getModuleInputBufferName()
Definition Module.h:1080
llvm::iterator_range< submodule_iterator > submodules()
Definition Module.h:1067
unsigned IsExternC
Whether this is an 'extern "C"' module (which implicitly puts all headers in it within an 'extern "C"...
Definition Module.h:595
std::optional< Header > getUmbrellaHeaderAsWritten() const
Retrieve the umbrella header as written.
Definition Module.h:985
OptionalDirectoryEntryRef Directory
The build directory of this module.
Definition Module.h:394
std::string PresumedModuleMapFile
The presumed file name for the module map defining this module.
Definition Module.h:398
ArrayRef< Header > getHeaders(HeaderKind HK) const
Definition Module.h:502
bool isAvailable() const
Determine whether this module is available for use within the current translation unit.
Definition Module.h:786
std::optional< DirectoryName > getUmbrellaDirAsWritten() const
Retrieve the umbrella directory as written.
Definition Module.h:977
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:240
void addTopHeader(FileEntryRef File)
Add a top-level header associated with this module.
Definition Module.cpp:272
@ 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:169
SourceLocation getLocation() const
Return a source location identifier for the specified offset in the current file.
Definition Token.h:142
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:104
tok::TokenKind getKind() const
Definition Token.h:99
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:158
bool StandardCXXModule
Whether this module file is a standard C++ module.
Definition ModuleFile.h:211
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:196
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:636
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
std::string createSpecificModuleCachePath(FileManager &FileMgr, StringRef ModuleCachePath, bool DisableModuleHash, std::string ContextHash)
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:487