LLVM 22.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,
56 AAQueryInfo &AAQI,
57 const Instruction *) {
60
61 // Get the attached MDNodes.
62 const MDNode *AScopes = LocA.AATags.Scope, *BScopes = LocB.AATags.Scope;
63
64 const MDNode *ANoAlias = LocA.AATags.NoAlias, *BNoAlias = LocB.AATags.NoAlias;
65
66 if (!mayAliasInScopes(AScopes, BNoAlias))
68
69 if (!mayAliasInScopes(BScopes, ANoAlias))
71
73}
74
76 const MemoryLocation &Loc,
77 AAQueryInfo &AAQI) {
79 return ModRefInfo::ModRef;
80
81 if (!mayAliasInScopes(Loc.AATags.Scope,
82 Call->getMetadata(LLVMContext::MD_noalias)))
84
85 if (!mayAliasInScopes(Call->getMetadata(LLVMContext::MD_alias_scope),
86 Loc.AATags.NoAlias))
88
89 return ModRefInfo::ModRef;
90}
91
93 const CallBase *Call2,
94 AAQueryInfo &AAQI) {
96 return ModRefInfo::ModRef;
97
98 if (!mayAliasInScopes(Call1->getMetadata(LLVMContext::MD_alias_scope),
99 Call2->getMetadata(LLVMContext::MD_noalias)))
101
102 if (!mayAliasInScopes(Call2->getMetadata(LLVMContext::MD_alias_scope),
103 Call1->getMetadata(LLVMContext::MD_noalias)))
105
106 return ModRefInfo::ModRef;
107}
108
109static void collectMDInDomain(const MDNode *List, const MDNode *Domain,
111 for (const MDOperand &MDOp : List->operands())
112 if (const MDNode *MD = dyn_cast<MDNode>(MDOp))
113 if (AliasScopeNode(MD).getDomain() == Domain)
114 Nodes.insert(MD);
115}
116
117/// Collect the set of scoped domains relevant to the noalias scopes.
119 const MDNode *NoAlias, SmallPtrSetImpl<const MDNode *> &Domains) const {
120 if (!NoAlias)
121 return;
122 assert(Domains.empty() && "Domains should be empty");
123 for (const MDOperand &MDOp : NoAlias->operands())
124 if (const MDNode *NAMD = dyn_cast<MDNode>(MDOp))
125 if (const MDNode *Domain = AliasScopeNode(NAMD).getDomain())
126 Domains.insert(Domain);
127}
128
129bool ScopedNoAliasAAResult::mayAliasInScopes(const MDNode *Scopes,
130 const MDNode *NoAlias) const {
131 if (!Scopes || !NoAlias)
132 return true;
133
134 // Collect the set of scope domains relevant to the noalias scopes.
136 collectScopedDomains(NoAlias, Domains);
137
138 // We alias unless, for some domain, the set of noalias scopes in that domain
139 // is a superset of the set of alias scopes in that domain.
140 for (const MDNode *Domain : Domains) {
142 collectMDInDomain(Scopes, Domain, ScopeNodes);
143 if (ScopeNodes.empty())
144 continue;
145
147 collectMDInDomain(NoAlias, Domain, NANodes);
148
149 // To not alias, all of the nodes in ScopeNodes must be in NANodes.
150 if (llvm::set_is_subset(ScopeNodes, NANodes))
151 return false;
152 }
153
154 return true;
155}
156
157AnalysisKey ScopedNoAliasAA::Key;
158
163
165
167 "Scoped NoAlias Alias Analysis", false, true)
168
170 return new ScopedNoAliasAAWrapperPass();
171}
172
174
176 Result.reset(new ScopedNoAliasAAResult());
177 return false;
178}
179
181 Result.reset();
182 return false;
183}
184
assert(UImm &&(UImm !=~static_cast< T >(0)) &&"Invalid immediate!")
static Domain getDomain(const ConstantRange &CR)
#define F(x, y, z)
Definition MD5.cpp:55
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:1589
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:1077
ArrayRef< MDOperand > operands() const
Definition Metadata.h:1443
Tracking metadata reference owned by Metadata.
Definition Metadata.h:899
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.
LLVM_ABI void collectScopedDomains(const MDNode *NoAlias, SmallPtrSetImpl< const MDNode * > &Domains) const
Collect the set of scoped domains relevant to the noalias scopes.
LLVM_ABI ModRefInfo getModRefInfo(const CallBase *Call, const MemoryLocation &Loc, AAQueryInfo &AAQI)
LLVM_ABI AliasResult alias(const MemoryLocation &LocA, const MemoryLocation &LocB, AAQueryInfo &AAQI, const Instruction *CtxI)
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.
decltype(auto) dyn_cast(const From &Val)
dyn_cast<X> - Return the argument parameter cast to the specified type.
Definition Casting.h:649
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:783
MDNode * NoAlias
The tag specifying the noalias scope.
Definition Metadata.h:786
A special type used by analysis passes to provide an address that identifies that particular analysis...
Definition Analysis.h:29