LLVM 22.0.0git
ScalarEvolutionPatternMatch.h
Go to the documentation of this file.
1//===----------------------------------------------------------------------===//
2// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
3// See https://llvm.org/LICENSE.txt for license information.
4// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
5//
6//===----------------------------------------------------------------------===//
7//
8// This file provides a simple and efficient mechanism for performing general
9// tree-based pattern matches on SCEVs, based on LLVM's IR pattern matchers.
10//
11//===----------------------------------------------------------------------===//
12
13#ifndef LLVM_ANALYSIS_SCALAREVOLUTIONPATTERNMATCH_H
14#define LLVM_ANALYSIS_SCALAREVOLUTIONPATTERNMATCH_H
15
17
18namespace llvm {
20
21template <typename Pattern> bool match(const SCEV *S, const Pattern &P) {
22 return P.match(S);
23}
24
25template <typename Predicate> struct cst_pred_ty : public Predicate {
26 cst_pred_ty() = default;
27 cst_pred_ty(uint64_t V) : Predicate(V) {}
28 bool match(const SCEV *S) const {
30 "no vector types expected from SCEVs");
31 auto *C = dyn_cast<SCEVConstant>(S);
32 return C && this->isValue(C->getAPInt());
33 }
34};
35
36struct is_zero {
37 bool isValue(const APInt &C) const { return C.isZero(); }
38};
39
40/// Match an integer 0.
42
43struct is_one {
44 bool isValue(const APInt &C) const { return C.isOne(); }
45};
46
47/// Match an integer 1.
49
51 bool isValue(const APInt &C) const { return C.isAllOnes(); }
52};
53
54/// Match an integer with all bits set.
58
59template <typename Class> struct class_match {
60 template <typename ITy> bool match(ITy *V) const { return isa<Class>(V); }
61};
62
70
71template <typename Class> struct bind_ty {
72 Class *&VR;
73
74 bind_ty(Class *&V) : VR(V) {}
75
76 template <typename ITy> bool match(ITy *V) const {
77 if (auto *CV = dyn_cast<Class>(V)) {
78 VR = CV;
79 return true;
80 }
81 return false;
82 }
83};
84
85/// Match a SCEV, capturing it if we match.
86inline bind_ty<const SCEV> m_SCEV(const SCEV *&V) { return V; }
88 return V;
89}
91 return V;
92}
93
95 return V;
96}
97
99 return V;
100}
101
102/// Match a specified const SCEV *.
104 const SCEV *Expr;
105
107
108 template <typename ITy> bool match(ITy *S) const { return S == Expr; }
109};
110
111/// Match if we have a specific specified SCEV.
112inline specificscev_ty m_scev_Specific(const SCEV *S) { return S; }
113
117 bool isValue(const APInt &C) const { return C == CV; }
118};
119
120/// Match an SCEV constant with a plain unsigned integer.
122
124 int64_t CV;
126 bool isValue(const APInt &C) const { return C.trySExtValue() == CV; }
127};
128
129/// Match an SCEV constant with a plain signed integer (sign-extended value will
130/// be matched)
132 return V;
133}
134
136 const APInt *&CR;
137
138 bind_cst_ty(const APInt *&Op0) : CR(Op0) {}
139
140 bool match(const SCEV *S) const {
142 "no vector types expected from SCEVs");
143 auto *C = dyn_cast<SCEVConstant>(S);
144 if (!C)
145 return false;
146 CR = &C->getAPInt();
147 return true;
148 }
149};
150
151/// Match an SCEV constant and bind it to an APInt.
152inline bind_cst_ty m_scev_APInt(const APInt *&C) { return C; }
153
154/// Match a unary SCEV.
155template <typename SCEVTy, typename Op0_t> struct SCEVUnaryExpr_match {
157
159
160 bool match(const SCEV *S) const {
161 auto *E = dyn_cast<SCEVTy>(S);
162 return E && E->getNumOperands() == 1 && Op0.match(E->getOperand(0));
163 }
164};
165
166template <typename SCEVTy, typename Op0_t>
170
171template <typename Op0_t>
172inline SCEVUnaryExpr_match<SCEVSignExtendExpr, Op0_t>
173m_scev_SExt(const Op0_t &Op0) {
175}
176
177template <typename Op0_t>
178inline SCEVUnaryExpr_match<SCEVZeroExtendExpr, Op0_t>
179m_scev_ZExt(const Op0_t &Op0) {
181}
182
183template <typename Op0_t>
184inline SCEVUnaryExpr_match<SCEVPtrToIntExpr, Op0_t>
188
189template <typename Op0_t>
190inline SCEVUnaryExpr_match<SCEVTruncateExpr, Op0_t>
191m_scev_Trunc(const Op0_t &Op0) {
193}
194
195/// Match a binary SCEV.
196template <typename SCEVTy, typename Op0_t, typename Op1_t,
198 bool Commutable = false>
202
204
205 bool match(const SCEV *S) const {
206 if (auto WrappingS = dyn_cast<SCEVNAryExpr>(S))
207 if (WrappingS->getNoWrapFlags(WrapFlags) != WrapFlags)
208 return false;
209
210 auto *E = dyn_cast<SCEVTy>(S);
211 return E && E->getNumOperands() == 2 &&
212 ((Op0.match(E->getOperand(0)) && Op1.match(E->getOperand(1))) ||
213 (Commutable && Op0.match(E->getOperand(1)) &&
214 Op1.match(E->getOperand(0))));
215 }
216};
217
218template <typename SCEVTy, typename Op0_t, typename Op1_t,
220 bool Commutable = false>
221inline SCEVBinaryExpr_match<SCEVTy, Op0_t, Op1_t, WrapFlags, Commutable>
226
227template <typename Op0_t, typename Op1_t>
228inline SCEVBinaryExpr_match<SCEVAddExpr, Op0_t, Op1_t>
229m_scev_Add(const Op0_t &Op0, const Op1_t &Op1) {
230 return m_scev_Binary<SCEVAddExpr>(Op0, Op1);
231}
232
233template <typename Op0_t, typename Op1_t>
234inline SCEVBinaryExpr_match<SCEVMulExpr, Op0_t, Op1_t>
235m_scev_Mul(const Op0_t &Op0, const Op1_t &Op1) {
236 return m_scev_Binary<SCEVMulExpr>(Op0, Op1);
237}
238
239template <typename Op0_t, typename Op1_t>
240inline SCEVBinaryExpr_match<SCEVMulExpr, Op0_t, Op1_t, SCEV::FlagAnyWrap, true>
241m_scev_c_Mul(const Op0_t &Op0, const Op1_t &Op1) {
243 Op1);
244}
245
246template <typename Op0_t, typename Op1_t>
247inline SCEVBinaryExpr_match<SCEVMulExpr, Op0_t, Op1_t, SCEV::FlagNUW, true>
248m_scev_c_NUWMul(const Op0_t &Op0, const Op1_t &Op1) {
250 Op1);
251}
252
253template <typename Op0_t, typename Op1_t>
254inline SCEVBinaryExpr_match<SCEVUDivExpr, Op0_t, Op1_t>
255m_scev_UDiv(const Op0_t &Op0, const Op1_t &Op1) {
256 return m_scev_Binary<SCEVUDivExpr>(Op0, Op1);
257}
258
259template <typename Op0_t, typename Op1_t>
260inline SCEVBinaryExpr_match<SCEVSMaxExpr, Op0_t, Op1_t>
261m_scev_SMax(const Op0_t &Op0, const Op1_t &Op1) {
262 return m_scev_Binary<SCEVSMaxExpr>(Op0, Op1);
263}
264
265template <typename Op0_t, typename Op1_t>
266inline SCEVBinaryExpr_match<SCEVMinMaxExpr, Op0_t, Op1_t>
267m_scev_MinMax(const Op0_t &Op0, const Op1_t &Op1) {
268 return m_scev_Binary<SCEVMinMaxExpr>(Op0, Op1);
269}
270
271/// Match unsigned remainder pattern.
272/// Matches patterns generated by getURemExpr.
273template <typename Op0_t, typename Op1_t> struct SCEVURem_match {
277
280
281 bool match(const SCEV *Expr) const {
282 if (Expr->getType()->isPointerTy())
283 return false;
284
285 // Try to match 'zext (trunc A to iB) to iY', which is used
286 // for URem with constant power-of-2 second operands. Make sure the size of
287 // the operand A matches the size of the whole expressions.
288 const SCEV *LHS;
290 Type *TruncTy = cast<SCEVZeroExtendExpr>(Expr)->getOperand()->getType();
291 // Bail out if the type of the LHS is larger than the type of the
292 // expression for now.
293 if (SE.getTypeSizeInBits(LHS->getType()) >
294 SE.getTypeSizeInBits(Expr->getType()))
295 return false;
296 if (LHS->getType() != Expr->getType())
297 LHS = SE.getZeroExtendExpr(LHS, Expr->getType());
298 const SCEV *RHS =
299 SE.getConstant(APInt(SE.getTypeSizeInBits(Expr->getType()), 1)
300 << SE.getTypeSizeInBits(TruncTy));
301 return Op0.match(LHS) && Op1.match(RHS);
302 }
303
304 const SCEV *A;
305 const SCEVMulExpr *Mul;
307 return false;
308
309 const auto MatchURemWithDivisor = [&](const SCEV *B) {
310 // (SomeExpr + (-(SomeExpr / B) * B)).
311 if (Expr == SE.getURemExpr(A, B))
312 return Op0.match(A) && Op1.match(B);
313 return false;
314 };
315
316 // (SomeExpr + (-1 * (SomeExpr / B) * B)).
317 if (Mul->getNumOperands() == 3 && isa<SCEVConstant>(Mul->getOperand(0)))
318 return MatchURemWithDivisor(Mul->getOperand(1)) ||
319 MatchURemWithDivisor(Mul->getOperand(2));
320
321 // (SomeExpr + ((-SomeExpr / B) * B)) or (SomeExpr + ((SomeExpr / B) * -B)).
322 if (Mul->getNumOperands() == 2)
323 return MatchURemWithDivisor(Mul->getOperand(1)) ||
324 MatchURemWithDivisor(Mul->getOperand(0)) ||
325 MatchURemWithDivisor(SE.getNegativeSCEV(Mul->getOperand(1))) ||
326 MatchURemWithDivisor(SE.getNegativeSCEV(Mul->getOperand(0)));
327 return false;
328 }
329};
330
331/// Match the mathematical pattern A - (A / B) * B, where A and B can be
332/// arbitrary expressions. Also match zext (trunc A to iB) to iY, which is used
333/// for URem with constant power-of-2 second operands. It's not always easy, as
334/// A and B can be folded (imagine A is X / 2, and B is 4, A / B becomes X / 8).
335template <typename Op0_t, typename Op1_t>
340
342
343/// Match an affine SCEVAddRecExpr.
344template <typename Op0_t, typename Op1_t, typename Loop_t>
347 Loop_t Loop;
348
350 : Ops(Op0, Op1), Loop(Loop) {}
351
352 bool match(const SCEV *S) const {
353 return Ops.match(S) && Loop.match(cast<SCEVAddRecExpr>(S)->getLoop());
354 }
355};
356
357/// Match a specified const Loop*.
359 const Loop *L;
360
361 specificloop_ty(const Loop *L) : L(L) {}
362
363 bool match(const Loop *L) const { return L == this->L; }
364};
365
366inline specificloop_ty m_SpecificLoop(const Loop *L) { return L; }
367
368inline bind_ty<const Loop> m_Loop(const Loop *&L) { return L; }
369
370template <typename Op0_t, typename Op1_t>
371inline SCEVAffineAddRec_match<Op0_t, Op1_t, class_match<const Loop>>
372m_scev_AffineAddRec(const Op0_t &Op0, const Op1_t &Op1) {
374 Op0, Op1, m_Loop());
375}
376
377template <typename Op0_t, typename Op1_t, typename Loop_t>
378inline SCEVAffineAddRec_match<Op0_t, Op1_t, Loop_t>
379m_scev_AffineAddRec(const Op0_t &Op0, const Op1_t &Op1, const Loop_t &L) {
381}
382
383} // namespace SCEVPatternMatch
384} // namespace llvm
385
386#endif
assert(UImm &&(UImm !=~static_cast< T >(0)) &&"Invalid immediate!")
static GCRegistry::Add< ErlangGC > A("erlang", "erlang-compatible garbage collector")
static GCRegistry::Add< CoreCLRGC > E("coreclr", "CoreCLR-compatible GC")
static GCRegistry::Add< OcamlGC > B("ocaml", "ocaml 3.10-compatible GC")
#define P(N)
Value * RHS
Value * LHS
Class for arbitrary precision integers.
Definition APInt.h:78
Represents a single loop in the control flow graph.
Definition LoopInfo.h:40
This node represents an addition of some number of SCEVs.
This class represents a constant integer value.
This node represents multiplication of some number of SCEVs.
This means that we are dealing with an entirely unknown SCEV value, and only represent it as its LLVM...
This class represents an analyzed expression in the program.
LLVM_ABI Type * getType() const
Return the LLVM type of this SCEV expression.
NoWrapFlags
NoWrapFlags are bitfield indices into SubclassData.
The main scalar evolution driver.
The instances of the Type class are immutable: once they are created, they are never changed.
Definition Type.h:45
bool isVectorTy() const
True if this is an instance of VectorType.
Definition Type.h:273
bool isPointerTy() const
True if this is an instance of PointerType.
Definition Type.h:267
@ C
The default llvm calling convention, compatible with C.
Definition CallingConv.h:34
cstval_pred_ty< Predicate, ConstantInt, AllowPoison > cst_pred_ty
specialization of cstval_pred_ty for ConstantInt
class_match< const SCEVVScale > m_SCEVVScale()
bind_cst_ty m_scev_APInt(const APInt *&C)
Match an SCEV constant and bind it to an APInt.
cst_pred_ty< is_all_ones > m_scev_AllOnes()
Match an integer with all bits set.
SCEVUnaryExpr_match< SCEVZeroExtendExpr, Op0_t > m_scev_ZExt(const Op0_t &Op0)
SCEVBinaryExpr_match< SCEVMinMaxExpr, Op0_t, Op1_t > m_scev_MinMax(const Op0_t &Op0, const Op1_t &Op1)
class_match< const SCEVConstant > m_SCEVConstant()
cst_pred_ty< is_one > m_scev_One()
Match an integer 1.
specificloop_ty m_SpecificLoop(const Loop *L)
cst_pred_ty< is_specific_signed_cst > m_scev_SpecificSInt(int64_t V)
Match an SCEV constant with a plain signed integer (sign-extended value will be matched)
SCEVBinaryExpr_match< SCEVTy, Op0_t, Op1_t, WrapFlags, Commutable > m_scev_Binary(const Op0_t &Op0, const Op1_t &Op1)
SCEVAffineAddRec_match< Op0_t, Op1_t, class_match< const Loop > > m_scev_AffineAddRec(const Op0_t &Op0, const Op1_t &Op1)
bind_ty< const SCEVMulExpr > m_scev_Mul(const SCEVMulExpr *&V)
SCEVUnaryExpr_match< SCEVTy, Op0_t > m_scev_Unary(const Op0_t &Op0)
SCEVUnaryExpr_match< SCEVSignExtendExpr, Op0_t > m_scev_SExt(const Op0_t &Op0)
cst_pred_ty< is_zero > m_scev_Zero()
Match an integer 0.
SCEVUnaryExpr_match< SCEVTruncateExpr, Op0_t > m_scev_Trunc(const Op0_t &Op0)
bool match(const SCEV *S, const Pattern &P)
SCEVBinaryExpr_match< SCEVUDivExpr, Op0_t, Op1_t > m_scev_UDiv(const Op0_t &Op0, const Op1_t &Op1)
specificscev_ty m_scev_Specific(const SCEV *S)
Match if we have a specific specified SCEV.
SCEVBinaryExpr_match< SCEVMulExpr, Op0_t, Op1_t, SCEV::FlagNUW, true > m_scev_c_NUWMul(const Op0_t &Op0, const Op1_t &Op1)
class_match< const Loop > m_Loop()
bind_ty< const SCEVAddExpr > m_scev_Add(const SCEVAddExpr *&V)
SCEVUnaryExpr_match< SCEVPtrToIntExpr, Op0_t > m_scev_PtrToInt(const Op0_t &Op0)
bind_ty< const SCEVUnknown > m_SCEVUnknown(const SCEVUnknown *&V)
cst_pred_ty< is_specific_cst > m_scev_SpecificInt(uint64_t V)
Match an SCEV constant with a plain unsigned integer.
SCEVBinaryExpr_match< SCEVMulExpr, Op0_t, Op1_t, SCEV::FlagAnyWrap, true > m_scev_c_Mul(const Op0_t &Op0, const Op1_t &Op1)
SCEVBinaryExpr_match< SCEVSMaxExpr, Op0_t, Op1_t > m_scev_SMax(const Op0_t &Op0, const Op1_t &Op1)
SCEVURem_match< Op0_t, Op1_t > m_scev_URem(Op0_t LHS, Op1_t RHS, ScalarEvolution &SE)
Match the mathematical pattern A - (A / B) * B, where A and B can be arbitrary expressions.
class_match< const SCEV > m_SCEV()
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:644
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...
Definition Casting.h:548
@ Mul
Product of integers.
decltype(auto) cast(const From &Val)
cast<X> - Return the argument parameter cast to the specified type.
Definition Casting.h:560
SCEVBinaryExpr_match< SCEVAddRecExpr, Op0_t, Op1_t > Ops
SCEVURem_match(Op0_t Op0, Op1_t Op1, ScalarEvolution &SE)