55#define DEBUG_TYPE "aa"
60STATISTIC(NumMayAlias,
"Number of MayAlias results");
61STATISTIC(NumMustAlias,
"Number of MustAlias results");
77 : TLI(Arg.TLI), AAs(
std::
move(Arg.AAs)), AADeps(
std::
move(Arg.AADeps)) {}
88 if (!PAC.preservedWhenStateless())
108 return alias(LocA, LocB, AAQIP,
nullptr);
117 for (
unsigned I = 0;
I < AAQI.
Depth; ++
I)
119 dbgs() <<
"Start " << *LocA.
Ptr <<
" @ " << LocA.
Size <<
", "
120 << *LocB.
Ptr <<
" @ " << LocB.
Size <<
"\n";
124 for (
const auto &AA : AAs) {
125 Result = AA->alias(LocA, LocB, AAQI, CtxI);
132 for (
unsigned I = 0;
I < AAQI.
Depth; ++
I)
134 dbgs() <<
"End " << *LocA.
Ptr <<
" @ " << LocA.
Size <<
", "
135 << *LocB.
Ptr <<
" @ " << LocB.
Size <<
" = " << Result <<
"\n";
138 if (AAQI.
Depth == 0) {
159 for (
const auto &AA : AAs) {
160 Result &= AA->getModRefInfoMask(Loc, AAQI, IgnoreLocals);
173 for (
const auto &AA : AAs) {
174 Result &= AA->getArgModRefInfo(Call, ArgIdx);
193 if (
const auto *Call1 = dyn_cast<CallBase>(
I)) {
198 if (
I->isFenceLike())
216 for (
const auto &AA : AAs) {
217 Result &= AA->getModRefInfo(Call, Loc, AAQI);
231 if (ME.doesNotAccessMemory())
236 if ((ArgMR | OtherMR) != OtherMR) {
242 const Value *Arg =
I.value();
245 unsigned ArgIdx =
I.index();
251 ArgMR &= AllArgsMask;
254 Result &= ArgMR | OtherMR;
269 for (
const auto &AA : AAs) {
270 Result &= AA->getModRefInfo(Call1, Call2, AAQI);
282 if (Call1B.doesNotAccessMemory())
286 if (Call2B.doesNotAccessMemory())
290 if (Call1B.onlyReadsMemory() && Call2B.onlyReadsMemory())
295 if (Call1B.onlyReadsMemory())
297 else if (Call1B.onlyWritesMemory())
303 if (Call2B.onlyAccessesArgPointees()) {
304 if (!Call2B.doesAccessArgPointees())
311 unsigned Call2ArgIdx = std::distance(Call2->
arg_begin(),
I);
331 R = (R | ArgMask) & Result;
341 if (Call1B.onlyAccessesArgPointees()) {
342 if (!Call1B.doesAccessArgPointees())
349 unsigned Call1ArgIdx = std::distance(Call1->
arg_begin(),
I);
360 R = (R | ArgModRefC1) & Result;
376 for (
const auto &AA : AAs) {
377 Result &= AA->getMemoryEffects(Call, AAQI);
380 if (Result.doesNotAccessMemory())
395 for (
const auto &AA : AAs) {
396 Result &= AA->getMemoryEffects(
F);
399 if (Result.doesNotAccessMemory())
418 OS <<
"PartialAlias";
451 OS <<
"InaccessibleMem: ";
603 const std::optional<MemoryLocation> &OptLoc,
605 if (OptLoc == std::nullopt) {
606 if (
const auto *Call = dyn_cast<CallBase>(
I))
612 switch (
I->getOpcode()) {
613 case Instruction::VAArg:
615 case Instruction::Load:
617 case Instruction::Store:
619 case Instruction::Fence:
621 case Instruction::AtomicCmpXchg:
623 case Instruction::AtomicRMW:
625 case Instruction::Call:
626 case Instruction::CallBr:
627 case Instruction::Invoke:
629 case Instruction::CatchPad:
631 case Instruction::CatchRet:
634 assert(!
I->mayReadOrWriteMemory() &&
635 "Unhandled memory access instruction!");
658 const auto *Call = dyn_cast<CallBase>(
I);
659 if (!Call || Call == Object)
670 for (
auto CI = Call->data_operands_begin(), CE = Call->data_operands_end();
671 CI != CE; ++CI, ++ArgNo) {
675 if (!(*CI)->getType()->isPointerTy() ||
676 (!Call->doesNotCapture(ArgNo) && ArgNo < Call->arg_size() &&
677 !Call->isByValArgument(ArgNo)))
689 if (Call->doesNotAccessMemory(ArgNo))
691 if (Call->onlyReadsMemory(ArgNo)) {
718 "Instructions not in same basic block!");
761 "Function Alias Analysis Results",
false,
true)
787 new AAResults(getAnalysis<TargetLibraryInfoWrapperPass>().getTLI(
F)));
794 AAR->addAAResult(getAnalysis<BasicAAWrapperPass>().getResult());
797 if (
auto *WrapperPass = getAnalysisIfAvailable<ScopedNoAliasAAWrapperPass>())
798 AAR->addAAResult(WrapperPass->getResult());
799 if (
auto *WrapperPass = getAnalysisIfAvailable<TypeBasedAAWrapperPass>())
800 AAR->addAAResult(WrapperPass->getResult());
801 if (
auto *WrapperPass = getAnalysisIfAvailable<GlobalsAAWrapperPass>())
802 AAR->addAAResult(WrapperPass->getResult());
803 if (
auto *WrapperPass = getAnalysisIfAvailable<SCEVAAWrapperPass>())
804 AAR->addAAResult(WrapperPass->getResult());
808 if (
auto *WrapperPass = getAnalysisIfAvailable<ExternalAAWrapperPass>())
810 WrapperPass->CB(*
this,
F, *AAR);
834 for (
auto &Getter : ResultGetters)
840 if (
const auto *Call = dyn_cast<CallBase>(V))
841 return Call->hasRetAttr(Attribute::NoAlias);
846 if (
const Argument *
A = dyn_cast<Argument>(V))
847 return A->hasNoAliasAttr() ||
A->hasByValAttr();
852 if (isa<AllocaInst>(V))
854 if (isa<GlobalValue>(V) && !isa<GlobalAlias>(V))
868 if (
auto *CB = dyn_cast<CallBase>(V))
875 if (isa<LoadInst>(V))
883 if (isa<IntToPtrInst>(V))
887 if (
auto *CE = dyn_cast<ConstantExpr>(V))
888 if (CE->getOpcode() == Instruction::IntToPtr)
895 bool &RequiresNoCaptureBeforeUnwind) {
896 RequiresNoCaptureBeforeUnwind =
false;
899 if (isa<AllocaInst>(Object))
903 if (
auto *
A = dyn_cast<Argument>(Object))
904 return A->hasByValAttr() ||
A->hasAttribute(Attribute::DeadOnUnwind);
910 RequiresNoCaptureBeforeUnwind =
true;
920 bool &ExplicitlyDereferenceableOnly) {
921 ExplicitlyDereferenceableOnly =
false;
925 if (isa<AllocaInst>(Object))
928 if (
auto *
A = dyn_cast<Argument>(Object)) {
929 if (
A->hasAttribute(Attribute::Writable)) {
930 ExplicitlyDereferenceableOnly =
true;
934 return A->hasByValAttr();
static cl::opt< bool > EnableAATrace("aa-trace", cl::Hidden, cl::init(false))
Print a trace of alias analysis queries and their results.
static bool isNoAliasOrByValArgument(const Value *V)
Function Alias Analysis Results
Atomic ordering constants.
This file contains the simple types necessary to represent the attributes associated with functions a...
This is the interface for LLVM's primary stateless and local alias analysis.
block Block Frequency Analysis
static GCRegistry::Add< ErlangGC > A("erlang", "erlang-compatible garbage collector")
static bool runOnFunction(Function &F, bool PostInlining)
This is the interface for a simple mod/ref and alias analysis over globals.
This file provides utility analysis objects describing memory locations.
This file declares a simple ARC-aware AliasAnalysis using special knowledge of Objective C to enhance...
#define INITIALIZE_PASS_DEPENDENCY(depName)
#define INITIALIZE_PASS_END(passName, arg, name, cfg, analysis)
#define INITIALIZE_PASS_BEGIN(passName, arg, name, cfg, analysis)
#define INITIALIZE_PASS(passName, arg, name, cfg, analysis)
assert(ImpDefSCC.getReg()==AMDGPU::SCC &&ImpDefSCC.isDef())
This is the interface for a SCEV-based alias analysis.
This is the interface for a metadata-based scoped no-alias analysis.
This file defines the 'Statistic' class, which is designed to be an easy way to expose various metric...
#define STATISTIC(VARNAME, DESC)
This is the interface for a metadata-based TBAA.
A manager for alias analyses.
Result run(Function &F, FunctionAnalysisManager &AM)
This class stores info we want to provide to or retain within an alias query.
unsigned Depth
Query depth used to distinguish recursive queries.
A wrapper pass to provide the legacy pass manager access to a suitably prepared AAResults object.
void getAnalysisUsage(AnalysisUsage &AU) const override
getAnalysisUsage - This function should be overriden by passes that need analysis information to do t...
ModRefInfo getModRefInfo(const Instruction *I, const std::optional< MemoryLocation > &OptLoc)
Check whether or not an instruction may read or write the optionally specified memory location.
AliasResult alias(const MemoryLocation &LocA, const MemoryLocation &LocB)
The main low level interface to the alias analysis implementation.
ModRefInfo getModRefInfoMask(const MemoryLocation &Loc, bool IgnoreLocals=false)
Returns a bitmask that should be unconditionally applied to the ModRef info of a memory location.
ModRefInfo callCapturesBefore(const Instruction *I, const MemoryLocation &MemLoc, DominatorTree *DT)
Return information about whether a particular call site modifies or reads the specified memory locati...
MemoryEffects getMemoryEffects(const CallBase *Call)
Return the behavior of the given call site.
bool invalidate(Function &F, const PreservedAnalyses &PA, FunctionAnalysisManager::Invalidator &Inv)
Handle invalidation events in the new pass manager.
AAResults(const TargetLibraryInfo &TLI)
ModRefInfo getArgModRefInfo(const CallBase *Call, unsigned ArgIdx)
Get the ModRef info associated with a pointer argument of a call.
bool canInstructionRangeModRef(const Instruction &I1, const Instruction &I2, const MemoryLocation &Loc, const ModRefInfo Mode)
Check if it is possible for the execution of the specified instructions to mod(according to the mode)...
bool canBasicBlockModify(const BasicBlock &BB, const MemoryLocation &Loc)
Check if it is possible for execution of the specified basic block to modify the location Loc.
The possible results of an alias query.
@ MayAlias
The two locations may or may not alias.
@ NoAlias
The two locations do not alias at all.
@ PartialAlias
The two locations alias, but only due to a partial overlap.
@ MustAlias
The two locations precisely alias each other.
constexpr int32_t getOffset() const
constexpr bool hasOffset() const
API to communicate dependencies between analyses during invalidation.
bool invalidate(IRUnitT &IR, const PreservedAnalyses &PA)
Trigger the invalidation of some other analysis pass if not already handled and return whether it was...
A container for analyses that lazily runs them and caches their results.
PassT::Result & getResult(IRUnitT &IR, ExtraArgTs... ExtraArgs)
Get the result of an analysis pass for a given IR unit.
Represent the analysis usage information of a pass.
AnalysisUsage & addUsedIfAvailable()
Add the specified Pass class to the set of analyses used by this pass.
void setPreservesAll()
Set by analyses that do not transform their input at all.
AnalysisUsage & addRequiredTransitive()
This class represents an incoming formal argument to a Function.
An instruction that atomically checks whether a specified value is in a memory location,...
AtomicOrdering getSuccessOrdering() const
Returns the success ordering constraint of this cmpxchg instruction.
an instruction that atomically reads a memory location, combines it with another value,...
AtomicOrdering getOrdering() const
Returns the ordering constraint of this rmw instruction.
Legacy wrapper pass to provide the BasicAAResult object.
LLVM Basic Block Representation.
InstListType::const_iterator const_iterator
const Instruction & front() const
const Instruction & back() const
Base class for all callable instructions (InvokeInst and CallInst) Holds everything related to callin...
User::op_iterator arg_begin()
Return the iterator pointing to the beginning of the argument list.
User::op_iterator arg_end()
Return the iterator pointing to the end of the argument list.
Concrete subclass of DominatorTreeBase that is used to compute a normal dominator tree.
An instruction for ordering other memory operations.
FunctionPass class - This class is used to implement most global optimizations.
Legacy wrapper pass to provide the GlobalsAAResult object.
ImmutablePass class - This class is used to provide information that does not need to be run.
An instruction for reading from memory.
MemoryEffectsBase getWithoutLoc(Location Loc) const
Get new MemoryEffectsBase with NoModRef on the given Loc.
ModRefInfo getModRef(Location Loc) const
Get ModRefInfo for the given Location.
static auto locations()
Returns iterator over all supported location kinds.
static MemoryEffectsBase unknown()
Create MemoryEffectsBase that can read and write any memory.
Representation for a specific memory location.
static MemoryLocation get(const LoadInst *LI)
Return a location with information about the memory reference by the given instruction.
LocationSize Size
The maximum size of the location, in address-units, or UnknownSize if the size is not known.
static MemoryLocation getBeforeOrAfter(const Value *Ptr, const AAMDNodes &AATags=AAMDNodes())
Return a location that may access any location before or after Ptr, while remaining within the underl...
const Value * Ptr
The address of the start of the location.
static MemoryLocation getForArgument(const CallBase *Call, unsigned ArgIdx, const TargetLibraryInfo *TLI)
Return a location representing a particular argument of a call.
static PassRegistry * getPassRegistry()
getPassRegistry - Access the global registry object, which is automatically initialized at applicatio...
A set of analyses that are preserved following a run of a transformation pass.
PreservedAnalysisChecker getChecker() const
Build a checker for this PreservedAnalyses and the specified analysis type.
Legacy wrapper pass to provide the SCEVAAResult object.
Legacy wrapper pass to provide the ScopedNoAliasAAResult object.
AAQueryInfo that uses SimpleCaptureInfo.
An instruction for storing to memory.
AtomicOrdering getOrdering() const
Returns the ordering constraint of this store instruction.
Analysis pass providing the TargetLibraryInfo.
Legacy wrapper pass to provide the TypeBasedAAResult object.
bool isPointerTy() const
True if this is an instance of PointerType.
This class represents the va_arg llvm instruction, which returns an argument of the specified type gi...
LLVM Value Representation.
Type * getType() const
All values are typed, get the type of this value.
const ParentTy * getParent() const
self_iterator getIterator()
This class implements an extremely fast bulk output stream that can only output to a stream.
initializer< Ty > init(const Ty &Val)
This is an optimization pass for GlobalISel generic memory operations.
cl::opt< bool > DisableBasicAA("disable-basic-aa", cl::Hidden, cl::init(false))
Allow disabling BasicAA from the AA results.
auto enumerate(FirstRange &&First, RestRanges &&...Rest)
Given two or more input ranges, returns a new range whose values are tuples (A, B,...
bool isStrongerThanMonotonic(AtomicOrdering AO)
bool PointerMayBeCapturedBefore(const Value *V, bool ReturnCaptures, bool StoreCaptures, const Instruction *I, const DominatorTree *DT, bool IncludeI=false, unsigned MaxUsesToExplore=0, const LoopInfo *LI=nullptr)
PointerMayBeCapturedBefore - Return true if this pointer value may be captured by the enclosing funct...
const Value * getUnderlyingObject(const Value *V, unsigned MaxLookup=6)
This method strips off any GEP address adjustments, pointer casts or llvm.threadlocal....
bool isNoAliasCall(const Value *V)
Return true if this pointer is returned by a noalias function.
void initializeAAResultsWrapperPassPass(PassRegistry &)
bool isIntrinsicReturningPointerAliasingArgumentWithoutCapturing(const CallBase *Call, bool MustPreserveNullness)
{launder,strip}.invariant.group returns pointer that aliases its argument, and it only captures point...
bool isModSet(const ModRefInfo MRI)
raw_ostream & dbgs()
dbgs() - This returns a reference to a raw_ostream for debugging messages.
bool isModOrRefSet(const ModRefInfo MRI)
bool isNotVisibleOnUnwind(const Value *Object, bool &RequiresNoCaptureBeforeUnwind)
Return true if Object memory is not visible after an unwind, in the sense that program semantics cann...
ModRefInfo
Flags indicating whether a memory access modifies or references memory.
@ Ref
The access may reference the value stored in memory.
@ ModRef
The access may reference and may modify the value stored in memory.
@ Mod
The access may modify the value stored in memory.
@ NoModRef
The access neither references nor modifies the value stored in memory.
IRMemLocation
The locations at which a function might access memory.
@ ArgMem
Access to memory via argument pointers.
@ InaccessibleMem
Memory that is inaccessible via LLVM IR.
raw_ostream & operator<<(raw_ostream &OS, const APFixedPoint &FX)
bool isIdentifiedFunctionLocal(const Value *V)
Return true if V is umabigously identified at the function-level.
OutputIt move(R &&Range, OutputIt Out)
Provide wrappers to std::move which take ranges instead of having to pass begin/end explicitly.
bool isEscapeSource(const Value *V)
Returns true if the pointer is one which would have been considered an escape by isNonEscapingLocalOb...
ImmutablePass * createExternalAAWrapperPass(std::function< void(Pass &, Function &, AAResults &)> Callback)
A wrapper pass around a callback which can be used to populate the AAResults in the AAResultsWrapperP...
void initializeExternalAAWrapperPassPass(PassRegistry &)
bool isNoModRef(const ModRefInfo MRI)
bool isIdentifiedObject(const Value *V)
Return true if this pointer refers to a distinct and identifiable object.
bool isStrongerThan(AtomicOrdering AO, AtomicOrdering Other)
Returns true if ao is stronger than other as defined by the AtomicOrdering lattice,...
bool isRefSet(const ModRefInfo MRI)
bool isWritableObject(const Value *Object, bool &ExplicitlyDereferenceableOnly)
Return true if the Object is writable, in the sense that any location based on this pointer that can ...
Implement std::hash so that hash_code can be used in STL containers.
A special type used by analysis passes to provide an address that identifies that particular analysis...
A wrapper pass for external alias analyses.
std::function< void(Pass &, Function &, AAResults &)> CallbackT