55#ifdef EXPENSIVE_CHECKS
65#define DEBUG_TYPE "attributor"
66#define VERBOSE_DEBUG_TYPE DEBUG_TYPE "-verbose"
69 "Determine what attributes are manifested in the IR");
71STATISTIC(NumFnDeleted,
"Number of function deleted");
73 "Number of functions with exact definitions");
75 "Number of functions without exact definitions");
76STATISTIC(NumFnShallowWrappersCreated,
"Number of shallow wrappers created");
78 "Number of abstract attributes timed out before fixpoint");
80 "Number of abstract attributes in a valid fixpoint state");
82 "Number of abstract attributes manifested in IR");
94 cl::desc(
"Maximal number of fixpoint iterations."),
100 cl::desc(
"Maximal number of callees specialized for "
105 "attributor-max-initialization-chain-length",
cl::Hidden,
107 "Maximal number of chained initializations (to avoid stack overflows)"),
113 cl::desc(
"Annotate call sites of function declarations."),
cl::init(
false));
120 cl::desc(
"Allow the Attributor to create shallow "
121 "wrappers for non-exact definitions."),
126 cl::desc(
"Allow the Attributor to use IP information "
127 "derived from non-exact functions via cloning"),
134 cl::desc(
"Comma separated list of attribute names that are "
135 "allowed to be seeded."),
139 "attributor-function-seed-allow-list",
cl::Hidden,
140 cl::desc(
"Comma separated list of function names that are "
141 "allowed to be seeded."),
147 cl::desc(
"Dump the dependency graph to dot files."),
151 "attributor-depgraph-dot-filename-prefix",
cl::Hidden,
152 cl::desc(
"The prefix used for the CallGraph dot file names."));
155 cl::desc(
"View the dependency graph."),
159 cl::desc(
"Print attribute dependencies"),
163 "attributor-enable-call-site-specific-deduction",
cl::Hidden,
164 cl::desc(
"Allow the Attributor to do call site specific analysis"),
169 cl::desc(
"Print Attributor's internal call graph"),
174 cl::desc(
"Try to simplify all loads."),
179 cl::desc(
"Should a closed world be assumed, or not. Default if not set."));
202enum class NVPTXAMDGPUAddressSpace :
unsigned {
211enum class SPIRVAddressSpace :
unsigned {
230 return AS ==
static_cast<unsigned>(SPIRVAddressSpace::Generic);
232 return AS ==
static_cast<unsigned>(NVPTXAMDGPUAddressSpace::Generic);
240 return AS ==
static_cast<unsigned>(SPIRVAddressSpace::Global);
242 return AS ==
static_cast<unsigned>(NVPTXAMDGPUAddressSpace::Global);
250 return AS ==
static_cast<unsigned>(SPIRVAddressSpace::Shared);
252 return AS ==
static_cast<unsigned>(NVPTXAMDGPUAddressSpace::Shared);
260 return AS ==
static_cast<unsigned>(SPIRVAddressSpace::Constant);
262 return AS ==
static_cast<unsigned>(NVPTXAMDGPUAddressSpace::Constant);
270 return AS ==
static_cast<unsigned>(SPIRVAddressSpace::Local);
272 return AS ==
static_cast<unsigned>(NVPTXAMDGPUAddressSpace::Local);
279 if (CB->hasFnAttr(Attribute::NoSync))
283 if (!CB->isConvergent() && !CB->mayReadOrWriteMemory())
292 if (!
I.mayReadOrWriteMemory())
299 const Value &V,
bool ForAnalysisOnly) {
301 if (!ForAnalysisOnly)
318 bool UsedAssumedInformation =
false;
320 if (
A.hasGlobalVariableSimplificationCallback(*GV)) {
321 auto AssumedGV =
A.getAssumedInitializerFromCallBack(
322 *GV, &QueryingAA, UsedAssumedInformation);
323 Initializer = *AssumedGV;
327 if (!GV->hasLocalLinkage()) {
330 if (!GV->hasDefinitiveInitializer() || !GV->isConstant())
335 assert(!GV->hasLocalLinkage() || GV->hasInitializer());
338 Initializer = GV->getInitializer();
342 int64_t StorageSize =
DL.getTypeStoreSize(&Ty);
343 if (StorageSize != RangePtr->
Size)
356 return I->getFunction() == Scope;
358 return A->getParent() == Scope;
371 return A->getParent() == Scope;
373 if (
I->getFunction() == Scope) {
377 return DT->dominates(
I, CtxI);
379 if (CtxI &&
I->getParent() == CtxI->
getParent())
382 [&](
const Instruction &AfterI) { return &AfterI == CtxI; });
389 if (V.getType() == &Ty)
396 if (
C->isNullValue() && !Ty.isPtrOrPtrVectorTy())
398 if (
C->getType()->isPointerTy() && Ty.isPointerTy())
400 if (
C->getType()->getPrimitiveSizeInBits() >= Ty.getPrimitiveSizeInBits()) {
401 if (
C->getType()->isIntegerTy() && Ty.isIntegerTy())
403 if (
C->getType()->isFloatingPointTy() && Ty.isFloatingPointTy())
410std::optional<Value *>
412 const std::optional<Value *> &
B,
425 Ty = (*A)->getType();
435template <
bool IsLoad,
typename Ty>
441 LLVM_DEBUG(
dbgs() <<
"Trying to determine the potential copies of " <<
I
442 <<
" (only exact: " << OnlyExact <<
")\n";);
444 Value &Ptr = *
I.getPointerOperand();
453 A.getInfoCache().getTargetLibraryInfoForFunction(*
I.getFunction());
455 auto Pred = [&](
Value &Obj) {
463 Ptr.getType()->getPointerAddressSpace()) &&
464 A.getAssumedSimplified(Ptr, QueryingAA, UsedAssumedInformation,
468 dbgs() <<
"Underlying object is a valid nullptr, giving up.\n";);
474 LLVM_DEBUG(
dbgs() <<
"Underlying object is not supported yet: " << Obj
479 if (!GV->hasLocalLinkage() &&
480 !(GV->isConstant() && GV->hasInitializer())) {
482 "linkage, not supported yet: "
487 bool NullOnly =
true;
488 bool NullRequired =
false;
489 auto CheckForNullOnlyAndUndef = [&](std::optional<Value *> V,
491 if (!V || *V ==
nullptr)
496 NullRequired = !IsExact;
505 LLVM_DEBUG(
dbgs() <<
"Underlying object written but stored value "
506 "cannot be converted to read type: "
523 if (NewCopies.
count(V)) {
528 if (
Value *V = AdjustWrittenValueType(Acc, *
SI->getValueOperand()))
529 if (NewCopies.
count(V)) {
542 CheckForNullOnlyAndUndef(Acc.
getContent(), IsExact);
543 if (OnlyExact && !IsExact && !NullOnly &&
549 if (NullRequired && !NullOnly) {
550 LLVM_DEBUG(
dbgs() <<
"Required all `null` accesses due to non exact "
551 "one, however found non-null one: "
562 if (PotentialValueOrigins)
568 LLVM_DEBUG(
dbgs() <<
"Underlying object written through a non-store "
569 "instruction not supported yet: "
573 Value *V = AdjustWrittenValueType(Acc, *
SI->getValueOperand());
577 if (PotentialValueOrigins)
582 if (!LI && OnlyExact) {
584 "instruction not supported yet: "
595 bool HasBeenWrittenTo =
false;
600 if (!PI || !PI->forallInterferingAccesses(
603 !IsLoad, CheckAccess,
604 HasBeenWrittenTo,
Range, SkipCB)) {
607 <<
"Failed to verify all interfering accesses for underlying object: "
612 if (IsLoad && !HasBeenWrittenTo && !
Range.isUnassigned()) {
615 A, QueryingAA, Obj, *
I.getType(), TLI,
DL, &
Range);
617 LLVM_DEBUG(
dbgs() <<
"Could not determine required initial value of "
618 "underlying object, abort!\n");
621 CheckForNullOnlyAndUndef(InitialValue,
true);
622 if (NullRequired && !NullOnly) {
623 LLVM_DEBUG(
dbgs() <<
"Non exact access but initial value that is not "
624 "null or undef, abort!\n");
628 NewCopies.
insert(InitialValue);
629 if (PotentialValueOrigins)
630 NewCopyOrigins.
insert(
nullptr);
640 if (!AAUO || !AAUO->forallUnderlyingObjects(Pred)) {
642 dbgs() <<
"Underlying objects stored into could not be determined\n";);
649 for (
const auto *PI : PIs) {
650 if (!PI->getState().isAtFixpoint())
651 UsedAssumedInformation =
true;
655 if (PotentialValueOrigins)
667 A, LI, PotentialValues, &PotentialValueOrigins, QueryingAA,
668 UsedAssumedInformation, OnlyExact);
676 A,
SI, PotentialCopies,
nullptr, QueryingAA, UsedAssumedInformation,
682 bool RequireReadNone,
bool &IsKnown) {
683 if (RequireReadNone) {
695 const auto *MemLocAA =
697 if (MemLocAA && MemLocAA->isAssumedReadNone()) {
698 IsKnown = MemLocAA->isKnownReadNone();
705 const auto *MemBehaviorAA =
708 (MemBehaviorAA->isAssumedReadNone() ||
709 (!RequireReadNone && MemBehaviorAA->isAssumedReadOnly()))) {
710 IsKnown = RequireReadNone ? MemBehaviorAA->isKnownReadNone()
711 : MemBehaviorAA->isKnownReadOnly();
736 std::function<
bool(
const Function &
F)> GoBackwardsCB) {
738 dbgs() <<
"[AA] isPotentiallyReachable @" << ToFn.
getName() <<
" from "
739 << FromI <<
" [GBCB: " <<
bool(GoBackwardsCB) <<
"][#ExS: "
740 << (ExclusionSet ? std::to_string(ExclusionSet->
size()) :
"none")
743 for (
auto *ES : *ExclusionSet)
744 dbgs() << *ES <<
"\n";
752 if (GoBackwardsCB && &ToFn != FromI.
getFunction() &&
753 !GoBackwardsCB(*FromI.
getFunction()) &&
A.getInfoCache().isKernel(ToFn) &&
755 LLVM_DEBUG(
dbgs() <<
"[AA] assume kernel cannot be reached from within the "
756 "module; success\n";);
765 if (!GoBackwardsCB && !ExclusionSet) {
767 <<
" is not checked backwards and does not have an "
768 "exclusion set, abort\n");
776 while (!Worklist.
empty()) {
778 if (!Visited.
insert(CurFromI).second)
782 if (FromFn == &ToFn) {
785 LLVM_DEBUG(
dbgs() <<
"[AA] check " << *ToI <<
" from " << *CurFromI
786 <<
" intraprocedurally\n");
789 bool Result = !ReachabilityAA || ReachabilityAA->isAssumedReachable(
790 A, *CurFromI, *ToI, ExclusionSet);
792 << (Result ?
"can potentially " :
"cannot ") <<
"reach "
793 << *ToI <<
" [Intra]\n");
803 Result = !ToReachabilityAA || ToReachabilityAA->isAssumedReachable(
804 A, EntryI, *ToI, ExclusionSet);
806 <<
" " << (Result ?
"can potentially " :
"cannot ")
807 <<
"reach @" << *ToI <<
" [ToFn]\n");
815 Result = !FnReachabilityAA || FnReachabilityAA->instructionCanReach(
816 A, *CurFromI, ToFn, ExclusionSet);
818 <<
" " << (Result ?
"can potentially " :
"cannot ")
819 <<
"reach @" << ToFn.
getName() <<
" [FromFn]\n");
828 bool Result = !ReachabilityAA || ReachabilityAA->isAssumedReachable(
829 A, *CurFromI, Ret, ExclusionSet);
831 << (Result ?
"can potentially " :
"cannot ") <<
"reach "
832 << Ret <<
" [Intra]\n");
837 bool UsedAssumedInformation =
false;
838 if (
A.checkForAllInstructions(ReturnInstCB, FromFn, &QueryingAA,
839 {Instruction::Ret}, UsedAssumedInformation)) {
844 if (!GoBackwardsCB) {
846 <<
" is not checked backwards, abort\n");
852 if (!GoBackwardsCB(*FromFn))
859 CallBase *CB = ACS.getInstruction();
871 Result = !
A.checkForAllCallSites(CheckCallSite, *FromFn,
873 &QueryingAA, UsedAssumedInformation);
875 LLVM_DEBUG(
dbgs() <<
"[AA] stepping back to call sites from " << *CurFromI
876 <<
" in @" << FromFn->
getName()
877 <<
" failed, give up\n");
881 LLVM_DEBUG(
dbgs() <<
"[AA] stepped back to call sites from " << *CurFromI
882 <<
" in @" << FromFn->
getName()
883 <<
" worklist size is: " << Worklist.
size() <<
"\n");
892 std::function<
bool(
const Function &
F)> GoBackwardsCB) {
894 return ::isPotentiallyReachable(
A, FromI, &ToI, *ToFn, QueryingAA,
895 ExclusionSet, GoBackwardsCB);
902 std::function<
bool(
const Function &
F)> GoBackwardsCB) {
903 return ::isPotentiallyReachable(
A, FromI,
nullptr, ToFn, QueryingAA,
904 ExclusionSet, GoBackwardsCB);
915 dbgs() <<
"[AA] Object '" << Obj
916 <<
"' is thread local; stack objects are thread local.\n");
919 bool IsKnownNoCapture;
924 << (IsAssumedNoCapture ?
"" :
"not") <<
" thread local; "
925 << (IsAssumedNoCapture ?
"non-" :
"")
926 <<
"captured stack object.\n");
927 return IsAssumedNoCapture;
930 if (GV->isConstant()) {
932 <<
"' is thread local; constant global\n");
935 if (GV->isThreadLocal()) {
937 <<
"' is thread local; thread local global\n");
942 if (
A.getInfoCache().IsTargetGPU()) {
944 Obj.getType()->getPointerAddressSpace())) {
946 <<
"' is thread local; GPU local memory\n");
950 A.getInfoCache().getModule(),
951 Obj.getType()->getPointerAddressSpace())) {
953 <<
"' is thread local; GPU constant memory\n");
958 LLVM_DEBUG(
dbgs() <<
"[AA] Object '" << Obj <<
"' is not thread local\n");
964 if (!
I.mayHaveSideEffects() && !
I.mayReadFromMemory())
969 auto AddLocationPtr = [&](std::optional<MemoryLocation>
Loc) {
972 dbgs() <<
"[AA] Access to unknown location; -> requires barriers\n");
995 for (
const Value *Ptr : Ptrs) {
1001 auto Pred = [&](
Value &Obj) {
1004 LLVM_DEBUG(
dbgs() <<
"[AA] Access to '" << Obj <<
"' via '" << *Ptr
1005 <<
"'; -> requires barrier\n");
1011 if (!UnderlyingObjsAA || !UnderlyingObjsAA->forallUnderlyingObjects(Pred))
1036 AB.addAttribute(Kind);
1050 if (!ForceReplace && Kind == Attribute::Memory) {
1054 AB.addMemoryAttr(ME);
1061 AB.addAttribute(Attr);
1068 AB.addAttribute(Attr);
1089 std::optional<Argument *> CBCandidateArg;
1093 for (
const Use *U : CallbackUses) {
1107 "ACS mapped into var-args arguments!");
1108 if (CBCandidateArg) {
1109 CBCandidateArg =
nullptr;
1117 if (CBCandidateArg && *CBCandidateArg)
1118 return *CBCandidateArg;
1123 if (Callee && Callee->arg_size() >
unsigned(ArgNo))
1124 return Callee->getArg(ArgNo);
1138 LLVM_DEBUG(
dbgs() <<
"[Attributor] Update " << HasChanged <<
" " << *
this
1148 InfoCache(InfoCache), Configuration(Configuration) {
1152 if (Fn->hasAddressTaken(
nullptr,
1158 InfoCache.IndirectlyCallableFunctions.push_back(Fn);
1165 "Did expect a valid position!");
1182 unsigned AttrsSize = Attrs.size();
1185 for (
const auto &It : A2K)
1188 return AttrsSize != Attrs.size();
1191template <
typename DescTy>
1197 if (AttrDescs.
empty())
1209 auto [Iter, Inserted] = AttrsMap.insert({AttrListAnchor, AL});
1215 AttributeSet AS =
AL.getAttributes(AttrIdx);
1217 AttrBuilder
AB(Ctx);
1220 for (
const DescTy &AttrDesc : AttrDescs)
1221 if (CB(AttrDesc, AS, AM, AB))
1227 AL =
AL.removeAttributesAtIndex(Ctx, AttrIdx, AM);
1228 AL =
AL.addAttributesAtIndex(Ctx, AttrIdx, AB);
1236 bool IgnoreSubsumingPositions,
1238 bool Implied =
false;
1239 bool HasAttr =
false;
1242 if (AttrSet.hasAttribute(Kind)) {
1243 Implied |= Kind != ImpliedAttributeKind;
1249 updateAttrMap<Attribute::AttrKind>(EquivIRP, AttrKinds, HasAttrCB);
1255 if (IgnoreSubsumingPositions)
1272 ImpliedAttributeKind)});
1279 bool IgnoreSubsumingPositions) {
1283 if (AttrSet.hasAttribute(Kind))
1284 Attrs.push_back(AttrSet.getAttribute(Kind));
1288 updateAttrMap<Attribute::AttrKind>(EquivIRP, AttrKinds, CollectAttrCB);
1292 if (IgnoreSubsumingPositions)
1303 if (!AttrSet.hasAttribute(Kind))
1305 AM.addAttribute(Kind);
1308 return updateAttrMap<Attribute::AttrKind>(IRP, AttrKinds, RemoveAttrCB);
1315 if (!AttrSet.hasAttribute(Attr))
1317 AM.addAttribute(Attr);
1321 return updateAttrMap<StringRef>(IRP, Attrs, RemoveAttrCB);
1326 bool ForceReplace) {
1332 return updateAttrMap<Attribute>(IRP, Attrs, AddAttrCB);
1336 IRPositions.emplace_back(IRP);
1340 auto CanIgnoreOperandBundles = [](
const CallBase &CB) {
1356 assert(CB &&
"Expected call site!");
1359 if (!CB->hasOperandBundles() || CanIgnoreOperandBundles(*CB))
1364 assert(CB &&
"Expected call site!");
1367 if (!CB->hasOperandBundles() || CanIgnoreOperandBundles(*CB)) {
1372 for (
const Argument &Arg : Callee->args())
1373 if (Arg.hasReturnedAttr()) {
1374 IRPositions.emplace_back(
1376 IRPositions.emplace_back(
1385 assert(CB &&
"Expected call site!");
1388 if (!CB->hasOperandBundles() || CanIgnoreOperandBundles(*CB)) {
1402void IRPosition::verify() {
1403#ifdef EXPENSIVE_CHECKS
1406 assert((CBContext ==
nullptr) &&
1407 "Invalid position must not have CallBaseContext!");
1409 "Expected a nullptr for an invalid position!");
1413 "Expected specialized kind for argument values!");
1417 "Expected function for a 'returned' position!");
1419 "Associated value mismatch!");
1422 assert((CBContext ==
nullptr) &&
1423 "'call site returned' position must not have CallBaseContext!");
1425 "Expected call base for 'call site returned' position!");
1427 "Associated value mismatch!");
1430 assert((CBContext ==
nullptr) &&
1431 "'call site function' position must not have CallBaseContext!");
1433 "Expected call base for 'call site function' position!");
1435 "Associated value mismatch!");
1439 "Expected function for a 'function' position!");
1441 "Associated value mismatch!");
1445 "Expected argument for a 'argument' position!");
1447 "Associated value mismatch!");
1450 assert((CBContext ==
nullptr) &&
1451 "'call site argument' position must not have CallBaseContext!");
1452 Use *U = getAsUsePtr();
1454 assert(U &&
"Expected use for a 'call site argument' position!");
1456 "Expected call base user for a 'call site argument' position!");
1458 "Expected call base argument operand for a 'call site argument' "
1462 "Argument number mismatch!");
1470std::optional<Constant *>
1473 bool &UsedAssumedInformation) {
1477 for (
auto &CB : SimplificationCallbacks.lookup(IRP)) {
1478 std::optional<Value *> SimplifiedV = CB(IRP, &
AA, UsedAssumedInformation);
1480 return std::nullopt;
1490 UsedAssumedInformation)) {
1492 return std::nullopt;
1506 for (
auto &CB : SimplificationCallbacks.lookup(IRP))
1507 return CB(IRP,
AA, UsedAssumedInformation);
1513 return std::nullopt;
1526 bool &UsedAssumedInformation,
bool RecurseForSelectAndPHI) {
1530 while (!Worklist.
empty()) {
1536 int NV = Values.
size();
1537 const auto &SimplificationCBs = SimplificationCallbacks.lookup(IRP);
1538 for (
const auto &CB : SimplificationCBs) {
1539 std::optional<Value *> CBResult = CB(IRP,
AA, UsedAssumedInformation);
1540 if (!CBResult.has_value())
1542 Value *V = *CBResult;
1551 if (SimplificationCBs.empty()) {
1554 const auto *PotentialValuesAA =
1556 if (PotentialValuesAA &&
1557 PotentialValuesAA->getAssumedSimplifiedValues(*
this, Values, S)) {
1558 UsedAssumedInformation |= !PotentialValuesAA->isAtFixpoint();
1567 if (!RecurseForSelectAndPHI)
1570 for (
int I = NV, E = Values.
size();
I < E; ++
I) {
1571 Value *V = Values[
I].getValue();
1574 if (!Seen.
insert(V).second)
1577 Values[
I] = Values[E - 1];
1591 bool &UsedAssumedInformation) {
1599 if (!Arg->hasPointeeInMemoryValueAttr())
1609 for (
auto &It : AAMap) {
1611 AA->~AbstractAttribute();
1617 bool &UsedAssumedInformation,
1618 bool CheckBBLivenessOnly,
DepClassTy DepClass) {
1619 if (!Configuration.UseLiveness)
1625 CheckBBLivenessOnly, DepClass);
1631 bool &UsedAssumedInformation,
1632 bool CheckBBLivenessOnly,
DepClassTy DepClass) {
1633 if (!Configuration.UseLiveness)
1638 UsedAssumedInformation, CheckBBLivenessOnly, DepClass);
1643 if (CB->isArgOperand(&U)) {
1647 UsedAssumedInformation, CheckBBLivenessOnly,
1653 UsedAssumedInformation, CheckBBLivenessOnly, DepClass);
1657 UsedAssumedInformation, CheckBBLivenessOnly, DepClass);
1659 if (!CheckBBLivenessOnly &&
SI->getPointerOperand() != U.get()) {
1667 UsedAssumedInformation =
true;
1674 UsedAssumedInformation, CheckBBLivenessOnly, DepClass);
1680 bool &UsedAssumedInformation,
1681 bool CheckBBLivenessOnly,
DepClassTy DepClass,
1682 bool CheckForDeadStore) {
1683 if (!Configuration.UseLiveness)
1688 if (ManifestAddedBlocks.contains(
I.getParent()))
1697 if (!FnLivenessAA || QueryingAA == FnLivenessAA)
1701 if (CheckBBLivenessOnly ? FnLivenessAA->
isAssumedDead(
I.getParent())
1706 UsedAssumedInformation =
true;
1710 if (CheckBBLivenessOnly)
1718 if (!IsDeadAA || QueryingAA == IsDeadAA)
1725 UsedAssumedInformation =
true;
1733 UsedAssumedInformation =
true;
1743 bool &UsedAssumedInformation,
1744 bool CheckBBLivenessOnly,
DepClassTy DepClass) {
1745 if (!Configuration.UseLiveness)
1756 isAssumedDead(*CtxI, QueryingAA, FnLivenessAA, UsedAssumedInformation,
1761 if (CheckBBLivenessOnly)
1774 if (!IsDeadAA || QueryingAA == IsDeadAA)
1781 UsedAssumedInformation =
true;
1792 if (!Configuration.UseLiveness)
1800 if (!FnLivenessAA || QueryingAA == FnLivenessAA)
1816 return Pred(Callee);
1820 if (!CallEdgesAA || CallEdgesAA->hasUnknownCallee())
1823 const auto &Callees = CallEdgesAA->getOptimisticEdges();
1824 return Pred(Callees.getArrayRef());
1834 bool CheckBBLivenessOnly,
DepClassTy LivenessDepClass,
1835 bool IgnoreDroppableUses,
1840 if (!CB(*
this, &QueryingAA))
1854 auto AddUsers = [&](
const Value &V,
const Use *OldUse) {
1855 for (
const Use &UU : V.uses()) {
1856 if (OldUse && EquivalentUseCB && !EquivalentUseCB(*OldUse, UU)) {
1858 "rejected by the equivalence call back: "
1868 AddUsers(V,
nullptr);
1871 <<
" initial uses to check\n");
1874 const auto *LivenessAA =
1879 while (!Worklist.
empty()) {
1885 dbgs() <<
"[Attributor] Check use: " << **U <<
" in " << Fn->getName()
1888 dbgs() <<
"[Attributor] Check use: " << **U <<
" in " << *U->getUser()
1891 bool UsedAssumedInformation =
false;
1892 if (
isAssumedDead(*U, &QueryingAA, LivenessAA, UsedAssumedInformation,
1893 CheckBBLivenessOnly, LivenessDepClass)) {
1895 dbgs() <<
"[Attributor] Dead use, skip!\n");
1898 if (IgnoreDroppableUses && U->getUser()->isDroppable()) {
1900 dbgs() <<
"[Attributor] Droppable user, skip!\n");
1905 if (&
SI->getOperandUse(0) == U) {
1906 if (!Visited.
insert(U).second)
1910 *
this, *
SI, PotentialCopies, QueryingAA, UsedAssumedInformation,
1914 <<
"[Attributor] Value is stored, continue with "
1915 << PotentialCopies.
size()
1916 <<
" potential copies instead!\n");
1917 for (
Value *PotentialCopy : PotentialCopies)
1918 if (!AddUsers(*PotentialCopy, U))
1925 bool Follow =
false;
1926 if (!Pred(*U, Follow))
1931 User &Usr = *U->getUser();
1932 AddUsers(Usr,
nullptr);
1940 bool RequireAllCallSites,
1941 bool &UsedAssumedInformation) {
1947 if (!AssociatedFunction) {
1948 LLVM_DEBUG(
dbgs() <<
"[Attributor] No function associated with " << IRP
1954 &QueryingAA, UsedAssumedInformation);
1959 bool RequireAllCallSites,
1961 bool &UsedAssumedInformation,
1962 bool CheckPotentiallyDead) {
1966 <<
"[Attributor] Function " << Fn.
getName()
1967 <<
" has no internal linkage, hence not all call sites are known\n");
1972 if (!CB(*
this, QueryingAA))
1976 for (
unsigned u = 0; u <
Uses.size(); ++u) {
1980 dbgs() <<
"[Attributor] Check use: " << Fn->
getName() <<
" in "
1981 << *U.getUser() <<
"\n";
1983 dbgs() <<
"[Attributor] Check use: " << *U <<
" in " << *U.getUser()
1986 if (!CheckPotentiallyDead &&
1987 isAssumedDead(U, QueryingAA,
nullptr, UsedAssumedInformation,
1990 dbgs() <<
"[Attributor] Dead use, skip!\n");
1994 if (CE->isCast() && CE->getType()->isPointerTy()) {
1996 dbgs() <<
"[Attributor] Use, is constant cast expression, add "
1997 << CE->getNumUses() <<
" uses of that expression instead!\n";
1999 for (
const Use &CEU : CE->uses())
2000 Uses.push_back(&CEU);
2008 <<
" has non call site use " << *U.get() <<
" in "
2009 << *U.getUser() <<
"\n");
2013 const Use *EffectiveUse =
2016 if (!RequireAllCallSites) {
2017 LLVM_DEBUG(
dbgs() <<
"[Attributor] User " << *EffectiveUse->getUser()
2018 <<
" is not a call of " << Fn.
getName()
2022 LLVM_DEBUG(
dbgs() <<
"[Attributor] User " << *EffectiveUse->getUser()
2023 <<
" is an invalid use of " << Fn.
getName() <<
"\n");
2031 unsigned MinArgsParams =
2033 for (
unsigned u = 0; u < MinArgsParams; ++u) {
2037 dbgs() <<
"[Attributor] Call site / callee argument type mismatch ["
2038 << u <<
"@" << Fn.
getName() <<
": "
2048 LLVM_DEBUG(
dbgs() <<
"[Attributor] Call site callback failed for "
2056bool Attributor::shouldPropagateCallBaseContext(
const IRPosition &IRP) {
2066 bool RecurseForSelectAndPHI) {
2070 if (!AssociatedFunction)
2073 bool UsedAssumedInformation =
false;
2077 UsedAssumedInformation, RecurseForSelectAndPHI))
2089 bool &UsedAssumedInformation,
bool CheckBBLivenessOnly =
false,
2090 bool CheckPotentiallyDead =
false) {
2091 for (
unsigned Opcode : Opcodes) {
2093 auto *Insts = OpcodeInstMap.
lookup(Opcode);
2099 if (
A && !CheckPotentiallyDead &&
2101 UsedAssumedInformation, CheckBBLivenessOnly)) {
2103 dbgs() <<
"[Attributor] Instruction " << *
I
2104 <<
" is potentially dead, skip!\n";);
2119 bool &UsedAssumedInformation,
2120 bool CheckBBLivenessOnly,
2121 bool CheckPotentiallyDead) {
2127 const auto *LivenessAA =
2128 CheckPotentiallyDead && QueryingAA
2132 auto &OpcodeInstMap = InfoCache.getOpcodeInstMapForFunction(*Fn);
2134 LivenessAA, Opcodes, UsedAssumedInformation,
2135 CheckBBLivenessOnly, CheckPotentiallyDead))
2144 bool &UsedAssumedInformation,
2145 bool CheckBBLivenessOnly,
2146 bool CheckPotentiallyDead) {
2150 UsedAssumedInformation, CheckBBLivenessOnly,
2151 CheckPotentiallyDead);
2156 bool &UsedAssumedInformation) {
2159 const Function *AssociatedFunction =
2161 if (!AssociatedFunction)
2165 const auto *LivenessAA =
2169 InfoCache.getReadOrWriteInstsForFunction(*AssociatedFunction)) {
2172 UsedAssumedInformation))
2182void Attributor::runTillFixpoint() {
2186 <<
" abstract attributes.\n");
2191 unsigned IterationCounter = 1;
2192 unsigned MaxIterations =
2202 LLVM_DEBUG(
dbgs() <<
"\n\n[Attributor] #Iteration: " << IterationCounter
2203 <<
", Worklist size: " << Worklist.
size() <<
"\n");
2208 for (
unsigned u = 0; u < InvalidAAs.
size(); ++u) {
2213 dbgs() <<
"[Attributor] InvalidAA: " << *InvalidAA
2214 <<
" has " << InvalidAA->
Deps.
size()
2215 <<
" required & optional dependences\n");
2216 for (
auto &DepIt : InvalidAA->
Deps) {
2220 dbgs() <<
" - recompute: " << *DepAA);
2225 <<
" - invalidate: " << *DepAA);
2229 InvalidAAs.
insert(DepAA);
2238 for (AbstractAttribute *ChangedAA : ChangedAAs) {
2239 for (
auto &DepIt : ChangedAA->Deps)
2241 ChangedAA->Deps.clear();
2244 LLVM_DEBUG(
dbgs() <<
"[Attributor] #Iteration: " << IterationCounter
2245 <<
", Worklist+Dependent size: " << Worklist.
size()
2254 for (AbstractAttribute *AA : Worklist) {
2255 const auto &AAState = AA->getState();
2256 if (!AAState.isAtFixpoint())
2258 ChangedAAs.push_back(AA);
2262 if (!AAState.isValidState())
2268 ChangedAAs.append(DG.SyntheticRoot.begin() + NumAAs,
2269 DG.SyntheticRoot.end());
2274 Worklist.insert_range(ChangedAAs);
2275 Worklist.insert_range(QueryAAsAwaitingUpdate);
2276 QueryAAsAwaitingUpdate.clear();
2278 }
while (!Worklist.empty() && (IterationCounter++ < MaxIterations));
2280 if (IterationCounter > MaxIterations && !Functions.empty()) {
2281 auto Remark = [&](OptimizationRemarkMissed ORM) {
2282 return ORM <<
"Attributor did not reach a fixpoint after "
2283 <<
ore::NV(
"Iterations", MaxIterations) <<
" iterations.";
2289 LLVM_DEBUG(
dbgs() <<
"\n[Attributor] Fixpoint iteration done after: "
2290 << IterationCounter <<
"/" << MaxIterations
2291 <<
" iterations\n");
2298 SmallPtrSet<AbstractAttribute *, 32> Visited;
2299 for (
unsigned u = 0;
u < ChangedAAs.size();
u++) {
2300 AbstractAttribute *ChangedAA = ChangedAAs[
u];
2301 if (!Visited.
insert(ChangedAA).second)
2304 AbstractState &State = ChangedAA->
getState();
2308 NumAttributesTimedOut++;
2311 for (
auto &DepIt : ChangedAA->
Deps)
2317 if (!Visited.
empty())
2318 dbgs() <<
"\n[Attributor] Finalized " << Visited.
size()
2319 <<
" abstract attributes.\n";
2325 "Non-query AAs should not be required to register for updates!");
2326 QueryAAsAwaitingUpdate.insert(&
AA);
2333 unsigned NumManifested = 0;
2334 unsigned NumAtFixpoint = 0;
2344 if (!State.isAtFixpoint())
2345 State.indicateOptimisticFixpoint();
2348 if (
AA->hasCallBaseContext())
2351 if (!State.isValidState())
2354 if (
AA->getCtxI() && !
isRunOn(*
AA->getAnchorScope()))
2358 bool UsedAssumedInformation =
false;
2369 AA->trackStatistics();
2370 LLVM_DEBUG(
dbgs() <<
"[Attributor] Manifest " << LocalChange <<
" : " << *
AA
2373 ManifestChange = ManifestChange | LocalChange;
2379 (void)NumManifested;
2380 (void)NumAtFixpoint;
2381 LLVM_DEBUG(
dbgs() <<
"\n[Attributor] Manifested " << NumManifested
2382 <<
" arguments while " << NumAtFixpoint
2383 <<
" were in a valid fixpoint state\n");
2385 NumAttributesManifested += NumManifested;
2386 NumAttributesValidFixpoint += NumAtFixpoint;
2391 for (
unsigned u = 0; u < NumFinalAAs; ++u)
2395 errs() <<
"Unexpected abstract attribute: "
2399 .getAssociatedValue()
2403 "remain unchanged!");
2406 for (
auto &It : AttrsMap) {
2407 AttributeList &
AL = It.getSecond();
2408 const IRPosition &IRP =
2411 : IRPosition::callsite_function(*
cast<CallBase>(It.getFirst()));
2415 return ManifestChange;
2418void Attributor::identifyDeadInternalFunctions() {
2420 if (!Configuration.DeleteFns)
2438 for (Function *
F : Functions)
2439 if (
F->hasLocalLinkage() && (
isModulePass() || !TLI->getLibFunc(*
F, LF)))
2442 SmallPtrSet<Function *, 8> LiveInternalFns;
2443 bool FoundLiveInternal =
true;
2444 while (FoundLiveInternal) {
2445 FoundLiveInternal =
false;
2446 for (Function *&
F : InternalFns) {
2450 bool UsedAssumedInformation =
false;
2452 [&](AbstractCallSite ACS) {
2454 return ToBeDeletedFunctions.count(Callee) ||
2455 (Functions.count(Callee) &&
Callee->hasLocalLinkage() &&
2456 !LiveInternalFns.
count(Callee));
2458 *
F,
true,
nullptr, UsedAssumedInformation)) {
2464 FoundLiveInternal =
true;
2468 for (Function *
F : InternalFns)
2470 ToBeDeletedFunctions.insert(
F);
2474 TimeTraceScope TimeScope(
"Attributor::cleanupIR");
2477 << ToBeDeletedFunctions.size() <<
" functions and "
2478 << ToBeDeletedBlocks.size() <<
" blocks and "
2479 << ToBeDeletedInsts.size() <<
" instructions and "
2480 << ToBeChangedValues.size() <<
" values and "
2481 << ToBeChangedUses.size() <<
" uses. To insert "
2482 << ToBeChangedToUnreachableInsts.size()
2483 <<
" unreachables.\n"
2484 <<
"Preserve manifest added " << ManifestAddedBlocks.size()
2490 auto ReplaceUse = [&](
Use *
U,
Value *NewV) {
2495 const auto &
Entry = ToBeChangedValues.lookup(NewV);
2503 "Cannot replace an instruction outside the current SCC!");
2509 if (CI->isMustTailCall() && !ToBeDeletedInsts.count(CI))
2514 for (
auto &Arg : RI->getFunction()->args())
2515 Arg.removeAttr(Attribute::Returned);
2519 <<
" instead of " << *OldV <<
"\n");
2523 CGModifiedFunctions.insert(
I->getFunction());
2530 if (CB->isArgOperand(U)) {
2531 unsigned Idx = CB->getArgOperandNo(U);
2532 CB->removeParamAttr(Idx, Attribute::NoUndef);
2534 if (Callee &&
Callee->arg_size() > Idx)
2535 Callee->removeParamAttr(Idx, Attribute::NoUndef);
2541 ToBeChangedToUnreachableInsts.insert(UserI);
2548 for (
auto &It : ToBeChangedUses) {
2550 Value *NewV = It.second;
2551 ReplaceUse(U, NewV);
2555 for (
auto &It : ToBeChangedValues) {
2556 Value *OldV = It.first;
2557 auto [NewV,
Done] = It.second;
2559 for (
auto &U : OldV->
uses())
2560 if (
Done || !
U.getUser()->isDroppable())
2562 for (Use *U :
Uses) {
2566 ReplaceUse(U, NewV);
2570 for (
const auto &V : InvokeWithDeadSuccessor)
2573 "Cannot replace an invoke outside the current SCC!");
2574 bool UnwindBBIsDead =
II->hasFnAttr(Attribute::NoUnwind);
2575 bool NormalBBIsDead =
II->hasFnAttr(Attribute::NoReturn);
2576 bool Invoke2CallAllowed =
2578 assert((UnwindBBIsDead || NormalBBIsDead) &&
2579 "Invoke does not have dead successors!");
2582 if (UnwindBBIsDead) {
2584 if (Invoke2CallAllowed) {
2589 ToBeChangedToUnreachableInsts.insert(NormalNextIP);
2591 assert(NormalBBIsDead &&
"Broken invariant!");
2594 ToBeChangedToUnreachableInsts.insert(&NormalDestBB->
front());
2597 for (Instruction *
I : TerminatorsToFold) {
2599 "Cannot replace a terminator outside the current SCC!");
2600 CGModifiedFunctions.insert(
I->getFunction());
2603 for (
const auto &V : ToBeChangedToUnreachableInsts)
2608 "Cannot replace an instruction outside the current SCC!");
2609 CGModifiedFunctions.insert(
I->getFunction());
2613 for (
const auto &V : ToBeDeletedInsts) {
2617 "Cannot delete an instruction outside the current SCC!");
2618 I->dropDroppableUses();
2619 CGModifiedFunctions.insert(
I->getFunction());
2620 if (!
I->getType()->isVoidTy())
2625 I->eraseFromParent();
2632 dbgs() <<
"[Attributor] DeadInsts size: " << DeadInsts.
size() <<
"\n";
2633 for (
auto &
I : DeadInsts)
2635 dbgs() <<
" - " << *
I <<
"\n";
2640 if (
unsigned NumDeadBlocks = ToBeDeletedBlocks.size()) {
2641 SmallVector<BasicBlock *, 8> ToBeDeletedBBs;
2642 ToBeDeletedBBs.
reserve(NumDeadBlocks);
2643 for (BasicBlock *BB : ToBeDeletedBlocks) {
2645 "Cannot delete a block outside the current SCC!");
2646 CGModifiedFunctions.insert(BB->
getParent());
2648 if (ManifestAddedBlocks.contains(BB))
2658 identifyDeadInternalFunctions();
2661 ChangeStatus ManifestChange = rewriteFunctionSignatures(CGModifiedFunctions);
2663 for (Function *Fn : CGModifiedFunctions)
2664 if (!ToBeDeletedFunctions.count(Fn) && Functions.count(Fn))
2665 Configuration.CGUpdater.reanalyzeFunction(*Fn);
2667 for (Function *Fn : ToBeDeletedFunctions) {
2668 if (!Functions.count(Fn))
2670 Configuration.CGUpdater.removeFunction(*Fn);
2673 if (!ToBeChangedUses.empty())
2676 if (!ToBeChangedToUnreachableInsts.empty())
2679 if (!ToBeDeletedFunctions.empty())
2682 if (!ToBeDeletedBlocks.empty())
2685 if (!ToBeDeletedInsts.empty())
2688 if (!InvokeWithDeadSuccessor.empty())
2691 if (!DeadInsts.empty())
2694 NumFnDeleted += ToBeDeletedFunctions.size();
2696 LLVM_DEBUG(
dbgs() <<
"[Attributor] Deleted " << ToBeDeletedFunctions.size()
2697 <<
" functions after manifest.\n");
2699#ifdef EXPENSIVE_CHECKS
2700 for (Function *
F : Functions) {
2701 if (ToBeDeletedFunctions.count(
F))
2707 return ManifestChange;
2712 AttributorCallGraph ACallGraph(*
this);
2717 Phase = AttributorPhase::UPDATE;
2730 Phase = AttributorPhase::MANIFEST;
2733 Phase = AttributorPhase::CLEANUP;
2739 return ManifestChange | CleanupChange;
2744 return AA.getName().str() +
2745 std::to_string(
AA.getIRPosition().getPositionKind());
2748 "We can update AA only in the update stage!");
2751 DependenceVector DV;
2752 DependenceStack.push_back(&DV);
2754 auto &AAState =
AA.getState();
2756 bool UsedAssumedInformation =
false;
2759 CS =
AA.update(*
this);
2761 if (!
AA.isQueryAA() && DV.empty() && !
AA.getState().isAtFixpoint()) {
2768 RerunCS =
AA.update(*
this);
2774 AAState.indicateOptimisticFixpoint();
2777 if (!AAState.isAtFixpoint())
2778 rememberDependences();
2782 DependenceVector *PoppedDV = DependenceStack.pop_back_val();
2784 assert(PoppedDV == &DV &&
"Inconsistent usage of the dependence stack!");
2790 assert(!
F.isDeclaration() &&
"Cannot create a wrapper around a declaration!");
2799 M.getFunctionList().insert(
F.getIterator(),
Wrapper);
2804 assert(
F.use_empty() &&
"Uses remained after wrapper was created!");
2809 F.setComdat(
nullptr);
2813 F.getAllMetadata(MDs);
2814 for (
auto MDIt : MDs)
2815 Wrapper->addMetadata(MDIt.first, *MDIt.second);
2816 Wrapper->setAttributes(
F.getAttributes());
2824 Args.push_back(&Arg);
2825 Arg.setName((FArgIt++)->
getName());
2833 NumFnShallowWrappersCreated++;
2837 if (
F.isDeclaration() ||
F.hasLocalLinkage() ||
2853 return InternalizedFns[&
F];
2871 F->getName() +
".internalized");
2874 for (
auto &Arg :
F->args()) {
2875 auto ArgName = Arg.getName();
2876 NewFArgIt->setName(ArgName);
2877 VMap[&Arg] = &(*NewFArgIt++);
2892 F->getAllMetadata(MDs);
2893 for (
auto MDIt : MDs)
2897 M.getFunctionList().insert(
F->getIterator(), Copied);
2905 auto &InternalizedFn = FnMap[
F];
2906 auto IsNotInternalized = [&](
Use &U) ->
bool {
2908 return !FnMap.
lookup(CB->getCaller());
2911 F->replaceUsesWithIf(InternalizedFn, IsNotInternalized);
2920 if (!Configuration.RewriteSignatures)
2943 LLVM_DEBUG(
dbgs() <<
"[Attributor] Cannot rewrite var-args functions\n");
2949 if (FnAttributeList.hasAttrSomewhere(Attribute::Nest) ||
2950 FnAttributeList.hasAttrSomewhere(Attribute::StructRet) ||
2951 FnAttributeList.hasAttrSomewhere(Attribute::InAlloca) ||
2952 FnAttributeList.hasAttrSomewhere(Attribute::Preallocated)) {
2954 dbgs() <<
"[Attributor] Cannot rewrite due to complex attribute\n");
2959 bool UsedAssumedInformation =
false;
2961 UsedAssumedInformation,
2963 LLVM_DEBUG(
dbgs() <<
"[Attributor] Cannot rewrite all call sites\n");
2969 return !CI->isMustTailCall();
2975 auto &OpcodeInstMap = InfoCache.getOpcodeInstMapForFunction(*Fn);
2977 nullptr, {Instruction::Call},
2978 UsedAssumedInformation)) {
2979 LLVM_DEBUG(
dbgs() <<
"[Attributor] Cannot rewrite due to instructions\n");
2990 LLVM_DEBUG(
dbgs() <<
"[Attributor] Register new rewrite of " << Arg <<
" in "
2992 << ReplacementTypes.
size() <<
" replacements\n");
2994 "Cannot register an invalid rewrite");
2998 ArgumentReplacementMap[Fn];
3004 std::unique_ptr<ArgumentReplacementInfo> &ARI = ARIs[Arg.
getArgNo()];
3005 if (ARI && ARI->getNumReplacementArgs() <= ReplacementTypes.
size()) {
3006 LLVM_DEBUG(
dbgs() <<
"[Attributor] Existing rewrite is preferred\n");
3014 LLVM_DEBUG(
dbgs() <<
"[Attributor] Register new rewrite of " << Arg <<
" in "
3016 << ReplacementTypes.
size() <<
" replacements\n");
3020 std::move(CalleeRepairCB),
3021 std::move(ACSRepairCB)));
3042 for (
auto &It : ArgumentReplacementMap) {
3046 if (!Functions.count(OldFn) || ToBeDeletedFunctions.count(OldFn))
3059 if (
const std::unique_ptr<ArgumentReplacementInfo> &ARI =
3060 ARIs[Arg.getArgNo()]) {
3061 NewArgumentTypes.
append(ARI->ReplacementTypes.begin(),
3062 ARI->ReplacementTypes.end());
3063 NewArgumentAttributes.
append(ARI->getNumReplacementArgs(),
3066 NewArgumentTypes.
push_back(Arg.getType());
3068 OldFnAttributeList.getParamAttrs(Arg.getArgNo()));
3072 uint64_t LargestVectorWidth = 0;
3073 for (
auto *
I : NewArgumentTypes)
3075 LargestVectorWidth =
3076 std::max(LargestVectorWidth,
3077 VT->getPrimitiveSizeInBits().getKnownMinValue());
3080 Type *RetTy = OldFnTy->getReturnType();
3083 FunctionType *NewFnTy =
3088 << *NewFnTy <<
"\n");
3093 Functions.insert(NewFn);
3106 Ctx, OldFnAttributeList.getFnAttrs(), OldFnAttributeList.getRetAttrs(),
3107 NewArgumentAttributes));
3108 AttributeFuncs::updateMinLegalVectorWidthAttr(*NewFn, LargestVectorWidth);
3116 return !
T->isPtrOrPtrVectorTy() ||
3132 auto CallSiteReplacementCreator = [&](AbstractCallSite ACS) {
3134 const AttributeList &OldCallAttributeList = OldCB->
getAttributes();
3137 SmallVector<Value *, 16> NewArgOperands;
3139 for (
unsigned OldArgNum = 0; OldArgNum < ARIs.
size(); ++OldArgNum) {
3140 unsigned NewFirstArgNum = NewArgOperands.
size();
3141 (void)NewFirstArgNum;
3142 if (
const std::unique_ptr<ArgumentReplacementInfo> &ARI =
3144 if (ARI->ACSRepairCB)
3145 ARI->ACSRepairCB(*ARI, ACS, NewArgOperands);
3146 assert(ARI->getNumReplacementArgs() + NewFirstArgNum ==
3147 NewArgOperands.
size() &&
3148 "ACS repair callback did not provide as many operand as new "
3149 "types were registered!");
3151 NewArgOperandAttributes.
append(ARI->ReplacementTypes.size(),
3156 OldCallAttributeList.getParamAttrs(OldArgNum));
3160 assert(NewArgOperands.
size() == NewArgOperandAttributes.
size() &&
3161 "Mismatch # argument operands vs. # argument operand attributes!");
3163 "Mismatch # argument operands vs. # function arguments!");
3172 II->getUnwindDest(), NewArgOperands,
3177 NewCI->setTailCallKind(
cast<CallInst>(OldCB)->getTailCallKind());
3182 NewCB->
copyMetadata(*OldCB, {LLVMContext::MD_prof, LLVMContext::MD_dbg});
3186 Ctx, OldCallAttributeList.getFnAttrs(),
3187 OldCallAttributeList.getRetAttrs(), NewArgOperandAttributes));
3189 AttributeFuncs::updateMinLegalVectorWidthAttr(*NewCB->
getCaller(),
3190 LargestVectorWidth);
3192 CallSitePairs.
push_back({OldCB, NewCB});
3197 bool UsedAssumedInformation =
false;
3199 true,
nullptr, UsedAssumedInformation,
3202 assert(
Success &&
"Assumed call site replacement to succeed!");
3207 for (
unsigned OldArgNum = 0; OldArgNum < ARIs.
size();
3208 ++OldArgNum, ++OldFnArgIt) {
3209 if (
const std::unique_ptr<ArgumentReplacementInfo> &ARI =
3211 if (ARI->CalleeRepairCB)
3212 ARI->CalleeRepairCB(*ARI, *NewFn, NewFnArgIt);
3213 if (ARI->ReplacementTypes.empty())
3216 NewFnArgIt += ARI->ReplacementTypes.size();
3218 NewFnArgIt->
takeName(&*OldFnArgIt);
3225 for (
auto &CallSitePair : CallSitePairs) {
3226 CallBase &OldCB = *CallSitePair.first;
3227 CallBase &NewCB = *CallSitePair.second;
3229 "Cannot handle call sites with different types!");
3236 Configuration.CGUpdater.replaceFunctionWith(*OldFn, *NewFn);
3240 if (ModifiedFns.
remove(OldFn))
3241 ModifiedFns.
insert(NewFn);
3249void InformationCache::initializeInformationCache(
const Function &CF,
3256 FI.IsKernel =
F.hasFnAttribute(
"kernel");
3262 DenseMap<const Value *, std::optional<short>> AssumeUsesMap;
3267 auto AddToAssumeUsesMap = [&](
const Value &
V) ->
void {
3268 SmallVector<const Instruction *> Worklist;
3271 while (!Worklist.
empty()) {
3273 std::optional<short> &NumUses = AssumeUsesMap[
I];
3275 NumUses =
I->getNumUses();
3276 NumUses = *NumUses - 1;
3279 AssumeOnlyValues.insert(
I);
3280 for (
const Value *
Op :
I->operands())
3287 bool IsInterestingOpcode =
false;
3294 switch (
I.getOpcode()) {
3297 "New call base instruction type needs to be known in the "
3300 case Instruction::Call:
3305 AssumeOnlyValues.insert(Assume);
3307 AddToAssumeUsesMap(*
Assume->getArgOperand(0));
3309 FI.ContainsMustTailCall =
true;
3312 getFunctionInfo(*Callee).CalledViaMustTail =
true;
3315 case Instruction::CallBr:
3316 case Instruction::Invoke:
3317 case Instruction::CleanupRet:
3318 case Instruction::CatchSwitch:
3319 case Instruction::AtomicRMW:
3320 case Instruction::AtomicCmpXchg:
3321 case Instruction::UncondBr:
3322 case Instruction::CondBr:
3323 case Instruction::Resume:
3324 case Instruction::Ret:
3325 case Instruction::Load:
3327 case Instruction::Store:
3329 case Instruction::Alloca:
3330 case Instruction::AddrSpaceCast:
3331 IsInterestingOpcode =
true;
3333 if (IsInterestingOpcode) {
3334 auto *&Insts = FI.OpcodeInstMap[
I.getOpcode()];
3339 if (
I.mayReadOrWriteMemory())
3340 FI.RWInsts.push_back(&
I);
3343 if (
F.hasFnAttribute(Attribute::AlwaysInline) &&
3345 InlineableFunctions.insert(&
F);
3348InformationCache::FunctionInfo::~FunctionInfo() {
3351 for (
auto &It : OpcodeInstMap)
3352 It.getSecond()->~InstructionVectorTy();
3357 assert(
A.isClosedWorldModule() &&
"Cannot see all indirect callees!");
3358 return IndirectlyCallableFunctions;
3364 return std::nullopt;
3375 if (DependenceStack.empty())
3379 DependenceStack.back()->push_back({&FromAA, &ToAA, DepClass});
3382void Attributor::rememberDependences() {
3383 assert(!DependenceStack.empty() &&
"No dependences to remember!");
3385 for (DepInfo &DI : *DependenceStack.back()) {
3388 "Expected required or optional dependence (1 bit)!");
3395template <Attribute::AttrKind AK,
typename AAType>
3396void Attributor::checkAndQueryIRAttr(
const IRPosition &IRP, AttributeSet Attrs,
3397 bool SkipHasAttrCheck) {
3399 if (SkipHasAttrCheck || !
Attrs.hasAttribute(AK))
3400 if (!Configuration.Allowed || Configuration.Allowed->count(&AAType::ID))
3403 getOrCreateAAFor<AAType>(IRP);
3409 if (!VisitedFunctions.insert(&
F).second)
3415 InformationCache::FunctionInfo &FI = InfoCache.getFunctionInfo(
F);
3417 for (
const Use &U :
F.uses())
3419 if (CB->isCallee(&U) && CB->isMustTailCall())
3420 FI.CalledViaMustTail =
true;
3425 auto Attrs =
F.getAttributes();
3426 auto FnAttrs = Attrs.getFnAttrs();
3442 checkAndQueryIRAttr<Attribute::MustProgress, AAMustProgress>(FPos, FnAttrs);
3445 checkAndQueryIRAttr<Attribute::NoFree, AANoFree>(FPos, FnAttrs);
3448 checkAndQueryIRAttr<Attribute::WillReturn, AAWillReturn>(FPos, FnAttrs);
3451 checkAndQueryIRAttr<Attribute::NoSync, AANoSync>(FPos, FnAttrs);
3456 if (IsIPOAmendable) {
3459 checkAndQueryIRAttr<Attribute::NoUnwind, AANoUnwind>(FPos, FnAttrs);
3462 checkAndQueryIRAttr<Attribute::NoReturn, AANoReturn>(FPos, FnAttrs);
3465 checkAndQueryIRAttr<Attribute::NoRecurse, AANoRecurse>(FPos, FnAttrs);
3468 if (Attrs.hasFnAttr(Attribute::Convergent))
3489 Type *ReturnType =
F.getReturnType();
3490 if (!ReturnType->isVoidTy()) {
3498 bool UsedAssumedInformation =
false;
3503 checkAndQueryIRAttr<Attribute::NoUndef, AANoUndef>(RetPos, RetAttrs);
3505 if (ReturnType->isPointerTy()) {
3511 checkAndQueryIRAttr<Attribute::NonNull, AANonNull>(RetPos, RetAttrs);
3514 checkAndQueryIRAttr<Attribute::NoAlias, AANoAlias>(RetPos, RetAttrs);
3519 }
else if (AttributeFuncs::isNoFPClassCompatibleType(ReturnType)) {
3527 auto ArgNo = Arg.getArgNo();
3530 if (!IsIPOAmendable) {
3531 if (Arg.getType()->isPointerTy())
3533 checkAndQueryIRAttr<Attribute::NoFree, AANoFree>(ArgPos, ArgAttrs);
3540 bool UsedAssumedInformation =
false;
3548 checkAndQueryIRAttr<Attribute::NoUndef, AANoUndef>(ArgPos, ArgAttrs);
3550 if (Arg.getType()->isPointerTy()) {
3552 checkAndQueryIRAttr<Attribute::NonNull, AANonNull>(ArgPos, ArgAttrs);
3555 checkAndQueryIRAttr<Attribute::NoAlias, AANoAlias>(ArgPos, ArgAttrs);
3564 checkAndQueryIRAttr<Attribute::Captures, AANoCapture>(
3565 ArgPos, ArgAttrs,
true);
3572 checkAndQueryIRAttr<Attribute::NoFree, AANoFree>(ArgPos, ArgAttrs);
3577 }
else if (AttributeFuncs::isNoFPClassCompatibleType(Arg.getType())) {
3605 !Callee->hasMetadata(LLVMContext::MD_callback))
3608 if (!Callee->getReturnType()->isVoidTy() && !CB.use_empty()) {
3610 bool UsedAssumedInformation =
false;
3614 if (AttributeFuncs::isNoFPClassCompatibleType(Callee->getReturnType()))
3618 const AttributeList &CBAttrs = CBFnPos.
getAttrList();
3619 for (
int I = 0, E = CB.arg_size();
I < E; ++
I) {
3630 bool UsedAssumedInformation =
false;
3635 checkAndQueryIRAttr<Attribute::NoUndef, AANoUndef>(CBArgPos, CBArgAttrs);
3637 Type *ArgTy = CB.getArgOperand(
I)->getType();
3640 if (AttributeFuncs::isNoFPClassCompatibleType(ArgTy))
3647 checkAndQueryIRAttr<Attribute::NonNull, AANonNull>(CBArgPos, CBArgAttrs);
3650 checkAndQueryIRAttr<Attribute::Captures, AANoCapture>(
3651 CBArgPos, CBArgAttrs,
true);
3654 checkAndQueryIRAttr<Attribute::NoAlias, AANoAlias>(CBArgPos, CBArgAttrs);
3664 if (!CBAttrs.hasParamAttr(
I, Attribute::ReadNone))
3668 checkAndQueryIRAttr<Attribute::NoFree, AANoFree>(CBArgPos, CBArgAttrs);
3673 auto &OpcodeInstMap = InfoCache.getOpcodeInstMapForFunction(
F);
3674 [[maybe_unused]]
bool Success;
3675 bool UsedAssumedInformation =
false;
3677 nullptr, OpcodeInstMap, CallSitePred,
nullptr,
nullptr,
3678 {(
unsigned)Instruction::Invoke, (
unsigned)Instruction::CallBr,
3680 UsedAssumedInformation);
3681 assert(
Success &&
"Expected the check call to be successful!");
3705 nullptr, OpcodeInstMap, LoadStorePred,
nullptr,
nullptr,
3706 {(
unsigned)Instruction::Load, (
unsigned)Instruction::Store},
3707 UsedAssumedInformation);
3708 assert(
Success &&
"Expected the check call to be successful!");
3711 auto AAAllocationInfoPred = [&](
Instruction &
I) ->
bool {
3717 nullptr, OpcodeInstMap, AAAllocationInfoPred,
nullptr,
nullptr,
3718 {(
unsigned)Instruction::Alloca}, UsedAssumedInformation);
3719 assert(
Success &&
"Expected the check call to be successful!");
3725 return isModulePass() && Configuration.IsClosedWorldModule;
3742 return OS <<
"fn_ret";
3744 return OS <<
"cs_ret";
3752 return OS <<
"cs_arg";
3774 return OS << static_cast<const AbstractState &>(S);
3788 OS <<
"set-state(< {";
3804 OS <<
"set-state(< {";
3810 OS <<
"@" <<
F->getName() <<
"[" << int(It.second) <<
"], ";
3812 OS << *It.first.getValue() <<
"[" << int(It.second) <<
"], ";
3825 OS <<
"] for CtxI ";
3832 OS <<
"<<null inst>>";
3841 for (
const auto &DepAA :
Deps) {
3842 auto *
AA = DepAA.getPointer();
3859 OS <<
" [ <unknown> ]";
3873 bool DeleteFns,
bool IsModulePass) {
3874 if (Functions.empty())
3878 dbgs() <<
"[Attributor] Run on module with " << Functions.size()
3887 AC.IsModulePass = IsModulePass;
3888 AC.DeleteFns = DeleteFns;
3892 IndirectCalleeTrackingMap;
3894 AC.IndirectCalleeSpecializationCallback =
3899 auto &Set = IndirectCalleeTrackingMap[&CB];
3901 Set = std::make_unique<SmallPtrSet<Function *, 8>>();
3903 return Set->contains(&Callee);
3904 Set->insert(&Callee);
3914 if (!
A.isFunctionIPOAmendable(*
F))
3922 unsigned FunSize = Functions.size();
3923 for (
unsigned u = 0; u < FunSize; u++) {
3925 if (!
F->isDeclaration() && !
F->isDefinitionExact() && !
F->use_empty() &&
3928 assert(NewF &&
"Could not internalize function.");
3929 Functions.insert(NewF);
3933 for (
const Use &U : NewF->
uses())
3935 auto *CallerF = CB->getCaller();
3943 if (
F->isDeclaration())
3946 if (
F->hasExactDefinition())
3947 NumFnWithExactDefinition++;
3949 NumFnWithoutExactDefinition++;
3954 if (
F->hasLocalLinkage()) {
3956 const auto *CB = dyn_cast<CallBase>(U.getUser());
3957 return CB && CB->isCallee(&U) &&
3958 Functions.count(const_cast<Function *>(CB->getCaller()));
3965 A.identifyDefaultAbstractAttributes(*
F);
3970 LLVM_DEBUG(
dbgs() <<
"[Attributor] Done with " << Functions.size()
3971 <<
" functions, result: " <<
Changed <<
".\n");
3980 bool IsModulePass) {
3981 if (Functions.empty())
3985 dbgs() <<
"[AttributorLight] Run on module with " << Functions.size()
3994 AC.IsModulePass = IsModulePass;
3995 AC.DeleteFns =
false;
4003 AC.Allowed = &Allowed;
4004 AC.UseLiveness =
false;
4009 if (
F->isDeclaration())
4012 if (
F->hasExactDefinition())
4013 NumFnWithExactDefinition++;
4015 NumFnWithoutExactDefinition++;
4020 if (AC.UseLiveness &&
F->hasLocalLinkage()) {
4022 const auto *CB = dyn_cast<CallBase>(U.getUser());
4023 return CB && CB->isCallee(&U) &&
4024 Functions.count(const_cast<Function *>(CB->getCaller()));
4031 A.identifyDefaultAbstractAttributes(*
F);
4048 for (
auto *U :
Changed->users()) {
4051 FAM.invalidate(*
Call->getFunction(), FuncPA);
4056 LLVM_DEBUG(
dbgs() <<
"[Attributor] Done with " << Functions.size()
4057 <<
" functions, result: " <<
Changed <<
".\n");
4064 static std::atomic<int> CallTimes;
4070 Prefix =
"dep_graph";
4072 Prefix +
"_" + std::to_string(CallTimes.load()) +
".dot";
4074 outs() <<
"Dependency graph dump to " <<
Filename <<
".\n";
4120 Functions.
insert(&
N.getFunction());
4122 if (Functions.empty())
4125 Module &M = *Functions.back()->getParent();
4176 Functions.
insert(&
N.getFunction());
4178 if (Functions.empty())
4181 Module &M = *Functions.back()->getParent();
4233 std::string AAString;
aarch64 falkor hwpf fix Falkor HW Prefetch Fix Late Phase
static unsigned getIntrinsicID(const SDNode *N)
assert(UImm &&(UImm !=~static_cast< T >(0)) &&"Invalid immediate!")
amdgpu aa AMDGPU Address space based Alias Analysis Wrapper
MachineBasicBlock MachineBasicBlock::iterator DebugLoc DL
Expand Atomic instructions
This file contains the simple types necessary to represent the attributes associated with functions a...
static cl::opt< bool > AllowShallowWrappers("attributor-allow-shallow-wrappers", cl::Hidden, cl::desc("Allow the Attributor to create shallow " "wrappers for non-exact definitions."), cl::init(false))
bool canMarkAsVisited(const User *Usr)
#define VERBOSE_DEBUG_TYPE
static cl::opt< bool > EnableHeapToStack("enable-heap-to-stack-conversion", cl::init(true), cl::Hidden)
static cl::list< std::string > SeedAllowList("attributor-seed-allow-list", cl::Hidden, cl::desc("Comma separated list of attribute names that are " "allowed to be seeded."), cl::CommaSeparated)
static bool runAttributorOnFunctions(InformationCache &InfoCache, SetVector< Function * > &Functions, AnalysisGetter &AG, CallGraphUpdater &CGUpdater, bool DeleteFns, bool IsModulePass)
}
static bool getPotentialCopiesOfMemoryValue(Attributor &A, Ty &I, SmallSetVector< Value *, 4 > &PotentialCopies, SmallSetVector< Instruction *, 4 > *PotentialValueOrigins, const AbstractAttribute &QueryingAA, bool &UsedAssumedInformation, bool OnlyExact)
static bool runAttributorLightOnFunctions(InformationCache &InfoCache, SetVector< Function * > &Functions, AnalysisGetter &AG, CallGraphUpdater &CGUpdater, FunctionAnalysisManager &FAM, bool IsModulePass)
static cl::opt< unsigned, true > MaxInitializationChainLengthX("attributor-max-initialization-chain-length", cl::Hidden, cl::desc("Maximal number of chained initializations (to avoid stack overflows)"), cl::location(MaxInitializationChainLength), cl::init(1024))
static cl::opt< unsigned > MaxSpecializationPerCB("attributor-max-specializations-per-call-base", cl::Hidden, cl::desc("Maximal number of callees specialized for " "a call base"), cl::init(UINT32_MAX))
static cl::opt< bool > SimplifyAllLoads("attributor-simplify-all-loads", cl::Hidden, cl::desc("Try to simplify all loads."), cl::init(true))
static bool addIfNotExistent(LLVMContext &Ctx, const Attribute &Attr, AttributeSet AttrSet, bool ForceReplace, AttrBuilder &AB)
Return true if the information provided by Attr was added to the attribute set AttrSet.
static cl::opt< bool > ViewDepGraph("attributor-view-dep-graph", cl::Hidden, cl::desc("View the dependency graph."), cl::init(false))
static bool isEqualOrWorse(const Attribute &New, const Attribute &Old)
Return true if New is equal or worse than Old.
static cl::opt< bool > AllowDeepWrapper("attributor-allow-deep-wrappers", cl::Hidden, cl::desc("Allow the Attributor to use IP information " "derived from non-exact functions via cloning"), cl::init(false))
static cl::opt< bool > DumpDepGraph("attributor-dump-dep-graph", cl::Hidden, cl::desc("Dump the dependency graph to dot files."), cl::init(false))
static cl::opt< bool > PrintCallGraph("attributor-print-call-graph", cl::Hidden, cl::desc("Print Attributor's internal call graph"), cl::init(false))
static bool checkForAllInstructionsImpl(Attributor *A, InformationCache::OpcodeInstMapTy &OpcodeInstMap, function_ref< bool(Instruction &)> Pred, const AbstractAttribute *QueryingAA, const AAIsDead *LivenessAA, ArrayRef< unsigned > Opcodes, bool &UsedAssumedInformation, bool CheckBBLivenessOnly=false, bool CheckPotentiallyDead=false)
static cl::opt< bool > PrintDependencies("attributor-print-dep", cl::Hidden, cl::desc("Print attribute dependencies"), cl::init(false))
static bool isAssumedReadOnlyOrReadNone(Attributor &A, const IRPosition &IRP, const AbstractAttribute &QueryingAA, bool RequireReadNone, bool &IsKnown)
static cl::opt< std::string > DepGraphDotFileNamePrefix("attributor-depgraph-dot-filename-prefix", cl::Hidden, cl::desc("The prefix used for the CallGraph dot file names."))
static cl::opt< bool > AnnotateDeclarationCallSites("attributor-annotate-decl-cs", cl::Hidden, cl::desc("Annotate call sites of function declarations."), cl::init(false))
static cl::opt< unsigned > SetFixpointIterations("attributor-max-iterations", cl::Hidden, cl::desc("Maximal number of fixpoint iterations."), cl::init(32))
static cl::list< std::string > FunctionSeedAllowList("attributor-function-seed-allow-list", cl::Hidden, cl::desc("Comma separated list of function names that are " "allowed to be seeded."), cl::CommaSeparated)
static cl::opt< bool > EnableCallSiteSpecific("attributor-enable-call-site-specific-deduction", cl::Hidden, cl::desc("Allow the Attributor to do call site specific analysis"), cl::init(false))
static cl::opt< bool > CloseWorldAssumption("attributor-assume-closed-world", cl::Hidden, cl::desc("Should a closed world be assumed, or not. Default if not set."))
static GCRegistry::Add< ErlangGC > A("erlang", "erlang-compatible garbage collector")
static GCRegistry::Add< OcamlGC > B("ocaml", "ocaml 3.10-compatible GC")
This file provides interfaces used to build and manipulate a call graph, which is a very useful tool ...
This file contains the declarations for the subclasses of Constant, which represent the different fla...
This file provides an implementation of debug counters.
#define DEBUG_COUNTER(VARNAME, COUNTERNAME, DESC)
Contains a collection of routines for determining if a given instruction is guaranteed to execute if ...
ConstantRange Range(APInt(BitWidth, Low), APInt(BitWidth, High))
uint64_t IntrinsicInst * II
static constexpr StringLiteral Filename
FunctionAnalysisManager FAM
This file defines the PointerIntPair class.
static StringRef getName(Value *V)
Remove Loads Into Fake Uses
This file defines the SmallPtrSet class.
This file defines the 'Statistic' class, which is designed to be an easy way to expose various metric...
#define STATISTIC(VARNAME, DESC)
#define DEBUG_WITH_TYPE(TYPE,...)
DEBUG_WITH_TYPE macro - This macro should be used by passes to emit debug information.
void print(OutputBuffer &OB) const
static const fltSemantics & IEEEsingle()
Class for arbitrary precision integers.
CallBase * getInstruction() const
Return the underlying instruction.
bool isCallbackCall() const
Return true if this ACS represents a callback call.
const Use & getCalleeUseForCallback() const
Return the use of the callee value in the underlying instruction.
static LLVM_ABI void getCallbackUses(const CallBase &CB, SmallVectorImpl< const Use * > &CallbackUses)
Add operand uses of CB that represent callback uses into CallbackUses.
bool isCallee(Value::const_user_iterator UI) const
Return true if UI is the use that defines the callee of this ACS.
Value * getCallArgOperand(Argument &Arg) const
Return the operand of the underlying instruction associated with Arg.
int getCallArgOperandNo(Argument &Arg) const
Return the operand index of the underlying instruction associated with Arg.
unsigned getNumArgOperands() const
Return the number of parameters of the callee.
Function * getCalledFunction() const
Return the function being called if this is a direct call, otherwise return null (if it's an indirect...
This templated class represents "all analyses that operate over <aparticular IR unit>" (e....
PassT::Result & getResult(IRUnitT &IR, ExtraArgTs... ExtraArgs)
Get the result of an analysis pass for a given IR unit.
This class represents an incoming formal argument to a Function.
const Function * getParent() const
unsigned getArgNo() const
Return the index of this formal argument in its containing function.
Represent a constant reference to an array (0 or more elements consecutively in memory),...
size_t size() const
Get the array size.
bool empty() const
Check if the array is empty.
This class stores enough information to efficiently remove some attributes from an existing AttrBuild...
This class holds the attributes for a particular argument, parameter, function, or return value.
LLVM_ABI MemoryEffects getMemoryEffects() const
LLVM_ABI bool hasAttribute(Attribute::AttrKind Kind) const
Return true if the attribute exists in this set.
LLVM_ABI Attribute getAttribute(Attribute::AttrKind Kind) const
Return the attribute object.
Functions, function parameters, and return types can have attributes to indicate how they should be t...
LLVM_ABI bool isStringAttribute() const
Return true if the attribute is a string (target-dependent) attribute.
LLVM_ABI bool isEnumAttribute() const
Return true if the attribute is an Attribute::AttrKind type.
LLVM_ABI bool isIntAttribute() const
Return true if the attribute is an integer attribute.
LLVM_ABI uint64_t getValueAsInt() const
Return the attribute's value as an integer.
LLVM_ABI bool isConstantRangeAttribute() const
Return true if the attribute is a ConstantRange attribute.
LLVM_ABI StringRef getKindAsString() const
Return the attribute's kind as a string.
static LLVM_ABI Attribute get(LLVMContext &Context, AttrKind Kind, uint64_t Val=0)
Return a uniquified Attribute object.
LLVM_ABI Attribute::AttrKind getKindAsEnum() const
Return the attribute's kind as an enum (Attribute::AttrKind).
LLVM_ABI MemoryEffects getMemoryEffects() const
Returns memory effects.
LLVM_ABI StringRef getValueAsString() const
Return the attribute's value as a string.
AttrKind
This enumeration lists the attributes that can be associated with parameters, function results,...
@ None
No attributes have been set.
LLVM Basic Block Representation.
const Function * getParent() const
Return the enclosing method, or null if none.
static BasicBlock * Create(LLVMContext &Context, const Twine &Name="", Function *Parent=nullptr, BasicBlock *InsertBefore=nullptr)
Creates a new BasicBlock.
const Instruction & front() const
LLVM_ABI const BasicBlock * getUniquePredecessor() const
Return the predecessor of this block if it has a unique predecessor block.
const Instruction * getTerminator() const LLVM_READONLY
Returns the terminator instruction; assumes that the block is well-formed.
Represents analyses that only rely on functions' control flow.
Base class for all callable instructions (InvokeInst and CallInst) Holds everything related to callin...
void setCallingConv(CallingConv::ID CC)
void addFnAttr(Attribute::AttrKind Kind)
Adds the attribute to the function.
LLVM_ABI void getOperandBundlesAsDefs(SmallVectorImpl< OperandBundleDef > &Defs) const
Return the list of operand bundles attached to this instruction as a vector of OperandBundleDefs.
CallingConv::ID getCallingConv() const
LLVM_ABI bool isMustTailCall() const
Tests if this call site must be tail call optimized.
Value * getCalledOperand() const
void setAttributes(AttributeList A)
Set the attributes for this call.
unsigned arg_size() const
AttributeList getAttributes() const
Return the attributes for this call.
LLVM_ABI Function * getCaller()
Helper to get the caller (the parent function).
Wrapper to unify "old style" CallGraph and "new style" LazyCallGraph.
LLVM_ABI void replaceFunctionWith(Function &OldFn, Function &NewFn)
Replace OldFn in the call graph (and SCC) with NewFn.
LLVM_ABI void reanalyzeFunction(Function &Fn)
After an CGSCC pass changes a function in ways that affect the call graph, this method can be called ...
void initialize(LazyCallGraph &LCG, LazyCallGraph::SCC &SCC, CGSCCAnalysisManager &AM, CGSCCUpdateResult &UR)
Initializers for usage outside of a CGSCC pass, inside a CGSCC pass in the old and new pass manager (...
This class represents a function call, abstracting a target machine's calling convention.
static CallInst * Create(FunctionType *Ty, Value *F, const Twine &NameStr="", InsertPosition InsertBefore=nullptr)
void setTailCall(bool IsTc=true)
A constant value that is initialized with an expression using other constant values.
static LLVM_ABI Constant * getPointerCast(Constant *C, Type *Ty)
Create a BitCast, AddrSpaceCast, or a PtrToInt cast constant expression.
static LLVM_ABI Constant * getTrunc(Constant *C, Type *Ty, bool OnlyIfReduced=false)
LLVM_ABI void print(raw_ostream &OS) const
Print out the bounds to a stream.
This is an important base class in LLVM.
static LLVM_ABI Constant * getNullValue(Type *Ty)
Constructor to create a '0' constant of arbitrary type.
A parsed version of the target data layout string in and methods for querying it.
static bool shouldExecute(CounterInfo &Counter)
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.
Implements a dense probed hash-table based set.
Analysis pass which computes a DominatorTree.
Concrete subclass of DominatorTreeBase that is used to compute a normal dominator tree.
A proxy from a FunctionAnalysisManager to an SCC.
Class to represent function types.
static LLVM_ABI FunctionType * get(Type *Result, ArrayRef< Type * > Params, bool isVarArg)
This static method is the primary way of constructing a FunctionType.
void setSubprogram(DISubprogram *SP)
Set the attached subprogram.
static Function * Create(FunctionType *Ty, LinkageTypes Linkage, unsigned AddrSpace, const Twine &N="", Module *M=nullptr)
void splice(Function::iterator ToIt, Function *FromF)
Transfer all blocks from FromF to this function at ToIt.
const BasicBlock & getEntryBlock() const
FunctionType * getFunctionType() const
Returns the FunctionType for me.
iterator_range< arg_iterator > args()
DISubprogram * getSubprogram() const
Get the attached subprogram.
MemoryEffects getMemoryEffects() const
bool hasParamAttribute(unsigned ArgNo, Attribute::AttrKind Kind) const
check if an attributes is in the list of attributes.
AttributeList getAttributes() const
Return the attribute list for this Function.
void setAttributes(AttributeList Attrs)
Set the attribute list for this Function.
LLVMContext & getContext() const
getContext - Return a reference to the LLVMContext associated with this function.
Type * getReturnType() const
Returns the type of the ret val.
void setMemoryEffects(MemoryEffects ME)
Argument * getArg(unsigned i) const
bool isVarArg() const
isVarArg - Return true if this function takes a variable number of arguments.
void copyAttributesFrom(const Function *Src)
copyAttributesFrom - copy all additional attributes (those not needed to create a Function) from the ...
bool hasMetadata() const
Return true if this GlobalObject has any metadata attached to it.
LLVM_ABI void addMetadata(unsigned KindID, MDNode &MD)
Add a metadata attachment.
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
bool hasLocalLinkage() const
void setLinkage(LinkageTypes LT)
unsigned getAddressSpace() const
Module * getParent()
Get the module that this global value is contained inside of...
void setDSOLocal(bool Local)
PointerType * getType() const
Global values are always pointers.
@ DefaultVisibility
The GV is visible.
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.
@ PrivateLinkage
Like Internal, but omit from symbol table.
@ InternalLinkage
Rename collisions when linking (static functions).
LLVM_ABI InstListType::iterator eraseFromParent()
This method unlinks 'this' from the containing basic block and deletes it.
LLVM_ABI const Function * getFunction() const
Return the function this instruction belongs to.
LLVM_ABI void copyMetadata(const Instruction &SrcInst, ArrayRef< unsigned > WL=ArrayRef< unsigned >())
Copy metadata from SrcInst to this instruction.
static InvokeInst * Create(FunctionType *Ty, Value *Func, BasicBlock *IfNormal, BasicBlock *IfException, ArrayRef< Value * > Args, const Twine &NameStr, InsertPosition InsertBefore=nullptr)
This is an important class for using LLVM in a threaded context.
A node in the call graph.
An SCC of the call graph.
A lazily constructed view of the call graph of a module.
An instruction for reading from memory.
This is the common base class for memset/memcpy/memmove.
This class wraps the llvm.memcpy/memmove intrinsics.
static MemoryEffectsBase argMemOnly(ModRefInfo MR=ModRefInfo::ModRef)
bool doesAccessArgPointees() const
Whether this function may access argument memory.
static LLVM_ABI MemoryLocation getForSource(const MemTransferInst *MTI)
Return a location representing the source of a memory transfer.
static LLVM_ABI MemoryLocation getForDest(const MemIntrinsic *MI)
Return a location representing the destination of a memory set or transfer.
static LLVM_ABI std::optional< MemoryLocation > getOrNone(const Instruction *Inst)
A Module instance is used to store all the information related to an LLVM module.
const FunctionListType & getFunctionList() const
Get the Module's list of functions (constant).
PointerIntPair - This class implements a pair of a pointer and small integer.
void * getOpaqueValue() const
PointerTy getPointer() const
static LLVM_ABI PoisonValue * get(Type *T)
Static factory methods - Return an 'poison' object of the specified 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.
PreservedAnalyses & preserveSet()
Mark an analysis set as preserved.
PreservedAnalyses & preserve()
Mark an analysis as preserved.
Return a value (possibly void), from a function.
static ReturnInst * Create(LLVMContext &C, Value *retVal=nullptr, InsertPosition InsertBefore=nullptr)
A vector that has set insertion semantics.
ArrayRef< value_type > getArrayRef() const
bool remove(const value_type &X)
Remove an item from the set vector.
size_type size() const
Determine the number of elements in the SetVector.
void insert_range(Range &&R)
size_type count(const_arg_type key) const
Count the number of elements of a given key in the SetVector.
typename vector_type::const_iterator iterator
void clear()
Completely clear the SetVector.
iterator begin()
Get an iterator to the beginning of the SetVector.
bool insert(const value_type &X)
Insert a new element into the SetVector.
A templated base class for SmallPtrSet which provides the typesafe interface that is common across al...
size_type count(ConstPtrType Ptr) const
count - Return 1 if the specified pointer is in the set, 0 otherwise.
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.
A SetVector that performs no allocations if smaller than a certain size.
This class consists of common code factored out of the SmallVector class to reduce code duplication b...
void reserve(size_type N)
void append(ItTy in_start, ItTy in_end)
Add the specified range to the end of the SmallVector.
void push_back(const T &Elt)
This is a 'vector' (really, a variable-sized array), optimized for the case when the array is small.
An instruction for storing to memory.
Represent a constant reference to a string, i.e.
A visitor class for IR positions.
LLVM_ABI SubsumingPositionIterator(const IRPosition &IRP)
Provides information about what library functions are available for the current target.
The TimeTraceScope is a helper class to call the begin and end functions of the time trace profiler.
Triple - Helper class for working with autoconf configuration names.
The instances of the Type class are immutable: once they are created, they are never changed.
bool isPointerTy() const
True if this is an instance of PointerType.
bool isVoidTy() const
Return true if this is 'void'.
static LLVM_ABI UndefValue * get(Type *T)
Static factory methods - Return an 'undef' object of the specified type.
A Use represents the edge between a Value definition and its users.
LLVM Value Representation.
Type * getType() const
All values are typed, get the type of this value.
LLVM_ABI void replaceAllUsesWith(Value *V)
Change all uses of this to point to a new Value.
LLVMContext & getContext() const
All values hold a context through their type.
LLVM_ABI const Value * stripPointerCasts() const
Strip off pointer casts, all-zero GEPs and address space casts.
iterator_range< use_iterator > uses()
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.
An efficient, type-erasing, non-owning reference to a callable.
const ParentTy * getParent() const
self_iterator getIterator()
NodeTy * getNextNode()
Get the next node, or nullptr for the list tail.
iterator insert(iterator where, pointer New)
A raw_ostream that writes to a file descriptor.
This class implements an extremely fast bulk output stream that can only output to a stream.
A raw_ostream that writes to an std::string.
#define llvm_unreachable(msg)
Marks that the current location is not supposed to be reachable.
Abstract Attribute helper functions.
LLVM_ABI bool isAssumedReadNone(Attributor &A, const IRPosition &IRP, const AbstractAttribute &QueryingAA, bool &IsKnown)
Return true if IRP is readnone.
LLVM_ABI bool isAssumedReadOnly(Attributor &A, const IRPosition &IRP, const AbstractAttribute &QueryingAA, bool &IsKnown)
Return true if IRP is readonly.
LLVM_ABI std::optional< Value * > combineOptionalValuesInAAValueLatice(const std::optional< Value * > &A, const std::optional< Value * > &B, Type *Ty)
Return the combination of A and B such that the result is a possible value of both.
LLVM_ABI bool isValidAtPosition(const ValueAndContext &VAC, InformationCache &InfoCache)
Return true if the value of VAC is a valid at the position of VAC, that is a constant,...
LLVM_ABI bool isAssumedThreadLocalObject(Attributor &A, Value &Obj, const AbstractAttribute &QueryingAA)
Return true if Obj is assumed to be a thread local object.
LLVM_ABI bool isGPUConstantAddressSpace(const Module &M, unsigned AS)
Check if the given address space AS corresponds to a GPU constant address space for the target triple...
LLVM_ABI bool isGPUGenericAddressSpace(const Module &M, unsigned AS)
Check if the given address space AS corresponds to a GPU generic address space for the target triple ...
LLVM_ABI bool isDynamicallyUnique(Attributor &A, const AbstractAttribute &QueryingAA, const Value &V, bool ForAnalysisOnly=true)
Return true if V is dynamically unique, that is, there are no two "instances" of V at runtime with di...
LLVM_ABI bool getPotentialCopiesOfStoredValue(Attributor &A, StoreInst &SI, SmallSetVector< Value *, 4 > &PotentialCopies, const AbstractAttribute &QueryingAA, bool &UsedAssumedInformation, bool OnlyExact=false)
Collect all potential values of the one stored by SI into PotentialCopies.
LLVM_ABI bool isGPUSharedAddressSpace(const Module &M, unsigned AS)
Check if the given address space AS corresponds to a GPU shared address space for the target triple i...
LLVM_ABI bool isGPULocalAddressSpace(const Module &M, unsigned AS)
Check if the given address space AS corresponds to a GPU local/private address space for the target t...
LLVM_ABI bool isPotentiallyAffectedByBarrier(Attributor &A, const Instruction &I, const AbstractAttribute &QueryingAA)
Return true if I is potentially affected by a barrier.
SmallPtrSet< Instruction *, 4 > InstExclusionSetTy
LLVM_ABI bool isGPU(const Module &M)
Return true iff M target a GPU (and we can use GPU AS reasoning).
LLVM_ABI Constant * getInitialValueForObj(Attributor &A, const AbstractAttribute &QueryingAA, Value &Obj, Type &Ty, const TargetLibraryInfo *TLI, const DataLayout &DL, RangeTy *RangePtr=nullptr)
Return the initial value of Obj with type Ty if that is a constant.
ValueScope
Flags to distinguish intra-procedural queries from potentially inter-procedural queries.
LLVM_ABI bool isValidInScope(const Value &V, const Function *Scope)
Return true if V is a valid value in Scope, that is a constant or an instruction/argument of Scope.
LLVM_ABI bool isGPUGlobalAddressSpace(const Module &M, unsigned AS)
Check if the given address space AS corresponds to a GPU global address space for the target triple i...
LLVM_ABI bool isPotentiallyReachable(Attributor &A, const Instruction &FromI, const Instruction &ToI, const AbstractAttribute &QueryingAA, const AA::InstExclusionSetTy *ExclusionSet=nullptr, std::function< bool(const Function &F)> GoBackwardsCB=nullptr)
Return true if ToI is potentially reachable from FromI without running into any instruction in Exclus...
LLVM_ABI bool isNoSyncInst(Attributor &A, const Instruction &I, const AbstractAttribute &QueryingAA)
Return true if I is a nosync instruction.
bool hasAssumedIRAttr(Attributor &A, const AbstractAttribute *QueryingAA, const IRPosition &IRP, DepClassTy DepClass, bool &IsKnown, bool IgnoreSubsumingPositions=false, const AAType **AAPtr=nullptr)
Helper to avoid creating an AA for IR Attributes that might already be set.
LLVM_ABI bool getPotentiallyLoadedValues(Attributor &A, LoadInst &LI, SmallSetVector< Value *, 4 > &PotentialValues, SmallSetVector< Instruction *, 4 > &PotentialValueOrigins, const AbstractAttribute &QueryingAA, bool &UsedAssumedInformation, bool OnlyExact=false)
Collect all potential values LI could read into PotentialValues.
LLVM_ABI Value * getWithType(Value &V, Type &Ty)
Try to convert V to type Ty without introducing new instructions.
constexpr char Attrs[]
Key for Kernel::Metadata::mAttrs.
@ C
The default llvm calling convention, compatible with C.
@ BasicBlock
Various leaf nodes.
initializer< Ty > init(const Ty &Val)
LocationClass< Ty > location(Ty &L)
DiagnosticInfoOptimizationBase::Argument NV
NodeAddr< UseNode * > Use
friend class Instruction
Iterator for Instructions in a `BasicBlock.
@ OF_TextWithCRLF
The file should be opened in text mode and use a carriage linefeed '\r '.
This is an optimization pass for GlobalISel generic memory operations.
FunctionAddr VTableAddr Value
bool all_of(R &&range, UnaryPredicate P)
Provide wrappers to std::all_of which take ranges instead of having to pass begin/end explicitly.
LLVM_ABI Constant * getInitialValueOfAllocation(const Value *V, const TargetLibraryInfo *TLI, Type *Ty)
If this is a call to an allocation function that initializes memory to a fixed value,...
Printable print(const GCNRegPressure &RP, const GCNSubtarget *ST=nullptr, unsigned DynamicVGPRBlockSize=0)
LLVM_ABI bool RecursivelyDeleteTriviallyDeadInstructions(Value *V, const TargetLibraryInfo *TLI=nullptr, MemorySSAUpdater *MSSAU=nullptr, std::function< void(Value *)> AboutToDeleteCallback=std::function< void(Value *)>())
If the specified value is a trivially dead instruction, delete it.
LLVM_ABI unsigned MaxInitializationChainLength
The value passed to the line option that defines the maximal initialization chain length.
LLVM_ABI bool ConstantFoldTerminator(BasicBlock *BB, bool DeleteDeadConditions=false, const TargetLibraryInfo *TLI=nullptr, DomTreeUpdater *DTU=nullptr)
If a terminator instruction is predicated on a constant value, convert it into an unconditional branc...
APInt operator&(APInt a, const APInt &b)
LLVM_ABI void detachDeadBlocks(ArrayRef< BasicBlock * > BBs, SmallVectorImpl< DominatorTree::UpdateType > *Updates, bool KeepOneInputPHIs=false)
Replace contents of every block in BBs with single unreachable instruction.
decltype(auto) dyn_cast(const From &Val)
dyn_cast<X> - Return the argument parameter cast to the specified type.
DenseMap< AssumeInst *, MinMax > Assume2KnowledgeMap
A mapping from intrinsics (=llvm.assume calls) to a value range (=knowledge) that is encoded in them.
LLVM_ABI bool verifyFunction(const Function &F, raw_ostream *OS=nullptr)
Check a function for errors, useful for use when debugging a pass.
LLVM_ABI CallInst * changeToCall(InvokeInst *II, DomTreeUpdater *DTU=nullptr)
This function converts the specified invoke into a normal call.
LLVM_ABI raw_fd_ostream & outs()
This returns a reference to a raw_fd_ostream for standard output.
auto dyn_cast_if_present(const Y &Val)
dyn_cast_if_present<X> - Functionally identical to dyn_cast, except that a null (or none in the case ...
iterator_range< T > make_range(T x, T y)
Convenience function for iterating over sub-ranges.
InnerAnalysisManagerProxy< FunctionAnalysisManager, Module > FunctionAnalysisManagerModuleProxy
Provide the FunctionAnalysisManager to Module proxy.
LLVM_ABI bool isNoAliasCall(const Value *V)
Return true if this pointer is returned by a noalias function.
MemoryEffectsBase< IRMemLocation > MemoryEffects
Summary of how a function affects memory in the program.
raw_ostream & WriteGraph(raw_ostream &O, const GraphType &G, bool ShortNames=false, const Twine &Title="")
bool isa_and_nonnull(const Y &Val)
AnalysisManager< LazyCallGraph::SCC, LazyCallGraph & > CGSCCAnalysisManager
The CGSCC analysis manager.
LLVM_ABI InlineResult isInlineViable(Function &Callee)
Check if it is mechanically possible to inline the function Callee, based on the contents of the func...
auto dyn_cast_or_null(const Y &Val)
bool any_of(R &&range, UnaryPredicate P)
Provide wrappers to std::any_of which take ranges instead of having to pass begin/end explicitly.
LLVM_ABI bool isInstructionTriviallyDead(Instruction *I, const TargetLibraryInfo *TLI=nullptr)
Return true if the result produced by the instruction is not used, and the instruction will return.
LLVM_ABI Constant * ConstantFoldLoadFromUniformValue(Constant *C, Type *Ty, const DataLayout &DL)
If C is a uniform value where all bits are the same (either all zero, all ones, all undef or all pois...
PotentialValuesState< std::pair< AA::ValueAndContext, AA::ValueScope > > PotentialLLVMValuesState
decltype(auto) get(const PointerIntPair< PointerTy, IntBits, IntType, PtrTraits, Info > &Pair)
LLVM_ABI bool NullPointerIsDefined(const Function *F, unsigned AS=0)
Check whether null pointer dereferencing is considered undefined behavior for a given function or an ...
LLVM_ABI raw_ostream & dbgs()
dbgs() - This returns a reference to a raw_ostream for debugging messages.
LLVM_ABI bool AreStatisticsEnabled()
Check if statistics are enabled.
LLVM_ABI Constant * ConstantFoldLoadFromConst(Constant *C, Type *Ty, const APInt &Offset, const DataLayout &DL)
Extract value of C at the given Offset reinterpreted as Ty.
class LLVM_GSL_OWNER SmallVector
Forward declaration of SmallVector so that calculateSmallVectorDefaultInlinedElements can reference s...
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...
@ Success
The lock was released successfully.
LLVM_ABI unsigned changeToUnreachable(Instruction *I, bool PreserveLCSSA=false, DomTreeUpdater *DTU=nullptr, MemorySSAUpdater *MSSAU=nullptr)
Insert an unreachable instruction before the specified instruction, making it and the rest of the cod...
LLVM_ABI raw_fd_ostream & errs()
This returns a reference to a raw_ostream for standard error.
@ Global
Append to llvm.global_dtors.
LLVM_ABI BasicBlock * SplitBlockPredecessors(BasicBlock *BB, ArrayRef< BasicBlock * > Preds, const char *Suffix, DominatorTree *DT, LoopInfo *LI=nullptr, MemorySSAUpdater *MSSAU=nullptr, bool PreserveLCSSA=false)
This method introduces at least one new basic block into the function and moves some of the predecess...
PotentialValuesState< APInt > PotentialConstantIntValuesState
bool operator&=(SparseBitVector< ElementSize > *LHS, const SparseBitVector< ElementSize > &RHS)
LLVM_ABI bool isPotentiallyReachable(const Instruction *From, const Instruction *To, const SmallPtrSetImpl< BasicBlock * > *ExclusionSet=nullptr, const DominatorTree *DT=nullptr, const LoopInfo *LI=nullptr, const CycleInfo *CI=nullptr)
Determine whether instruction 'To' is reachable from 'From', without passing through any blocks in Ex...
DWARFExpression::Operation Op
void ViewGraph(const GraphType &G, const Twine &Name, bool ShortNames=false, const Twine &Title="", GraphProgram::Name Program=GraphProgram::DOT)
ViewGraph - Emit a dot graph, run 'dot', run gv on the postscript file, then cleanup.
raw_ostream & operator<<(raw_ostream &OS, const APFixedPoint &FX)
ArrayRef(const T &OneElt) -> ArrayRef< T >
ValueMap< const Value *, WeakTrackingVH > ValueToValueMapTy
LLVM_ABI void CloneFunctionInto(Function *NewFunc, const Function *OldFunc, ValueToValueMapTy &VMap, CloneFunctionChangeType Changes, SmallVectorImpl< ReturnInst * > &Returns, const char *NameSuffix="", ClonedCodeInfo *CodeInfo=nullptr, ValueMapTypeRemapper *TypeMapper=nullptr, ValueMaterializer *Materializer=nullptr)
Clone OldFunc into NewFunc, transforming the old arguments into references to VMap values.
decltype(auto) cast(const From &Val)
cast<X> - Return the argument parameter cast to the specified type.
void erase_if(Container &C, UnaryPredicate P)
Provide a container algorithm similar to C++ Library Fundamentals v2's erase_if which is equivalent t...
iterator_range< pointer_iterator< WrappedIteratorT > > make_pointer_range(RangeT &&Range)
LLVM_ABI bool isAllocationFn(const Value *V, const TargetLibraryInfo *TLI)
Tests if a value is a call or invoke to a library function that allocates or reallocates memory (eith...
bool is_contained(R &&Range, const E &Element)
Returns true if Element is found in Range.
LLVM_ABI void fillMapFromAssume(AssumeInst &Assume, RetainedKnowledgeMap &Result)
Insert into the map all the informations contained in the operand bundles of the llvm....
bool operator|=(SparseBitVector< ElementSize > &LHS, const SparseBitVector< ElementSize > *RHS)
LLVM_ABI Constant * ConstantFoldCastInstruction(unsigned opcode, Constant *V, Type *DestTy)
@ OPTIONAL
The target may be valid if the source is not.
@ NONE
Do not track a dependence between source and target.
@ REQUIRED
The target cannot be valid if the source is not.
AnalysisManager< Function > FunctionAnalysisManager
Convenience typedef for the Function analysis manager.
APInt operator|(APInt a, const APInt &b)
BumpPtrAllocatorImpl<> BumpPtrAllocator
The standard BumpPtrAllocator which just uses the default template parameters.
AnalysisManager< Module > ModuleAnalysisManager
Convenience typedef for the Module analysis manager.
static LLVM_ABI const char ID
Unique ID (due to the unique address)
static LLVM_ABI const char ID
Unique ID (due to the unique address)
DepSetTy Deps
Set of dependency graph nodes which should be updated if this one is updated.
PointerIntPair< AADepGraphNode *, 1 > DepTy
The data structure for the dependency graph.
LLVM_ABI void viewGraph()
AADepGraphNode SyntheticRoot
There is no root node for the dependency graph.
LLVM_ABI void print()
Print dependency graph.
LLVM_ABI void dumpGraph()
Dump graph to file.
AADepGraphNode * GetEntryNode()
An abstract interface to track if a value leaves it's defining function instance.
bool isAssumedUniqueForAnalysis() const
Return true if we assume that the underlying value is unique in its scope wrt.
An abstract Attribute for computing reachability between functions.
static LLVM_ABI const char ID
Unique ID (due to the unique address)
An abstract interface to determine reachability of point A to B.
static LLVM_ABI const char ID
Unique ID (due to the unique address)
An abstract interface for liveness abstract attribute.
virtual bool isKnownDead() const =0
Returns true if the underlying value is known dead.
virtual bool isAssumedDead() const =0
The query functions are protected such that other attributes need to go through the Attributor interf...
virtual bool isRemovableStore() const
Return true if the underlying value is a store that is known to be removable.
static bool mayCatchAsynchronousExceptions(const Function &F)
Determine if F might catch asynchronous exceptions.
An abstract interface for memory access kind related attributes (readnone/readonly/writeonly).
static LLVM_ABI const char ID
Unique ID (due to the unique address)
An abstract interface for all memory location attributes (readnone/argmemonly/inaccessiblememonly/ina...
static LLVM_ABI const char ID
Unique ID (due to the unique address)
static LLVM_ABI const char ID
Unique ID (due to the unique address)
static LLVM_ABI const char ID
Unique ID (due to the unique address)
static LLVM_ABI const char ID
Unique ID (due to the unique address)
static LLVM_ABI const char ID
Unique ID (due to the unique address)
static LLVM_ABI const char ID
Unique ID (due to the unique address)
static LLVM_ABI const char ID
Unique ID (due to the unique address)
static LLVM_ABI const char ID
Unique ID (due to the unique address)
static LLVM_ABI bool isNonRelaxedAtomic(const Instruction *I)
Helper function used to determine whether an instruction is non-relaxed atomic.
static LLVM_ABI const char ID
Unique ID (due to the unique address)
static LLVM_ABI const char ID
Unique ID (due to the unique address)
bool isWrittenValueUnknown() const
Return true if the value written cannot be determined at all.
std::optional< Value * > getContent() const
Return the written value which can be llvm::null if it is not yet determined.
bool isWriteOrAssumption() const
Return true if this is a write access.
bool isRead() const
Return true if this is a read access.
Value * getWrittenValue() const
Return the value writen, if any.
Instruction * getLocalInst() const
Return the instruction that causes the access with respect to the local scope of the associated attri...
Instruction * getRemoteInst() const
Return the actual instruction that causes the access.
bool isWrittenValueYetUndetermined() const
Return true if the value written is not known yet.
AccessKind getKind() const
Return the access kind.
An abstract interface for struct information.
static LLVM_ABI Value * getSingleValue(Attributor &A, const AbstractAttribute &AA, const IRPosition &IRP, SmallVectorImpl< AA::ValueAndContext > &Values)
Extract the single value in Values if any.
An abstract attribute for getting all assumption underlying objects.
static LLVM_ABI const char ID
Unique ID (due to the unique address)
static LLVM_ABI const char ID
Unique ID (due to the unique address)
Helper to represent an access offset and size, with logic to deal with uncertainty and check for over...
bool offsetOrSizeAreUnknown() const
Return true if offset or size are unknown.
const Instruction * getCtxI() const
Base struct for all "concrete attribute" deductions.
ChangeStatus update(Attributor &A)
Hook for the Attributor to trigger an update of the internal state.
friend struct Attributor
}
virtual void printWithDeps(raw_ostream &OS) const
void print(raw_ostream &OS) const
Helper functions, for debug purposes only.
virtual StateType & getState()=0
Return the internal abstract state for inspection.
virtual const std::string getAsStr(Attributor *A) const =0
This function should return the "summarized" assumed state as string.
virtual ChangeStatus updateImpl(Attributor &A)=0
The actual update/transfer function which has to be implemented by the derived classes.
const IRPosition & getIRPosition() const
Return an IR position, see struct IRPosition.
An interface to query the internal state of an abstract attribute.
virtual ChangeStatus indicatePessimisticFixpoint()=0
Indicate that the abstract state should converge to the pessimistic state.
virtual bool isAtFixpoint() const =0
Return if this abstract state is fixed, thus does not need to be updated if information changes as it...
virtual bool isValidState() const =0
Return if this abstract state is in a valid state.
Wrapper for FunctionAnalysisManager.
LLVM_ABI PreservedAnalyses run(LazyCallGraph::SCC &C, CGSCCAnalysisManager &AM, LazyCallGraph &CG, CGSCCUpdateResult &UR)
void populateAll() const
Force populate the entire call graph.
Configuration for the Attributor.
std::optional< unsigned > MaxFixpointIterations
Maximum number of iterations to run until fixpoint.
LLVM_ABI PreservedAnalyses run(LazyCallGraph::SCC &C, CGSCCAnalysisManager &AM, LazyCallGraph &CG, CGSCCUpdateResult &UR)
LLVM_ABI PreservedAnalyses run(Module &M, ModuleAnalysisManager &AM)
LLVM_ABI PreservedAnalyses run(Module &M, ModuleAnalysisManager &AM)
Helper struct used in the communication between an abstract attribute (AA) that wants to change the s...
std::function< void( const ArgumentReplacementInfo &, Function &, Function::arg_iterator)> CalleeRepairCBTy
Callee repair callback type.
std::function< void(const ArgumentReplacementInfo &, AbstractCallSite, SmallVectorImpl< Value * > &)> ACSRepairCBTy
Abstract call site (ACS) repair callback type.
The fixpoint analysis framework that orchestrates the attribute deduction.
LLVM_ABI bool registerFunctionSignatureRewrite(Argument &Arg, ArrayRef< Type * > ReplacementTypes, ArgumentReplacementInfo::CalleeRepairCBTy &&CalleeRepairCB, ArgumentReplacementInfo::ACSRepairCBTy &&ACSRepairCB)
Register a rewrite for a function signature.
LLVM_ABI bool checkForAllCallees(function_ref< bool(ArrayRef< const Function * > Callees)> Pred, const AbstractAttribute &QueryingAA, const CallBase &CB)
Check Pred on all potential Callees of CB.
bool isModulePass() const
Return true if this is a module pass, false otherwise.
LLVM_ABI bool isValidFunctionSignatureRewrite(Argument &Arg, ArrayRef< Type * > ReplacementTypes)
Check if we can rewrite a function signature.
static LLVM_ABI bool isInternalizable(Function &F)
Returns true if the function F can be internalized.
LLVM_ABI ChangeStatus removeAttrs(const IRPosition &IRP, ArrayRef< Attribute::AttrKind > AttrKinds)
Remove all AttrKinds attached to IRP.
void emitRemark(Instruction *I, StringRef RemarkName, RemarkCallBack &&RemarkCB) const
Emit a remark generically.
bool isRunOn(Function &Fn) const
Return true if we derive attributes for Fn.
LLVM_ABI bool isAssumedDead(const AbstractAttribute &AA, const AAIsDead *LivenessAA, bool &UsedAssumedInformation, bool CheckBBLivenessOnly=false, DepClassTy DepClass=DepClassTy::OPTIONAL)
Return true if AA (or its context instruction) is assumed dead.
LLVM_ABI bool checkForAllInstructions(function_ref< bool(Instruction &)> Pred, const Function *Fn, const AbstractAttribute *QueryingAA, ArrayRef< unsigned > Opcodes, bool &UsedAssumedInformation, bool CheckBBLivenessOnly=false, bool CheckPotentiallyDead=false)
Check Pred on all instructions in Fn with an opcode present in Opcodes.
LLVM_ABI void recordDependence(const AbstractAttribute &FromAA, const AbstractAttribute &ToAA, DepClassTy DepClass)
Explicitly record a dependence from FromAA to ToAA, that is if FromAA changes ToAA should be updated ...
static LLVM_ABI void createShallowWrapper(Function &F)
Create a shallow wrapper for F such that F has internal linkage afterwards.
const AAType * getAAFor(const AbstractAttribute &QueryingAA, const IRPosition &IRP, DepClassTy DepClass)
Lookup an abstract attribute of type AAType at position IRP.
std::optional< Value * > getAssumedSimplified(const IRPosition &IRP, const AbstractAttribute &AA, bool &UsedAssumedInformation, AA::ValueScope S)
If V is assumed simplified, return it, if it is unclear yet, return std::nullopt, otherwise return nu...
static LLVM_ABI Function * internalizeFunction(Function &F, bool Force=false)
Make another copy of the function F such that the copied version has internal linkage afterwards and ...
bool isFunctionIPOAmendable(const Function &F)
Determine whether the function F is IPO amendable.
const AAType * getOrCreateAAFor(IRPosition IRP, const AbstractAttribute *QueryingAA, DepClassTy DepClass, bool ForceUpdate=false, bool UpdateAfterInit=true)
The version of getAAFor that allows to omit a querying abstract attribute.
LLVM_ABI bool checkForAllReadWriteInstructions(function_ref< bool(Instruction &)> Pred, AbstractAttribute &QueryingAA, bool &UsedAssumedInformation)
Check Pred on all Read/Write instructions.
LLVM_ABI bool checkForAllReturnedValues(function_ref< bool(Value &)> Pred, const AbstractAttribute &QueryingAA, AA::ValueScope S=AA::ValueScope::Intraprocedural, bool RecurseForSelectAndPHI=true)
Check Pred on all values potentially returned by the function associated with QueryingAA.
LLVM_ABI bool isClosedWorldModule() const
Return true if the module contains the whole world, thus, no outside functions exist.
LLVM_ABI std::optional< Constant * > getAssumedConstant(const IRPosition &IRP, const AbstractAttribute &AA, bool &UsedAssumedInformation)
If IRP is assumed to be a constant, return it, if it is unclear yet, return std::nullopt,...
LLVM_ABI Attributor(SetVector< Function * > &Functions, InformationCache &InfoCache, AttributorConfig Configuration)
Constructor.
LLVM_ABI void getAttrs(const IRPosition &IRP, ArrayRef< Attribute::AttrKind > AKs, SmallVectorImpl< Attribute > &Attrs, bool IgnoreSubsumingPositions=false)
Return the attributes of any kind in AKs existing in the IR at a position that will affect this one.
InformationCache & getInfoCache()
Return the internal information cache.
LLVM_ABI std::optional< Value * > translateArgumentToCallSiteContent(std::optional< Value * > V, CallBase &CB, const AbstractAttribute &AA, bool &UsedAssumedInformation)
Translate V from the callee context into the call site context.
LLVM_ABI bool checkForAllUses(function_ref< bool(const Use &, bool &)> Pred, const AbstractAttribute &QueryingAA, const Value &V, bool CheckBBLivenessOnly=false, DepClassTy LivenessDepClass=DepClassTy::OPTIONAL, bool IgnoreDroppableUses=true, function_ref< bool(const Use &OldU, const Use &NewU)> EquivalentUseCB=nullptr)
Check Pred on all (transitive) uses of V.
LLVM_ABI ChangeStatus manifestAttrs(const IRPosition &IRP, ArrayRef< Attribute > DeducedAttrs, bool ForceReplace=false)
Attach DeducedAttrs to IRP, if ForceReplace is set we do this even if the same attribute kind was alr...
LLVM_ABI bool hasAttr(const IRPosition &IRP, ArrayRef< Attribute::AttrKind > AKs, bool IgnoreSubsumingPositions=false, Attribute::AttrKind ImpliedAttributeKind=Attribute::None)
Return true if any kind in AKs existing in the IR at a position that will affect this one.
LLVM_ABI void registerForUpdate(AbstractAttribute &AA)
Allows a query AA to request an update if a new query was received.
std::function< bool(Attributor &, const AbstractAttribute *)> VirtualUseCallbackTy
LLVM_ABI void identifyDefaultAbstractAttributes(Function &F)
Determine opportunities to derive 'default' attributes in F and create abstract attribute objects for...
LLVM_ABI bool getAssumedSimplifiedValues(const IRPosition &IRP, const AbstractAttribute *AA, SmallVectorImpl< AA::ValueAndContext > &Values, AA::ValueScope S, bool &UsedAssumedInformation, bool RecurseForSelectAndPHI=true)
Try to simplify IRP and in the scope S.
BumpPtrAllocator & Allocator
The allocator used to allocate memory, e.g. for AbstractAttributes.
LLVM_ABI ChangeStatus run()
Run the analyses until a fixpoint is reached or enforced (timeout).
static LLVM_ABI bool internalizeFunctions(SmallPtrSetImpl< Function * > &FnSet, DenseMap< Function *, Function * > &FnMap)
Make copies of each function in the set FnSet such that the copied version has internal linkage after...
LLVM_ABI bool checkForAllCallSites(function_ref< bool(AbstractCallSite)> Pred, const AbstractAttribute &QueryingAA, bool RequireAllCallSites, bool &UsedAssumedInformation)
Check Pred on all function call sites.
LLVM_ABI bool getAttrsFromAssumes(const IRPosition &IRP, Attribute::AttrKind AK, SmallVectorImpl< Attribute > &Attrs)
Return the attributes of kind AK existing in the IR as operand bundles of an llvm....
bool isKnown(base_t BitsEncoding=BestState) const
Return true if the bits set in BitsEncoding are "known bits".
Support structure for SCC passes to communicate updates the call graph back to the CGSCC pass manager...
static std::string getNodeLabel(const AADepGraphNode *Node, const AADepGraph *DG)
DOTGraphTraits(bool isSimple=false)
DefaultDOTGraphTraits(bool simple=false)
Represent subnormal handling kind for floating point instruction inputs and outputs.
@ Dynamic
Denormals have unknown treatment.
static NodeRef DepGetVal(const DepTy &DT)
PointerIntPair< AADepGraphNode *, 1 > DepTy
static ChildIteratorType child_end(NodeRef N)
static NodeRef getEntryNode(AADepGraphNode *DGN)
mapped_iterator< AADepGraphNode::DepSetTy::iterator, decltype(&DepGetVal)> ChildIteratorType
PointerIntPair< AADepGraphNode *, 1 > EdgeRef
static ChildIteratorType child_begin(NodeRef N)
AADepGraphNode::DepSetTy::iterator ChildEdgeIteratorType
static NodeRef getEntryNode(AADepGraph *DG)
mapped_iterator< AADepGraphNode::DepSetTy::iterator, decltype(&DepGetVal)> nodes_iterator
static nodes_iterator nodes_begin(AADepGraph *DG)
static nodes_iterator nodes_end(AADepGraph *DG)
typename AADepGraph *::UnknownGraphTypeError NodeRef
Helper to describe and deal with positions in the LLVM-IR.
Function * getAssociatedFunction() const
Return the associated function, if any.
void setAttrList(const AttributeList &AttrList) const
Update the attributes associated with this function or call site scope.
unsigned getAttrIdx() const
Return the index in the attribute list for this position.
bool hasCallBaseContext() const
Check if the position has any call base context.
static const IRPosition callsite_returned(const CallBase &CB)
Create a position describing the returned value of CB.
static const IRPosition returned(const Function &F, const CallBaseContext *CBContext=nullptr)
Create a position describing the returned value of F.
LLVM_ABI Argument * getAssociatedArgument() const
Return the associated argument, if any.
static const IRPosition value(const Value &V, const CallBaseContext *CBContext=nullptr)
Create a position describing the value of V.
AttributeList getAttrList() const
Return the attributes associated with this function or call site scope.
static const IRPosition inst(const Instruction &I, const CallBaseContext *CBContext=nullptr)
Create a position describing the instruction I.
static const IRPosition callsite_argument(const CallBase &CB, unsigned ArgNo)
Create a position describing the argument of CB at position ArgNo.
Kind
The positions we distinguish in the IR.
@ IRP_ARGUMENT
An attribute for a function argument.
@ IRP_RETURNED
An attribute for the function return value.
@ IRP_CALL_SITE
An attribute for a call site (function scope).
@ IRP_CALL_SITE_RETURNED
An attribute for a call site return value.
@ IRP_FUNCTION
An attribute for a function (scope).
@ IRP_FLOAT
A position that is not associated with a spot suitable for attributes.
@ IRP_CALL_SITE_ARGUMENT
An attribute for a call site argument.
@ IRP_INVALID
An invalid position.
Instruction * getCtxI() const
Return the context instruction, if any.
static const IRPosition argument(const Argument &Arg, const CallBaseContext *CBContext=nullptr)
Create a position describing the argument Arg.
static const IRPosition function(const Function &F, const CallBaseContext *CBContext=nullptr)
Create a position describing the function scope of F.
const CallBaseContext * getCallBaseContext() const
Get the call base context from the position.
Value & getAssociatedValue() const
Return the value this abstract attribute is associated with.
Value & getAnchorValue() const
Return the value this abstract attribute is anchored with.
Value * getAttrListAnchor() const
Return the value attributes are attached to.
int getCallSiteArgNo() const
Return the call site argument number of the associated value if it is an argument or call site argume...
Kind getPositionKind() const
Return the associated position kind.
static const IRPosition callsite_function(const CallBase &CB)
Create a position describing the function scope of CB.
Function * getAnchorScope() const
Return the Function surrounding the anchor value.
State for an integer range.
ConstantRange getKnown() const
Return the known state encoding.
ConstantRange getAssumed() const
Return the assumed state encoding.
uint32_t getBitWidth() const
Return associated values' bit width.
A "must be executed context" for a given program point PP is the set of instructions,...
iterator & end()
Return an universal end iterator.
bool findInContextOf(const Instruction *I, const Instruction *PP)
Helper to look for I in the context of PP.
iterator & begin(const Instruction *PP)
Return an iterator to explore the context around PP.
bool undefIsContained() const
Returns whether this state contains an undef value or not.
bool isValidState() const override
See AbstractState::isValidState(...)
const SetTy & getAssumedSet() const
Return this set.