LLVM 23.0.0git
ScopedNoAliasAA.cpp
Go to the documentation of this file.
1//===- ScopedNoAliasAA.cpp - Scoped No-Alias Alias Analysis ---------------===//
2//
3// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4// See https://llvm.org/LICENSE.txt for license information.
5// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6//
7//===----------------------------------------------------------------------===//
8//
9// This file defines the ScopedNoAlias alias-analysis pass, which implements
10// metadata-based scoped no-alias support.
11//
12// Alias-analysis scopes are defined by an id (which can be a string or some
13// other metadata node), a domain node, and an optional descriptive string.
14// A domain is defined by an id (which can be a string or some other metadata
15// node), and an optional descriptive string.
16//
17// !dom0 = metadata !{ metadata !"domain of foo()" }
18// !scope1 = metadata !{ metadata !scope1, metadata !dom0, metadata !"scope 1" }
19// !scope2 = metadata !{ metadata !scope2, metadata !dom0, metadata !"scope 2" }
20//
21// Loads and stores can be tagged with an alias-analysis scope, and also, with
22// a noalias tag for a specific scope:
23//
24// ... = load %ptr1, !alias.scope !{ !scope1 }
25// ... = load %ptr2, !alias.scope !{ !scope1, !scope2 }, !noalias !{ !scope1 }
26//
27// When evaluating an aliasing query, if one of the instructions is associated
28// has a set of noalias scopes in some domain that is a superset of the alias
29// scopes in that domain of some other instruction, then the two memory
30// accesses are assumed not to alias.
31//
32//===----------------------------------------------------------------------===//
33
38#include "llvm/IR/InstrTypes.h"
39#include "llvm/IR/LLVMContext.h"
40#include "llvm/IR/Metadata.h"
42#include "llvm/Pass.h"
45
46using namespace llvm;
47
48// A handy option for disabling scoped no-alias functionality. The same effect
49// can also be achieved by stripping the associated metadata tags from IR, but
50// this option is sometimes more convenient.
51static cl::opt<bool> EnableScopedNoAlias("enable-scoped-noalias",
52 cl::init(true), cl::Hidden);
53
55 const MemoryLocation &LocB) {
58
59 // Get the attached MDNodes.
60 const MDNode *AScopes = LocA.AATags.Scope, *BScopes = LocB.AATags.Scope;
61
62 const MDNode *ANoAlias = LocA.AATags.NoAlias, *BNoAlias = LocB.AATags.NoAlias;
63
64 if (!mayAliasInScopes(AScopes, BNoAlias))
66
67 if (!mayAliasInScopes(BScopes, ANoAlias))
69
71}
72
74 const MemoryLocation &LocB,
75 AAQueryInfo &, const Instruction *) {
76 return alias(LocA, LocB);
77}
78
80 const MemoryLocation &Loc,
81 AAQueryInfo &AAQI) {
83 return ModRefInfo::ModRef;
84
85 if (!mayAliasInScopes(Loc.AATags.Scope,
86 Call->getMetadata(LLVMContext::MD_noalias)))
88
89 if (!mayAliasInScopes(Call->getMetadata(LLVMContext::MD_alias_scope),
90 Loc.AATags.NoAlias))
92
93 return ModRefInfo::ModRef;
94}
95
97 const CallBase *Call2,
98 AAQueryInfo &AAQI) {
100 return ModRefInfo::ModRef;
101
102 if (!mayAliasInScopes(Call1->getMetadata(LLVMContext::MD_alias_scope),
103 Call2->getMetadata(LLVMContext::MD_noalias)))
105
106 if (!mayAliasInScopes(Call2->getMetadata(LLVMContext::MD_alias_scope),
107 Call1->getMetadata(LLVMContext::MD_noalias)))
109
110 return ModRefInfo::ModRef;
111}
112
113static void collectMDInDomain(const MDNode *List, const MDNode *Domain,
115 for (const MDOperand &MDOp : List->operands())
116 if (const MDNode *MD = dyn_cast<MDNode>(MDOp))
117 if (AliasScopeNode(MD).getDomain() == Domain)
118 Nodes.insert(MD);
119}
120
121/// Collect the set of scoped domains relevant to the noalias scopes.
123 const MDNode *NoAlias, SmallPtrSetImpl<const MDNode *> &Domains) {
124 if (!NoAlias)
125 return;
126 assert(Domains.empty() && "Domains should be empty");
127 for (const MDOperand &MDOp : NoAlias->operands())
128 if (const MDNode *NAMD = dyn_cast<MDNode>(MDOp))
129 if (const MDNode *Domain = AliasScopeNode(NAMD).getDomain())
130 Domains.insert(Domain);
131}
132
134 const MDNode *NoAlias) {
135 if (!Scopes || !NoAlias)
136 return true;
137
138 // Collect the set of scope domains relevant to the noalias scopes.
140 collectScopedDomains(NoAlias, Domains);
141
142 // We alias unless, for some domain, the set of noalias scopes in that domain
143 // is a superset of the set of alias scopes in that domain.
144 for (const MDNode *Domain : Domains) {
146 collectMDInDomain(Scopes, Domain, ScopeNodes);
147 if (ScopeNodes.empty())
148 continue;
149
151 collectMDInDomain(NoAlias, Domain, NANodes);
152
153 // To not alias, all of the nodes in ScopeNodes must be in NANodes.
154 if (llvm::set_is_subset(ScopeNodes, NANodes))
155 return false;
156 }
157
158 return true;
159}
160
161AnalysisKey ScopedNoAliasAA::Key;
162
167
169
171 "Scoped NoAlias Alias Analysis", false, true)
172
174 return new ScopedNoAliasAAWrapperPass();
175}
176
178
180 Result.reset(new ScopedNoAliasAAResult());
181 return false;
182}
183
185 Result.reset();
186 return false;
187}
188
assert(UImm &&(UImm !=~static_cast< T >(0)) &&"Invalid immediate!")
static Domain getDomain(const ConstantRange &CR)
#define F(x, y, z)
Definition MD5.cpp:54
This file provides utility analysis objects describing memory locations.
This file contains the declarations for metadata subclasses.
#define INITIALIZE_PASS(passName, arg, name, cfg, analysis)
Definition PassSupport.h:56
static void collectMDInDomain(const MDNode *List, const MDNode *Domain, SmallPtrSetImpl< const MDNode * > &Nodes)
static cl::opt< bool > EnableScopedNoAlias("enable-scoped-noalias", cl::init(true), cl::Hidden)
This is the interface for a metadata-based scoped no-alias analysis.
This file defines generic set operations that may be used on set's of different types,...
This file defines the SmallPtrSet class.
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.
This is a simple wrapper around an MDNode which provides a higher-level interface by hiding the detai...
Definition Metadata.h:1596
Represent the analysis usage information of a pass.
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...
ImmutablePass class - This class is used to provide information that does not need to be run.
Definition Pass.h:285
ImmutablePass(char &pid)
Definition Pass.h:287
MDNode * getMetadata(unsigned KindID) const
Get the metadata of given kind attached to this Instruction.
Metadata node.
Definition Metadata.h:1080
ArrayRef< MDOperand > operands() const
Definition Metadata.h:1442
Tracking metadata reference owned by Metadata.
Definition Metadata.h:902
Representation for a specific memory location.
AAMDNodes AATags
The metadata nodes which describes the aliasing of the location (each member is null if that kind of ...
A Module instance is used to store all the information related to an LLVM module.
Definition Module.h:67
A simple AA result which uses scoped-noalias metadata to answer queries.
static LLVM_ABI bool mayAliasInScopes(const MDNode *Scopes, const MDNode *NoAlias)
static LLVM_ABI AliasResult alias(const MemoryLocation &LocA, const MemoryLocation &LocB)
static LLVM_ABI void collectScopedDomains(const MDNode *NoAlias, SmallPtrSetImpl< const MDNode * > &Domains)
Collect the set of scoped domains relevant to the noalias scopes.
LLVM_ABI ModRefInfo getModRefInfo(const CallBase *Call, const MemoryLocation &Loc, AAQueryInfo &AAQI)
Legacy wrapper pass to provide the ScopedNoAliasAAResult object.
bool doInitialization(Module &M) override
doInitialization - Virtual method overridden by subclasses to do any necessary initialization before ...
bool doFinalization(Module &M) override
doFinalization - Virtual method overriden by subclasses to do any necessary clean up after all passes...
void getAnalysisUsage(AnalysisUsage &AU) const override
getAnalysisUsage - This function should be overriden by passes that need analysis information to do t...
LLVM_ABI ScopedNoAliasAAResult run(Function &F, FunctionAnalysisManager &AM)
A templated base class for SmallPtrSet which provides the typesafe interface that is common across al...
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.
CallInst * Call
initializer< Ty > init(const Ty &Val)
This is an optimization pass for GlobalISel generic memory operations.
Definition Types.h:26
decltype(auto) dyn_cast(const From &Val)
dyn_cast<X> - Return the argument parameter cast to the specified type.
Definition Casting.h:643
bool set_is_subset(const S1Ty &S1, const S2Ty &S2)
set_is_subset(A, B) - Return true iff A in B
LLVM_ABI ImmutablePass * createScopedNoAliasAAWrapperPass()
ModRefInfo
Flags indicating whether a memory access modifies or references memory.
Definition ModRef.h:28
@ ModRef
The access may reference and may modify the value stored in memory.
Definition ModRef.h:36
@ NoModRef
The access neither references nor modifies the value stored in memory.
Definition ModRef.h:30
AnalysisManager< Function > FunctionAnalysisManager
Convenience typedef for the Function analysis manager.
MDNode * Scope
The tag for alias scope specification (used with noalias).
Definition Metadata.h:786
MDNode * NoAlias
The tag specifying the noalias scope.
Definition Metadata.h:789
A special type used by analysis passes to provide an address that identifies that particular analysis...
Definition Analysis.h:29