1 //===- AssumeBundleQueries.cpp - tool to query assume bundles ---*- C++ -*-===//
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
7 //===----------------------------------------------------------------------===//
9 #include "llvm/Analysis/AssumeBundleQueries.h"
10 #include "llvm/ADT/Statistic.h"
11 #include "llvm/Analysis/AssumptionCache.h"
12 #include "llvm/Analysis/ValueTracking.h"
13 #include "llvm/IR/Instruction.h"
14 #include "llvm/IR/Instructions.h"
15 #include "llvm/IR/IntrinsicInst.h"
16 #include "llvm/IR/PatternMatch.h"
17 #include "llvm/Support/DebugCounter.h"
19 #define DEBUG_TYPE "assume-queries"
22 using namespace llvm::PatternMatch
;
24 STATISTIC(NumAssumeQueries
, "Number of Queries into an assume assume bundles");
26 NumUsefullAssumeQueries
,
27 "Number of Queries into an assume assume bundles that were satisfied");
29 DEBUG_COUNTER(AssumeQueryCounter
, "assume-queries-counter",
30 "Controls which assumes gets created");
32 static bool bundleHasArgument(const CallBase::BundleOpInfo
&BOI
, unsigned Idx
) {
33 return BOI
.End
- BOI
.Begin
> Idx
;
36 static Value
*getValueFromBundleOpInfo(AssumeInst
&Assume
,
37 const CallBase::BundleOpInfo
&BOI
,
39 assert(bundleHasArgument(BOI
, Idx
) && "index out of range");
40 return (Assume
.op_begin() + BOI
.Begin
+ Idx
)->get();
43 bool llvm::hasAttributeInAssume(AssumeInst
&Assume
, Value
*IsOn
,
44 StringRef AttrName
, uint64_t *ArgVal
) {
45 assert(Attribute::isExistingAttribute(AttrName
) &&
46 "this attribute doesn't exist");
47 assert((ArgVal
== nullptr || Attribute::isIntAttrKind(
48 Attribute::getAttrKindFromName(AttrName
))) &&
49 "requested value for an attribute that has no argument");
50 if (Assume
.bundle_op_infos().empty())
53 for (auto &BOI
: Assume
.bundle_op_infos()) {
54 if (BOI
.Tag
->getKey() != AttrName
)
56 if (IsOn
&& (BOI
.End
- BOI
.Begin
<= ABA_WasOn
||
57 IsOn
!= getValueFromBundleOpInfo(Assume
, BOI
, ABA_WasOn
)))
60 assert(BOI
.End
- BOI
.Begin
> ABA_Argument
);
62 cast
<ConstantInt
>(getValueFromBundleOpInfo(Assume
, BOI
, ABA_Argument
))
70 void llvm::fillMapFromAssume(AssumeInst
&Assume
, RetainedKnowledgeMap
&Result
) {
71 for (auto &Bundles
: Assume
.bundle_op_infos()) {
72 std::pair
<Value
*, Attribute::AttrKind
> Key
{
73 nullptr, Attribute::getAttrKindFromName(Bundles
.Tag
->getKey())};
74 if (bundleHasArgument(Bundles
, ABA_WasOn
))
75 Key
.first
= getValueFromBundleOpInfo(Assume
, Bundles
, ABA_WasOn
);
77 if (Key
.first
== nullptr && Key
.second
== Attribute::None
)
79 if (!bundleHasArgument(Bundles
, ABA_Argument
)) {
80 Result
[Key
][&Assume
] = {0, 0};
83 auto *CI
= dyn_cast
<ConstantInt
>(
84 getValueFromBundleOpInfo(Assume
, Bundles
, ABA_Argument
));
87 uint64_t Val
= CI
->getZExtValue();
88 auto Lookup
= Result
.find(Key
);
89 if (Lookup
== Result
.end() || !Lookup
->second
.count(&Assume
)) {
90 Result
[Key
][&Assume
] = {Val
, Val
};
93 Lookup
->second
[&Assume
].Min
= std::min(Val
, Lookup
->second
[&Assume
].Min
);
94 Lookup
->second
[&Assume
].Max
= std::max(Val
, Lookup
->second
[&Assume
].Max
);
99 llvm::getKnowledgeFromBundle(AssumeInst
&Assume
,
100 const CallBase::BundleOpInfo
&BOI
) {
101 RetainedKnowledge Result
;
102 Result
.AttrKind
= Attribute::getAttrKindFromName(BOI
.Tag
->getKey());
103 if (bundleHasArgument(BOI
, ABA_WasOn
))
104 Result
.WasOn
= getValueFromBundleOpInfo(Assume
, BOI
, ABA_WasOn
);
105 auto GetArgOr1
= [&](unsigned Idx
) -> uint64_t {
106 if (auto *ConstInt
= dyn_cast
<ConstantInt
>(
107 getValueFromBundleOpInfo(Assume
, BOI
, ABA_Argument
+ Idx
)))
108 return ConstInt
->getZExtValue();
111 if (BOI
.End
- BOI
.Begin
> ABA_Argument
)
112 Result
.ArgValue
= GetArgOr1(0);
113 if (Result
.AttrKind
== Attribute::Alignment
)
114 if (BOI
.End
- BOI
.Begin
> ABA_Argument
+ 1)
115 Result
.ArgValue
= MinAlign(Result
.ArgValue
, GetArgOr1(1));
119 RetainedKnowledge
llvm::getKnowledgeFromOperandInAssume(AssumeInst
&Assume
,
121 CallBase::BundleOpInfo BOI
= Assume
.getBundleOpInfoForOperand(Idx
);
122 return getKnowledgeFromBundle(Assume
, BOI
);
125 bool llvm::isAssumeWithEmptyBundle(const AssumeInst
&Assume
) {
126 return none_of(Assume
.bundle_op_infos(),
127 [](const CallBase::BundleOpInfo
&BOI
) {
128 return BOI
.Tag
->getKey() != IgnoreBundleTag
;
132 static CallInst::BundleOpInfo
*getBundleFromUse(const Use
*U
) {
133 if (!match(U
->getUser(),
134 m_Intrinsic
<Intrinsic::assume
>(m_Unless(m_Specific(U
->get())))))
136 auto *Intr
= cast
<IntrinsicInst
>(U
->getUser());
137 return &Intr
->getBundleOpInfoForOperand(U
->getOperandNo());
141 llvm::getKnowledgeFromUse(const Use
*U
,
142 ArrayRef
<Attribute::AttrKind
> AttrKinds
) {
143 CallInst::BundleOpInfo
* Bundle
= getBundleFromUse(U
);
145 return RetainedKnowledge::none();
146 RetainedKnowledge RK
=
147 getKnowledgeFromBundle(*cast
<AssumeInst
>(U
->getUser()), *Bundle
);
148 if (llvm::is_contained(AttrKinds
, RK
.AttrKind
))
150 return RetainedKnowledge::none();
154 llvm::getKnowledgeForValue(const Value
*V
,
155 ArrayRef
<Attribute::AttrKind
> AttrKinds
,
157 function_ref
<bool(RetainedKnowledge
, Instruction
*,
158 const CallBase::BundleOpInfo
*)>
161 if (!DebugCounter::shouldExecute(AssumeQueryCounter
))
162 return RetainedKnowledge::none();
164 for (AssumptionCache::ResultElem
&Elem
: AC
->assumptionsFor(V
)) {
165 auto *II
= cast_or_null
<AssumeInst
>(Elem
.Assume
);
166 if (!II
|| Elem
.Index
== AssumptionCache::ExprResultIdx
)
168 if (RetainedKnowledge RK
= getKnowledgeFromBundle(
169 *II
, II
->bundle_op_info_begin()[Elem
.Index
])) {
172 if (is_contained(AttrKinds
, RK
.AttrKind
) &&
173 Filter(RK
, II
, &II
->bundle_op_info_begin()[Elem
.Index
])) {
174 NumUsefullAssumeQueries
++;
179 return RetainedKnowledge::none();
181 for (const auto &U
: V
->uses()) {
182 CallInst::BundleOpInfo
* Bundle
= getBundleFromUse(&U
);
185 if (RetainedKnowledge RK
=
186 getKnowledgeFromBundle(*cast
<AssumeInst
>(U
.getUser()), *Bundle
))
187 if (is_contained(AttrKinds
, RK
.AttrKind
) &&
188 Filter(RK
, cast
<Instruction
>(U
.getUser()), Bundle
)) {
189 NumUsefullAssumeQueries
++;
193 return RetainedKnowledge::none();
196 RetainedKnowledge
llvm::getKnowledgeValidInContext(
197 const Value
*V
, ArrayRef
<Attribute::AttrKind
> AttrKinds
,
198 const Instruction
*CtxI
, const DominatorTree
*DT
, AssumptionCache
*AC
) {
199 return getKnowledgeForValue(V
, AttrKinds
, AC
,
200 [&](auto, Instruction
*I
, auto) {
201 return isValidAssumeForContext(I
, CtxI
, DT
);