142418abaSMehdi Amini //===- FunctionImport.cpp - ThinLTO Summary-based Function Import ---------===//
242418abaSMehdi Amini //
342418abaSMehdi Amini //                     The LLVM Compiler Infrastructure
442418abaSMehdi Amini //
542418abaSMehdi Amini // This file is distributed under the University of Illinois Open Source
642418abaSMehdi Amini // License. See LICENSE.TXT for details.
742418abaSMehdi Amini //
842418abaSMehdi Amini //===----------------------------------------------------------------------===//
942418abaSMehdi Amini //
1042418abaSMehdi Amini // This file implements Function import based on summaries.
1142418abaSMehdi Amini //
1242418abaSMehdi Amini //===----------------------------------------------------------------------===//
1342418abaSMehdi Amini 
1442418abaSMehdi Amini #include "llvm/Transforms/IPO/FunctionImport.h"
1542418abaSMehdi Amini 
1601e32130SMehdi Amini #include "llvm/ADT/SmallVector.h"
17d29478f7STeresa Johnson #include "llvm/ADT/Statistic.h"
1842418abaSMehdi Amini #include "llvm/ADT/StringSet.h"
1942418abaSMehdi Amini #include "llvm/IR/AutoUpgrade.h"
2042418abaSMehdi Amini #include "llvm/IR/DiagnosticPrinter.h"
2142418abaSMehdi Amini #include "llvm/IR/IntrinsicInst.h"
2242418abaSMehdi Amini #include "llvm/IR/Module.h"
2342418abaSMehdi Amini #include "llvm/IRReader/IRReader.h"
2442418abaSMehdi Amini #include "llvm/Linker/Linker.h"
2526ab5772STeresa Johnson #include "llvm/Object/ModuleSummaryIndexObjectFile.h"
2642418abaSMehdi Amini #include "llvm/Support/CommandLine.h"
2742418abaSMehdi Amini #include "llvm/Support/Debug.h"
2842418abaSMehdi Amini #include "llvm/Support/SourceMgr.h"
29488a800aSTeresa Johnson #include "llvm/Transforms/Utils/FunctionImportUtils.h"
307e88d0daSMehdi Amini 
3101e32130SMehdi Amini #define DEBUG_TYPE "function-import"
327e88d0daSMehdi Amini 
3342418abaSMehdi Amini using namespace llvm;
3442418abaSMehdi Amini 
35d29478f7STeresa Johnson STATISTIC(NumImported, "Number of functions imported");
36d29478f7STeresa Johnson 
3739303619STeresa Johnson /// Limit on instruction count of imported functions.
3839303619STeresa Johnson static cl::opt<unsigned> ImportInstrLimit(
3939303619STeresa Johnson     "import-instr-limit", cl::init(100), cl::Hidden, cl::value_desc("N"),
4039303619STeresa Johnson     cl::desc("Only import functions with less than N instructions"));
4139303619STeresa Johnson 
4240641748SMehdi Amini static cl::opt<float>
4340641748SMehdi Amini     ImportInstrFactor("import-instr-evolution-factor", cl::init(0.7),
4440641748SMehdi Amini                       cl::Hidden, cl::value_desc("x"),
4540641748SMehdi Amini                       cl::desc("As we import functions, multiply the "
4640641748SMehdi Amini                                "`import-instr-limit` threshold by this factor "
4740641748SMehdi Amini                                "before processing newly imported functions"));
4840641748SMehdi Amini 
49d29478f7STeresa Johnson static cl::opt<bool> PrintImports("print-imports", cl::init(false), cl::Hidden,
50d29478f7STeresa Johnson                                   cl::desc("Print imported functions"));
51d29478f7STeresa Johnson 
5242418abaSMehdi Amini // Load lazily a module from \p FileName in \p Context.
5342418abaSMehdi Amini static std::unique_ptr<Module> loadFile(const std::string &FileName,
5442418abaSMehdi Amini                                         LLVMContext &Context) {
5542418abaSMehdi Amini   SMDiagnostic Err;
5642418abaSMehdi Amini   DEBUG(dbgs() << "Loading '" << FileName << "'\n");
576cba37ceSTeresa Johnson   // Metadata isn't loaded until functions are imported, to minimize
586cba37ceSTeresa Johnson   // the memory overhead.
59a1080ee6STeresa Johnson   std::unique_ptr<Module> Result =
60a1080ee6STeresa Johnson       getLazyIRFileModule(FileName, Err, Context,
61a1080ee6STeresa Johnson                           /* ShouldLazyLoadMetadata = */ true);
6242418abaSMehdi Amini   if (!Result) {
6342418abaSMehdi Amini     Err.print("function-import", errs());
64d7ad221cSMehdi Amini     report_fatal_error("Abort");
6542418abaSMehdi Amini   }
6642418abaSMehdi Amini 
6742418abaSMehdi Amini   return Result;
6842418abaSMehdi Amini }
6942418abaSMehdi Amini 
707e88d0daSMehdi Amini namespace {
7140641748SMehdi Amini 
7201e32130SMehdi Amini /// Given a list of possible callee implementation for a call site, select one
7301e32130SMehdi Amini /// that fits the \p Threshold.
7401e32130SMehdi Amini ///
7501e32130SMehdi Amini /// FIXME: select "best" instead of first that fits. But what is "best"?
7601e32130SMehdi Amini /// - The smallest: more likely to be inlined.
7701e32130SMehdi Amini /// - The one with the least outgoing edges (already well optimized).
7801e32130SMehdi Amini /// - One from a module already being imported from in order to reduce the
7901e32130SMehdi Amini ///   number of source modules parsed/linked.
8001e32130SMehdi Amini /// - One that has PGO data attached.
8101e32130SMehdi Amini /// - [insert you fancy metric here]
8201e32130SMehdi Amini static const FunctionSummary *
8301e32130SMehdi Amini selectCallee(const GlobalValueInfoList &CalleeInfoList, unsigned Threshold) {
8401e32130SMehdi Amini   auto It = llvm::find_if(
8501e32130SMehdi Amini       CalleeInfoList, [&](const std::unique_ptr<GlobalValueInfo> &GlobInfo) {
8601e32130SMehdi Amini         assert(GlobInfo->summary() &&
8701e32130SMehdi Amini                "We should not have a Global Info without summary");
8801e32130SMehdi Amini         auto *Summary = cast<FunctionSummary>(GlobInfo->summary());
8940641748SMehdi Amini 
9001e32130SMehdi Amini         if (GlobalValue::isWeakAnyLinkage(Summary->linkage()))
9101e32130SMehdi Amini           return false;
927e88d0daSMehdi Amini 
9301e32130SMehdi Amini         if (Summary->instCount() > Threshold)
9401e32130SMehdi Amini           return false;
957e88d0daSMehdi Amini 
9601e32130SMehdi Amini         return true;
9701e32130SMehdi Amini       });
9801e32130SMehdi Amini   if (It == CalleeInfoList.end())
9901e32130SMehdi Amini     return nullptr;
1007e88d0daSMehdi Amini 
10101e32130SMehdi Amini   return cast<FunctionSummary>((*It)->summary());
102434e9561SRafael Espindola }
1037e88d0daSMehdi Amini 
10401e32130SMehdi Amini /// Return the summary for the function \p GUID that fits the \p Threshold, or
10501e32130SMehdi Amini /// null if there's no match.
106*ad5741b0SMehdi Amini static const FunctionSummary *selectCallee(GlobalValue::GUID GUID,
107*ad5741b0SMehdi Amini                                            unsigned Threshold,
10801e32130SMehdi Amini                                            const ModuleSummaryIndex &Index) {
10901e32130SMehdi Amini   auto CalleeInfoList = Index.findGlobalValueInfoList(GUID);
11001e32130SMehdi Amini   if (CalleeInfoList == Index.end()) {
11101e32130SMehdi Amini     return nullptr; // This function does not have a summary
1127e88d0daSMehdi Amini   }
11301e32130SMehdi Amini   return selectCallee(CalleeInfoList->second, Threshold);
11401e32130SMehdi Amini }
1157e88d0daSMehdi Amini 
11601e32130SMehdi Amini /// Return true if the global \p GUID is exported by module \p ExportModulePath.
11701e32130SMehdi Amini static bool isGlobalExported(const ModuleSummaryIndex &Index,
118*ad5741b0SMehdi Amini                              StringRef ExportModulePath,
119*ad5741b0SMehdi Amini                              GlobalValue::GUID GUID) {
12001e32130SMehdi Amini   auto CalleeInfoList = Index.findGlobalValueInfoList(GUID);
12101e32130SMehdi Amini   if (CalleeInfoList == Index.end())
12201e32130SMehdi Amini     // This global does not have a summary, it is not part of the ThinLTO
12301e32130SMehdi Amini     // process
12401e32130SMehdi Amini     return false;
12501e32130SMehdi Amini   auto DefinedInCalleeModule = llvm::find_if(
12601e32130SMehdi Amini       CalleeInfoList->second,
12701e32130SMehdi Amini       [&](const std::unique_ptr<GlobalValueInfo> &GlobInfo) {
12801e32130SMehdi Amini         auto *Summary = GlobInfo->summary();
12901e32130SMehdi Amini         assert(Summary && "Unexpected GlobalValueInfo without summary");
13001e32130SMehdi Amini         return Summary->modulePath() == ExportModulePath;
13101e32130SMehdi Amini       });
13201e32130SMehdi Amini   return (DefinedInCalleeModule != CalleeInfoList->second.end());
13301e32130SMehdi Amini }
1347e88d0daSMehdi Amini 
13501e32130SMehdi Amini using EdgeInfo = std::pair<const FunctionSummary *, unsigned /* Threshold */>;
13601e32130SMehdi Amini 
13701e32130SMehdi Amini /// Compute the list of functions to import for a given caller. Mark these
13801e32130SMehdi Amini /// imported functions and the symbols they reference in their source module as
13901e32130SMehdi Amini /// exported from their source module.
14001e32130SMehdi Amini static void computeImportForFunction(
14101e32130SMehdi Amini     StringRef ModulePath, const FunctionSummary &Summary,
14201e32130SMehdi Amini     const ModuleSummaryIndex &Index, unsigned Threshold,
143*ad5741b0SMehdi Amini     const std::map<GlobalValue::GUID, FunctionSummary *> &DefinedFunctions,
14401e32130SMehdi Amini     SmallVectorImpl<EdgeInfo> &Worklist,
14501e32130SMehdi Amini     FunctionImporter::ImportMapTy &ImportsForModule,
14601e32130SMehdi Amini     StringMap<FunctionImporter::ExportSetTy> &ExportLists) {
14701e32130SMehdi Amini   for (auto &Edge : Summary.calls()) {
14801e32130SMehdi Amini     auto GUID = Edge.first;
14901e32130SMehdi Amini     DEBUG(dbgs() << " edge -> " << GUID << " Threshold:" << Threshold << "\n");
15001e32130SMehdi Amini 
15101e32130SMehdi Amini     if (DefinedFunctions.count(GUID)) {
15201e32130SMehdi Amini       DEBUG(dbgs() << "ignored! Target already in destination module.\n");
1537e88d0daSMehdi Amini       continue;
154d450da32STeresa Johnson     }
15540641748SMehdi Amini 
15601e32130SMehdi Amini     auto *CalleeSummary = selectCallee(GUID, Threshold, Index);
15701e32130SMehdi Amini     if (!CalleeSummary) {
15801e32130SMehdi Amini       DEBUG(dbgs() << "ignored! No qualifying callee with summary found.\n");
1597e88d0daSMehdi Amini       continue;
1607e88d0daSMehdi Amini     }
16101e32130SMehdi Amini     assert(CalleeSummary->instCount() <= Threshold &&
16201e32130SMehdi Amini            "selectCallee() didn't honor the threshold");
16301e32130SMehdi Amini 
16401e32130SMehdi Amini     auto &ProcessedThreshold =
16501e32130SMehdi Amini         ImportsForModule[CalleeSummary->modulePath()][GUID];
16601e32130SMehdi Amini     /// Since the traversal of the call graph is DFS, we can revisit a function
16701e32130SMehdi Amini     /// a second time with a higher threshold. In this case, it is added back to
16801e32130SMehdi Amini     /// the worklist with the new threshold.
16901e32130SMehdi Amini     if (ProcessedThreshold && ProcessedThreshold > Threshold) {
17001e32130SMehdi Amini       DEBUG(dbgs() << "ignored! Target was already seen with Threshold "
17101e32130SMehdi Amini                    << ProcessedThreshold << "\n");
17201e32130SMehdi Amini       continue;
17301e32130SMehdi Amini     }
17401e32130SMehdi Amini     // Mark this function as imported in this module, with the current Threshold
17501e32130SMehdi Amini     ProcessedThreshold = Threshold;
17601e32130SMehdi Amini 
17701e32130SMehdi Amini     // Make exports in the source module.
17801e32130SMehdi Amini     auto ExportModulePath = CalleeSummary->modulePath();
17901e32130SMehdi Amini     auto ExportList = ExportLists[ExportModulePath];
18001e32130SMehdi Amini     ExportList.insert(GUID);
18101e32130SMehdi Amini     // Mark all functions and globals referenced by this function as exported to
18201e32130SMehdi Amini     // the outside if they are defined in the same source module.
18301e32130SMehdi Amini     for (auto &Edge : CalleeSummary->calls()) {
18401e32130SMehdi Amini       auto CalleeGUID = Edge.first;
18501e32130SMehdi Amini       if (isGlobalExported(Index, ExportModulePath, CalleeGUID))
18601e32130SMehdi Amini         ExportList.insert(CalleeGUID);
18701e32130SMehdi Amini     }
18801e32130SMehdi Amini     for (auto &GUID : CalleeSummary->refs()) {
18901e32130SMehdi Amini       if (isGlobalExported(Index, ExportModulePath, GUID))
19001e32130SMehdi Amini         ExportList.insert(GUID);
1917e88d0daSMehdi Amini     }
1927e88d0daSMehdi Amini 
19301e32130SMehdi Amini     // Insert the newly imported function to the worklist.
19401e32130SMehdi Amini     Worklist.push_back(std::make_pair(CalleeSummary, Threshold));
195d450da32STeresa Johnson   }
196d450da32STeresa Johnson }
197d450da32STeresa Johnson 
19801e32130SMehdi Amini /// Given the list of globals defined in a module, compute the list of imports
19901e32130SMehdi Amini /// as well as the list of "exports", i.e. the list of symbols referenced from
20001e32130SMehdi Amini /// another module (that may require promotion).
20101e32130SMehdi Amini static void ComputeImportForModule(
20201e32130SMehdi Amini     StringRef ModulePath,
203*ad5741b0SMehdi Amini     const std::map<GlobalValue::GUID, FunctionSummary *> &DefinedFunctions,
20426ab5772STeresa Johnson     const ModuleSummaryIndex &Index,
20501e32130SMehdi Amini     FunctionImporter::ImportMapTy &ImportsForModule,
20601e32130SMehdi Amini     StringMap<FunctionImporter::ExportSetTy> &ExportLists) {
20701e32130SMehdi Amini   // Worklist contains the list of function imported in this module, for which
20801e32130SMehdi Amini   // we will analyse the callees and may import further down the callgraph.
20901e32130SMehdi Amini   SmallVector<EdgeInfo, 128> Worklist;
21001e32130SMehdi Amini 
21101e32130SMehdi Amini   // Populate the worklist with the import for the functions in the current
21201e32130SMehdi Amini   // module
21301e32130SMehdi Amini   for (auto &FuncInfo : DefinedFunctions) {
21401e32130SMehdi Amini     auto *Summary = FuncInfo.second;
21501e32130SMehdi Amini     DEBUG(dbgs() << "Initalize import for " << FuncInfo.first << "\n");
21601e32130SMehdi Amini     computeImportForFunction(ModulePath, *Summary, Index, ImportInstrLimit,
21701e32130SMehdi Amini                              DefinedFunctions, Worklist, ImportsForModule,
21801e32130SMehdi Amini                              ExportLists);
21901e32130SMehdi Amini   }
22001e32130SMehdi Amini 
22142418abaSMehdi Amini   while (!Worklist.empty()) {
22201e32130SMehdi Amini     auto FuncInfo = Worklist.pop_back_val();
22301e32130SMehdi Amini     auto *Summary = FuncInfo.first;
22401e32130SMehdi Amini     auto Threshold = FuncInfo.second;
22542418abaSMehdi Amini 
2267e88d0daSMehdi Amini     // Process the newly imported functions and add callees to the worklist.
22740641748SMehdi Amini     // Adjust the threshold
22840641748SMehdi Amini     Threshold = Threshold * ImportInstrFactor;
22901e32130SMehdi Amini 
23001e32130SMehdi Amini     computeImportForFunction(ModulePath, *Summary, Index, Threshold,
23101e32130SMehdi Amini                              DefinedFunctions, Worklist, ImportsForModule,
23201e32130SMehdi Amini                              ExportLists);
233c8c55170SMehdi Amini   }
23442418abaSMehdi Amini }
235ffe2e4aaSMehdi Amini 
23601e32130SMehdi Amini } // anonymous namespace
23701e32130SMehdi Amini 
23801e32130SMehdi Amini /// Compute all the import and export for every module in the Index.
23901e32130SMehdi Amini void llvm::ComputeCrossModuleImport(
24001e32130SMehdi Amini     const ModuleSummaryIndex &Index,
24101e32130SMehdi Amini     StringMap<FunctionImporter::ImportMapTy> &ImportLists,
24201e32130SMehdi Amini     StringMap<FunctionImporter::ExportSetTy> &ExportLists) {
24301e32130SMehdi Amini   auto ModuleCount = Index.modulePaths().size();
24401e32130SMehdi Amini 
24501e32130SMehdi Amini   // Collect for each module the list of function it defines.
24601e32130SMehdi Amini   // GUID -> Summary
247*ad5741b0SMehdi Amini   StringMap<std::map<GlobalValue::GUID, FunctionSummary *>>
248*ad5741b0SMehdi Amini       Module2FunctionInfoMap(ModuleCount);
24901e32130SMehdi Amini 
25001e32130SMehdi Amini   for (auto &GlobalList : Index) {
25101e32130SMehdi Amini     auto GUID = GlobalList.first;
25201e32130SMehdi Amini     for (auto &GlobInfo : GlobalList.second) {
25301e32130SMehdi Amini       auto *Summary = dyn_cast_or_null<FunctionSummary>(GlobInfo->summary());
25401e32130SMehdi Amini       if (!Summary)
25501e32130SMehdi Amini         /// Ignore global variable, focus on functions
25601e32130SMehdi Amini         continue;
25701e32130SMehdi Amini       DEBUG(dbgs() << "Adding definition: Module '" << Summary->modulePath()
25801e32130SMehdi Amini                    << "' defines '" << GUID << "'\n");
25901e32130SMehdi Amini       Module2FunctionInfoMap[Summary->modulePath()][GUID] = Summary;
26001e32130SMehdi Amini     }
26101e32130SMehdi Amini   }
26201e32130SMehdi Amini 
26301e32130SMehdi Amini   // For each module that has function defined, compute the import/export lists.
26401e32130SMehdi Amini   for (auto &DefinedFunctions : Module2FunctionInfoMap) {
26501e32130SMehdi Amini     auto &ImportsForModule = ImportLists[DefinedFunctions.first()];
26601e32130SMehdi Amini     DEBUG(dbgs() << "Computing import for Module '" << DefinedFunctions.first()
26701e32130SMehdi Amini                  << "'\n");
26801e32130SMehdi Amini     ComputeImportForModule(DefinedFunctions.first(), DefinedFunctions.second,
26901e32130SMehdi Amini                            Index, ImportsForModule, ExportLists);
27001e32130SMehdi Amini   }
27101e32130SMehdi Amini 
27201e32130SMehdi Amini #ifndef NDEBUG
27301e32130SMehdi Amini   DEBUG(dbgs() << "Import/Export lists for " << ImportLists.size()
27401e32130SMehdi Amini                << " modules:\n");
27501e32130SMehdi Amini   for (auto &ModuleImports : ImportLists) {
27601e32130SMehdi Amini     auto ModName = ModuleImports.first();
27701e32130SMehdi Amini     auto &Exports = ExportLists[ModName];
27801e32130SMehdi Amini     DEBUG(dbgs() << "* Module " << ModName << " exports " << Exports.size()
27901e32130SMehdi Amini                  << " functions. Imports from " << ModuleImports.second.size()
28001e32130SMehdi Amini                  << " modules.\n");
28101e32130SMehdi Amini     for (auto &Src : ModuleImports.second) {
28201e32130SMehdi Amini       auto SrcModName = Src.first();
28301e32130SMehdi Amini       DEBUG(dbgs() << " - " << Src.second.size() << " functions imported from "
28401e32130SMehdi Amini                    << SrcModName << "\n");
28501e32130SMehdi Amini     }
28601e32130SMehdi Amini   }
28701e32130SMehdi Amini #endif
28801e32130SMehdi Amini }
28901e32130SMehdi Amini 
290c8c55170SMehdi Amini // Automatically import functions in Module \p DestModule based on the summaries
291c8c55170SMehdi Amini // index.
292c8c55170SMehdi Amini //
29301e32130SMehdi Amini bool FunctionImporter::importFunctions(
29401e32130SMehdi Amini     Module &DestModule, const FunctionImporter::ImportMapTy &ImportList) {
2955411d051SMehdi Amini   DEBUG(dbgs() << "Starting import for Module "
296311fef6eSMehdi Amini                << DestModule.getModuleIdentifier() << "\n");
297c8c55170SMehdi Amini   unsigned ImportedCount = 0;
298c8c55170SMehdi Amini 
299c8c55170SMehdi Amini   // Linker that will be used for importing function
3009d2bfc48SRafael Espindola   Linker TheLinker(DestModule);
3017e88d0daSMehdi Amini   // Do the actual import of functions now, one Module at a time
30201e32130SMehdi Amini   std::set<StringRef> ModuleNameOrderedList;
30301e32130SMehdi Amini   for (auto &FunctionsToImportPerModule : ImportList) {
30401e32130SMehdi Amini     ModuleNameOrderedList.insert(FunctionsToImportPerModule.first());
30501e32130SMehdi Amini   }
30601e32130SMehdi Amini   for (auto &Name : ModuleNameOrderedList) {
3077e88d0daSMehdi Amini     // Get the module for the import
30801e32130SMehdi Amini     const auto &FunctionsToImportPerModule = ImportList.find(Name);
30901e32130SMehdi Amini     assert(FunctionsToImportPerModule != ImportList.end());
31001e32130SMehdi Amini     std::unique_ptr<Module> SrcModule = ModuleLoader(Name);
3117e88d0daSMehdi Amini     assert(&DestModule.getContext() == &SrcModule->getContext() &&
3127e88d0daSMehdi Amini            "Context mismatch");
3137e88d0daSMehdi Amini 
3146cba37ceSTeresa Johnson     // If modules were created with lazy metadata loading, materialize it
3156cba37ceSTeresa Johnson     // now, before linking it (otherwise this will be a noop).
3166cba37ceSTeresa Johnson     SrcModule->materializeMetadata();
3176cba37ceSTeresa Johnson     UpgradeDebugInfo(*SrcModule);
318e5a61917STeresa Johnson 
31901e32130SMehdi Amini     auto &ImportGUIDs = FunctionsToImportPerModule->second;
32001e32130SMehdi Amini     // Find the globals to import
32101e32130SMehdi Amini     DenseSet<const GlobalValue *> GlobalsToImport;
32201e32130SMehdi Amini     for (auto &GV : *SrcModule) {
32301e32130SMehdi Amini       if (GV.hasName() && ImportGUIDs.count(GV.getGUID())) {
32401e32130SMehdi Amini         GV.materialize();
32501e32130SMehdi Amini         GlobalsToImport.insert(&GV);
32601e32130SMehdi Amini       }
32701e32130SMehdi Amini     }
32801e32130SMehdi Amini     for (auto &GV : SrcModule->aliases()) {
32901e32130SMehdi Amini       if (!GV.hasName())
33001e32130SMehdi Amini         continue;
33101e32130SMehdi Amini       auto GUID = GV.getGUID();
33201e32130SMehdi Amini       if (ImportGUIDs.count(GUID)) {
33301e32130SMehdi Amini         // Alias can't point to "available_externally". However when we import
3349aae395fSTeresa Johnson         // linkOnceODR the linkage does not change. So we import the alias
3359aae395fSTeresa Johnson         // and aliasee only in this case.
33601e32130SMehdi Amini         const GlobalObject *GO = GV.getBaseObject();
33701e32130SMehdi Amini         if (!GO->hasLinkOnceODRLinkage())
33801e32130SMehdi Amini           continue;
3399aae395fSTeresa Johnson         GV.materialize();
3409aae395fSTeresa Johnson         GlobalsToImport.insert(&GV);
34101e32130SMehdi Amini         GlobalsToImport.insert(GO);
34201e32130SMehdi Amini       }
34301e32130SMehdi Amini     }
34401e32130SMehdi Amini     for (auto &GV : SrcModule->globals()) {
34501e32130SMehdi Amini       if (!GV.hasName())
34601e32130SMehdi Amini         continue;
347efeae0e2STeresa Johnson       auto GUID = GV.getGUID();
34801e32130SMehdi Amini       if (ImportGUIDs.count(GUID)) {
34901e32130SMehdi Amini         GV.materialize();
35001e32130SMehdi Amini         GlobalsToImport.insert(&GV);
35101e32130SMehdi Amini       }
35201e32130SMehdi Amini     }
35301e32130SMehdi Amini 
3547e88d0daSMehdi Amini     // Link in the specified functions.
35501e32130SMehdi Amini     if (renameModuleForThinLTO(*SrcModule, Index, &GlobalsToImport))
3568d05185aSMehdi Amini       return true;
3578d05185aSMehdi Amini 
358d29478f7STeresa Johnson     if (PrintImports) {
359d29478f7STeresa Johnson       for (const auto *GV : GlobalsToImport)
360d29478f7STeresa Johnson         dbgs() << DestModule.getSourceFileName() << ": Import " << GV->getName()
361d29478f7STeresa Johnson                << " from " << SrcModule->getSourceFileName() << "\n";
362d29478f7STeresa Johnson     }
363d29478f7STeresa Johnson 
364434e9561SRafael Espindola     if (TheLinker.linkInModule(std::move(SrcModule), Linker::Flags::None,
36501e32130SMehdi Amini                                &GlobalsToImport))
3667e88d0daSMehdi Amini       report_fatal_error("Function Import: link error");
3677e88d0daSMehdi Amini 
36801e32130SMehdi Amini     ImportedCount += GlobalsToImport.size();
3697e88d0daSMehdi Amini   }
370e5a61917STeresa Johnson 
371d29478f7STeresa Johnson   NumImported += ImportedCount;
372d29478f7STeresa Johnson 
3737e88d0daSMehdi Amini   DEBUG(dbgs() << "Imported " << ImportedCount << " functions for Module "
374c8c55170SMehdi Amini                << DestModule.getModuleIdentifier() << "\n");
375c8c55170SMehdi Amini   return ImportedCount;
37642418abaSMehdi Amini }
37742418abaSMehdi Amini 
37842418abaSMehdi Amini /// Summary file to use for function importing when using -function-import from
37942418abaSMehdi Amini /// the command line.
38042418abaSMehdi Amini static cl::opt<std::string>
38142418abaSMehdi Amini     SummaryFile("summary-file",
38242418abaSMehdi Amini                 cl::desc("The summary file to use for function importing."));
38342418abaSMehdi Amini 
38442418abaSMehdi Amini static void diagnosticHandler(const DiagnosticInfo &DI) {
38542418abaSMehdi Amini   raw_ostream &OS = errs();
38642418abaSMehdi Amini   DiagnosticPrinterRawOStream DP(OS);
38742418abaSMehdi Amini   DI.print(DP);
38842418abaSMehdi Amini   OS << '\n';
38942418abaSMehdi Amini }
39042418abaSMehdi Amini 
39126ab5772STeresa Johnson /// Parse the summary index out of an IR file and return the summary
39242418abaSMehdi Amini /// index object if found, or nullptr if not.
39326ab5772STeresa Johnson static std::unique_ptr<ModuleSummaryIndex>
39426ab5772STeresa Johnson getModuleSummaryIndexForFile(StringRef Path, std::string &Error,
39542418abaSMehdi Amini                              DiagnosticHandlerFunction DiagnosticHandler) {
39642418abaSMehdi Amini   std::unique_ptr<MemoryBuffer> Buffer;
39742418abaSMehdi Amini   ErrorOr<std::unique_ptr<MemoryBuffer>> BufferOrErr =
39842418abaSMehdi Amini       MemoryBuffer::getFile(Path);
39942418abaSMehdi Amini   if (std::error_code EC = BufferOrErr.getError()) {
40042418abaSMehdi Amini     Error = EC.message();
40142418abaSMehdi Amini     return nullptr;
40242418abaSMehdi Amini   }
40342418abaSMehdi Amini   Buffer = std::move(BufferOrErr.get());
40426ab5772STeresa Johnson   ErrorOr<std::unique_ptr<object::ModuleSummaryIndexObjectFile>> ObjOrErr =
40526ab5772STeresa Johnson       object::ModuleSummaryIndexObjectFile::create(Buffer->getMemBufferRef(),
40642418abaSMehdi Amini                                                    DiagnosticHandler);
40742418abaSMehdi Amini   if (std::error_code EC = ObjOrErr.getError()) {
40842418abaSMehdi Amini     Error = EC.message();
40942418abaSMehdi Amini     return nullptr;
41042418abaSMehdi Amini   }
41142418abaSMehdi Amini   return (*ObjOrErr)->takeIndex();
41242418abaSMehdi Amini }
41342418abaSMehdi Amini 
414fe2b5415SBenjamin Kramer namespace {
41542418abaSMehdi Amini /// Pass that performs cross-module function import provided a summary file.
41642418abaSMehdi Amini class FunctionImportPass : public ModulePass {
41726ab5772STeresa Johnson   /// Optional module summary index to use for importing, otherwise
4185fcbdb71STeresa Johnson   /// the summary-file option must be specified.
41926ab5772STeresa Johnson   const ModuleSummaryIndex *Index;
42042418abaSMehdi Amini 
42142418abaSMehdi Amini public:
42242418abaSMehdi Amini   /// Pass identification, replacement for typeid
42342418abaSMehdi Amini   static char ID;
42442418abaSMehdi Amini 
4255fcbdb71STeresa Johnson   /// Specify pass name for debug output
4265fcbdb71STeresa Johnson   const char *getPassName() const override {
4275fcbdb71STeresa Johnson     return "Function Importing";
4285fcbdb71STeresa Johnson   }
4295fcbdb71STeresa Johnson 
43026ab5772STeresa Johnson   explicit FunctionImportPass(const ModuleSummaryIndex *Index = nullptr)
4315fcbdb71STeresa Johnson       : ModulePass(ID), Index(Index) {}
43242418abaSMehdi Amini 
43342418abaSMehdi Amini   bool runOnModule(Module &M) override {
4345fcbdb71STeresa Johnson     if (SummaryFile.empty() && !Index)
4355fcbdb71STeresa Johnson       report_fatal_error("error: -function-import requires -summary-file or "
4365fcbdb71STeresa Johnson                          "file from frontend\n");
43726ab5772STeresa Johnson     std::unique_ptr<ModuleSummaryIndex> IndexPtr;
4385fcbdb71STeresa Johnson     if (!SummaryFile.empty()) {
4395fcbdb71STeresa Johnson       if (Index)
4405fcbdb71STeresa Johnson         report_fatal_error("error: -summary-file and index from frontend\n");
44142418abaSMehdi Amini       std::string Error;
44226ab5772STeresa Johnson       IndexPtr =
44326ab5772STeresa Johnson           getModuleSummaryIndexForFile(SummaryFile, Error, diagnosticHandler);
4445fcbdb71STeresa Johnson       if (!IndexPtr) {
4455fcbdb71STeresa Johnson         errs() << "Error loading file '" << SummaryFile << "': " << Error
4465fcbdb71STeresa Johnson                << "\n";
44742418abaSMehdi Amini         return false;
44842418abaSMehdi Amini       }
4495fcbdb71STeresa Johnson       Index = IndexPtr.get();
4505fcbdb71STeresa Johnson     }
45142418abaSMehdi Amini 
45201e32130SMehdi Amini     // First step is collecting the import/export lists
45301e32130SMehdi Amini     // The export list is not used yet, but could limit the amount of renaming
45401e32130SMehdi Amini     // performed in renameModuleForThinLTO()
45501e32130SMehdi Amini     StringMap<FunctionImporter::ImportMapTy> ImportLists;
45601e32130SMehdi Amini     StringMap<FunctionImporter::ExportSetTy> ExportLists;
45701e32130SMehdi Amini     ComputeCrossModuleImport(*Index, ImportLists, ExportLists);
45801e32130SMehdi Amini     auto &ImportList = ImportLists[M.getModuleIdentifier()];
45901e32130SMehdi Amini 
46001e32130SMehdi Amini     // Next we need to promote to global scope and rename any local values that
4611b00f2d9STeresa Johnson     // are potentially exported to other modules.
46201e32130SMehdi Amini     if (renameModuleForThinLTO(M, *Index, nullptr)) {
4631b00f2d9STeresa Johnson       errs() << "Error renaming module\n";
4641b00f2d9STeresa Johnson       return false;
4651b00f2d9STeresa Johnson     }
4661b00f2d9STeresa Johnson 
46742418abaSMehdi Amini     // Perform the import now.
468d16c8065SMehdi Amini     auto ModuleLoader = [&M](StringRef Identifier) {
469d16c8065SMehdi Amini       return loadFile(Identifier, M.getContext());
470d16c8065SMehdi Amini     };
4719d2bfc48SRafael Espindola     FunctionImporter Importer(*Index, ModuleLoader);
47201e32130SMehdi Amini     return Importer.importFunctions(M, ImportList);
47342418abaSMehdi Amini   }
47442418abaSMehdi Amini };
475fe2b5415SBenjamin Kramer } // anonymous namespace
47642418abaSMehdi Amini 
47742418abaSMehdi Amini char FunctionImportPass::ID = 0;
47842418abaSMehdi Amini INITIALIZE_PASS_BEGIN(FunctionImportPass, "function-import",
47942418abaSMehdi Amini                       "Summary Based Function Import", false, false)
48042418abaSMehdi Amini INITIALIZE_PASS_END(FunctionImportPass, "function-import",
48142418abaSMehdi Amini                     "Summary Based Function Import", false, false)
48242418abaSMehdi Amini 
48342418abaSMehdi Amini namespace llvm {
48426ab5772STeresa Johnson Pass *createFunctionImportPass(const ModuleSummaryIndex *Index = nullptr) {
4855fcbdb71STeresa Johnson   return new FunctionImportPass(Index);
4865fcbdb71STeresa Johnson }
48742418abaSMehdi Amini }
488