[clang] [clang] Move VFS overlays from `HeaderSearchOptions` to `FileSystemOptions` (PR #86534)

Jan Svoboda via cfe-commits cfe-commits at lists.llvm.org
Sat Nov 2 14:58:29 PDT 2024


https://github.com/jansvoboda11 updated https://github.com/llvm/llvm-project/pull/86534

>From d49ef2de83b37156ab73c0028676d9b0425ef2ed Mon Sep 17 00:00:00 2001
From: Jan Svoboda <jan_svoboda at apple.com>
Date: Mon, 25 Mar 2024 10:15:33 -0700
Subject: [PATCH] [clang] Move VFS overlays from `HeaderSearchOptions` to
 `FileSystemOptions`

---
 clang/include/clang/Basic/FileSystemOptions.h |   3 +
 clang/include/clang/Lex/HeaderSearchOptions.h |   7 -
 .../include/clang/Serialization/ASTBitCodes.h |   6 +-
 clang/lib/Frontend/ASTUnit.cpp                |  37 ++---
 clang/lib/Frontend/CompilerInstance.cpp       |   4 +-
 clang/lib/Frontend/CompilerInvocation.cpp     |  18 +--
 clang/lib/Frontend/FrontendActions.cpp        |  12 +-
 clang/lib/Lex/HeaderSearch.cpp                |   3 +-
 clang/lib/Serialization/ASTReader.cpp         |  26 ++--
 clang/lib/Serialization/ASTWriter.cpp         |  30 ++---
 .../DependencyScanningWorker.cpp              |  56 ++++----
 .../DependencyScanning/ModuleDepCollector.cpp | 127 +++++++++---------
 12 files changed, 167 insertions(+), 162 deletions(-)

diff --git a/clang/include/clang/Basic/FileSystemOptions.h b/clang/include/clang/Basic/FileSystemOptions.h
index 458af0c7b6592c..d6c895007669cb 100644
--- a/clang/include/clang/Basic/FileSystemOptions.h
+++ b/clang/include/clang/Basic/FileSystemOptions.h
@@ -24,6 +24,9 @@ class FileSystemOptions {
   /// If set, paths are resolved as if the working directory was
   /// set to the value of WorkingDir.
   std::string WorkingDir;
+
+  /// The set of user-provided virtual filesystem overlay files.
+  std::vector<std::string> VFSOverlayFiles;
 };
 
 } // end namespace clang
diff --git a/clang/include/clang/Lex/HeaderSearchOptions.h b/clang/include/clang/Lex/HeaderSearchOptions.h
index 83a95e9ad90a7f..a1e8ebe9dbb58f 100644
--- a/clang/include/clang/Lex/HeaderSearchOptions.h
+++ b/clang/include/clang/Lex/HeaderSearchOptions.h
@@ -187,9 +187,6 @@ class HeaderSearchOptions {
   /// of computing the module hash.
   llvm::SmallSetVector<llvm::CachedHashString, 16> ModulesIgnoreMacros;
 
-  /// The set of user-provided virtual filesystem overlay files.
-  std::vector<std::string> VFSOverlayFiles;
-
   /// Include the compiler builtin includes.
   LLVM_PREFERRED_TYPE(bool)
   unsigned UseBuiltinIncludes : 1;
@@ -307,10 +304,6 @@ class HeaderSearchOptions {
     SystemHeaderPrefixes.emplace_back(Prefix, IsSystemHeader);
   }
 
-  void AddVFSOverlayFile(StringRef Name) {
-    VFSOverlayFiles.push_back(std::string(Name));
-  }
-
   void AddPrebuiltModulePath(StringRef Name) {
     PrebuiltModulePaths.push_back(std::string(Name));
   }
diff --git a/clang/include/clang/Serialization/ASTBitCodes.h b/clang/include/clang/Serialization/ASTBitCodes.h
index b6193866fc7134..06a1507fbfd4d5 100644
--- a/clang/include/clang/Serialization/ASTBitCodes.h
+++ b/clang/include/clang/Serialization/ASTBitCodes.h
@@ -392,9 +392,6 @@ enum OptionsRecordTypes {
   /// Record code for the target options table.
   TARGET_OPTIONS,
 
-  /// Record code for the filesystem options table.
-  FILE_SYSTEM_OPTIONS,
-
   /// Record code for the headers search options table.
   HEADER_SEARCH_OPTIONS,
 
@@ -413,6 +410,9 @@ enum UnhashedControlBlockRecordTypes {
   /// Record code for the diagnostic options table.
   DIAGNOSTIC_OPTIONS,
 
+  /// Record code for the filesystem options table.
+  FILE_SYSTEM_OPTIONS,
+
   /// Record code for the headers search paths.
   HEADER_SEARCH_PATHS,
 
diff --git a/clang/lib/Frontend/ASTUnit.cpp b/clang/lib/Frontend/ASTUnit.cpp
index 4aec928f9eb0a5..26dfca106f2e39 100644
--- a/clang/lib/Frontend/ASTUnit.cpp
+++ b/clang/lib/Frontend/ASTUnit.cpp
@@ -517,6 +517,7 @@ namespace {
 class ASTInfoCollector : public ASTReaderListener {
   Preprocessor &PP;
   ASTContext *Context;
+  FileSystemOptions &FSOpts;
   HeaderSearchOptions &HSOpts;
   PreprocessorOptions &PPOpts;
   LangOptions &LangOpt;
@@ -524,17 +525,17 @@ class ASTInfoCollector : public ASTReaderListener {
   IntrusiveRefCntPtr<TargetInfo> &Target;
   unsigned &Counter;
   bool InitializedLanguage = false;
-  bool InitializedHeaderSearchPaths = false;
+  bool InitializedFileSystem = false;
 
 public:
   ASTInfoCollector(Preprocessor &PP, ASTContext *Context,
-                   HeaderSearchOptions &HSOpts, PreprocessorOptions &PPOpts,
-                   LangOptions &LangOpt,
+                   FileSystemOptions &FSOpts, HeaderSearchOptions &HSOpts,
+                   PreprocessorOptions &PPOpts, LangOptions &LangOpt,
                    std::shared_ptr<TargetOptions> &TargetOpts,
                    IntrusiveRefCntPtr<TargetInfo> &Target, unsigned &Counter)
-      : PP(PP), Context(Context), HSOpts(HSOpts), PPOpts(PPOpts),
-        LangOpt(LangOpt), TargetOpts(TargetOpts), Target(Target),
-        Counter(Counter) {}
+      : PP(PP), Context(Context), FSOpts(FSOpts), HSOpts(HSOpts),
+        PPOpts(PPOpts), LangOpt(LangOpt), TargetOpts(TargetOpts),
+        Target(Target), Counter(Counter) {}
 
   bool ReadLanguageOptions(const LangOptions &LangOpts,
                            StringRef ModuleFilename, bool Complain,
@@ -568,7 +569,6 @@ class ASTInfoCollector : public ASTReaderListener {
         this->HSOpts.ForceCheckCXX20ModulesInputFiles;
     llvm::SaveAndRestore X(this->HSOpts.UserEntries);
     llvm::SaveAndRestore Y(this->HSOpts.SystemHeaderPrefixes);
-    llvm::SaveAndRestore Z(this->HSOpts.VFSOverlayFiles);
 
     this->HSOpts = HSOpts;
     this->HSOpts.ForceCheckCXX20ModulesInputFiles =
@@ -577,24 +577,29 @@ class ASTInfoCollector : public ASTReaderListener {
     return false;
   }
 
-  bool ReadHeaderSearchPaths(const HeaderSearchOptions &HSOpts,
+  bool ReadFileSystemOptions(const FileSystemOptions &FSOpts,
                              bool Complain) override {
-    if (InitializedHeaderSearchPaths)
+    if (InitializedFileSystem)
       return false;
 
-    this->HSOpts.UserEntries = HSOpts.UserEntries;
-    this->HSOpts.SystemHeaderPrefixes = HSOpts.SystemHeaderPrefixes;
-    this->HSOpts.VFSOverlayFiles = HSOpts.VFSOverlayFiles;
+    this->FSOpts.VFSOverlayFiles = FSOpts.VFSOverlayFiles;
 
     // Initialize the FileManager. We can't do this in update(), since that
     // performs the initialization too late (once both target and language
     // options are read).
     PP.getFileManager().setVirtualFileSystem(createVFSFromOverlayFiles(
-        HSOpts.VFSOverlayFiles, PP.getDiagnostics(),
+        FSOpts.VFSOverlayFiles, PP.getDiagnostics(),
         PP.getFileManager().getVirtualFileSystemPtr()));
 
-    InitializedHeaderSearchPaths = true;
+    InitializedFileSystem = true;
+
+    return false;
+  }
 
+  bool ReadHeaderSearchPaths(const HeaderSearchOptions &HSOpts,
+                             bool Complain) override {
+    this->HSOpts.UserEntries = HSOpts.UserEntries;
+    this->HSOpts.SystemHeaderPrefixes = HSOpts.SystemHeaderPrefixes;
     return false;
   }
 
@@ -867,8 +872,8 @@ std::unique_ptr<ASTUnit> ASTUnit::LoadFromASTFile(
 
   unsigned Counter = 0;
   AST->Reader->setListener(std::make_unique<ASTInfoCollector>(
-      *AST->PP, AST->Ctx.get(), *AST->HSOpts, *AST->PPOpts, *AST->LangOpts,
-      AST->TargetOpts, AST->Target, Counter));
+      *AST->PP, AST->Ctx.get(), AST->FileSystemOpts, *AST->HSOpts, *AST->PPOpts,
+      *AST->LangOpts, AST->TargetOpts, AST->Target, Counter));
 
   // Attach the AST reader to the AST context as an external AST
   // source, so that declarations will be deserialized from the
diff --git a/clang/lib/Frontend/CompilerInstance.cpp b/clang/lib/Frontend/CompilerInstance.cpp
index 240305b33824b8..63fcb9c51d85dd 100644
--- a/clang/lib/Frontend/CompilerInstance.cpp
+++ b/clang/lib/Frontend/CompilerInstance.cpp
@@ -261,12 +261,12 @@ static void collectIncludePCH(CompilerInstance &CI,
 
 static void collectVFSEntries(CompilerInstance &CI,
                               std::shared_ptr<ModuleDependencyCollector> MDC) {
-  if (CI.getHeaderSearchOpts().VFSOverlayFiles.empty())
+  if (CI.getFileSystemOpts().VFSOverlayFiles.empty())
     return;
 
   // Collect all VFS found.
   SmallVector<llvm::vfs::YAMLVFSEntry, 16> VFSEntries;
-  for (const std::string &VFSFile : CI.getHeaderSearchOpts().VFSOverlayFiles) {
+  for (const std::string &VFSFile : CI.getFileSystemOpts().VFSOverlayFiles) {
     llvm::ErrorOr<std::unique_ptr<llvm::MemoryBuffer>> Buffer =
         llvm::MemoryBuffer::getFile(VFSFile);
     if (!Buffer)
diff --git a/clang/lib/Frontend/CompilerInvocation.cpp b/clang/lib/Frontend/CompilerInvocation.cpp
index d8261e12b08b5c..eb1f2e9f9f05af 100644
--- a/clang/lib/Frontend/CompilerInvocation.cpp
+++ b/clang/lib/Frontend/CompilerInvocation.cpp
@@ -2421,6 +2421,9 @@ static void GenerateFileSystemArgs(const FileSystemOptions &Opts,
   GENERATE_OPTION_WITH_MARSHALLING(Consumer, __VA_ARGS__)
 #include "clang/Driver/Options.inc"
 #undef FILE_SYSTEM_OPTION_WITH_MARSHALLING
+
+  for (const std::string &F : Opts.VFSOverlayFiles)
+    GenerateArg(Consumer, OPT_ivfsoverlay, F);
 }
 
 static bool ParseFileSystemArgs(FileSystemOptions &Opts, const ArgList &Args,
@@ -2434,6 +2437,9 @@ static bool ParseFileSystemArgs(FileSystemOptions &Opts, const ArgList &Args,
 #include "clang/Driver/Options.inc"
 #undef FILE_SYSTEM_OPTION_WITH_MARSHALLING
 
+  for (const auto *A : Args.filtered(OPT_ivfsoverlay, OPT_vfsoverlay))
+    Opts.VFSOverlayFiles.push_back(std::string(A->getValue()));
+
   return Diags.getNumErrors() == NumErrorsBefore;
 }
 
@@ -3270,9 +3276,6 @@ static void GenerateHeaderSearchArgs(const HeaderSearchOptions &Opts,
                                         : OPT_no_system_header_prefix;
     GenerateArg(Consumer, Opt, P.Prefix);
   }
-
-  for (const std::string &F : Opts.VFSOverlayFiles)
-    GenerateArg(Consumer, OPT_ivfsoverlay, F);
 }
 
 static bool ParseHeaderSearchArgs(HeaderSearchOptions &Opts, ArgList &Args,
@@ -3411,9 +3414,6 @@ static bool ParseHeaderSearchArgs(HeaderSearchOptions &Opts, ArgList &Args,
     Opts.AddSystemHeaderPrefix(
         A->getValue(), A->getOption().matches(OPT_system_header_prefix));
 
-  for (const auto *A : Args.filtered(OPT_ivfsoverlay, OPT_vfsoverlay))
-    Opts.AddVFSOverlayFile(A->getValue());
-
   return Diags.getNumErrors() == NumErrorsBefore;
 }
 
@@ -4959,7 +4959,7 @@ bool CompilerInvocation::CreateFromArgsImpl(
   // to determine the PGO type.
   if (!Res.getCodeGenOpts().ProfileInstrumentUsePath.empty()) {
     auto FS =
-        createVFSFromOverlayFiles(Res.getHeaderSearchOpts().VFSOverlayFiles,
+        createVFSFromOverlayFiles(Res.getFileSystemOpts().VFSOverlayFiles,
                                   Diags, llvm::vfs::getRealFileSystem());
     setPGOUseInstrumentor(Res.getCodeGenOpts(),
                           Res.getCodeGenOpts().ProfileInstrumentUsePath, *FS,
@@ -5051,7 +5051,7 @@ std::string CompilerInvocation::getModuleHash() const {
   if (hsOpts.ModulesStrictContextHash) {
     HBuilder.addRange(hsOpts.SystemHeaderPrefixes);
     HBuilder.addRange(hsOpts.UserEntries);
-    HBuilder.addRange(hsOpts.VFSOverlayFiles);
+    HBuilder.addRange(getFileSystemOpts().VFSOverlayFiles);
 
     const DiagnosticOptions &diagOpts = getDiagnosticOpts();
 #define DIAGOPT(Name, Bits, Default) HBuilder.add(diagOpts.Name);
@@ -5171,7 +5171,7 @@ IntrusiveRefCntPtr<llvm::vfs::FileSystem>
 clang::createVFSFromCompilerInvocation(
     const CompilerInvocation &CI, DiagnosticsEngine &Diags,
     IntrusiveRefCntPtr<llvm::vfs::FileSystem> BaseFS) {
-  return createVFSFromOverlayFiles(CI.getHeaderSearchOpts().VFSOverlayFiles,
+  return createVFSFromOverlayFiles(CI.getFileSystemOpts().VFSOverlayFiles,
                                    Diags, std::move(BaseFS));
 }
 
diff --git a/clang/lib/Frontend/FrontendActions.cpp b/clang/lib/Frontend/FrontendActions.cpp
index e943f143d4c158..d8a1d21648e057 100644
--- a/clang/lib/Frontend/FrontendActions.cpp
+++ b/clang/lib/Frontend/FrontendActions.cpp
@@ -706,6 +706,15 @@ namespace {
       return false;
     }
 
+    bool ReadFileSystemOptions(const FileSystemOptions &FSOpts,
+                               bool Complain) override {
+      Out.indent(2) << "File system options:\n";
+      Out.indent(4) << "VFS overlay files:\n";
+      for (const auto &Overlay : FSOpts.VFSOverlayFiles)
+        Out.indent(6) << Overlay << "\n";
+      return false;
+    }
+
     bool ReadHeaderSearchPaths(const HeaderSearchOptions &HSOpts,
                                bool Complain) override {
       Out.indent(2) << "Header search paths:\n";
@@ -715,9 +724,6 @@ namespace {
       Out.indent(4) << "System header prefixes:\n";
       for (const auto &Prefix : HSOpts.SystemHeaderPrefixes)
         Out.indent(6) << Prefix.Prefix << "\n";
-      Out.indent(4) << "VFS overlay files:\n";
-      for (const auto &Overlay : HSOpts.VFSOverlayFiles)
-        Out.indent(6) << Overlay << "\n";
       return false;
     }
 
diff --git a/clang/lib/Lex/HeaderSearch.cpp b/clang/lib/Lex/HeaderSearch.cpp
index 052be1395161d4..a3c6877d538a09 100644
--- a/clang/lib/Lex/HeaderSearch.cpp
+++ b/clang/lib/Lex/HeaderSearch.cpp
@@ -157,7 +157,8 @@ std::vector<bool> HeaderSearch::collectVFSUsageAndClear() const {
       RFS->clearHasBeenUsed();
     }
   });
-  assert(VFSUsage.size() == getHeaderSearchOpts().VFSOverlayFiles.size() &&
+  assert(VFSUsage.size() ==
+             FileMgr.getFileSystemOpts().VFSOverlayFiles.size() &&
          "A different number of RedirectingFileSystem's were present than "
          "-ivfsoverlay options passed to Clang!");
   // VFS visit order is the opposite of VFSOverlayFiles order.
diff --git a/clang/lib/Serialization/ASTReader.cpp b/clang/lib/Serialization/ASTReader.cpp
index 8d8f9378cfeabe..c6a125547d145c 100644
--- a/clang/lib/Serialization/ASTReader.cpp
+++ b/clang/lib/Serialization/ASTReader.cpp
@@ -2844,14 +2844,6 @@ ASTReader::ASTReadResult ASTReader::ReadOptionsBlock(
       break;
     }
 
-    case FILE_SYSTEM_OPTIONS: {
-      bool Complain = (ClientLoadCapabilities & ARR_ConfigurationMismatch) == 0;
-      if (!AllowCompatibleConfigurationMismatch &&
-          ParseFileSystemOptions(Record, Complain, Listener))
-        Result = ConfigurationMismatch;
-      break;
-    }
-
     case HEADER_SEARCH_OPTIONS: {
       bool Complain = (ClientLoadCapabilities & ARR_ConfigurationMismatch) == 0;
       if (!AllowCompatibleConfigurationMismatch &&
@@ -5015,6 +5007,13 @@ ASTReader::ASTReadResult ASTReader::readUnhashedControlBlockImpl(
         Result = OutOfDate; // Don't return early.  Read the signature.
       break;
     }
+    case FILE_SYSTEM_OPTIONS: {
+      bool Complain = (ClientLoadCapabilities & ARR_ConfigurationMismatch) == 0;
+      if (Listener && !AllowCompatibleConfigurationMismatch &&
+          ParseFileSystemOptions(Record, Complain, *Listener))
+        Result = ConfigurationMismatch;
+      break;
+    }
     case HEADER_SEARCH_PATHS: {
       bool Complain = (ClientLoadCapabilities & ARR_ConfigurationMismatch) == 0;
       if (Listener && !AllowCompatibleConfigurationMismatch &&
@@ -6155,7 +6154,12 @@ bool ASTReader::ParseFileSystemOptions(const RecordData &Record, bool Complain,
                                        ASTReaderListener &Listener) {
   FileSystemOptions FSOpts;
   unsigned Idx = 0;
+
   FSOpts.WorkingDir = ReadString(Record, Idx);
+
+  for (unsigned N = Record[Idx++]; N; --N)
+    FSOpts.VFSOverlayFiles.emplace_back(ReadString(Record, Idx));
+
   return Listener.ReadFileSystemOptions(FSOpts, Complain);
 }
 
@@ -6207,12 +6211,6 @@ bool ASTReader::ParseHeaderSearchPaths(const RecordData &Record, bool Complain,
     HSOpts.SystemHeaderPrefixes.emplace_back(std::move(Prefix), IsSystemHeader);
   }
 
-  // VFS overlay files.
-  for (unsigned N = Record[Idx++]; N; --N) {
-    std::string VFSOverlayFile = ReadString(Record, Idx);
-    HSOpts.VFSOverlayFiles.emplace_back(std::move(VFSOverlayFile));
-  }
-
   return Listener.ReadHeaderSearchPaths(HSOpts, Complain);
 }
 
diff --git a/clang/lib/Serialization/ASTWriter.cpp b/clang/lib/Serialization/ASTWriter.cpp
index 569c688f793d81..5fb98e9fb9425c 100644
--- a/clang/lib/Serialization/ASTWriter.cpp
+++ b/clang/lib/Serialization/ASTWriter.cpp
@@ -868,7 +868,6 @@ void ASTWriter::WriteBlockInfoBlock() {
   BLOCK(OPTIONS_BLOCK);
   RECORD(LANGUAGE_OPTIONS);
   RECORD(TARGET_OPTIONS);
-  RECORD(FILE_SYSTEM_OPTIONS);
   RECORD(HEADER_SEARCH_OPTIONS);
   RECORD(PREPROCESSOR_OPTIONS);
 
@@ -1113,6 +1112,7 @@ void ASTWriter::WriteBlockInfoBlock() {
   RECORD(SIGNATURE);
   RECORD(AST_BLOCK_HASH);
   RECORD(DIAGNOSTIC_OPTIONS);
+  RECORD(FILE_SYSTEM_OPTIONS);
   RECORD(HEADER_SEARCH_PATHS);
   RECORD(DIAG_PRAGMA_MAPPINGS);
 
@@ -1342,6 +1342,19 @@ void ASTWriter::writeUnhashedControlBlock(Preprocessor &PP,
     Record.clear();
   }
 
+  // File system options.
+  const FileSystemOptions &FSOpts =
+      Context.getSourceManager().getFileManager().getFileSystemOpts();
+
+  AddString(FSOpts.WorkingDir, Record);
+
+  Record.push_back(FSOpts.VFSOverlayFiles.size());
+  for (StringRef VFSOverlayFile : FSOpts.VFSOverlayFiles)
+    AddString(VFSOverlayFile, Record);
+
+  Stream.EmitRecord(FILE_SYSTEM_OPTIONS, Record);
+  Record.clear();
+
   // Header search paths.
   if (!HSOpts.ModulesSkipHeaderSearchPaths) {
     // Include entries.
@@ -1361,11 +1374,6 @@ void ASTWriter::writeUnhashedControlBlock(Preprocessor &PP,
       Record.push_back(HSOpts.SystemHeaderPrefixes[I].IsSystemHeader);
     }
 
-    // VFS overlay files.
-    Record.push_back(HSOpts.VFSOverlayFiles.size());
-    for (StringRef VFSOverlayFile : HSOpts.VFSOverlayFiles)
-      AddString(VFSOverlayFile, Record);
-
     Stream.EmitRecord(HEADER_SEARCH_PATHS, Record);
   }
 
@@ -1607,13 +1615,6 @@ void ASTWriter::WriteControlBlock(Preprocessor &PP, ASTContext &Context,
   }
   Stream.EmitRecord(TARGET_OPTIONS, Record);
 
-  // File system options.
-  Record.clear();
-  const FileSystemOptions &FSOpts =
-      Context.getSourceManager().getFileManager().getFileSystemOpts();
-  AddString(FSOpts.WorkingDir, Record);
-  Stream.EmitRecord(FILE_SYSTEM_OPTIONS, Record);
-
   // Header search options.
   Record.clear();
   const HeaderSearchOptions &HSOpts =
@@ -5019,8 +5020,7 @@ void ASTWriter::computeNonAffectingInputFiles() {
   FileManager &FileMgr = PP->getFileManager();
   FileMgr.trackVFSUsage(true);
   // Lookup the paths in the VFS to trigger `-ivfsoverlay` usage tracking.
-  for (StringRef Path :
-       PP->getHeaderSearchInfo().getHeaderSearchOpts().VFSOverlayFiles)
+  for (StringRef Path : FileMgr.getFileSystemOpts().VFSOverlayFiles)
     FileMgr.getVirtualFileSystem().exists(Path);
   for (unsigned I = 1; I != N; ++I) {
     if (IsSLocAffecting[I]) {
diff --git a/clang/lib/Tooling/DependencyScanning/DependencyScanningWorker.cpp b/clang/lib/Tooling/DependencyScanning/DependencyScanningWorker.cpp
index d77187bfb1f2b8..ee3805c3f0d46c 100644
--- a/clang/lib/Tooling/DependencyScanning/DependencyScanningWorker.cpp
+++ b/clang/lib/Tooling/DependencyScanning/DependencyScanningWorker.cpp
@@ -61,25 +61,22 @@ class DependencyConsumerForwarder : public DependencyFileGenerator {
   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<std::string> 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);
-      }
+static bool checkFileSystemOpts(const FileSystemOptions &FSOpts,
+                                const FileSystemOptions &ExistingFSOpts,
+                                DiagnosticsEngine *Diags) {
+  if (FSOpts.VFSOverlayFiles != ExistingFSOpts.VFSOverlayFiles) {
+    if (Diags) {
+      Diags->Report(diag::warn_pch_vfsoverlay_mismatch);
+      auto VFSNote = [&](int Type, ArrayRef<std::string> 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, FSOpts.VFSOverlayFiles);
+      VFSNote(1, ExistingFSOpts.VFSOverlayFiles);
     }
   }
   return false;
@@ -94,11 +91,11 @@ class PrebuiltModuleListener : public ASTReaderListener {
   PrebuiltModuleListener(PrebuiltModuleFilesT &PrebuiltModuleFiles,
                          llvm::SmallVector<std::string> &NewModuleFiles,
                          PrebuiltModuleVFSMapT &PrebuiltModuleVFSMap,
-                         const HeaderSearchOptions &HSOpts,
+                         const FileSystemOptions &FSOpts,
                          const LangOptions &LangOpts, DiagnosticsEngine &Diags)
       : PrebuiltModuleFiles(PrebuiltModuleFiles),
         NewModuleFiles(NewModuleFiles),
-        PrebuiltModuleVFSMap(PrebuiltModuleVFSMap), ExistingHSOpts(HSOpts),
+        PrebuiltModuleVFSMap(PrebuiltModuleVFSMap), ExistingFSOpts(FSOpts),
         ExistingLangOpts(LangOpts), Diags(Diags) {}
 
   bool needsImportVisitation() const override { return true; }
@@ -113,20 +110,23 @@ class PrebuiltModuleListener : public ASTReaderListener {
     CurrentFile = Filename;
   }
 
-  bool ReadHeaderSearchPaths(const HeaderSearchOptions &HSOpts,
+  bool ReadFileSystemOptions(const FileSystemOptions &FSOpts,
                              bool Complain) override {
-    std::vector<std::string> VFSOverlayFiles = HSOpts.VFSOverlayFiles;
     PrebuiltModuleVFSMap.insert(
-        {CurrentFile, llvm::StringSet<>(VFSOverlayFiles)});
-    return checkHeaderSearchPaths(
-        HSOpts, ExistingHSOpts, Complain ? &Diags : nullptr, ExistingLangOpts);
+        {CurrentFile, llvm::StringSet<>(FSOpts.VFSOverlayFiles)});
+
+    if (!ExistingLangOpts.Modules)
+      return false;
+
+    return checkFileSystemOpts(FSOpts, ExistingFSOpts,
+                               Complain ? &Diags : nullptr);
   }
 
 private:
   PrebuiltModuleFilesT &PrebuiltModuleFiles;
   llvm::SmallVector<std::string> &NewModuleFiles;
   PrebuiltModuleVFSMapT &PrebuiltModuleVFSMap;
-  const HeaderSearchOptions &ExistingHSOpts;
+  const FileSystemOptions &ExistingFSOpts;
   const LangOptions &ExistingLangOpts;
   DiagnosticsEngine &Diags;
   std::string CurrentFile;
@@ -142,7 +142,7 @@ static bool visitPrebuiltModule(StringRef PrebuiltModuleFilename,
   // List of module files to be processed.
   llvm::SmallVector<std::string> Worklist;
   PrebuiltModuleListener Listener(ModuleFiles, Worklist, PrebuiltModuleVFSMap,
-                                  CI.getHeaderSearchOpts(), CI.getLangOpts(),
+                                  CI.getFileSystemOpts(), CI.getLangOpts(),
                                   Diags);
 
   Listener.visitModuleFile(PrebuiltModuleFilename,
diff --git a/clang/lib/Tooling/DependencyScanning/ModuleDepCollector.cpp b/clang/lib/Tooling/DependencyScanning/ModuleDepCollector.cpp
index 637416cd1fc621..0a99ae9237536a 100644
--- a/clang/lib/Tooling/DependencyScanning/ModuleDepCollector.cpp
+++ b/clang/lib/Tooling/DependencyScanning/ModuleDepCollector.cpp
@@ -29,72 +29,71 @@ const std::vector<std::string> &ModuleDeps::getBuildArguments() {
   return std::get<std::vector<std::string>>(BuildInfo);
 }
 
+static void optimizeHeaderSearchOpts(HeaderSearchOptions &Opts,
+                                     const serialization::ModuleFile &MF) {
+  // Only preserve search paths that were used during the dependency scan.
+  std::vector<HeaderSearchOptions::Entry> Entries;
+  std::swap(Opts.UserEntries, Entries);
+
+  llvm::BitVector SearchPathUsage(Entries.size());
+  llvm::DenseSet<const serialization::ModuleFile *> Visited;
+  std::function<void(const serialization::ModuleFile *)> VisitMF =
+      [&](const serialization::ModuleFile *MF) {
+        SearchPathUsage |= MF->SearchPathUsage;
+        Visited.insert(MF);
+        for (const serialization::ModuleFile *Import : MF->Imports)
+          if (!Visited.contains(Import))
+            VisitMF(Import);
+      };
+  VisitMF(&MF);
+
+  if (SearchPathUsage.size() != Entries.size())
+    llvm::report_fatal_error(
+        "Inconsistent search path options between modules detected");
+
+  for (auto Idx : SearchPathUsage.set_bits())
+    Opts.UserEntries.push_back(std::move(Entries[Idx]));
+}
+
 static void
-optimizeHeaderSearchOpts(HeaderSearchOptions &Opts, ASTReader &Reader,
-                         const serialization::ModuleFile &MF,
-                         const PrebuiltModuleVFSMapT &PrebuiltModuleVFSMap,
-                         ScanningOptimizations OptimizeArgs) {
-  if (any(OptimizeArgs & ScanningOptimizations::HeaderSearch)) {
-    // Only preserve search paths that were used during the dependency scan.
-    std::vector<HeaderSearchOptions::Entry> Entries;
-    std::swap(Opts.UserEntries, Entries);
-
-    llvm::BitVector SearchPathUsage(Entries.size());
-    llvm::DenseSet<const serialization::ModuleFile *> Visited;
-    std::function<void(const serialization::ModuleFile *)> VisitMF =
-        [&](const serialization::ModuleFile *MF) {
-          SearchPathUsage |= MF->SearchPathUsage;
-          Visited.insert(MF);
+optimizeFileSystemOpts(FileSystemOptions &Opts,
+                       const serialization::ModuleFile &MF,
+                       const PrebuiltModuleVFSMapT &PrebuiltModuleVFSMap) {
+  std::vector<std::string> VFSOverlayFiles;
+  std::swap(Opts.VFSOverlayFiles, VFSOverlayFiles);
+
+  llvm::BitVector VFSUsage(VFSOverlayFiles.size());
+  llvm::DenseSet<const serialization::ModuleFile *> Visited;
+  std::function<void(const serialization::ModuleFile *)> VisitMF =
+      [&](const serialization::ModuleFile *MF) {
+        Visited.insert(MF);
+        if (MF->Kind == serialization::MK_ImplicitModule) {
+          VFSUsage |= MF->VFSUsage;
+          // We only need to recurse into implicit modules. Other module types
+          // will have the correct set of VFSs for anything they depend on.
           for (const serialization::ModuleFile *Import : MF->Imports)
             if (!Visited.contains(Import))
               VisitMF(Import);
-        };
-    VisitMF(&MF);
-
-    if (SearchPathUsage.size() != Entries.size())
-      llvm::report_fatal_error(
-          "Inconsistent search path options between modules detected");
-
-    for (auto Idx : SearchPathUsage.set_bits())
-      Opts.UserEntries.push_back(std::move(Entries[Idx]));
-  }
-  if (any(OptimizeArgs & ScanningOptimizations::VFS)) {
-    std::vector<std::string> VFSOverlayFiles;
-    std::swap(Opts.VFSOverlayFiles, VFSOverlayFiles);
-
-    llvm::BitVector VFSUsage(VFSOverlayFiles.size());
-    llvm::DenseSet<const serialization::ModuleFile *> Visited;
-    std::function<void(const serialization::ModuleFile *)> VisitMF =
-        [&](const serialization::ModuleFile *MF) {
-          Visited.insert(MF);
-          if (MF->Kind == serialization::MK_ImplicitModule) {
-            VFSUsage |= MF->VFSUsage;
-            // We only need to recurse into implicit modules. Other module types
-            // will have the correct set of VFSs for anything they depend on.
-            for (const serialization::ModuleFile *Import : MF->Imports)
-              if (!Visited.contains(Import))
-                VisitMF(Import);
-          } else {
-            // This is not an implicitly built module, so it may have different
-            // VFS options. Fall back to a string comparison instead.
-            auto VFSMap = PrebuiltModuleVFSMap.find(MF->FileName);
-            if (VFSMap == PrebuiltModuleVFSMap.end())
-              return;
-            for (std::size_t I = 0, E = VFSOverlayFiles.size(); I != E; ++I) {
-              if (VFSMap->second.contains(VFSOverlayFiles[I]))
-                VFSUsage[I] = true;
-            }
+        } else {
+          // This is not an implicitly built module, so it may have different
+          // VFS options. Fall back to a string comparison instead.
+          auto VFSMap = PrebuiltModuleVFSMap.find(MF->FileName);
+          if (VFSMap == PrebuiltModuleVFSMap.end())
+            return;
+          for (std::size_t I = 0, E = VFSOverlayFiles.size(); I != E; ++I) {
+            if (VFSMap->second.contains(VFSOverlayFiles[I]))
+              VFSUsage[I] = true;
           }
-        };
-    VisitMF(&MF);
+        }
+      };
+  VisitMF(&MF);
 
-    if (VFSUsage.size() != VFSOverlayFiles.size())
-      llvm::report_fatal_error(
-          "Inconsistent -ivfsoverlay options between modules detected");
+  if (VFSUsage.size() != VFSOverlayFiles.size())
+    llvm::report_fatal_error(
+        "Inconsistent -ivfsoverlay options between modules detected");
 
-    for (auto Idx : VFSUsage.set_bits())
-      Opts.VFSOverlayFiles.push_back(std::move(VFSOverlayFiles[Idx]));
-  }
+  for (auto Idx : VFSUsage.set_bits())
+    Opts.VFSOverlayFiles.push_back(std::move(VFSOverlayFiles[Idx]));
 }
 
 static void optimizeDiagnosticOpts(DiagnosticOptions &Opts,
@@ -628,12 +627,12 @@ ModuleDepCollectorPP::handleTopLevelModule(const Module *M) {
   CowCompilerInvocation CI =
       MDC.getInvocationAdjustedForModuleBuildWithoutOutputs(
           MD, [&](CowCompilerInvocation &BuildInvocation) {
-            if (any(MDC.OptimizeArgs & (ScanningOptimizations::HeaderSearch |
-                                        ScanningOptimizations::VFS)))
+            if (any(MDC.OptimizeArgs & ScanningOptimizations::HeaderSearch))
               optimizeHeaderSearchOpts(BuildInvocation.getMutHeaderSearchOpts(),
-                                       *MDC.ScanInstance.getASTReader(), *MF,
-                                       MDC.PrebuiltModuleVFSMap,
-                                       MDC.OptimizeArgs);
+                                       *MF);
+            if (any(MDC.OptimizeArgs & ScanningOptimizations::VFS))
+              optimizeFileSystemOpts(BuildInvocation.getMutFileSystemOpts(),
+                                     *MF, MDC.PrebuiltModuleVFSMap);
             if (any(MDC.OptimizeArgs & ScanningOptimizations::SystemWarnings))
               optimizeDiagnosticOpts(
                   BuildInvocation.getMutDiagnosticOpts(),



More information about the cfe-commits mailing list