54#include <system_error>
60#define DEBUG_TYPE "function-import"
63 "Number of functions thin link decided to import");
65 "Number of hot functions thin link decided to import");
67 "Number of critical functions thin link decided to import");
69 "Number of global variables thin link decided to import");
70STATISTIC(NumImportedFunctions,
"Number of functions imported in backend");
72 "Number of global variables imported in backend");
73STATISTIC(NumImportedModules,
"Number of modules imported from");
74STATISTIC(NumDeadSymbols,
"Number of dead stripped symbols in index");
75STATISTIC(NumLiveSymbols,
"Number of live symbols in index");
82 cl::desc(
"Import functions with noinline attribute"));
87 cl::desc(
"Only import functions with less than N instructions"));
91 cl::desc(
"Only import first N functions if N>=0 (default -1)"));
96 cl::desc(
"As we import functions, multiply the "
97 "`import-instr-limit` threshold by this factor "
98 "before processing newly imported functions"));
103 cl::desc(
"As we import functions called from hot callsite, multiply the "
104 "`import-instr-limit` threshold by this factor "
105 "before processing newly imported functions"));
109 cl::desc(
"Multiply the `import-instr-limit` threshold for hot callsites"));
115 "Multiply the `import-instr-limit` threshold for critical callsites"));
120 cl::desc(
"Multiply the `import-instr-limit` threshold for cold callsites"));
123 cl::desc(
"Print imported functions"));
127 cl::desc(
"Print information for functions rejected for importing"));
134 cl::desc(
"Enable import metadata like 'thinlto_src_module' and "
135 "'thinlto_src_file'"));
141 cl::desc(
"The summary file to use for function importing."));
147 cl::desc(
"Import all external functions in index."));
157 cl::desc(
"If true, import function declaration as fallback if the function "
158 "definition is not imported."));
169 "thinlto-workload-def",
170 cl::desc(
"Pass a workload definition. This is a file containing a JSON "
171 "dictionary. The keys are root functions, the values are lists of "
172 "functions to import in the module defining the root. It is "
173 "assumed -funique-internal-linkage-names was used, to ensure "
174 "local linkage functions have unique names. For example: \n"
176 " \"rootFunction_1\": [\"function_to_import_1\", "
177 "\"function_to_import_2\"], \n"
178 " \"rootFunction_2\": [\"function_to_import_3\", "
179 "\"function_to_import_4\"] \n"
186 "thinlto-move-ctxprof-trees",
187 cl::desc(
"Move contextual profiling roots and the graphs under them in "
188 "their own module."),
197static std::unique_ptr<Module>
loadFile(
const std::string &FileName,
203 std::unique_ptr<Module> Result =
207 Err.print(
"function-import",
errs());
236 ArrayRef<std::unique_ptr<GlobalValueSummary>> CalleeSummaryList,
240 [&Index, CalleeSummaryList,
241 CallerModulePath](
const std::unique_ptr<GlobalValueSummary> &SummaryPtr)
244 auto *GVSummary = SummaryPtr.get();
245 if (!Index.isGlobalValueLive(GVSummary))
282 if (Summary->notEligibleToImport())
307 ArrayRef<std::unique_ptr<GlobalValueSummary>> CalleeSummaryList,
308 unsigned Threshold,
StringRef CallerModulePath,
312 TooLargeOrNoInlineSummary =
nullptr;
313 auto QualifiedCandidates =
315 for (
auto QualifiedValue : QualifiedCandidates) {
316 Reason = QualifiedValue.first;
325 if ((Summary->instCount() > Threshold) && !Summary->fflags().AlwaysInline &&
327 TooLargeOrNoInlineSummary = Summary;
334 TooLargeOrNoInlineSummary = Summary;
353 auto [Def, Decl] = IDs.createImportIDs(FromModule, GUID);
354 if (!Imports.insert(Def).second)
366 auto [Def, Decl] = IDs.createImportIDs(FromModule, GUID);
369 if (!Imports.contains(Def))
370 Imports.insert(Decl);
376 for (
const auto &[SrcMod, GUID, ImportType] : *
this)
383std::optional<GlobalValueSummary::ImportKind>
386 if (
auto IDPair = IDs.getImportIDs(FromModule, GUID)) {
387 auto [Def, Decl] = *IDPair;
388 if (Imports.contains(Def))
390 if (Imports.contains(Decl))
406 bool shouldImportGlobal(
const ValueInfo &VI) {
407 const auto &GVS = DefinedGVSummaries.
find(VI.getGUID());
408 if (GVS == DefinedGVSummaries.end())
420 if (VI.getSummaryList().size() > 1 &&
422 !IsPrevailing(VI.getGUID(), GVS->second))
431 for (
const auto &VI : Summary.refs()) {
432 if (!shouldImportGlobal(VI)) {
434 dbgs() <<
"Ref ignored! Target already in destination module.\n");
440 for (
const auto &RefSummary : VI.getSummaryList()) {
458 bool CanImportDecl =
false;
460 Summary.modulePath()) ||
461 !Index.canImportGlobalVar(GVS,
true,
464 ImportList.maybeAddDeclaration(RefSummary->modulePath(),
472 if (ImportList.addDefinition(RefSummary->modulePath(), VI.getGUID()) !=
478 NumImportedGlobalVarsThinLink++;
483 (*ExportLists)[RefSummary->modulePath()].insert(VI);
487 if (!Index.isWriteOnly(GVS))
501 : Index(Index), DefinedGVSummaries(DefinedGVSummaries),
502 IsPrevailing(IsPrevailing), ImportList(ImportList),
503 ExportLists(ExportLists) {}
507 onImportingSummaryImpl(Summary, Worklist);
508 while (!Worklist.
empty())
509 onImportingSummaryImpl(*Worklist.
pop_back_val(), Worklist);
517 void computeImportForFunction(
549 static std::unique_ptr<ModuleImportsManager>
583 if (SetIter == Workloads.end()) {
585 <<
" does not contain the root of any context.\n");
587 ModName, ImportList);
590 <<
" contains the root(s) of context(s).\n");
594 auto &ValueInfos = SetIter->second;
596 auto It = DefinedGVSummaries.
find(VI.getGUID());
597 if (It != DefinedGVSummaries.
end() &&
600 dbgs() <<
"[Workload] " << VI.name()
601 <<
" has the prevailing variant already in the module "
602 << ModName <<
". No need to import\n");
612 [&](
const auto &Candidate) {
614 <<
" from " << Candidate.second->modulePath()
615 <<
" ImportFailureReason: "
617 return Candidate.first ==
620 [](
const auto &Candidate) {
return Candidate.second; });
621 if (PotentialCandidates.empty()) {
623 <<
" because can't find eligible Callee. Guid is: "
624 << VI.getGUID() <<
"\n");
642 PotentialCandidates, [&](
const auto *Candidate) {
645 if (PrevailingCandidates.empty()) {
646 GVS = *PotentialCandidates.begin();
651 <<
"[Workload] Found multiple non-prevailing candidates for "
653 <<
". This is unexpected. Are module paths passed to the "
654 "compiler unique for the modules passed to the linker?");
663 GVS = *PrevailingCandidates.begin();
670 if (ExportingModule == ModName) {
672 <<
" because its defining module is the same as the "
676 LLVM_DEBUG(
dbgs() <<
"[Workload][Including]" << VI.name() <<
" from "
677 << ExportingModule <<
" : " << VI.getGUID() <<
"\n");
681 (*ExportLists)[ExportingModule].insert(VI);
686 void loadFromJson() {
693 if (!NameToValueInfo.
insert(std::make_pair(VI.name(), VI)).second)
696 auto DbgReportIfAmbiguous = [&](
StringRef Name) {
698 dbgs() <<
"[Workload] Function name " << Name
699 <<
" present in the workload definition is ambiguous. Consider "
700 "compiling with -funique-internal-linkage-names.";
705 if (std::error_code EC = BufferOrErr.getError()) {
709 auto Buffer = std::move(BufferOrErr.get());
710 std::map<std::string, std::vector<std::string>> WorkloadDefs;
723 for (
const auto &Workload : WorkloadDefs) {
724 const auto &Root = Workload.first;
725 DbgReportIfAmbiguous(Root);
727 const auto &AllCallees = Workload.second;
728 auto RootIt = NameToValueInfo.
find(Root);
729 if (RootIt == NameToValueInfo.
end()) {
731 <<
" not found in this linkage unit.\n");
734 auto RootVI = RootIt->second;
735 if (RootVI.getSummaryList().size() != 1) {
737 <<
" should have exactly one summary, but has "
738 << RootVI.getSummaryList().size() <<
". Skipping.\n");
742 RootVI.getSummaryList().
front()->modulePath();
743 LLVM_DEBUG(
dbgs() <<
"[Workload] Root defining module for " << Root
744 <<
" is : " << RootDefiningModule <<
"\n");
745 auto &Set = Workloads[RootDefiningModule];
746 for (
const auto &Callee : AllCallees) {
748 DbgReportIfAmbiguous(Callee);
749 auto ElemIt = NameToValueInfo.
find(Callee);
750 if (ElemIt == NameToValueInfo.
end()) {
754 Set.insert(ElemIt->second);
759 void loadFromCtxProf() {
762 if (std::error_code EC = BufferOrErr.getError()) {
766 auto Buffer = std::move(BufferOrErr.get());
774 const auto &CtxMap = Ctx->Contexts;
776 for (
const auto &[RootGuid, Root] : CtxMap) {
779 ContainedGUIDs.
clear();
781 auto RootVI =
Index.getValueInfo(RootGuid);
784 <<
" not found in this linkage unit.\n");
787 if (RootVI.getSummaryList().size() != 1) {
789 <<
" should have exactly one summary, but has "
790 << RootVI.getSummaryList().size() <<
". Skipping.\n");
793 std::string RootDefiningModule =
794 RootVI.getSummaryList().front()->modulePath().str();
796 RootDefiningModule = std::to_string(RootGuid);
798 dbgs() <<
"[Workload] Moving " << RootGuid
799 <<
" to a module with the filename without extension : "
800 << RootDefiningModule <<
"\n");
802 LLVM_DEBUG(
dbgs() <<
"[Workload] Root defining module for " << RootGuid
803 <<
" is : " << RootDefiningModule <<
"\n");
805 auto &Set = Workloads[RootDefiningModule];
806 Root.getContainedGuids(ContainedGUIDs);
807 Roots.insert(RootVI);
808 for (
auto Guid : ContainedGUIDs)
825 "Pass only one of: -thinlto-pgo-ctx-prof or -thinlto-workload-def");
833 for (
const auto &[Root, Set] : Workloads) {
834 dbgs() <<
"[Workload] Root: " << Root <<
" we have " << Set.size()
835 <<
" distinct callees.\n";
836 for (
const auto &VI : Set) {
837 dbgs() <<
"[Workload] Root: " << Root
838 <<
" Would include: " << VI.getGUID() <<
"\n";
851 LLVM_DEBUG(
dbgs() <<
"[Workload] Using the regular imports manager.\n");
852 return std::unique_ptr<ModuleImportsManager>(
855 LLVM_DEBUG(
dbgs() <<
"[Workload] Using the contextual imports manager.\n");
872 return "InterposableLinkage";
874 return "LocalLinkageNotInModule";
876 return "NotEligible";
886void ModuleImportsManager::computeImportForFunction(
887 const FunctionSummary &Summary,
const unsigned Threshold,
889 SmallVectorImpl<EdgeInfo> &Worklist, GlobalsImporter &GVImporter,
890 FunctionImporter::ImportMapTy &ImportList,
893 static int ImportCount = 0;
895 ValueInfo
VI =
Edge.first;
896 LLVM_DEBUG(
dbgs() <<
" edge -> " << VI <<
" Threshold:" << Threshold
905 if (DefinedGVSummaries.
count(
VI.getGUID())) {
909 LLVM_DEBUG(
dbgs() <<
"ignored! Target already in destination module.\n");
913 if (!canImport(VI)) {
915 dbgs() <<
"Skipping over " <<
VI.getGUID()
916 <<
" because its import is handled in a different module.");
917 assert(
VI.getSummaryList().size() == 1 &&
918 "The root was expected to be an external symbol");
932 const auto NewThreshold =
933 Threshold * GetBonusMultiplier(
Edge.second.getHotness());
935 auto IT = ImportThresholds.insert(std::make_pair(
936 VI.getGUID(), std::make_tuple(NewThreshold,
nullptr,
nullptr)));
937 bool PreviouslyVisited = !
IT.second;
938 auto &ProcessedThreshold = std::get<0>(
IT.first->second);
939 auto &CalleeSummary = std::get<1>(
IT.first->second);
940 auto &FailureInfo = std::get<2>(
IT.first->second);
944 bool IsCriticalCallsite =
947 const FunctionSummary *ResolvedCalleeSummary =
nullptr;
949 assert(PreviouslyVisited);
954 if (NewThreshold <= ProcessedThreshold) {
956 dbgs() <<
"ignored! Target was already imported with Threshold "
957 << ProcessedThreshold <<
"\n");
961 ProcessedThreshold = NewThreshold;
966 if (PreviouslyVisited && NewThreshold <= ProcessedThreshold) {
968 dbgs() <<
"ignored! Target was already rejected with Threshold "
969 << ProcessedThreshold <<
"\n");
972 "Expected FailureInfo for previously rejected candidate");
973 FailureInfo->Attempts++;
981 const GlobalValueSummary *SummaryForDeclImport =
nullptr;
984 Summary.modulePath(), SummaryForDeclImport, Reason);
985 if (!CalleeSummary) {
989 StringRef DeclSourceModule = SummaryForDeclImport->
modulePath();
998 if (PreviouslyVisited) {
999 ProcessedThreshold = NewThreshold;
1002 "Expected FailureInfo for previously rejected candidate");
1003 FailureInfo->Reason = Reason;
1004 FailureInfo->Attempts++;
1005 FailureInfo->MaxHotness =
1006 std::max(FailureInfo->MaxHotness,
Edge.second.getHotness());
1010 "Expected no FailureInfo for newly rejected candidate");
1011 FailureInfo = std::make_unique<FunctionImporter::ImportFailureInfo>(
1012 VI,
Edge.second.getHotness(), Reason, 1);
1015 std::string Msg = std::string(
"Failed to import function ") +
1016 VI.name().str() +
" due to " +
1021 "Error importing module: ");
1025 <<
"ignored! No qualifying callee with summary found.\n");
1031 CalleeSummary = CalleeSummary->getBaseObject();
1035 (ResolvedCalleeSummary->
instCount() <= NewThreshold)) &&
1036 "selectCallee() didn't honor the threshold");
1038 auto ExportModulePath = ResolvedCalleeSummary->
modulePath();
1044 NumImportedFunctionsThinLink++;
1046 NumImportedHotFunctionsThinLink++;
1047 if (IsCriticalCallsite)
1048 NumImportedCriticalFunctionsThinLink++;
1055 (*ExportLists)[ExportModulePath].insert(VI);
1058 auto GetAdjustedThreshold = [](
unsigned Threshold,
bool IsHotCallsite) {
1067 const auto AdjThreshold = GetAdjustedThreshold(Threshold, IsHotCallsite);
1072 Worklist.
emplace_back(ResolvedCalleeSummary, AdjThreshold);
1088 for (
const auto &GVSummary : DefinedGVSummaries) {
1092 auto VI =
Index.getValueInfo(GVSummary.first);
1094 if (!
Index.isGlobalValueLive(GVSummary.second)) {
1104 computeImportForFunction(*FuncSummary,
ImportInstrLimit, DefinedGVSummaries,
1105 Worklist, GVI, ImportList, ImportThresholds);
1109 while (!Worklist.
empty()) {
1111 auto *Summary = std::get<0>(GVInfo);
1112 auto Threshold = std::get<1>(GVInfo);
1114 computeImportForFunction(*Summary, Threshold, DefinedGVSummaries, Worklist,
1115 GVI, ImportList, ImportThresholds);
1121 dbgs() <<
"Missed imports into module " << ModName <<
"\n";
1122 for (
auto &
I : ImportThresholds) {
1123 auto &ProcessedThreshold = std::get<0>(
I.second);
1124 auto &CalleeSummary = std::get<1>(
I.second);
1125 auto &FailureInfo = std::get<2>(
I.second);
1130 if (!FailureInfo->VI.getSummaryList().empty())
1132 FailureInfo->VI.getSummaryList()[0]->getBaseObject());
1133 dbgs() << FailureInfo->VI
1135 <<
", Threshold = " << ProcessedThreshold
1136 <<
", Size = " << (FS ? (int)FS->instCount() : -1)
1138 <<
", Attempts = " << FailureInfo->Attempts <<
"\n";
1145 auto SL = VI.getSummaryList();
1153 if (
const auto &VI = Index.getValueInfo(
G))
1162 unsigned NumGVS = 0;
1163 for (
auto &VI : ExportSet)
1181 for (
const auto &[FromModule, GUID,
Type] : ImportList) {
1200 for (
const auto &ImportPerModule : ImportLists)
1201 for (
const auto &[FromModule, GUID, ImportType] : ImportPerModule.second)
1202 FlattenedImports.
insert(GUID);
1210 auto IsReadOrWriteOnlyVarNeedingImporting = [&](
StringRef ModulePath,
1213 Index.findSummaryInModule(VI, ModulePath));
1214 return GVS && (Index.isReadOnly(GVS) || Index.isWriteOnly(GVS)) &&
1220 for (
auto &ExportPerModule : ExportLists)
1221 for (
auto &VI : ExportPerModule.second)
1222 if (!FlattenedImports.
count(VI.getGUID()) &&
1223 IsReadOrWriteOnlyVarNeedingImporting(ExportPerModule.first, VI))
1240 for (
const auto &DefinedGVSummaries : ModuleToDefinedGVSummaries) {
1241 auto &ImportList = ImportLists[DefinedGVSummaries.first];
1243 << DefinedGVSummaries.first <<
"'\n");
1244 MIS->computeImportForModule(DefinedGVSummaries.second,
1245 DefinedGVSummaries.first, ImportList);
1253 for (
auto &ELI : ExportLists) {
1257 const auto &DefinedGVSummaries =
1258 ModuleToDefinedGVSummaries.
lookup(ELI.first);
1259 for (
auto &EI : ELI.second) {
1266 auto DS = DefinedGVSummaries.
find(EI.getGUID());
1270 auto *S = DS->getSecond();
1271 S = S->getBaseObject();
1277 if (!Index.isWriteOnly(GVS))
1290 [&](
ValueInfo VI) {
return !DefinedGVSummaries.
count(VI.getGUID()); });
1291 ELI.second.insert_range(NewExports);
1298 for (
const auto &ModuleImports : ImportLists) {
1299 auto ModName = ModuleImports.first;
1300 auto &Exports = ExportLists[ModName];
1305 << Exports.
size() - NumGVS <<
" functions and " << NumGVS
1306 <<
" vars. Imports from " << Histogram.
size()
1308 for (
const auto &[SrcModName,
Stats] : Histogram) {
1310 <<
" function definitions and "
1312 <<
" function declarations imported from " << SrcModName
1315 <<
" global vars imported from " << SrcModName <<
"\n");
1327 LLVM_DEBUG(
dbgs() <<
"* Module " << ModulePath <<
" imports from "
1328 << Histogram.
size() <<
" modules.\n");
1329 for (
const auto &[SrcModName,
Stats] : Histogram) {
1331 <<
" function definitions and "
1333 <<
" function declarations imported from " << SrcModName
1336 << SrcModName <<
"\n");
1358 Index.collectDefinedFunctionsForModule(ModulePath, FunctionSummaryMap);
1361 LLVM_DEBUG(
dbgs() <<
"Computing import for Module '" << ModulePath <<
"'\n");
1363 MIS->computeImportForModule(FunctionSummaryMap, ModulePath, ImportList);
1378 for (
const auto &GlobalList : Index) {
1380 if (GlobalList.second.getSummaryList().empty())
1383 auto GUID = GlobalList.first;
1384 assert(GlobalList.second.getSummaryList().size() == 1 &&
1385 "Expected individual combined index to have one summary per GUID");
1386 auto &Summary = GlobalList.second.getSummaryList()[0];
1389 if (Summary->modulePath() == ModulePath)
1392 ImportList.
addGUID(Summary->modulePath(), GUID, Summary->importType());
1405 for (
auto &EI : FS->mutableCalls()) {
1406 if (!EI.first.getSummaryList().empty())
1408 auto GUID = Index.getGUIDFromOriginalID(EI.first.getGUID());
1412 auto VI = Index.getValueInfo(GUID);
1414 VI.getSummaryList(),
1415 [&](
const std::unique_ptr<GlobalValueSummary> &SummaryPtr) {
1425 return SummaryPtr->getSummaryKind() ==
1426 GlobalValueSummary::GlobalVarKind;
1434 for (
const auto &Entry : Index) {
1435 for (
const auto &S : Entry.second.getSummaryList()) {
1446 assert(!Index.withGlobalValueDeadStripping());
1449 GUIDPreservedSymbols.
empty()) {
1454 unsigned LiveSymbols = 0;
1456 Worklist.
reserve(GUIDPreservedSymbols.
size() * 2);
1457 for (
auto GUID : GUIDPreservedSymbols) {
1458 ValueInfo VI = Index.getValueInfo(GUID);
1461 for (
const auto &S : VI.getSummaryList())
1466 for (
const auto &Entry : Index) {
1467 auto VI = Index.getValueInfo(Entry);
1468 for (
const auto &S : Entry.second.getSummaryList()) {
1492 [](
const std::unique_ptr<llvm::GlobalValueSummary> &S) {
1503 bool KeepAliveLinkage =
false;
1504 bool Interposable =
false;
1505 for (
const auto &S : VI.getSummaryList()) {
1509 KeepAliveLinkage =
true;
1511 Interposable =
true;
1515 if (!KeepAliveLinkage)
1520 "Interposable and available_externally/linkonce_odr/weak_odr "
1525 for (
const auto &S : VI.getSummaryList())
1531 while (!Worklist.
empty()) {
1533 for (
const auto &Summary : VI.getSummaryList()) {
1538 visit(AS->getAliaseeVI(),
true);
1541 for (
auto Ref : Summary->refs())
1544 for (
auto Call : FS->calls())
1548 Index.setWithGlobalValueDeadStripping();
1550 unsigned DeadSymbols = Index.size() - LiveSymbols;
1551 LLVM_DEBUG(
dbgs() << LiveSymbols <<
" symbols Live, and " << DeadSymbols
1552 <<
" symbols Dead \n");
1553 NumDeadSymbols += DeadSymbols;
1554 NumLiveSymbols += LiveSymbols;
1562 bool ImportEnabled) {
1567 Index.propagateAttributes(GUIDPreservedSymbols);
1579 ModuleToSummariesForIndex[std::string(ModulePath)] =
1580 ModuleToDefinedGVSummaries.
lookup(ModulePath);
1589 auto It = Map.find(
Key);
1590 if (It == Map.end())
1591 std::tie(It, std::ignore) =
1597 for (
const auto &[FromModule, GUID, ImportType] : ImportList) {
1598 auto &SummariesForIndex =
1599 LookupOrCreate(ModuleToSummariesForIndex, FromModule);
1601 const auto &DefinedGVSummaries = ModuleToDefinedGVSummaries.
at(FromModule);
1602 const auto &DS = DefinedGVSummaries.
find(GUID);
1603 assert(DS != DefinedGVSummaries.
end() &&
1604 "Expected a defined summary for imported global value");
1606 DecSummaries.
insert(DS->second);
1608 SummariesForIndex[GUID] = DS->second;
1624 for (
auto &[ModPath, SummariesForIndex] : ModuleToSummariesForIndex) {
1625 if (ModPath == ModulePath)
1627 auto It = ModuleToDefinedGVSummaries.
find(ModPath);
1628 if (It == ModuleToDefinedGVSummaries.
end())
1630 for (
const auto &[GUID, Summary] : It->second) {
1631 if (Summary->noRenameOnPromotion()) {
1632 DecSummaries.
insert(Summary);
1633 SummariesForIndex.try_emplace(GUID, Summary);
1650 [&](
StringRef M) { ImportsOS << M <<
"\n"; });
1660 for (
const auto &ILI : ModuleToSummariesForIndex)
1664 if (ILI.first != ModulePath)
1674 F->setComdat(
nullptr);
1676 V->setInitializer(
nullptr);
1679 V->setComdat(
nullptr);
1708 auto FinalizeInModule = [&](
GlobalValue &GV,
bool Propagate =
false) {
1710 const auto &GS = DefinedGlobals.
find(GV.
getGUID());
1711 if (GS == DefinedGlobals.
end())
1718 if (FS->fflags().ReadNone && !
F->doesNotAccessMemory())
1719 F->setDoesNotAccessMemory();
1721 if (FS->fflags().ReadOnly && !
F->onlyReadsMemory())
1722 F->setOnlyReadsMemory();
1724 if (FS->fflags().NoRecurse && !
F->doesNotRecurse())
1725 F->setDoesNotRecurse();
1727 if (FS->fflags().NoUnwind && !
F->doesNotThrow())
1728 F->setDoesNotThrow();
1732 auto NewLinkage = GS->second->linkage();
1771 GS->second->canAutoHide()) {
1777 <<
"` from " << GV.
getLinkage() <<
" to " << NewLinkage
1785 if (GO && GO->isDeclarationForLinker() && GO->hasComdat()) {
1786 if (GO->getComdat()->getName() == GO->getName())
1787 NonPrevailingComdats.
insert(GO->getComdat());
1788 GO->setComdat(
nullptr);
1793 for (
auto &GV : TheModule)
1795 for (
auto &GV : TheModule.
globals())
1796 FinalizeInModule(GV);
1797 for (
auto &GV : TheModule.
aliases())
1798 FinalizeInModule(GV);
1803 if (NonPrevailingComdats.
empty())
1806 if (
auto *
C = GO.getComdat();
C && NonPrevailingComdats.
count(
C)) {
1807 GO.setComdat(
nullptr);
1818 for (
auto &GA : TheModule.
aliases()) {
1819 if (GA.hasAvailableExternallyLinkage())
1822 assert(Obj &&
"aliasee without an base object is unimplemented");
1823 if (Obj->hasAvailableExternallyLinkage()) {
1837 auto MustPreserveGV = [&](
const GlobalValue &GV) ->
bool {
1847 auto GS = DefinedGlobals.
find(GV.getGUID());
1848 if (GS == DefinedGlobals.
end()) {
1859 GS = DefinedGlobals.
find(
1861 if (GS == DefinedGlobals.
end()) {
1868 GS = DefinedGlobals.
find(
1899 for (
auto &GV : M.globals())
1902 if (!GV.isDeclaration() && GV.hasAttribute(
"thinlto-internalize")) {
1916 if (!
F.hasAvailableExternallyLinkage() ||
1917 !
F.hasMetadata(LLVMContext::MD_implicit_ref))
1920 F.getMetadata(LLVMContext::MD_implicit_ref, MDs);
1922 auto *
Op = MD->getOperand(0).get();
1927 if (GV->hasAvailableExternallyLinkage() && Seen.
insert(GV).second)
1931 if (!ToProtect.
empty())
1941 unsigned ImportedCount = 0, ImportedGVCount = 0;
1948 for (
auto &
F : DestModule)
1949 if (!
F.isDeclaration() && MoveSymbolGUIDSet.
contains(
F.getGUID()))
1959 if (!SrcModuleOrErr)
1961 std::unique_ptr<Module> SrcModule = std::move(*SrcModuleOrErr);
1963 "Context mismatch");
1967 if (
Error Err = SrcModule->materializeMetadata())
1968 return std::move(Err);
1977 auto GUID =
F.getGUID();
1978 auto MaybeImportType = ImportList.
getImportType(ModName, GUID);
1979 bool ImportDefinition =
1983 <<
" importing function"
1984 << (ImportDefinition
1986 : (MaybeImportType ?
" declaration " :
" "))
1987 << GUID <<
" " <<
F.getName() <<
" from "
1988 << SrcModule->getSourceFileName() <<
"\n");
1989 if (ImportDefinition) {
1990 if (
Error Err =
F.materialize())
1991 return std::move(Err);
1998 "thinlto_src_module",
2000 {MDString::get(DestModule.getContext(),
2001 SrcModule->getModuleIdentifier())}));
2005 {MDString::get(DestModule.getContext(),
2006 SrcModule->getSourceFileName())}));
2017 auto GUID = GV.getGUID();
2018 auto MaybeImportType = ImportList.
getImportType(ModName, GUID);
2019 bool ImportDefinition =
2023 <<
" importing global"
2024 << (ImportDefinition
2026 : (MaybeImportType ?
" declaration " :
" "))
2027 << GUID <<
" " << GV.getName() <<
" from "
2028 << SrcModule->getSourceFileName() <<
"\n");
2029 if (ImportDefinition) {
2030 if (
Error Err = GV.materialize())
2031 return std::move(Err);
2032 ImportedGVCount += GlobalsToImport.
insert(&GV);
2041 auto GUID = GA.getGUID();
2042 auto MaybeImportType = ImportList.
getImportType(ModName, GUID);
2043 bool ImportDefinition =
2047 <<
" importing alias"
2048 << (ImportDefinition
2050 : (MaybeImportType ?
" declaration " :
" "))
2051 << GUID <<
" " << GA.getName() <<
" from "
2052 << SrcModule->getSourceFileName() <<
"\n");
2053 if (ImportDefinition) {
2054 if (
Error Err = GA.materialize())
2055 return std::move(Err);
2059 return std::move(Err);
2062 <<
" " << GO->
getName() <<
" from "
2063 << SrcModule->getSourceFileName() <<
"\n");
2068 "thinlto_src_module",
2070 {MDString::get(DestModule.getContext(),
2071 SrcModule->getModuleIdentifier())}));
2075 {MDString::get(DestModule.getContext(),
2076 SrcModule->getSourceFileName())}));
2078 GlobalsToImport.
insert(Fn);
2091 SrcModule->setPartialSampleProfileRatio(Index);
2098 for (
const auto *GV : GlobalsToImport)
2100 <<
" from " << SrcModule->getSourceFileName() <<
"\n";
2103 if (
Error Err = Mover.
move(std::move(SrcModule),
2107 Twine(
"Function Import: link error: ") +
2110 ImportedCount += GlobalsToImport.
size();
2111 NumImportedModules++;
2118 if (ImportedGVCount > 0)
2121 NumImportedFunctions += (ImportedCount - ImportedGVCount);
2122 NumImportedGlobalVars += ImportedGVCount;
2125 LLVM_DEBUG(
dbgs() <<
"Imported " << ImportedCount - ImportedGVCount
2126 <<
" functions for Module "
2129 <<
" global variables for Module "
2131 return ImportedCount;
2141 if (!IndexPtrOrErr) {
2146 std::unique_ptr<ModuleSummaryIndex> Index = std::move(*IndexPtrOrErr);
2156 *Index, ImportList);
2159 isPrevailing, *Index, ImportList);
2165 for (
auto &
I : *Index) {
2166 for (
auto &S :
I.second.getSummaryList()) {
2168 S->setExternalLinkageForTest();
2178 auto ModuleLoader = [&M](
StringRef Identifier) {
2179 return loadFile(std::string(Identifier), M.getContext());
2188 "Error importing module: ");
assert(UImm &&(UImm !=~static_cast< T >(0)) &&"Invalid immediate!")
static cl::opt< ITMode > IT(cl::desc("IT block support"), cl::Hidden, cl::init(DefaultIT), cl::values(clEnumValN(DefaultIT, "arm-default-it", "Generate any type of IT block"), clEnumValN(RestrictedIT, "arm-restrict-it", "Disallow complex IT blocks")))
static auto qualifyCalleeCandidates(const ModuleSummaryIndex &Index, ArrayRef< std::unique_ptr< GlobalValueSummary > > CalleeSummaryList, StringRef CallerModulePath)
Given a list of possible callee implementation for a call site, qualify the legality of importing eac...
static DenseMap< StringRef, ImportStatistics > collectImportStatistics(const ModuleSummaryIndex &Index, const FunctionImporter::ImportMapTy &ImportList)
static unsigned numGlobalVarSummaries(const ModuleSummaryIndex &Index, FunctionImporter::ExportSetTy &ExportSet)
static bool checkVariableImport(const ModuleSummaryIndex &Index, FunctionImporter::ImportListsTy &ImportLists, DenseMap< StringRef, FunctionImporter::ExportSetTy > &ExportLists)
static bool doImportingForModuleForTest(Module &M, function_ref< bool(GlobalValue::GUID, const GlobalValueSummary *)> isPrevailing)
static const char * getFailureName(FunctionImporter::ImportFailureReason Reason)
static void internalizeGVsAfterImport(Module &M)
void updateValueInfoForIndirectCalls(ModuleSummaryIndex &Index, FunctionSummary *FS)
static std::unique_ptr< Module > loadFile(const std::string &FileName, LLVMContext &Context)
static bool shouldSkipLocalInAnotherModule(const GlobalValueSummary *RefSummary, size_t NumDefs, StringRef ImporterModule)
static bool isGlobalVarSummary(const ModuleSummaryIndex &Index, ValueInfo VI)
static Function * replaceAliasWithAliasee(Module *SrcModule, GlobalAlias *GA)
Make alias a clone of its aliasee.
static void protectImplicitRefGlobals(Module &M)
When a function carrying !implicit.ref is imported via ThinLTO, the referenced global arrives as avai...
static void dumpImportListForModule(const ModuleSummaryIndex &Index, StringRef ModulePath, FunctionImporter::ImportMapTy &ImportList)
static void ComputeCrossModuleImportForModuleFromIndexForTest(StringRef ModulePath, const ModuleSummaryIndex &Index, FunctionImporter::ImportMapTy &ImportList)
Mark all external summaries in Index for import into the given module.
static const GlobalValueSummary * selectCallee(const ModuleSummaryIndex &Index, ArrayRef< std::unique_ptr< GlobalValueSummary > > CalleeSummaryList, unsigned Threshold, StringRef CallerModulePath, const GlobalValueSummary *&TooLargeOrNoInlineSummary, FunctionImporter::ImportFailureReason &Reason)
Given a list of possible callee implementation for a call site, select one that fits the Threshold fo...
static void ComputeCrossModuleImportForModuleForTest(StringRef ModulePath, function_ref< bool(GlobalValue::GUID, const GlobalValueSummary *)> isPrevailing, const ModuleSummaryIndex &Index, FunctionImporter::ImportMapTy &ImportList)
Compute all the imports for the given module using the Index.
Module.h This file contains the declarations for the Module class.
This file supports working with JSON data.
block placement Basic Block Placement Stats
static cl::opt< std::string > OutputFilename("o", cl::desc("Output filename"), cl::value_desc("filename"), cl::init("-"))
static cl::opt< bool > PropagateAttrs("propagate-attrs", cl::init(true), cl::Hidden, cl::desc("Propagate attributes in index"))
ModuleSummaryIndex.h This file contains the declarations the classes that hold the module index and s...
static constexpr StringLiteral Filename
Reader for contextual iFDO profile, which comes in bitstream format.
static void visit(BasicBlock &Start, std::function< bool(BasicBlock *)> op)
std::pair< BasicBlock *, BasicBlock * > Edge
This file implements a set that has insertion order iteration characteristics.
This file defines the SmallVector class.
This file defines the 'Statistic' class, which is designed to be an easy way to expose various metric...
#define STATISTIC(VARNAME, DESC)
Import globals referenced by a function or other globals that are being imported, if importing such g...
void onImportingSummary(const GlobalValueSummary &Summary)
GlobalsImporter(const ModuleSummaryIndex &Index, const GVSummaryMapTy &DefinedGVSummaries, function_ref< bool(GlobalValue::GUID, const GlobalValueSummary *)> IsPrevailing, FunctionImporter::ImportMapTy &ImportList, DenseMap< StringRef, FunctionImporter::ExportSetTy > *ExportLists)
virtual bool canImport(ValueInfo VI)
DenseMap< StringRef, FunctionImporter::ExportSetTy > *const ExportLists
virtual ~ModuleImportsManager()=default
static std::unique_ptr< ModuleImportsManager > create(function_ref< bool(GlobalValue::GUID, const GlobalValueSummary *)> IsPrevailing, const ModuleSummaryIndex &Index, DenseMap< StringRef, FunctionImporter::ExportSetTy > *ExportLists=nullptr)
function_ref< bool(GlobalValue::GUID, const GlobalValueSummary *)> IsPrevailing
ModuleImportsManager(function_ref< bool(GlobalValue::GUID, const GlobalValueSummary *)> IsPrevailing, const ModuleSummaryIndex &Index, DenseMap< StringRef, FunctionImporter::ExportSetTy > *ExportLists=nullptr)
const ModuleSummaryIndex & Index
virtual void computeImportForModule(const GVSummaryMapTy &DefinedGVSummaries, StringRef ModName, FunctionImporter::ImportMapTy &ImportList)
Given the list of globals defined in a module, compute the list of imports as well as the list of "ex...
WorkloadImportsManager(function_ref< bool(GlobalValue::GUID, const GlobalValueSummary *)> IsPrevailing, const ModuleSummaryIndex &Index, DenseMap< StringRef, FunctionImporter::ExportSetTy > *ExportLists)
Represent a constant reference to an array (0 or more elements consecutively in memory),...
ValueT & at(const_arg_type_t< KeyT > Val)
Return the entry for the specified key, or abort if no such entry exists.
ValueT lookup(const_arg_type_t< KeyT > Val) const
Return the entry for the specified key, or a default constructed value if no such entry exists.
iterator find(const_arg_type_t< KeyT > Val)
size_type count(const_arg_type_t< KeyT > Val) const
Return 1 if the specified key is in the map, 0 otherwise.
Implements a dense probed hash-table based set.
Lightweight error class with error context and mandatory checking.
static ErrorSuccess success()
Create a success value.
Tagged union holding either a T or a Error.
Error takeError()
Take ownership of the stored error.
LLVM_ABI PreservedAnalyses run(Module &M, ModuleAnalysisManager &AM)
The map maintains the list of imports.
LLVM_ABI AddDefinitionStatus addDefinition(StringRef FromModule, GlobalValue::GUID GUID)
void addGUID(StringRef FromModule, GlobalValue::GUID GUID, GlobalValueSummary::ImportKind ImportKind)
LLVM_ABI SmallVector< StringRef, 0 > getSourceModules() const
LLVM_ABI std::optional< GlobalValueSummary::ImportKind > getImportType(StringRef FromModule, GlobalValue::GUID GUID) const
LLVM_ABI void maybeAddDeclaration(StringRef FromModule, GlobalValue::GUID GUID)
The function importer is automatically importing function from other modules based on the provided su...
LLVM_ABI Expected< bool > importFunctions(Module &M, const ImportMapTy &ImportList)
Import functions in Module M based on the supplied import list.
DenseMap< GlobalValue::GUID, std::tuple< unsigned, const GlobalValueSummary *, std::unique_ptr< ImportFailureInfo > > > ImportThresholdsTy
Map of callee GUID considered for import into a given module to a pair consisting of the largest thre...
ImportFailureReason
The different reasons selectCallee will chose not to import a candidate.
@ LocalLinkageNotInModule
DenseSet< ValueInfo > ExportSetTy
The set contains an entry for every global value that the module exports.
Function summary information to aid decisions and implementation of importing.
unsigned instCount() const
Get the instruction count recorded for this function.
FFlags fflags() const
Get function summary flags.
static Function * Create(FunctionType *Ty, LinkageTypes Linkage, unsigned AddrSpace, const Twine &N="", Module *M=nullptr)
LLVM_ABI const GlobalObject * getAliaseeObject() const
Function and variable summary information to aid decisions and implementation of importing.
StringRef modulePath() const
Get the path to the module containing this function.
GlobalValue::LinkageTypes linkage() const
Return linkage type recorded for this global value.
static LLVM_ABI GUID getGUIDAssumingExternalLinkage(StringRef GlobalName)
Return a 64-bit global unique ID constructed from the name of a global symbol.
VisibilityTypes getVisibility() const
bool isImplicitDSOLocal() const
static bool isLocalLinkage(LinkageTypes Linkage)
LLVM_ABI bool isDeclaration() const
Return true if the primary definition of this global value is outside of the current translation unit...
LinkageTypes getLinkage() const
uint64_t GUID
Declare a type to represent a global unique identifier for a global value.
ThreadLocalMode getThreadLocalMode() const
static bool isAvailableExternallyLinkage(LinkageTypes Linkage)
void setLinkage(LinkageTypes LT)
unsigned getAddressSpace() const
GUID getGUID() const
Return a 64-bit global unique ID constructed from global value name (i.e.
Module * getParent()
Get the module that this global value is contained inside of...
LLVM_ABI const GlobalObject * getAliaseeObject() const
void setDSOLocal(bool Local)
PointerType * getType() const
Global values are always pointers.
@ DefaultVisibility
The GV is visible.
@ HiddenVisibility
The GV is hidden.
static LLVM_ABI std::string getGlobalIdentifier(StringRef Name, GlobalValue::LinkageTypes Linkage, StringRef FileName)
Return the modified name for a global value suitable to be used as the key for a global lookup (e....
void setVisibility(VisibilityTypes V)
static bool isInterposableLinkage(LinkageTypes Linkage)
Whether the definition of this global may be replaced by something non-equivalent at link time.
LLVM_ABI Error materialize()
Make sure this GlobalValue is fully read.
LLVM_ABI bool canBeOmittedFromSymbolTable() const
True if GV can be left out of the object symbol table.
@ InternalLinkage
Rename collisions when linking (static functions).
@ WeakODRLinkage
Same, but only replaced by something equivalent.
@ ExternalLinkage
Externally visible function.
@ AvailableExternallyLinkage
Available for inspection, not emission.
@ LinkOnceODRLinkage
Same, but only replaced by something equivalent.
Type * getValueType() const
LLVM_ABI Error move(std::unique_ptr< Module > Src, ArrayRef< GlobalValue * > ValuesToLink, LazyCallback AddLazyFor, bool IsPerformingImport)
Move in the provide values in ValuesToLink from Src.
This is an important class for using LLVM in a threaded context.
static MDTuple * get(LLVMContext &Context, ArrayRef< Metadata * > MDs)
static ErrorOr< std::unique_ptr< MemoryBuffer > > getFileOrSTDIN(const Twine &Filename, bool IsText=false, bool RequiresNullTerminator=true, std::optional< Align > Alignment=std::nullopt)
Open the specified file as a MemoryBuffer, or open stdin if the Filename is "-".
Class to hold module path string table and global value map, and encapsulate methods for operating on...
static StringRef getOriginalNameBeforePromote(StringRef Name)
Helper to obtain the unpromoted name for a global value (or the original name if not promoted).
A Module instance is used to store all the information related to an LLVM module.
LLVMContext & getContext() const
Get the global data context.
const std::string & getSourceFileName() const
Get the module's original source file name.
iterator_range< alias_iterator > aliases()
iterator_range< global_iterator > globals()
const std::string & getModuleIdentifier() const
Get the module identifier which is, essentially, the name of the module.
iterator_range< global_object_iterator > global_objects()
LLVM_ABI Expected< PGOCtxProfile > loadProfiles()
unsigned getAddressSpace() const
Return the address space of the Pointer type.
A set of analyses that are preserved following a run of a transformation pass.
static PreservedAnalyses none()
Convenience factory function for the empty preserved set.
static PreservedAnalyses all()
Construct a special preserved set that preserves all passes.
Instances of this class encapsulate one diagnostic report, allowing printing to a raw_ostream as a ca...
A vector that has set insertion semantics.
ArrayRef< value_type > getArrayRef() const
size_type size() const
Determine the number of elements in the SetVector.
Vector takeVector()
Clear the SetVector and return the underlying vector.
bool contains(const_arg_type key) const
Check if the SetVector contains the given key.
void clear()
Completely clear the SetVector.
bool insert(const value_type &X)
Insert a new element into the SetVector.
std::pair< iterator, bool > insert(PtrType Ptr)
Inserts Ptr if and only if there is no element in the container equal to Ptr.
SmallPtrSet - This class implements a set which is optimized for holding SmallSize or less elements.
This class consists of common code factored out of the SmallVector class to reduce code duplication b...
reference emplace_back(ArgTypes &&... Args)
void reserve(size_type N)
void push_back(const T &Elt)
This is a 'vector' (really, a variable-sized array), optimized for the case when the array is small.
StringMap - This is an unconventional map that is specialized for handling keys that are "strings",...
iterator find(StringRef Key)
size_type count(StringRef Key) const
count - Return 1 if the element is in the map, 0 otherwise.
bool insert(MapEntryTy *KeyValue)
insert - Insert the specified key/value pair into the map.
Represent a constant reference to a string, i.e.
constexpr size_t size() const
Get the string size.
char front() const
Get the first character in the string.
size_t find(char C, size_t From=0) const
Search for the first character C in the string.
StringSet - A wrapper for StringMap that provides set-like functionality.
std::pair< typename Base::iterator, bool > insert(StringRef key)
The TimeTraceScope is a helper class to call the begin and end functions of the time trace profiler.
Twine - A lightweight data structure for efficiently representing the concatenation of temporary valu...
The instances of the Type class are immutable: once they are created, they are never changed.
bool isFunctionTy() const
True if this is an instance of FunctionType.
LLVM_ABI void replaceAllUsesWith(Value *V)
Change all uses of this to point to a new Value.
LLVM_ABI StringRef getName() const
Return a constant reference to the value's name.
LLVM_ABI void takeName(Value *V)
Transfer the name from V to this value.
std::pair< iterator, bool > insert(const ValueT &V)
void insert_range(Range &&R)
bool remove_if(Predicate Pred)
Remove all elements for which Pred returns true.
bool contains(const_arg_type_t< ValueT > V) const
Check if the set contains the given element.
size_type count(const_arg_type_t< ValueT > V) const
Return 1 if the specified key is in the set, 0 otherwise.
An efficient, type-erasing, non-owning reference to a callable.
The root is the trivial Path to the root value.
A raw_ostream that writes to a file descriptor.
#define llvm_unreachable(msg)
Marks that the current location is not supposed to be reachable.
@ C
The default llvm calling convention, compatible with C.
initializer< Ty > init(const Ty &Val)
LLVM_ABI llvm::Expected< Value > parse(llvm::StringRef JSON)
Parses the provided JSON source, or returns a ParseError.
bool fromJSON(const Value &E, std::string &Out, Path P)
@ OF_Text
The file should be opened in text mode on platforms like z/OS that make this distinction.
LLVM_ABI StringRef filename(StringRef path LLVM_LIFETIME_BOUND, Style style=Style::native)
Get filename.
This is an optimization pass for GlobalISel generic memory operations.
LLVM_ABI void logAllUnhandledErrors(Error E, raw_ostream &OS, Twine ErrorBanner={})
Log all errors (if any) in E to OS.
static cl::opt< float > ImportHotMultiplier("import-hot-multiplier", cl::init(10.0), cl::Hidden, cl::value_desc("x"), cl::desc("Multiply the `import-instr-limit` threshold for hot callsites"))
static cl::opt< bool > CtxprofMoveRootsToOwnModule("thinlto-move-ctxprof-trees", cl::desc("Move contextual profiling roots and the graphs under them in " "their own module."), cl::Hidden, cl::init(false))
Error createFileError(const Twine &F, Error E)
Concatenate a source file path and/or name with an Error.
bool internalizeModule(Module &TheModule, std::function< bool(const GlobalValue &)> MustPreserveGV)
Helper function to internalize functions and variables in a Module.
static cl::opt< float > ImportColdMultiplier("import-cold-multiplier", cl::init(0), cl::Hidden, cl::value_desc("N"), cl::desc("Multiply the `import-instr-limit` threshold for cold callsites"))
std::error_code make_error_code(BitcodeError E)
cl::list< GlobalValue::GUID > MoveSymbolGUID
const char * getHotnessName(CalleeInfo::HotnessType HT)
decltype(auto) dyn_cast(const From &Val)
dyn_cast<X> - Return the argument parameter cast to the specified type.
static cl::opt< std::string > WorkloadDefinitions("thinlto-workload-def", cl::desc("Pass a workload definition. This is a file containing a JSON " "dictionary. The keys are root functions, the values are lists of " "functions to import in the module defining the root. It is " "assumed -funique-internal-linkage-names was used, to ensure " "local linkage functions have unique names. For example: \n" "{\n" " \"rootFunction_1\": [\"function_to_import_1\", " "\"function_to_import_2\"], \n" " \"rootFunction_2\": [\"function_to_import_3\", " "\"function_to_import_4\"] \n" "}"), cl::Hidden)
Pass a workload description file - an example of workload would be the functions executed to satisfy ...
DenseMap< GlobalValue::GUID, GlobalValueSummary * > GVSummaryMapTy
Map of global value GUID to its summary, used to identify values defined in a particular module,...
cl::opt< std::string > UseCtxProfile("use-ctx-profile", cl::init(""), cl::Hidden, cl::desc("Use the specified contextual profile file"))
static cl::opt< bool > PrintImportFailures("print-import-failures", cl::init(false), cl::Hidden, cl::desc("Print information for functions rejected for importing"))
iterator_range< early_inc_iterator_impl< detail::IterOfRange< RangeT > > > make_early_inc_range(RangeT &&Range)
Make a range that does early increment to allow mutation of the underlying range without disrupting i...
LLVM_ABI bool convertToDeclaration(GlobalValue &GV)
Converts value GV to declaration, or replaces with a declaration if it is an alias.
static cl::opt< unsigned > ImportInstrLimit("import-instr-limit", cl::init(100), cl::Hidden, cl::value_desc("N"), cl::desc("Only import functions with less than N instructions"))
Limit on instruction count of imported functions.
LLVM_ABI void renameModuleForThinLTO(Module &M, const ModuleSummaryIndex &Index, bool ClearDSOLocalOnDeclarations, SetVector< GlobalValue * > *GlobalsToImport=nullptr)
Perform in-place global value handling on the given Module for exported local functions renamed and p...
Error createStringError(std::error_code EC, char const *Fmt, const Ts &... Vals)
Create formatted StringError object.
static cl::opt< float > ImportHotInstrFactor("import-hot-evolution-factor", cl::init(1.0), cl::Hidden, cl::value_desc("x"), cl::desc("As we import functions called from hot callsite, multiply the " "`import-instr-limit` threshold by this factor " "before processing newly imported functions"))
static cl::opt< bool > PrintImports("print-imports", cl::init(false), cl::Hidden, cl::desc("Print imported functions"))
auto map_range(ContainerTy &&C, FuncTy F)
Return a range that applies F to the elements of C.
LLVM_ABI void ComputeCrossModuleImport(const ModuleSummaryIndex &Index, const DenseMap< StringRef, GVSummaryMapTy > &ModuleToDefinedGVSummaries, function_ref< bool(GlobalValue::GUID, const GlobalValueSummary *)> isPrevailing, FunctionImporter::ImportListsTy &ImportLists, DenseMap< StringRef, FunctionImporter::ExportSetTy > &ExportLists)
Compute all the imports and exports for every module in the Index.
auto dyn_cast_or_null(const Y &Val)
static cl::opt< float > ImportInstrFactor("import-instr-evolution-factor", cl::init(0.7), cl::Hidden, cl::value_desc("x"), cl::desc("As we import functions, multiply the " "`import-instr-limit` threshold by this factor " "before processing newly imported functions"))
bool any_of(R &&range, UnaryPredicate P)
Provide wrappers to std::any_of which take ranges instead of having to pass begin/end explicitly.
void sort(IteratorTy Start, IteratorTy End)
cl::opt< bool > AlwaysRenamePromotedLocals("always-rename-promoted-locals", cl::init(true), cl::Hidden, cl::desc("Always rename promoted locals."))
LLVM_ABI void computeDeadSymbolsAndUpdateIndirectCalls(ModuleSummaryIndex &Index, const DenseSet< GlobalValue::GUID > &GUIDPreservedSymbols, function_ref< PrevailingType(GlobalValue::GUID)> isPrevailing)
Compute all the symbols that are "dead": i.e these that can't be reached in the graph from any of the...
LLVM_ABI raw_ostream & dbgs()
dbgs() - This returns a reference to a raw_ostream for debugging messages.
auto make_first_range(ContainerTy &&c)
Given a container of pairs, return a range over the first elements.
LLVM_ABI void report_fatal_error(Error Err, bool gen_crash_diag=true)
std::map< std::string, GVSummaryMapTy, std::less<> > ModuleToSummariesForIndexTy
Map of a module name to the GUIDs and summaries we will import from that module.
bool hasSingleElement(ContainerTy &&C)
Returns true if the given container only contains a single element.
iterator_range< filter_iterator< detail::IterOfRange< RangeT >, PredicateT > > make_filter_range(RangeT &&Range, PredicateT Pred)
Convenience function that takes a range of elements and a predicate, and return a new filter_iterator...
static cl::opt< bool > ImportAllIndex("import-all-index", cl::desc("Import all external functions in index."))
Used when testing importing from distributed indexes via opt.
static cl::opt< int > ImportCutoff("import-cutoff", cl::init(-1), cl::Hidden, cl::value_desc("N"), cl::desc("Only import first N functions if N>=0 (default -1)"))
static cl::opt< bool > ImportDeclaration("import-declaration", cl::init(false), cl::Hidden, cl::desc("If true, import function declaration as fallback if the function " "definition is not imported."))
This is a test-only option.
bool isa(const From &Val)
isa<X> - Return true if the parameter to the template is an instance of one of the template type argu...
LLVM_ATTRIBUTE_VISIBILITY_DEFAULT AnalysisKey InnerAnalysisManagerProxy< AnalysisManagerT, IRUnitT, ExtraArgTs... >::Key
Error make_error(ArgTs &&... Args)
Make a Error instance representing failure using the given error info type.
LLVM_ABI void updateIndirectCalls(ModuleSummaryIndex &Index)
Update call edges for indirect calls to local functions added from SamplePGO when needed.
LLVM_ABI raw_fd_ostream & errs()
This returns a reference to a raw_ostream for standard error.
@ Ref
The access may reference the value stored in memory.
cl::opt< bool > EnableMemProfContextDisambiguation
Enable MemProf context disambiguation for thin link.
LLVM_ABI void appendToCompilerUsed(Module &M, ArrayRef< GlobalValue * > Values)
Adds global values to the llvm.compiler.used list.
LLVM_ABI std::unique_ptr< Module > getLazyIRFileModule(StringRef Filename, SMDiagnostic &Err, LLVMContext &Context, bool ShouldLazyLoadMetadata=false)
If the given file holds a bitcode image, return a Module for it which does lazy deserialization of fu...
LLVM_ABI void thinLTOInternalizeModule(Module &TheModule, const GVSummaryMapTy &DefinedGlobals)
Internalize TheModule based on the information recorded in the summaries during global summary-based ...
cl::opt< bool > ForceImportAll
DWARFExpression::Operation Op
LLVM_ABI void gatherImportedSummariesForModule(StringRef ModulePath, const DenseMap< StringRef, GVSummaryMapTy > &ModuleToDefinedGVSummaries, const FunctionImporter::ImportMapTy &ImportList, ModuleToSummariesForIndexTy &ModuleToSummariesForIndex, GVSummaryPtrSet &DecSummaries)
Compute the set of summaries needed for a ThinLTO backend compilation of ModulePath.
static cl::opt< std::string > SummaryFile("summary-file", cl::desc("The summary file to use for function importing."))
Summary file to use for function importing when using -function-import from the command line.
std::string toString(const APInt &I, unsigned Radix, bool Signed, bool formatAsCLiteral=false, bool UpperCase=true, bool InsertSeparators=false)
ValueMap< const Value *, WeakTrackingVH > ValueToValueMapTy
LLVM_ABI void processImportsFiles(StringRef ModulePath, const ModuleToSummariesForIndexTy &ModuleToSummariesForIndex, function_ref< void(const std::string &)> F)
Call F passing each of the files module ModulePath will import from.
static cl::opt< float > ImportCriticalMultiplier("import-critical-multiplier", cl::init(100.0), cl::Hidden, cl::value_desc("x"), cl::desc("Multiply the `import-instr-limit` threshold for critical callsites"))
decltype(auto) cast(const From &Val)
cast<X> - Return the argument parameter cast to the specified type.
PrevailingType
PrevailingType enum used as a return type of callback passed to computeDeadSymbolsAndUpdateIndirectCa...
LLVM_ABI Error errorCodeToError(std::error_code EC)
Helper for converting an std::error_code to a Error.
LLVM_ABI bool UpgradeDebugInfo(Module &M)
Check the debug info version number, if it is out-dated, drop the debug info.
static cl::opt< bool > EnableImportMetadata("enable-import-metadata", cl::init(false), cl::Hidden, cl::desc("Enable import metadata like 'thinlto_src_module' and " "'thinlto_src_file'"))
SmallPtrSet< GlobalValueSummary *, 0 > GVSummaryPtrSet
A set of global value summary pointers.
LLVM_ABI Function * CloneFunction(Function *F, ValueToValueMapTy &VMap, ClonedCodeInfo *CodeInfo=nullptr)
Return a copy of the specified function and add it to that function's module.
LLVM_ABI void computeDeadSymbolsWithConstProp(ModuleSummaryIndex &Index, const DenseSet< GlobalValue::GUID > &GUIDPreservedSymbols, function_ref< PrevailingType(GlobalValue::GUID)> isPrevailing, bool ImportEnabled)
Compute dead symbols and run constant propagation in combined index after that.
LLVM_ABI Error EmitImportsFiles(StringRef ModulePath, StringRef OutputFilename, const ModuleToSummariesForIndexTy &ModuleToSummariesForIndex)
Emit into OutputFilename the files module ModulePath will import from.
static cl::opt< bool > ComputeDead("compute-dead", cl::init(true), cl::Hidden, cl::desc("Compute dead symbols"))
LLVM_ABI Expected< std::unique_ptr< ModuleSummaryIndex > > getModuleSummaryIndexForFile(StringRef Path, bool IgnoreEmptyThinLTOIndexFile=false)
Parse the module summary index out of an IR file and return the module summary index object if found,...
AnalysisManager< Module > ModuleAnalysisManager
Convenience typedef for the Module analysis manager.
LLVM_ABI void thinLTOFinalizeInModule(Module &TheModule, const GVSummaryMapTy &DefinedGlobals, bool PropagateAttrs)
Based on the information recorded in the summaries during global summary-based analysis:
Struct that holds a reference to a particular GUID in a global value summary.