clang 23.0.0git
FrontendActions.cpp
Go to the documentation of this file.
1//===--- FrontendActions.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
11#include "clang/AST/Decl.h"
15#include "clang/Basic/Module.h"
30#include "llvm/Config/llvm-config.h" // for LLVM_HOST_TRIPLE
31#include "llvm/Support/ErrorHandling.h"
32#include "llvm/Support/FileSystem.h"
33#include "llvm/Support/MemoryBuffer.h"
34#include "llvm/Support/YAMLTraits.h"
35#include "llvm/Support/raw_ostream.h"
36#include <memory>
37#include <optional>
38#include <system_error>
39
40using namespace clang;
41
42namespace {
43CodeCompleteConsumer *GetCodeCompletionConsumer(CompilerInstance &CI) {
45 : nullptr;
46}
47
48void EnsureSemaIsCreated(CompilerInstance &CI, FrontendAction &Action) {
49 if (Action.hasCodeCompletionSupport() &&
52
53 if (!CI.hasSema())
55 GetCodeCompletionConsumer(CI));
56}
57} // namespace
58
59//===----------------------------------------------------------------------===//
60// Custom Actions
61//===----------------------------------------------------------------------===//
62
63std::unique_ptr<ASTConsumer>
64InitOnlyAction::CreateASTConsumer(CompilerInstance &CI, StringRef InFile) {
65 return std::make_unique<ASTConsumer>();
66}
67
69}
70
71// Basically PreprocessOnlyAction::ExecuteAction.
73 Preprocessor &PP = getCompilerInstance().getPreprocessor();
74
75 // Ignore unknown pragmas.
76 PP.IgnorePragmas();
77
78 Token Tok;
79 // Start parsing the specified input file.
81 do {
82 PP.Lex(Tok);
83 } while (Tok.isNot(tok::eof));
84}
85
86std::unique_ptr<ASTConsumer>
87ReadPCHAndPreprocessAction::CreateASTConsumer(CompilerInstance &CI,
88 StringRef InFile) {
89 return std::make_unique<ASTConsumer>();
90}
91
92//===----------------------------------------------------------------------===//
93// AST Consumer Actions
94//===----------------------------------------------------------------------===//
95
96std::unique_ptr<ASTConsumer>
98 if (std::unique_ptr<raw_ostream> OS =
99 CI.createDefaultOutputFile(false, InFile))
100 return CreateASTPrinter(std::move(OS), CI.getFrontendOpts().ASTDumpFilter);
101 return nullptr;
102}
103
104std::unique_ptr<ASTConsumer>
106 const FrontendOptions &Opts = CI.getFrontendOpts();
107 return CreateASTDumper(nullptr /*Dump to stdout.*/, Opts.ASTDumpFilter,
108 Opts.ASTDumpDecls, Opts.ASTDumpAll,
110 Opts.ASTDumpFormat);
111}
112
113std::unique_ptr<ASTConsumer>
117
118std::unique_ptr<ASTConsumer>
120 return CreateASTViewer();
121}
122
123std::unique_ptr<ASTConsumer>
125 std::string Sysroot;
126 if (!ComputeASTConsumerArguments(CI, /*ref*/ Sysroot))
127 return nullptr;
128
129 std::string OutputFile;
130 std::unique_ptr<raw_pwrite_stream> OS =
131 CreateOutputFile(CI, InFile, /*ref*/ OutputFile);
132 if (!OS)
133 return nullptr;
134
136 Sysroot.clear();
137
138 const auto &FrontendOpts = CI.getFrontendOpts();
139 auto Buffer = std::make_shared<PCHBuffer>();
140 std::vector<std::unique_ptr<ASTConsumer>> Consumers;
141 Consumers.push_back(std::make_unique<PCHGenerator>(
142 CI.getPreprocessor(), CI.getModuleCache(), OutputFile, Sysroot, Buffer,
143 CI.getCodeGenOpts(), FrontendOpts.ModuleFileExtensions,
145 FrontendOpts.IncludeTimestamps, FrontendOpts.BuildingImplicitModule,
146 +CI.getLangOpts().CacheGeneratedPCH));
147 Consumers.push_back(CI.getPCHContainerWriter().CreatePCHContainerGenerator(
148 CI, std::string(InFile), OutputFile, std::move(OS), Buffer));
149
150 return std::make_unique<MultiplexConsumer>(std::move(Consumers));
151}
152
154 std::string &Sysroot) {
155 Sysroot = CI.getHeaderSearchOpts().Sysroot;
156 if (CI.getFrontendOpts().RelocatablePCH && Sysroot.empty()) {
157 CI.getDiagnostics().Report(diag::err_relocatable_without_isysroot);
158 return false;
159 }
160
161 return true;
162}
163
164std::unique_ptr<llvm::raw_pwrite_stream>
166 std::string &OutputFile) {
167 // Because this is exposed via libclang we must disable RemoveFileOnSignal.
168 std::unique_ptr<raw_pwrite_stream> OS = CI.createDefaultOutputFile(
169 /*Binary=*/true, InFile, /*Extension=*/"", /*RemoveFileOnSignal=*/false);
170 if (!OS)
171 return nullptr;
172
173 OutputFile = CI.getFrontendOpts().OutputFile;
174 return OS;
175}
176
178 if (getCompilerInstance().getPreprocessorOpts().AllowPCHWithCompilerErrors)
179 return false;
181}
182
187
188std::vector<std::unique_ptr<ASTConsumer>>
190 StringRef InFile) {
191 std::unique_ptr<raw_pwrite_stream> OS = CreateOutputFile(CI, InFile);
192 if (!OS)
193 return {};
194
195 std::string OutputFile = CI.getFrontendOpts().OutputFile;
196 std::string Sysroot;
197
198 auto Buffer = std::make_shared<PCHBuffer>();
199 std::vector<std::unique_ptr<ASTConsumer>> Consumers;
200
201 Consumers.push_back(std::make_unique<PCHGenerator>(
202 CI.getPreprocessor(), CI.getModuleCache(), OutputFile, Sysroot, Buffer,
204 /*AllowASTWithErrors=*/
206 /*IncludeTimestamps=*/
209 /*BuildingImplicitModule=*/+CI.getFrontendOpts().BuildingImplicitModule,
210 /*ShouldCacheASTInMemory=*/
212 Consumers.push_back(CI.getPCHContainerWriter().CreatePCHContainerGenerator(
213 CI, std::string(InFile), OutputFile, std::move(OS), Buffer));
214 return Consumers;
215}
216
217std::unique_ptr<ASTConsumer>
219 StringRef InFile) {
220 std::vector<std::unique_ptr<ASTConsumer>> Consumers =
221 CreateMultiplexConsumer(CI, InFile);
222 if (Consumers.empty())
223 return nullptr;
224
225 return std::make_unique<MultiplexConsumer>(std::move(Consumers));
226}
227
232
233bool GenerateModuleFromModuleMapAction::BeginSourceFileAction(
234 CompilerInstance &CI) {
235 if (!CI.getLangOpts().Modules) {
236 CI.getDiagnostics().Report(diag::err_module_build_requires_fmodules);
237 return false;
238 }
239
241}
242
243std::unique_ptr<raw_pwrite_stream>
244GenerateModuleFromModuleMapAction::CreateOutputFile(CompilerInstance &CI,
245 StringRef InFile) {
246 // If no output file was provided, figure out where this module would go
247 // in the module cache.
248 if (CI.getFrontendOpts().OutputFile.empty()) {
249 StringRef ModuleMapFile = CI.getFrontendOpts().OriginalModuleMap;
250 if (ModuleMapFile.empty())
251 ModuleMapFile = InFile;
252
254 ModuleFileName FileName = HS.getCachedModuleFileName(
255 CI.getLangOpts().CurrentModule, ModuleMapFile);
257 }
258
259 // Because this is exposed via libclang we must disable RemoveFileOnSignal.
260 return CI.createDefaultOutputFile(/*Binary=*/true, InFile, /*Extension=*/"",
261 /*RemoveFileOnSignal=*/false,
262 /*CreateMissingDirectories=*/true,
263 /*ForceUseTemporary=*/true);
264}
265
267 CompilerInstance &CI) {
268 for (const auto &FIF : CI.getFrontendOpts().Inputs) {
269 if (const auto InputFormat = FIF.getKind().getFormat();
270 InputFormat != InputKind::Format::Source) {
272 diag::err_frontend_action_unsupported_input_format)
273 << "module interface compilation" << FIF.getFile() << InputFormat;
274 return false;
275 }
276 }
278}
279
286
287std::unique_ptr<ASTConsumer>
289 StringRef InFile) {
290 std::vector<std::unique_ptr<ASTConsumer>> Consumers;
291
293 !CI.getFrontendOpts().ModuleOutputPath.empty()) {
294 Consumers.push_back(std::make_unique<ReducedBMIGenerator>(
298 }
299
300 Consumers.push_back(std::make_unique<CXX20ModulesGenerator>(
304
305 return std::make_unique<MultiplexConsumer>(std::move(Consumers));
306}
307
308std::unique_ptr<raw_pwrite_stream>
310 StringRef InFile) {
311 return CI.createDefaultOutputFile(/*Binary=*/true, InFile, "pcm");
312}
313
314std::unique_ptr<ASTConsumer>
315GenerateReducedModuleInterfaceAction::CreateASTConsumer(CompilerInstance &CI,
316 StringRef InFile) {
317 return std::make_unique<ReducedBMIGenerator>(
320}
321
322bool GenerateHeaderUnitAction::BeginSourceFileAction(CompilerInstance &CI) {
323 if (!CI.getLangOpts().CPlusPlusModules) {
324 CI.getDiagnostics().Report(diag::err_module_interface_requires_cpp_modules);
325 return false;
326 }
327 CI.getLangOpts().setCompilingModule(LangOptions::CMK_HeaderUnit);
329}
330
331std::unique_ptr<raw_pwrite_stream>
332GenerateHeaderUnitAction::CreateOutputFile(CompilerInstance &CI,
333 StringRef InFile) {
334 return CI.createDefaultOutputFile(/*Binary=*/true, InFile, "pcm");
335}
336
339
340std::unique_ptr<ASTConsumer>
342 return std::make_unique<ASTConsumer>();
343}
344
345std::unique_ptr<ASTConsumer>
347 StringRef InFile) {
348 return std::make_unique<ASTConsumer>();
349}
350
351std::unique_ptr<ASTConsumer>
353 return std::make_unique<ASTConsumer>();
354}
355
359 const std::string &Sysroot = CI.getHeaderSearchOpts().Sysroot;
360 std::unique_ptr<ASTReader> Reader(new ASTReader(
364 Sysroot.empty() ? "" : Sysroot.c_str(),
366 /*AllowASTWithCompilerErrors*/ false,
367 /*AllowConfigurationMismatch*/ true,
368 /*ValidateSystemInputs*/ true, /*ForceValidateUserInputs*/ true));
369
373}
374
375namespace {
376struct TemplightEntry {
377 std::string Name;
378 std::string Kind;
379 std::string Event;
380 std::string DefinitionLocation;
381 std::string PointOfInstantiation;
382};
383} // namespace
384
385namespace llvm {
386namespace yaml {
387template <> struct MappingTraits<TemplightEntry> {
388 static void mapping(IO &io, TemplightEntry &fields) {
389 io.mapRequired("name", fields.Name);
390 io.mapRequired("kind", fields.Kind);
391 io.mapRequired("event", fields.Event);
392 io.mapRequired("orig", fields.DefinitionLocation);
393 io.mapRequired("poi", fields.PointOfInstantiation);
394 }
395};
396} // namespace yaml
397} // namespace llvm
398
399namespace {
400class DefaultTemplateInstCallback : public TemplateInstantiationCallback {
401 using CodeSynthesisContext = Sema::CodeSynthesisContext;
402
403public:
404 void initialize(const Sema &) override {}
405
406 void finalize(const Sema &) override {}
407
408 void atTemplateBegin(const Sema &TheSema,
409 const CodeSynthesisContext &Inst) override {
410 displayTemplightEntry<true>(llvm::outs(), TheSema, Inst);
411 }
412
413 void atTemplateEnd(const Sema &TheSema,
414 const CodeSynthesisContext &Inst) override {
415 displayTemplightEntry<false>(llvm::outs(), TheSema, Inst);
416 }
417
418private:
419 static std::string toString(CodeSynthesisContext::SynthesisKind Kind) {
420 switch (Kind) {
421 case CodeSynthesisContext::TemplateInstantiation:
422 return "TemplateInstantiation";
423 case CodeSynthesisContext::DefaultTemplateArgumentInstantiation:
424 return "DefaultTemplateArgumentInstantiation";
425 case CodeSynthesisContext::DefaultFunctionArgumentInstantiation:
426 return "DefaultFunctionArgumentInstantiation";
427 case CodeSynthesisContext::ExplicitTemplateArgumentSubstitution:
428 return "ExplicitTemplateArgumentSubstitution";
429 case CodeSynthesisContext::DeducedTemplateArgumentSubstitution:
430 return "DeducedTemplateArgumentSubstitution";
431 case CodeSynthesisContext::LambdaExpressionSubstitution:
432 return "LambdaExpressionSubstitution";
433 case CodeSynthesisContext::PriorTemplateArgumentSubstitution:
434 return "PriorTemplateArgumentSubstitution";
435 case CodeSynthesisContext::DefaultTemplateArgumentChecking:
436 return "DefaultTemplateArgumentChecking";
437 case CodeSynthesisContext::ExceptionSpecEvaluation:
438 return "ExceptionSpecEvaluation";
439 case CodeSynthesisContext::ExceptionSpecInstantiation:
440 return "ExceptionSpecInstantiation";
441 case CodeSynthesisContext::DeclaringSpecialMember:
442 return "DeclaringSpecialMember";
443 case CodeSynthesisContext::DeclaringImplicitEqualityComparison:
444 return "DeclaringImplicitEqualityComparison";
445 case CodeSynthesisContext::DefiningSynthesizedFunction:
446 return "DefiningSynthesizedFunction";
447 case CodeSynthesisContext::RewritingOperatorAsSpaceship:
448 return "RewritingOperatorAsSpaceship";
449 case CodeSynthesisContext::Memoization:
450 return "Memoization";
451 case CodeSynthesisContext::ConstraintsCheck:
452 return "ConstraintsCheck";
453 case CodeSynthesisContext::ConstraintSubstitution:
454 return "ConstraintSubstitution";
455 case CodeSynthesisContext::ConstraintNormalization:
456 return "ConstraintNormalization";
457 case CodeSynthesisContext::RequirementParameterInstantiation:
458 return "RequirementParameterInstantiation";
459 case CodeSynthesisContext::ParameterMappingSubstitution:
460 return "ParameterMappingSubstitution";
461 case CodeSynthesisContext::RequirementInstantiation:
462 return "RequirementInstantiation";
463 case CodeSynthesisContext::NestedRequirementConstraintsCheck:
464 return "NestedRequirementConstraintsCheck";
465 case CodeSynthesisContext::InitializingStructuredBinding:
466 return "InitializingStructuredBinding";
467 case CodeSynthesisContext::MarkingClassDllexported:
468 return "MarkingClassDllexported";
469 case CodeSynthesisContext::BuildingBuiltinDumpStructCall:
470 return "BuildingBuiltinDumpStructCall";
471 case CodeSynthesisContext::BuildingDeductionGuides:
472 return "BuildingDeductionGuides";
473 case CodeSynthesisContext::TypeAliasTemplateInstantiation:
474 return "TypeAliasTemplateInstantiation";
475 case CodeSynthesisContext::PartialOrderingTTP:
476 return "PartialOrderingTTP";
477 case CodeSynthesisContext::SYCLKernelLaunchLookup:
478 return "SYCLKernelLaunchLookup";
479 case CodeSynthesisContext::SYCLKernelLaunchOverloadResolution:
480 return "SYCLKernelLaunchOverloadResolution";
481 }
482 return "";
483 }
484
485 template <bool BeginInstantiation>
486 static void displayTemplightEntry(llvm::raw_ostream &Out, const Sema &TheSema,
487 const CodeSynthesisContext &Inst) {
488 std::string YAML;
489 {
490 llvm::raw_string_ostream OS(YAML);
491 llvm::yaml::Output YO(OS);
492 TemplightEntry Entry =
493 getTemplightEntry<BeginInstantiation>(TheSema, Inst);
494 llvm::yaml::EmptyContext Context;
495 llvm::yaml::yamlize(YO, Entry, true, Context);
496 }
497 Out << "---" << YAML << "\n";
498 }
499
500 static void printEntryName(const Sema &TheSema, const Decl *Entity,
501 llvm::raw_string_ostream &OS) {
502 auto *NamedTemplate = cast<NamedDecl>(Entity);
503
504 PrintingPolicy Policy = TheSema.Context.getPrintingPolicy();
505 // FIXME: Also ask for FullyQualifiedNames?
506 Policy.SuppressDefaultTemplateArgs = false;
507 NamedTemplate->getNameForDiagnostic(OS, Policy, true);
508
509 if (!OS.str().empty())
510 return;
511
512 Decl *Ctx = Decl::castFromDeclContext(NamedTemplate->getDeclContext());
513 NamedDecl *NamedCtx = dyn_cast_or_null<NamedDecl>(Ctx);
514
515 if (const auto *Decl = dyn_cast<TagDecl>(NamedTemplate)) {
516 if (const auto *R = dyn_cast<RecordDecl>(Decl)) {
517 if (R->isLambda()) {
518 OS << "lambda at ";
519 Decl->getLocation().print(OS, TheSema.getSourceManager());
520 return;
521 }
522 }
523 OS << "unnamed " << Decl->getKindName();
524 return;
525 }
526
527 assert(NamedCtx && "NamedCtx cannot be null");
528
529 if (const auto *Decl = dyn_cast<ParmVarDecl>(NamedTemplate)) {
530 OS << "unnamed function parameter " << Decl->getFunctionScopeIndex()
531 << " ";
532 if (Decl->getFunctionScopeDepth() > 0)
533 OS << "(at depth " << Decl->getFunctionScopeDepth() << ") ";
534 OS << "of ";
535 NamedCtx->getNameForDiagnostic(OS, TheSema.getLangOpts(), true);
536 return;
537 }
538
539 if (const auto *Decl = dyn_cast<TemplateTypeParmDecl>(NamedTemplate)) {
540 if (const Type *Ty = Decl->getTypeForDecl()) {
541 if (const auto *TTPT = dyn_cast_or_null<TemplateTypeParmType>(Ty)) {
542 OS << "unnamed template type parameter " << TTPT->getIndex() << " ";
543 if (TTPT->getDepth() > 0)
544 OS << "(at depth " << TTPT->getDepth() << ") ";
545 OS << "of ";
546 NamedCtx->getNameForDiagnostic(OS, TheSema.getLangOpts(), true);
547 return;
548 }
549 }
550 }
551
552 if (const auto *Decl = dyn_cast<NonTypeTemplateParmDecl>(NamedTemplate)) {
553 OS << "unnamed template non-type parameter " << Decl->getIndex() << " ";
554 if (Decl->getDepth() > 0)
555 OS << "(at depth " << Decl->getDepth() << ") ";
556 OS << "of ";
557 NamedCtx->getNameForDiagnostic(OS, TheSema.getLangOpts(), true);
558 return;
559 }
560
561 if (const auto *Decl = dyn_cast<TemplateTemplateParmDecl>(NamedTemplate)) {
562 OS << "unnamed template template parameter " << Decl->getIndex() << " ";
563 if (Decl->getDepth() > 0)
564 OS << "(at depth " << Decl->getDepth() << ") ";
565 OS << "of ";
566 NamedCtx->getNameForDiagnostic(OS, TheSema.getLangOpts(), true);
567 return;
568 }
569
570 llvm_unreachable("Failed to retrieve a name for this entry!");
571 OS << "unnamed identifier";
572 }
573
574 template <bool BeginInstantiation>
575 static TemplightEntry getTemplightEntry(const Sema &TheSema,
576 const CodeSynthesisContext &Inst) {
577 TemplightEntry Entry;
578 Entry.Kind = toString(Inst.Kind);
579 Entry.Event = BeginInstantiation ? "Begin" : "End";
580 llvm::raw_string_ostream OS(Entry.Name);
581 printEntryName(TheSema, Inst.Entity, OS);
582 const PresumedLoc DefLoc =
583 TheSema.getSourceManager().getPresumedLoc(Inst.Entity->getLocation());
584 if (!DefLoc.isInvalid())
585 Entry.DefinitionLocation = std::string(DefLoc.getFilename()) + ":" +
586 std::to_string(DefLoc.getLine()) + ":" +
587 std::to_string(DefLoc.getColumn());
588 const PresumedLoc PoiLoc =
589 TheSema.getSourceManager().getPresumedLoc(Inst.PointOfInstantiation);
590 if (!PoiLoc.isInvalid()) {
591 Entry.PointOfInstantiation = std::string(PoiLoc.getFilename()) + ":" +
592 std::to_string(PoiLoc.getLine()) + ":" +
593 std::to_string(PoiLoc.getColumn());
594 }
595 return Entry;
596 }
597};
598} // namespace
599
600std::unique_ptr<ASTConsumer>
602 return std::make_unique<ASTConsumer>();
603}
604
607
608 // This part is normally done by ASTFrontEndAction, but needs to happen
609 // before Templight observers can be created
610 // FIXME: Move the truncation aspect of this into Sema, we delayed this till
611 // here so the source manager would be initialized.
612 EnsureSemaIsCreated(CI, *this);
613
614 CI.getSema().TemplateInstCallbacks.push_back(
615 std::make_unique<DefaultTemplateInstCallback>());
617}
618
619namespace {
620 /// AST reader listener that dumps module information for a module
621 /// file.
622 class DumpModuleInfoListener : public ASTReaderListener {
623 llvm::raw_ostream &Out;
625
626 public:
627 DumpModuleInfoListener(llvm::raw_ostream &Out, FileManager &FileMgr)
628 : Out(Out), FileMgr(FileMgr) {}
629
630#define DUMP_BOOLEAN(Value, Text) \
631 Out.indent(4) << Text << ": " << (Value? "Yes" : "No") << "\n"
632
633 bool ReadFullVersionInformation(StringRef FullVersion) override {
634 Out.indent(2)
635 << "Generated by "
636 << (FullVersion == getClangFullRepositoryVersion()? "this"
637 : "a different")
638 << " Clang: " << FullVersion << "\n";
640 }
641
642 void ReadModuleName(StringRef ModuleName) override {
643 Out.indent(2) << "Module name: " << ModuleName << "\n";
644 }
645 void ReadModuleMapFile(StringRef ModuleMapPath) override {
646 Out.indent(2) << "Module map file: " << ModuleMapPath << "\n";
647 }
648
649 bool ReadLanguageOptions(const LangOptions &LangOpts,
650 StringRef ModuleFilename, bool Complain,
651 bool AllowCompatibleDifferences) override {
652 // FIXME: Replace with C++20 `using enum LangOptions::CompatibilityKind`.
653 using CK = LangOptions::CompatibilityKind;
654
655 Out.indent(2) << "Language options:\n";
656#define LANGOPT(Name, Bits, Default, Compatibility, Description) \
657 if constexpr (CK::Compatibility != CK::Benign) \
658 DUMP_BOOLEAN(LangOpts.Name, Description);
659#define ENUM_LANGOPT(Name, Type, Bits, Default, Compatibility, Description) \
660 if constexpr (CK::Compatibility != CK::Benign) \
661 Out.indent(4) << Description << ": " \
662 << static_cast<unsigned>(LangOpts.get##Name()) << "\n";
663#define VALUE_LANGOPT(Name, Bits, Default, Compatibility, Description) \
664 if constexpr (CK::Compatibility != CK::Benign) \
665 Out.indent(4) << Description << ": " << LangOpts.Name << "\n";
666#include "clang/Basic/LangOptions.def"
667
668 if (!LangOpts.ModuleFeatures.empty()) {
669 Out.indent(4) << "Module features:\n";
670 for (StringRef Feature : LangOpts.ModuleFeatures)
671 Out.indent(6) << Feature << "\n";
672 }
673
674 return false;
675 }
676
677 bool ReadTargetOptions(const TargetOptions &TargetOpts,
678 StringRef ModuleFilename, bool Complain,
679 bool AllowCompatibleDifferences) override {
680 Out.indent(2) << "Target options:\n";
681 Out.indent(4) << " Triple: " << TargetOpts.Triple << "\n";
682 Out.indent(4) << " CPU: " << TargetOpts.CPU << "\n";
683 Out.indent(4) << " TuneCPU: " << TargetOpts.TuneCPU << "\n";
684 Out.indent(4) << " ABI: " << TargetOpts.ABI << "\n";
685
686 if (!TargetOpts.FeaturesAsWritten.empty()) {
687 Out.indent(4) << "Target features:\n";
688 for (unsigned I = 0, N = TargetOpts.FeaturesAsWritten.size();
689 I != N; ++I) {
690 Out.indent(6) << TargetOpts.FeaturesAsWritten[I] << "\n";
691 }
692 }
693
694 return false;
695 }
696
697 bool ReadDiagnosticOptions(DiagnosticOptions &DiagOpts,
698 StringRef ModuleFilename,
699 bool Complain) override {
700 Out.indent(2) << "Diagnostic options:\n";
701#define DIAGOPT(Name, Bits, Default) DUMP_BOOLEAN(DiagOpts.Name, #Name);
702#define ENUM_DIAGOPT(Name, Type, Bits, Default) \
703 Out.indent(4) << #Name << ": " << DiagOpts.get##Name() << "\n";
704#define VALUE_DIAGOPT(Name, Bits, Default) \
705 Out.indent(4) << #Name << ": " << DiagOpts.Name << "\n";
706#include "clang/Basic/DiagnosticOptions.def"
707
708 Out.indent(4) << "Diagnostic flags:\n";
709 for (const std::string &Warning : DiagOpts.Warnings)
710 Out.indent(6) << "-W" << Warning << "\n";
711 for (const std::string &Remark : DiagOpts.Remarks)
712 Out.indent(6) << "-R" << Remark << "\n";
713
714 return false;
715 }
716
717 bool ReadHeaderSearchOptions(const HeaderSearchOptions &HSOpts,
718 StringRef ModuleFilename,
719 StringRef ContextHash,
720 bool Complain) override {
721 std::string SpecificModuleCachePath = createSpecificModuleCachePath(
722 FileMgr, HSOpts.ModuleCachePath, HSOpts.DisableModuleHash,
723 std::string(ContextHash));
724
725 Out.indent(2) << "Header search options:\n";
726 Out.indent(4) << "System root [-isysroot=]: '" << HSOpts.Sysroot << "'\n";
727 Out.indent(4) << "Resource dir [ -resource-dir=]: '" << HSOpts.ResourceDir << "'\n";
728 Out.indent(4) << "Module Cache: '" << SpecificModuleCachePath << "'\n";
730 "Use builtin include directories [-nobuiltininc]");
732 "Use standard system include directories [-nostdinc]");
734 "Use standard C++ include directories [-nostdinc++]");
735 DUMP_BOOLEAN(HSOpts.UseLibcxx,
736 "Use libc++ (rather than libstdc++) [-stdlib=]");
737 return false;
738 }
739
740 bool ReadHeaderSearchPaths(const HeaderSearchOptions &HSOpts,
741 bool Complain) override {
742 Out.indent(2) << "Header search paths:\n";
743 Out.indent(4) << "User entries:\n";
744 for (const auto &Entry : HSOpts.UserEntries)
745 Out.indent(6) << Entry.Path << "\n";
746 Out.indent(4) << "System header prefixes:\n";
747 for (const auto &Prefix : HSOpts.SystemHeaderPrefixes)
748 Out.indent(6) << Prefix.Prefix << "\n";
749 Out.indent(4) << "VFS overlay files:\n";
750 for (const auto &Overlay : HSOpts.VFSOverlayFiles)
751 Out.indent(6) << Overlay << "\n";
752 return false;
753 }
754
755 bool ReadPreprocessorOptions(const PreprocessorOptions &PPOpts,
756 StringRef ModuleFilename, bool ReadMacros,
757 bool Complain,
758 std::string &SuggestedPredefines) override {
759 Out.indent(2) << "Preprocessor options:\n";
761 "Uses compiler/target-specific predefines [-undef]");
763 "Uses detailed preprocessing record (for indexing)");
764
765 if (ReadMacros) {
766 Out.indent(4) << "Predefined macros:\n";
767 }
768
769 for (std::vector<std::pair<std::string, bool/*isUndef*/> >::const_iterator
770 I = PPOpts.Macros.begin(), IEnd = PPOpts.Macros.end();
771 I != IEnd; ++I) {
772 Out.indent(6);
773 if (I->second)
774 Out << "-U";
775 else
776 Out << "-D";
777 Out << I->first << "\n";
778 }
779 return false;
780 }
781
782 /// Indicates that a particular module file extension has been read.
783 void readModuleFileExtension(
784 const ModuleFileExtensionMetadata &Metadata) override {
785 Out.indent(2) << "Module file extension '"
786 << Metadata.BlockName << "' " << Metadata.MajorVersion
787 << "." << Metadata.MinorVersion;
788 if (!Metadata.UserInfo.empty()) {
789 Out << ": ";
790 Out.write_escaped(Metadata.UserInfo);
791 }
792
793 Out << "\n";
794 }
795
796 /// Tells the \c ASTReaderListener that we want to receive the
797 /// input files of the AST file via \c visitInputFile.
798 bool needsInputFileVisitation() override { return true; }
799
800 /// Tells the \c ASTReaderListener that we want to receive the
801 /// input files of the AST file via \c visitInputFile.
802 bool needsSystemInputFileVisitation() override { return true; }
803
804 /// Indicates that the AST file contains particular input file.
805 ///
806 /// \returns true to continue receiving the next input file, false to stop.
807 bool visitInputFileAsRequested(StringRef FilenameAsRequested,
808 StringRef Filename, bool isSystem,
809 bool isOverridden, time_t StoredTime,
810 bool isExplicitModule) override {
811
812 Out.indent(2) << "Input file: " << FilenameAsRequested;
813
814 if (isSystem || isOverridden || isExplicitModule) {
815 Out << " [";
816 if (isSystem) {
817 Out << "System";
818 if (isOverridden || isExplicitModule)
819 Out << ", ";
820 }
821 if (isOverridden) {
822 Out << "Overridden";
823 if (isExplicitModule)
824 Out << ", ";
825 }
826 if (isExplicitModule)
827 Out << "ExplicitModule";
828
829 Out << "]";
830 }
831
832 Out << "\n";
833
834 if (StoredTime > 0)
835 Out.indent(4) << "MTime: " << llvm::itostr(StoredTime) << "\n";
836
837 return true;
838 }
839
840 /// Returns true if this \c ASTReaderListener wants to receive the
841 /// imports of the AST file via \c visitImport, false otherwise.
842 bool needsImportVisitation() const override { return true; }
843
844 /// If needsImportVisitation returns \c true, this is called for each
845 /// AST file imported by this AST file.
846 void visitImport(StringRef ModuleName, StringRef Filename) override {
847 Out.indent(2) << "Imports module '" << ModuleName
848 << "': " << Filename.str() << "\n";
849 }
850#undef DUMP_BOOLEAN
851 };
852}
853
855 // The Object file reader also supports raw ast files and there is no point in
856 // being strict about the module file format in -module-file-info mode.
858 return true;
859}
860
861static StringRef ModuleKindName(Module::ModuleKind MK) {
862 switch (MK) {
864 return "Module Map Module";
866 return "Interface Unit";
868 return "Implementation Unit";
870 return "Partition Interface";
872 return "Partition Implementation";
874 return "Header Unit";
876 return "Global Module Fragment";
878 return "Implicit Module Fragment";
880 return "Private Module Fragment";
881 }
882 llvm_unreachable("unknown module kind!");
883}
884
887
888 // Don't process files of type other than module to avoid crash
889 if (!isCurrentFileAST()) {
890 CI.getDiagnostics().Report(diag::err_file_is_not_module)
891 << getCurrentFile();
892 return;
893 }
894
895 // Set up the output file.
896 StringRef OutputFileName = CI.getFrontendOpts().OutputFile;
897 if (!OutputFileName.empty() && OutputFileName != "-") {
898 std::error_code EC;
899 OutputStream.reset(new llvm::raw_fd_ostream(
900 OutputFileName.str(), EC, llvm::sys::fs::OF_TextWithCRLF));
901 }
902 llvm::raw_ostream &Out = OutputStream ? *OutputStream : llvm::outs();
903
904 Out << "Information for module file '" << getCurrentFile() << "':\n";
905 auto &FileMgr = CI.getFileManager();
906 auto Buffer = FileMgr.getBufferForFile(getCurrentFile());
907 StringRef Magic = (*Buffer)->getMemBufferRef().getBuffer();
908 bool IsRaw = Magic.starts_with("CPCH");
909 Out << " Module format: " << (IsRaw ? "raw" : "obj") << "\n";
910
911 Preprocessor &PP = CI.getPreprocessor();
912 DumpModuleInfoListener Listener(Out, CI.getFileManager());
913 const HeaderSearchOptions &HSOpts =
914 PP.getHeaderSearchInfo().getHeaderSearchOpts();
915
916 // The FrontendAction::BeginSourceFile () method loads the AST so that much
917 // of the information is already available and modules should have been
918 // loaded.
919
921 if (LO.CPlusPlusModules && !LO.CurrentModule.empty()) {
923 unsigned SubModuleCount = R->getTotalNumSubmodules();
924 serialization::ModuleFile &MF = R->getModuleManager().getPrimaryModule();
925 Out << " ====== C++20 Module structure ======\n";
926
927 if (MF.ModuleName != LO.CurrentModule)
928 Out << " Mismatched module names : " << MF.ModuleName << " and "
929 << LO.CurrentModule << "\n";
930
931 struct SubModInfo {
932 unsigned Idx;
933 Module *Mod;
935 std::string &Name;
936 bool Seen;
937 };
938 std::map<std::string, SubModInfo> SubModMap;
939 auto PrintSubMapEntry = [&](std::string Name, Module::ModuleKind Kind) {
940 Out << " " << ModuleKindName(Kind) << " '" << Name << "'";
941 auto I = SubModMap.find(Name);
942 if (I == SubModMap.end())
943 Out << " was not found in the sub modules!\n";
944 else {
945 I->second.Seen = true;
946 Out << " is at index #" << I->second.Idx << "\n";
947 }
948 };
949 Module *Primary = nullptr;
950 for (unsigned Idx = 0; Idx <= SubModuleCount; ++Idx) {
951 Module *M = R->getModule(Idx);
952 if (!M)
953 continue;
954 if (M->Name == LO.CurrentModule) {
955 Primary = M;
956 Out << " " << ModuleKindName(M->Kind) << " '" << LO.CurrentModule
957 << "' is the Primary Module at index #" << Idx << "\n";
958 SubModMap.insert({M->Name, {Idx, M, M->Kind, M->Name, true}});
959 } else
960 SubModMap.insert({M->Name, {Idx, M, M->Kind, M->Name, false}});
961 }
962 if (Primary) {
963 if (!Primary->submodules().empty())
964 Out << " Sub Modules:\n";
965 for (auto *MI : Primary->submodules()) {
966 PrintSubMapEntry(MI->Name, MI->Kind);
967 }
968 if (!Primary->Imports.empty())
969 Out << " Imports:\n";
970 for (auto *IMP : Primary->Imports) {
971 PrintSubMapEntry(IMP->Name, IMP->Kind);
972 }
973 if (!Primary->Exports.empty())
974 Out << " Exports:\n";
975 for (unsigned MN = 0, N = Primary->Exports.size(); MN != N; ++MN) {
976 if (Module *M = Primary->Exports[MN].getPointer()) {
977 PrintSubMapEntry(M->Name, M->Kind);
978 }
979 }
980 }
981
982 // Emit the macro definitions in the module file so that we can know how
983 // much definitions in the module file quickly.
984 // TODO: Emit the macro definition bodies completely.
985 {
986 std::vector<StringRef> MacroNames;
987 for (const auto &M : R->getPreprocessor().macros()) {
988 if (M.first->isFromAST())
989 MacroNames.push_back(M.first->getName());
990 }
991 llvm::sort(MacroNames);
992 if (!MacroNames.empty())
993 Out << " Macro Definitions:\n";
994 for (StringRef Name : MacroNames)
995 Out << " " << Name << "\n";
996 }
997
998 // Now let's print out any modules we did not see as part of the Primary.
999 for (const auto &SM : SubModMap) {
1000 if (!SM.second.Seen && SM.second.Mod) {
1001 Out << " " << ModuleKindName(SM.second.Kind) << " '" << SM.first
1002 << "' at index #" << SM.second.Idx
1003 << " has no direct reference in the Primary\n";
1004 }
1005 }
1006 Out << " ====== ======\n";
1007 }
1008
1009 // The reminder of the output is produced from the listener as the AST
1010 // FileCcontrolBlock is (re-)parsed.
1014 /*FindModuleFileExtensions=*/true, Listener,
1016}
1017
1018//===----------------------------------------------------------------------===//
1019// Preprocessor Actions
1020//===----------------------------------------------------------------------===//
1021
1025
1026 // Start lexing the specified input file.
1027 llvm::MemoryBufferRef FromFile = SM.getBufferOrFake(SM.getMainFileID());
1028 Lexer RawLex(SM.getMainFileID(), FromFile, SM, PP.getLangOpts());
1029 RawLex.SetKeepWhitespaceMode(true);
1030
1031 Token RawTok;
1032 RawLex.LexFromRawLexer(RawTok);
1033 while (RawTok.isNot(tok::eof)) {
1034 PP.DumpToken(RawTok, true);
1035 llvm::errs() << "\n";
1036 RawLex.LexFromRawLexer(RawTok);
1037 }
1038}
1039
1042 // Start preprocessing the specified input file.
1043 Token Tok;
1045 do {
1046 PP.Lex(Tok);
1047 PP.DumpToken(Tok, true);
1048 llvm::errs() << "\n";
1049 } while (Tok.isNot(tok::eof));
1050}
1051
1054
1055 // Ignore unknown pragmas.
1056 PP.IgnorePragmas();
1057
1058 Token Tok;
1059 // Start parsing the specified input file.
1061 do {
1062 PP.Lex(Tok);
1063 } while (Tok.isNot(tok::eof));
1064}
1065
1068 // Output file may need to be set to 'Binary', to avoid converting Unix style
1069 // line feeds (<LF>) to Microsoft style line feeds (<CR><LF>) on Windows.
1070 //
1071 // Look to see what type of line endings the file uses. If there's a
1072 // CRLF, then we won't open the file up in binary mode. If there is
1073 // just an LF or CR, then we will open the file up in binary mode.
1074 // In this fashion, the output format should match the input format, unless
1075 // the input format has inconsistent line endings.
1076 //
1077 // This should be a relatively fast operation since most files won't have
1078 // all of their source code on a single line. However, that is still a
1079 // concern, so if we scan for too long, we'll just assume the file should
1080 // be opened in binary mode.
1081
1082 bool BinaryMode = false;
1083 if (llvm::Triple(LLVM_HOST_TRIPLE).isOSWindows()) {
1084 BinaryMode = true;
1085 const SourceManager &SM = CI.getSourceManager();
1086 if (std::optional<llvm::MemoryBufferRef> Buffer =
1087 SM.getBufferOrNone(SM.getMainFileID())) {
1088 const char *cur = Buffer->getBufferStart();
1089 const char *end = Buffer->getBufferEnd();
1090 const char *next = (cur != end) ? cur + 1 : end;
1091
1092 // Limit ourselves to only scanning 256 characters into the source
1093 // file. This is mostly a check in case the file has no
1094 // newlines whatsoever.
1095 if (end - cur > 256)
1096 end = cur + 256;
1097
1098 while (next < end) {
1099 if (*cur == 0x0D) { // CR
1100 if (*next == 0x0A) // CRLF
1101 BinaryMode = false;
1102
1103 break;
1104 } else if (*cur == 0x0A) // LF
1105 break;
1106
1107 ++cur;
1108 ++next;
1109 }
1110 }
1111 }
1112
1113 std::unique_ptr<raw_ostream> OS =
1115 if (!OS) return;
1116
1117 // If we're preprocessing a module map, start by dumping the contents of the
1118 // module itself before switching to the input buffer.
1119 auto &Input = getCurrentInput();
1120 if (Input.getKind().getFormat() == InputKind::ModuleMap) {
1121 if (Input.isFile()) {
1122 (*OS) << "# 1 \"";
1123 OS->write_escaped(Input.getFile());
1124 (*OS) << "\"\n";
1125 }
1126 getCurrentModule()->print(*OS);
1127 (*OS) << "#pragma clang module contents\n";
1128 }
1129
1132}
1133
1135 switch (getCurrentFileKind().getLanguage()) {
1136 case Language::C:
1137 case Language::CXX:
1138 case Language::ObjC:
1139 case Language::ObjCXX:
1140 case Language::OpenCL:
1142 case Language::CUDA:
1143 case Language::HIP:
1144 case Language::HLSL:
1145 case Language::CIR:
1146 break;
1147
1148 case Language::Unknown:
1149 case Language::Asm:
1150 case Language::LLVM_IR:
1151 // We can't do anything with these.
1152 return;
1153 }
1154
1155 // We don't expect to find any #include directives in a preprocessed input.
1156 if (getCurrentFileKind().isPreprocessed())
1157 return;
1158
1160 auto Buffer = CI.getFileManager().getBufferForFile(getCurrentFile());
1161 if (Buffer) {
1162 unsigned Preamble =
1163 Lexer::ComputePreamble((*Buffer)->getBuffer(), CI.getLangOpts()).Size;
1164 llvm::outs().write((*Buffer)->getBufferStart(), Preamble);
1165 }
1166}
1167
1170 std::unique_ptr<raw_ostream> OSP =
1172 if (!OSP)
1173 return;
1174
1175 raw_ostream &OS = *OSP;
1176 const Preprocessor &PP = CI.getPreprocessor();
1177 const LangOptions &LangOpts = PP.getLangOpts();
1178
1179 // FIXME: Rather than manually format the JSON (which is awkward due to
1180 // needing to remove trailing commas), this should make use of a JSON library.
1181 // FIXME: Instead of printing enums as an integral value and specifying the
1182 // type as a separate field, use introspection to print the enumerator.
1183
1184 OS << "{\n";
1185 OS << "\n\"features\" : [\n";
1186 {
1188#define FEATURE(Name, Predicate) \
1189 ("\t{\"" #Name "\" : " + llvm::Twine(Predicate ? "true" : "false") + "},\n") \
1190 .toVector(Str);
1191#include "clang/Basic/Features.def"
1192#undef FEATURE
1193 // Remove the newline and comma from the last entry to ensure this remains
1194 // valid JSON.
1195 OS << Str.substr(0, Str.size() - 2);
1196 }
1197 OS << "\n],\n";
1198
1199 OS << "\n\"extensions\" : [\n";
1200 {
1202#define EXTENSION(Name, Predicate) \
1203 ("\t{\"" #Name "\" : " + llvm::Twine(Predicate ? "true" : "false") + "},\n") \
1204 .toVector(Str);
1205#include "clang/Basic/Features.def"
1206#undef EXTENSION
1207 // Remove the newline and comma from the last entry to ensure this remains
1208 // valid JSON.
1209 OS << Str.substr(0, Str.size() - 2);
1210 }
1211 OS << "\n]\n";
1212
1213 OS << "}";
1214}
1215
1219 llvm::MemoryBufferRef FromFile = SM.getBufferOrFake(SM.getMainFileID());
1220
1224 FromFile.getBuffer(), Tokens, Directives, &CI.getDiagnostics(),
1225 SM.getLocForStartOfFile(SM.getMainFileID()))) {
1226 assert(CI.getDiagnostics().hasErrorOccurred() &&
1227 "no errors reported for failure");
1228
1229 // Preprocess the source when verifying the diagnostics to capture the
1230 // 'expected' comments.
1231 if (CI.getDiagnosticOpts().VerifyDiagnostics) {
1232 // Make sure we don't emit new diagnostics!
1236 Token Tok;
1237 do {
1238 PP.Lex(Tok);
1239 } while (Tok.isNot(tok::eof));
1240 }
1241 return;
1242 }
1243 printDependencyDirectivesAsSource(FromFile.getBuffer(), Directives,
1244 llvm::outs());
1245}
1246
1247//===----------------------------------------------------------------------===//
1248// HLSL Specific Actions
1249//===----------------------------------------------------------------------===//
1250
1252private:
1253 Sema &Actions;
1254 StringRef RootSigName;
1255 llvm::dxbc::RootSignatureVersion Version;
1256
1257 std::optional<StringLiteral *> processStringLiteral(ArrayRef<Token> Tokens) {
1258 for (Token Tok : Tokens)
1259 if (!tok::isStringLiteral(Tok.getKind()))
1260 return std::nullopt;
1261
1262 ExprResult StringResult = Actions.ActOnUnevaluatedStringLiteral(Tokens);
1263 if (StringResult.isInvalid())
1264 return std::nullopt;
1265
1266 if (auto Signature = dyn_cast<StringLiteral>(StringResult.get()))
1267 return Signature;
1268
1269 return std::nullopt;
1270 }
1271
1272public:
1273 void MacroDefined(const Token &MacroNameTok,
1274 const MacroDirective *MD) override {
1275 if (RootSigName != MacroNameTok.getIdentifierInfo()->getName())
1276 return;
1277
1278 const MacroInfo *MI = MD->getMacroInfo();
1279 auto Signature = processStringLiteral(MI->tokens());
1280 if (!Signature.has_value()) {
1281 Actions.getDiagnostics().Report(MI->getDefinitionLoc(),
1282 diag::err_expected_string_literal)
1283 << /*in attributes...*/ 4 << "RootSignature";
1284 return;
1285 }
1286
1287 IdentifierInfo *DeclIdent =
1288 hlsl::ParseHLSLRootSignature(Actions, Version, *Signature);
1289 Actions.HLSL().SetRootSignatureOverride(DeclIdent);
1290 }
1291
1292 InjectRootSignatureCallback(Sema &Actions, StringRef RootSigName,
1293 llvm::dxbc::RootSignatureVersion Version)
1294 : PPCallbacks(), Actions(Actions), RootSigName(RootSigName),
1295 Version(Version) {}
1296};
1297
1299 // Pre-requisites to invoke
1301 if (!CI.hasASTContext() || !CI.hasPreprocessor())
1303
1304 // InjectRootSignatureCallback requires access to invoke Sema to lookup/
1305 // register a root signature declaration. The wrapped action is required to
1306 // account for this by only creating a Sema if one doesn't already exist
1307 // (like we have done, and, ASTFrontendAction::ExecuteAction)
1308 if (!CI.hasSema())
1310 /*CodeCompleteConsumer=*/nullptr);
1311 Sema &S = CI.getSema();
1312
1313 auto &TargetInfo = CI.getASTContext().getTargetInfo();
1314 bool IsRootSignatureTarget =
1315 TargetInfo.getTriple().getEnvironment() == llvm::Triple::RootSignature;
1316 StringRef HLSLEntry = TargetInfo.getTargetOpts().HLSLEntry;
1317
1318 // Register HLSL specific callbacks
1319 auto LangOpts = CI.getLangOpts();
1320 StringRef RootSigName =
1321 IsRootSignatureTarget ? HLSLEntry : LangOpts.HLSLRootSigOverride;
1322
1323 auto MacroCallback = std::make_unique<InjectRootSignatureCallback>(
1324 S, RootSigName, LangOpts.HLSLRootSigVer);
1325
1326 Preprocessor &PP = CI.getPreprocessor();
1327 PP.addPPCallbacks(std::move(MacroCallback));
1328
1329 // If we are targeting a root signature, invoke custom handling
1330 if (IsRootSignatureTarget)
1331 return hlsl::HandleRootSignatureTarget(S, HLSLEntry);
1332 else // otherwise, invoke as normal
1334}
1335
1337 std::unique_ptr<FrontendAction> WrappedAction)
This is the interface for scanning header and source files to get the minimum necessary preprocessor ...
Defines the clang::FileManager interface and associated types.
Token Tok
The Token.
#define DUMP_BOOLEAN(Value, Text)
static StringRef ModuleKindName(Module::ModuleKind MK)
Defines the clang::Module class, which describes a module in the source code.
#define SM(sm)
Defines the clang::Preprocessor interface.
static std::string toString(const clang::SanitizerSet &Sanitizers)
Produce a string containing comma-separated names of sanitizers in Sanitizers set.
void MacroDefined(const Token &MacroNameTok, const MacroDirective *MD) override
Hook called whenever a macro definition is seen.
InjectRootSignatureCallback(Sema &Actions, StringRef RootSigName, llvm::dxbc::RootSignatureVersion Version)
const clang::PrintingPolicy & getPrintingPolicy() const
Definition ASTContext.h:851
const TargetInfo & getTargetInfo() const
Definition ASTContext.h:917
std::unique_ptr< ASTConsumer > CreateASTConsumer(CompilerInstance &CI, StringRef InFile) override
Create the AST consumer object for this action, if supported.
std::unique_ptr< ASTConsumer > CreateASTConsumer(CompilerInstance &CI, StringRef InFile) override
Create the AST consumer object for this action, if supported.
void ExecuteAction() override
Implement the ExecuteAction interface by running Sema on the already-initialized AST consumer.
std::unique_ptr< ASTConsumer > CreateASTConsumer(CompilerInstance &CI, StringRef InFile) override
Create the AST consumer object for this action, if supported.
Abstract interface for callback invocations by the ASTReader.
Definition ASTReader.h:117
virtual bool ReadFullVersionInformation(StringRef FullVersion)
Receives the full Clang version information.
Definition ASTReader.h:125
Reads an AST files chain containing the contents of a translation unit.
Definition ASTReader.h:427
@ ARR_ConfigurationMismatch
The client can handle an AST file that cannot load because it's compiled configuration doesn't match ...
Definition ASTReader.h:1860
static bool readASTFileControlBlock(StringRef Filename, FileManager &FileMgr, const ModuleCache &ModCache, const PCHContainerReader &PCHContainerRdr, bool FindModuleFileExtensions, ASTReaderListener &Listener, bool ValidateDiagnosticOptions, unsigned ClientLoadCapabilities=ARR_ConfigurationMismatch|ARR_OutOfDate)
Read the control block for the named AST file.
const LangOptions & getLangOpts() const
Definition ASTUnit.h:476
IntrusiveRefCntPtr< ASTReader > getASTReader() const
Definition ASTUnit.cpp:654
std::unique_ptr< ASTConsumer > CreateASTConsumer(CompilerInstance &CI, StringRef InFile) override
Create the AST consumer object for this action, if supported.
PtrTy get() const
Definition Ownership.h:171
bool isInvalid() const
Definition Ownership.h:167
Abstract interface for a consumer of code-completion information.
CompilerInstance - Helper class for managing a single instance of the Clang compiler.
const PCHContainerReader & getPCHContainerReader() const
Return the appropriate PCHContainerReader depending on the current CodeGenOptions.
DiagnosticsEngine & getDiagnostics() const
Get the current diagnostics engine.
std::unique_ptr< raw_pwrite_stream > createDefaultOutputFile(bool Binary=true, StringRef BaseInput="", StringRef Extension="", bool RemoveFileOnSignal=true, bool CreateMissingDirectories=false, bool ForceUseTemporary=false)
Create the default output file (from the invocation's options) and add it to the list of tracked outp...
FileManager & getFileManager() const
Return the current file manager to the caller.
PreprocessorOutputOptions & getPreprocessorOutputOpts()
ModuleCache & getModuleCache() const
Preprocessor & getPreprocessor() const
Return the current preprocessor.
ASTContext & getASTContext() const
FrontendOptions & getFrontendOpts()
HeaderSearchOptions & getHeaderSearchOpts()
bool hasCodeCompletionConsumer() const
const PCHContainerWriter & getPCHContainerWriter() const
Return the appropriate PCHContainerWriter depending on the current CodeGenOptions.
PreprocessorOptions & getPreprocessorOpts()
void createCodeCompletionConsumer()
Create a code completion consumer using the invocation; note that this will cause the source manager ...
DiagnosticOptions & getDiagnosticOpts()
CodeGenOptions & getCodeGenOpts()
SourceManager & getSourceManager() const
Return the current source manager.
CodeCompleteConsumer & getCodeCompletionConsumer() const
void createSema(TranslationUnitKind TUKind, CodeCompleteConsumer *CompletionConsumer)
Create the Sema object to be used for parsing.
static Decl * castFromDeclContext(const DeclContext *)
std::vector< std::string > Remarks
The list of -R... options used to alter the diagnostic mappings, with the prefixes removed.
std::vector< std::string > Warnings
The list of -W... options used to alter the diagnostic mappings, with the prefixes removed.
DiagnosticBuilder Report(SourceLocation Loc, unsigned DiagID)
Issue the message to the client.
bool hasErrorOccurred() const
Definition Diagnostic.h:880
void setSuppressAllDiagnostics(bool Val)
Suppress all diagnostics, to silence the front end when we know that we don't want any more diagnosti...
Definition Diagnostic.h:735
std::unique_ptr< ASTConsumer > CreateASTConsumer(CompilerInstance &CI, StringRef InFile) override
Create the AST consumer object for this action, if supported.
void ExecuteAction() override
Implement the ExecuteAction interface by running Sema on the already-initialized AST consumer.
bool BeginInvocation(CompilerInstance &CI) override
Callback before starting processing a single input, giving the opportunity to modify the CompilerInvo...
void ExecuteAction() override
Callback to run the program action, using the initialized compiler instance.
void ExecuteAction() override
Callback to run the program action, using the initialized compiler instance.
Implements support for file system lookup, file system caching, and directory search management.
Definition FileManager.h:53
llvm::ErrorOr< std::unique_ptr< llvm::MemoryBuffer > > getBufferForFile(FileEntryRef Entry, bool isVolatile=false, bool RequiresNullTerminator=true, std::optional< int64_t > MaybeLimit=std::nullopt, bool IsText=true)
Open the specified file as a MemoryBuffer, returning a new MemoryBuffer if successful,...
Abstract base class for actions which can be performed by the frontend.
const FrontendInputFile & getCurrentInput() const
InputKind getCurrentFileKind() const
virtual bool shouldEraseOutputFiles()
Callback at the end of processing a single input, to determine if the output files should be erased o...
ASTUnit & getCurrentASTUnit() const
CompilerInstance & getCompilerInstance() const
virtual bool PrepareToExecuteAction(CompilerInstance &CI)
Prepare to execute the action on the given CompilerInstance.
Module * getCurrentModule() const
StringRef getCurrentFile() const
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.
virtual TranslationUnitKind getTranslationUnitKind()
For AST-based actions, the kind of translation unit we're handling.
virtual bool hasCodeCompletionSupport() const
Does this action support use with code completion?
StringRef getCurrentFileOrBufferName() const
bool isCurrentFileAST() const
FrontendOptions - Options for controlling the behavior of the frontend.
unsigned BuildingImplicitModule
Whether we are performing an implicit module build.
unsigned AllowPCMWithCompilerErrors
Output (and read) PCM files regardless of compiler errors.
unsigned IncludeTimestamps
Whether timestamps should be written to the produced PCH file.
std::string ASTDumpFilter
If given, filter dumped AST Decl nodes by this substring.
unsigned ASTDumpLookups
Whether we include lookup table dumps in AST dumps.
ASTDumpOutputFormat ASTDumpFormat
Specifies the output format of the AST.
std::string OutputFile
The output file, if any.
unsigned GenReducedBMI
Whether to generate reduced BMI for C++20 named modules.
std::vector< std::shared_ptr< ModuleFileExtension > > ModuleFileExtensions
The list of module file extensions.
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::string ModuleOutputPath
Output Path for module output file.
unsigned ASTDumpDeclTypes
Whether we include declaration type dumps in AST dumps.
unsigned ASTDumpAll
Whether we deserialize all decls when forming AST dumps.
unsigned RelocatablePCH
When generating PCH files, instruct the AST writer to create relocatable PCH files.
SmallVector< FrontendInputFile, 0 > Inputs
The input files and their types.
unsigned ASTDumpDecls
Whether we include declaration dumps in AST dumps.
bool shouldEraseOutputFiles() override
Callback at the end of processing a single input, to determine if the output files should be erased o...
std::vector< std::unique_ptr< ASTConsumer > > CreateMultiplexConsumer(CompilerInstance &CI, StringRef InFile)
std::unique_ptr< ASTConsumer > CreateASTConsumer(CompilerInstance &CI, StringRef InFile) override
Create the AST consumer object for this action, if supported.
bool BeginSourceFileAction(CompilerInstance &CI) override
Callback at the start of processing a single input.
bool PrepareToExecuteAction(CompilerInstance &CI) override
Prepare to execute the action on the given CompilerInstance.
std::unique_ptr< raw_pwrite_stream > CreateOutputFile(CompilerInstance &CI, StringRef InFile) override
std::unique_ptr< ASTConsumer > CreateASTConsumer(CompilerInstance &CI, StringRef InFile) override
Create the AST consumer object for this action, if supported.
static std::unique_ptr< llvm::raw_pwrite_stream > CreateOutputFile(CompilerInstance &CI, StringRef InFile, std::string &OutputFile)
Creates file to write the PCH into and returns a stream to write it into.
bool BeginSourceFileAction(CompilerInstance &CI) override
Callback at the start of processing a single input.
static bool ComputeASTConsumerArguments(CompilerInstance &CI, std::string &Sysroot)
Compute the AST consumer arguments that will be used to create the PCHGenerator instance returned by ...
std::unique_ptr< ASTConsumer > CreateASTConsumer(CompilerInstance &CI, StringRef InFile) override
Create the AST consumer object for this action, if supported.
bool shouldEraseOutputFiles() override
Callback at the end of processing a single input, to determine if the output files should be erased o...
void ExecuteAction() override
Callback to run the program action, using the initialized compiler instance.
HLSLFrontendAction(std::unique_ptr< FrontendAction > WrappedAction)
HeaderSearchOptions - Helper class for storing options related to the initialization of the HeaderSea...
std::vector< SystemHeaderPrefix > SystemHeaderPrefixes
User-specified system header prefixes.
std::string ModuleFormat
The module/pch container format.
std::string Sysroot
If non-empty, the directory to use as a "virtual system root" for include paths.
std::string ModuleCachePath
The directory used for the module cache.
std::vector< std::string > VFSOverlayFiles
The set of user-provided virtual filesystem overlay files.
unsigned UseLibcxx
Use libc++ instead of the default libstdc++.
unsigned UseBuiltinIncludes
Include the compiler builtin includes.
unsigned UseStandardCXXIncludes
Include the system standard C++ library include search directories.
std::vector< Entry > UserEntries
User specified include entries.
std::string ResourceDir
The directory which holds the compiler resource files (builtin includes, etc.).
unsigned UseStandardSystemIncludes
Include the system standard include search directories.
unsigned DisableModuleHash
Whether we should disable the use of the hash string within the module cache.
ModuleFileName getCachedModuleFileName(Module *Module)
Retrieve the name of the cached module file that should be used to load the given module.
One of these records is kept for each identifier that is lexed.
StringRef getName() const
Return the actual identifier string.
@ CMK_HeaderUnit
Compiling a module header unit.
@ CMK_ModuleInterface
Compiling a C++ modules interface unit.
Keeps track of the various options that can be enabled, which controls the dialect of C or C++ that i...
std::string HLSLRootSigOverride
The HLSL root signature that will be used to overide the root signature used for the shader entry poi...
llvm::dxbc::RootSignatureVersion HLSLRootSigVer
The HLSL root signature version for dxil.
std::string CurrentModule
The name of the current module, of which the main source file is a part.
std::vector< std::string > ModuleFeatures
The names of any features to enable in module 'requires' decls in addition to the hard-coded list in ...
Lexer - This provides a simple interface that turns a text buffer into a stream of tokens.
Definition Lexer.h:78
void SetKeepWhitespaceMode(bool Val)
SetKeepWhitespaceMode - This method lets clients enable or disable whitespace retention mode.
Definition Lexer.h:254
bool LexFromRawLexer(Token &Result)
LexFromRawLexer - Lex a token from a designated raw lexer (one with no associated preprocessor object...
Definition Lexer.h:236
static PreambleBounds ComputePreamble(StringRef Buffer, const LangOptions &LangOpts, unsigned MaxLines=0)
Compute the preamble of the given file.
Definition Lexer.cpp:645
Encapsulates changes to the "macros namespace" (the location where the macro name became active,...
Definition MacroInfo.h:313
const MacroInfo * getMacroInfo() const
Definition MacroInfo.h:416
Encapsulates the data about a macro definition (e.g.
Definition MacroInfo.h:39
SourceLocation getDefinitionLoc() const
Return the location that the macro was defined at.
Definition MacroInfo.h:125
ArrayRef< Token > tokens() const
Definition MacroInfo.h:249
static ModuleFileName makeExplicit(std::string Name)
Creates a file name for an explicit module.
Definition Module.h:111
Describes a module or submodule.
Definition Module.h:246
SmallVector< ExportDecl, 2 > Exports
The set of export declarations.
Definition Module.h:574
void print(raw_ostream &OS, unsigned Indent=0, bool Dump=false) const
Print the module map for this module to the given stream.
Definition Module.cpp:482
ModuleKind Kind
The kind of this module.
Definition Module.h:291
llvm::SmallSetVector< Module *, 2 > Imports
The set of modules imported by this module, and on which this module depends.
Definition Module.h:561
std::string Name
The name of this module.
Definition Module.h:249
llvm::iterator_range< submodule_iterator > submodules()
Definition Module.h:952
@ ModuleImplementationUnit
This is a C++20 module implementation unit.
Definition Module.h:269
@ ModuleMapModule
This is a module that was defined by a module map and built out of header files.
Definition Module.h:260
@ ImplicitGlobalModuleFragment
This is an implicit fragment of the global module which contains only language linkage declarations (...
Definition Module.h:287
@ ModulePartitionInterface
This is a C++20 module partition interface.
Definition Module.h:272
@ ModuleInterfaceUnit
This is a C++20 module interface unit.
Definition Module.h:266
@ ModuleHeaderUnit
This is a C++20 header unit.
Definition Module.h:263
@ ModulePartitionImplementation
This is a C++20 module partition implementation.
Definition Module.h:275
@ PrivateModuleFragment
This is the private module fragment within some C++ module.
Definition Module.h:282
@ ExplicitGlobalModuleFragment
This is the explicit Global Module Fragment of a modular TU.
Definition Module.h:279
virtual void getNameForDiagnostic(raw_ostream &OS, const PrintingPolicy &Policy, bool Qualified) const
Appends a human-readable name for this declaration into the given stream.
Definition Decl.cpp:1846
virtual std::unique_ptr< ASTConsumer > CreatePCHContainerGenerator(CompilerInstance &CI, const std::string &MainFileName, const std::string &OutputFileName, std::unique_ptr< llvm::raw_pwrite_stream > OS, std::shared_ptr< PCHBuffer > Buffer) const =0
Return an ASTConsumer that can be chained with a PCHGenerator that produces a wrapper file format con...
This interface provides a way to observe the actions of the preprocessor as it does its thing.
Definition PPCallbacks.h:37
void ExecuteAction() override
Callback to run the program action, using the initialized compiler instance.
std::pair< unsigned, bool > PrecompiledPreambleBytes
If non-zero, the implicit PCH include is actually a precompiled preamble that covers this number of b...
bool DetailedRecord
Whether we should maintain a detailed record of all macro definitions and expansions.
bool UsePredefines
Initialize the preprocessor with the compiler and target specific predefines.
std::vector< std::pair< std::string, bool > > Macros
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 DumpToken(const Token &Tok, bool DumpFlags=false) const
Print the token to stderr, used for debugging.
void IgnorePragmas()
Install empty handlers for all pragmas (making them ignored).
Definition Pragma.cpp:2219
void Lex(Token &Result)
Lex the next token for this preprocessor.
void addPPCallbacks(std::unique_ptr< PPCallbacks > C)
void EnterMainSourceFile()
Enter the specified FileID as the main source file, which implicitly adds the builtin defines etc.
SourceManager & getSourceManager() const
HeaderSearch & getHeaderSearchInfo() const
const LangOptions & getLangOpts() const
unsigned getColumn() const
Return the presumed column number of this location.
const char * getFilename() const
Return the presumed filename of this location.
unsigned getLine() const
Return the presumed line number of this location.
bool isInvalid() const
Return true if this object is invalid or uninitialized.
void ExecuteAction() override
Callback to run the program action, using the initialized compiler instance.
void ExecuteAction() override
Callback to run the program action, using the initialized compiler instance.
void ExecuteAction() override
Callback to run the program action, using the initialized compiler instance.
Sema - This implements semantic analysis and AST building for C.
Definition Sema.h:868
ASTContext & Context
Definition Sema.h:1304
const LangOptions & getLangOpts() const
Definition Sema.h:932
std::vector< std::unique_ptr< TemplateInstantiationCallback > > TemplateInstCallbacks
The template instantiation callbacks to trace or track instantiations (objects can be chained).
Definition Sema.h:13728
SourceManager & getSourceManager() const
Definition Sema.h:937
Encodes a location in the source.
This class handles loading and caching of source files into memory.
PresumedLoc getPresumedLoc(SourceLocation Loc, bool UseLineDirectives=true) const
Returns the "presumed" location of a SourceLocation specifies.
std::unique_ptr< ASTConsumer > CreateASTConsumer(CompilerInstance &CI, StringRef InFile) override
Create the AST consumer object for this action, if supported.
Exposes information about the current target.
Definition TargetInfo.h:227
TargetOptions & getTargetOpts() const
Retrieve the target options.
Definition TargetInfo.h:327
const llvm::Triple & getTriple() const
Returns the target triple of the primary target.
std::string Triple
The name of the target triple to compile for.
std::string ABI
If given, the name of the target ABI to use.
std::string TuneCPU
If given, the name of the target CPU to tune code for.
std::string CPU
If given, the name of the target CPU to generate code for.
std::vector< std::string > FeaturesAsWritten
The list of target specific features to enable or disable, as written on the command line.
std::string HLSLEntry
The entry point name for HLSL shader being compiled as specified by -E.
This is a base class for callbacks that will be notified at every template instantiation.
std::unique_ptr< ASTConsumer > CreateASTConsumer(CompilerInstance &CI, StringRef InFile) override
Create the AST consumer object for this action, if supported.
void ExecuteAction() override
Implement the ExecuteAction interface by running Sema on the already-initialized AST consumer.
Token - This structure provides full information about a lexed token.
Definition Token.h:36
IdentifierInfo * getIdentifierInfo() const
Definition Token.h:197
bool isNot(tok::TokenKind K) const
Definition Token.h:111
std::unique_ptr< ASTConsumer > CreateASTConsumer(CompilerInstance &CI, StringRef InFile) override
Create the AST consumer object for this action, if supported.
void ExecuteAction() override
Implement the ExecuteAction interface by running Sema on the already-initialized AST consumer.
void ExecuteAction() override
Callback to run the program action, using the initialized compiler instance.
WrapperFrontendAction(std::unique_ptr< FrontendAction > WrappedAction)
Construct a WrapperFrontendAction from an existing action, taking ownership of it.
TranslationUnitKind getTranslationUnitKind() override
For AST-based actions, the kind of translation unit we're handling.
std::unique_ptr< FrontendAction > WrappedAction
Information about a module that has been loaded by the ASTReader.
Definition ModuleFile.h:145
std::string ModuleName
The name of the module.
Definition ModuleFile.h:166
Defines the clang::TargetInfo interface.
@ HeaderSearch
Remove unused header search paths including header maps.
@ OS
Indicates that the tracking object is a descendant of a referenced-counted OSObject,...
void HandleRootSignatureTarget(Sema &S, StringRef EntryRootSig)
IdentifierInfo * ParseHLSLRootSignature(Sema &Actions, llvm::dxbc::RootSignatureVersion Version, StringLiteral *Signature)
std::variant< struct RequiresDecl, struct HeaderDecl, struct UmbrellaDirDecl, struct ModuleDecl, struct ExcludeDecl, struct ExportDecl, struct ExportAsDecl, struct ExternModuleDecl, struct UseDecl, struct LinkDecl, struct ConfigMacrosDecl, struct ConflictDecl > Decl
All declarations that can appear in a module declaration.
@ MK_PCH
File is a PCH file treated as such.
Definition ModuleFile.h:51
@ MK_Preamble
File is a PCH file treated as the preamble.
Definition ModuleFile.h:54
bool isStringLiteral(TokenKind K)
Return true if this is a C or C++ string-literal (or C++11 user-defined-string-literal) token.
Definition TokenKinds.h:93
The JSON file list parser is used to communicate input to InstallAPI.
void atTemplateEnd(TemplateInstantiationCallbackPtrs &Callbacks, const Sema &TheSema, const Sema::CodeSynthesisContext &Inst)
void atTemplateBegin(TemplateInstantiationCallbackPtrs &Callbacks, const Sema &TheSema, const Sema::CodeSynthesisContext &Inst)
void printDependencyDirectivesAsSource(StringRef Source, ArrayRef< dependency_directives_scan::Directive > Directives, llvm::raw_ostream &OS)
Print the previously scanned dependency directives as minimized source text.
void initialize(TemplateInstantiationCallbackPtrs &Callbacks, const Sema &TheSema)
bool scanSourceForDependencyDirectives(StringRef Input, SmallVectorImpl< dependency_directives_scan::Token > &Tokens, SmallVectorImpl< dependency_directives_scan::Directive > &Directives, DiagnosticsEngine *Diags=nullptr, SourceLocation InputSourceLoc=SourceLocation())
Scan the input for the preprocessor directives that might have an effect on the dependencies for a co...
std::unique_ptr< ASTConsumer > CreateASTDeclNodeLister()
@ C
Languages that the frontend can parse and compile.
@ CIR
LLVM IR & CIR: we accept these so that we can run the optimizer on them, and compile them to assembly...
@ Asm
Assembly: we accept this only so that we can preprocess it.
std::string createSpecificModuleCachePath(FileManager &FileMgr, StringRef ModuleCachePath, bool DisableModuleHash, std::string ContextHash)
void DoPrintPreprocessedInput(Preprocessor &PP, raw_ostream *OS, const PreprocessorOutputOptions &Opts)
DoPrintPreprocessedInput - Implement -E mode.
@ Type
The name was classified as a type.
Definition Sema.h:564
std::unique_ptr< ASTConsumer > CreateASTDumper(std::unique_ptr< raw_ostream > OS, StringRef FilterString, bool DumpDecls, bool Deserialize, bool DumpLookups, bool DumpDeclTypes, ASTDumpOutputFormat Format)
std::unique_ptr< ASTConsumer > CreateASTPrinter(std::unique_ptr< raw_ostream > OS, StringRef FilterString)
std::unique_ptr< ASTConsumer > CreateASTViewer()
@ None
Perform validation, don't disable it.
std::string getClangFullRepositoryVersion()
Retrieves the full repository version that is an amalgamation of the information in getClangRepositor...
Definition Version.cpp:68
void finalize(TemplateInstantiationCallbackPtrs &Callbacks, const Sema &TheSema)
U cast(CodeGen::Address addr)
Definition Address.h:327
ActionResult< Expr * > ExprResult
Definition Ownership.h:249
unsigned MajorVersion
The major version of the extension data.
std::string UserInfo
A string containing additional user information that will be stored with the metadata.
std::string BlockName
The name used to identify this particular extension block within the resulting module file.
unsigned MinorVersion
The minor version of the extension data.
unsigned Size
Size of the preamble in bytes.
Definition Lexer.h:62
unsigned SuppressDefaultTemplateArgs
When true, attempt to suppress template arguments that match the default argument for the parameter.
static void mapping(IO &io, TemplightEntry &fields)