34#define DEBUG_TYPE "globalsmodref-aa"
37 "Number of global vars without address taken");
38STATISTIC(NumNonAddrTakenFunctions,
"Number of functions without address taken");
39STATISTIC(NumNoMemFunctions,
"Number of functions that do not access memory");
40STATISTIC(NumReadMemFunctions,
"Number of functions that only read memory");
41STATISTIC(NumIndirectGlobalVars,
"Number of indirect global objects");
69 struct alignas(8) AlignedMap {
70 AlignedMap() =
default;
71 AlignedMap(
const AlignedMap &Arg) =
default;
72 GlobalInfoMapType Map;
76 struct AlignedMapPointerTraits {
77 static inline void *getAsVoidPointer(AlignedMap *
P) {
return P; }
78 static inline AlignedMap *getFromVoidPointer(
void *
P) {
79 return (AlignedMap *)
P;
81 static constexpr int NumLowBitsAvailable = 3;
82 static_assert(
alignof(AlignedMap) >= (1 << NumLowBitsAvailable),
83 "AlignedMap insufficiently aligned to have enough low bits.");
91 enum { MayReadAnyGlobal = 4 };
95 "ModRef and the MayReadAnyGlobal flag bits overlap.");
97 AlignedMapPointerTraits::NumLowBitsAvailable) == 0,
98 "Insufficient low bits to store our flag and ModRef info.");
103 delete Info.getPointer();
109 : Info(nullptr, Arg.Info.getInt()) {
110 if (
const auto *ArgPtr = Arg.Info.
getPointer())
111 Info.setPointer(
new AlignedMap(*ArgPtr));
114 : Info(Arg.Info.getPointer(), Arg.Info.getInt()) {
115 Arg.Info.setPointerAndInt(
nullptr, 0);
118 delete Info.getPointer();
119 Info.setPointerAndInt(
nullptr, RHS.Info.getInt());
120 if (
const auto *RHSPtr = RHS.Info.getPointer())
121 Info.setPointer(
new AlignedMap(*RHSPtr));
125 delete Info.getPointer();
126 Info.setPointerAndInt(RHS.Info.getPointer(), RHS.Info.getInt());
144 Info.setInt(Info.getInt() |
static_cast<int>(NewMRI));
159 if (AlignedMap *
P = Info.getPointer()) {
160 auto I =
P->Map.find(&GV);
161 if (
I !=
P->Map.end())
162 GlobalMRI |=
I->second;
176 for (
const auto &
G :
P->Map)
181 AlignedMap *
P = Info.getPointer();
183 P =
new AlignedMap();
186 auto &GlobalMRI =
P->Map[&GV];
193 if (AlignedMap *
P = Info.getPointer())
206void GlobalsAAResult::DeletionCallbackHandle::deleted() {
209 GAR->FunctionInfos.erase(
F);
212 if (GAR->NonAddressTakenGlobals.erase(GV)) {
215 if (GAR->IndirectGlobals.erase(GV)) {
217 for (
auto I = GAR->AllocsForIndirectGlobals.begin(),
218 E = GAR->AllocsForIndirectGlobals.end();
221 GAR->AllocsForIndirectGlobals.erase(
I);
226 for (
auto &FIPair : GAR->FunctionInfos)
227 FIPair.second.eraseModRefInfoForGlobal(*GV);
232 GAR->AllocsForIndirectGlobals.erase(V);
236 GAR->Handles.erase(I);
250GlobalsAAResult::getFunctionInfo(
const Function *
F) {
251 auto I = FunctionInfos.find(
F);
252 if (
I != FunctionInfos.end())
261void GlobalsAAResult::AnalyzeGlobals(
Module &M) {
264 if (
F.hasLocalLinkage()) {
265 if (!AnalyzeUsesOfPointer(&
F)) {
267 NonAddressTakenGlobals.insert(&
F);
269 Handles.emplace_front(*
this, &
F);
270 Handles.front().I = Handles.begin();
271 ++NumNonAddrTakenFunctions;
273 UnknownFunctionsWithLocalLinkage =
true;
276 SmallPtrSet<Function *, 16> Readers, Writers;
277 for (GlobalVariable &GV :
M.globals())
278 if (GV.hasLocalLinkage()) {
279 if (!AnalyzeUsesOfPointer(&GV, &Readers,
280 GV.isConstant() ?
nullptr : &Writers)) {
282 NonAddressTakenGlobals.insert(&GV);
283 Handles.emplace_front(*
this, &GV);
284 Handles.front().I = Handles.begin();
286 for (Function *Reader : Readers) {
287 if (TrackedFunctions.
insert(Reader).second) {
288 Handles.emplace_front(*
this, Reader);
289 Handles.front().I = Handles.begin();
294 if (!GV.isConstant())
295 for (Function *Writer : Writers) {
296 if (TrackedFunctions.
insert(Writer).second) {
297 Handles.emplace_front(*
this, Writer);
298 Handles.front().I = Handles.begin();
302 ++NumNonAddrTakenGlobalVars;
305 if (GV.getValueType()->isPointerTy() &&
306 AnalyzeIndirectGlobalMemory(&GV))
307 ++NumIndirectGlobalVars;
320bool GlobalsAAResult::AnalyzeUsesOfPointer(
Value *V,
321 SmallPtrSetImpl<Function *> *Readers,
322 SmallPtrSetImpl<Function *> *Writers,
323 GlobalValue *OkayStoreDest) {
324 if (!
V->getType()->isPointerTy())
327 for (Use &U :
V->uses()) {
331 Readers->
insert(LI->getParent()->getParent());
333 if (V ==
SI->getOperand(1)) {
335 Writers->
insert(
SI->getParent()->getParent());
336 }
else if (
SI->getOperand(1) != OkayStoreDest) {
340 if (AnalyzeUsesOfPointer(
I, Readers, Writers))
344 if (AnalyzeUsesOfPointer(
I, Readers, Writers, OkayStoreDest))
348 if (
II->getIntrinsicID() == Intrinsic::threadlocal_address &&
349 V ==
II->getArgOperand(0)) {
350 if (AnalyzeUsesOfPointer(
II, Readers, Writers))
372 if (!
F || !
F->isDeclaration())
413bool GlobalsAAResult::AnalyzeIndirectGlobalMemory(GlobalVariable *GV) {
416 std::vector<Value *> AllocRelatedValues;
420 if (!
C->isNullValue())
425 for (User *U : GV->
users()) {
430 if (AnalyzeUsesOfPointer(LI))
435 if (
SI->getOperand(0) == GV)
450 if (AnalyzeUsesOfPointer(
Ptr,
nullptr,
nullptr,
455 AllocRelatedValues.push_back(
Ptr);
464 while (!AllocRelatedValues.empty()) {
465 AllocsForIndirectGlobals[AllocRelatedValues.back()] = GV;
466 Handles.emplace_front(*
this, AllocRelatedValues.back());
467 Handles.front().I = Handles.begin();
468 AllocRelatedValues.pop_back();
470 IndirectGlobals.
insert(GV);
471 Handles.emplace_front(*
this, GV);
472 Handles.front().I = Handles.begin();
476void GlobalsAAResult::CollectSCCMembership(CallGraph &CG) {
480 for (scc_iterator<CallGraph *>
I =
scc_begin(&CG); !
I.isAtEnd(); ++
I) {
481 const std::vector<CallGraphNode *> &
SCC = *
I;
482 assert(!
SCC.empty() &&
"SCC with no functions?");
484 for (
auto *CGN : SCC)
485 if (Function *
F = CGN->getFunction())
486 FunctionToSCCMap[
F] = SCCID;
495void GlobalsAAResult::AnalyzeCallGraph(CallGraph &CG,
Module &M) {
498 for (scc_iterator<CallGraph *>
I =
scc_begin(&CG); !
I.isAtEnd(); ++
I) {
499 const std::vector<CallGraphNode *> &
SCC = *
I;
500 assert(!
SCC.empty() &&
"SCC with no functions?");
504 if (!
F || !
F->isDefinitionExact()) {
508 for (
auto *Node : SCC)
509 FunctionInfos.erase(
Node->getFunction());
514 Handles.emplace_front(*
this,
F);
515 Handles.front().I = Handles.begin();
516 bool KnowNothing =
false;
525 auto MaySyncOrCallIntoModule = [](
const Function &
F) {
526 return !
F.isDeclaration() || !
F.hasNoSync() ||
527 !
F.hasFnAttribute(Attribute::NoCallback);
532 for (
unsigned i = 0, e =
SCC.size(); i != e && !KnowNothing; ++i) {
538 if (
F->isDeclaration() ||
F->hasOptNone()) {
540 if (
F->doesNotAccessMemory()) {
542 }
else if (
F->onlyReadsMemory()) {
544 if (!
F->onlyAccessesArgMemory() && MaySyncOrCallIntoModule(*
F))
547 FI.setMayReadAnyGlobal();
550 if (!
F->onlyAccessesArgMemory())
551 FI.setMayReadAnyGlobal();
552 if (MaySyncOrCallIntoModule(*
F)) {
561 CI !=
E && !KnowNothing; ++CI)
562 if (Function *Callee = CI->second->getFunction()) {
565 FI.addFunctionInfo(*CalleeFI);
569 CallGraphNode *CalleeNode = CG[
Callee];
581 for (
auto *Node : SCC)
582 FunctionInfos.erase(
Node->getFunction());
587 for (
auto *Node : SCC) {
594 if (
Node->getFunction()->hasOptNone())
608 if (
I.mayReadFromMemory())
610 if (
I.mayWriteToMemory())
616 ++NumReadMemFunctions;
625 for (
unsigned i = 1, e =
SCC.size(); i != e; ++i)
626 FunctionInfos[
SCC[i]->getFunction()] = CachedFI;
675 for (
const Value *
Op : PN->incoming_values()) {
684 }
while (!Inputs.
empty());
715bool GlobalsAAResult::isNonEscapingGlobalNoAlias(
const GlobalValue *GV,
717 const Instruction *CtxI) {
727 if (!
V->getType()->isPointerTy())
730 SmallPtrSet<const Value *, 8> Visited;
731 SmallVector<const Value *, 8> Inputs;
748 if (GVar && InputGVar &&
749 !GVar->isDeclaration() && !InputGVar->isDeclaration() &&
750 !GVar->isInterposable() && !InputGVar->isInterposable()) {
751 Type *GVType = GVar->getInitializer()->getType();
752 Type *InputGVType = InputGVar->getInitializer()->getType();
754 (DL.getTypeAllocSize(GVType) > 0) &&
755 (DL.getTypeAllocSize(InputGVType) > 0))
807 for (
const Value *
Op : PN->incoming_values()) {
819 }
while (!Inputs.
empty());
826 ModuleAnalysisManager::Invalidator &) {
830 return !PAC.preservedWhenStateless();
852 if (GV1 && !NonAddressTakenGlobals.count(GV1))
854 if (GV2 && !NonAddressTakenGlobals.count(GV2))
859 if (GV1 && GV2 && GV1 != GV2)
866 if ((GV1 || GV2) && GV1 != GV2)
871 if ((GV1 || GV2) && GV1 != GV2) {
873 const Value *UV = GV1 ? UV2 : UV1;
874 if (isNonEscapingGlobalNoAlias(GV, UV, CtxI))
888 if (IndirectGlobals.count(GV))
892 if (IndirectGlobals.count(GV))
898 GV1 = AllocsForIndirectGlobals.lookup(UV1);
900 GV2 = AllocsForIndirectGlobals.lookup(UV2);
905 if (GV1 && GV2 && GV1 != GV2)
912 if ((GV1 || GV2) && GV1 != GV2)
921 if (
Call->doesNotAccessMemory())
928 for (
const auto &
A :
Call->args()) {
940 return ConservativeResult;
943 return ConservativeResult;
963 if (NonAddressTakenGlobals.count(GV))
965 Known = FI->getModRefInfoForGlobal(*GV) |
966 getModRefInfoForArgument(
Call, GV, AAQI);
971GlobalsAAResult::GlobalsAAResult(
976GlobalsAAResult::GlobalsAAResult(GlobalsAAResult &&Arg)
978 NonAddressTakenGlobals(
std::
move(Arg.NonAddressTakenGlobals)),
979 IndirectGlobals(
std::
move(Arg.IndirectGlobals)),
980 AllocsForIndirectGlobals(
std::
move(Arg.AllocsForIndirectGlobals)),
981 FunctionInfos(
std::
move(Arg.FunctionInfos)),
982 Handles(
std::
move(Arg.Handles)) {
984 for (
auto &
H : Handles) {
995 GlobalsAAResult Result(M.getDataLayout(), GetTLI);
998 Result.CollectSCCMembership(CG);
1001 Result.AnalyzeGlobals(M);
1004 Result.AnalyzeCallGraph(CG, M);
1025 G->NonAddressTakenGlobals.clear();
1026 G->UnknownFunctionsWithLocalLinkage =
false;
1027 G->IndirectGlobals.clear();
1028 G->AllocsForIndirectGlobals.clear();
1029 G->FunctionInfos.clear();
1030 G->FunctionToSCCMap.clear();
1032 G->CollectSCCMembership(CG);
1033 G->AnalyzeGlobals(M);
1034 G->AnalyzeCallGraph(CG, M);
1041 "Globals Alias Analysis",
false,
true)
assert(UImm &&(UImm !=~static_cast< T >(0)) &&"Invalid immediate!")
MachineBasicBlock MachineBasicBlock::iterator DebugLoc DL
Expand Atomic instructions
static GCRegistry::Add< ErlangGC > A("erlang", "erlang-compatible garbage collector")
static GCRegistry::Add< CoreCLRGC > E("coreclr", "CoreCLR-compatible GC")
This file provides interfaces used to build and manipulate a call graph, which is a very useful tool ...
static cl::opt< bool > EnableUnsafeGlobalsModRefAliasResults("enable-unsafe-globalsmodref-alias-results", cl::init(false), cl::Hidden)
static bool isNonEscapingGlobalNoAliasWithLoad(const GlobalValue *GV, const Value *V, int &Depth, const DataLayout &DL)
This is the interface for a simple mod/ref and alias analysis over globals.
Module.h This file contains the declarations for the Module class.
This header defines various interfaces for pass management in LLVM.
Machine Check Debug Module
uint64_t IntrinsicInst * II
FunctionAnalysisManager FAM
#define INITIALIZE_PASS_DEPENDENCY(depName)
#define INITIALIZE_PASS_END(passName, arg, name, cfg, analysis)
#define INITIALIZE_PASS_BEGIN(passName, arg, name, cfg, analysis)
This builds on the llvm/ADT/GraphTraits.h file to find the strongly connected components (SCCs) of a ...
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)
The mod/ref information collected for a particular function.
FunctionInfo & operator=(FunctionInfo &&RHS)
void eraseModRefInfoForGlobal(const GlobalValue &GV)
Clear a global's ModRef info.
void setMayReadAnyGlobal()
Sets this function as potentially reading from any global.
void addModRefInfo(ModRefInfo NewMRI)
Adds new ModRefInfo for this function to its state.
void addFunctionInfo(const FunctionInfo &FI)
Add mod/ref info from another function into ours, saturating towards ModRef.
ModRefInfo getModRefInfo() const
Returns the ModRefInfo info for this function.
FunctionInfo()=default
Checks to document the invariants of the bit packing here.
FunctionInfo & operator=(const FunctionInfo &RHS)
void addModRefInfoForGlobal(const GlobalValue &GV, ModRefInfo NewMRI)
ModRefInfo globalClearMayReadAnyGlobal(int I) const
This method clears MayReadAnyGlobal bit added by GlobalsAAResult to return the corresponding ModRefIn...
bool mayReadAnyGlobal() const
Returns whether this function may read any global variable, and we don't know which global.
FunctionInfo(FunctionInfo &&Arg)
FunctionInfo(const FunctionInfo &Arg)
ModRefInfo getModRefInfoForGlobal(const GlobalValue &GV) const
Returns the ModRefInfo info for this function w.r.t.
This class stores info we want to provide to or retain within an alias query.
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.
PassT::Result * getCachedResult(IRUnitT &IR) const
Get the cached result of an analysis pass for a given IR unit.
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 & addRequired()
void setPreservesAll()
Set by analyses that do not transform their input at all.
Base class for all callable instructions (InvokeInst and CallInst) Holds everything related to callin...
bool doesNotCapture(unsigned OpNo) const
Determine whether this data operand is not captured.
Function * getCalledFunction() const
Returns the function called, or null if this is an indirect function invocation or the function signa...
bool hasFnAttr(Attribute::AttrKind Kind) const
Determine whether this call has the given attribute.
unsigned getArgOperandNo(const Use *U) const
Given a use for a arg operand, get the arg operand number that corresponds to it.
bool isArgOperand(const Use *U) const
bool isDataOperand(const Use *U) const
An analysis pass to compute the CallGraph for a Module.
std::vector< CallRecord >::iterator iterator
The ModulePass which wraps up a CallGraph and the logic to build it.
The basic data container for the call graph of a Module of IR.
A parsed version of the target data layout string in and methods for querying it.
bool hasLocalLinkage() const
const Constant * getInitializer() const
getInitializer - Return the initializer for this global variable.
An alias analysis result set for globals.
LLVM_ABI ~GlobalsAAResult()
LLVM_ABI ModRefInfo getModRefInfo(const CallBase *Call, const MemoryLocation &Loc, AAQueryInfo &AAQI)
LLVM_ABI bool invalidate(Module &M, const PreservedAnalyses &PA, ModuleAnalysisManager::Invalidator &)
static LLVM_ABI GlobalsAAResult analyzeModule(Module &M, std::function< const TargetLibraryInfo &(Function &F)> GetTLI, CallGraph &CG)
LLVM_ABI MemoryEffects getMemoryEffects(const Function *F)
getMemoryEffects - Return the behavior of the specified function if called from the specified call si...
LLVM_ABI AliasResult alias(const MemoryLocation &LocA, const MemoryLocation &LocB, AAQueryInfo &AAQI, const Instruction *CtxI)
alias - If one of the pointers is to a global that we are tracking, and the other is some random poin...
Legacy wrapper pass to provide the GlobalsAAResult object.
void getAnalysisUsage(AnalysisUsage &AU) const override
getAnalysisUsage - This function should be overriden by passes that need analysis information to do t...
bool runOnModule(Module &M) override
runOnModule - Virtual method overriden by subclasses to process the module being operated on.
bool doFinalization(Module &M) override
doFinalization - Virtual method overriden by subclasses to do any necessary clean up after all passes...
Analysis pass providing a never-invalidated alias analysis result.
LLVM_ABI GlobalsAAResult run(Module &M, ModuleAnalysisManager &AM)
LLVM_ABI const Function * getFunction() const
Return the function this instruction belongs to.
An instruction for reading from memory.
static MemoryEffectsBase unknown()
Representation for a specific memory location.
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.
ModulePass class - This class is used to implement unstructured interprocedural optimizations and ana...
A Module instance is used to store all the information related to an LLVM module.
unsigned getOpcode() const
Return the opcode for this Instruction or ConstantExpr.
AnalysisType & getAnalysis() const
getAnalysis<AnalysisType>() - This function is used by subclasses to get to the analysis information ...
PointerIntPair - This class implements a pair of a pointer and small integer.
void setPointerAndInt(PointerTy PtrVal, IntType IntVal) &
PointerTy getPointer() const
A set of analyses that are preserved following a run of a transformation pass.
static PreservedAnalyses all()
Construct a special preserved set that preserves all passes.
PreservedAnalysisChecker getChecker() const
Build a checker for this PreservedAnalyses and the specified analysis type.
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.
void push_back(const T &Elt)
This is a 'vector' (really, a variable-sized array), optimized for the case when the array is small.
Analysis pass providing the TargetLibraryInfo.
Provides information about what library functions are available for the current target.
bool isSized(SmallPtrSetImpl< Type * > *Visited=nullptr) const
Return true if it makes sense to take the size of this type.
Value * getValPtr() const
LLVM Value Representation.
iterator_range< user_iterator > users()
LLVM_ABI const Value * stripPointerCastsForAliasAnalysis() const
Strip off pointer casts, all-zero GEPs, single-argument phi nodes and invariant group info.
const ParentTy * getParent() const
@ C
The default llvm calling convention, compatible with C.
initializer< Ty > init(const Ty &Val)
@ User
could "use" a pointer
NodeAddr< NodeBase * > Node
LLVM_ABI iterator begin() const
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.
decltype(auto) dyn_cast(const From &Val)
dyn_cast<X> - Return the argument parameter cast to the specified type.
scc_iterator< T > scc_begin(const T &G)
Construct the begin iterator for a deduced graph type T.
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.
LLVM_ABI ModulePass * createGlobalsAAWrapperPass()
bool isModSet(const ModRefInfo MRI)
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 ...
bool isModOrRefSet(const ModRefInfo MRI)
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...
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.
DWARFExpression::Operation Op
LLVM_ABI Value * getFreedOperand(const CallBase *CB, const TargetLibraryInfo *TLI)
If this if a call to a free function, return the freed operand.
bool isModAndRefSet(const ModRefInfo MRI)
OutputIt move(R &&Range, OutputIt Out)
Provide wrappers to std::move which take ranges instead of having to pass begin/end explicitly.
bool is_contained(R &&Range, const E &Element)
Returns true if Element is found in Range.
AnalysisManager< Function > FunctionAnalysisManager
Convenience typedef for the Function analysis manager.
LLVM_ABI const Value * getUnderlyingObject(const Value *V, unsigned MaxLookup=MaxLookupSearchDepth)
This method strips off any GEP address adjustments, pointer casts or llvm.threadlocal....
LLVM_ABI void getUnderlyingObjects(const Value *V, SmallVectorImpl< const Value * > &Objects, const LoopInfo *LI=nullptr, unsigned MaxLookup=MaxLookupSearchDepth)
This method is similar to getUnderlyingObject except that it can look through phi and select instruct...
LLVM_ABI bool isIdentifiedObject(const Value *V)
Return true if this pointer refers to a distinct and identifiable object.
AnalysisManager< Module > ModuleAnalysisManager
Convenience typedef for the Module analysis manager.
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...
LLVM_ABI PreservedAnalyses run(Module &M, ModuleAnalysisManager &AM)