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);
1338 IRPositions.emplace_back(IRP);
1342 auto CanIgnoreOperandBundles = [](
const CallBase &CB) {
1358 assert(CB &&
"Expected call site!");
1361 if (!CB->hasOperandBundles() || CanIgnoreOperandBundles(*CB))
1366 assert(CB &&
"Expected call site!");
1369 if (!CB->hasOperandBundles() || CanIgnoreOperandBundles(*CB)) {
1374 for (
const Argument &Arg : Callee->args())
1375 if (Arg.hasReturnedAttr()) {
1376 IRPositions.emplace_back(
1378 IRPositions.emplace_back(
1387 assert(CB &&
"Expected call site!");
1390 if (!CB->hasOperandBundles() || CanIgnoreOperandBundles(*CB)) {
1404void IRPosition::verify() {
1405#ifdef EXPENSIVE_CHECKS
1408 assert((CBContext ==
nullptr) &&
1409 "Invalid position must not have CallBaseContext!");
1411 "Expected a nullptr for an invalid position!");
1415 "Expected specialized kind for argument values!");
1419 "Expected function for a 'returned' position!");
1421 "Associated value mismatch!");
1424 assert((CBContext ==
nullptr) &&
1425 "'call site returned' position must not have CallBaseContext!");
1427 "Expected call base for 'call site returned' position!");
1429 "Associated value mismatch!");
1432 assert((CBContext ==
nullptr) &&
1433 "'call site function' position must not have CallBaseContext!");
1435 "Expected call base for 'call site function' position!");
1437 "Associated value mismatch!");
1441 "Expected function for a 'function' position!");
1443 "Associated value mismatch!");
1447 "Expected argument for a 'argument' position!");
1449 "Associated value mismatch!");
1452 assert((CBContext ==
nullptr) &&
1453 "'call site argument' position must not have CallBaseContext!");
1454 Use *U = getAsUsePtr();
1456 assert(U &&
"Expected use for a 'call site argument' position!");
1458 "Expected call base user for a 'call site argument' position!");
1460 "Expected call base argument operand for a 'call site argument' "
1464 "Argument number mismatch!");
1472std::optional<Constant *>
1475 bool &UsedAssumedInformation) {
1479 for (
auto &CB : SimplificationCallbacks.lookup(IRP)) {
1480 std::optional<Value *> SimplifiedV = CB(IRP, &
AA, UsedAssumedInformation);
1482 return std::nullopt;
1492 UsedAssumedInformation)) {
1494 return std::nullopt;
1508 for (
auto &CB : SimplificationCallbacks.lookup(IRP))
1509 return CB(IRP,
AA, UsedAssumedInformation);
1515 return std::nullopt;
1528 bool &UsedAssumedInformation,
bool RecurseForSelectAndPHI) {
1532 while (!Worklist.
empty()) {
1538 int NV = Values.
size();
1539 const auto &SimplificationCBs = SimplificationCallbacks.lookup(IRP);
1540 for (
const auto &CB : SimplificationCBs) {
1541 std::optional<Value *> CBResult = CB(IRP,
AA, UsedAssumedInformation);
1542 if (!CBResult.has_value())
1544 Value *V = *CBResult;
1553 if (SimplificationCBs.empty()) {
1556 const auto *PotentialValuesAA =
1558 if (PotentialValuesAA &&
1559 PotentialValuesAA->getAssumedSimplifiedValues(*
this, Values, S)) {
1560 UsedAssumedInformation |= !PotentialValuesAA->isAtFixpoint();
1569 if (!RecurseForSelectAndPHI)
1572 for (
int I = NV, E = Values.
size();
I < E; ++
I) {
1573 Value *V = Values[
I].getValue();
1576 if (!Seen.
insert(V).second)
1579 Values[
I] = Values[E - 1];
1593 bool &UsedAssumedInformation) {
1601 if (!Arg->hasPointeeInMemoryValueAttr())
1611 for (
auto &It : AAMap) {
1613 AA->~AbstractAttribute();
1619 bool &UsedAssumedInformation,
1620 bool CheckBBLivenessOnly,
DepClassTy DepClass) {
1621 if (!Configuration.UseLiveness)
1627 CheckBBLivenessOnly, DepClass);
1633 bool &UsedAssumedInformation,
1634 bool CheckBBLivenessOnly,
DepClassTy DepClass) {
1635 if (!Configuration.UseLiveness)
1640 UsedAssumedInformation, CheckBBLivenessOnly, DepClass);
1645 if (CB->isArgOperand(&U)) {
1649 UsedAssumedInformation, CheckBBLivenessOnly,
1655 UsedAssumedInformation, CheckBBLivenessOnly, DepClass);
1659 UsedAssumedInformation, CheckBBLivenessOnly, DepClass);
1661 if (!CheckBBLivenessOnly &&
SI->getPointerOperand() != U.get()) {
1669 UsedAssumedInformation =
true;
1676 UsedAssumedInformation, CheckBBLivenessOnly, DepClass);
1682 bool &UsedAssumedInformation,
1683 bool CheckBBLivenessOnly,
DepClassTy DepClass,
1684 bool CheckForDeadStore) {
1685 if (!Configuration.UseLiveness)
1690 if (ManifestAddedBlocks.contains(
I.getParent()))
1699 if (!FnLivenessAA || QueryingAA == FnLivenessAA)
1703 if (CheckBBLivenessOnly ? FnLivenessAA->
isAssumedDead(
I.getParent())
1708 UsedAssumedInformation =
true;
1712 if (CheckBBLivenessOnly)
1720 if (!IsDeadAA || QueryingAA == IsDeadAA)
1727 UsedAssumedInformation =
true;
1735 UsedAssumedInformation =
true;
1745 bool &UsedAssumedInformation,
1746 bool CheckBBLivenessOnly,
DepClassTy DepClass) {
1747 if (!Configuration.UseLiveness)
1758 isAssumedDead(*CtxI, QueryingAA, FnLivenessAA, UsedAssumedInformation,
1763 if (CheckBBLivenessOnly)
1776 if (!IsDeadAA || QueryingAA == IsDeadAA)
1783 UsedAssumedInformation =
true;
1794 if (!Configuration.UseLiveness)
1802 if (!FnLivenessAA || QueryingAA == FnLivenessAA)
1818 return Pred(Callee);
1822 if (!CallEdgesAA || CallEdgesAA->hasUnknownCallee())
1825 const auto &Callees = CallEdgesAA->getOptimisticEdges();
1826 return Pred(Callees.getArrayRef());
1836 bool CheckBBLivenessOnly,
DepClassTy LivenessDepClass,
1837 bool IgnoreDroppableUses,
1842 if (!CB(*
this, &QueryingAA))
1856 auto AddUsers = [&](
const Value &V,
const Use *OldUse) {
1857 for (
const Use &UU : V.uses()) {
1858 if (OldUse && EquivalentUseCB && !EquivalentUseCB(*OldUse, UU)) {
1860 "rejected by the equivalence call back: "
1870 AddUsers(V,
nullptr);
1873 <<
" initial uses to check\n");
1876 const auto *LivenessAA =
1881 while (!Worklist.
empty()) {
1887 dbgs() <<
"[Attributor] Check use: " << **U <<
" in " << Fn->getName()
1890 dbgs() <<
"[Attributor] Check use: " << **U <<
" in " << *U->getUser()
1893 bool UsedAssumedInformation =
false;
1894 if (
isAssumedDead(*U, &QueryingAA, LivenessAA, UsedAssumedInformation,
1895 CheckBBLivenessOnly, LivenessDepClass)) {
1897 dbgs() <<
"[Attributor] Dead use, skip!\n");
1900 if (IgnoreDroppableUses && U->getUser()->isDroppable()) {
1902 dbgs() <<
"[Attributor] Droppable user, skip!\n");
1907 if (&
SI->getOperandUse(0) == U) {
1908 if (!Visited.
insert(U).second)
1912 *
this, *
SI, PotentialCopies, QueryingAA, UsedAssumedInformation,
1916 <<
"[Attributor] Value is stored, continue with "
1917 << PotentialCopies.
size()
1918 <<
" potential copies instead!\n");
1919 for (
Value *PotentialCopy : PotentialCopies)
1920 if (!AddUsers(*PotentialCopy, U))
1927 bool Follow =
false;
1928 if (!Pred(*U, Follow))
1933 User &Usr = *U->getUser();
1934 AddUsers(Usr,
nullptr);
1942 bool RequireAllCallSites,
1943 bool &UsedAssumedInformation) {
1949 if (!AssociatedFunction) {
1950 LLVM_DEBUG(
dbgs() <<
"[Attributor] No function associated with " << IRP
1956 &QueryingAA, UsedAssumedInformation);
1961 bool RequireAllCallSites,
1963 bool &UsedAssumedInformation,
1964 bool CheckPotentiallyDead) {
1968 <<
"[Attributor] Function " << Fn.
getName()
1969 <<
" has no internal linkage, hence not all call sites are known\n");
1974 if (!CB(*
this, QueryingAA))
1978 for (
unsigned u = 0; u <
Uses.size(); ++u) {
1982 dbgs() <<
"[Attributor] Check use: " << Fn->
getName() <<
" in "
1983 << *U.getUser() <<
"\n";
1985 dbgs() <<
"[Attributor] Check use: " << *U <<
" in " << *U.getUser()
1988 if (!CheckPotentiallyDead &&
1989 isAssumedDead(U, QueryingAA,
nullptr, UsedAssumedInformation,
1992 dbgs() <<
"[Attributor] Dead use, skip!\n");
1996 if (CE->isCast() && CE->getType()->isPointerTy()) {
1998 dbgs() <<
"[Attributor] Use, is constant cast expression, add "
1999 << CE->getNumUses() <<
" uses of that expression instead!\n";
2001 for (
const Use &CEU : CE->uses())
2002 Uses.push_back(&CEU);
2010 <<
" has non call site use " << *U.get() <<
" in "
2011 << *U.getUser() <<
"\n");
2015 const Use *EffectiveUse =
2018 if (!RequireAllCallSites) {
2019 LLVM_DEBUG(
dbgs() <<
"[Attributor] User " << *EffectiveUse->getUser()
2020 <<
" is not a call of " << Fn.
getName()
2024 LLVM_DEBUG(
dbgs() <<
"[Attributor] User " << *EffectiveUse->getUser()
2025 <<
" is an invalid use of " << Fn.
getName() <<
"\n");
2033 unsigned MinArgsParams =
2035 for (
unsigned u = 0; u < MinArgsParams; ++u) {
2039 dbgs() <<
"[Attributor] Call site / callee argument type mismatch ["
2040 << u <<
"@" << Fn.
getName() <<
": "
2050 LLVM_DEBUG(
dbgs() <<
"[Attributor] Call site callback failed for "
2058bool Attributor::shouldPropagateCallBaseContext(
const IRPosition &IRP) {
2068 bool RecurseForSelectAndPHI) {
2072 if (!AssociatedFunction)
2075 bool UsedAssumedInformation =
false;
2079 UsedAssumedInformation, RecurseForSelectAndPHI))
2091 bool &UsedAssumedInformation,
bool CheckBBLivenessOnly =
false,
2092 bool CheckPotentiallyDead =
false) {
2093 for (
unsigned Opcode : Opcodes) {
2095 auto *Insts = OpcodeInstMap.
lookup(Opcode);
2101 if (
A && !CheckPotentiallyDead &&
2103 UsedAssumedInformation, CheckBBLivenessOnly)) {
2105 dbgs() <<
"[Attributor] Instruction " << *
I
2106 <<
" is potentially dead, skip!\n";);
2121 bool &UsedAssumedInformation,
2122 bool CheckBBLivenessOnly,
2123 bool CheckPotentiallyDead) {
2129 const auto *LivenessAA =
2130 CheckPotentiallyDead && QueryingAA
2134 auto &OpcodeInstMap = InfoCache.getOpcodeInstMapForFunction(*Fn);
2136 LivenessAA, Opcodes, UsedAssumedInformation,
2137 CheckBBLivenessOnly, CheckPotentiallyDead))
2146 bool &UsedAssumedInformation,
2147 bool CheckBBLivenessOnly,
2148 bool CheckPotentiallyDead) {
2152 UsedAssumedInformation, CheckBBLivenessOnly,
2153 CheckPotentiallyDead);
2158 bool &UsedAssumedInformation) {
2161 const Function *AssociatedFunction =
2163 if (!AssociatedFunction)
2167 const auto *LivenessAA =
2171 InfoCache.getReadOrWriteInstsForFunction(*AssociatedFunction)) {
2174 UsedAssumedInformation))
2184void Attributor::runTillFixpoint() {
2188 <<
" abstract attributes.\n");
2193 unsigned IterationCounter = 1;
2194 unsigned MaxIterations =
2204 LLVM_DEBUG(
dbgs() <<
"\n\n[Attributor] #Iteration: " << IterationCounter
2205 <<
", Worklist size: " << Worklist.
size() <<
"\n");
2210 for (
unsigned u = 0; u < InvalidAAs.
size(); ++u) {
2215 dbgs() <<
"[Attributor] InvalidAA: " << *InvalidAA
2216 <<
" has " << InvalidAA->
Deps.
size()
2217 <<
" required & optional dependences\n");
2218 for (
auto &DepIt : InvalidAA->
Deps) {
2222 dbgs() <<
" - recompute: " << *DepAA);
2227 <<
" - invalidate: " << *DepAA);
2231 InvalidAAs.
insert(DepAA);
2240 for (AbstractAttribute *ChangedAA : ChangedAAs) {
2241 for (
auto &DepIt : ChangedAA->Deps)
2243 ChangedAA->Deps.clear();
2246 LLVM_DEBUG(
dbgs() <<
"[Attributor] #Iteration: " << IterationCounter
2247 <<
", Worklist+Dependent size: " << Worklist.
size()
2256 for (AbstractAttribute *AA : Worklist) {
2257 const auto &AAState = AA->getState();
2258 if (!AAState.isAtFixpoint())
2260 ChangedAAs.push_back(AA);
2264 if (!AAState.isValidState())
2270 ChangedAAs.append(DG.SyntheticRoot.begin() + NumAAs,
2271 DG.SyntheticRoot.end());
2276 Worklist.insert_range(ChangedAAs);
2277 Worklist.insert_range(QueryAAsAwaitingUpdate);
2278 QueryAAsAwaitingUpdate.clear();
2280 }
while (!Worklist.empty() && (IterationCounter++ < MaxIterations));
2282 if (IterationCounter > MaxIterations && !Functions.empty()) {
2283 auto Remark = [&](OptimizationRemarkMissed ORM) {
2284 return ORM <<
"Attributor did not reach a fixpoint after "
2285 <<
ore::NV(
"Iterations", MaxIterations) <<
" iterations.";
2291 LLVM_DEBUG(
dbgs() <<
"\n[Attributor] Fixpoint iteration done after: "
2292 << IterationCounter <<
"/" << MaxIterations
2293 <<
" iterations\n");
2300 SmallPtrSet<AbstractAttribute *, 32> Visited;
2301 for (
unsigned u = 0;
u < ChangedAAs.size();
u++) {
2302 AbstractAttribute *ChangedAA = ChangedAAs[
u];
2303 if (!Visited.
insert(ChangedAA).second)
2306 AbstractState &State = ChangedAA->
getState();
2310 NumAttributesTimedOut++;
2313 for (
auto &DepIt : ChangedAA->
Deps)
2319 if (!Visited.
empty())
2320 dbgs() <<
"\n[Attributor] Finalized " << Visited.
size()
2321 <<
" abstract attributes.\n";
2327 "Non-query AAs should not be required to register for updates!");
2328 QueryAAsAwaitingUpdate.insert(&
AA);
2335 unsigned NumManifested = 0;
2336 unsigned NumAtFixpoint = 0;
2346 if (!State.isAtFixpoint())
2347 State.indicateOptimisticFixpoint();
2350 if (
AA->hasCallBaseContext())
2353 if (!State.isValidState())
2356 if (
AA->getCtxI() && !
isRunOn(*
AA->getAnchorScope()))
2360 bool UsedAssumedInformation =
false;
2371 AA->trackStatistics();
2372 LLVM_DEBUG(
dbgs() <<
"[Attributor] Manifest " << LocalChange <<
" : " << *
AA
2375 ManifestChange = ManifestChange | LocalChange;
2381 (void)NumManifested;
2382 (void)NumAtFixpoint;
2383 LLVM_DEBUG(
dbgs() <<
"\n[Attributor] Manifested " << NumManifested
2384 <<
" arguments while " << NumAtFixpoint
2385 <<
" were in a valid fixpoint state\n");
2387 NumAttributesManifested += NumManifested;
2388 NumAttributesValidFixpoint += NumAtFixpoint;
2393 for (
unsigned u = 0; u < NumFinalAAs; ++u)
2397 errs() <<
"Unexpected abstract attribute: "
2401 .getAssociatedValue()
2405 "remain unchanged!");
2408 for (
auto &It : AttrsMap) {
2409 AttributeList &
AL = It.getSecond();
2410 const IRPosition &IRP =
2413 : IRPosition::callsite_function(*
cast<CallBase>(It.getFirst()));
2417 return ManifestChange;
2420void Attributor::identifyDeadInternalFunctions() {
2422 if (!Configuration.DeleteFns)
2440 for (Function *
F : Functions)
2441 if (
F->hasLocalLinkage() && (
isModulePass() || !TLI->getLibFunc(*
F, LF)))
2444 SmallPtrSet<Function *, 8> LiveInternalFns;
2445 bool FoundLiveInternal =
true;
2446 while (FoundLiveInternal) {
2447 FoundLiveInternal =
false;
2448 for (Function *&
F : InternalFns) {
2452 bool UsedAssumedInformation =
false;
2454 [&](AbstractCallSite ACS) {
2456 return ToBeDeletedFunctions.count(Callee) ||
2457 (Functions.count(Callee) &&
Callee->hasLocalLinkage() &&
2458 !LiveInternalFns.
count(Callee));
2460 *
F,
true,
nullptr, UsedAssumedInformation)) {
2466 FoundLiveInternal =
true;
2470 for (Function *
F : InternalFns)
2472 ToBeDeletedFunctions.insert(
F);
2476 TimeTraceScope TimeScope(
"Attributor::cleanupIR");
2479 << ToBeDeletedFunctions.size() <<
" functions and "
2480 << ToBeDeletedBlocks.size() <<
" blocks and "
2481 << ToBeDeletedInsts.size() <<
" instructions and "
2482 << ToBeChangedValues.size() <<
" values and "
2483 << ToBeChangedUses.size() <<
" uses. To insert "
2484 << ToBeChangedToUnreachableInsts.size()
2485 <<
" unreachables.\n"
2486 <<
"Preserve manifest added " << ManifestAddedBlocks.size()
2492 auto ReplaceUse = [&](
Use *
U,
Value *NewV) {
2497 const auto &
Entry = ToBeChangedValues.lookup(NewV);
2505 "Cannot replace an instruction outside the current SCC!");
2511 if (CI->isMustTailCall() && !ToBeDeletedInsts.count(CI))
2516 for (
auto &Arg : RI->getFunction()->args())
2517 Arg.removeAttr(Attribute::Returned);
2521 <<
" instead of " << *OldV <<
"\n");
2525 CGModifiedFunctions.insert(
I->getFunction());
2532 if (CB->isArgOperand(U)) {
2533 unsigned Idx = CB->getArgOperandNo(U);
2534 CB->removeParamAttr(Idx, Attribute::NoUndef);
2536 if (Callee &&
Callee->arg_size() > Idx)
2537 Callee->removeParamAttr(Idx, Attribute::NoUndef);
2543 ToBeChangedToUnreachableInsts.insert(UserI);
2550 for (
auto &It : ToBeChangedUses) {
2552 Value *NewV = It.second;
2553 ReplaceUse(U, NewV);
2557 for (
auto &It : ToBeChangedValues) {
2558 Value *OldV = It.first;
2559 auto [NewV,
Done] = It.second;
2561 for (
auto &U : OldV->
uses())
2562 if (
Done || !
U.getUser()->isDroppable())
2564 for (Use *U :
Uses) {
2568 ReplaceUse(U, NewV);
2572 for (
const auto &V : InvokeWithDeadSuccessor)
2575 "Cannot replace an invoke outside the current SCC!");
2576 bool UnwindBBIsDead =
II->hasFnAttr(Attribute::NoUnwind);
2577 bool NormalBBIsDead =
II->hasFnAttr(Attribute::NoReturn);
2578 bool Invoke2CallAllowed =
2580 assert((UnwindBBIsDead || NormalBBIsDead) &&
2581 "Invoke does not have dead successors!");
2584 if (UnwindBBIsDead) {
2586 if (Invoke2CallAllowed) {
2591 ToBeChangedToUnreachableInsts.insert(NormalNextIP);
2593 assert(NormalBBIsDead &&
"Broken invariant!");
2596 ToBeChangedToUnreachableInsts.insert(&NormalDestBB->
front());
2599 for (Instruction *
I : TerminatorsToFold) {
2601 "Cannot replace a terminator outside the current SCC!");
2602 CGModifiedFunctions.insert(
I->getFunction());
2605 for (
const auto &V : ToBeChangedToUnreachableInsts)
2610 "Cannot replace an instruction outside the current SCC!");
2611 CGModifiedFunctions.insert(
I->getFunction());
2615 for (
const auto &V : ToBeDeletedInsts) {
2619 "Cannot delete an instruction outside the current SCC!");
2620 I->dropDroppableUses();
2621 CGModifiedFunctions.insert(
I->getFunction());
2622 if (!
I->getType()->isVoidTy())
2627 I->eraseFromParent();
2634 dbgs() <<
"[Attributor] DeadInsts size: " << DeadInsts.
size() <<
"\n";
2635 for (
auto &
I : DeadInsts)
2637 dbgs() <<
" - " << *
I <<
"\n";
2642 if (
unsigned NumDeadBlocks = ToBeDeletedBlocks.size()) {
2643 SmallVector<BasicBlock *, 8> ToBeDeletedBBs;
2644 ToBeDeletedBBs.
reserve(NumDeadBlocks);
2645 for (BasicBlock *BB : ToBeDeletedBlocks) {
2647 "Cannot delete a block outside the current SCC!");
2648 CGModifiedFunctions.insert(BB->
getParent());
2650 if (ManifestAddedBlocks.contains(BB))
2660 identifyDeadInternalFunctions();
2663 ChangeStatus ManifestChange = rewriteFunctionSignatures(CGModifiedFunctions);
2665 for (Function *Fn : CGModifiedFunctions)
2666 if (!ToBeDeletedFunctions.count(Fn) && Functions.count(Fn))
2667 Configuration.CGUpdater.reanalyzeFunction(*Fn);
2669 for (Function *Fn : ToBeDeletedFunctions) {
2670 if (!Functions.count(Fn))
2672 Configuration.CGUpdater.removeFunction(*Fn);
2675 if (!ToBeChangedUses.empty())
2678 if (!ToBeChangedToUnreachableInsts.empty())
2681 if (!ToBeDeletedFunctions.empty())
2684 if (!ToBeDeletedBlocks.empty())
2687 if (!ToBeDeletedInsts.empty())
2690 if (!InvokeWithDeadSuccessor.empty())
2693 if (!DeadInsts.empty())
2696 NumFnDeleted += ToBeDeletedFunctions.size();
2698 LLVM_DEBUG(
dbgs() <<
"[Attributor] Deleted " << ToBeDeletedFunctions.size()
2699 <<
" functions after manifest.\n");
2701#ifdef EXPENSIVE_CHECKS
2702 for (Function *
F : Functions) {
2703 if (ToBeDeletedFunctions.count(
F))
2709 return ManifestChange;
2714 AttributorCallGraph ACallGraph(*
this);
2719 Phase = AttributorPhase::UPDATE;
2732 Phase = AttributorPhase::MANIFEST;
2735 Phase = AttributorPhase::CLEANUP;
2741 return ManifestChange | CleanupChange;
2746 return AA.getName().str() +
2747 std::to_string(
AA.getIRPosition().getPositionKind());
2750 "We can update AA only in the update stage!");
2753 DependenceVector DV;
2754 DependenceStack.push_back(&DV);
2756 auto &AAState =
AA.getState();
2758 bool UsedAssumedInformation =
false;
2761 CS =
AA.update(*
this);
2763 if (!
AA.isQueryAA() && DV.empty() && !
AA.getState().isAtFixpoint()) {
2770 RerunCS =
AA.update(*
this);
2776 AAState.indicateOptimisticFixpoint();
2779 if (!AAState.isAtFixpoint())
2780 rememberDependences();
2784 DependenceVector *PoppedDV = DependenceStack.pop_back_val();
2786 assert(PoppedDV == &DV &&
"Inconsistent usage of the dependence stack!");
2792 assert(!
F.isDeclaration() &&
"Cannot create a wrapper around a declaration!");
2801 M.getFunctionList().insert(
F.getIterator(),
Wrapper);
2806 assert(
F.use_empty() &&
"Uses remained after wrapper was created!");
2811 F.setComdat(
nullptr);
2815 F.getAllMetadata(MDs);
2816 for (
auto MDIt : MDs)
2817 Wrapper->addMetadata(MDIt.first, *MDIt.second);
2818 Wrapper->setAttributes(
F.getAttributes());
2826 Args.push_back(&Arg);
2827 Arg.setName((FArgIt++)->
getName());
2835 NumFnShallowWrappersCreated++;
2839 if (
F.isDeclaration() ||
F.hasLocalLinkage() ||
2855 return InternalizedFns[&
F];
2873 F->getName() +
".internalized");
2876 for (
auto &Arg :
F->args()) {
2877 auto ArgName = Arg.getName();
2878 NewFArgIt->setName(ArgName);
2879 VMap[&Arg] = &(*NewFArgIt++);
2894 F->getAllMetadata(MDs);
2895 for (
auto MDIt : MDs)
2899 M.getFunctionList().insert(
F->getIterator(), Copied);
2907 auto &InternalizedFn = FnMap[
F];
2908 auto IsNotInternalized = [&](
Use &U) ->
bool {
2910 return !FnMap.
lookup(CB->getCaller());
2913 F->replaceUsesWithIf(InternalizedFn, IsNotInternalized);
2922 if (!Configuration.RewriteSignatures)
2945 LLVM_DEBUG(
dbgs() <<
"[Attributor] Cannot rewrite var-args functions\n");
2951 if (FnAttributeList.hasAttrSomewhere(Attribute::Nest) ||
2952 FnAttributeList.hasAttrSomewhere(Attribute::StructRet) ||
2953 FnAttributeList.hasAttrSomewhere(Attribute::InAlloca) ||
2954 FnAttributeList.hasAttrSomewhere(Attribute::Preallocated)) {
2956 dbgs() <<
"[Attributor] Cannot rewrite due to complex attribute\n");
2961 bool UsedAssumedInformation =
false;
2963 UsedAssumedInformation,
2965 LLVM_DEBUG(
dbgs() <<
"[Attributor] Cannot rewrite all call sites\n");
2971 return !CI->isMustTailCall();
2977 auto &OpcodeInstMap = InfoCache.getOpcodeInstMapForFunction(*Fn);
2979 nullptr, {Instruction::Call},
2980 UsedAssumedInformation)) {
2981 LLVM_DEBUG(
dbgs() <<
"[Attributor] Cannot rewrite due to instructions\n");
2992 LLVM_DEBUG(
dbgs() <<
"[Attributor] Register new rewrite of " << Arg <<
" in "
2994 << ReplacementTypes.
size() <<
" replacements\n");
2996 "Cannot register an invalid rewrite");
3000 ArgumentReplacementMap[Fn];
3006 std::unique_ptr<ArgumentReplacementInfo> &ARI = ARIs[Arg.
getArgNo()];
3007 if (ARI && ARI->getNumReplacementArgs() <= ReplacementTypes.
size()) {
3008 LLVM_DEBUG(
dbgs() <<
"[Attributor] Existing rewrite is preferred\n");
3016 LLVM_DEBUG(
dbgs() <<
"[Attributor] Register new rewrite of " << Arg <<
" in "
3018 << ReplacementTypes.
size() <<
" replacements\n");
3022 std::move(CalleeRepairCB),
3023 std::move(ACSRepairCB)));
3044 for (
auto &It : ArgumentReplacementMap) {
3048 if (!Functions.count(OldFn) || ToBeDeletedFunctions.count(OldFn))
3061 if (
const std::unique_ptr<ArgumentReplacementInfo> &ARI =
3062 ARIs[Arg.getArgNo()]) {
3063 NewArgumentTypes.
append(ARI->ReplacementTypes.begin(),
3064 ARI->ReplacementTypes.end());
3065 NewArgumentAttributes.
append(ARI->getNumReplacementArgs(),
3068 NewArgumentTypes.
push_back(Arg.getType());
3070 OldFnAttributeList.getParamAttrs(Arg.getArgNo()));
3074 uint64_t LargestVectorWidth = 0;
3075 for (
auto *
I : NewArgumentTypes)
3077 LargestVectorWidth =
3078 std::max(LargestVectorWidth,
3079 VT->getPrimitiveSizeInBits().getKnownMinValue());
3082 Type *RetTy = OldFnTy->getReturnType();
3085 FunctionType *NewFnTy =
3090 << *NewFnTy <<
"\n");
3095 Functions.insert(NewFn);
3108 Ctx, OldFnAttributeList.getFnAttrs(), OldFnAttributeList.getRetAttrs(),
3109 NewArgumentAttributes));
3110 AttributeFuncs::updateMinLegalVectorWidthAttr(*NewFn, LargestVectorWidth);
3118 return !
T->isPtrOrPtrVectorTy() ||
3134 auto CallSiteReplacementCreator = [&](AbstractCallSite ACS) {
3136 const AttributeList &OldCallAttributeList = OldCB->
getAttributes();
3139 SmallVector<Value *, 16> NewArgOperands;
3141 for (
unsigned OldArgNum = 0; OldArgNum < ARIs.
size(); ++OldArgNum) {
3142 unsigned NewFirstArgNum = NewArgOperands.
size();
3143 (void)NewFirstArgNum;
3144 if (
const std::unique_ptr<ArgumentReplacementInfo> &ARI =
3146 if (ARI->ACSRepairCB)
3147 ARI->ACSRepairCB(*ARI, ACS, NewArgOperands);
3148 assert(ARI->getNumReplacementArgs() + NewFirstArgNum ==
3149 NewArgOperands.
size() &&
3150 "ACS repair callback did not provide as many operand as new "
3151 "types were registered!");
3153 NewArgOperandAttributes.
append(ARI->ReplacementTypes.size(),
3158 OldCallAttributeList.getParamAttrs(OldArgNum));
3162 assert(NewArgOperands.
size() == NewArgOperandAttributes.
size() &&
3163 "Mismatch # argument operands vs. # argument operand attributes!");
3165 "Mismatch # argument operands vs. # function arguments!");
3174 II->getUnwindDest(), NewArgOperands,
3179 NewCI->setTailCallKind(
cast<CallInst>(OldCB)->getTailCallKind());
3184 NewCB->
copyMetadata(*OldCB, {LLVMContext::MD_prof, LLVMContext::MD_dbg});
3188 Ctx, OldCallAttributeList.getFnAttrs(),
3189 OldCallAttributeList.getRetAttrs(), NewArgOperandAttributes));
3191 AttributeFuncs::updateMinLegalVectorWidthAttr(*NewCB->
getCaller(),
3192 LargestVectorWidth);
3194 CallSitePairs.
push_back({OldCB, NewCB});
3199 bool UsedAssumedInformation =
false;
3201 true,
nullptr, UsedAssumedInformation,
3204 assert(
Success &&
"Assumed call site replacement to succeed!");
3209 for (
unsigned OldArgNum = 0; OldArgNum < ARIs.
size();
3210 ++OldArgNum, ++OldFnArgIt) {
3211 if (
const std::unique_ptr<ArgumentReplacementInfo> &ARI =
3213 if (ARI->CalleeRepairCB)
3214 ARI->CalleeRepairCB(*ARI, *NewFn, NewFnArgIt);
3215 if (ARI->ReplacementTypes.empty())
3218 NewFnArgIt += ARI->ReplacementTypes.size();
3220 NewFnArgIt->
takeName(&*OldFnArgIt);
3227 for (
auto &CallSitePair : CallSitePairs) {
3228 CallBase &OldCB = *CallSitePair.first;
3229 CallBase &NewCB = *CallSitePair.second;
3231 "Cannot handle call sites with different types!");
3238 Configuration.CGUpdater.replaceFunctionWith(*OldFn, *NewFn);
3242 if (ModifiedFns.
remove(OldFn))
3243 ModifiedFns.
insert(NewFn);
3251void InformationCache::initializeInformationCache(
const Function &CF,
3258 FI.IsKernel =
F.hasFnAttribute(
"kernel");
3264 DenseMap<const Value *, std::optional<short>> AssumeUsesMap;
3269 auto AddToAssumeUsesMap = [&](
const Value &
V) ->
void {
3270 SmallVector<const Instruction *> Worklist;
3273 while (!Worklist.
empty()) {
3275 std::optional<short> &NumUses = AssumeUsesMap[
I];
3277 NumUses =
I->getNumUses();
3278 NumUses = *NumUses - 1;
3281 AssumeOnlyValues.insert(
I);
3282 for (
const Value *
Op :
I->operands())
3289 bool IsInterestingOpcode =
false;
3296 switch (
I.getOpcode()) {
3299 "New call base instruction type needs to be known in the "
3302 case Instruction::Call:
3307 AssumeOnlyValues.insert(Assume);
3309 AddToAssumeUsesMap(*
Assume->getArgOperand(0));
3311 FI.ContainsMustTailCall =
true;
3314 getFunctionInfo(*Callee).CalledViaMustTail =
true;
3317 case Instruction::CallBr:
3318 case Instruction::Invoke:
3319 case Instruction::CleanupRet:
3320 case Instruction::CatchSwitch:
3321 case Instruction::AtomicRMW:
3322 case Instruction::AtomicCmpXchg:
3323 case Instruction::UncondBr:
3324 case Instruction::CondBr:
3325 case Instruction::Resume:
3326 case Instruction::Ret:
3327 case Instruction::Load:
3329 case Instruction::Store:
3331 case Instruction::Alloca:
3332 case Instruction::AddrSpaceCast:
3333 IsInterestingOpcode =
true;
3335 if (IsInterestingOpcode) {
3336 auto *&Insts = FI.OpcodeInstMap[
I.getOpcode()];
3341 if (
I.mayReadOrWriteMemory())
3342 FI.RWInsts.push_back(&
I);
3345 if (
F.hasFnAttribute(Attribute::AlwaysInline) &&
3347 InlineableFunctions.insert(&
F);
3350InformationCache::FunctionInfo::~FunctionInfo() {
3353 for (
auto &It : OpcodeInstMap)
3354 It.getSecond()->~InstructionVectorTy();
3359 assert(
A.isClosedWorldModule() &&
"Cannot see all indirect callees!");
3360 return IndirectlyCallableFunctions;
3366 return std::nullopt;
3377 if (DependenceStack.empty())
3381 DependenceStack.back()->push_back({&FromAA, &ToAA, DepClass});
3384void Attributor::rememberDependences() {
3385 assert(!DependenceStack.empty() &&
"No dependences to remember!");
3387 for (DepInfo &DI : *DependenceStack.back()) {
3390 "Expected required or optional dependence (1 bit)!");
3397template <Attribute::AttrKind AK,
typename AAType>
3398void Attributor::checkAndQueryIRAttr(
const IRPosition &IRP, AttributeSet Attrs,
3399 bool SkipHasAttrCheck) {
3401 if (SkipHasAttrCheck || !
Attrs.hasAttribute(AK))
3402 if (!Configuration.Allowed || Configuration.Allowed->count(&AAType::ID))
3405 getOrCreateAAFor<AAType>(IRP);
3411 if (!VisitedFunctions.insert(&
F).second)
3417 InformationCache::FunctionInfo &FI = InfoCache.getFunctionInfo(
F);
3419 for (
const Use &U :
F.uses())
3421 if (CB->isCallee(&U) && CB->isMustTailCall())
3422 FI.CalledViaMustTail =
true;
3427 auto Attrs =
F.getAttributes();
3428 auto FnAttrs = Attrs.getFnAttrs();
3444 checkAndQueryIRAttr<Attribute::MustProgress, AAMustProgress>(FPos, FnAttrs);
3447 checkAndQueryIRAttr<Attribute::NoFree, AANoFree>(FPos, FnAttrs);
3450 checkAndQueryIRAttr<Attribute::WillReturn, AAWillReturn>(FPos, FnAttrs);
3453 checkAndQueryIRAttr<Attribute::NoSync, AANoSync>(FPos, FnAttrs);
3458 if (IsIPOAmendable) {
3461 checkAndQueryIRAttr<Attribute::NoUnwind, AANoUnwind>(FPos, FnAttrs);
3464 checkAndQueryIRAttr<Attribute::NoReturn, AANoReturn>(FPos, FnAttrs);
3467 checkAndQueryIRAttr<Attribute::NoRecurse, AANoRecurse>(FPos, FnAttrs);
3470 if (Attrs.hasFnAttr(Attribute::Convergent))
3491 Type *ReturnType =
F.getReturnType();
3492 if (!ReturnType->isVoidTy()) {
3500 bool UsedAssumedInformation =
false;
3505 checkAndQueryIRAttr<Attribute::NoUndef, AANoUndef>(RetPos, RetAttrs);
3507 if (ReturnType->isPointerTy()) {
3513 checkAndQueryIRAttr<Attribute::NonNull, AANonNull>(RetPos, RetAttrs);
3516 checkAndQueryIRAttr<Attribute::NoAlias, AANoAlias>(RetPos, RetAttrs);
3521 }
else if (AttributeFuncs::isNoFPClassCompatibleType(ReturnType)) {
3529 auto ArgNo = Arg.getArgNo();
3532 if (!IsIPOAmendable) {
3533 if (Arg.getType()->isPointerTy())
3535 checkAndQueryIRAttr<Attribute::NoFree, AANoFree>(ArgPos, ArgAttrs);
3542 bool UsedAssumedInformation =
false;
3550 checkAndQueryIRAttr<Attribute::NoUndef, AANoUndef>(ArgPos, ArgAttrs);
3552 if (Arg.getType()->isPointerTy()) {
3554 checkAndQueryIRAttr<Attribute::NonNull, AANonNull>(ArgPos, ArgAttrs);
3557 checkAndQueryIRAttr<Attribute::NoAlias, AANoAlias>(ArgPos, ArgAttrs);
3566 checkAndQueryIRAttr<Attribute::Captures, AANoCapture>(
3567 ArgPos, ArgAttrs,
true);
3574 checkAndQueryIRAttr<Attribute::NoFree, AANoFree>(ArgPos, ArgAttrs);
3579 }
else if (AttributeFuncs::isNoFPClassCompatibleType(Arg.getType())) {
3607 !Callee->hasMetadata(LLVMContext::MD_callback))
3610 if (!Callee->getReturnType()->isVoidTy() && !CB.use_empty()) {
3612 bool UsedAssumedInformation =
false;
3616 if (AttributeFuncs::isNoFPClassCompatibleType(Callee->getReturnType()))
3620 const AttributeList &CBAttrs = CBFnPos.
getAttrList();
3621 for (
int I = 0, E = CB.arg_size();
I < E; ++
I) {
3632 bool UsedAssumedInformation =
false;
3637 checkAndQueryIRAttr<Attribute::NoUndef, AANoUndef>(CBArgPos, CBArgAttrs);
3639 Type *ArgTy = CB.getArgOperand(
I)->getType();
3642 if (AttributeFuncs::isNoFPClassCompatibleType(ArgTy))
3649 checkAndQueryIRAttr<Attribute::NonNull, AANonNull>(CBArgPos, CBArgAttrs);
3652 checkAndQueryIRAttr<Attribute::Captures, AANoCapture>(
3653 CBArgPos, CBArgAttrs,
true);
3656 checkAndQueryIRAttr<Attribute::NoAlias, AANoAlias>(CBArgPos, CBArgAttrs);
3666 if (!CBAttrs.hasParamAttr(
I, Attribute::ReadNone))
3670 checkAndQueryIRAttr<Attribute::NoFree, AANoFree>(CBArgPos, CBArgAttrs);
3675 auto &OpcodeInstMap = InfoCache.getOpcodeInstMapForFunction(
F);
3676 [[maybe_unused]]
bool Success;
3677 bool UsedAssumedInformation =
false;
3679 nullptr, OpcodeInstMap, CallSitePred,
nullptr,
nullptr,
3680 {(
unsigned)Instruction::Invoke, (
unsigned)Instruction::CallBr,
3682 UsedAssumedInformation);
3683 assert(
Success &&
"Expected the check call to be successful!");
3707 nullptr, OpcodeInstMap, LoadStorePred,
nullptr,
nullptr,
3708 {(
unsigned)Instruction::Load, (
unsigned)Instruction::Store},
3709 UsedAssumedInformation);
3710 assert(
Success &&
"Expected the check call to be successful!");
3713 auto AAAllocationInfoPred = [&](
Instruction &
I) ->
bool {
3719 nullptr, OpcodeInstMap, AAAllocationInfoPred,
nullptr,
nullptr,
3720 {(
unsigned)Instruction::Alloca}, UsedAssumedInformation);
3721 assert(
Success &&
"Expected the check call to be successful!");
3727 return isModulePass() && Configuration.IsClosedWorldModule;
3744 return OS <<
"fn_ret";
3746 return OS <<
"cs_ret";
3754 return OS <<
"cs_arg";
3776 return OS << static_cast<const AbstractState &>(S);
3790 OS <<
"set-state(< {";
3806 OS <<
"set-state(< {";
3812 OS <<
"@" <<
F->getName() <<
"[" << int(It.second) <<
"], ";
3814 OS << *It.first.getValue() <<
"[" << int(It.second) <<
"], ";
3827 OS <<
"] for CtxI ";
3834 OS <<
"<<null inst>>";
3843 for (
const auto &DepAA :
Deps) {
3844 auto *
AA = DepAA.getPointer();
3861 OS <<
" [ <unknown> ]";
3875 bool DeleteFns,
bool IsModulePass) {
3876 if (Functions.empty())
3880 dbgs() <<
"[Attributor] Run on module with " << Functions.size()
3889 AC.IsModulePass = IsModulePass;
3890 AC.DeleteFns = DeleteFns;
3894 IndirectCalleeTrackingMap;
3896 AC.IndirectCalleeSpecializationCallback =
3901 auto &Set = IndirectCalleeTrackingMap[&CB];
3903 Set = std::make_unique<SmallPtrSet<Function *, 8>>();
3905 return Set->contains(&Callee);
3906 Set->insert(&Callee);
3916 if (!
A.isFunctionIPOAmendable(*
F))
3924 unsigned FunSize = Functions.size();
3925 for (
unsigned u = 0; u < FunSize; u++) {
3927 if (!
F->isDeclaration() && !
F->isDefinitionExact() && !
F->use_empty() &&
3930 assert(NewF &&
"Could not internalize function.");
3931 Functions.insert(NewF);
3935 for (
const Use &U : NewF->
uses())
3937 auto *CallerF = CB->getCaller();
3945 if (
F->isDeclaration())
3948 if (
F->hasExactDefinition())
3949 NumFnWithExactDefinition++;
3951 NumFnWithoutExactDefinition++;
3956 if (
F->hasLocalLinkage()) {
3958 const auto *CB = dyn_cast<CallBase>(U.getUser());
3959 return CB && CB->isCallee(&U) &&
3960 Functions.count(const_cast<Function *>(CB->getCaller()));
3967 A.identifyDefaultAbstractAttributes(*
F);
3972 LLVM_DEBUG(
dbgs() <<
"[Attributor] Done with " << Functions.size()
3973 <<
" functions, result: " <<
Changed <<
".\n");
3982 bool IsModulePass) {
3983 if (Functions.empty())
3987 dbgs() <<
"[AttributorLight] Run on module with " << Functions.size()
3996 AC.IsModulePass = IsModulePass;
3997 AC.DeleteFns =
false;
4005 AC.Allowed = &Allowed;
4006 AC.UseLiveness =
false;
4011 if (
F->isDeclaration())
4014 if (
F->hasExactDefinition())
4015 NumFnWithExactDefinition++;
4017 NumFnWithoutExactDefinition++;
4022 if (AC.UseLiveness &&
F->hasLocalLinkage()) {
4024 const auto *CB = dyn_cast<CallBase>(U.getUser());
4025 return CB && CB->isCallee(&U) &&
4026 Functions.count(const_cast<Function *>(CB->getCaller()));
4033 A.identifyDefaultAbstractAttributes(*
F);
4050 for (
auto *U :
Changed->users()) {
4053 FAM.invalidate(*
Call->getFunction(), FuncPA);
4058 LLVM_DEBUG(
dbgs() <<
"[Attributor] Done with " << Functions.size()
4059 <<
" functions, result: " <<
Changed <<
".\n");
4066 static std::atomic<int> CallTimes;
4072 Prefix =
"dep_graph";
4074 Prefix +
"_" + std::to_string(CallTimes.load()) +
".dot";
4076 outs() <<
"Dependency graph dump to " <<
Filename <<
".\n";
4122 Functions.
insert(&
N.getFunction());
4124 if (Functions.empty())
4127 Module &M = *Functions.back()->getParent();
4178 Functions.
insert(&
N.getFunction());
4180 if (Functions.empty())
4183 Module &M = *Functions.back()->getParent();
4235 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.
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.