jansvoboda11 created this revision. jansvoboda11 added reviewers: Bigcheese, dexonsmith. jansvoboda11 requested review of this revision. Herald added a project: clang. Herald added a subscriber: cfe-commits.
This patch uses the new `CompilerInvocation::generateCC1CommandLine` to generate the full canonical command line for modular dependencies, instead of only appending additional arguments. Repository: rG LLVM Github Monorepo https://reviews.llvm.org/D100534 Files: clang/include/clang/Tooling/DependencyScanning/ModuleDepCollector.h clang/lib/Tooling/DependencyScanning/ModuleDepCollector.cpp clang/test/ClangScanDeps/modules-full.cpp
Index: clang/test/ClangScanDeps/modules-full.cpp =================================================================== --- clang/test/ClangScanDeps/modules-full.cpp +++ clang/test/ClangScanDeps/modules-full.cpp @@ -31,11 +31,11 @@ // CHECK-NEXT: ], // CHECK-NEXT: "clang-modulemap-file": "[[PREFIX]]/Inputs/module.modulemap", // CHECK-NEXT: "command-line": [ -// CHECK-NEXT: "-fno-implicit-modules", -// CHECK-NEXT: "-fno-implicit-module-maps", -// CHECK-NEXT: "-fmodule-file=[[PREFIX]]/module-cache/[[CONTEXT_HASH_H1]]/header2-{{[A-Z0-9]+}}.pcm", -// CHECK-NEXT: "-fmodule-map-file=[[PREFIX]]/Inputs/module.modulemap" -// CHECK-NEXT: ], +// CHECK: "-fmodule-map-file=[[PREFIX]]/Inputs/module.modulemap", +// CHECK: "-fmodule-file=[[PREFIX]]/module-cache/[[CONTEXT_HASH_H1]]/header2-{{[A-Z0-9]+}}.pcm", +// CHECK-NOT: "-fimplicit-module-maps", +// CHECK: "-fno-implicit-modules", +// CHECK: ], // CHECK-NEXT: "context-hash": "[[CONTEXT_HASH_H1]]", // CHECK-NEXT: "file-deps": [ // CHECK-NEXT: "[[PREFIX]]/Inputs/header.h", @@ -47,9 +47,9 @@ // CHECK-NEXT: "clang-module-deps": [], // CHECK-NEXT: "clang-modulemap-file": "[[PREFIX]]/Inputs/module.modulemap", // CHECK-NEXT: "command-line": [ -// CHECK-NEXT: "-fno-implicit-modules", -// CHECK-NEXT: "-fno-implicit-module-maps" -// CHECK-NEXT: ], +// CHECK-NOT: "-fimplicit-module-maps", +// CHECK: "-fno-implicit-modules", +// CHECK: ], // CHECK-NEXT: "context-hash": "[[CONTEXT_HASH_H2:[A-Z0-9]+]]", // CHECK-NEXT: "file-deps": [ // CHECK-NEXT: "[[PREFIX]]/Inputs/header.h", @@ -61,9 +61,9 @@ // CHECK-NEXT: "clang-module-deps": [], // CHECK-NEXT: "clang-modulemap-file": "[[PREFIX]]/Inputs/module.modulemap", // CHECK-NEXT: "command-line": [ -// CHECK-NEXT: "-fno-implicit-modules", -// CHECK-NEXT: "-fno-implicit-module-maps" -// CHECK-NEXT: ], +// CHECK-NOT: "-fimplicit-module-maps", +// CHECK: "-fno-implicit-modules", +// CHECK: ], // CHECK-NEXT: "context-hash": "[[CONTEXT_HASH_H1]]", // CHECK-NEXT: "file-deps": [ // CHECK-NEXT: "[[PREFIX]]/Inputs/header2.h", Index: clang/lib/Tooling/DependencyScanning/ModuleDepCollector.cpp =================================================================== --- clang/lib/Tooling/DependencyScanning/ModuleDepCollector.cpp +++ clang/lib/Tooling/DependencyScanning/ModuleDepCollector.cpp @@ -12,32 +12,60 @@ #include "clang/Frontend/CompilerInstance.h" #include "clang/Lex/Preprocessor.h" #include "clang/Tooling/DependencyScanning/DependencyScanningWorker.h" +#include "llvm/Support/StringSaver.h" using namespace clang; using namespace tooling; using namespace dependencies; +static CompilerInvocation +getFullCommandLineCompilerInvocation(const ModuleDeps &Deps) { + CompilerInvocation CI = Deps.Invocation; + + // Remove options incompatible with explicit module build. + CI.getFrontendOpts().Inputs.clear(); + CI.getFrontendOpts().OutputFile.clear(); + + CI.getFrontendOpts().ProgramAction = frontend::GenerateModule; + CI.getLangOpts()->ModuleName = Deps.ID.ModuleName; + CI.getFrontendOpts().IsSystemModule = Deps.IsSystem; + + CI.getLangOpts()->ImplicitModules = false; + CI.getHeaderSearchOpts().ImplicitModuleMaps = false; + + return CI; +} + +static std::vector<std::string> +serializeCompilerInvocation(CompilerInvocation &CI) { + // Set up string allocator. + llvm::BumpPtrAllocator Alloc; + llvm::StringSaver Strings(Alloc); + auto SA = [&Strings](const Twine &Arg) { return Strings.save(Arg).data(); }; + SmallVector<const char *, 32> Args; + + // Synthesize full command line from the CompilerInvocation. + CI.generateCC1CommandLine(Args, SA); + + // Convert arguments to the return type. + std::vector<std::string> Ret; + Ret.reserve(Args.size()); + for (const char *Arg : Args) + Ret.emplace_back(Arg); + + return Ret; +} + std::vector<std::string> ModuleDeps::getFullCommandLine( std::function<StringRef(ModuleID)> LookupPCMPath, std::function<const ModuleDeps &(ModuleID)> LookupModuleDeps) const { - // TODO: Build full command line. That also means capturing the original - // command line into NonPathCommandLine. - - std::vector<std::string> Ret{ - "-fno-implicit-modules", - "-fno-implicit-module-maps", - }; + CompilerInvocation CI = getFullCommandLineCompilerInvocation(*this); - std::vector<std::string> PCMPaths; - std::vector<std::string> ModMapPaths; dependencies::detail::collectPCMAndModuleMapPaths( - ClangModuleDeps, LookupPCMPath, LookupModuleDeps, PCMPaths, ModMapPaths); - for (const std::string &PCMPath : PCMPaths) - Ret.push_back("-fmodule-file=" + PCMPath); - for (const std::string &ModMapPath : ModMapPaths) - Ret.push_back("-fmodule-map-file=" + ModMapPath); + ClangModuleDeps, LookupPCMPath, LookupModuleDeps, + CI.getFrontendOpts().ModuleFiles, CI.getFrontendOpts().ModuleMapFiles); - return Ret; + return serializeCompilerInvocation(CI); } void dependencies::detail::collectPCMAndModuleMapPaths( @@ -149,10 +177,12 @@ .getModuleMap() .getContainingModuleMapFile(M); + MD.Invocation = Instance.getInvocation(); MD.ClangModuleMapFile = std::string(ModuleMap ? ModuleMap->getName() : ""); MD.ID.ModuleName = M->getFullModuleName(); MD.ImplicitModulePCMPath = std::string(M->getASTFile()->getName()); MD.ID.ContextHash = MDC.ContextHash; + MD.IsSystem = M->IsSystem; serialization::ModuleFile *MF = MDC.Instance.getASTReader()->getModuleManager().lookup(M->getASTFile()); MDC.Instance.getASTReader()->visitInputFiles( Index: clang/include/clang/Tooling/DependencyScanning/ModuleDepCollector.h =================================================================== --- clang/include/clang/Tooling/DependencyScanning/ModuleDepCollector.h +++ clang/include/clang/Tooling/DependencyScanning/ModuleDepCollector.h @@ -12,6 +12,7 @@ #include "clang/Basic/LLVM.h" #include "clang/Basic/SourceManager.h" +#include "clang/Frontend/CompilerInvocation.h" #include "clang/Frontend/Utils.h" #include "clang/Lex/HeaderSearch.h" #include "clang/Lex/PPCallbacks.h" @@ -47,6 +48,9 @@ /// The identifier of the module. ModuleID ID; + /// Whether this is a "system" module. + bool IsSystem; + /// The path to the modulemap file which defines this module. /// /// This can be used to explicitly build this module. This file will @@ -71,6 +75,10 @@ // the primary TU. bool ImportedByMainFile = false; + /// The compiler invocation associated with the translation unit that imports + /// this module. + CompilerInvocation Invocation; + /// Gets the full command line suitable for passing to clang. /// /// \param LookupPCMPath This function is called to fill in `-fmodule-file=`
_______________________________________________ cfe-commits mailing list cfe-commits@lists.llvm.org https://lists.llvm.org/cgi-bin/mailman/listinfo/cfe-commits