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.
106ad5741b0SMehdi Amini static const FunctionSummary *selectCallee(GlobalValue::GUID GUID,
107ad5741b0SMehdi 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,
118ad5741b0SMehdi Amini                              StringRef ExportModulePath,
119ad5741b0SMehdi 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(
1413255eec1STeresa Johnson     const FunctionSummary &Summary, const ModuleSummaryIndex &Index,
1423255eec1STeresa Johnson     unsigned Threshold,
143ad5741b0SMehdi 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()) {
148*2d5487cfSTeresa Johnson     auto GUID = Edge.first.getGUID();
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()) {
184*2d5487cfSTeresa Johnson       auto CalleeGUID = Edge.first.getGUID();
18501e32130SMehdi Amini       if (isGlobalExported(Index, ExportModulePath, CalleeGUID))
18601e32130SMehdi Amini         ExportList.insert(CalleeGUID);
18701e32130SMehdi Amini     }
188*2d5487cfSTeresa Johnson     for (auto &Ref : CalleeSummary->refs()) {
189*2d5487cfSTeresa Johnson       auto GUID = Ref.getGUID();
19001e32130SMehdi Amini       if (isGlobalExported(Index, ExportModulePath, GUID))
19101e32130SMehdi Amini         ExportList.insert(GUID);
1927e88d0daSMehdi Amini     }
1937e88d0daSMehdi Amini 
19401e32130SMehdi Amini     // Insert the newly imported function to the worklist.
19501e32130SMehdi Amini     Worklist.push_back(std::make_pair(CalleeSummary, Threshold));
196d450da32STeresa Johnson   }
197d450da32STeresa Johnson }
198d450da32STeresa Johnson 
19901e32130SMehdi Amini /// Given the list of globals defined in a module, compute the list of imports
20001e32130SMehdi Amini /// as well as the list of "exports", i.e. the list of symbols referenced from
20101e32130SMehdi Amini /// another module (that may require promotion).
20201e32130SMehdi Amini static void ComputeImportForModule(
203ad5741b0SMehdi 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");
2163255eec1STeresa Johnson     computeImportForFunction(*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 
2303255eec1STeresa Johnson     computeImportForFunction(*Summary, Index, Threshold, DefinedFunctions,
2313255eec1STeresa Johnson                              Worklist, ImportsForModule, ExportLists);
232c8c55170SMehdi Amini   }
23342418abaSMehdi Amini }
234ffe2e4aaSMehdi Amini 
23501e32130SMehdi Amini } // anonymous namespace
23601e32130SMehdi Amini 
23701e32130SMehdi Amini /// Compute all the import and export for every module in the Index.
23801e32130SMehdi Amini void llvm::ComputeCrossModuleImport(
23901e32130SMehdi Amini     const ModuleSummaryIndex &Index,
24001e32130SMehdi Amini     StringMap<FunctionImporter::ImportMapTy> &ImportLists,
24101e32130SMehdi Amini     StringMap<FunctionImporter::ExportSetTy> &ExportLists) {
24201e32130SMehdi Amini   auto ModuleCount = Index.modulePaths().size();
24301e32130SMehdi Amini 
24401e32130SMehdi Amini   // Collect for each module the list of function it defines.
24501e32130SMehdi Amini   // GUID -> Summary
246ad5741b0SMehdi Amini   StringMap<std::map<GlobalValue::GUID, FunctionSummary *>>
247ad5741b0SMehdi Amini       Module2FunctionInfoMap(ModuleCount);
24801e32130SMehdi Amini 
24901e32130SMehdi Amini   for (auto &GlobalList : Index) {
25001e32130SMehdi Amini     auto GUID = GlobalList.first;
25101e32130SMehdi Amini     for (auto &GlobInfo : GlobalList.second) {
25201e32130SMehdi Amini       auto *Summary = dyn_cast_or_null<FunctionSummary>(GlobInfo->summary());
25301e32130SMehdi Amini       if (!Summary)
25401e32130SMehdi Amini         /// Ignore global variable, focus on functions
25501e32130SMehdi Amini         continue;
25601e32130SMehdi Amini       DEBUG(dbgs() << "Adding definition: Module '" << Summary->modulePath()
25701e32130SMehdi Amini                    << "' defines '" << GUID << "'\n");
25801e32130SMehdi Amini       Module2FunctionInfoMap[Summary->modulePath()][GUID] = Summary;
25901e32130SMehdi Amini     }
26001e32130SMehdi Amini   }
26101e32130SMehdi Amini 
26201e32130SMehdi Amini   // For each module that has function defined, compute the import/export lists.
26301e32130SMehdi Amini   for (auto &DefinedFunctions : Module2FunctionInfoMap) {
26401e32130SMehdi Amini     auto &ImportsForModule = ImportLists[DefinedFunctions.first()];
26501e32130SMehdi Amini     DEBUG(dbgs() << "Computing import for Module '" << DefinedFunctions.first()
26601e32130SMehdi Amini                  << "'\n");
2673255eec1STeresa Johnson     ComputeImportForModule(DefinedFunctions.second, Index, ImportsForModule,
2683255eec1STeresa Johnson                            ExportLists);
26901e32130SMehdi Amini   }
27001e32130SMehdi Amini 
27101e32130SMehdi Amini #ifndef NDEBUG
27201e32130SMehdi Amini   DEBUG(dbgs() << "Import/Export lists for " << ImportLists.size()
27301e32130SMehdi Amini                << " modules:\n");
27401e32130SMehdi Amini   for (auto &ModuleImports : ImportLists) {
27501e32130SMehdi Amini     auto ModName = ModuleImports.first();
27601e32130SMehdi Amini     auto &Exports = ExportLists[ModName];
27701e32130SMehdi Amini     DEBUG(dbgs() << "* Module " << ModName << " exports " << Exports.size()
27801e32130SMehdi Amini                  << " functions. Imports from " << ModuleImports.second.size()
27901e32130SMehdi Amini                  << " modules.\n");
28001e32130SMehdi Amini     for (auto &Src : ModuleImports.second) {
28101e32130SMehdi Amini       auto SrcModName = Src.first();
28201e32130SMehdi Amini       DEBUG(dbgs() << " - " << Src.second.size() << " functions imported from "
28301e32130SMehdi Amini                    << SrcModName << "\n");
28401e32130SMehdi Amini     }
28501e32130SMehdi Amini   }
28601e32130SMehdi Amini #endif
28701e32130SMehdi Amini }
28801e32130SMehdi Amini 
289c8c55170SMehdi Amini // Automatically import functions in Module \p DestModule based on the summaries
290c8c55170SMehdi Amini // index.
291c8c55170SMehdi Amini //
29201e32130SMehdi Amini bool FunctionImporter::importFunctions(
29301e32130SMehdi Amini     Module &DestModule, const FunctionImporter::ImportMapTy &ImportList) {
2945411d051SMehdi Amini   DEBUG(dbgs() << "Starting import for Module "
295311fef6eSMehdi Amini                << DestModule.getModuleIdentifier() << "\n");
296c8c55170SMehdi Amini   unsigned ImportedCount = 0;
297c8c55170SMehdi Amini 
298c8c55170SMehdi Amini   // Linker that will be used for importing function
2999d2bfc48SRafael Espindola   Linker TheLinker(DestModule);
3007e88d0daSMehdi Amini   // Do the actual import of functions now, one Module at a time
30101e32130SMehdi Amini   std::set<StringRef> ModuleNameOrderedList;
30201e32130SMehdi Amini   for (auto &FunctionsToImportPerModule : ImportList) {
30301e32130SMehdi Amini     ModuleNameOrderedList.insert(FunctionsToImportPerModule.first());
30401e32130SMehdi Amini   }
30501e32130SMehdi Amini   for (auto &Name : ModuleNameOrderedList) {
3067e88d0daSMehdi Amini     // Get the module for the import
30701e32130SMehdi Amini     const auto &FunctionsToImportPerModule = ImportList.find(Name);
30801e32130SMehdi Amini     assert(FunctionsToImportPerModule != ImportList.end());
30901e32130SMehdi Amini     std::unique_ptr<Module> SrcModule = ModuleLoader(Name);
3107e88d0daSMehdi Amini     assert(&DestModule.getContext() == &SrcModule->getContext() &&
3117e88d0daSMehdi Amini            "Context mismatch");
3127e88d0daSMehdi Amini 
3136cba37ceSTeresa Johnson     // If modules were created with lazy metadata loading, materialize it
3146cba37ceSTeresa Johnson     // now, before linking it (otherwise this will be a noop).
3156cba37ceSTeresa Johnson     SrcModule->materializeMetadata();
3166cba37ceSTeresa Johnson     UpgradeDebugInfo(*SrcModule);
317e5a61917STeresa Johnson 
31801e32130SMehdi Amini     auto &ImportGUIDs = FunctionsToImportPerModule->second;
31901e32130SMehdi Amini     // Find the globals to import
32001e32130SMehdi Amini     DenseSet<const GlobalValue *> GlobalsToImport;
32101e32130SMehdi Amini     for (auto &GV : *SrcModule) {
3220beb858eSTeresa Johnson       if (!GV.hasName())
3230beb858eSTeresa Johnson         continue;
3240beb858eSTeresa Johnson       auto GUID = GV.getGUID();
3250beb858eSTeresa Johnson       auto Import = ImportGUIDs.count(GUID);
3260beb858eSTeresa Johnson       DEBUG(dbgs() << (Import ? "Is" : "Not") << " importing " << GUID << " "
3270beb858eSTeresa Johnson                    << GV.getName() << " from " << SrcModule->getSourceFileName()
3280beb858eSTeresa Johnson                    << "\n");
3290beb858eSTeresa Johnson       if (Import) {
33001e32130SMehdi Amini         GV.materialize();
33101e32130SMehdi Amini         GlobalsToImport.insert(&GV);
33201e32130SMehdi Amini       }
33301e32130SMehdi Amini     }
33401e32130SMehdi Amini     for (auto &GV : SrcModule->aliases()) {
33501e32130SMehdi Amini       if (!GV.hasName())
33601e32130SMehdi Amini         continue;
33701e32130SMehdi Amini       auto GUID = GV.getGUID();
3380beb858eSTeresa Johnson       auto Import = ImportGUIDs.count(GUID);
3390beb858eSTeresa Johnson       DEBUG(dbgs() << (Import ? "Is" : "Not") << " importing " << GUID << " "
3400beb858eSTeresa Johnson                    << GV.getName() << " from " << SrcModule->getSourceFileName()
3410beb858eSTeresa Johnson                    << "\n");
3420beb858eSTeresa Johnson       if (Import) {
34301e32130SMehdi Amini         // Alias can't point to "available_externally". However when we import
3449aae395fSTeresa Johnson         // linkOnceODR the linkage does not change. So we import the alias
3459aae395fSTeresa Johnson         // and aliasee only in this case.
34601e32130SMehdi Amini         const GlobalObject *GO = GV.getBaseObject();
34701e32130SMehdi Amini         if (!GO->hasLinkOnceODRLinkage())
34801e32130SMehdi Amini           continue;
3499aae395fSTeresa Johnson         GV.materialize();
3509aae395fSTeresa Johnson         GlobalsToImport.insert(&GV);
35101e32130SMehdi Amini         GlobalsToImport.insert(GO);
35201e32130SMehdi Amini       }
35301e32130SMehdi Amini     }
35401e32130SMehdi Amini     for (auto &GV : SrcModule->globals()) {
35501e32130SMehdi Amini       if (!GV.hasName())
35601e32130SMehdi Amini         continue;
357efeae0e2STeresa Johnson       auto GUID = GV.getGUID();
3580beb858eSTeresa Johnson       auto Import = ImportGUIDs.count(GUID);
3590beb858eSTeresa Johnson       DEBUG(dbgs() << (Import ? "Is" : "Not") << " importing " << GUID << " "
3600beb858eSTeresa Johnson                    << GV.getName() << " from " << SrcModule->getSourceFileName()
3610beb858eSTeresa Johnson                    << "\n");
3620beb858eSTeresa Johnson       if (Import) {
36301e32130SMehdi Amini         GV.materialize();
36401e32130SMehdi Amini         GlobalsToImport.insert(&GV);
36501e32130SMehdi Amini       }
36601e32130SMehdi Amini     }
36701e32130SMehdi Amini 
3687e88d0daSMehdi Amini     // Link in the specified functions.
36901e32130SMehdi Amini     if (renameModuleForThinLTO(*SrcModule, Index, &GlobalsToImport))
3708d05185aSMehdi Amini       return true;
3718d05185aSMehdi Amini 
372d29478f7STeresa Johnson     if (PrintImports) {
373d29478f7STeresa Johnson       for (const auto *GV : GlobalsToImport)
374d29478f7STeresa Johnson         dbgs() << DestModule.getSourceFileName() << ": Import " << GV->getName()
375d29478f7STeresa Johnson                << " from " << SrcModule->getSourceFileName() << "\n";
376d29478f7STeresa Johnson     }
377d29478f7STeresa Johnson 
378434e9561SRafael Espindola     if (TheLinker.linkInModule(std::move(SrcModule), Linker::Flags::None,
37901e32130SMehdi Amini                                &GlobalsToImport))
3807e88d0daSMehdi Amini       report_fatal_error("Function Import: link error");
3817e88d0daSMehdi Amini 
38201e32130SMehdi Amini     ImportedCount += GlobalsToImport.size();
3837e88d0daSMehdi Amini   }
384e5a61917STeresa Johnson 
385d29478f7STeresa Johnson   NumImported += ImportedCount;
386d29478f7STeresa Johnson 
3877e88d0daSMehdi Amini   DEBUG(dbgs() << "Imported " << ImportedCount << " functions for Module "
388c8c55170SMehdi Amini                << DestModule.getModuleIdentifier() << "\n");
389c8c55170SMehdi Amini   return ImportedCount;
39042418abaSMehdi Amini }
39142418abaSMehdi Amini 
39242418abaSMehdi Amini /// Summary file to use for function importing when using -function-import from
39342418abaSMehdi Amini /// the command line.
39442418abaSMehdi Amini static cl::opt<std::string>
39542418abaSMehdi Amini     SummaryFile("summary-file",
39642418abaSMehdi Amini                 cl::desc("The summary file to use for function importing."));
39742418abaSMehdi Amini 
39842418abaSMehdi Amini static void diagnosticHandler(const DiagnosticInfo &DI) {
39942418abaSMehdi Amini   raw_ostream &OS = errs();
40042418abaSMehdi Amini   DiagnosticPrinterRawOStream DP(OS);
40142418abaSMehdi Amini   DI.print(DP);
40242418abaSMehdi Amini   OS << '\n';
40342418abaSMehdi Amini }
40442418abaSMehdi Amini 
40526ab5772STeresa Johnson /// Parse the summary index out of an IR file and return the summary
40642418abaSMehdi Amini /// index object if found, or nullptr if not.
40726ab5772STeresa Johnson static std::unique_ptr<ModuleSummaryIndex>
40826ab5772STeresa Johnson getModuleSummaryIndexForFile(StringRef Path, std::string &Error,
40942418abaSMehdi Amini                              DiagnosticHandlerFunction DiagnosticHandler) {
41042418abaSMehdi Amini   std::unique_ptr<MemoryBuffer> Buffer;
41142418abaSMehdi Amini   ErrorOr<std::unique_ptr<MemoryBuffer>> BufferOrErr =
41242418abaSMehdi Amini       MemoryBuffer::getFile(Path);
41342418abaSMehdi Amini   if (std::error_code EC = BufferOrErr.getError()) {
41442418abaSMehdi Amini     Error = EC.message();
41542418abaSMehdi Amini     return nullptr;
41642418abaSMehdi Amini   }
41742418abaSMehdi Amini   Buffer = std::move(BufferOrErr.get());
41826ab5772STeresa Johnson   ErrorOr<std::unique_ptr<object::ModuleSummaryIndexObjectFile>> ObjOrErr =
41926ab5772STeresa Johnson       object::ModuleSummaryIndexObjectFile::create(Buffer->getMemBufferRef(),
42042418abaSMehdi Amini                                                    DiagnosticHandler);
42142418abaSMehdi Amini   if (std::error_code EC = ObjOrErr.getError()) {
42242418abaSMehdi Amini     Error = EC.message();
42342418abaSMehdi Amini     return nullptr;
42442418abaSMehdi Amini   }
42542418abaSMehdi Amini   return (*ObjOrErr)->takeIndex();
42642418abaSMehdi Amini }
42742418abaSMehdi Amini 
428fe2b5415SBenjamin Kramer namespace {
42942418abaSMehdi Amini /// Pass that performs cross-module function import provided a summary file.
43042418abaSMehdi Amini class FunctionImportPass : public ModulePass {
43126ab5772STeresa Johnson   /// Optional module summary index to use for importing, otherwise
4325fcbdb71STeresa Johnson   /// the summary-file option must be specified.
43326ab5772STeresa Johnson   const ModuleSummaryIndex *Index;
43442418abaSMehdi Amini 
43542418abaSMehdi Amini public:
43642418abaSMehdi Amini   /// Pass identification, replacement for typeid
43742418abaSMehdi Amini   static char ID;
43842418abaSMehdi Amini 
4395fcbdb71STeresa Johnson   /// Specify pass name for debug output
4405fcbdb71STeresa Johnson   const char *getPassName() const override {
4415fcbdb71STeresa Johnson     return "Function Importing";
4425fcbdb71STeresa Johnson   }
4435fcbdb71STeresa Johnson 
44426ab5772STeresa Johnson   explicit FunctionImportPass(const ModuleSummaryIndex *Index = nullptr)
4455fcbdb71STeresa Johnson       : ModulePass(ID), Index(Index) {}
44642418abaSMehdi Amini 
44742418abaSMehdi Amini   bool runOnModule(Module &M) override {
4485fcbdb71STeresa Johnson     if (SummaryFile.empty() && !Index)
4495fcbdb71STeresa Johnson       report_fatal_error("error: -function-import requires -summary-file or "
4505fcbdb71STeresa Johnson                          "file from frontend\n");
45126ab5772STeresa Johnson     std::unique_ptr<ModuleSummaryIndex> IndexPtr;
4525fcbdb71STeresa Johnson     if (!SummaryFile.empty()) {
4535fcbdb71STeresa Johnson       if (Index)
4545fcbdb71STeresa Johnson         report_fatal_error("error: -summary-file and index from frontend\n");
45542418abaSMehdi Amini       std::string Error;
45626ab5772STeresa Johnson       IndexPtr =
45726ab5772STeresa Johnson           getModuleSummaryIndexForFile(SummaryFile, Error, diagnosticHandler);
4585fcbdb71STeresa Johnson       if (!IndexPtr) {
4595fcbdb71STeresa Johnson         errs() << "Error loading file '" << SummaryFile << "': " << Error
4605fcbdb71STeresa Johnson                << "\n";
46142418abaSMehdi Amini         return false;
46242418abaSMehdi Amini       }
4635fcbdb71STeresa Johnson       Index = IndexPtr.get();
4645fcbdb71STeresa Johnson     }
46542418abaSMehdi Amini 
46601e32130SMehdi Amini     // First step is collecting the import/export lists
46701e32130SMehdi Amini     // The export list is not used yet, but could limit the amount of renaming
46801e32130SMehdi Amini     // performed in renameModuleForThinLTO()
46901e32130SMehdi Amini     StringMap<FunctionImporter::ImportMapTy> ImportLists;
47001e32130SMehdi Amini     StringMap<FunctionImporter::ExportSetTy> ExportLists;
47101e32130SMehdi Amini     ComputeCrossModuleImport(*Index, ImportLists, ExportLists);
47201e32130SMehdi Amini     auto &ImportList = ImportLists[M.getModuleIdentifier()];
47301e32130SMehdi Amini 
47401e32130SMehdi Amini     // Next we need to promote to global scope and rename any local values that
4751b00f2d9STeresa Johnson     // are potentially exported to other modules.
47601e32130SMehdi Amini     if (renameModuleForThinLTO(M, *Index, nullptr)) {
4771b00f2d9STeresa Johnson       errs() << "Error renaming module\n";
4781b00f2d9STeresa Johnson       return false;
4791b00f2d9STeresa Johnson     }
4801b00f2d9STeresa Johnson 
48142418abaSMehdi Amini     // Perform the import now.
482d16c8065SMehdi Amini     auto ModuleLoader = [&M](StringRef Identifier) {
483d16c8065SMehdi Amini       return loadFile(Identifier, M.getContext());
484d16c8065SMehdi Amini     };
4859d2bfc48SRafael Espindola     FunctionImporter Importer(*Index, ModuleLoader);
48601e32130SMehdi Amini     return Importer.importFunctions(M, ImportList);
48742418abaSMehdi Amini   }
48842418abaSMehdi Amini };
489fe2b5415SBenjamin Kramer } // anonymous namespace
49042418abaSMehdi Amini 
49142418abaSMehdi Amini char FunctionImportPass::ID = 0;
49242418abaSMehdi Amini INITIALIZE_PASS_BEGIN(FunctionImportPass, "function-import",
49342418abaSMehdi Amini                       "Summary Based Function Import", false, false)
49442418abaSMehdi Amini INITIALIZE_PASS_END(FunctionImportPass, "function-import",
49542418abaSMehdi Amini                     "Summary Based Function Import", false, false)
49642418abaSMehdi Amini 
49742418abaSMehdi Amini namespace llvm {
49826ab5772STeresa Johnson Pass *createFunctionImportPass(const ModuleSummaryIndex *Index = nullptr) {
4995fcbdb71STeresa Johnson   return new FunctionImportPass(Index);
5005fcbdb71STeresa Johnson }
50142418abaSMehdi Amini }
502