53#include <system_error>
59#define DEBUG_TYPE "function-import"
62 "Number of functions thin link decided to import");
64 "Number of hot functions thin link decided to import");
66 "Number of critical functions thin link decided to import");
68 "Number of global variables thin link decided to import");
69STATISTIC(NumImportedFunctions,
"Number of functions imported in backend");
71 "Number of global variables imported in backend");
72STATISTIC(NumImportedModules,
"Number of modules imported from");
73STATISTIC(NumDeadSymbols,
"Number of dead stripped symbols in index");
74STATISTIC(NumLiveSymbols,
"Number of live symbols in index");
81 cl::desc(
"Import functions with noinline attribute"));
86 cl::desc(
"Only import functions with less than N instructions"));
90 cl::desc(
"Only import first N functions if N>=0 (default -1)"));
95 cl::desc(
"As we import functions, multiply the "
96 "`import-instr-limit` threshold by this factor "
97 "before processing newly imported functions"));
102 cl::desc(
"As we import functions called from hot callsite, multiply the "
103 "`import-instr-limit` threshold by this factor "
104 "before processing newly imported functions"));
108 cl::desc(
"Multiply the `import-instr-limit` threshold for hot callsites"));
114 "Multiply the `import-instr-limit` threshold for critical callsites"));
119 cl::desc(
"Multiply the `import-instr-limit` threshold for cold callsites"));
122 cl::desc(
"Print imported functions"));
126 cl::desc(
"Print information for functions rejected for importing"));
133 cl::desc(
"Enable import metadata like 'thinlto_src_module' and "
134 "'thinlto_src_file'"));
140 cl::desc(
"The summary file to use for function importing."));
146 cl::desc(
"Import all external functions in index."));
156 cl::desc(
"If true, import function declaration as fallback if the function "
157 "definition is not imported."));
168 "thinlto-workload-def",
169 cl::desc(
"Pass a workload definition. This is a file containing a JSON "
170 "dictionary. The keys are root functions, the values are lists of "
171 "functions to import in the module defining the root. It is "
172 "assumed -funique-internal-linkage-names was used, to ensure "
173 "local linkage functions have unique names. For example: \n"
175 " \"rootFunction_1\": [\"function_to_import_1\", "
176 "\"function_to_import_2\"], \n"
177 " \"rootFunction_2\": [\"function_to_import_3\", "
178 "\"function_to_import_4\"] \n"
185 "thinlto-move-ctxprof-trees",
186 cl::desc(
"Move contextual profiling roots and the graphs under them in "
187 "their own module."),
196static std::unique_ptr<Module>
loadFile(
const std::string &FileName,
202 std::unique_ptr<Module> Result =
206 Err.print(
"function-import",
errs());
235 ArrayRef<std::unique_ptr<GlobalValueSummary>> CalleeSummaryList,
239 [&Index, CalleeSummaryList,
240 CallerModulePath](
const std::unique_ptr<GlobalValueSummary> &SummaryPtr)
243 auto *GVSummary = SummaryPtr.get();
244 if (!Index.isGlobalValueLive(GVSummary))
281 if (Summary->notEligibleToImport())
306 ArrayRef<std::unique_ptr<GlobalValueSummary>> CalleeSummaryList,
307 unsigned Threshold,
StringRef CallerModulePath,
311 TooLargeOrNoInlineSummary =
nullptr;
312 auto QualifiedCandidates =
314 for (
auto QualifiedValue : QualifiedCandidates) {
315 Reason = QualifiedValue.first;
324 if ((Summary->instCount() > Threshold) && !Summary->fflags().AlwaysInline &&
326 TooLargeOrNoInlineSummary = Summary;
333 TooLargeOrNoInlineSummary = Summary;
352 auto [Def, Decl] = IDs.createImportIDs(FromModule, GUID);
353 if (!Imports.insert(Def).second)
365 auto [Def, Decl] = IDs.createImportIDs(FromModule, GUID);
368 if (!Imports.contains(Def))
369 Imports.insert(Decl);
375 for (
const auto &[SrcMod, GUID, ImportType] : *
this)
382std::optional<GlobalValueSummary::ImportKind>
385 if (
auto IDPair = IDs.getImportIDs(FromModule, GUID)) {
386 auto [Def, Decl] = *IDPair;
387 if (Imports.contains(Def))
389 if (Imports.contains(Decl))
405 bool shouldImportGlobal(
const ValueInfo &VI) {
406 const auto &GVS = DefinedGVSummaries.
find(VI.getGUID());
407 if (GVS == DefinedGVSummaries.end())
419 if (VI.getSummaryList().size() > 1 &&
421 !IsPrevailing(VI.getGUID(), GVS->second))
430 for (
const auto &VI : Summary.refs()) {
431 if (!shouldImportGlobal(VI)) {
433 dbgs() <<
"Ref ignored! Target already in destination module.\n");
439 for (
const auto &RefSummary : VI.getSummaryList()) {
457 bool CanImportDecl =
false;
459 Summary.modulePath()) ||
460 !Index.canImportGlobalVar(GVS,
true,
463 ImportList.maybeAddDeclaration(RefSummary->modulePath(),
471 if (ImportList.addDefinition(RefSummary->modulePath(), VI.getGUID()) !=
477 NumImportedGlobalVarsThinLink++;
482 (*ExportLists)[RefSummary->modulePath()].insert(VI);
486 if (!Index.isWriteOnly(GVS))
500 : Index(Index), DefinedGVSummaries(DefinedGVSummaries),
501 IsPrevailing(IsPrevailing), ImportList(ImportList),
502 ExportLists(ExportLists) {}
506 onImportingSummaryImpl(Summary, Worklist);
507 while (!Worklist.
empty())
508 onImportingSummaryImpl(*Worklist.
pop_back_val(), Worklist);
516 void computeImportForFunction(
548 static std::unique_ptr<ModuleImportsManager>
582 if (SetIter == Workloads.end()) {
584 <<
" does not contain the root of any context.\n");
586 ModName, ImportList);
589 <<
" contains the root(s) of context(s).\n");
593 auto &ValueInfos = SetIter->second;
595 auto It = DefinedGVSummaries.
find(VI.getGUID());
596 if (It != DefinedGVSummaries.
end() &&
599 dbgs() <<
"[Workload] " << VI.name()
600 <<
" has the prevailing variant already in the module "
601 << ModName <<
". No need to import\n");
611 [&](
const auto &Candidate) {
613 <<
" from " << Candidate.second->modulePath()
614 <<
" ImportFailureReason: "
616 return Candidate.first ==
619 [](
const auto &Candidate) {
return Candidate.second; });
620 if (PotentialCandidates.empty()) {
622 <<
" because can't find eligible Callee. Guid is: "
623 << VI.getGUID() <<
"\n");
641 PotentialCandidates, [&](
const auto *Candidate) {
644 if (PrevailingCandidates.empty()) {
645 GVS = *PotentialCandidates.begin();
650 <<
"[Workload] Found multiple non-prevailing candidates for "
652 <<
". This is unexpected. Are module paths passed to the "
653 "compiler unique for the modules passed to the linker?");
662 GVS = *PrevailingCandidates.begin();
669 if (ExportingModule == ModName) {
671 <<
" because its defining module is the same as the "
675 LLVM_DEBUG(
dbgs() <<
"[Workload][Including]" << VI.name() <<
" from "
676 << ExportingModule <<
" : " << VI.getGUID() <<
"\n");
680 (*ExportLists)[ExportingModule].insert(VI);
685 void loadFromJson() {
692 if (!NameToValueInfo.
insert(std::make_pair(VI.name(), VI)).second)
695 auto DbgReportIfAmbiguous = [&](
StringRef Name) {
697 dbgs() <<
"[Workload] Function name " << Name
698 <<
" present in the workload definition is ambiguous. Consider "
699 "compiling with -funique-internal-linkage-names.";
704 if (std::error_code EC = BufferOrErr.getError()) {
708 auto Buffer = std::move(BufferOrErr.get());
709 std::map<std::string, std::vector<std::string>> WorkloadDefs;
722 for (
const auto &Workload : WorkloadDefs) {
723 const auto &Root = Workload.first;
724 DbgReportIfAmbiguous(Root);
726 const auto &AllCallees = Workload.second;
727 auto RootIt = NameToValueInfo.
find(Root);
728 if (RootIt == NameToValueInfo.
end()) {
730 <<
" not found in this linkage unit.\n");
733 auto RootVI = RootIt->second;
734 if (RootVI.getSummaryList().size() != 1) {
736 <<
" should have exactly one summary, but has "
737 << RootVI.getSummaryList().size() <<
". Skipping.\n");
741 RootVI.getSummaryList().
front()->modulePath();
742 LLVM_DEBUG(
dbgs() <<
"[Workload] Root defining module for " << Root
743 <<
" is : " << RootDefiningModule <<
"\n");
744 auto &Set = Workloads[RootDefiningModule];
745 for (
const auto &Callee : AllCallees) {
747 DbgReportIfAmbiguous(Callee);
748 auto ElemIt = NameToValueInfo.
find(Callee);
749 if (ElemIt == NameToValueInfo.
end()) {
753 Set.insert(ElemIt->second);
758 void loadFromCtxProf() {
761 if (std::error_code EC = BufferOrErr.getError()) {
765 auto Buffer = std::move(BufferOrErr.get());
773 const auto &CtxMap = Ctx->Contexts;
775 for (
const auto &[RootGuid, Root] : CtxMap) {
778 ContainedGUIDs.
clear();
780 auto RootVI =
Index.getValueInfo(RootGuid);
783 <<
" not found in this linkage unit.\n");
786 if (RootVI.getSummaryList().size() != 1) {
788 <<
" should have exactly one summary, but has "
789 << RootVI.getSummaryList().size() <<
". Skipping.\n");
792 std::string RootDefiningModule =
793 RootVI.getSummaryList().front()->modulePath().str();
795 RootDefiningModule = std::to_string(RootGuid);
797 dbgs() <<
"[Workload] Moving " << RootGuid
798 <<
" to a module with the filename without extension : "
799 << RootDefiningModule <<
"\n");
801 LLVM_DEBUG(
dbgs() <<
"[Workload] Root defining module for " << RootGuid
802 <<
" is : " << RootDefiningModule <<
"\n");
804 auto &Set = Workloads[RootDefiningModule];
805 Root.getContainedGuids(ContainedGUIDs);
806 Roots.insert(RootVI);
807 for (
auto Guid : ContainedGUIDs)
824 "Pass only one of: -thinlto-pgo-ctx-prof or -thinlto-workload-def");
832 for (
const auto &[Root, Set] : Workloads) {
833 dbgs() <<
"[Workload] Root: " << Root <<
" we have " << Set.size()
834 <<
" distinct callees.\n";
835 for (
const auto &VI : Set) {
836 dbgs() <<
"[Workload] Root: " << Root
837 <<
" Would include: " << VI.getGUID() <<
"\n";
850 LLVM_DEBUG(
dbgs() <<
"[Workload] Using the regular imports manager.\n");
851 return std::unique_ptr<ModuleImportsManager>(
854 LLVM_DEBUG(
dbgs() <<
"[Workload] Using the contextual imports manager.\n");
871 return "InterposableLinkage";
873 return "LocalLinkageNotInModule";
875 return "NotEligible";
885void ModuleImportsManager::computeImportForFunction(
886 const FunctionSummary &Summary,
const unsigned Threshold,
888 SmallVectorImpl<EdgeInfo> &Worklist, GlobalsImporter &GVImporter,
889 FunctionImporter::ImportMapTy &ImportList,
892 static int ImportCount = 0;
894 ValueInfo
VI =
Edge.first;
895 LLVM_DEBUG(
dbgs() <<
" edge -> " << VI <<
" Threshold:" << Threshold
904 if (DefinedGVSummaries.
count(
VI.getGUID())) {
908 LLVM_DEBUG(
dbgs() <<
"ignored! Target already in destination module.\n");
912 if (!canImport(VI)) {
914 dbgs() <<
"Skipping over " <<
VI.getGUID()
915 <<
" because its import is handled in a different module.");
916 assert(
VI.getSummaryList().size() == 1 &&
917 "The root was expected to be an external symbol");
931 const auto NewThreshold =
932 Threshold * GetBonusMultiplier(
Edge.second.getHotness());
934 auto IT = ImportThresholds.insert(std::make_pair(
935 VI.getGUID(), std::make_tuple(NewThreshold,
nullptr,
nullptr)));
936 bool PreviouslyVisited = !
IT.second;
937 auto &ProcessedThreshold = std::get<0>(
IT.first->second);
938 auto &CalleeSummary = std::get<1>(
IT.first->second);
939 auto &FailureInfo = std::get<2>(
IT.first->second);
943 bool IsCriticalCallsite =
946 const FunctionSummary *ResolvedCalleeSummary =
nullptr;
948 assert(PreviouslyVisited);
953 if (NewThreshold <= ProcessedThreshold) {
955 dbgs() <<
"ignored! Target was already imported with Threshold "
956 << ProcessedThreshold <<
"\n");
960 ProcessedThreshold = NewThreshold;
965 if (PreviouslyVisited && NewThreshold <= ProcessedThreshold) {
967 dbgs() <<
"ignored! Target was already rejected with Threshold "
968 << ProcessedThreshold <<
"\n");
971 "Expected FailureInfo for previously rejected candidate");
972 FailureInfo->Attempts++;
980 const GlobalValueSummary *SummaryForDeclImport =
nullptr;
983 Summary.modulePath(), SummaryForDeclImport, Reason);
984 if (!CalleeSummary) {
988 StringRef DeclSourceModule = SummaryForDeclImport->
modulePath();
997 if (PreviouslyVisited) {
998 ProcessedThreshold = NewThreshold;
1001 "Expected FailureInfo for previously rejected candidate");
1002 FailureInfo->Reason = Reason;
1003 FailureInfo->Attempts++;
1004 FailureInfo->MaxHotness =
1005 std::max(FailureInfo->MaxHotness,
Edge.second.getHotness());
1009 "Expected no FailureInfo for newly rejected candidate");
1010 FailureInfo = std::make_unique<FunctionImporter::ImportFailureInfo>(
1011 VI,
Edge.second.getHotness(), Reason, 1);
1014 std::string Msg = std::string(
"Failed to import function ") +
1015 VI.name().str() +
" due to " +
1020 "Error importing module: ");
1024 <<
"ignored! No qualifying callee with summary found.\n");
1030 CalleeSummary = CalleeSummary->getBaseObject();
1034 (ResolvedCalleeSummary->
instCount() <= NewThreshold)) &&
1035 "selectCallee() didn't honor the threshold");
1037 auto ExportModulePath = ResolvedCalleeSummary->
modulePath();
1043 NumImportedFunctionsThinLink++;
1045 NumImportedHotFunctionsThinLink++;
1046 if (IsCriticalCallsite)
1047 NumImportedCriticalFunctionsThinLink++;
1054 (*ExportLists)[ExportModulePath].insert(VI);
1057 auto GetAdjustedThreshold = [](
unsigned Threshold,
bool IsHotCallsite) {
1066 const auto AdjThreshold = GetAdjustedThreshold(Threshold, IsHotCallsite);
1071 Worklist.
emplace_back(ResolvedCalleeSummary, AdjThreshold);
1087 for (
const auto &GVSummary : DefinedGVSummaries) {
1091 auto VI =
Index.getValueInfo(GVSummary.first);
1093 if (!
Index.isGlobalValueLive(GVSummary.second)) {
1103 computeImportForFunction(*FuncSummary,
ImportInstrLimit, DefinedGVSummaries,
1104 Worklist, GVI, ImportList, ImportThresholds);
1108 while (!Worklist.
empty()) {
1110 auto *Summary = std::get<0>(GVInfo);
1111 auto Threshold = std::get<1>(GVInfo);
1113 computeImportForFunction(*Summary, Threshold, DefinedGVSummaries, Worklist,
1114 GVI, ImportList, ImportThresholds);
1120 dbgs() <<
"Missed imports into module " << ModName <<
"\n";
1121 for (
auto &
I : ImportThresholds) {
1122 auto &ProcessedThreshold = std::get<0>(
I.second);
1123 auto &CalleeSummary = std::get<1>(
I.second);
1124 auto &FailureInfo = std::get<2>(
I.second);
1129 if (!FailureInfo->VI.getSummaryList().empty())
1131 FailureInfo->VI.getSummaryList()[0]->getBaseObject());
1132 dbgs() << FailureInfo->VI
1134 <<
", Threshold = " << ProcessedThreshold
1135 <<
", Size = " << (FS ? (int)FS->instCount() : -1)
1137 <<
", Attempts = " << FailureInfo->Attempts <<
"\n";
1144 auto SL = VI.getSummaryList();
1152 if (
const auto &VI = Index.getValueInfo(
G))
1161 unsigned NumGVS = 0;
1162 for (
auto &VI : ExportSet)
1180 for (
const auto &[FromModule, GUID,
Type] : ImportList) {
1199 for (
const auto &ImportPerModule : ImportLists)
1200 for (
const auto &[FromModule, GUID, ImportType] : ImportPerModule.second)
1201 FlattenedImports.
insert(GUID);
1209 auto IsReadOrWriteOnlyVarNeedingImporting = [&](
StringRef ModulePath,
1212 Index.findSummaryInModule(VI, ModulePath));
1213 return GVS && (Index.isReadOnly(GVS) || Index.isWriteOnly(GVS)) &&
1219 for (
auto &ExportPerModule : ExportLists)
1220 for (
auto &VI : ExportPerModule.second)
1221 if (!FlattenedImports.
count(VI.getGUID()) &&
1222 IsReadOrWriteOnlyVarNeedingImporting(ExportPerModule.first, VI))
1239 for (
const auto &DefinedGVSummaries : ModuleToDefinedGVSummaries) {
1240 auto &ImportList = ImportLists[DefinedGVSummaries.first];
1242 << DefinedGVSummaries.first <<
"'\n");
1243 MIS->computeImportForModule(DefinedGVSummaries.second,
1244 DefinedGVSummaries.first, ImportList);
1252 for (
auto &ELI : ExportLists) {
1256 const auto &DefinedGVSummaries =
1257 ModuleToDefinedGVSummaries.
lookup(ELI.first);
1258 for (
auto &EI : ELI.second) {
1265 auto DS = DefinedGVSummaries.
find(EI.getGUID());
1269 auto *S = DS->getSecond();
1270 S = S->getBaseObject();
1276 if (!Index.isWriteOnly(GVS))
1289 [&](
ValueInfo VI) {
return !DefinedGVSummaries.
count(VI.getGUID()); });
1290 ELI.second.insert_range(NewExports);
1297 for (
const auto &ModuleImports : ImportLists) {
1298 auto ModName = ModuleImports.first;
1299 auto &Exports = ExportLists[ModName];
1304 << Exports.
size() - NumGVS <<
" functions and " << NumGVS
1305 <<
" vars. Imports from " << Histogram.
size()
1307 for (
const auto &[SrcModName,
Stats] : Histogram) {
1309 <<
" function definitions and "
1311 <<
" function declarations imported from " << SrcModName
1314 <<
" global vars imported from " << SrcModName <<
"\n");
1326 LLVM_DEBUG(
dbgs() <<
"* Module " << ModulePath <<
" imports from "
1327 << Histogram.
size() <<
" modules.\n");
1328 for (
const auto &[SrcModName,
Stats] : Histogram) {
1330 <<
" function definitions and "
1332 <<
" function declarations imported from " << SrcModName
1335 << SrcModName <<
"\n");
1357 Index.collectDefinedFunctionsForModule(ModulePath, FunctionSummaryMap);
1360 LLVM_DEBUG(
dbgs() <<
"Computing import for Module '" << ModulePath <<
"'\n");
1362 MIS->computeImportForModule(FunctionSummaryMap, ModulePath, ImportList);
1377 for (
const auto &GlobalList : Index) {
1379 if (GlobalList.second.getSummaryList().empty())
1382 auto GUID = GlobalList.first;
1383 assert(GlobalList.second.getSummaryList().size() == 1 &&
1384 "Expected individual combined index to have one summary per GUID");
1385 auto &Summary = GlobalList.second.getSummaryList()[0];
1388 if (Summary->modulePath() == ModulePath)
1391 ImportList.
addGUID(Summary->modulePath(), GUID, Summary->importType());
1404 for (
auto &EI : FS->mutableCalls()) {
1405 if (!EI.first.getSummaryList().empty())
1407 auto GUID = Index.getGUIDFromOriginalID(EI.first.getGUID());
1411 auto VI = Index.getValueInfo(GUID);
1413 VI.getSummaryList(),
1414 [&](
const std::unique_ptr<GlobalValueSummary> &SummaryPtr) {
1424 return SummaryPtr->getSummaryKind() ==
1425 GlobalValueSummary::GlobalVarKind;
1433 for (
const auto &Entry : Index) {
1434 for (
const auto &S : Entry.second.getSummaryList()) {
1445 assert(!Index.withGlobalValueDeadStripping());
1448 GUIDPreservedSymbols.
empty()) {
1453 unsigned LiveSymbols = 0;
1455 Worklist.
reserve(GUIDPreservedSymbols.
size() * 2);
1456 for (
auto GUID : GUIDPreservedSymbols) {
1457 ValueInfo VI = Index.getValueInfo(GUID);
1460 for (
const auto &S : VI.getSummaryList())
1465 for (
const auto &Entry : Index) {
1466 auto VI = Index.getValueInfo(Entry);
1467 for (
const auto &S : Entry.second.getSummaryList()) {
1491 [](
const std::unique_ptr<llvm::GlobalValueSummary> &S) {
1502 bool KeepAliveLinkage =
false;
1503 bool Interposable =
false;
1504 for (
const auto &S : VI.getSummaryList()) {
1508 KeepAliveLinkage =
true;
1510 Interposable =
true;
1514 if (!KeepAliveLinkage)
1519 "Interposable and available_externally/linkonce_odr/weak_odr "
1524 for (
const auto &S : VI.getSummaryList())
1530 while (!Worklist.
empty()) {
1532 for (
const auto &Summary : VI.getSummaryList()) {
1537 visit(AS->getAliaseeVI(),
true);
1540 for (
auto Ref : Summary->refs())
1543 for (
auto Call : FS->calls())
1547 Index.setWithGlobalValueDeadStripping();
1549 unsigned DeadSymbols = Index.size() - LiveSymbols;
1550 LLVM_DEBUG(
dbgs() << LiveSymbols <<
" symbols Live, and " << DeadSymbols
1551 <<
" symbols Dead \n");
1552 NumDeadSymbols += DeadSymbols;
1553 NumLiveSymbols += LiveSymbols;
1561 bool ImportEnabled) {
1566 Index.propagateAttributes(GUIDPreservedSymbols);
1578 ModuleToSummariesForIndex[std::string(ModulePath)] =
1579 ModuleToDefinedGVSummaries.
lookup(ModulePath);
1588 auto It = Map.find(
Key);
1589 if (It == Map.end())
1590 std::tie(It, std::ignore) =
1596 for (
const auto &[FromModule, GUID, ImportType] : ImportList) {
1597 auto &SummariesForIndex =
1598 LookupOrCreate(ModuleToSummariesForIndex, FromModule);
1600 const auto &DefinedGVSummaries = ModuleToDefinedGVSummaries.
at(FromModule);
1601 const auto &DS = DefinedGVSummaries.
find(GUID);
1602 assert(DS != DefinedGVSummaries.
end() &&
1603 "Expected a defined summary for imported global value");
1605 DecSummaries.insert(DS->second);
1607 SummariesForIndex[GUID] = DS->second;
1623 for (
auto &[ModPath, SummariesForIndex] : ModuleToSummariesForIndex) {
1624 if (ModPath == ModulePath)
1626 auto It = ModuleToDefinedGVSummaries.
find(ModPath);
1627 if (It == ModuleToDefinedGVSummaries.
end())
1629 for (
const auto &[GUID, Summary] : It->second) {
1630 if (Summary->noRenameOnPromotion()) {
1631 DecSummaries.insert(Summary);
1632 SummariesForIndex.try_emplace(GUID, Summary);
1649 [&](
StringRef M) { ImportsOS << M <<
"\n"; });
1659 for (
const auto &ILI : ModuleToSummariesForIndex)
1663 if (ILI.first != ModulePath)
1672 UniqueID = GO->getMetadata(LLVMContext::MD_unique_id);
1678 F->setMetadata(LLVMContext::MD_unique_id,
UniqueID);
1679 F->setComdat(
nullptr);
1681 V->setInitializer(
nullptr);
1685 V->setMetadata(LLVMContext::MD_unique_id,
UniqueID);
1686 V->setComdat(
nullptr);
1715 auto FinalizeInModule = [&](
GlobalValue &GV,
bool Propagate =
false) {
1718 if (GS == DefinedGlobals.
end())
1725 if (FS->fflags().ReadNone && !
F->doesNotAccessMemory())
1726 F->setDoesNotAccessMemory();
1728 if (FS->fflags().ReadOnly && !
F->onlyReadsMemory())
1729 F->setOnlyReadsMemory();
1731 if (FS->fflags().NoRecurse && !
F->doesNotRecurse())
1732 F->setDoesNotRecurse();
1734 if (FS->fflags().NoUnwind && !
F->doesNotThrow())
1735 F->setDoesNotThrow();
1739 auto NewLinkage = GS->second->linkage();
1778 GS->second->canAutoHide()) {
1784 <<
"` from " << GV.
getLinkage() <<
" to " << NewLinkage
1792 if (GO && GO->isDeclarationForLinker() && GO->hasComdat()) {
1793 if (GO->getComdat()->getName() == GO->getName())
1794 NonPrevailingComdats.
insert(GO->getComdat());
1795 GO->setComdat(
nullptr);
1800 for (
auto &GV : TheModule)
1802 for (
auto &GV : TheModule.
globals())
1803 FinalizeInModule(GV);
1804 for (
auto &GV : TheModule.
aliases())
1805 FinalizeInModule(GV);
1810 if (NonPrevailingComdats.
empty())
1813 if (
auto *
C = GO.getComdat();
C && NonPrevailingComdats.
count(
C)) {
1814 GO.setComdat(
nullptr);
1825 for (
auto &GA : TheModule.
aliases()) {
1826 if (GA.hasAvailableExternallyLinkage())
1829 assert(Obj &&
"aliasee without an base object is unimplemented");
1830 if (Obj->hasAvailableExternallyLinkage()) {
1844 auto MustPreserveGV = [&](
const GlobalValue &GV) ->
bool {
1854 auto GS = DefinedGlobals.
find(GV.getGUIDOrFallback());
1855 if (GS == DefinedGlobals.
end()) {
1866 GS = DefinedGlobals.
find(
1868 if (GS == DefinedGlobals.
end()) {
1875 GS = DefinedGlobals.
find(
1906 for (
auto &GV : M.globals())
1909 if (!GV.isDeclaration() && GV.hasAttribute(
"thinlto-internalize")) {
1921 unsigned ImportedCount = 0, ImportedGVCount = 0;
1928 for (
auto &
F : DestModule)
1929 if (!
F.isDeclaration() && MoveSymbolGUIDSet.
contains(
F.getGUIDOrFallback()))
1939 if (!SrcModuleOrErr)
1941 std::unique_ptr<Module> SrcModule = std::move(*SrcModuleOrErr);
1943 "Context mismatch");
1947 if (
Error Err = SrcModule->materializeMetadata())
1948 return std::move(Err);
1957 auto GUID =
F.getGUIDOrFallback();
1958 auto MaybeImportType = ImportList.
getImportType(ModName, GUID);
1959 bool ImportDefinition =
1963 <<
" importing function"
1964 << (ImportDefinition
1966 : (MaybeImportType ?
" declaration " :
" "))
1967 << GUID <<
" " <<
F.getName() <<
" from "
1968 << SrcModule->getSourceFileName() <<
"\n");
1969 if (ImportDefinition) {
1970 if (
Error Err =
F.materialize())
1971 return std::move(Err);
1978 "thinlto_src_module",
1980 {MDString::get(DestModule.getContext(),
1981 SrcModule->getModuleIdentifier())}));
1985 {MDString::get(DestModule.getContext(),
1986 SrcModule->getSourceFileName())}));
1997 auto GUID = GV.getGUIDOrFallback();
1998 auto MaybeImportType = ImportList.
getImportType(ModName, GUID);
1999 bool ImportDefinition =
2003 <<
" importing global"
2004 << (ImportDefinition
2006 : (MaybeImportType ?
" declaration " :
" "))
2007 << GUID <<
" " << GV.getName() <<
" from "
2008 << SrcModule->getSourceFileName() <<
"\n");
2009 if (ImportDefinition) {
2010 if (
Error Err = GV.materialize())
2011 return std::move(Err);
2012 ImportedGVCount += GlobalsToImport.
insert(&GV);
2021 auto GUID = GA.getGUIDOrFallback();
2022 auto MaybeImportType = ImportList.
getImportType(ModName, GUID);
2023 bool ImportDefinition =
2027 <<
" importing alias"
2028 << (ImportDefinition
2030 : (MaybeImportType ?
" declaration " :
" "))
2031 << GUID <<
" " << GA.getName() <<
" from "
2032 << SrcModule->getSourceFileName() <<
"\n");
2033 if (ImportDefinition) {
2034 if (
Error Err = GA.materialize())
2035 return std::move(Err);
2039 return std::move(Err);
2045 <<
" " << GO->
getName() <<
" from "
2046 << SrcModule->getSourceFileName() <<
"\n");
2051 "thinlto_src_module",
2053 {MDString::get(DestModule.getContext(),
2054 SrcModule->getModuleIdentifier())}));
2058 {MDString::get(DestModule.getContext(),
2059 SrcModule->getSourceFileName())}));
2061 GlobalsToImport.
insert(Fn);
2074 SrcModule->setPartialSampleProfileRatio(Index);
2081 for (
const auto *GV : GlobalsToImport)
2083 <<
" from " << SrcModule->getSourceFileName() <<
"\n";
2086 if (
Error Err = Mover.
move(std::move(SrcModule),
2090 Twine(
"Function Import: link error: ") +
2093 ImportedCount += GlobalsToImport.
size();
2094 NumImportedModules++;
2099 NumImportedFunctions += (ImportedCount - ImportedGVCount);
2100 NumImportedGlobalVars += ImportedGVCount;
2103 LLVM_DEBUG(
dbgs() <<
"Imported " << ImportedCount - ImportedGVCount
2104 <<
" functions for Module "
2107 <<
" global variables for Module "
2109 return ImportedCount;
2119 if (!IndexPtrOrErr) {
2124 std::unique_ptr<ModuleSummaryIndex> Index = std::move(*IndexPtrOrErr);
2134 *Index, ImportList);
2137 isPrevailing, *Index, ImportList);
2143 for (
auto &
I : *Index) {
2144 for (
auto &S :
I.second.getSummaryList()) {
2146 S->setExternalLinkageForTest();
2156 auto ModuleLoader = [&M](
StringRef Identifier) {
2157 return loadFile(std::string(Identifier), M.getContext());
2166 "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")))
This file defines the DenseMap class.
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 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.
GUID getGUIDOrFallback() const
Return the GUID for this value if it has been assigned, otherwise fall back to computing it based on ...
ThreadLocalMode getThreadLocalMode() const
static bool isAvailableExternallyLinkage(LinkageTypes Linkage)
void setLinkage(LinkageTypes LT)
unsigned getAddressSpace() const
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.
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.
std::unordered_set< GlobalValueSummary * > GVSummaryPtrSet
A set of global value summary pointers.
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 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
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'"))
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.