clang 23.0.0git
DependencyScannerImpl.cpp
Go to the documentation of this file.
1//===- DependencyScannerImpl.cpp - Implements module dependency scanning --===//
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
15#include "clang/Driver/Driver.h"
17#include "llvm/ADT/IntrusiveRefCntPtr.h"
18#include "llvm/ADT/ScopeExit.h"
19#include "llvm/Support/AdvisoryLock.h"
20#include "llvm/Support/CrashRecoveryContext.h"
21#include "llvm/Support/VirtualFileSystem.h"
22#include "llvm/TargetParser/Host.h"
23
24#include <mutex>
25#include <thread>
26
27using namespace clang;
28using namespace dependencies;
29
30namespace {
31/// Forwards the gatherered dependencies to the consumer.
32class DependencyConsumerForwarder : public DependencyFileGenerator {
33public:
34 DependencyConsumerForwarder(std::unique_ptr<DependencyOutputOptions> Opts,
35 StringRef WorkingDirectory, DependencyConsumer &C)
36 : DependencyFileGenerator(*Opts), WorkingDirectory(WorkingDirectory),
37 Opts(std::move(Opts)), C(C) {}
38
39 void finishedMainFile(DiagnosticsEngine &Diags) override {
40 C.handleDependencyOutputOpts(*Opts);
41 llvm::SmallString<256> CanonPath;
42 for (const auto &File : getDependencies()) {
43 CanonPath = File;
44 llvm::sys::path::remove_dots(CanonPath, /*remove_dot_dot=*/true);
45 llvm::sys::path::make_absolute(WorkingDirectory, CanonPath);
46 C.handleFileDependency(CanonPath);
47 }
48 }
49
50private:
51 StringRef WorkingDirectory;
52 std::unique_ptr<DependencyOutputOptions> Opts;
53 DependencyConsumer &C;
54};
55
56static bool checkHeaderSearchPaths(const HeaderSearchOptions &HSOpts,
57 const HeaderSearchOptions &ExistingHSOpts,
58 DiagnosticsEngine *Diags,
59 const LangOptions &LangOpts) {
60 if (LangOpts.Modules) {
61 if (HSOpts.VFSOverlayFiles != ExistingHSOpts.VFSOverlayFiles) {
62 if (Diags) {
63 Diags->Report(diag::warn_pch_vfsoverlay_mismatch);
64 auto VFSNote = [&](int Type, ArrayRef<std::string> VFSOverlays) {
65 if (VFSOverlays.empty()) {
66 Diags->Report(diag::note_pch_vfsoverlay_empty) << Type;
67 } else {
68 std::string Files = llvm::join(VFSOverlays, "\n");
69 Diags->Report(diag::note_pch_vfsoverlay_files) << Type << Files;
70 }
71 };
72 VFSNote(0, HSOpts.VFSOverlayFiles);
73 VFSNote(1, ExistingHSOpts.VFSOverlayFiles);
74 }
75 }
76 }
77 return false;
78}
79
80using PrebuiltModuleFilesT = decltype(HeaderSearchOptions::PrebuiltModuleFiles);
81
82/// A listener that collects the imported modules and the input
83/// files. While visiting, collect vfsoverlays and file inputs that determine
84/// whether prebuilt modules fully resolve in stable directories.
85class PrebuiltModuleListener : public ASTReaderListener {
86public:
87 PrebuiltModuleListener(PrebuiltModuleFilesT &PrebuiltModuleFiles,
88 llvm::SmallVector<std::string> &NewModuleFiles,
89 PrebuiltModulesAttrsMap &PrebuiltModulesASTMap,
90 const HeaderSearchOptions &HSOpts,
91 const LangOptions &LangOpts, DiagnosticsEngine &Diags,
92 const ArrayRef<StringRef> StableDirs)
93 : PrebuiltModuleFiles(PrebuiltModuleFiles),
94 NewModuleFiles(NewModuleFiles),
95 PrebuiltModulesASTMap(PrebuiltModulesASTMap), ExistingHSOpts(HSOpts),
96 ExistingLangOpts(LangOpts), Diags(Diags), StableDirs(StableDirs) {}
97
98 bool needsImportVisitation() const override { return true; }
99 bool needsInputFileVisitation() override { return true; }
100 bool needsSystemInputFileVisitation() override { return true; }
101
102 /// Accumulate the modules are transitively depended on by the initial
103 /// prebuilt module.
104 void visitImport(StringRef ModuleName, StringRef Filename) override {
105 if (PrebuiltModuleFiles.insert({ModuleName.str(), Filename.str()}).second)
106 NewModuleFiles.push_back(Filename.str());
107
108 auto PrebuiltMapEntry = PrebuiltModulesASTMap.try_emplace(Filename);
109 PrebuiltModuleASTAttrs &PrebuiltModule = PrebuiltMapEntry.first->second;
110 if (PrebuiltMapEntry.second)
111 PrebuiltModule.setInStableDir(!StableDirs.empty());
112
113 if (auto It = PrebuiltModulesASTMap.find(CurrentFile);
114 It != PrebuiltModulesASTMap.end() && CurrentFile != Filename)
115 PrebuiltModule.addDependent(It->getKey());
116 }
117
118 /// For each input file discovered, check whether it's external path is in a
119 /// stable directory. Traversal is stopped if the current module is not
120 /// considered stable.
121 bool visitInputFileAsRequested(StringRef FilenameAsRequested,
122 StringRef Filename, bool isSystem,
123 bool isOverridden, time_t StoredTime,
124 bool isExplicitModule) override {
125 if (StableDirs.empty())
126 return false;
127 auto PrebuiltEntryIt = PrebuiltModulesASTMap.find(CurrentFile);
128 if ((PrebuiltEntryIt == PrebuiltModulesASTMap.end()) ||
129 (!PrebuiltEntryIt->second.isInStableDir()))
130 return false;
131
132 PrebuiltEntryIt->second.setInStableDir(
133 isPathInStableDir(StableDirs, Filename));
134 return PrebuiltEntryIt->second.isInStableDir();
135 }
136
137 /// Update which module that is being actively traversed.
138 void visitModuleFile(StringRef Filename,
139 serialization::ModuleKind Kind) override {
140 // If the CurrentFile is not
141 // considered stable, update any of it's transitive dependents.
142 auto PrebuiltEntryIt = PrebuiltModulesASTMap.find(CurrentFile);
143 if ((PrebuiltEntryIt != PrebuiltModulesASTMap.end()) &&
144 !PrebuiltEntryIt->second.isInStableDir())
145 PrebuiltEntryIt->second.updateDependentsNotInStableDirs(
146 PrebuiltModulesASTMap);
147 CurrentFile = Filename;
148 }
149
150 /// Check the header search options for a given module when considering
151 /// if the module comes from stable directories.
152 bool ReadHeaderSearchOptions(const HeaderSearchOptions &HSOpts,
153 StringRef ModuleFilename, StringRef ContextHash,
154 bool Complain) override {
155
156 auto PrebuiltMapEntry = PrebuiltModulesASTMap.try_emplace(CurrentFile);
157 PrebuiltModuleASTAttrs &PrebuiltModule = PrebuiltMapEntry.first->second;
158 if (PrebuiltMapEntry.second)
159 PrebuiltModule.setInStableDir(!StableDirs.empty());
160
161 if (PrebuiltModule.isInStableDir())
162 PrebuiltModule.setInStableDir(areOptionsInStableDir(StableDirs, HSOpts));
163
164 return false;
165 }
166
167 /// Accumulate vfsoverlays used to build these prebuilt modules.
168 bool ReadHeaderSearchPaths(const HeaderSearchOptions &HSOpts,
169 bool Complain) override {
170
171 auto PrebuiltMapEntry = PrebuiltModulesASTMap.try_emplace(CurrentFile);
172 PrebuiltModuleASTAttrs &PrebuiltModule = PrebuiltMapEntry.first->second;
173 if (PrebuiltMapEntry.second)
174 PrebuiltModule.setInStableDir(!StableDirs.empty());
175
176 PrebuiltModule.setVFS(
177 llvm::StringSet<>(llvm::from_range, HSOpts.VFSOverlayFiles));
178
179 return checkHeaderSearchPaths(
180 HSOpts, ExistingHSOpts, Complain ? &Diags : nullptr, ExistingLangOpts);
181 }
182
183private:
184 PrebuiltModuleFilesT &PrebuiltModuleFiles;
185 llvm::SmallVector<std::string> &NewModuleFiles;
186 PrebuiltModulesAttrsMap &PrebuiltModulesASTMap;
187 const HeaderSearchOptions &ExistingHSOpts;
188 const LangOptions &ExistingLangOpts;
189 DiagnosticsEngine &Diags;
190 std::string CurrentFile;
191 const ArrayRef<StringRef> StableDirs;
192};
193
194/// Visit the given prebuilt module and collect all of the modules it
195/// transitively imports and contributing input files.
196static bool visitPrebuiltModule(StringRef PrebuiltModuleFilename,
198 PrebuiltModuleFilesT &ModuleFiles,
199 PrebuiltModulesAttrsMap &PrebuiltModulesASTMap,
200 DiagnosticsEngine &Diags,
201 const ArrayRef<StringRef> StableDirs) {
202 // List of module files to be processed.
204
205 PrebuiltModuleListener Listener(ModuleFiles, Worklist, PrebuiltModulesASTMap,
207 Diags, StableDirs);
208
209 Listener.visitModuleFile(PrebuiltModuleFilename,
212 PrebuiltModuleFilename, CI.getFileManager(), CI.getModuleCache(),
214 /*FindModuleFileExtensions=*/false, Listener,
215 /*ValidateDiagnosticOptions=*/false, ASTReader::ARR_OutOfDate))
216 return true;
217
218 while (!Worklist.empty()) {
219 Listener.visitModuleFile(Worklist.back(), serialization::MK_ExplicitModule);
221 Worklist.pop_back_val(), CI.getFileManager(), CI.getModuleCache(),
223 /*FindModuleFileExtensions=*/false, Listener,
224 /*ValidateDiagnosticOptions=*/false))
225 return true;
226 }
227 return false;
228}
229
230/// Transform arbitrary file name into an object-like file name.
231static std::string makeObjFileName(StringRef FileName) {
232 SmallString<128> ObjFileName(FileName);
233 llvm::sys::path::replace_extension(ObjFileName, "o");
234 return std::string(ObjFileName);
235}
236
237/// Deduce the dependency target based on the output file and input files.
238static std::string
239deduceDepTarget(const std::string &OutputFile,
240 const SmallVectorImpl<FrontendInputFile> &InputFiles) {
241 if (OutputFile != "-")
242 return OutputFile;
243
244 if (InputFiles.empty() || !InputFiles.front().isFile())
245 return "clang-scan-deps\\ dependency";
246
247 return makeObjFileName(InputFiles.front().getFile());
248}
249
250// Clang implements -D and -U by splatting text into a predefines buffer. This
251// allows constructs such as `-DFඞ=3 "-D F\u{0D9E} 4 3 2”` to be accepted and
252// define the same macro, or adding C++ style comments before the macro name.
253//
254// This function checks that the first non-space characters in the macro
255// obviously form an identifier that can be uniqued on without lexing. Failing
256// to do this could lead to changing the final definition of a macro.
257//
258// We could set up a preprocessor and actually lex the name, but that's very
259// heavyweight for a situation that will almost never happen in practice.
260static std::optional<StringRef> getSimpleMacroName(StringRef Macro) {
261 StringRef Name = Macro.split("=").first.ltrim(" \t");
262 std::size_t I = 0;
263
264 auto FinishName = [&]() -> std::optional<StringRef> {
265 StringRef SimpleName = Name.slice(0, I);
266 if (SimpleName.empty())
267 return std::nullopt;
268 return SimpleName;
269 };
270
271 for (; I != Name.size(); ++I) {
272 switch (Name[I]) {
273 case '(': // Start of macro parameter list
274 case ' ': // End of macro name
275 case '\t':
276 return FinishName();
277 case '_':
278 continue;
279 default:
280 if (llvm::isAlnum(Name[I]))
281 continue;
282 return std::nullopt;
283 }
284 }
285 return FinishName();
286}
287} // namespace
288
290 using MacroOpt = std::pair<StringRef, std::size_t>;
291 std::vector<MacroOpt> SimpleNames;
292 SimpleNames.reserve(PPOpts.Macros.size());
293 std::size_t Index = 0;
294 for (const auto &M : PPOpts.Macros) {
295 auto SName = getSimpleMacroName(M.first);
296 // Skip optimizing if we can't guarantee we can preserve relative order.
297 if (!SName)
298 return;
299 SimpleNames.emplace_back(*SName, Index);
300 ++Index;
301 }
302
303 llvm::stable_sort(SimpleNames, llvm::less_first());
304 // Keep the last instance of each macro name by going in reverse
305 auto NewEnd = std::unique(
306 SimpleNames.rbegin(), SimpleNames.rend(),
307 [](const MacroOpt &A, const MacroOpt &B) { return A.first == B.first; });
308 SimpleNames.erase(SimpleNames.begin(), NewEnd.base());
309
310 // Apply permutation.
311 decltype(PPOpts.Macros) NewMacros;
312 NewMacros.reserve(SimpleNames.size());
313 for (std::size_t I = 0, E = SimpleNames.size(); I != E; ++I) {
314 std::size_t OriginalIndex = SimpleNames[I].second;
315 // We still emit undefines here as they may be undefining a predefined macro
316 NewMacros.push_back(std::move(PPOpts.Macros[OriginalIndex]));
317 }
318 std::swap(PPOpts.Macros, NewMacros);
319}
320
321namespace {
322class ScanningDependencyDirectivesGetter : public DependencyDirectivesGetter {
324
325public:
326 ScanningDependencyDirectivesGetter(FileManager &FileMgr) : DepFS(nullptr) {
327 FileMgr.getVirtualFileSystem().visit([&](llvm::vfs::FileSystem &FS) {
328 auto *DFS = llvm::dyn_cast<DependencyScanningWorkerFilesystem>(&FS);
329 if (DFS) {
330 assert(!DepFS && "Found multiple scanning VFSs");
331 DepFS = DFS;
332 }
333 });
334 assert(DepFS && "Did not find scanning VFS");
335 }
336
337 std::unique_ptr<DependencyDirectivesGetter>
338 cloneFor(FileManager &FileMgr) override {
339 return std::make_unique<ScanningDependencyDirectivesGetter>(FileMgr);
340 }
341
342 std::optional<ArrayRef<dependency_directives_scan::Directive>>
343 operator()(FileEntryRef File) override {
344 return DepFS->getDirectiveTokens(File.getName());
345 }
346};
347
348/// Sanitize diagnostic options for dependency scan.
349void sanitizeDiagOpts(DiagnosticOptions &DiagOpts) {
350 // Don't print 'X warnings and Y errors generated'.
351 DiagOpts.ShowCarets = false;
352 // Don't write out diagnostic file.
353 DiagOpts.DiagnosticSerializationFile.clear();
354 // Don't emit warnings except for scanning specific warnings.
355 // TODO: It would be useful to add a more principled way to ignore all
356 // warnings that come from source code. The issue is that we need to
357 // ignore warnings that could be surpressed by
358 // `#pragma clang diagnostic`, while still allowing some scanning
359 // warnings for things we're not ready to turn into errors yet.
360 // See `test/ClangScanDeps/diagnostic-pragmas.c` for an example.
361 llvm::erase_if(DiagOpts.Warnings, [](StringRef Warning) {
362 return llvm::StringSwitch<bool>(Warning)
363 .Cases({"pch-vfs-diff", "error=pch-vfs-diff"}, false)
364 .StartsWith("no-error=", false)
365 .Default(true);
366 });
367}
368} // namespace
369
370std::unique_ptr<DiagnosticOptions>
372 std::vector<const char *> CLI;
373 for (const std::string &Arg : CommandLine)
374 CLI.push_back(Arg.c_str());
375 auto DiagOpts = CreateAndPopulateDiagOpts(CLI);
376 sanitizeDiagOpts(*DiagOpts);
377 return DiagOpts;
378}
379
381 ArrayRef<std::string> CommandLine,
383 std::vector<const char *> CCommandLine(CommandLine.size(), nullptr);
384 llvm::transform(CommandLine, CCommandLine.begin(),
385 [](const std::string &Str) { return Str.c_str(); });
386 DiagOpts = CreateAndPopulateDiagOpts(CCommandLine);
387 sanitizeDiagOpts(*DiagOpts);
389 /*ShouldOwnClient=*/false);
390}
391
392std::unique_ptr<CompilerInvocation>
394 DiagnosticsEngine &Diags) {
395 llvm::opt::ArgStringList Argv;
396 for (const std::string &Str : ArrayRef(CommandLine).drop_front())
397 Argv.push_back(Str.c_str());
398
399 auto Invocation = std::make_unique<CompilerInvocation>();
400 if (!CompilerInvocation::CreateFromArgs(*Invocation, Argv, Diags)) {
401 // FIXME: Should we just go on like cc1_main does?
402 return nullptr;
403 }
404 return Invocation;
405}
406
408 CompilerInstance &ScanInstance,
410 DiagnosticConsumer *DiagConsumer, DependencyScanningService &Service,
412 ScanInstance.setBuildingModule(false);
413 ScanInstance.createVirtualFileSystem(FS, DiagConsumer);
414 ScanInstance.createDiagnostics(DiagConsumer, /*ShouldOwnClient=*/false);
415 ScanInstance.createFileManager();
416 ScanInstance.createSourceManager();
417
418 // Use DepFS for getting the dependency directives if requested to do so.
420 DepFS->resetBypassedPathPrefix();
421 SmallString<256> ModulesCachePath;
424 ModulesCachePath);
425 if (!ModulesCachePath.empty())
426 DepFS->setBypassedPathPrefix(ModulesCachePath);
427
429 std::make_unique<ScanningDependencyDirectivesGetter>(
430 ScanInstance.getFileManager()));
431 }
432}
433
434std::shared_ptr<CompilerInvocation> dependencies::createScanCompilerInvocation(
435 const CompilerInvocation &Invocation,
436 const DependencyScanningService &Service,
437 DependencyActionController &Controller) {
438 auto ScanInvocation = std::make_shared<CompilerInvocation>(Invocation);
439
440 sanitizeDiagOpts(ScanInvocation->getDiagnosticOpts());
441
442 ScanInvocation->getPreprocessorOpts().AllowPCHWithDifferentModulesCachePath =
443 true;
444
445 if (ScanInvocation->getHeaderSearchOpts().ModulesValidateOncePerBuildSession)
446 ScanInvocation->getHeaderSearchOpts().BuildSessionTimestamp =
448
449 ScanInvocation->getFrontendOpts().DisableFree = false;
450 ScanInvocation->getFrontendOpts().GenerateGlobalModuleIndex = false;
451 ScanInvocation->getFrontendOpts().UseGlobalModuleIndex = false;
452 ScanInvocation->getFrontendOpts().GenReducedBMI = false;
453 ScanInvocation->getFrontendOpts().ModuleOutputPath.clear();
454 // This will prevent us compiling individual modules asynchronously since
455 // FileManager is not thread-safe, but it does improve performance for now.
456 ScanInvocation->getFrontendOpts().ModulesShareFileManager = true;
457 ScanInvocation->getHeaderSearchOpts().ModuleFormat = "raw";
458 ScanInvocation->getHeaderSearchOpts().ModulesIncludeVFSUsage =
460
461 // Consider different header search and diagnostic options to create
462 // different modules. This avoids the unsound aliasing of module PCMs.
463 //
464 // TODO: Implement diagnostic bucketing to reduce the impact of strict
465 // context hashing.
466 ScanInvocation->getHeaderSearchOpts().ModulesStrictContextHash = true;
467 ScanInvocation->getHeaderSearchOpts().ModulesSerializeOnlyPreprocessor = true;
468 ScanInvocation->getHeaderSearchOpts().ModulesSkipDiagnosticOptions = true;
469 ScanInvocation->getHeaderSearchOpts().ModulesSkipHeaderSearchPaths = true;
470 ScanInvocation->getHeaderSearchOpts().ModulesSkipPragmaDiagnosticMappings =
471 true;
472 ScanInvocation->getHeaderSearchOpts().ModulesForceValidateUserHeaders = false;
473
474 // Ensure that the scanner does not create new dependency collectors,
475 // and thus won't write out the extra '.d' files to disk.
476 ScanInvocation->getDependencyOutputOpts() = {};
477
478 Controller.initializeScanInvocation(*ScanInvocation);
479
480 return ScanInvocation;
481}
482
485 // Create a collection of stable directories derived from the ScanInstance
486 // for determining whether module dependencies would fully resolve from
487 // those directories.
489 const StringRef Sysroot = ScanInstance.getHeaderSearchOpts().Sysroot;
490 if (!Sysroot.empty() && (llvm::sys::path::root_directory(Sysroot) != Sysroot))
491 StableDirs = {Sysroot, ScanInstance.getHeaderSearchOpts().ResourceDir};
492 return StableDirs;
493}
494
495std::optional<PrebuiltModulesAttrsMap>
497 CompilerInstance &ScanInstance, llvm::SmallVector<StringRef> &StableDirs) {
498 // Store a mapping of prebuilt module files and their properties like header
499 // search options. This will prevent the implicit build to create duplicate
500 // modules and will force reuse of the existing prebuilt module files
501 // instead.
502 PrebuiltModulesAttrsMap PrebuiltModulesASTMap;
503
504 if (!ScanInstance.getPreprocessorOpts().ImplicitPCHInclude.empty())
505 if (visitPrebuiltModule(
506 ScanInstance.getPreprocessorOpts().ImplicitPCHInclude, ScanInstance,
508 PrebuiltModulesASTMap, ScanInstance.getDiagnostics(), StableDirs))
509 return {};
510
511 return PrebuiltModulesASTMap;
512}
513
514std::unique_ptr<DependencyOutputOptions>
516 const CompilerInvocation &Invocation) {
517 auto Opts = std::make_unique<DependencyOutputOptions>(
518 Invocation.getDependencyOutputOpts());
519 // We need at least one -MT equivalent for the generator of make dependency
520 // files to work.
521 if (Opts->Targets.empty())
522 Opts->Targets = {deduceDepTarget(Invocation.getFrontendOpts().OutputFile,
523 Invocation.getFrontendOpts().Inputs)};
524 Opts->IncludeSystemHeaders = true;
525
526 return Opts;
527}
528
529std::shared_ptr<ModuleDepCollector>
531 CompilerInstance &ScanInstance,
532 std::unique_ptr<DependencyOutputOptions> DepOutputOpts,
533 StringRef WorkingDirectory, DependencyConsumer &Consumer,
535 DependencyActionController &Controller,
536 PrebuiltModulesAttrsMap PrebuiltModulesASTMap,
537 llvm::SmallVector<StringRef> &StableDirs) {
538 std::shared_ptr<ModuleDepCollector> MDC;
539 switch (Service.getOpts().Format) {
541 ScanInstance.addDependencyCollector(
542 std::make_shared<DependencyConsumerForwarder>(
543 std::move(DepOutputOpts), WorkingDirectory, Consumer));
544 break;
547 MDC = std::make_shared<ModuleDepCollector>(
548 Service, std::move(DepOutputOpts), ScanInstance, Consumer, Controller,
549 Inv, std::move(PrebuiltModulesASTMap), StableDirs);
550 ScanInstance.addDependencyCollector(MDC);
551 break;
552 }
553
554 return MDC;
555}
556
557/// Manages (and terminates) the asynchronous compilation of modules.
559 std::mutex Mutex;
560 bool Stop = false;
561 // FIXME: Have the service own a thread pool and use that instead.
562 std::vector<std::thread> Compiles;
563
564public:
565 /// Registers the module compilation, unless this instance is about to be
566 /// destroyed.
567 void add(llvm::unique_function<void()> Compile) {
568 std::lock_guard<std::mutex> Lock(Mutex);
569 if (!Stop)
570 Compiles.emplace_back(std::move(Compile));
571 }
572
574 {
575 // Prevent registration of further module compiles.
576 std::lock_guard<std::mutex> Lock(Mutex);
577 Stop = true;
578 }
579
580 // Wait for outstanding module compiles to finish.
581 for (std::thread &Compile : Compiles)
582 Compile.join();
583 }
584};
585
598
599/// The preprocessor callback that takes care of initiating an asynchronous
600/// module compilation if needed.
606
611
612 void moduleLoadSkipped(Module *M) override {
613 M = M->getTopLevelModule();
614
615 HeaderSearch &HS = CI.getPreprocessor().getHeaderSearchInfo();
616 ModuleCache &ModCache = CI.getModuleCache();
617 std::string ModuleFileName = HS.getCachedModuleFileName(M);
618
619 uint64_t Timestamp = ModCache.getModuleTimestamp(ModuleFileName);
620 // Someone else already built/validated the PCM.
621 if (Timestamp > CI.getHeaderSearchOpts().BuildSessionTimestamp)
622 return;
623
624 if (!CI.getASTReader())
625 CI.createASTReader();
627 // Only calling ReadASTCore() to avoid the expensive eager deserialization
628 // of the clang::Module objects in ReadAST().
629 // FIXME: Consider doing this in the new thread depending on how expensive
630 // the read turns out to be.
631 switch (CI.getASTReader()->ReadASTCore(
633 nullptr, Imported, {}, {}, {},
637 // We successfully read a valid, up-to-date PCM.
638 // FIXME: This could update the timestamp. Regular calls to
639 // ASTReader::ReadAST() would do so unless they encountered corrupted
640 // AST block, corrupted extension block, or did not read the expected
641 // top-level module.
642 return;
645 // The most interesting case.
646 break;
647 default:
648 // Let the regular scan diagnose this.
649 return;
650 }
651
652 ModCache.prepareForGetLock(ModuleFileName);
653 auto Lock = ModCache.getLock(ModuleFileName);
654 bool Owned;
655 llvm::Error LockErr = Lock->tryLock().moveInto(Owned);
656 // Someone else is building the PCM right now.
657 if (!LockErr && !Owned)
658 return;
659 // We should build the PCM.
661 llvm::makeIntrusiveRefCnt<DependencyScanningWorkerFilesystem>(
662 Service.getSharedCache(), Service.getOpts().MakeVFS());
663 VFS = createVFSFromCompilerInvocation(CI.getInvocation(),
664 CI.getDiagnostics(), std::move(VFS));
665 auto DC = std::make_unique<DiagnosticConsumer>();
666 auto MC = makeInProcessModuleCache(Service.getModuleCacheEntries());
667 CompilerInstance::ThreadSafeCloneConfig CloneConfig(std::move(VFS), *DC,
668 std::move(MC));
669 auto ModCI1 = CI.cloneForModuleCompile(SourceLocation(), M, ModuleFileName,
670 CloneConfig);
671 auto ModCI2 = CI.cloneForModuleCompile(SourceLocation(), M, ModuleFileName,
672 CloneConfig);
673
674 auto ModController = Controller.clone();
675
676 // Note: This lock belongs to a module cache that might not outlive the
677 // thread. This works, because the in-process lock only refers to an object
678 // managed by the service, which does outlive the thread.
679 Compiles.add([Lock = std::move(Lock), ModCI1 = std::move(ModCI1),
680 ModCI2 = std::move(ModCI2), DC = std::move(DC),
681 ModController = std::move(ModController), Service = &Service,
682 Compiles = &Compiles] {
683 llvm::CrashRecoveryContext CRC;
684 (void)CRC.RunSafely([&] {
685 // Quickly discovers and compiles modules for the real scan below.
686 SingleModuleWithAsyncModuleCompiles Action1(*Service, *ModController,
687 *Compiles);
688 (void)ModCI1->ExecuteAction(Action1);
689 // The real scan below.
690 ModCI2->getPreprocessorOpts().SingleModuleParseMode = false;
691 GenerateModuleFromModuleMapAction Action2;
692 (void)ModCI2->ExecuteAction(Action2);
693 });
694 });
695 }
696};
697
698/// Runs the preprocessor on a TU with single-module-parse-mode and compiles
699/// modules asynchronously without blocking or importing them.
717
719 CompilerInstance &CI) {
722 std::make_unique<AsyncModuleCompile>(CI, Service, Controller, Compiles));
723 return true;
724}
725
727 std::string Executable,
728 std::unique_ptr<CompilerInvocation> OriginalInvocation,
730 std::shared_ptr<PCHContainerOperations> PCHContainerOps,
731 DiagnosticConsumer *DiagConsumer) {
732 // Making sure that we canonicalize the defines early to avoid unnecessary
733 // variants in both the scanner and in the resulting explicit command lines.
734 if (any(Service.getOpts().OptimizeArgs & ScanningOptimizations::Macros))
735 canonicalizeDefines(OriginalInvocation->getPreprocessorOpts());
736
737 if (Scanned) {
738 CompilerInstance &ScanInstance = *ScanInstanceStorage;
739
740 // Scanning runs once for the first -cc1 invocation in a chain of driver
741 // jobs. For any dependent jobs, reuse the scanning result and just
742 // update the new invocation.
743 // FIXME: to support multi-arch builds, each arch requires a separate scan
744 if (MDC)
745 MDC->applyDiscoveredDependencies(*OriginalInvocation);
746
747 if (!Controller.finalize(ScanInstance, *OriginalInvocation))
748 return false;
749
750 Consumer.handleBuildCommand(
751 {Executable, OriginalInvocation->getCC1CommandLine()});
752 return true;
753 }
754
755 Scanned = true;
756
757 // Create a compiler instance to handle the actual work.
758 auto ScanInvocation =
759 createScanCompilerInvocation(*OriginalInvocation, Service, Controller);
760
761 // Quickly discovers and compiles modules for the real scan below.
762 std::optional<AsyncModuleCompiles> AsyncCompiles;
763 if (Service.getOpts().AsyncScanModules) {
764 auto ModCache = makeInProcessModuleCache(Service.getModuleCacheEntries());
765 auto ScanInstanceStorage = std::make_unique<CompilerInstance>(
766 std::make_shared<CompilerInvocation>(*ScanInvocation), PCHContainerOps,
767 std::move(ModCache));
768 CompilerInstance &ScanInstance = *ScanInstanceStorage;
769
770 DiagnosticConsumer DiagConsumer;
771 initializeScanCompilerInstance(ScanInstance, FS, &DiagConsumer, Service,
772 DepFS);
773
774 // FIXME: Do this only once.
775 SmallVector<StringRef> StableDirs = getInitialStableDirs(ScanInstance);
776 auto MaybePrebuiltModulesASTMap =
777 computePrebuiltModulesASTMap(ScanInstance, StableDirs);
778 if (!MaybePrebuiltModulesASTMap)
779 return false;
780
781 // Normally this would be handled by GeneratePCHAction
783 ScanInstance.getLangOpts().CompilingPCH = true;
784
785 AsyncCompiles.emplace();
786 SingleTUWithAsyncModuleCompiles Action(Service, Controller, *AsyncCompiles);
787 (void)ScanInstance.ExecuteAction(Action);
788 }
789
790 auto ModCache = makeInProcessModuleCache(Service.getModuleCacheEntries());
791 ScanInstanceStorage.emplace(std::move(ScanInvocation),
792 std::move(PCHContainerOps), std::move(ModCache));
793 CompilerInstance &ScanInstance = *ScanInstanceStorage;
794
795 initializeScanCompilerInstance(ScanInstance, FS, DiagConsumer, Service,
796 DepFS);
797
798 llvm::SmallVector<StringRef> StableDirs = getInitialStableDirs(ScanInstance);
799 auto MaybePrebuiltModulesASTMap =
800 computePrebuiltModulesASTMap(ScanInstance, StableDirs);
801 if (!MaybePrebuiltModulesASTMap)
802 return false;
803
804 auto DepOutputOpts = createDependencyOutputOptions(*OriginalInvocation);
805
807 ScanInstance, std::move(DepOutputOpts), WorkingDirectory, Consumer,
808 Service, *OriginalInvocation, Controller, *MaybePrebuiltModulesASTMap,
809 StableDirs);
810
811 if (ScanInstance.getDiagnostics().hasErrorOccurred())
812 return false;
813
814 if (!Controller.initialize(ScanInstance, *OriginalInvocation))
815 return false;
816
818 const bool Result = ScanInstance.ExecuteAction(Action);
819
820 if (Result) {
821 if (MDC)
822 MDC->applyDiscoveredDependencies(*OriginalInvocation);
823
824 if (!Controller.finalize(ScanInstance, *OriginalInvocation))
825 return false;
826
827 Consumer.handleBuildCommand(
828 {Executable, OriginalInvocation->getCC1CommandLine()});
829 }
830
831 return Result;
832}
Manages (and terminates) the asynchronous compilation of modules.
void add(llvm::unique_function< void()> Compile)
Registers the module compilation, unless this instance is about to be destroyed.
Abstract interface for callback invocations by the ASTReader.
Definition ASTReader.h:117
@ ARR_Missing
The client can handle an AST file that cannot load because it is missing.
Definition ASTReader.h:1847
@ ARR_OutOfDate
The client can handle an AST file that cannot load because it is out-of-date relative to its input fi...
Definition ASTReader.h:1851
@ ARR_TreatModuleWithErrorsAsOutOfDate
If a module file is marked with errors treat it as out-of-date so the caller can rebuild it.
Definition ASTReader.h:1864
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.
@ Success
The control block was read successfully.
Definition ASTReader.h:450
@ OutOfDate
The AST file is out-of-date relative to its input files, and needs to be regenerated.
Definition ASTReader.h:460
@ Missing
The AST file was missing.
Definition ASTReader.h:456
Configuration object for making the result of cloneForModuleCompile() thread-safe.
CompilerInstance - Helper class for managing a single instance of the Clang compiler.
void createDiagnostics(DiagnosticConsumer *Client=nullptr, bool ShouldOwnClient=true)
Create the diagnostics engine using the invocation's diagnostic options and replace any existing one ...
const PCHContainerReader & getPCHContainerReader() const
Return the appropriate PCHContainerReader depending on the current CodeGenOptions.
DiagnosticsEngine & getDiagnostics() const
Get the current diagnostics engine.
void createFileManager()
Create the file manager and replace any existing one with it.
FileManager & getFileManager() const
Return the current file manager to the caller.
ModuleCache & getModuleCache() const
void addDependencyCollector(std::shared_ptr< DependencyCollector > Listener)
Preprocessor & getPreprocessor() const
Return the current preprocessor.
void createVirtualFileSystem(IntrusiveRefCntPtr< llvm::vfs::FileSystem > BaseFS=llvm::vfs::getRealFileSystem(), DiagnosticConsumer *DC=nullptr)
Create a virtual file system instance based on the invocation.
FrontendOptions & getFrontendOpts()
HeaderSearchOptions & getHeaderSearchOpts()
void createSourceManager()
Create the source manager and replace any existing one with it.
CompilerInvocation & getInvocation()
PreprocessorOptions & getPreprocessorOpts()
bool ExecuteAction(FrontendAction &Act)
ExecuteAction - Execute the provided action against the compiler's CompilerInvocation object.
void setDependencyDirectivesGetter(std::unique_ptr< DependencyDirectivesGetter > Getter)
Helper class for holding the data necessary to invoke the compiler.
PreprocessorOptions & getPreprocessorOpts()
static bool CreateFromArgs(CompilerInvocation &Res, ArrayRef< const char * > CommandLineArgs, DiagnosticsEngine &Diags, const char *Argv0=nullptr)
Create a compiler invocation from a list of input options.
DependencyOutputOptions & getDependencyOutputOpts()
FrontendOptions & getFrontendOpts()
Functor that returns the dependency directives for a given file.
Builds a dependency file when attached to a Preprocessor (for includes) and ASTReader (for module imp...
Definition Utils.h:104
Abstract interface, implemented by clients of the front-end, which formats and prints fully processed...
Options for controlling the compiler diagnostics engine.
std::vector< std::string > Warnings
The list of -W... options used to alter the diagnostic mappings, with the prefixes removed.
std::string DiagnosticSerializationFile
The file to serialize diagnostics to (non-appending).
Concrete class used by the front-end to report problems and issues.
Definition Diagnostic.h:232
DiagnosticBuilder Report(SourceLocation Loc, unsigned DiagID)
Issue the message to the client.
bool hasErrorOccurred() const
Definition Diagnostic.h:880
Implements support for file system lookup, file system caching, and directory search management.
Definition FileManager.h:53
std::string OutputFile
The output file, if any.
SmallVector< FrontendInputFile, 0 > Inputs
The input files and their types.
frontend::ActionKind ProgramAction
The frontend action to perform.
HeaderSearchOptions - Helper class for storing options related to the initialization of the HeaderSea...
std::map< std::string, std::string, std::less<> > PrebuiltModuleFiles
The mapping of module names to prebuilt module files.
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.
std::string ResourceDir
The directory which holds the compiler resource files (builtin includes, etc.).
Encapsulates the information needed to find the file referenced by a #include or #include_next,...
std::string getCachedModuleFileName(Module *Module)
Retrieve the name of the cached module file that should be used to load the given module.
Keeps track of the various options that can be enabled, which controls the dialect of C or C++ that i...
The module cache used for compiling modules implicitly.
Definition ModuleCache.h:25
virtual void prepareForGetLock(StringRef ModuleFilename)=0
May perform any work that only needs to be performed once for multiple calls getLock() with the same ...
virtual std::time_t getModuleTimestamp(StringRef ModuleFilename)=0
Returns 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.
void setBuildingModule(bool BuildingModuleFlag)
Flag indicating whether this instance is building a module.
Describes a module or submodule.
Definition Module.h:144
Module * getTopLevelModule()
Retrieve the top-level module for this (sub)module, which may be this module.
Definition Module.h:722
This interface provides a way to observe the actions of the preprocessor as it does its thing.
Definition PPCallbacks.h:37
PreprocessorOptions - This class is used for passing the various options used in preprocessor initial...
std::string ImplicitPCHInclude
The implicit PCH included at the start of the translation unit, or empty.
bool SingleModuleParseMode
When enabled, preprocessor is in a mode for parsing a single module only.
std::vector< std::pair< std::string, bool > > Macros
void addPPCallbacks(std::unique_ptr< PPCallbacks > C)
Preprocessor-based frontend action that also loads PCH files.
Encodes a location in the source.
The base class of the type hierarchy.
Definition TypeBase.h:1839
Dependency scanner callbacks that are used during scanning to influence the behaviour of the scan - f...
virtual void initializeScanInvocation(CompilerInvocation &ScanInvocation)
Initializes the scan invocation.
bool runInvocation(std::string Executable, std::unique_ptr< CompilerInvocation > Invocation, IntrusiveRefCntPtr< llvm::vfs::FileSystem > FS, std::shared_ptr< PCHContainerOperations > PCHContainerOps, DiagnosticConsumer *DiagConsumer)
The dependency scanning service contains shared configuration and state that is used by the individua...
const DependencyScanningServiceOptions & getOpts() const
A virtual file system optimized for the dependency discovery.
void setVFS(llvm::StringSet<> &&VFS)
Update the VFSMap to the one discovered from serializing the AST file.
bool isInStableDir() const
Read-only access to whether the module is made up of dependencies in stable directories.
void addDependent(StringRef ModuleFile)
Add a direct dependent module file, so it can be updated if the current module is from stable directo...
void setInStableDir(bool V=false)
Update whether the prebuilt module resolves entirely in a stable directories.
SmallVector< StringRef > getInitialStableDirs(const CompilerInstance &ScanInstance)
bool areOptionsInStableDir(const ArrayRef< StringRef > Directories, const HeaderSearchOptions &HSOpts)
Determine if options collected from a module's compilation can safely be considered as stable.
@ VFS
Remove unused -ivfsoverlay arguments.
std::shared_ptr< CompilerInvocation > createScanCompilerInvocation(const CompilerInvocation &Invocation, const DependencyScanningService &Service, DependencyActionController &Controller)
Creates a CompilerInvocation suitable for the dependency scanner.
void initializeScanCompilerInstance(CompilerInstance &ScanInstance, IntrusiveRefCntPtr< llvm::vfs::FileSystem > FS, DiagnosticConsumer *DiagConsumer, DependencyScanningService &Service, IntrusiveRefCntPtr< DependencyScanningWorkerFilesystem > DepFS)
llvm::StringMap< PrebuiltModuleASTAttrs > PrebuiltModulesAttrsMap
Attributes loaded from AST files of prebuilt modules collected prior to ModuleDepCollector creation.
std::unique_ptr< CompilerInvocation > createCompilerInvocation(ArrayRef< std::string > CommandLine, DiagnosticsEngine &Diags)
void canonicalizeDefines(PreprocessorOptions &PPOpts)
Canonicalizes command-line macro defines (e.g. removing "-DX -UX").
std::unique_ptr< DependencyOutputOptions > createDependencyOutputOptions(const CompilerInvocation &Invocation)
Creates dependency output options to be reported to the dependency consumer, deducing missing informa...
std::optional< PrebuiltModulesAttrsMap > computePrebuiltModulesASTMap(CompilerInstance &ScanInstance, SmallVector< StringRef > &StableDirs)
std::shared_ptr< ModuleCache > makeInProcessModuleCache(ModuleCacheEntries &Entries)
std::shared_ptr< ModuleDepCollector > initializeScanInstanceDependencyCollector(CompilerInstance &ScanInstance, std::unique_ptr< DependencyOutputOptions > DepOutputOpts, StringRef WorkingDirectory, DependencyConsumer &Consumer, DependencyScanningService &Service, CompilerInvocation &Inv, DependencyActionController &Controller, PrebuiltModulesAttrsMap PrebuiltModulesASTMap, llvm::SmallVector< StringRef > &StableDirs)
Create the dependency collector that will collect the produced dependencies.
std::unique_ptr< DiagnosticOptions > createDiagOptions(ArrayRef< std::string > CommandLine)
bool isPathInStableDir(const ArrayRef< StringRef > Directories, const StringRef Input)
Determine if Input can be resolved within a stable directory.
@ Make
This is the Makefile compatible dep format.
@ Full
This outputs the full clang module dependency graph suitable for use for explicitly building modules.
@ P1689
This outputs the dependency graph for standard c++ modules in P1689R5 format.
@ DependencyDirectivesScan
This mode is used to compute the dependencies by running the preprocessor with special kind of lexing...
@ GeneratePCH
Generate pre-compiled header.
ModuleKind
Specifies the kind of module that has been loaded.
Definition ModuleFile.h:43
@ MK_ExplicitModule
File is an explicitly-loaded module.
Definition ModuleFile.h:48
@ MK_ImplicitModule
File is an implicitly-loaded module.
Definition ModuleFile.h:45
The JSON file list parser is used to communicate input to InstallAPI.
std::unique_ptr< DiagnosticOptions > CreateAndPopulateDiagOpts(ArrayRef< const char * > Argv)
nullptr
This class represents a compute construct, representing a 'Kind' of ‘parallel’, 'serial',...
@ Result
The result type of a method or function.
Definition TypeBase.h:905
IntrusiveRefCntPtr< llvm::vfs::FileSystem > createVFSFromCompilerInvocation(const CompilerInvocation &CI, DiagnosticsEngine &Diags)
void normalizeModuleCachePath(FileManager &FileMgr, StringRef Path, SmallVectorImpl< char > &NormalizedPath)
int __ovld __cnfn any(char)
Returns 1 if the most significant bit in any component of x is set; otherwise returns 0.
DependencyActionController & Controller
AsyncModuleCompile(CompilerInstance &CI, DependencyScanningService &Service, DependencyActionController &Controller, AsyncModuleCompiles &Compiles)
void moduleLoadSkipped(Module *M) override
Callback invoked whenever a module load was skipped due to enabled single-module-parse-mode.
AsyncModuleCompiles & Compiles
DependencyScanningService & Service
bool BeginSourceFileAction(CompilerInstance &CI) override
Callback at the start of processing a single input.
SingleModuleWithAsyncModuleCompiles(DependencyScanningService &Service, DependencyActionController &Controller, AsyncModuleCompiles &Compiles)
Runs the preprocessor on a TU with single-module-parse-mode and compiles modules asynchronously witho...
bool BeginSourceFileAction(CompilerInstance &CI) override
Callback at the start of processing a single input.
SingleTUWithAsyncModuleCompiles(DependencyScanningService &Service, DependencyActionController &Controller, AsyncModuleCompiles &Compiles)
DependencyScanningService & Service
DependencyActionController & Controller
ScanningMode Mode
Whether to use optimized dependency directive scan or full preprocessing.
std::time_t BuildSessionTimestamp
The build session timestamp for validate-once-per-build-session logic.
ScanningOptimizations OptimizeArgs
How to optimize resulting explicit module command lines.
ScanningOutputFormat Format
What output format are we expected to produce.
IntrusiveRefCntPtr< DiagnosticsEngine > DiagEngine
DiagnosticsEngineWithDiagOpts(ArrayRef< std::string > CommandLine, IntrusiveRefCntPtr< llvm::vfs::FileSystem > FS, DiagnosticConsumer &DC)