Skip to content

Commit 01e3213

Browse files
committed
ThinLTO: use the callgraph from the combined index to drive the FunctionImporter
Summary: Now that the summary contains the full reference/call graph, we can replace the existing function importer that loads and inspect the IR to iteratively walk the call graph by a traversal based purely on the summary information. Decouple the actual importing decision from any IR manipulation. Reviewers: tejohnson Subscribers: llvm-commits, joker.eph Differential Revision: http://reviews.llvm.org/D18343 From: Mehdi Amini <mehdi.amini@apple.com> llvm-svn: 264503
1 parent 4022168 commit 01e3213

File tree

7 files changed

+324
-270
lines changed

7 files changed

+324
-270
lines changed

llvm/include/llvm/Transforms/IPO/FunctionImport.h

+38-7
Original file line numberDiff line numberDiff line change
@@ -11,7 +11,10 @@
1111
#define LLVM_FUNCTIONIMPORT_H
1212

1313
#include "llvm/ADT/StringMap.h"
14+
1415
#include <functional>
16+
#include <map>
17+
#include <unordered_set>
1518

1619
namespace llvm {
1720
class LLVMContext;
@@ -21,23 +24,51 @@ class ModuleSummaryIndex;
2124
/// The function importer is automatically importing function from other modules
2225
/// based on the provided summary informations.
2326
class FunctionImporter {
27+
public:
28+
/// Set of functions to import from a source module. Each entry is a map
29+
/// containing all the functions to import for a source module.
30+
/// The keys is the GUID identifying a function to import, and the value
31+
/// is the threshold applied when deciding to import it.
32+
typedef std::map<uint64_t, unsigned> FunctionsToImportTy;
2433

25-
/// The summaries index used to trigger importing.
26-
const ModuleSummaryIndex &Index;
34+
/// The map contains an entry for every module to import from, the key being
35+
/// the module identifier to pass to the ModuleLoader. The value is the set of
36+
/// functions to import.
37+
typedef StringMap<FunctionsToImportTy> ImportMapTy;
2738

28-
/// Factory function to load a Module for a given identifier
29-
std::function<std::unique_ptr<Module>(StringRef Identifier)> ModuleLoader;
39+
/// The set contains an entry for every global value the module exports.
40+
typedef std::unordered_set<uint64_t> ExportSetTy;
3041

31-
public:
3242
/// Create a Function Importer.
3343
FunctionImporter(
3444
const ModuleSummaryIndex &Index,
3545
std::function<std::unique_ptr<Module>(StringRef Identifier)> ModuleLoader)
3646
: Index(Index), ModuleLoader(ModuleLoader) {}
3747

38-
/// Import functions in Module \p M based on the summary informations.
39-
bool importFunctions(Module &M);
48+
/// Import functions in Module \p M based on the supplied import list.
49+
bool importFunctions(Module &M, const ImportMapTy &ImportList);
50+
51+
private:
52+
/// The summaries index used to trigger importing.
53+
const ModuleSummaryIndex &Index;
54+
55+
/// Factory function to load a Module for a given identifier
56+
std::function<std::unique_ptr<Module>(StringRef Identifier)> ModuleLoader;
4057
};
58+
59+
/// Compute all the imports and exports for every module in the Index.
60+
///
61+
/// \p ImportLists will be populated with an entry for every Module we are
62+
/// importing into. This entry is itself a map that can be passed to
63+
/// FunctionImporter::importFunctions() above (see description there).
64+
///
65+
/// \p ExportLists contains for each Module the set of globals (GUID) that will
66+
/// be imported by another module, or referenced by such a function. I.e. this
67+
/// is the set of globals that need to be promoted/renamed appropriately.
68+
void ComputeCrossModuleImport(
69+
const ModuleSummaryIndex &Index,
70+
StringMap<FunctionImporter::ImportMapTy> &ImportLists,
71+
StringMap<FunctionImporter::ExportSetTy> &ExportLists);
4172
}
4273

4374
#endif // LLVM_FUNCTIONIMPORT_H

llvm/lib/LTO/ThinLTOCodeGenerator.cpp

+26-8
Original file line numberDiff line numberDiff line change
@@ -131,12 +131,13 @@ static void promoteModule(Module &TheModule, const ModuleSummaryIndex &Index) {
131131
report_fatal_error("renameModuleForThinLTO failed");
132132
}
133133

134-
static void crossImportIntoModule(Module &TheModule,
135-
const ModuleSummaryIndex &Index,
136-
StringMap<MemoryBufferRef> &ModuleMap) {
134+
static void
135+
crossImportIntoModule(Module &TheModule, const ModuleSummaryIndex &Index,
136+
StringMap<MemoryBufferRef> &ModuleMap,
137+
const FunctionImporter::ImportMapTy &ImportList) {
137138
ModuleLoader Loader(TheModule.getContext(), ModuleMap);
138139
FunctionImporter Importer(Index, Loader);
139-
Importer.importFunctions(TheModule);
140+
Importer.importFunctions(TheModule, ImportList);
140141
}
141142

142143
static void optimizeModule(Module &TheModule, TargetMachine &TM) {
@@ -185,6 +186,7 @@ std::unique_ptr<MemoryBuffer> codegenModule(Module &TheModule,
185186
static std::unique_ptr<MemoryBuffer>
186187
ProcessThinLTOModule(Module &TheModule, const ModuleSummaryIndex &Index,
187188
StringMap<MemoryBufferRef> &ModuleMap, TargetMachine &TM,
189+
const FunctionImporter::ImportMapTy &ImportList,
188190
ThinLTOCodeGenerator::CachingOptions CacheOptions,
189191
StringRef SaveTempsDir, unsigned count) {
190192

@@ -200,7 +202,7 @@ ProcessThinLTOModule(Module &TheModule, const ModuleSummaryIndex &Index,
200202
// Save temps: after promotion.
201203
saveTempBitcode(TheModule, SaveTempsDir, count, ".2.promoted.bc");
202204

203-
crossImportIntoModule(TheModule, Index, ModuleMap);
205+
crossImportIntoModule(TheModule, Index, ModuleMap, ImportList);
204206

205207
// Save temps: after cross-module import.
206208
saveTempBitcode(TheModule, SaveTempsDir, count, ".3.imported.bc");
@@ -317,7 +319,15 @@ void ThinLTOCodeGenerator::promote(Module &TheModule,
317319
void ThinLTOCodeGenerator::crossModuleImport(Module &TheModule,
318320
ModuleSummaryIndex &Index) {
319321
auto ModuleMap = generateModuleMap(Modules);
320-
crossImportIntoModule(TheModule, Index, ModuleMap);
322+
323+
// Generate import/export list
324+
auto ModuleCount = Index.modulePaths().size();
325+
StringMap<FunctionImporter::ImportMapTy> ImportLists(ModuleCount);
326+
StringMap<FunctionImporter::ExportSetTy> ExportLists(ModuleCount);
327+
ComputeCrossModuleImport(Index, ImportLists, ExportLists);
328+
auto &ImportList = ImportLists[TheModule.getModuleIdentifier()];
329+
330+
crossImportIntoModule(TheModule, Index, ModuleMap, ImportList);
321331
}
322332

323333
/**
@@ -358,6 +368,13 @@ void ThinLTOCodeGenerator::run() {
358368

359369
// Prepare the module map.
360370
auto ModuleMap = generateModuleMap(Modules);
371+
auto ModuleCount = Modules.size();
372+
373+
// Collect the import/export lists for all modules from the call-graph in the
374+
// combined index.
375+
StringMap<FunctionImporter::ImportMapTy> ImportLists(ModuleCount);
376+
StringMap<FunctionImporter::ExportSetTy> ExportLists(ModuleCount);
377+
ComputeCrossModuleImport(*Index, ImportLists, ExportLists);
361378

362379
// Parallel optimizer + codegen
363380
{
@@ -376,9 +393,10 @@ void ThinLTOCodeGenerator::run() {
376393
saveTempBitcode(*TheModule, SaveTempsDir, count, ".0.original.bc");
377394
}
378395

396+
auto &ImportList = ImportLists[TheModule->getModuleIdentifier()];
379397
ProducedBinaries[count] = ProcessThinLTOModule(
380-
*TheModule, *Index, ModuleMap, *TMBuilder.create(), CacheOptions,
381-
SaveTempsDir, count);
398+
*TheModule, *Index, ModuleMap, *TMBuilder.create(), ImportList,
399+
CacheOptions, SaveTempsDir, count);
382400
}, count);
383401
count++;
384402
}

0 commit comments

Comments
 (0)