From 6712f9799f6865aeb40163805cca5f9349cabc45 Mon Sep 17 00:00:00 2001 From: Jan Svoboda Date: Wed, 16 Aug 2023 17:26:37 -0700 Subject: [PATCH] [clang][deps] Load module map file from PCM --- clang/lib/Frontend/CompilerInstance.cpp | 5 +- clang/lib/Lex/ModuleMap.cpp | 6 +- clang/lib/Serialization/ASTWriter.cpp | 14 +++-- .../DependencyScanning/ModuleDepCollector.cpp | 18 +++--- .../modules-inferred-vfs-mod-map.m | 63 +++++++++++++++++++ 5 files changed, 86 insertions(+), 20 deletions(-) create mode 100644 clang/test/ClangScanDeps/modules-inferred-vfs-mod-map.m diff --git a/clang/lib/Frontend/CompilerInstance.cpp b/clang/lib/Frontend/CompilerInstance.cpp index 49c1d8e553a1d..367e32e132cec 100644 --- a/clang/lib/Frontend/CompilerInstance.cpp +++ b/clang/lib/Frontend/CompilerInstance.cpp @@ -1338,7 +1338,8 @@ static bool compileModule(CompilerInstance &ImportingInstance, Result = compileModuleImpl( ImportingInstance, ImportLoc, Module->getTopLevelModuleName(), FrontendInputFile(ModuleMapFilePath, IK, +Module->IsSystem), - ModMap.getModuleMapFileForUniquing(Module)->getName(), ModuleFileName); + ModMap.getModuleMapFileForUniquing(Module)->getNameAsRequested(), + ModuleFileName); } else { // FIXME: We only need to fake up an input file here as a way of // transporting the module's directory to the module map parser. We should @@ -1355,7 +1356,7 @@ static bool compileModule(CompilerInstance &ImportingInstance, Result = compileModuleImpl( ImportingInstance, ImportLoc, Module->getTopLevelModuleName(), FrontendInputFile(FakeModuleMapFile, IK, +Module->IsSystem), - ModMap.getModuleMapFileForUniquing(Module)->getName(), + ModMap.getModuleMapFileForUniquing(Module)->getNameAsRequested(), ModuleFileName, [&](CompilerInstance &Instance) { std::unique_ptr ModuleMapBuffer = diff --git a/clang/lib/Lex/ModuleMap.cpp b/clang/lib/Lex/ModuleMap.cpp index bee3a484f84bd..acbac82d28df9 100644 --- a/clang/lib/Lex/ModuleMap.cpp +++ b/clang/lib/Lex/ModuleMap.cpp @@ -1307,6 +1307,9 @@ void ModuleMap::setInferredModuleAllowedBy(Module *M, std::error_code ModuleMap::canonicalizeModuleMapPath(SmallVectorImpl &Path) { + FileManager &FM = SourceMgr.getFileManager(); + FM.makeAbsolutePath(Path); + StringRef Dir = llvm::sys::path::parent_path({Path.data(), Path.size()}); // Do not canonicalize within the framework; the module map parser expects @@ -1317,8 +1320,7 @@ ModuleMap::canonicalizeModuleMapPath(SmallVectorImpl &Path) { Dir = Parent; } - FileManager &FM = SourceMgr.getFileManager(); - auto DirEntry = FM.getDirectoryRef(Dir.empty() ? "." : Dir); + auto DirEntry = FM.getDirectoryRef(Dir); if (!DirEntry) return llvm::errorToErrorCode(DirEntry.takeError()); diff --git a/clang/lib/Serialization/ASTWriter.cpp b/clang/lib/Serialization/ASTWriter.cpp index 65bee806d2c55..348d71d0e6ade 100644 --- a/clang/lib/Serialization/ASTWriter.cpp +++ b/clang/lib/Serialization/ASTWriter.cpp @@ -1372,11 +1372,15 @@ void ASTWriter::WriteControlBlock(Preprocessor &PP, ASTContext &Context, Record.clear(); auto &Map = PP.getHeaderSearchInfo().getModuleMap(); - AddPath(WritingModule->PresumedModuleMapFile.empty() - ? Map.getModuleMapFileForUniquing(WritingModule) - ->getNameAsRequested() - : StringRef(WritingModule->PresumedModuleMapFile), - Record); + + if (!WritingModule->PresumedModuleMapFile.empty()) { + AddPath(WritingModule->PresumedModuleMapFile, Record); + } else { + auto FE = Map.getModuleMapFileForUniquing(WritingModule); + SmallString<128> ModuleMapPath = FE->getNameAsRequested(); + Map.canonicalizeModuleMapPath(ModuleMapPath); + AddPath(ModuleMapPath, Record); + } // Additional module map files. if (auto *AdditionalModMaps = diff --git a/clang/lib/Tooling/DependencyScanning/ModuleDepCollector.cpp b/clang/lib/Tooling/DependencyScanning/ModuleDepCollector.cpp index 5e028c4431fe9..06359b40f2732 100644 --- a/clang/lib/Tooling/DependencyScanning/ModuleDepCollector.cpp +++ b/clang/lib/Tooling/DependencyScanning/ModuleDepCollector.cpp @@ -482,16 +482,9 @@ ModuleDepCollectorPP::handleTopLevelModule(const Module *M) { MD.ID.ModuleName = M->getFullModuleName(); MD.IsSystem = M->IsSystem; - ModuleMap &ModMapInfo = - MDC.ScanInstance.getPreprocessor().getHeaderSearchInfo().getModuleMap(); - - OptionalFileEntryRef ModuleMap = ModMapInfo.getModuleMapFileForUniquing(M); - - if (ModuleMap) { - SmallString<128> Path = ModuleMap->getNameAsRequested(); - ModMapInfo.canonicalizeModuleMapPath(Path); - MD.ClangModuleMapFile = std::string(Path); - } + assert(!M->PresumedModuleMapFile.empty() && + "Precompiled module missing presumed module map"); + MD.ClangModuleMapFile = M->PresumedModuleMapFile; serialization::ModuleFile *MF = MDC.ScanInstance.getASTReader()->getModuleManager().lookup( @@ -505,7 +498,10 @@ ModuleDepCollectorPP::handleTopLevelModule(const Module *M) { // handle it like normal. With explicitly built modules we don't need // to play VFS tricks, so replace it with the correct module map. if (StringRef(IFI.Filename).endswith("__inferred_module.map")) { - MDC.addFileDep(MD, ModuleMap->getName()); + auto FE = MDC.ScanInstance.getFileManager().getOptionalFileRef( + MD.ClangModuleMapFile); + assert(FE && "Module map of precompiled module still exists"); + MDC.addFileDep(MD, FE->getName()); return; } MDC.addFileDep(MD, IFI.Filename); diff --git a/clang/test/ClangScanDeps/modules-inferred-vfs-mod-map.m b/clang/test/ClangScanDeps/modules-inferred-vfs-mod-map.m new file mode 100644 index 0000000000000..d14779e73a51d --- /dev/null +++ b/clang/test/ClangScanDeps/modules-inferred-vfs-mod-map.m @@ -0,0 +1,63 @@ +// This test checks that we report the module map that allowed inferring using +// its on-disk path in file dependencies. + +// RUN: rm -rf %t +// RUN: split-file %s %t + +//--- frameworks/Inferred.framework/Headers/Inferred.h +//--- frameworks/Inferred.framework/Frameworks/Sub.framework/Headers/Sub.h +//--- real/module.modulemap +framework module * {} +//--- vfsoverlay.json.template +{ + "version": 0, + "case-sensitive": "false", + "use-external-names": true, + "roots": [ + { + "contents": [ + { + "external-contents": "DIR/real/module.modulemap", + "name": "module.modulemap", + "type": "file" + } + ], + "name": "DIR/frameworks", + "type": "directory" + } + ] +} +//--- tu.m +#include + +//--- cdb.json.template +[{ + "directory": "DIR", + "file": "DIR/tu.m", + "command": "clang -fmodules -fimplicit-module-maps -fmodules-cache-path=DIR/cache -F DIR/frameworks -ivfsoverlay DIR/vfsoverlay.json -c DIR/tu.m -o DIR/tu.o" +}] + +// RUN: sed "s|DIR|%/t|g" %t/vfsoverlay.json.template > %t/vfsoverlay.json +// RUN: sed "s|DIR|%/t|g" %t/cdb.json.template > %t/cdb.json +// RUN: clang-scan-deps -compilation-database %t/cdb.json -format experimental-full > %t/result.json +// RUN: cat %t/result.json | sed 's:\\\\\?:/:g' | FileCheck %s -DPREFIX=%/t + +// CHECK: { +// CHECK-NEXT: "modules": [ +// CHECK-NEXT: { +// CHECK-NEXT: "clang-module-deps": [], +// CHECK-NEXT: "clang-modulemap-file": "[[PREFIX]]/frameworks/module.modulemap", +// CHECK-NEXT: "command-line": [ +// CHECK: ], +// CHECK-NEXT: "context-hash": "{{.*}}", +// CHECK-NEXT: "file-deps": [ +// CHECK-NEXT: "[[PREFIX]]/frameworks/Inferred.framework/Frameworks/Sub.framework/Headers/Sub.h", +// CHECK-NEXT: "[[PREFIX]]/frameworks/Inferred.framework/Headers/Inferred.h", +// CHECK-NEXT: "[[PREFIX]]/real/module.modulemap" +// CHECK-NEXT: ], +// CHECK-NEXT: "name": "Inferred" +// CHECK-NEXT: } +// CHECK-NEXT: ], +// CHECK-NEXT: "translation-units": [ +// CHECK: ] +// CHECK: }