23#include "clang/Config/config.h"
46#include "llvm/ADT/IntrusiveRefCntPtr.h"
47#include "llvm/ADT/STLExtras.h"
48#include "llvm/ADT/ScopeExit.h"
49#include "llvm/ADT/Statistic.h"
50#include "llvm/Config/llvm-config.h"
51#include "llvm/Plugins/PassPlugin.h"
52#include "llvm/Support/AdvisoryLock.h"
53#include "llvm/Support/BuryPointer.h"
54#include "llvm/Support/CrashRecoveryContext.h"
55#include "llvm/Support/Errc.h"
56#include "llvm/Support/FileSystem.h"
57#include "llvm/Support/MemoryBuffer.h"
58#include "llvm/Support/Path.h"
59#include "llvm/Support/Signals.h"
60#include "llvm/Support/SmallVectorMemoryBuffer.h"
61#include "llvm/Support/TimeProfiler.h"
62#include "llvm/Support/Timer.h"
63#include "llvm/Support/VirtualFileSystem.h"
64#include "llvm/Support/VirtualOutputBackends.h"
65#include "llvm/Support/VirtualOutputError.h"
66#include "llvm/Support/raw_ostream.h"
67#include "llvm/TargetParser/Host.h"
74CompilerInstance::CompilerInstance(
75 std::shared_ptr<CompilerInvocation> Invocation,
76 std::shared_ptr<PCHContainerOperations> PCHContainerOps,
77 std::shared_ptr<ModuleCache> ModCache)
79 Invocation(
std::move(Invocation)),
80 ModCache(ModCache ?
std::move(ModCache)
82 ThePCHContainerOperations(
std::move(PCHContainerOps)) {
83 assert(this->Invocation &&
"Invocation must not be null");
87 assert(OutputFiles.empty() &&
"Still output files in flight?");
91 return (BuildGlobalModuleIndex ||
92 (TheASTReader && TheASTReader->isGlobalIndexUnavailable() &&
94 !DisableGeneratingGlobalModuleIndex;
99 Diagnostics = std::move(
Value);
103 OwnedVerboseOutputStream.reset();
104 VerboseOutputStream = &
Value;
108 OwnedVerboseOutputStream.swap(
Value);
109 VerboseOutputStream = OwnedVerboseOutputStream.get();
133 auto &TO = AuxTargetOpts = std::make_unique<TargetOptions>();
174 assert(
Value ==
nullptr ||
176 FileMgr = std::move(
Value);
181 SourceMgr = std::move(
Value);
185 PP = std::move(
Value);
190 Context = std::move(
Value);
192 if (Context && Consumer)
201 Consumer = std::move(
Value);
203 if (Context && Consumer)
208 CompletionConsumer.reset(
Value);
212 return std::move(TheSema);
219 assert(ModCache.get() == &Reader->getModuleManager().getModuleCache() &&
220 "Expected ASTReader to use the same PCM cache");
221 TheASTReader = std::move(Reader);
224std::shared_ptr<ModuleDependencyCollector>
226 return ModuleDepCollector;
230 std::shared_ptr<ModuleDependencyCollector> Collector) {
231 ModuleDepCollector = std::move(Collector);
235 std::shared_ptr<ModuleDependencyCollector> MDC) {
238 for (
auto &Name : HeaderMapFileNames)
243 std::shared_ptr<ModuleDependencyCollector> MDC) {
250 auto PCHDir =
FileMgr.getOptionalDirectoryRef(PCHInclude);
252 MDC->addFile(PCHInclude);
258 llvm::sys::path::native(PCHDir->getName(), DirNative);
259 llvm::vfs::FileSystem &FS =
FileMgr.getVirtualFileSystem();
261 for (llvm::vfs::directory_iterator Dir = FS.dir_begin(DirNative, EC), DirEnd;
262 Dir != DirEnd && !EC; Dir.increment(EC)) {
271 MDC->addFile(Dir->path());
276 std::shared_ptr<ModuleDependencyCollector> MDC) {
280 if (
auto *RedirectingVFS = dyn_cast<llvm::vfs::RedirectingFileSystem>(&VFS))
281 llvm::vfs::collectVFSEntries(*RedirectingVFS, VFSEntries);
284 for (
auto &E : VFSEntries)
285 MDC->addFile(E.VPath, E.RPath);
299 llvm::makeIntrusiveRefCnt<llvm::vfs::TracingFileSystem>(std::move(VFS));
307 std::unique_ptr<raw_ostream> StreamOwner;
308 raw_ostream *OS = &llvm::errs();
311 auto FileOS = std::make_unique<llvm::raw_fd_ostream>(
313 llvm::sys::fs::OF_Append | llvm::sys::fs::OF_TextWithCRLF);
315 Diags.
Report(diag::warn_fe_cc_log_diagnostics_failure)
318 FileOS->SetUnbuffered();
320 StreamOwner = std::move(FileOS);
325 auto Logger = std::make_unique<LogDiagnosticPrinter>(*OS, DiagOpts,
326 std::move(StreamOwner));
340 StringRef OutputFile) {
341 auto SerializedConsumer =
346 Diags.
takeClient(), std::move(SerializedConsumer)));
349 Diags.
getClient(), std::move(SerializedConsumer)));
354 bool ShouldOwnClient) {
363 auto Diags = llvm::makeIntrusiveRefCnt<DiagnosticsEngine>(
369 Diags->setClient(Client, ShouldOwnClient);
376 if (Opts.VerifyDiagnostics)
395 assert(VFS &&
"CompilerInstance needs a VFS for creating FileManager");
402 assert(Diagnostics &&
"DiagnosticsEngine needed for creating SourceManager");
403 assert(FileMgr &&
"FileManager needed for creating SourceManager");
404 SourceMgr = llvm::makeIntrusiveRefCnt<SourceManager>(
getDiagnostics(),
418 FileMgr.getVirtualFileRef(RB.first, RB.second->getBufferSize(), 0);
425 SourceMgr.overrideFileContents(FromFile, RB.second->getMemBufferRef());
427 SourceMgr.overrideFileContents(
428 FromFile, std::unique_ptr<llvm::MemoryBuffer>(RB.second));
436 Diags.
Report(diag::err_fe_remap_missing_to_file) << RF.first << RF.second;
446 SourceMgr.overrideFileContents(FromFile, *ToFile);
449 SourceMgr.setOverridenFilesKeepOriginalName(
459 TheASTReader.reset();
465 PP = std::make_shared<Preprocessor>(Invocation->getPreprocessorOpts(),
474 PP->createPreprocessingRecord();
478 PP->getFileManager(), PPOpts);
487 const llvm::Triple *HeaderSearchTriple = &PP->getTargetInfo().getTriple();
488 if (PP->getTargetInfo().getTriple().getOS() == llvm::Triple::CUDA &&
489 PP->getAuxTargetInfo())
490 HeaderSearchTriple = &PP->getAuxTargetInfo()->getTriple();
493 PP->getLangOpts(), *HeaderSearchTriple);
497 if (PP->getLangOpts().Modules && PP->getLangOpts().ImplicitModules) {
501 PP->getHeaderSearchInfo().initializeModuleCachePath(
516 ModuleDepCollector = std::make_shared<ModuleDependencyCollector>(
522 if (ModuleDepCollector) {
533 for (
auto &Listener : DependencyCollectors)
534 Listener->attachToPreprocessor(*PP);
541 if (OutputPath ==
"-")
554 if (GetDependencyDirectives)
555 PP->setDependencyDirectivesGetter(*GetDependencyDirectives);
562 auto Context = llvm::makeIntrusiveRefCnt<ASTContext>(
563 getLangOpts(), PP.getSourceManager(), PP.getIdentifierTable(),
564 PP.getSelectorTable(), PP.getBuiltinInfo(), PP.TUKind);
581 void ReadModuleName(StringRef ModuleName)
override {
584 LoadedModules.push_back(ModuleName.str());
589 for (
const std::string &LoadedModule : LoadedModules)
592 LoadedModules.clear();
595 void markAllUnavailable() {
596 for (
const std::string &LoadedModule : LoadedModules) {
599 M->HasIncompatibleModuleFile =
true;
603 SmallVector<Module *, 2> Stack;
605 while (!Stack.empty()) {
606 Module *Current = Stack.pop_back_val();
610 llvm::append_range(Stack, SubmodulesRange);
614 LoadedModules.clear();
621 bool AllowPCHWithCompilerErrors,
void *DeserializationListener,
622 bool OwnDeserializationListener) {
629 DeserializationListener, OwnDeserializationListener,
Preamble,
634 StringRef Path, StringRef Sysroot,
639 ArrayRef<std::shared_ptr<ModuleFileExtension>> Extensions,
640 ArrayRef<std::shared_ptr<DependencyCollector>> DependencyCollectors,
641 void *DeserializationListener,
bool OwnDeserializationListener,
642 bool Preamble,
bool UseGlobalModuleIndex) {
644 PP.getHeaderSearchInfo().getHeaderSearchOpts();
646 auto Reader = llvm::makeIntrusiveRefCnt<ASTReader>(
647 PP, ModCache, &Context, PCHContainerRdr, CodeGenOpts, Extensions,
648 Sysroot.empty() ?
"" : Sysroot.data(), DisableValidation,
649 AllowPCHWithCompilerErrors,
false,
656 Context.setExternalSource(Reader);
658 Reader->setDeserializationListener(
660 OwnDeserializationListener);
662 for (
auto &Listener : DependencyCollectors)
663 Listener->attachToASTReader(*Reader);
665 auto Listener = std::make_unique<ReadModuleNames>(PP);
666 auto &ListenerRef = *Listener;
668 std::move(Listener));
677 PP.setPredefines(Reader->getSuggestedPredefines());
678 ListenerRef.registerAll();
694 ListenerRef.markAllUnavailable();
695 Context.setExternalSource(
nullptr);
721 if (!CompletionConsumer) {
734 timerGroup.reset(
new llvm::TimerGroup(
"clang",
"Clang time report"));
735 FrontendTimer.reset(
new llvm::Timer(
"frontend",
"Front end", *timerGroup));
755 TUKind, CompletionConsumer));
761 if (ExternalSemaSrc) {
762 TheSema->addExternalSource(ExternalSemaSrc);
763 ExternalSemaSrc->InitializeSema(*TheSema);
769 (void)TheSema->APINotes.loadCurrentModuleAPINotes(
781 for (
auto &O : OutputFiles)
782 llvm::handleAllErrors(
784 [&](
const llvm::vfs::TempFileOutputError &E) {
785 getDiagnostics().Report(diag::err_unable_to_rename_temp)
786 << E.getTempPath() << E.getOutputPath()
787 << E.convertToErrorCode().message();
789 [&](
const llvm::vfs::OutputError &E) {
790 getDiagnostics().Report(diag::err_fe_unable_to_open_output)
791 << E.getOutputPath() << E.convertToErrorCode().message();
793 [&](
const llvm::ErrorInfoBase &EIB) {
794 getDiagnostics().Report(diag::err_fe_unable_to_open_output)
795 << O.getPath() << EIB.message();
799 if (DeleteBuiltModules) {
800 for (
auto &
Module : BuiltModules)
801 llvm::sys::fs::remove(
Module.second);
802 BuiltModules.clear();
807 bool Binary, StringRef InFile, StringRef Extension,
bool RemoveFileOnSignal,
808 bool CreateMissingDirectories,
bool ForceUseTemporary) {
810 std::optional<SmallString<128>> PathStorage;
811 if (OutputPath.empty()) {
812 if (InFile ==
"-" || Extension.empty()) {
815 PathStorage.emplace(InFile);
816 llvm::sys::path::replace_extension(*PathStorage, Extension);
817 OutputPath = *PathStorage;
823 CreateMissingDirectories);
827 return std::make_unique<llvm::raw_null_ostream>();
834 assert(!OutputMgr &&
"Already has an output manager");
835 OutputMgr = std::move(NewOutputs);
839 assert(!OutputMgr &&
"Already has an output manager");
840 OutputMgr = llvm::makeIntrusiveRefCnt<llvm::vfs::OnDiskOutputBackend>();
854std::unique_ptr<raw_pwrite_stream>
856 bool RemoveFileOnSignal,
bool UseTemporary,
857 bool CreateMissingDirectories) {
859 createOutputFileImpl(OutputPath,
Binary, RemoveFileOnSignal, UseTemporary,
860 CreateMissingDirectories);
862 return std::move(*OS);
864 << OutputPath << errorToErrorCode(OS.takeError()).message();
869CompilerInstance::createOutputFileImpl(StringRef OutputPath,
bool Binary,
870 bool RemoveFileOnSignal,
872 bool CreateMissingDirectories) {
873 assert((!CreateMissingDirectories || UseTemporary) &&
874 "CreateMissingDirectories is only allowed when using temporary files");
878 std::optional<SmallString<128>> AbsPath;
879 if (OutputPath !=
"-" && !llvm::sys::path::is_absolute(OutputPath)) {
881 "File Manager is required to fix up relative path.\n");
883 AbsPath.emplace(OutputPath);
885 OutputPath = *AbsPath;
893 .setDiscardOnSignal(RemoveFileOnSignal)
894 .setAtomicWrite(UseTemporary)
895 .setImplyCreateDirectories(UseTemporary && CreateMissingDirectories));
897 return O.takeError();
899 O->discardOnDestroy([](llvm::Error E) { consumeError(std::move(E)); });
900 OutputFiles.push_back(std::move(*O));
901 return OutputFiles.back().createProxy();
923 SourceMgr.setMainFileID(SourceMgr.createFileID(Input.
getBuffer(), Kind));
924 assert(SourceMgr.getMainFileID().isValid() &&
925 "Couldn't establish MainFileID!");
929 StringRef InputFile = Input.
getFile();
932 auto FileOrErr = InputFile ==
"-"
934 : FileMgr.getFileRef(InputFile,
true);
936 auto EC = llvm::errorToErrorCode(FileOrErr.takeError());
937 if (InputFile !=
"-")
938 Diags.
Report(diag::err_fe_error_reading) << InputFile << EC.message();
940 Diags.
Report(diag::err_fe_error_reading_stdin) << EC.message();
944 SourceMgr.setMainFileID(
947 assert(SourceMgr.getMainFileID().isValid() &&
948 "Couldn't establish MainFileID!");
954void CompilerInstance::PrepareForExecution() {
971 assert(
hasDiagnostics() &&
"Diagnostics engine is not initialized!");
973 assert(!
getFrontendOpts().ShowVersion &&
"Client must handle '-version'!");
975 llvm::TimeTraceScope TimeScope(
"ExecuteCompiler");
977 PrepareForExecution();
998 OS <<
"clang -cc1 version " CLANG_VERSION_STRING <<
" based upon LLVM "
999 << LLVM_VERSION_STRING <<
" default target "
1000 << llvm::sys::getDefaultTargetTriple() <<
"\n";
1003 llvm::EnableStatistics(
false);
1016 ModuleImportResults.clear();
1019 if (llvm::Error Err = Act.
Execute()) {
1020 consumeError(std::move(Err));
1033 llvm::PrintStatistics(OS);
1036 if (!StatsFile.empty()) {
1037 llvm::sys::fs::OpenFlags FileFlags = llvm::sys::fs::OF_TextWithCRLF;
1039 FileFlags |= llvm::sys::fs::OF_Append;
1042 std::make_unique<llvm::raw_fd_ostream>(StatsFile, EC, FileFlags);
1045 << StatsFile << EC.message();
1047 llvm::PrintStatisticsJSON(*StatS);
1066 OS << NumWarnings <<
" warning" << (NumWarnings == 1 ?
"" :
"s");
1067 if (NumWarnings && NumErrors)
1070 OS << NumErrors <<
" error" << (NumErrors == 1 ?
"" :
"s");
1071 if (NumWarnings || NumErrors) {
1075 OS <<
" when compiling for host";
1077 OS <<
" when compiling for "
1090 if (llvm::sys::DynamicLibrary::LoadLibraryPermanently(Path.c_str(), &
Error))
1097 if (
auto PassPlugin = llvm::PassPlugin::Load(Path)) {
1098 PassPlugins.emplace_back(std::make_unique<llvm::PassPlugin>(*PassPlugin));
1101 << Path <<
toString(PassPlugin.takeError());
1106 for (
const FrontendPluginRegistry::entry &Plugin :
1107 FrontendPluginRegistry::entries()) {
1108 std::unique_ptr<PluginASTAction> P(Plugin.instantiate());
1120 if (LangOpts.OpenCL)
1129std::unique_ptr<CompilerInstance> CompilerInstance::cloneForModuleCompileImpl(
1132 std::optional<ThreadSafeCloneConfig> ThreadSafeConfig) {
1134 auto Invocation = std::make_shared<CompilerInvocation>(
getInvocation());
1136 PreprocessorOptions &PPOpts = Invocation->getPreprocessorOpts();
1140 Invocation->resetNonModularOptions();
1144 HeaderSearchOptions &HSOpts = Invocation->getHeaderSearchOpts();
1145 llvm::erase_if(PPOpts.
Macros,
1146 [&HSOpts](
const std::pair<std::string, bool> &def) {
1147 StringRef MacroDef = def.first;
1148 return HSOpts.ModulesIgnoreMacros.contains(
1149 llvm::CachedHashString(MacroDef.split(
'=').first));
1153 Invocation->getLangOpts().ModuleName =
1157 Invocation->getLangOpts().CurrentModule = std::string(ModuleName);
1162 FrontendOptions &FrontendOpts = Invocation->getFrontendOpts();
1163 FrontendOpts.
OutputFile = ModuleFileName.str();
1170 FrontendOpts.
Inputs = {std::move(Input)};
1175 DiagnosticOptions &DiagOpts = Invocation->getDiagnosticOpts();
1177 DiagOpts.VerifyDiagnostics = 0;
1180 "Module hash mismatch!");
1182 std::shared_ptr<ModuleCache> ModCache;
1183 if (ThreadSafeConfig) {
1184 ModCache = ThreadSafeConfig->getModuleCache();
1186 ModCache = this->ModCache;
1190 auto InstancePtr = std::make_unique<CompilerInstance>(
1192 auto &Instance = *InstancePtr;
1194 auto &
Inv = Instance.getInvocation();
1196 if (ThreadSafeConfig) {
1197 Instance.setVirtualFileSystem(ThreadSafeConfig->getVFS());
1198 Instance.createFileManager();
1204 Instance.createFileManager();
1207 if (ThreadSafeConfig) {
1208 Instance.createDiagnostics(&ThreadSafeConfig->getDiagConsumer(),
1211 Instance.createDiagnostics(
1216 Instance.getDiagnostics().setSuppressSystemWarnings(
false);
1218 Instance.createSourceManager();
1219 SourceManager &SourceMgr = Instance.getSourceManager();
1221 if (ThreadSafeConfig) {
1227 SourceMgr.pushModuleBuildStack(
1232 Instance.FailedModules = FailedModules;
1237 if (GetDependencyDirectives)
1238 Instance.GetDependencyDirectives =
1239 GetDependencyDirectives->cloneFor(Instance.getFileManager());
1241 if (ThreadSafeConfig) {
1242 Instance.setModuleDepCollector(ThreadSafeConfig->getModuleDepCollector());
1249 Inv.getDependencyOutputOpts() = DependencyOutputOptions();
1255class PrettyStackTraceBuildModule :
public llvm::PrettyStackTraceEntry {
1256 StringRef ModuleName;
1257 StringRef ModuleFileName;
1260 PrettyStackTraceBuildModule(StringRef ModuleName, StringRef ModuleFileName)
1261 : ModuleName(ModuleName), ModuleFileName(ModuleFileName) {}
1262 void print(raw_ostream &OS)
const override {
1263 OS <<
"Building module '" << ModuleName <<
"' as '" << ModuleFileName
1269std::unique_ptr<llvm::MemoryBuffer>
1272 CompilerInstance &Instance) {
1273 PrettyStackTraceBuildModule CrashInfo(ModuleName,
ModuleFileName);
1274 llvm::TimeTraceScope TimeScope(
"Module Compile", ModuleName);
1291 bool Crashed = !llvm::CrashRecoveryContext().RunSafelyOnNewStack(
1293 auto OS = std::make_unique<llvm::raw_svector_ostream>(Buffer);
1295 std::unique_ptr<FrontendAction> Action =
1296 std::make_unique<GenerateModuleFromModuleMapAction>(std::move(OS));
1298 if (
auto WrapGenModuleAction = Instance.getGenModuleActionWrapper())
1299 Action = WrapGenModuleAction(Instance.getFrontendOpts(),
1302 Instance.ExecuteAction(*Action);
1314 FailedModules = std::move(Instance.FailedModules);
1319 Instance.setSema(
nullptr);
1320 Instance.setASTConsumer(
nullptr);
1323 Instance.clearOutputFiles(
true);
1337 if (Instance.getDiagnostics().hasErrorOccurred() &&
1338 !Instance.getFrontendOpts().AllowPCMWithCompilerErrors)
1341 return std::make_unique<llvm::SmallVectorMemoryBuffer>(
1342 std::move(Buffer), Instance.getFrontendOpts().OutputFile);
1347 StringRef Filename = llvm::sys::path::filename(
File.getName());
1349 if (Filename ==
"module_private.map")
1350 llvm::sys::path::append(PublicFilename,
"module.map");
1351 else if (Filename ==
"module.private.modulemap")
1352 llvm::sys::path::append(PublicFilename,
"module.modulemap");
1354 return std::nullopt;
1355 return FileMgr.getOptionalFileRef(PublicFilename);
1360 std::optional<ThreadSafeCloneConfig> ThreadSafeConfig) {
1376 while (Loc.
isValid() && isModuleMap(SourceMgr.getFileCharacteristic(Loc))) {
1377 ModuleMapFID = SourceMgr.getFileID(Loc);
1378 Loc = SourceMgr.getIncludeLoc(ModuleMapFID);
1382 SourceMgr.getFileEntryRefForID(ModuleMapFID);
1383 assert(ModuleMapFile &&
"Top-level module map with no FileID");
1390 ModuleMapFile = PublicMMFile;
1402 return cloneForModuleCompileImpl(
1403 ImportLoc, ModuleName,
1406 std::move(ThreadSafeConfig));
1414 llvm::sys::path::append(FakeModuleMapFile,
"__inferred_module.map");
1416 std::string InferredModuleMapContent;
1417 llvm::raw_string_ostream OS(InferredModuleMapContent);
1420 auto Instance = cloneForModuleCompileImpl(
1421 ImportLoc, ModuleName,
1424 std::move(ThreadSafeConfig));
1426 std::unique_ptr<llvm::MemoryBuffer> ModuleMapBuffer =
1427 llvm::MemoryBuffer::getMemBufferCopy(InferredModuleMapContent);
1428 FileEntryRef ModuleMapFile = Instance->getFileManager().getVirtualFileRef(
1429 FakeModuleMapFile, InferredModuleMapContent.size(), 0);
1430 Instance->getSourceManager().overrideFileContents(ModuleMapFile,
1431 std::move(ModuleMapBuffer));
1443 bool *OutOfDate,
bool *Missing) {
1454 ModuleLoadCapabilities);
1484 std::unique_ptr<llvm::MemoryBuffer> Buffer;
1496 diag::err_module_not_built)
1508 diag::err_module_not_written)
1526 StringRef ExtractedBuffer = Rdr.
ExtractPCH(*Buffer);
1529 Buffer = llvm::MemoryBuffer::getMemBufferCopy(ExtractedBuffer);
1567 if (llvm::Error Err = Lock->tryLock().moveInto(Owned)) {
1590 switch (Lock->waitForUnlockFor(std::chrono::seconds(Timeout))) {
1591 case llvm::WaitForUnlockResult::Success:
1593 case llvm::WaitForUnlockResult::OwnerDied:
1595 case llvm::WaitForUnlockResult::Timeout:
1602 Lock->unsafeUnlock();
1607 bool OutOfDate =
false;
1608 bool Missing =
false;
1612 if (!OutOfDate && !Missing)
1672 for (
auto *MD = LatestLocalMD; MD; MD = MD->getPrevious()) {
1674 FileID FID = SourceMgr.getFileID(MDLoc);
1680 !SourceMgr.isWrittenInCommandLineFile(MDLoc))
1682 if (
auto *DMD = dyn_cast<DefMacroDirective>(MD))
1683 CmdLineDefinition = DMD->getMacroInfo();
1688 if (CurrentDefinition == CmdLineDefinition) {
1690 }
else if (!CurrentDefinition) {
1693 PP.
Diag(ImportLoc, diag::warn_module_config_macro_undef)
1695 auto LatestDef = LatestLocalMD->getDefinition();
1696 assert(LatestDef.isUndefined() &&
1697 "predefined macro went away with no #undef?");
1698 PP.
Diag(LatestDef.getUndefLocation(), diag::note_module_def_undef_here)
1701 }
else if (!CmdLineDefinition) {
1704 PP.
Diag(ImportLoc, diag::warn_module_config_macro_undef)
1706 PP.
Diag(CurrentDefinition->getDefinitionLoc(),
1707 diag::note_module_def_undef_here)
1709 }
else if (!CurrentDefinition->isIdenticalTo(*CmdLineDefinition, PP,
1712 PP.
Diag(ImportLoc, diag::warn_module_config_macro_undef)
1714 PP.
Diag(CurrentDefinition->getDefinitionLoc(),
1715 diag::note_module_def_undef_here)
1723 for (
const StringRef ConMacro : TopModule->
ConfigMacros) {
1739 .getHeaderSearchInfo()
1740 .getSpecificModuleCachePath()
1747 std::string Sysroot = HSOpts.
Sysroot;
1750 std::unique_ptr<llvm::Timer> ReadTimer;
1753 ReadTimer = std::make_unique<llvm::Timer>(
"reading_modules",
1754 "Reading modules", *timerGroup);
1755 TheASTReader = llvm::makeIntrusiveRefCnt<ASTReader>(
1759 Sysroot.empty() ?
"" : Sysroot.c_str(),
1768 TheASTReader->setDeserializationListener(
1775 TheASTReader->InitializeSema(
getSema());
1779 for (
auto &Listener : DependencyCollectors)
1780 Listener->attachToASTReader(*TheASTReader);
1787 Timer.init(
"preloading." + std::string(
FileName.str()),
1788 "Preloading " + std::string(
FileName.str()), *timerGroup);
1789 llvm::TimeRegion TimeLoading(timerGroup ? &Timer :
nullptr);
1797 bool ConfigMismatchIsRecoverable =
1802 auto Listener = std::make_unique<ReadModuleNames>(*PP);
1803 auto &ListenerRef = *Listener;
1805 std::move(Listener));
1808 switch (TheASTReader->ReadAST(
1811 &LoadedModuleFile)) {
1815 ListenerRef.registerAll();
1821 diag::warn_ast_file_config_mismatch)
1825 ListenerRef.markAllUnavailable();
1837 MS_PrebuiltModulePath,
1838 MS_ModuleBuildPragma
1846 const std::map<std::string, std::string, std::less<>> &BuiltModules,
1848 assert(ModuleFilename.
empty() &&
"Already has a module source?");
1852 auto BuiltModuleIt = BuiltModules.find(ModuleName);
1853 if (BuiltModuleIt != BuiltModules.end()) {
1855 return MS_ModuleBuildPragma;
1865 if (!ModuleFilename.
empty())
1866 return MS_PrebuiltModulePath;
1872 return MS_ModuleCache;
1875 return MS_ModuleNotFound;
1880 bool IsInclusionDirective) {
1884 HS.
lookupModule(ModuleName, ImportLoc,
true, !IsInclusionDirective);
1894 ModuleFileName ModuleFilename;
1895 ModuleSource Source =
1897 SourceLocation ModuleNameLoc = ModuleNameRange.
getBegin();
1898 if (Source == MS_ModuleNotFound) {
1901 << ModuleName << ModuleNameRange;
1904 if (ModuleFilename.
empty()) {
1923 Timer.init(
"loading." + std::string(ModuleFilename.
str()),
1924 "Loading " + std::string(ModuleFilename.
str()), *timerGroup);
1925 llvm::TimeRegion TimeLoading(timerGroup ? &Timer :
nullptr);
1926 llvm::TimeTraceScope TimeScope(
"Module Load", ModuleName);
1930 unsigned ARRFlags = Source == MS_ModuleCache
1933 : Source == MS_PrebuiltModulePath
1937 Source == MS_PrebuiltModulePath
1939 : Source == MS_ModuleBuildPragma
1942 ImportLoc, ARRFlags)) {
1946 assert(Source != MS_ModuleCache &&
1947 "missing module, but file loaded from cache");
1951 M = HS.
lookupModule(ModuleName, ImportLoc,
true, !IsInclusionDirective);
1956 getASTReader()->getModuleManager().makeKey(ModuleFilename))
1961 return ModuleLoadResult();
1970 if (Source == MS_PrebuiltModulePath)
1974 diag::warn_ast_file_config_mismatch)
1983 return ModuleLoadResult();
1987 return ModuleLoadResult();
1991 if (Source != MS_ModuleCache) {
1995 return ModuleLoadResult();
1999 assert(M &&
"missing module, but trying to compile for cache");
2003 ModuleBuildStack::iterator Pos = ModPath.begin(), PosEnd = ModPath.end();
2004 for (; Pos != PosEnd; ++Pos) {
2005 if (Pos->first == ModuleName)
2009 if (Pos != PosEnd) {
2010 SmallString<256> CyclePath;
2011 for (; Pos != PosEnd; ++Pos) {
2012 CyclePath += Pos->first;
2013 CyclePath +=
" -> ";
2015 CyclePath += ModuleName;
2018 << ModuleName << CyclePath;
2023 if (FailedModules.contains(ModuleName)) {
2025 << ModuleName << SourceRange(ImportLoc, ModuleNameLoc);
2033 "undiagnosed error in compileModuleAndReadAST");
2034 FailedModules.insert(ModuleName);
2046 bool IsInclusionDirective) {
2048 StringRef ModuleName = Path[0].getIdentifierInfo()->getName();
2055 auto CacheIt = ModuleImportResults.find(ImportLoc);
2056 if (CacheIt != ModuleImportResults.end()) {
2057 if (CacheIt->second && ModuleName !=
getLangOpts().CurrentModule)
2058 TheASTReader->makeModuleVisible(CacheIt->second,
Visibility, ImportLoc);
2059 return CacheIt->second;
2074 }
else if (ModuleName ==
getLangOpts().CurrentModule) {
2076 Module = PP->getHeaderSearchInfo().lookupModule(
2077 ModuleName, ImportLoc,
true,
2078 !IsInclusionDirective);
2090 ModuleName, ImportLoc,
true, !IsInclusionDirective);
2093 PPCb->moduleLoadSkipped(
Module);
2096 std::vector<clang::Module *> Worklist{
Module};
2097 while (!Worklist.empty()) {
2099 Worklist.pop_back();
2102 Worklist.push_back(SubM);
2107 SourceLocation ModuleNameEndLoc = Path.back().getLoc().getLocWithOffset(
2108 Path.back().getIdentifierInfo()->getLength());
2111 IsInclusionDirective);
2115 DisableGeneratingGlobalModuleIndex =
true;
2127 bool MapPrivateSubModToTopLevel =
false;
2128 for (
unsigned I = 1, N = Path.size(); I != N; ++I) {
2129 StringRef Name = Path[I].getIdentifierInfo()->getName();
2138 PrivateModule.append(
"_Private");
2141 auto &II = PP->getIdentifierTable().get(
2142 PrivateModule, PP->getIdentifierInfo(
Module->
Name)->getTokenID());
2143 PrivPath.emplace_back(Path[0].getLoc(), &II);
2147 if (PP->getHeaderSearchInfo().lookupModule(PrivateModule, ImportLoc,
true,
2148 !IsInclusionDirective) ||
2150 PP->getHeaderSearchInfo()) != MS_ModuleNotFound)
2153 MapPrivateSubModToTopLevel =
true;
2154 PP->markClangModuleAsAffecting(
Module);
2156 diag::warn_no_priv_submodule_use_toplevel, ImportLoc)) {
2158 diag::warn_no_priv_submodule_use_toplevel)
2161 <<
SourceRange(Path[0].getLoc(), Path[I].getLoc())
2165 diag::note_private_top_level_defined);
2173 unsigned BestEditDistance = (std::numeric_limits<unsigned>::max)();
2177 Name.edit_distance(SubModule->Name,
2178 true, BestEditDistance);
2179 if (ED <= BestEditDistance) {
2180 if (ED < BestEditDistance) {
2182 BestEditDistance = ED;
2185 Best.push_back(SubModule->Name);
2190 if (Best.size() == 1) {
2192 diag::err_no_submodule_suggest)
2194 << Best[0] <<
SourceRange(Path[0].getLoc(), Path[I - 1].getLoc())
2207 <<
SourceRange(Path[0].getLoc(), Path[I - 1].getLoc());
2226 <<
SourceRange(Path.front().getLoc(), Path.back().getLoc());
2235 <<
SourceRange(Path.front().getLoc(), Path.back().getLoc());
2254 StringRef ModuleName,
2258 for (
auto &
C : CleanModuleName)
2267 if (std::error_code EC = llvm::sys::fs::createTemporaryFile(
2273 std::string ModuleMapFileName = (CleanModuleName +
".map").str();
2280 std::string NullTerminatedSource(Source.str());
2282 auto Other = cloneForModuleCompileImpl(ImportLoc, ModuleName, Input,
2288 ModuleMapFileName, NullTerminatedSource.size(), 0);
2289 Other->getSourceManager().overrideFileContents(
2290 ModuleMapFile, llvm::MemoryBuffer::getMemBuffer(NullTerminatedSource));
2292 Other->BuiltModules = std::move(BuiltModules);
2293 Other->DeleteBuiltModules =
false;
2296 std::unique_ptr<llvm::MemoryBuffer> Buffer =
2298 BuiltModules = std::move(
Other->BuiltModules);
2301 llvm::raw_fd_ostream OS(FD,
true);
2302 BuiltModules[std::string(ModuleName)] = std::string(
ModuleFileName);
2303 OS << Buffer->getBuffer();
2316 TheASTReader->makeModuleVisible(Mod,
Visibility, ImportLoc);
2322 .getHeaderSearchInfo()
2323 .getSpecificModuleCachePath()
2333 TheASTReader->loadGlobalIndex();
2338 llvm::sys::fs::create_directories(
2339 getPreprocessor().getHeaderSearchInfo().getSpecificModuleCachePath());
2343 .getHeaderSearchInfo()
2344 .getSpecificModuleCachePath())) {
2349 consumeError(std::move(Err));
2352 TheASTReader->resetForReload();
2353 TheASTReader->loadGlobalIndex();
2354 GlobalIndex = TheASTReader->getGlobalIndex();
2358 if (!HaveFullGlobalModuleIndex && GlobalIndex && !
buildingModule()) {
2364 bool RecreateIndex =
false;
2367 Module *TheModule = I->second;
2370 Path.emplace_back(TriggerLoc,
2372 std::reverse(Path.begin(), Path.end());
2375 RecreateIndex =
true;
2378 if (RecreateIndex) {
2382 .getHeaderSearchInfo()
2383 .getSpecificModuleCachePath())) {
2385 consumeError(std::move(Err));
2388 TheASTReader->resetForReload();
2389 TheASTReader->loadGlobalIndex();
2390 GlobalIndex = TheASTReader->getGlobalIndex();
2392 HaveFullGlobalModuleIndex =
true;
2426 ExternalSemaSrc = std::move(ESS);
Defines the clang::ASTContext interface.
Defines the Diagnostic-related interfaces.
static void collectVFSEntries(CompilerInstance &CI, std::shared_ptr< ModuleDependencyCollector > MDC)
static bool EnableCodeCompletion(Preprocessor &PP, StringRef Filename, unsigned Line, unsigned Column)
static bool compileModuleAndReadAST(CompilerInstance &ImportingInstance, SourceLocation ImportLoc, SourceLocation ModuleNameLoc, Module *Module, ModuleFileName ModuleFileName)
Compile a module in a separate compiler instance and read the AST, returning true if the module compi...
static void SetupSerializedDiagnostics(DiagnosticOptions &DiagOpts, DiagnosticsEngine &Diags, StringRef OutputFile)
static bool readASTAfterCompileModule(CompilerInstance &ImportingInstance, SourceLocation ImportLoc, SourceLocation ModuleNameLoc, Module *Module, ModuleFileName ModuleFileName, bool *OutOfDate, bool *Missing)
Read the AST right after compiling the module.
static CompileOrReadResult compileModuleBehindLockOrRead(CompilerInstance &ImportingInstance, SourceLocation ImportLoc, SourceLocation ModuleNameLoc, Module *Module, ModuleFileName ModuleFileName)
Attempt to compile the module in a separate compiler instance behind a lock (to avoid building the sa...
static Language getLanguageFromOptions(const LangOptions &LangOpts)
Determine the appropriate source input kind based on language options.
static void checkConfigMacro(Preprocessor &PP, StringRef ConfigMacro, Module *Mod, SourceLocation ImportLoc)
Diagnose differences between the current definition of the given configuration macro and the definiti...
static void collectHeaderMaps(const HeaderSearch &HS, std::shared_ptr< ModuleDependencyCollector > MDC)
CompileOrReadResult
The result of compileModuleBehindLockOrRead().
@ Compiled
We successfully compiled the module and we still need to read it.
@ Read
We read a module file compiled by another instance.
@ FailedToRead
We failed to read the module file compiled by another instance.
@ FailedToCompile
We failed to compile the module.
static void InitializeFileRemapping(DiagnosticsEngine &Diags, SourceManager &SourceMgr, FileManager &FileMgr, const PreprocessorOptions &InitOpts)
static void collectIncludePCH(CompilerInstance &CI, std::shared_ptr< ModuleDependencyCollector > MDC)
static OptionalFileEntryRef getPublicModuleMap(FileEntryRef File, FileManager &FileMgr)
static ModuleSource selectModuleSource(Module *M, StringRef ModuleName, ModuleFileName &ModuleFilename, const std::map< std::string, std::string, std::less<> > &BuiltModules, HeaderSearch &HS)
Select a source for loading the named module and compute the filename to load it from.
static bool compileModuleImpl(CompilerInstance &ImportingInstance, SourceLocation ImportLoc, SourceLocation ModuleNameLoc, Module *Module, ModuleFileName ModuleFileName)
Compile a module in a separate compiler instance.
static void checkConfigMacros(Preprocessor &PP, Module *M, SourceLocation ImportLoc)
static void SetUpDiagnosticLog(DiagnosticOptions &DiagOpts, const CodeGenOptions *CodeGenOpts, DiagnosticsEngine &Diags)
Defines the clang::FileManager interface and associated types.
Defines the clang::FrontendAction interface and various convenience abstract classes (clang::ASTFront...
static void print(llvm::raw_ostream &OS, const T &V, ASTContext &ASTCtx, QualType Ty)
static StringRef getTriple(const Command &Job)
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.
Defines the SourceManager interface.
Defines utilities for dealing with stack allocation and stack space.
Defines version macros and version-related utility functions for Clang.
virtual void Initialize(ASTContext &Context)
Initialize - This is called to initialize the consumer, providing the ASTContext.
Holds long-lived AST nodes (such as types and decls) that can be referred to throughout the semantic ...
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.
Abstract interface for callback invocations by the ASTReader.
RAII object to temporarily add an AST callback listener.
@ ARR_Missing
The client can handle an AST file that cannot load because it is missing.
@ ARR_None
The client can't handle any AST loading failures.
@ ARR_ConfigurationMismatch
The client can handle an AST file that cannot load because it's compiled configuration doesn't match ...
@ ARR_OutOfDate
The client can handle an AST file that cannot load because it is out-of-date relative to its input fi...
@ ARR_TreatModuleWithErrorsAsOutOfDate
If a module file is marked with errors treat it as out-of-date so the caller can rebuild it.
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.
ASTReadResult
The result of reading the control block of an AST file, which can fail for various reasons.
@ Success
The control block was read successfully.
@ ConfigurationMismatch
The AST file was written with a different language/target configuration.
@ OutOfDate
The AST file is out-of-date relative to its input files, and needs to be regenerated.
@ Failure
The AST file itself appears corrupted.
@ VersionMismatch
The AST file was written by a different version of Clang.
@ HadErrors
The AST file has errors.
@ Missing
The AST file was missing.
ChainedDiagnosticConsumer - Chain two diagnostic clients so that diagnostics go to the first client a...
Abstract interface for a consumer of code-completion information.
Options controlling the behavior of code completion.
CodeGenOptions - Track various options which control how the code is optimized and passed to the back...
std::string DwarfDebugFlags
The string to embed in the debug information for the compile unit, if non-empty.
CompilerInstance - Helper class for managing a single instance of the Clang compiler.
bool hasOutputManager() const
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
~CompilerInstance() override
void createPreprocessor(TranslationUnitKind TUKind)
Create the preprocessor, using the invocation, file, and source managers, and replace any existing on...
bool lookupMissingImports(StringRef Name, SourceLocation TriggerLoc) override
Check global module index for missing imports.
void setOutputManager(IntrusiveRefCntPtr< llvm::vfs::OutputBackend > NewOutputs)
Set the output manager.
void createDiagnostics(DiagnosticConsumer *Client=nullptr, bool ShouldOwnClient=true)
Create the diagnostics engine using the invocation's diagnostic options and replace any existing one ...
DependencyOutputOptions & getDependencyOutputOpts()
bool hasFileManager() const
TargetInfo * getAuxTarget() const
const PCHContainerReader & getPCHContainerReader() const
Return the appropriate PCHContainerReader depending on the current CodeGenOptions.
DiagnosticsEngine & getDiagnostics() const
Get the current diagnostics engine.
GlobalModuleIndex * loadGlobalModuleIndex(SourceLocation TriggerLoc) override
Load, create, or return global module.
raw_ostream & getVerboseOutputStream()
Get the current stream for verbose output.
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...
void setExternalSemaSource(IntrusiveRefCntPtr< ExternalSemaSource > ESS)
GenModuleActionWrapperFunc getGenModuleActionWrapper() const
ModuleLoadResult loadModule(SourceLocation ImportLoc, ModuleIdPath Path, Module::NameVisibilityKind Visibility, bool IsInclusionDirective) override
Attempt to load the given module.
FileSystemOptions & getFileSystemOpts()
llvm::Timer & getFrontendTimer() const
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.
FileManager & getFileManager() const
Return the current file manager to the caller.
void setBuildGlobalModuleIndex(bool Build)
Set the flag indicating whether we should (re)build the global module index.
void createOutputManager()
Create an output manager.
std::unique_ptr< Sema > takeSema()
std::unique_ptr< llvm::MemoryBuffer > compileModule(SourceLocation ImportLoc, StringRef ModuleName, StringRef ModuleFileName, CompilerInstance &Instance)
Compile a module file for the given module, using the options provided by the importing compiler inst...
void printDiagnosticStats()
At the end of a compilation, print the number of warnings/errors.
void setASTConsumer(std::unique_ptr< ASTConsumer > Value)
setASTConsumer - Replace the current AST consumer; the compiler instance takes ownership of Value.
PreprocessorOutputOptions & getPreprocessorOutputOpts()
IntrusiveRefCntPtr< FileManager > getFileManagerPtr() const
ModuleCache & getModuleCache() const
IntrusiveRefCntPtr< ASTReader > getASTReader() const
void setTarget(TargetInfo *Value)
Replace the current Target.
void setModuleDepCollector(std::shared_ptr< ModuleDependencyCollector > Collector)
void addDependencyCollector(std::shared_ptr< DependencyCollector > Listener)
void createASTContext()
Create the AST context.
std::unique_ptr< raw_pwrite_stream > createOutputFile(StringRef OutputPath, bool Binary, bool RemoveFileOnSignal, bool UseTemporary, bool CreateMissingDirectories=false)
Create a new output file, optionally deriving the output path name, and add it to the list of tracked...
bool hasASTContext() const
void createModuleFromSource(SourceLocation ImportLoc, StringRef ModuleName, StringRef Source) override
Attempt to create the given module from the specified source buffer.
Preprocessor & getPreprocessor() const
Return the current preprocessor.
ASTContext & getASTContext() const
IntrusiveRefCntPtr< llvm::vfs::FileSystem > getVirtualFileSystemPtr() const
void LoadRequestedPlugins()
Load the list of plugins requested in the FrontendOptions.
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()
std::shared_ptr< ModuleDependencyCollector > getModuleDepCollector() const
bool hasDiagnostics() const
void setSema(Sema *S)
Replace the current Sema; the compiler instance takes ownership of S.
std::unique_ptr< CompilerInstance > cloneForModuleCompile(SourceLocation ImportLoc, const Module *Module, StringRef ModuleFileName, std::optional< ThreadSafeCloneConfig > ThreadSafeConfig=std::nullopt)
Creates a new CompilerInstance for compiling a module.
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()
void createFrontendTimer()
Create the frontend timer and replace any existing one with it.
void createSourceManager()
Create the source manager and replace any existing one with it.
CompilerInvocation & getInvocation()
void setVerboseOutputStream(raw_ostream &Value)
Replace the current stream for verbose output.
PreprocessorOptions & getPreprocessorOpts()
ASTConsumer & getASTConsumer() const
void setFileManager(IntrusiveRefCntPtr< FileManager > Value)
Replace the current file manager.
TargetInfo & getTarget() const
llvm::vfs::OutputBackend & getOutputManager()
llvm::vfs::FileSystem & getVirtualFileSystem() const
void createCodeCompletionConsumer()
Create a code completion consumer using the invocation; note that this will cause the source manager ...
void setCodeCompletionConsumer(CodeCompleteConsumer *Value)
setCodeCompletionConsumer - Replace the current code completion consumer; the compiler instance takes...
bool ExecuteAction(FrontendAction &Act)
ExecuteAction - Execute the provided action against the compiler's CompilerInvocation object.
std::shared_ptr< PCHContainerOperations > getPCHContainerOperations() const
void clearOutputFiles(bool EraseFiles)
clearOutputFiles - Clear the output file list.
DiagnosticOptions & getDiagnosticOpts()
LangOptions & getLangOpts()
llvm::vfs::OutputBackend & getOrCreateOutputManager()
CodeGenOptions & getCodeGenOpts()
SourceManager & getSourceManager() const
Return the current source manager.
void setDiagnostics(llvm::IntrusiveRefCntPtr< DiagnosticsEngine > Value)
setDiagnostics - Replace the current diagnostics engine.
bool shouldBuildGlobalModuleIndex() const
Indicates whether we should (re)build the global module index.
bool hasSourceManager() const
bool hasASTConsumer() const
APINotesOptions & getAPINotesOpts()
std::unique_ptr< raw_pwrite_stream > createNullOutputFile()
void setAuxTarget(TargetInfo *Value)
Replace the current AuxTarget.
void makeModuleVisible(Module *Mod, Module::NameVisibilityKind Visibility, SourceLocation ImportLoc) override
Make the given module visible.
bool hasPreprocessor() const
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.
LangOptions & getLangOpts()
Mutable getters.
FrontendOptions & getFrontendOpts()
std::string computeContextHash() const
Compute the context hash - a string that uniquely identifies compiler settings.
DependencyOutputOptions - Options for controlling the compiler dependency file generation.
ShowIncludesDestination ShowIncludesDest
Destination of cl.exe style /showIncludes info.
std::string DOTOutputFile
The file to write GraphViz-formatted header dependencies to.
std::string ModuleDependencyOutputDir
The directory to copy module dependencies to when collecting them.
std::string OutputFile
The file to write dependency output to.
std::string HeaderIncludeOutputFile
The file to write header include output to.
unsigned ShowHeaderIncludes
Show header inclusions (-H).
Abstract interface, implemented by clients of the front-end, which formats and prints fully processed...
unsigned getNumErrors() const
unsigned getNumWarnings() const
static llvm::IntrusiveRefCntPtr< DiagnosticIDs > create()
Options for controlling the compiler diagnostics engine.
std::string DiagnosticLogFile
The file to log diagnostic output to.
std::vector< std::string > SystemHeaderWarningsModules
The list of -Wsystem-headers-in-module=... options used to override whether -Wsystem-headers is enabl...
std::string DiagnosticSerializationFile
The file to serialize diagnostics to (non-appending).
Concrete class used by the front-end to report problems and issues.
DiagnosticBuilder Report(SourceLocation Loc, unsigned DiagID)
Issue the message to the client.
bool hasErrorOccurred() const
void setClient(DiagnosticConsumer *client, bool ShouldOwnClient=true)
Set the diagnostic client associated with this diagnostic object.
std::unique_ptr< DiagnosticConsumer > takeClient()
Return the current diagnostic client along with ownership of that client.
DiagnosticConsumer * getClient()
Level getDiagnosticLevel(unsigned DiagID, SourceLocation Loc) const
Based on the way the client configured the DiagnosticsEngine object, classify the specified diagnosti...
bool ownsClient() const
Determine whether this DiagnosticsEngine object own its client.
StringRef getName() const
A reference to a FileEntry that includes the name of the file as it was accessed by the FileManager's...
StringRef getName() const
The name of this FileEntry.
StringRef getNameAsRequested() const
The name of this FileEntry, as originally requested without applying any remappings for VFS 'use-exte...
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.
void AddStats(const FileManager &Other)
Import statistics from a child FileManager and add them to this current FileManager.
static bool fixupRelativePath(const FileSystemOptions &FileSystemOpts, SmallVectorImpl< char > &Path)
OptionalFileEntryRef getOptionalFileRef(StringRef Filename, bool OpenFile=false, bool CacheFailure=true, bool IsText=true)
Get a FileEntryRef if it exists, without doing anything on error.
static FixItHint CreateReplacement(CharSourceRange RemoveRange, StringRef Code)
Create a code modification hint that replaces the given source range with the given code string.
Abstract base class for actions which can be performed by the frontend.
virtual void EndSourceFile()
Perform any per-file post processing, deallocate per-file objects, and run statistics and output file...
bool PrepareToExecute(CompilerInstance &CI)
Prepare the action to execute on the given compiler instance.
llvm::Error Execute()
Set the source manager's main input file, and run the action.
bool BeginSourceFile(CompilerInstance &CI, const FrontendInputFile &Input)
Prepare the action for processing the input file Input.
virtual bool isModelParsingAction() const
Is this action invoked on a model file?
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 BuildingImplicitModuleUsesLock
Whether to use a filesystem lock when building implicit modules.
unsigned ModulesShareFileManager
Whether to share the FileManager when building modules.
std::optional< std::string > AuxTargetCPU
Auxiliary target CPU for CUDA/HIP compilation.
std::string StatsFile
Filename to write statistics to.
std::string OutputFile
The output file, if any.
std::string ActionName
The name of the action to run when using a plugin action.
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,...
unsigned GenerateGlobalModuleIndex
Whether we can generate the global module index if needed.
unsigned DisableFree
Disable memory freeing on exit.
SmallVector< FrontendInputFile, 0 > Inputs
The input files and their types.
frontend::ActionKind ProgramAction
The frontend action to perform.
std::optional< std::vector< std::string > > AuxTargetFeatures
Auxiliary target features for CUDA/HIP compilation.
unsigned ImplicitModulesLockTimeoutSeconds
The time in seconds to wait on an implicit module lock before timing out.
A global index for a set of module files, providing information about the identifiers within those mo...
llvm::SmallPtrSet< ModuleFile *, 4 > HitSet
A set of module files in which we found a result.
bool lookupIdentifier(llvm::StringRef Name, HitSet &Hits)
Look for all of the module files with information about the given identifier, e.g....
static llvm::Error writeIndex(FileManager &FileMgr, const PCHContainerReader &PCHContainerRdr, llvm::StringRef Path)
Write a global index into the given.
One of these records is kept for each identifier that is lexed.
bool hadMacroDefinition() const
Returns true if this identifier was #defined to some value at any moment.
llvm::MemoryBuffer & addBuiltPCM(llvm::StringRef Filename, std::unique_ptr< llvm::MemoryBuffer > Buffer, off_t Size, time_t ModTime)
Store a just-built PCM under the Filename.
@ FPE_Default
Used internally to represent initial unspecified value.
@ FPE_Ignore
Assume that floating-point exceptions are masked.
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.
Encapsulates the data about a macro definition (e.g.
The module cache used for compiling modules implicitly.
virtual std::error_code write(StringRef Path, llvm::MemoryBufferRef Buffer, off_t &Size, time_t &ModTime)=0
Write the PCM contents to the given path in the module cache.
virtual InMemoryModuleCache & getInMemoryModuleCache()=0
Returns this process's view of the module cache.
virtual void updateModuleTimestamp(StringRef ModuleFilename)=0
Updates the timestamp denoting the last time inputs of the module file were validated.
virtual std::unique_ptr< llvm::AdvisoryLock > getLock(StringRef ModuleFilename)=0
Returns lock for the given module file. The lock is initially unlocked.
Identifies a module file to be loaded.
bool empty() const
Checks whether the module file name is empty.
static ModuleFileName makeExplicit(std::string Name)
Creates a file name for an explicit module.
StringRef str() const
Returns the plain module file name.
Describes the result of attempting to load a module.
bool buildingModule() const
Returns true if this instance is building a module.
ModuleLoader(bool BuildingModule=false)
llvm::StringMap< Module * >::const_iterator module_iterator
module_iterator module_begin() const
OptionalFileEntryRef getModuleMapFileForUniquing(const Module *M) const
std::optional< Module * > getCachedModuleLoad(const IdentifierInfo &II)
Return a cached module load.
module_iterator module_end() const
FileID getContainingModuleMapFileID(const Module *Module) const
Retrieve the module map file containing the definition of the given module.
void resolveLinkAsDependencies(Module *Mod)
Use PendingLinkAsModule information to mark top level link names that are going to be replaced by exp...
void cacheModuleLoad(const IdentifierInfo &II, Module *M)
Cache a module load. M might be nullptr.
void loadAllParsedModules()
Module * findOrLoadModule(StringRef Name)
Describes a module or submodule.
StringRef getTopLevelModuleName() const
Retrieve the name of the top-level module.
std::vector< std::string > ConfigMacros
The set of "configuration macros", which are macros that (intentionally) change how this module is bu...
unsigned IsUnimportable
Whether this module has declared itself unimportable, either because it's missing a requirement from ...
NameVisibilityKind
Describes the visibility of the various names within a particular module.
@ Hidden
All of the names in this module are hidden.
void print(raw_ostream &OS, unsigned Indent=0, bool Dump=false) const
Print the module map for this module to the given stream.
const ModuleFileKey * getASTFileKey() const
The serialized AST file key for this module, if one was created.
SourceLocation DefinitionLoc
The location of the module definition.
unsigned IsSystem
Whether this is a "system" module (which assumes that all headers in it are system headers).
std::string Name
The name of this module.
llvm::iterator_range< submodule_iterator > submodules()
OptionalDirectoryEntryRef Directory
The build directory of this module.
ModuleRef findSubmodule(StringRef Name) const
Find the submodule with the given name.
unsigned IsFromModuleFile
Whether this module was loaded from a module file.
unsigned HasIncompatibleModuleFile
Whether we tried and failed to load a module file for this module.
std::string getFullModuleName(bool AllowStringLiterals=false) const
Retrieve the full name of this module, including the path from its top-level module.
unsigned IsAvailable
Whether this module is available in the current translation unit.
Module * getTopLevelModule()
Retrieve the top-level module for this (sub)module, which may be this module.
This abstract interface provides operations for unwrapping containers for serialized ASTs (precompile...
virtual llvm::StringRef ExtractPCH(llvm::MemoryBufferRef Buffer) const =0
Returns the serialized AST inside the PCH container Buffer.
This interface provides a way to observe the actions of the preprocessor as it does its thing.
@ ReplaceAction
Replace the main action.
PreprocessorOptions - This class is used for passing the various options used in preprocessor initial...
std::vector< std::pair< std::string, std::string > > RemappedFiles
The set of file remappings, which take existing files on the system (the first part of each pair) and...
std::pair< unsigned, bool > PrecompiledPreambleBytes
If non-zero, the implicit PCH include is actually a precompiled preamble that covers this number of b...
bool RemappedFilesKeepOriginalName
True if the SourceManager should report the original file name for contents of files that were remapp...
bool RetainRemappedFileBuffers
Whether the compiler instance should retain (i.e., not free) the buffers associated with remapped fil...
bool DetailedRecord
Whether we should maintain a detailed record of all macro definitions and expansions.
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.
std::vector< std::pair< std::string, bool > > Macros
std::vector< std::pair< std::string, llvm::MemoryBuffer * > > RemappedFileBuffers
The set of file-to-buffer remappings, which take existing files on the system (the first part of each...
Engages in a tight little dance with the lexer to efficiently preprocess tokens.
const MacroInfo * getMacroInfo(const IdentifierInfo *II) const
MacroDirective * getLocalMacroDirectiveHistory(const IdentifierInfo *II) const
Given an identifier, return the latest non-imported macro directive for that identifier.
bool SetCodeCompletionPoint(FileEntryRef File, unsigned Line, unsigned Column)
Specify the point at which code-completion will be performed.
IdentifierInfo * getIdentifierInfo(StringRef Name) const
Return information about the specified preprocessor identifier token.
SourceManager & getSourceManager() const
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.
FileManager & getFileManager() const
FileID getPredefinesFileID() const
Returns the FileID for the preprocessor predefines.
HeaderSearch & getHeaderSearchInfo() const
DiagnosticsEngine & getDiagnostics() const
DiagnosticBuilder Diag(SourceLocation Loc, unsigned DiagID) const
Forwarding function for diagnostics.
A simple code-completion consumer that prints the results it receives in a simple format.
Sema - This implements semantic analysis and AST building for C.
ASTReaderListenter implementation to set SuggestedPredefines of ASTReader which is required to use a ...
Encodes a location in the source.
bool isValid() const
Return true if this is a valid SourceLocation object.
This class handles loading and caching of source files into memory.
ModuleBuildStack getModuleBuildStack() const
Retrieve the module build stack.
A trivial tuple used to represent a source range.
SourceLocation getBegin() const
CharacteristicKind getFileCharacteristic() const
Return whether this is a system header or not.
This is a discriminated union of FileInfo and ExpansionInfo.
const FileInfo & getFile() const
Exposes information about the current target.
static TargetInfo * CreateTargetInfo(DiagnosticsEngine &Diags, TargetOptions &Opts)
Construct a target for the given options.
const llvm::Triple & getTriple() const
Returns the target triple of the primary target.
virtual void setAuxTarget(const TargetInfo *Aux)
void noSignedCharForObjCBool()
virtual void adjust(DiagnosticsEngine &Diags, LangOptions &Opts, const TargetInfo *Aux)
Set forced language options.
std::string CPU
If given, the name of the target CPU to generate code for.
VerifyDiagnosticConsumer - Create a diagnostic client which will use markers in the input source to c...
Information about a module that has been loaded by the ASTReader.
Defines the clang::TargetInfo interface.
CharacteristicKind
Indicates whether a file or directory holds normal user code, system code, or system code which is im...
@ HeaderSearch
Remove unused header search paths including header maps.
@ OS
Indicates that the tracking object is a descendant of a referenced-counted OSObject,...
@ PluginAction
Run a plugin action,.
@ RewriteObjC
ObjC->C Rewriter.
bool Inv(InterpState &S, CodePtr OpPC)
@ MK_PCH
File is a PCH file treated as such.
@ MK_Preamble
File is a PCH file treated as the preamble.
@ MK_ExplicitModule
File is an explicitly-loaded module.
@ MK_ImplicitModule
File is an implicitly-loaded module.
@ MK_PrebuiltModule
File is from a prebuilt module path.
std::unique_ptr< DiagnosticConsumer > create(StringRef OutputFile, DiagnosticOptions &DiagOpts, bool MergeChildRecords=false)
Returns a DiagnosticConsumer that serializes diagnostics to a bitcode file.
The JSON file list parser is used to communicate input to InstallAPI.
CustomizableOptional< FileEntryRef > OptionalFileEntryRef
ArrayRef< IdentifierLoc > ModuleIdPath
A sequence of identifier/location pairs used to describe a particular module or submodule,...
ArrayRef< std::pair< std::string, FullSourceLoc > > ModuleBuildStack
The stack used when building modules on demand, which is used to provide a link between the source ma...
void ApplyHeaderSearchOptions(HeaderSearch &HS, const HeaderSearchOptions &HSOpts, const LangOptions &Lang, const llvm::Triple &triple)
Apply the header search options to get given HeaderSearch object.
std::shared_ptr< ModuleCache > createCrossProcessModuleCache()
Creates new ModuleCache backed by a file system directory that may be operated on by multiple process...
nullptr
This class represents a compute construct, representing a 'Kind' of ‘parallel’, 'serial',...
void InitializePreprocessor(Preprocessor &PP, const PreprocessorOptions &PPOpts, const PCHContainerReader &PCHContainerRdr, const FrontendOptions &FEOpts, const CodeGenOptions &CodeGenOpts)
InitializePreprocessor - Initialize the preprocessor getting it and the environment ready to process ...
LLVM_READONLY bool isAlphanumeric(unsigned char c)
Return true if this character is an ASCII letter or digit: [a-zA-Z0-9].
@ Module
Module linkage, which indicates that the entity can be referred to from other translation units withi...
Language
The language for the input, used to select and validate the language standard and possible actions.
@ C
Languages that the frontend can parse and compile.
@ Result
The result type of a method or function.
IntrusiveRefCntPtr< llvm::vfs::FileSystem > createVFSFromCompilerInvocation(const CompilerInvocation &CI, DiagnosticsEngine &Diags)
constexpr size_t DesiredStackSize
The amount of stack space that Clang would like to be provided with.
void noteBottomOfStack(bool ForceSet=false)
Call this once on each thread, as soon after starting the thread as feasible, to note the approximate...
void ProcessWarningOptions(DiagnosticsEngine &Diags, const DiagnosticOptions &Opts, llvm::vfs::FileSystem &VFS, bool ReportDiags=true)
ProcessWarningOptions - Initialize the diagnostic client and process the warning options specified on...
TranslationUnitKind
Describes the kind of translation unit being processed.
void AttachHeaderIncludeGen(Preprocessor &PP, const DependencyOutputOptions &DepOpts, bool ShowAllHeaders=false, StringRef OutputPath={}, bool ShowDepth=true, bool MSStyle=false)
AttachHeaderIncludeGen - Create a header include list generator, and attach it to the given preproces...
DisableValidationForModuleKind
Whether to disable the normal validation performed on precompiled headers and module files when they ...
@ Other
Other implicit parameter.
Visibility
Describes the different kinds of visibility that a declaration may have.
void AttachDependencyGraphGen(Preprocessor &PP, StringRef OutputFile, StringRef SysRoot)
AttachDependencyGraphGen - Create a dependency graph generator, and attach it to the given preprocess...
A source location that has been parsed on the command line.