//===- DependencyScanningWorker.cpp - clang-scan-deps worker --------------===// // // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. // See https://llvm.org/LICENSE.txt for license information. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception // //===----------------------------------------------------------------------===// #include "clang/Tooling/DependencyScanning/DependencyScanningWorker.h" #include "clang/Basic/DiagnosticDriver.h" #include "clang/Basic/DiagnosticFrontend.h" #include "clang/Basic/DiagnosticSerialization.h" #include "clang/CodeGen/ObjectFilePCHContainerOperations.h" #include "clang/Driver/Compilation.h" #include "clang/Driver/Driver.h" #include "clang/Driver/Job.h" #include "clang/Driver/Tool.h" #include "clang/Frontend/CompilerInstance.h" #include "clang/Frontend/CompilerInvocation.h" #include "clang/Frontend/FrontendActions.h" #include "clang/Frontend/TextDiagnosticPrinter.h" #include "clang/Frontend/Utils.h" #include "clang/Lex/PreprocessorOptions.h" #include "clang/Tooling/DependencyScanning/DependencyScanningService.h" #include "clang/Tooling/DependencyScanning/ModuleDepCollector.h" #include "clang/Tooling/Tooling.h" #include "llvm/ADT/ScopeExit.h" #include "llvm/Support/Allocator.h" #include "llvm/Support/Error.h" #include "llvm/TargetParser/Host.h" #include using namespace clang; using namespace tooling; using namespace dependencies; namespace { /// Forwards the gatherered dependencies to the consumer. class DependencyConsumerForwarder : public DependencyFileGenerator { public: DependencyConsumerForwarder(std::unique_ptr Opts, StringRef WorkingDirectory, DependencyConsumer &C) : DependencyFileGenerator(*Opts), WorkingDirectory(WorkingDirectory), Opts(std::move(Opts)), C(C) {} void finishedMainFile(DiagnosticsEngine &Diags) override { C.handleDependencyOutputOpts(*Opts); llvm::SmallString<256> CanonPath; for (const auto &File : getDependencies()) { CanonPath = File; llvm::sys::path::remove_dots(CanonPath, /*remove_dot_dot=*/true); llvm::sys::fs::make_absolute(WorkingDirectory, CanonPath); C.handleFileDependency(CanonPath); } } private: StringRef WorkingDirectory; std::unique_ptr Opts; DependencyConsumer &C; }; static bool checkHeaderSearchPaths(const HeaderSearchOptions &HSOpts, const HeaderSearchOptions &ExistingHSOpts, DiagnosticsEngine *Diags, const LangOptions &LangOpts) { if (LangOpts.Modules) { if (HSOpts.VFSOverlayFiles != ExistingHSOpts.VFSOverlayFiles) { if (Diags) { Diags->Report(diag::warn_pch_vfsoverlay_mismatch); auto VFSNote = [&](int Type, ArrayRef VFSOverlays) { if (VFSOverlays.empty()) { Diags->Report(diag::note_pch_vfsoverlay_empty) << Type; } else { std::string Files = llvm::join(VFSOverlays, "\n"); Diags->Report(diag::note_pch_vfsoverlay_files) << Type << Files; } }; VFSNote(0, HSOpts.VFSOverlayFiles); VFSNote(1, ExistingHSOpts.VFSOverlayFiles); } } } return false; } using PrebuiltModuleFilesT = decltype(HeaderSearchOptions::PrebuiltModuleFiles); /// A listener that collects the imported modules and optionally the input /// files. class PrebuiltModuleListener : public ASTReaderListener { public: PrebuiltModuleListener(PrebuiltModuleFilesT &PrebuiltModuleFiles, llvm::SmallVector &NewModuleFiles, PrebuiltModuleVFSMapT &PrebuiltModuleVFSMap, const HeaderSearchOptions &HSOpts, const LangOptions &LangOpts, DiagnosticsEngine &Diags) : PrebuiltModuleFiles(PrebuiltModuleFiles), NewModuleFiles(NewModuleFiles), PrebuiltModuleVFSMap(PrebuiltModuleVFSMap), ExistingHSOpts(HSOpts), ExistingLangOpts(LangOpts), Diags(Diags) {} bool needsImportVisitation() const override { return true; } void visitImport(StringRef ModuleName, StringRef Filename) override { if (PrebuiltModuleFiles.insert({ModuleName.str(), Filename.str()}).second) NewModuleFiles.push_back(Filename.str()); } void visitModuleFile(StringRef Filename, serialization::ModuleKind Kind) override { CurrentFile = Filename; } bool ReadHeaderSearchPaths(const HeaderSearchOptions &HSOpts, bool Complain) override { std::vector VFSOverlayFiles = HSOpts.VFSOverlayFiles; PrebuiltModuleVFSMap.insert( {CurrentFile, llvm::StringSet<>(VFSOverlayFiles)}); return checkHeaderSearchPaths( HSOpts, ExistingHSOpts, Complain ? &Diags : nullptr, ExistingLangOpts); } private: PrebuiltModuleFilesT &PrebuiltModuleFiles; llvm::SmallVector &NewModuleFiles; PrebuiltModuleVFSMapT &PrebuiltModuleVFSMap; const HeaderSearchOptions &ExistingHSOpts; const LangOptions &ExistingLangOpts; DiagnosticsEngine &Diags; std::string CurrentFile; }; /// Visit the given prebuilt module and collect all of the modules it /// transitively imports and contributing input files. static bool visitPrebuiltModule(StringRef PrebuiltModuleFilename, CompilerInstance &CI, PrebuiltModuleFilesT &ModuleFiles, PrebuiltModuleVFSMapT &PrebuiltModuleVFSMap, DiagnosticsEngine &Diags) { // List of module files to be processed. llvm::SmallVector Worklist; PrebuiltModuleListener Listener(ModuleFiles, Worklist, PrebuiltModuleVFSMap, CI.getHeaderSearchOpts(), CI.getLangOpts(), Diags); Listener.visitModuleFile(PrebuiltModuleFilename, serialization::MK_ExplicitModule); if (ASTReader::readASTFileControlBlock( PrebuiltModuleFilename, CI.getFileManager(), CI.getModuleCache(), CI.getPCHContainerReader(), /*FindModuleFileExtensions=*/false, Listener, /*ValidateDiagnosticOptions=*/false, ASTReader::ARR_OutOfDate)) return true; while (!Worklist.empty()) { Listener.visitModuleFile(Worklist.back(), serialization::MK_ExplicitModule); if (ASTReader::readASTFileControlBlock( Worklist.pop_back_val(), CI.getFileManager(), CI.getModuleCache(), CI.getPCHContainerReader(), /*FindModuleFileExtensions=*/false, Listener, /*ValidateDiagnosticOptions=*/false)) return true; } return false; } /// Transform arbitrary file name into an object-like file name. static std::string makeObjFileName(StringRef FileName) { SmallString<128> ObjFileName(FileName); llvm::sys::path::replace_extension(ObjFileName, "o"); return std::string(ObjFileName); } /// Deduce the dependency target based on the output file and input files. static std::string deduceDepTarget(const std::string &OutputFile, const SmallVectorImpl &InputFiles) { if (OutputFile != "-") return OutputFile; if (InputFiles.empty() || !InputFiles.front().isFile()) return "clang-scan-deps\\ dependency"; return makeObjFileName(InputFiles.front().getFile()); } /// Sanitize diagnostic options for dependency scan. static void sanitizeDiagOpts(DiagnosticOptions &DiagOpts) { // Don't print 'X warnings and Y errors generated'. DiagOpts.ShowCarets = false; // Don't write out diagnostic file. DiagOpts.DiagnosticSerializationFile.clear(); // Don't emit warnings except for scanning specific warnings. // TODO: It would be useful to add a more principled way to ignore all // warnings that come from source code. The issue is that we need to // ignore warnings that could be surpressed by // `#pragma clang diagnostic`, while still allowing some scanning // warnings for things we're not ready to turn into errors yet. // See `test/ClangScanDeps/diagnostic-pragmas.c` for an example. llvm::erase_if(DiagOpts.Warnings, [](StringRef Warning) { return llvm::StringSwitch(Warning) .Cases("pch-vfs-diff", "error=pch-vfs-diff", false) .StartsWith("no-error=", false) .Default(true); }); } // Clang implements -D and -U by splatting text into a predefines buffer. This // allows constructs such as `-DFඞ=3 "-D F\u{0D9E} 4 3 2”` to be accepted and // define the same macro, or adding C++ style comments before the macro name. // // This function checks that the first non-space characters in the macro // obviously form an identifier that can be uniqued on without lexing. Failing // to do this could lead to changing the final definition of a macro. // // We could set up a preprocessor and actually lex the name, but that's very // heavyweight for a situation that will almost never happen in practice. static std::optional getSimpleMacroName(StringRef Macro) { StringRef Name = Macro.split("=").first.ltrim(" \t"); std::size_t I = 0; auto FinishName = [&]() -> std::optional { StringRef SimpleName = Name.slice(0, I); if (SimpleName.empty()) return std::nullopt; return SimpleName; }; for (; I != Name.size(); ++I) { switch (Name[I]) { case '(': // Start of macro parameter list case ' ': // End of macro name case '\t': return FinishName(); case '_': continue; default: if (llvm::isAlnum(Name[I])) continue; return std::nullopt; } } return FinishName(); } static void canonicalizeDefines(PreprocessorOptions &PPOpts) { using MacroOpt = std::pair; std::vector SimpleNames; SimpleNames.reserve(PPOpts.Macros.size()); std::size_t Index = 0; for (const auto &M : PPOpts.Macros) { auto SName = getSimpleMacroName(M.first); // Skip optimizing if we can't guarantee we can preserve relative order. if (!SName) return; SimpleNames.emplace_back(*SName, Index); ++Index; } llvm::stable_sort(SimpleNames, llvm::less_first()); // Keep the last instance of each macro name by going in reverse auto NewEnd = std::unique( SimpleNames.rbegin(), SimpleNames.rend(), [](const MacroOpt &A, const MacroOpt &B) { return A.first == B.first; }); SimpleNames.erase(SimpleNames.begin(), NewEnd.base()); // Apply permutation. decltype(PPOpts.Macros) NewMacros; NewMacros.reserve(SimpleNames.size()); for (std::size_t I = 0, E = SimpleNames.size(); I != E; ++I) { std::size_t OriginalIndex = SimpleNames[I].second; // We still emit undefines here as they may be undefining a predefined macro NewMacros.push_back(std::move(PPOpts.Macros[OriginalIndex])); } std::swap(PPOpts.Macros, NewMacros); } /// A clang tool that runs the preprocessor in a mode that's optimized for /// dependency scanning for the given compiler invocation. class DependencyScanningAction : public tooling::ToolAction { public: DependencyScanningAction( StringRef WorkingDirectory, DependencyConsumer &Consumer, DependencyActionController &Controller, llvm::IntrusiveRefCntPtr DepFS, ScanningOutputFormat Format, ScanningOptimizations OptimizeArgs, bool EagerLoadModules, bool DisableFree, std::optional ModuleName = std::nullopt) : WorkingDirectory(WorkingDirectory), Consumer(Consumer), Controller(Controller), DepFS(std::move(DepFS)), Format(Format), OptimizeArgs(OptimizeArgs), EagerLoadModules(EagerLoadModules), DisableFree(DisableFree), ModuleName(ModuleName) {} bool runInvocation(std::shared_ptr Invocation, FileManager *DriverFileMgr, std::shared_ptr PCHContainerOps, DiagnosticConsumer *DiagConsumer) override { // Make a deep copy of the original Clang invocation. CompilerInvocation OriginalInvocation(*Invocation); // Restore the value of DisableFree, which may be modified by Tooling. OriginalInvocation.getFrontendOpts().DisableFree = DisableFree; if (any(OptimizeArgs & ScanningOptimizations::Macros)) canonicalizeDefines(OriginalInvocation.getPreprocessorOpts()); if (Scanned) { // Scanning runs once for the first -cc1 invocation in a chain of driver // jobs. For any dependent jobs, reuse the scanning result and just // update the LastCC1Arguments to correspond to the new invocation. // FIXME: to support multi-arch builds, each arch requires a separate scan setLastCC1Arguments(std::move(OriginalInvocation)); return true; } Scanned = true; // Create a compiler instance to handle the actual work. ScanInstanceStorage.emplace(std::move(PCHContainerOps)); CompilerInstance &ScanInstance = *ScanInstanceStorage; ScanInstance.setInvocation(std::move(Invocation)); // Create the compiler's actual diagnostics engine. sanitizeDiagOpts(ScanInstance.getDiagnosticOpts()); ScanInstance.createDiagnostics(DiagConsumer, /*ShouldOwnClient=*/false); if (!ScanInstance.hasDiagnostics()) return false; // Some DiagnosticConsumers require that finish() is called. auto DiagConsumerFinisher = llvm::make_scope_exit([DiagConsumer]() { DiagConsumer->finish(); }); ScanInstance.getPreprocessorOpts().AllowPCHWithDifferentModulesCachePath = true; ScanInstance.getFrontendOpts().GenerateGlobalModuleIndex = false; ScanInstance.getFrontendOpts().UseGlobalModuleIndex = false; ScanInstance.getFrontendOpts().ModulesShareFileManager = false; ScanInstance.getHeaderSearchOpts().ModuleFormat = "raw"; ScanInstance.getHeaderSearchOpts().ModulesIncludeVFSUsage = any(OptimizeArgs & ScanningOptimizations::VFS); // Support for virtual file system overlays. auto FS = createVFSFromCompilerInvocation( ScanInstance.getInvocation(), ScanInstance.getDiagnostics(), DriverFileMgr->getVirtualFileSystemPtr()); // Create a new FileManager to match the invocation's FileSystemOptions. auto *FileMgr = ScanInstance.createFileManager(FS); ScanInstance.createSourceManager(*FileMgr); // Store the list of prebuilt module files into header search options. This // will prevent the implicit build to create duplicate modules and will // force reuse of the existing prebuilt module files instead. PrebuiltModuleVFSMapT PrebuiltModuleVFSMap; if (!ScanInstance.getPreprocessorOpts().ImplicitPCHInclude.empty()) if (visitPrebuiltModule( ScanInstance.getPreprocessorOpts().ImplicitPCHInclude, ScanInstance, ScanInstance.getHeaderSearchOpts().PrebuiltModuleFiles, PrebuiltModuleVFSMap, ScanInstance.getDiagnostics())) return false; // Use the dependency scanning optimized file system if requested to do so. if (DepFS) ScanInstance.getPreprocessorOpts().DependencyDirectivesForFile = [LocalDepFS = DepFS](FileEntryRef File) -> std::optional> { if (llvm::ErrorOr Entry = LocalDepFS->getOrCreateFileSystemEntry(File.getName())) if (LocalDepFS->ensureDirectiveTokensArePopulated(*Entry)) return Entry->getDirectiveTokens(); return std::nullopt; }; // Create the dependency collector that will collect the produced // dependencies. // // This also moves the existing dependency output options from the // invocation to the collector. The options in the invocation are reset, // which ensures that the compiler won't create new dependency collectors, // and thus won't write out the extra '.d' files to disk. auto Opts = std::make_unique(); std::swap(*Opts, ScanInstance.getInvocation().getDependencyOutputOpts()); // We need at least one -MT equivalent for the generator of make dependency // files to work. if (Opts->Targets.empty()) Opts->Targets = { deduceDepTarget(ScanInstance.getFrontendOpts().OutputFile, ScanInstance.getFrontendOpts().Inputs)}; Opts->IncludeSystemHeaders = true; switch (Format) { case ScanningOutputFormat::Make: ScanInstance.addDependencyCollector( std::make_shared( std::move(Opts), WorkingDirectory, Consumer)); break; case ScanningOutputFormat::P1689: case ScanningOutputFormat::Full: MDC = std::make_shared( std::move(Opts), ScanInstance, Consumer, Controller, OriginalInvocation, std::move(PrebuiltModuleVFSMap), OptimizeArgs, EagerLoadModules, Format == ScanningOutputFormat::P1689); ScanInstance.addDependencyCollector(MDC); break; } // Consider different header search and diagnostic options to create // different modules. This avoids the unsound aliasing of module PCMs. // // TODO: Implement diagnostic bucketing to reduce the impact of strict // context hashing. ScanInstance.getHeaderSearchOpts().ModulesStrictContextHash = true; ScanInstance.getHeaderSearchOpts().ModulesSkipDiagnosticOptions = true; ScanInstance.getHeaderSearchOpts().ModulesSkipHeaderSearchPaths = true; ScanInstance.getHeaderSearchOpts().ModulesSkipPragmaDiagnosticMappings = true; // Avoid some checks and module map parsing when loading PCM files. ScanInstance.getPreprocessorOpts().ModulesCheckRelocated = false; std::unique_ptr Action; if (ModuleName) Action = std::make_unique(*ModuleName); else Action = std::make_unique(); if (ScanInstance.getDiagnostics().hasErrorOccurred()) return false; // Each action is responsible for calling finish. DiagConsumerFinisher.release(); const bool Result = ScanInstance.ExecuteAction(*Action); if (Result) setLastCC1Arguments(std::move(OriginalInvocation)); // Propagate the statistics to the parent FileManager. DriverFileMgr->AddStats(ScanInstance.getFileManager()); return Result; } bool hasScanned() const { return Scanned; } /// Take the cc1 arguments corresponding to the most recent invocation used /// with this action. Any modifications implied by the discovered dependencies /// will have already been applied. std::vector takeLastCC1Arguments() { std::vector Result; std::swap(Result, LastCC1Arguments); // Reset LastCC1Arguments to empty. return Result; } private: void setLastCC1Arguments(CompilerInvocation &&CI) { if (MDC) MDC->applyDiscoveredDependencies(CI); LastCC1Arguments = CI.getCC1CommandLine(); } private: StringRef WorkingDirectory; DependencyConsumer &Consumer; DependencyActionController &Controller; llvm::IntrusiveRefCntPtr DepFS; ScanningOutputFormat Format; ScanningOptimizations OptimizeArgs; bool EagerLoadModules; bool DisableFree; std::optional ModuleName; std::optional ScanInstanceStorage; std::shared_ptr MDC; std::vector LastCC1Arguments; bool Scanned = false; }; } // end anonymous namespace DependencyScanningWorker::DependencyScanningWorker( DependencyScanningService &Service, llvm::IntrusiveRefCntPtr FS) : Format(Service.getFormat()), OptimizeArgs(Service.getOptimizeArgs()), EagerLoadModules(Service.shouldEagerLoadModules()) { PCHContainerOps = std::make_shared(); // We need to read object files from PCH built outside the scanner. PCHContainerOps->registerReader( std::make_unique()); // The scanner itself writes only raw ast files. PCHContainerOps->registerWriter(std::make_unique()); switch (Service.getMode()) { case ScanningMode::DependencyDirectivesScan: DepFS = new DependencyScanningWorkerFilesystem(Service.getSharedCache(), FS); BaseFS = DepFS; break; case ScanningMode::CanonicalPreprocessing: DepFS = nullptr; BaseFS = FS; break; } } llvm::Error DependencyScanningWorker::computeDependencies( StringRef WorkingDirectory, const std::vector &CommandLine, DependencyConsumer &Consumer, DependencyActionController &Controller, std::optional ModuleName) { std::vector CLI; for (const std::string &Arg : CommandLine) CLI.push_back(Arg.c_str()); auto DiagOpts = CreateAndPopulateDiagOpts(CLI); sanitizeDiagOpts(*DiagOpts); // Capture the emitted diagnostics and report them to the client // in the case of a failure. std::string DiagnosticOutput; llvm::raw_string_ostream DiagnosticsOS(DiagnosticOutput); TextDiagnosticPrinter DiagPrinter(DiagnosticsOS, DiagOpts.release()); if (computeDependencies(WorkingDirectory, CommandLine, Consumer, Controller, DiagPrinter, ModuleName)) return llvm::Error::success(); return llvm::make_error(DiagnosticsOS.str(), llvm::inconvertibleErrorCode()); } static bool forEachDriverJob( ArrayRef ArgStrs, DiagnosticsEngine &Diags, FileManager &FM, llvm::function_ref Callback) { SmallVector Argv; Argv.reserve(ArgStrs.size()); for (const std::string &Arg : ArgStrs) Argv.push_back(Arg.c_str()); llvm::vfs::FileSystem *FS = &FM.getVirtualFileSystem(); std::unique_ptr Driver = std::make_unique( Argv[0], llvm::sys::getDefaultTargetTriple(), Diags, "clang LLVM compiler", FS); Driver->setTitle("clang_based_tool"); llvm::BumpPtrAllocator Alloc; bool CLMode = driver::IsClangCL( driver::getDriverMode(Argv[0], ArrayRef(Argv).slice(1))); if (llvm::Error E = driver::expandResponseFiles(Argv, CLMode, Alloc, FS)) { Diags.Report(diag::err_drv_expand_response_file) << llvm::toString(std::move(E)); return false; } const std::unique_ptr Compilation( Driver->BuildCompilation(llvm::ArrayRef(Argv))); if (!Compilation) return false; if (Compilation->containsError()) return false; for (const driver::Command &Job : Compilation->getJobs()) { if (!Callback(Job)) return false; } return true; } static bool createAndRunToolInvocation( std::vector CommandLine, DependencyScanningAction &Action, FileManager &FM, std::shared_ptr &PCHContainerOps, DiagnosticsEngine &Diags, DependencyConsumer &Consumer) { // Save executable path before providing CommandLine to ToolInvocation std::string Executable = CommandLine[0]; ToolInvocation Invocation(std::move(CommandLine), &Action, &FM, PCHContainerOps); Invocation.setDiagnosticConsumer(Diags.getClient()); Invocation.setDiagnosticOptions(&Diags.getDiagnosticOptions()); if (!Invocation.run()) return false; std::vector Args = Action.takeLastCC1Arguments(); Consumer.handleBuildCommand({std::move(Executable), std::move(Args)}); return true; } bool DependencyScanningWorker::computeDependencies( StringRef WorkingDirectory, const std::vector &CommandLine, DependencyConsumer &Consumer, DependencyActionController &Controller, DiagnosticConsumer &DC, std::optional ModuleName) { // Reset what might have been modified in the previous worker invocation. BaseFS->setCurrentWorkingDirectory(WorkingDirectory); std::optional> ModifiedCommandLine; llvm::IntrusiveRefCntPtr ModifiedFS; // If we're scanning based on a module name alone, we don't expect the client // to provide us with an input file. However, the driver really wants to have // one. Let's just make it up to make the driver happy. if (ModuleName) { auto OverlayFS = llvm::makeIntrusiveRefCnt(BaseFS); auto InMemoryFS = llvm::makeIntrusiveRefCnt(); InMemoryFS->setCurrentWorkingDirectory(WorkingDirectory); OverlayFS->pushOverlay(InMemoryFS); ModifiedFS = OverlayFS; SmallString<128> FakeInputPath; // TODO: We should retry the creation if the path already exists. llvm::sys::fs::createUniquePath(*ModuleName + "-%%%%%%%%.input", FakeInputPath, /*MakeAbsolute=*/false); InMemoryFS->addFile(FakeInputPath, 0, llvm::MemoryBuffer::getMemBuffer("")); ModifiedCommandLine = CommandLine; ModifiedCommandLine->emplace_back(FakeInputPath); } const std::vector &FinalCommandLine = ModifiedCommandLine ? *ModifiedCommandLine : CommandLine; auto &FinalFS = ModifiedFS ? ModifiedFS : BaseFS; auto FileMgr = llvm::makeIntrusiveRefCnt(FileSystemOptions{}, FinalFS); std::vector FinalCCommandLine(FinalCommandLine.size(), nullptr); llvm::transform(FinalCommandLine, FinalCCommandLine.begin(), [](const std::string &Str) { return Str.c_str(); }); auto DiagOpts = CreateAndPopulateDiagOpts(FinalCCommandLine); sanitizeDiagOpts(*DiagOpts); IntrusiveRefCntPtr Diags = CompilerInstance::createDiagnostics(DiagOpts.release(), &DC, /*ShouldOwnClient=*/false); // Although `Diagnostics` are used only for command-line parsing, the // custom `DiagConsumer` might expect a `SourceManager` to be present. SourceManager SrcMgr(*Diags, *FileMgr); Diags->setSourceManager(&SrcMgr); // DisableFree is modified by Tooling for running // in-process; preserve the original value, which is // always true for a driver invocation. bool DisableFree = true; DependencyScanningAction Action(WorkingDirectory, Consumer, Controller, DepFS, Format, OptimizeArgs, EagerLoadModules, DisableFree, ModuleName); bool Success = false; if (FinalCommandLine[1] == "-cc1") { Success = createAndRunToolInvocation(FinalCommandLine, Action, *FileMgr, PCHContainerOps, *Diags, Consumer); } else { Success = forEachDriverJob( FinalCommandLine, *Diags, *FileMgr, [&](const driver::Command &Cmd) { if (StringRef(Cmd.getCreator().getName()) != "clang") { // Non-clang command. Just pass through to the dependency // consumer. Consumer.handleBuildCommand( {Cmd.getExecutable(), {Cmd.getArguments().begin(), Cmd.getArguments().end()}}); return true; } // Insert -cc1 comand line options into Argv std::vector Argv; Argv.push_back(Cmd.getExecutable()); Argv.insert(Argv.end(), Cmd.getArguments().begin(), Cmd.getArguments().end()); // Create an invocation that uses the underlying file // system to ensure that any file system requests that // are made by the driver do not go through the // dependency scanning filesystem. return createAndRunToolInvocation(std::move(Argv), Action, *FileMgr, PCHContainerOps, *Diags, Consumer); }); } if (Success && !Action.hasScanned()) Diags->Report(diag::err_fe_expected_compiler_job) << llvm::join(FinalCommandLine, " "); return Success && Action.hasScanned(); } DependencyActionController::~DependencyActionController() {}