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);
1340 IRPositions.emplace_back(IRP);
1344 auto CanIgnoreOperandBundles = [](
const CallBase &CB) {
1360 assert(CB &&
"Expected call site!");
1363 if (!CB->hasOperandBundles() || CanIgnoreOperandBundles(*CB))
1368 assert(CB &&
"Expected call site!");
1371 if (!CB->hasOperandBundles() || CanIgnoreOperandBundles(*CB)) {
1376 for (
const Argument &Arg : Callee->args())
1377 if (Arg.hasReturnedAttr()) {
1378 IRPositions.emplace_back(
1380 IRPositions.emplace_back(
1389 assert(CB &&
"Expected call site!");
1392 if (!CB->hasOperandBundles() || CanIgnoreOperandBundles(*CB)) {
1406void IRPosition::verify() {
1407#ifdef EXPENSIVE_CHECKS
1410 assert((CBContext ==
nullptr) &&
1411 "Invalid position must not have CallBaseContext!");
1413 "Expected a nullptr for an invalid position!");
1417 "Expected specialized kind for argument values!");
1421 "Expected function for a 'returned' position!");
1423 "Associated value mismatch!");
1426 assert((CBContext ==
nullptr) &&
1427 "'call site returned' position must not have CallBaseContext!");
1429 "Expected call base for 'call site returned' position!");
1431 "Associated value mismatch!");
1434 assert((CBContext ==
nullptr) &&
1435 "'call site function' position must not have CallBaseContext!");
1437 "Expected call base for 'call site function' position!");
1439 "Associated value mismatch!");
1443 "Expected function for a 'function' position!");
1445 "Associated value mismatch!");
1449 "Expected argument for a 'argument' position!");
1451 "Associated value mismatch!");
1454 assert((CBContext ==
nullptr) &&
1455 "'call site argument' position must not have CallBaseContext!");
1456 Use *U = getAsUsePtr();
1458 assert(U &&
"Expected use for a 'call site argument' position!");
1460 "Expected call base user for a 'call site argument' position!");
1462 "Expected call base argument operand for a 'call site argument' "
1466 "Argument number mismatch!");
1474std::optional<Constant *>
1477 bool &UsedAssumedInformation) {
1481 for (
auto &CB : SimplificationCallbacks.lookup(IRP)) {
1482 std::optional<Value *> SimplifiedV = CB(IRP, &
AA, UsedAssumedInformation);
1484 return std::nullopt;
1494 UsedAssumedInformation)) {
1496 return std::nullopt;
1510 for (
auto &CB : SimplificationCallbacks.lookup(IRP))
1511 return CB(IRP,
AA, UsedAssumedInformation);
1517 return std::nullopt;
1530 bool &UsedAssumedInformation,
bool RecurseForSelectAndPHI) {
1534 while (!Worklist.
empty()) {
1540 int NV = Values.
size();
1541 const auto &SimplificationCBs = SimplificationCallbacks.lookup(IRP);
1542 for (
const auto &CB : SimplificationCBs) {
1543 std::optional<Value *> CBResult = CB(IRP,
AA, UsedAssumedInformation);
1544 if (!CBResult.has_value())
1546 Value *V = *CBResult;
1555 if (SimplificationCBs.empty()) {
1558 const auto *PotentialValuesAA =
1560 if (PotentialValuesAA &&
1561 PotentialValuesAA->getAssumedSimplifiedValues(*
this, Values, S)) {
1562 UsedAssumedInformation |= !PotentialValuesAA->isAtFixpoint();
1571 if (!RecurseForSelectAndPHI)
1574 for (
int I = NV, E = Values.
size();
I < E; ++
I) {
1575 Value *V = Values[
I].getValue();
1578 if (!Seen.
insert(V).second)
1581 Values[
I] = Values[E - 1];
1595 bool &UsedAssumedInformation) {
1603 if (!Arg->hasPointeeInMemoryValueAttr())
1613 for (
auto &It : AAMap) {
1615 AA->~AbstractAttribute();
1621 bool &UsedAssumedInformation,
1622 bool CheckBBLivenessOnly,
DepClassTy DepClass) {
1623 if (!Configuration.UseLiveness)
1629 CheckBBLivenessOnly, DepClass);
1635 bool &UsedAssumedInformation,
1636 bool CheckBBLivenessOnly,
DepClassTy DepClass) {
1637 if (!Configuration.UseLiveness)
1642 UsedAssumedInformation, CheckBBLivenessOnly, DepClass);
1647 if (CB->isArgOperand(&U)) {
1651 UsedAssumedInformation, CheckBBLivenessOnly,
1657 UsedAssumedInformation, CheckBBLivenessOnly, DepClass);
1661 UsedAssumedInformation, CheckBBLivenessOnly, DepClass);
1663 if (!CheckBBLivenessOnly &&
SI->getPointerOperand() != U.get()) {
1671 UsedAssumedInformation =
true;
1678 UsedAssumedInformation, CheckBBLivenessOnly, DepClass);
1684 bool &UsedAssumedInformation,
1685 bool CheckBBLivenessOnly,
DepClassTy DepClass,
1686 bool CheckForDeadStore) {
1687 if (!Configuration.UseLiveness)
1692 if (ManifestAddedBlocks.contains(
I.getParent()))
1701 if (!FnLivenessAA || QueryingAA == FnLivenessAA)
1705 if (CheckBBLivenessOnly ? FnLivenessAA->
isAssumedDead(
I.getParent())
1710 UsedAssumedInformation =
true;
1714 if (CheckBBLivenessOnly)
1722 if (!IsDeadAA || QueryingAA == IsDeadAA)
1729 UsedAssumedInformation =
true;
1737 UsedAssumedInformation =
true;
1747 bool &UsedAssumedInformation,
1748 bool CheckBBLivenessOnly,
DepClassTy DepClass) {
1749 if (!Configuration.UseLiveness)
1760 isAssumedDead(*CtxI, QueryingAA, FnLivenessAA, UsedAssumedInformation,
1765 if (CheckBBLivenessOnly)
1778 if (!IsDeadAA || QueryingAA == IsDeadAA)
1785 UsedAssumedInformation =
true;
1796 if (!Configuration.UseLiveness)
1804 if (!FnLivenessAA || QueryingAA == FnLivenessAA)
1820 return Pred(Callee);
1824 if (!CallEdgesAA || CallEdgesAA->hasUnknownCallee())
1827 const auto &Callees = CallEdgesAA->getOptimisticEdges();
1828 return Pred(Callees.getArrayRef());
1838 bool CheckBBLivenessOnly,
DepClassTy LivenessDepClass,
1839 bool IgnoreDroppableUses,
1844 if (!CB(*
this, &QueryingAA))
1858 auto AddUsers = [&](
const Value &V,
const Use *OldUse) {
1859 for (
const Use &UU : V.uses()) {
1860 if (OldUse && EquivalentUseCB && !EquivalentUseCB(*OldUse, UU)) {
1862 "rejected by the equivalence call back: "
1872 AddUsers(V,
nullptr);
1875 <<
" initial uses to check\n");
1878 const auto *LivenessAA =
1883 while (!Worklist.
empty()) {
1889 dbgs() <<
"[Attributor] Check use: " << **U <<
" in " << Fn->getName()
1892 dbgs() <<
"[Attributor] Check use: " << **U <<
" in " << *U->getUser()
1895 bool UsedAssumedInformation =
false;
1896 if (
isAssumedDead(*U, &QueryingAA, LivenessAA, UsedAssumedInformation,
1897 CheckBBLivenessOnly, LivenessDepClass)) {
1899 dbgs() <<
"[Attributor] Dead use, skip!\n");
1902 if (IgnoreDroppableUses && U->getUser()->isDroppable()) {
1904 dbgs() <<
"[Attributor] Droppable user, skip!\n");
1909 if (&
SI->getOperandUse(0) == U) {
1910 if (!Visited.
insert(U).second)
1914 *
this, *
SI, PotentialCopies, QueryingAA, UsedAssumedInformation,
1918 <<
"[Attributor] Value is stored, continue with "
1919 << PotentialCopies.
size()
1920 <<
" potential copies instead!\n");
1921 for (
Value *PotentialCopy : PotentialCopies)
1922 if (!AddUsers(*PotentialCopy, U))
1929 bool Follow =
false;
1930 if (!Pred(*U, Follow))
1935 User &Usr = *U->getUser();
1936 AddUsers(Usr,
nullptr);
1944 bool RequireAllCallSites,
1945 bool &UsedAssumedInformation) {
1951 if (!AssociatedFunction) {
1952 LLVM_DEBUG(
dbgs() <<
"[Attributor] No function associated with " << IRP
1958 &QueryingAA, UsedAssumedInformation);
1963 bool RequireAllCallSites,
1965 bool &UsedAssumedInformation,
1966 bool CheckPotentiallyDead) {
1970 <<
"[Attributor] Function " << Fn.
getName()
1971 <<
" has no internal linkage, hence not all call sites are known\n");
1976 if (!CB(*
this, QueryingAA))
1980 for (
unsigned u = 0; u <
Uses.size(); ++u) {
1984 dbgs() <<
"[Attributor] Check use: " << Fn->
getName() <<
" in "
1985 << *U.getUser() <<
"\n";
1987 dbgs() <<
"[Attributor] Check use: " << *U <<
" in " << *U.getUser()
1990 if (!CheckPotentiallyDead &&
1991 isAssumedDead(U, QueryingAA,
nullptr, UsedAssumedInformation,
1994 dbgs() <<
"[Attributor] Dead use, skip!\n");
1998 if (CE->isCast() && CE->getType()->isPointerTy()) {
2000 dbgs() <<
"[Attributor] Use, is constant cast expression, add "
2001 << CE->getNumUses() <<
" uses of that expression instead!\n";
2003 for (
const Use &CEU : CE->uses())
2004 Uses.push_back(&CEU);
2012 <<
" has non call site use " << *U.get() <<
" in "
2013 << *U.getUser() <<
"\n");
2017 const Use *EffectiveUse =
2020 if (!RequireAllCallSites) {
2021 LLVM_DEBUG(
dbgs() <<
"[Attributor] User " << *EffectiveUse->getUser()
2022 <<
" is not a call of " << Fn.
getName()
2026 LLVM_DEBUG(
dbgs() <<
"[Attributor] User " << *EffectiveUse->getUser()
2027 <<
" is an invalid use of " << Fn.
getName() <<
"\n");
2035 unsigned MinArgsParams =
2037 for (
unsigned u = 0; u < MinArgsParams; ++u) {
2041 dbgs() <<
"[Attributor] Call site / callee argument type mismatch ["
2042 << u <<
"@" << Fn.
getName() <<
": "
2052 LLVM_DEBUG(
dbgs() <<
"[Attributor] Call site callback failed for "
2060bool Attributor::shouldPropagateCallBaseContext(
const IRPosition &IRP) {
2070 bool RecurseForSelectAndPHI) {
2074 if (!AssociatedFunction)
2077 bool UsedAssumedInformation =
false;
2081 UsedAssumedInformation, RecurseForSelectAndPHI))
2093 bool &UsedAssumedInformation,
bool CheckBBLivenessOnly =
false,
2094 bool CheckPotentiallyDead =
false) {
2095 for (
unsigned Opcode : Opcodes) {
2097 auto *Insts = OpcodeInstMap.
lookup(Opcode);
2103 if (
A && !CheckPotentiallyDead &&
2105 UsedAssumedInformation, CheckBBLivenessOnly)) {
2107 dbgs() <<
"[Attributor] Instruction " << *
I
2108 <<
" is potentially dead, skip!\n";);
2123 bool &UsedAssumedInformation,
2124 bool CheckBBLivenessOnly,
2125 bool CheckPotentiallyDead) {
2131 const auto *LivenessAA =
2132 CheckPotentiallyDead && QueryingAA
2136 auto &OpcodeInstMap = InfoCache.getOpcodeInstMapForFunction(*Fn);
2138 LivenessAA, Opcodes, UsedAssumedInformation,
2139 CheckBBLivenessOnly, CheckPotentiallyDead))
2148 bool &UsedAssumedInformation,
2149 bool CheckBBLivenessOnly,
2150 bool CheckPotentiallyDead) {
2154 UsedAssumedInformation, CheckBBLivenessOnly,
2155 CheckPotentiallyDead);
2160 bool &UsedAssumedInformation) {
2163 const Function *AssociatedFunction =
2165 if (!AssociatedFunction)
2169 const auto *LivenessAA =
2173 InfoCache.getReadOrWriteInstsForFunction(*AssociatedFunction)) {
2176 UsedAssumedInformation))
2186void Attributor::runTillFixpoint() {
2190 <<
" abstract attributes.\n");
2195 unsigned IterationCounter = 1;
2196 unsigned MaxIterations =
2206 LLVM_DEBUG(
dbgs() <<
"\n\n[Attributor] #Iteration: " << IterationCounter
2207 <<
", Worklist size: " << Worklist.
size() <<
"\n");
2212 for (
unsigned u = 0; u < InvalidAAs.
size(); ++u) {
2217 dbgs() <<
"[Attributor] InvalidAA: " << *InvalidAA
2218 <<
" has " << InvalidAA->
Deps.
size()
2219 <<
" required & optional dependences\n");
2220 for (
auto &DepIt : InvalidAA->
Deps) {
2224 dbgs() <<
" - recompute: " << *DepAA);
2229 <<
" - invalidate: " << *DepAA);
2233 InvalidAAs.
insert(DepAA);
2242 for (AbstractAttribute *ChangedAA : ChangedAAs) {
2243 for (
auto &DepIt : ChangedAA->Deps)
2245 ChangedAA->Deps.clear();
2248 LLVM_DEBUG(
dbgs() <<
"[Attributor] #Iteration: " << IterationCounter
2249 <<
", Worklist+Dependent size: " << Worklist.
size()
2258 for (AbstractAttribute *AA : Worklist) {
2259 const auto &AAState = AA->getState();
2260 if (!AAState.isAtFixpoint())
2262 ChangedAAs.push_back(AA);
2266 if (!AAState.isValidState())
2272 ChangedAAs.append(DG.SyntheticRoot.begin() + NumAAs,
2273 DG.SyntheticRoot.end());
2278 Worklist.insert_range(ChangedAAs);
2279 Worklist.insert_range(QueryAAsAwaitingUpdate);
2280 QueryAAsAwaitingUpdate.clear();
2282 }
while (!Worklist.empty() && (IterationCounter++ < MaxIterations));
2284 if (IterationCounter > MaxIterations && !Functions.empty()) {
2285 auto Remark = [&](OptimizationRemarkMissed ORM) {
2286 return ORM <<
"Attributor did not reach a fixpoint after "
2287 <<
ore::NV(
"Iterations", MaxIterations) <<
" iterations.";
2293 LLVM_DEBUG(
dbgs() <<
"\n[Attributor] Fixpoint iteration done after: "
2294 << IterationCounter <<
"/" << MaxIterations
2295 <<
" iterations\n");
2302 SmallPtrSet<AbstractAttribute *, 32> Visited;
2303 for (
unsigned u = 0;
u < ChangedAAs.size();
u++) {
2304 AbstractAttribute *ChangedAA = ChangedAAs[
u];
2305 if (!Visited.
insert(ChangedAA).second)
2308 AbstractState &State = ChangedAA->
getState();
2312 NumAttributesTimedOut++;
2315 for (
auto &DepIt : ChangedAA->
Deps)
2321 if (!Visited.
empty())
2322 dbgs() <<
"\n[Attributor] Finalized " << Visited.
size()
2323 <<
" abstract attributes.\n";
2329 "Non-query AAs should not be required to register for updates!");
2330 QueryAAsAwaitingUpdate.insert(&
AA);
2337 unsigned NumManifested = 0;
2338 unsigned NumAtFixpoint = 0;
2348 if (!State.isAtFixpoint())
2349 State.indicateOptimisticFixpoint();
2352 if (
AA->hasCallBaseContext())
2355 if (!State.isValidState())
2358 if (
AA->getCtxI() && !
isRunOn(*
AA->getAnchorScope()))
2362 bool UsedAssumedInformation =
false;
2373 AA->trackStatistics();
2374 LLVM_DEBUG(
dbgs() <<
"[Attributor] Manifest " << LocalChange <<
" : " << *
AA
2377 ManifestChange = ManifestChange | LocalChange;
2383 (void)NumManifested;
2384 (void)NumAtFixpoint;
2385 LLVM_DEBUG(
dbgs() <<
"\n[Attributor] Manifested " << NumManifested
2386 <<
" arguments while " << NumAtFixpoint
2387 <<
" were in a valid fixpoint state\n");
2389 NumAttributesManifested += NumManifested;
2390 NumAttributesValidFixpoint += NumAtFixpoint;
2395 for (
unsigned u = 0; u < NumFinalAAs; ++u)
2399 errs() <<
"Unexpected abstract attribute: "
2403 .getAssociatedValue()
2407 "remain unchanged!");
2410 for (
auto &It : AttrsMap) {
2411 AttributeList &
AL = It.getSecond();
2412 const IRPosition &IRP =
2415 : IRPosition::callsite_function(*
cast<CallBase>(It.getFirst()));
2419 return ManifestChange;
2422void Attributor::identifyDeadInternalFunctions() {
2424 if (!Configuration.DeleteFns)
2442 for (Function *
F : Functions)
2443 if (
F->hasLocalLinkage() && (
isModulePass() || !TLI->getLibFunc(*
F, LF)))
2446 SmallPtrSet<Function *, 8> LiveInternalFns;
2447 bool FoundLiveInternal =
true;
2448 while (FoundLiveInternal) {
2449 FoundLiveInternal =
false;
2450 for (Function *&
F : InternalFns) {
2454 bool UsedAssumedInformation =
false;
2456 [&](AbstractCallSite ACS) {
2458 return ToBeDeletedFunctions.count(Callee) ||
2459 (Functions.count(Callee) &&
Callee->hasLocalLinkage() &&
2460 !LiveInternalFns.
count(Callee));
2462 *
F,
true,
nullptr, UsedAssumedInformation)) {
2468 FoundLiveInternal =
true;
2472 for (Function *
F : InternalFns)
2474 ToBeDeletedFunctions.insert(
F);
2478 TimeTraceScope TimeScope(
"Attributor::cleanupIR");
2481 << ToBeDeletedFunctions.size() <<
" functions and "
2482 << ToBeDeletedBlocks.size() <<
" blocks and "
2483 << ToBeDeletedInsts.size() <<
" instructions and "
2484 << ToBeChangedValues.size() <<
" values and "
2485 << ToBeChangedUses.size() <<
" uses. To insert "
2486 << ToBeChangedToUnreachableInsts.size()
2487 <<
" unreachables.\n"
2488 <<
"Preserve manifest added " << ManifestAddedBlocks.size()
2494 auto ReplaceUse = [&](
Use *
U,
Value *NewV) {
2499 const auto &
Entry = ToBeChangedValues.lookup(NewV);
2507 "Cannot replace an instruction outside the current SCC!");
2513 if (CI->isMustTailCall() && !ToBeDeletedInsts.count(CI))
2518 for (
auto &Arg : RI->getFunction()->args())
2519 Arg.removeAttr(Attribute::Returned);
2523 <<
" instead of " << *OldV <<
"\n");
2527 CGModifiedFunctions.insert(
I->getFunction());
2534 if (CB->isArgOperand(U)) {
2535 unsigned Idx = CB->getArgOperandNo(U);
2536 CB->removeParamAttr(Idx, Attribute::NoUndef);
2538 if (Callee &&
Callee->arg_size() > Idx)
2539 Callee->removeParamAttr(Idx, Attribute::NoUndef);
2545 ToBeChangedToUnreachableInsts.insert(UserI);
2552 for (
auto &It : ToBeChangedUses) {
2554 Value *NewV = It.second;
2555 ReplaceUse(U, NewV);
2559 for (
auto &It : ToBeChangedValues) {
2560 Value *OldV = It.first;
2561 auto [NewV,
Done] = It.second;
2563 for (
auto &U : OldV->
uses())
2564 if (
Done || !
U.getUser()->isDroppable())
2566 for (Use *U :
Uses) {
2570 ReplaceUse(U, NewV);
2574 for (
const auto &V : InvokeWithDeadSuccessor)
2577 "Cannot replace an invoke outside the current SCC!");
2578 bool UnwindBBIsDead =
II->hasFnAttr(Attribute::NoUnwind);
2579 bool NormalBBIsDead =
II->hasFnAttr(Attribute::NoReturn);
2580 bool Invoke2CallAllowed =
2582 assert((UnwindBBIsDead || NormalBBIsDead) &&
2583 "Invoke does not have dead successors!");
2586 if (UnwindBBIsDead) {
2588 if (Invoke2CallAllowed) {
2593 ToBeChangedToUnreachableInsts.insert(NormalNextIP);
2595 assert(NormalBBIsDead &&
"Broken invariant!");
2598 ToBeChangedToUnreachableInsts.insert(&NormalDestBB->
front());
2601 for (Instruction *
I : TerminatorsToFold) {
2603 "Cannot replace a terminator outside the current SCC!");
2604 CGModifiedFunctions.insert(
I->getFunction());
2607 for (
const auto &V : ToBeChangedToUnreachableInsts)
2612 "Cannot replace an instruction outside the current SCC!");
2613 CGModifiedFunctions.insert(
I->getFunction());
2617 for (
const auto &V : ToBeDeletedInsts) {
2621 "Cannot delete an instruction outside the current SCC!");
2622 I->dropDroppableUses();
2623 CGModifiedFunctions.insert(
I->getFunction());
2624 if (!
I->getType()->isVoidTy())
2629 I->eraseFromParent();
2636 dbgs() <<
"[Attributor] DeadInsts size: " << DeadInsts.
size() <<
"\n";
2637 for (
auto &
I : DeadInsts)
2639 dbgs() <<
" - " << *
I <<
"\n";
2644 if (
unsigned NumDeadBlocks = ToBeDeletedBlocks.size()) {
2645 SmallVector<BasicBlock *, 8> ToBeDeletedBBs;
2646 ToBeDeletedBBs.
reserve(NumDeadBlocks);
2647 for (BasicBlock *BB : ToBeDeletedBlocks) {
2649 "Cannot delete a block outside the current SCC!");
2650 CGModifiedFunctions.insert(BB->
getParent());
2652 if (ManifestAddedBlocks.contains(BB))
2662 identifyDeadInternalFunctions();
2665 ChangeStatus ManifestChange = rewriteFunctionSignatures(CGModifiedFunctions);
2667 for (Function *Fn : CGModifiedFunctions)
2668 if (!ToBeDeletedFunctions.count(Fn) && Functions.count(Fn))
2669 Configuration.CGUpdater.reanalyzeFunction(*Fn);
2671 for (Function *Fn : ToBeDeletedFunctions) {
2672 if (!Functions.count(Fn))
2674 Configuration.CGUpdater.removeFunction(*Fn);
2677 if (!ToBeChangedUses.empty())
2680 if (!ToBeChangedToUnreachableInsts.empty())
2683 if (!ToBeDeletedFunctions.empty())
2686 if (!ToBeDeletedBlocks.empty())
2689 if (!ToBeDeletedInsts.empty())
2692 if (!InvokeWithDeadSuccessor.empty())
2695 if (!DeadInsts.empty())
2698 NumFnDeleted += ToBeDeletedFunctions.size();
2700 LLVM_DEBUG(
dbgs() <<
"[Attributor] Deleted " << ToBeDeletedFunctions.size()
2701 <<
" functions after manifest.\n");
2703#ifdef EXPENSIVE_CHECKS
2704 for (Function *
F : Functions) {
2705 if (ToBeDeletedFunctions.count(
F))
2711 return ManifestChange;
2716 AttributorCallGraph ACallGraph(*
this);
2721 Phase = AttributorPhase::UPDATE;
2734 Phase = AttributorPhase::MANIFEST;
2737 Phase = AttributorPhase::CLEANUP;
2743 return ManifestChange | CleanupChange;
2748 return AA.getName().str() +
2749 std::to_string(
AA.getIRPosition().getPositionKind());
2752 "We can update AA only in the update stage!");
2755 DependenceVector DV;
2756 DependenceStack.push_back(&DV);
2758 auto &AAState =
AA.getState();
2760 bool UsedAssumedInformation =
false;
2763 CS =
AA.update(*
this);
2765 if (!
AA.isQueryAA() && DV.empty() && !
AA.getState().isAtFixpoint()) {
2772 RerunCS =
AA.update(*
this);
2778 AAState.indicateOptimisticFixpoint();
2781 if (!AAState.isAtFixpoint())
2782 rememberDependences();
2786 DependenceVector *PoppedDV = DependenceStack.pop_back_val();
2788 assert(PoppedDV == &DV &&
"Inconsistent usage of the dependence stack!");
2794 assert(!
F.isDeclaration() &&
"Cannot create a wrapper around a declaration!");
2803 M.getFunctionList().insert(
F.getIterator(),
Wrapper);
2808 assert(
F.use_empty() &&
"Uses remained after wrapper was created!");
2813 F.setComdat(
nullptr);
2817 F.getAllMetadata(MDs);
2818 for (
auto MDIt : MDs)
2819 Wrapper->addMetadata(MDIt.first, *MDIt.second);
2820 Wrapper->setAttributes(
F.getAttributes());
2828 Args.push_back(&Arg);
2829 Arg.setName((FArgIt++)->
getName());
2837 NumFnShallowWrappersCreated++;
2841 if (
F.isDeclaration() ||
F.hasLocalLinkage() ||
2857 return InternalizedFns[&
F];
2875 F->getName() +
".internalized");
2878 for (
auto &Arg :
F->args()) {
2879 auto ArgName = Arg.getName();
2880 NewFArgIt->setName(ArgName);
2881 VMap[&Arg] = &(*NewFArgIt++);
2896 F->getAllMetadata(MDs);
2897 for (
auto MDIt : MDs)
2901 M.getFunctionList().insert(
F->getIterator(), Copied);
2909 auto &InternalizedFn = FnMap[
F];
2910 auto IsNotInternalized = [&](
Use &U) ->
bool {
2912 return !FnMap.
lookup(CB->getCaller());
2915 F->replaceUsesWithIf(InternalizedFn, IsNotInternalized);
2924 if (!Configuration.RewriteSignatures)
2947 LLVM_DEBUG(
dbgs() <<
"[Attributor] Cannot rewrite var-args functions\n");
2953 if (FnAttributeList.hasAttrSomewhere(Attribute::Nest) ||
2954 FnAttributeList.hasAttrSomewhere(Attribute::StructRet) ||
2955 FnAttributeList.hasAttrSomewhere(Attribute::InAlloca) ||
2956 FnAttributeList.hasAttrSomewhere(Attribute::Preallocated)) {
2958 dbgs() <<
"[Attributor] Cannot rewrite due to complex attribute\n");
2963 bool UsedAssumedInformation =
false;
2965 UsedAssumedInformation,
2967 LLVM_DEBUG(
dbgs() <<
"[Attributor] Cannot rewrite all call sites\n");
2973 return !CI->isMustTailCall();
2979 auto &OpcodeInstMap = InfoCache.getOpcodeInstMapForFunction(*Fn);
2981 nullptr, {Instruction::Call},
2982 UsedAssumedInformation)) {
2983 LLVM_DEBUG(
dbgs() <<
"[Attributor] Cannot rewrite due to instructions\n");
2994 LLVM_DEBUG(
dbgs() <<
"[Attributor] Register new rewrite of " << Arg <<
" in "
2996 << ReplacementTypes.
size() <<
" replacements\n");
2998 "Cannot register an invalid rewrite");
3002 ArgumentReplacementMap[Fn];
3008 std::unique_ptr<ArgumentReplacementInfo> &ARI = ARIs[Arg.
getArgNo()];
3009 if (ARI && ARI->getNumReplacementArgs() <= ReplacementTypes.
size()) {
3010 LLVM_DEBUG(
dbgs() <<
"[Attributor] Existing rewrite is preferred\n");
3018 LLVM_DEBUG(
dbgs() <<
"[Attributor] Register new rewrite of " << Arg <<
" in "
3020 << ReplacementTypes.
size() <<
" replacements\n");
3024 std::move(CalleeRepairCB),
3025 std::move(ACSRepairCB)));
3046 for (
auto &It : ArgumentReplacementMap) {
3050 if (!Functions.count(OldFn) || ToBeDeletedFunctions.count(OldFn))
3063 if (
const std::unique_ptr<ArgumentReplacementInfo> &ARI =
3064 ARIs[Arg.getArgNo()]) {
3065 NewArgumentTypes.
append(ARI->ReplacementTypes.begin(),
3066 ARI->ReplacementTypes.end());
3067 NewArgumentAttributes.
append(ARI->getNumReplacementArgs(),
3070 NewArgumentTypes.
push_back(Arg.getType());
3072 OldFnAttributeList.getParamAttrs(Arg.getArgNo()));
3076 uint64_t LargestVectorWidth = 0;
3077 for (
auto *
I : NewArgumentTypes)
3079 LargestVectorWidth =
3080 std::max(LargestVectorWidth,
3081 VT->getPrimitiveSizeInBits().getKnownMinValue());
3084 Type *RetTy = OldFnTy->getReturnType();
3087 FunctionType *NewFnTy =
3092 << *NewFnTy <<
"\n");
3097 Functions.insert(NewFn);
3110 Ctx, OldFnAttributeList.getFnAttrs(), OldFnAttributeList.getRetAttrs(),
3111 NewArgumentAttributes));
3112 AttributeFuncs::updateMinLegalVectorWidthAttr(*NewFn, LargestVectorWidth);
3120 return !
T->isPtrOrPtrVectorTy() ||
3136 auto CallSiteReplacementCreator = [&](AbstractCallSite ACS) {
3138 const AttributeList &OldCallAttributeList = OldCB->
getAttributes();
3141 SmallVector<Value *, 16> NewArgOperands;
3143 for (
unsigned OldArgNum = 0; OldArgNum < ARIs.
size(); ++OldArgNum) {
3144 unsigned NewFirstArgNum = NewArgOperands.
size();
3145 (void)NewFirstArgNum;
3146 if (
const std::unique_ptr<ArgumentReplacementInfo> &ARI =
3148 if (ARI->ACSRepairCB)
3149 ARI->ACSRepairCB(*ARI, ACS, NewArgOperands);
3150 assert(ARI->getNumReplacementArgs() + NewFirstArgNum ==
3151 NewArgOperands.
size() &&
3152 "ACS repair callback did not provide as many operand as new "
3153 "types were registered!");
3155 NewArgOperandAttributes.
append(ARI->ReplacementTypes.size(),
3160 OldCallAttributeList.getParamAttrs(OldArgNum));
3164 assert(NewArgOperands.
size() == NewArgOperandAttributes.
size() &&
3165 "Mismatch # argument operands vs. # argument operand attributes!");
3167 "Mismatch # argument operands vs. # function arguments!");
3176 II->getUnwindDest(), NewArgOperands,
3181 NewCI->setTailCallKind(
cast<CallInst>(OldCB)->getTailCallKind());
3186 NewCB->
copyMetadata(*OldCB, {LLVMContext::MD_prof, LLVMContext::MD_dbg});
3190 Ctx, OldCallAttributeList.getFnAttrs(),
3191 OldCallAttributeList.getRetAttrs(), NewArgOperandAttributes));
3193 AttributeFuncs::updateMinLegalVectorWidthAttr(*NewCB->
getCaller(),
3194 LargestVectorWidth);
3196 CallSitePairs.
push_back({OldCB, NewCB});
3201 bool UsedAssumedInformation =
false;
3203 true,
nullptr, UsedAssumedInformation,
3206 assert(
Success &&
"Assumed call site replacement to succeed!");
3211 for (
unsigned OldArgNum = 0; OldArgNum < ARIs.
size();
3212 ++OldArgNum, ++OldFnArgIt) {
3213 if (
const std::unique_ptr<ArgumentReplacementInfo> &ARI =
3215 if (ARI->CalleeRepairCB)
3216 ARI->CalleeRepairCB(*ARI, *NewFn, NewFnArgIt);
3217 if (ARI->ReplacementTypes.empty())
3220 NewFnArgIt += ARI->ReplacementTypes.size();
3222 NewFnArgIt->
takeName(&*OldFnArgIt);
3229 for (
auto &CallSitePair : CallSitePairs) {
3230 CallBase &OldCB = *CallSitePair.first;
3231 CallBase &NewCB = *CallSitePair.second;
3233 "Cannot handle call sites with different types!");
3240 Configuration.CGUpdater.replaceFunctionWith(*OldFn, *NewFn);
3244 if (ModifiedFns.
remove(OldFn))
3245 ModifiedFns.
insert(NewFn);
3253void InformationCache::initializeInformationCache(
const Function &CF,
3260 FI.IsKernel =
F.hasFnAttribute(
"kernel");
3266 DenseMap<const Value *, std::optional<short>> AssumeUsesMap;
3271 auto AddToAssumeUsesMap = [&](
const Value &
V) ->
void {
3272 SmallVector<const Instruction *> Worklist;
3275 while (!Worklist.
empty()) {
3277 std::optional<short> &NumUses = AssumeUsesMap[
I];
3279 NumUses =
I->getNumUses();
3280 NumUses = *NumUses - 1;
3283 AssumeOnlyValues.insert(
I);
3284 for (
const Value *
Op :
I->operands())
3291 bool IsInterestingOpcode =
false;
3298 switch (
I.getOpcode()) {
3301 "New call base instruction type needs to be known in the "
3304 case Instruction::Call:
3309 AssumeOnlyValues.insert(Assume);
3311 AddToAssumeUsesMap(*
Assume->getArgOperand(0));
3313 FI.ContainsMustTailCall =
true;
3316 getFunctionInfo(*Callee).CalledViaMustTail =
true;
3319 case Instruction::CallBr:
3320 case Instruction::Invoke:
3321 case Instruction::CleanupRet:
3322 case Instruction::CatchSwitch:
3323 case Instruction::AtomicRMW:
3324 case Instruction::AtomicCmpXchg:
3325 case Instruction::UncondBr:
3326 case Instruction::CondBr:
3327 case Instruction::Resume:
3328 case Instruction::Ret:
3329 case Instruction::Load:
3331 case Instruction::Store:
3333 case Instruction::Alloca:
3334 case Instruction::AddrSpaceCast:
3335 IsInterestingOpcode =
true;
3337 if (IsInterestingOpcode) {
3338 auto *&Insts = FI.OpcodeInstMap[
I.getOpcode()];
3343 if (
I.mayReadOrWriteMemory())
3344 FI.RWInsts.push_back(&
I);
3347 if (
F.hasFnAttribute(Attribute::AlwaysInline) &&
3349 InlineableFunctions.insert(&
F);
3352InformationCache::FunctionInfo::~FunctionInfo() {
3355 for (
auto &It : OpcodeInstMap)
3356 It.getSecond()->~InstructionVectorTy();
3361 assert(
A.isClosedWorldModule() &&
"Cannot see all indirect callees!");
3362 return IndirectlyCallableFunctions;
3368 return std::nullopt;
3379 if (DependenceStack.empty())
3383 DependenceStack.back()->push_back({&FromAA, &ToAA, DepClass});
3386void Attributor::rememberDependences() {
3387 assert(!DependenceStack.empty() &&
"No dependences to remember!");
3389 for (DepInfo &DI : *DependenceStack.back()) {
3392 "Expected required or optional dependence (1 bit)!");
3399template <Attribute::AttrKind AK,
typename AAType>
3400void Attributor::checkAndQueryIRAttr(
const IRPosition &IRP, AttributeSet Attrs,
3401 bool SkipHasAttrCheck) {
3403 if (SkipHasAttrCheck || !
Attrs.hasAttribute(AK))
3404 if (!Configuration.Allowed || Configuration.Allowed->count(&AAType::ID))
3407 getOrCreateAAFor<AAType>(IRP);
3413 if (!VisitedFunctions.insert(&
F).second)
3419 InformationCache::FunctionInfo &FI = InfoCache.getFunctionInfo(
F);
3421 for (
const Use &U :
F.uses())
3423 if (CB->isCallee(&U) && CB->isMustTailCall())
3424 FI.CalledViaMustTail =
true;
3429 auto Attrs =
F.getAttributes();
3430 auto FnAttrs = Attrs.getFnAttrs();
3446 checkAndQueryIRAttr<Attribute::MustProgress, AAMustProgress>(FPos, FnAttrs);
3449 checkAndQueryIRAttr<Attribute::NoFree, AANoFree>(FPos, FnAttrs);
3452 checkAndQueryIRAttr<Attribute::WillReturn, AAWillReturn>(FPos, FnAttrs);
3455 checkAndQueryIRAttr<Attribute::NoSync, AANoSync>(FPos, FnAttrs);
3460 if (IsIPOAmendable) {
3463 checkAndQueryIRAttr<Attribute::NoUnwind, AANoUnwind>(FPos, FnAttrs);
3466 checkAndQueryIRAttr<Attribute::NoReturn, AANoReturn>(FPos, FnAttrs);
3469 checkAndQueryIRAttr<Attribute::NoRecurse, AANoRecurse>(FPos, FnAttrs);
3472 if (Attrs.hasFnAttr(Attribute::Convergent))
3493 Type *ReturnType =
F.getReturnType();
3494 if (!ReturnType->isVoidTy()) {
3502 bool UsedAssumedInformation =
false;
3507 checkAndQueryIRAttr<Attribute::NoUndef, AANoUndef>(RetPos, RetAttrs);
3509 if (ReturnType->isPointerTy()) {
3515 checkAndQueryIRAttr<Attribute::NonNull, AANonNull>(RetPos, RetAttrs);
3518 checkAndQueryIRAttr<Attribute::NoAlias, AANoAlias>(RetPos, RetAttrs);
3523 }
else if (AttributeFuncs::isNoFPClassCompatibleType(ReturnType)) {
3531 auto ArgNo = Arg.getArgNo();
3534 if (!IsIPOAmendable) {
3535 if (Arg.getType()->isPointerTy())
3537 checkAndQueryIRAttr<Attribute::NoFree, AANoFree>(ArgPos, ArgAttrs);
3544 bool UsedAssumedInformation =
false;
3552 checkAndQueryIRAttr<Attribute::NoUndef, AANoUndef>(ArgPos, ArgAttrs);
3554 if (Arg.getType()->isPointerTy()) {
3556 checkAndQueryIRAttr<Attribute::NonNull, AANonNull>(ArgPos, ArgAttrs);
3559 checkAndQueryIRAttr<Attribute::NoAlias, AANoAlias>(ArgPos, ArgAttrs);
3568 checkAndQueryIRAttr<Attribute::Captures, AANoCapture>(
3569 ArgPos, ArgAttrs,
true);
3576 checkAndQueryIRAttr<Attribute::NoFree, AANoFree>(ArgPos, ArgAttrs);
3581 }
else if (AttributeFuncs::isNoFPClassCompatibleType(Arg.getType())) {
3609 !Callee->hasMetadata(LLVMContext::MD_callback))
3612 if (!Callee->getReturnType()->isVoidTy() && !CB.use_empty()) {
3614 bool UsedAssumedInformation =
false;
3618 if (AttributeFuncs::isNoFPClassCompatibleType(Callee->getReturnType()))
3622 const AttributeList &CBAttrs = CBFnPos.
getAttrList();
3623 for (
int I = 0, E = CB.arg_size();
I < E; ++
I) {
3634 bool UsedAssumedInformation =
false;
3639 checkAndQueryIRAttr<Attribute::NoUndef, AANoUndef>(CBArgPos, CBArgAttrs);
3641 Type *ArgTy = CB.getArgOperand(
I)->getType();
3644 if (AttributeFuncs::isNoFPClassCompatibleType(ArgTy))
3651 checkAndQueryIRAttr<Attribute::NonNull, AANonNull>(CBArgPos, CBArgAttrs);
3654 checkAndQueryIRAttr<Attribute::Captures, AANoCapture>(
3655 CBArgPos, CBArgAttrs,
true);
3658 checkAndQueryIRAttr<Attribute::NoAlias, AANoAlias>(CBArgPos, CBArgAttrs);
3668 if (!CBAttrs.hasParamAttr(
I, Attribute::ReadNone))
3672 checkAndQueryIRAttr<Attribute::NoFree, AANoFree>(CBArgPos, CBArgAttrs);
3677 auto &OpcodeInstMap = InfoCache.getOpcodeInstMapForFunction(
F);
3678 [[maybe_unused]]
bool Success;
3679 bool UsedAssumedInformation =
false;
3681 nullptr, OpcodeInstMap, CallSitePred,
nullptr,
nullptr,
3682 {(
unsigned)Instruction::Invoke, (
unsigned)Instruction::CallBr,
3684 UsedAssumedInformation);
3685 assert(
Success &&
"Expected the check call to be successful!");
3709 nullptr, OpcodeInstMap, LoadStorePred,
nullptr,
nullptr,
3710 {(
unsigned)Instruction::Load, (
unsigned)Instruction::Store},
3711 UsedAssumedInformation);
3712 assert(
Success &&
"Expected the check call to be successful!");
3715 auto AAAllocationInfoPred = [&](
Instruction &
I) ->
bool {
3721 nullptr, OpcodeInstMap, AAAllocationInfoPred,
nullptr,
nullptr,
3722 {(
unsigned)Instruction::Alloca}, UsedAssumedInformation);
3723 assert(
Success &&
"Expected the check call to be successful!");
3729 return isModulePass() && Configuration.IsClosedWorldModule;
3746 return OS <<
"fn_ret";
3748 return OS <<
"cs_ret";
3756 return OS <<
"cs_arg";
3778 return OS << static_cast<const AbstractState &>(S);
3792 OS <<
"set-state(< {";
3808 OS <<
"set-state(< {";
3814 OS <<
"@" <<
F->getName() <<
"[" << int(It.second) <<
"], ";
3816 OS << *It.first.getValue() <<
"[" << int(It.second) <<
"], ";
3829 OS <<
"] for CtxI ";
3836 OS <<
"<<null inst>>";
3845 for (
const auto &DepAA :
Deps) {
3846 auto *
AA = DepAA.getPointer();
3863 OS <<
" [ <unknown> ]";
3877 bool DeleteFns,
bool IsModulePass) {
3878 if (Functions.empty())
3882 dbgs() <<
"[Attributor] Run on module with " << Functions.size()
3891 AC.IsModulePass = IsModulePass;
3892 AC.DeleteFns = DeleteFns;
3896 IndirectCalleeTrackingMap;
3898 AC.IndirectCalleeSpecializationCallback =
3903 auto &Set = IndirectCalleeTrackingMap[&CB];
3905 Set = std::make_unique<SmallPtrSet<Function *, 8>>();
3907 return Set->contains(&Callee);
3908 Set->insert(&Callee);
3918 if (!
A.isFunctionIPOAmendable(*
F))
3926 unsigned FunSize = Functions.size();
3927 for (
unsigned u = 0; u < FunSize; u++) {
3929 if (!
F->isDeclaration() && !
F->isDefinitionExact() && !
F->use_empty() &&
3932 assert(NewF &&
"Could not internalize function.");
3933 Functions.insert(NewF);
3937 for (
const Use &U : NewF->
uses())
3939 auto *CallerF = CB->getCaller();
3947 if (
F->isDeclaration())
3950 if (
F->hasExactDefinition())
3951 NumFnWithExactDefinition++;
3953 NumFnWithoutExactDefinition++;
3958 if (
F->hasLocalLinkage()) {
3960 const auto *CB = dyn_cast<CallBase>(U.getUser());
3961 return CB && CB->isCallee(&U) &&
3962 Functions.count(const_cast<Function *>(CB->getCaller()));
3969 A.identifyDefaultAbstractAttributes(*
F);
3974 LLVM_DEBUG(
dbgs() <<
"[Attributor] Done with " << Functions.size()
3975 <<
" functions, result: " <<
Changed <<
".\n");
3984 bool IsModulePass) {
3985 if (Functions.empty())
3989 dbgs() <<
"[AttributorLight] Run on module with " << Functions.size()
3998 AC.IsModulePass = IsModulePass;
3999 AC.DeleteFns =
false;
4007 AC.Allowed = &Allowed;
4008 AC.UseLiveness =
false;
4013 if (
F->isDeclaration())
4016 if (
F->hasExactDefinition())
4017 NumFnWithExactDefinition++;
4019 NumFnWithoutExactDefinition++;
4024 if (AC.UseLiveness &&
F->hasLocalLinkage()) {
4026 const auto *CB = dyn_cast<CallBase>(U.getUser());
4027 return CB && CB->isCallee(&U) &&
4028 Functions.count(const_cast<Function *>(CB->getCaller()));
4035 A.identifyDefaultAbstractAttributes(*
F);
4052 for (
auto *U :
Changed->users()) {
4055 FAM.invalidate(*
Call->getFunction(), FuncPA);
4060 LLVM_DEBUG(
dbgs() <<
"[Attributor] Done with " << Functions.size()
4061 <<
" functions, result: " <<
Changed <<
".\n");
4068 static std::atomic<int> CallTimes;
4074 Prefix =
"dep_graph";
4076 Prefix +
"_" + std::to_string(CallTimes.load()) +
".dot";
4078 outs() <<
"Dependency graph dump to " <<
Filename <<
".\n";
4124 Functions.
insert(&
N.getFunction());
4126 if (Functions.empty())
4129 Module &M = *Functions.back()->getParent();
4180 Functions.
insert(&
N.getFunction());
4182 if (Functions.empty())
4185 Module &M = *Functions.back()->getParent();
4237 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.
An information struct used to provide DenseMap with the various necessary components for a given valu...
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.
static LLVM_ABI const IRPosition TombstoneKey
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 LLVM_ABI const IRPosition EmptyKey
Special DenseMap key values.
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.