LLVM 22.0.0git
VPlanPatternMatch.h
Go to the documentation of this file.
1//===- VPlanPatternMatch.h - Match on VPValues and recipes ------*- C++ -*-===//
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 provides a simple and efficient mechanism for performing general
10// tree-based pattern matches on the VPlan values and recipes, based on
11// LLVM's IR pattern matchers.
12//
13//===----------------------------------------------------------------------===//
14
15#ifndef LLVM_TRANSFORM_VECTORIZE_VPLANPATTERNMATCH_H
16#define LLVM_TRANSFORM_VECTORIZE_VPLANPATTERNMATCH_H
17
18#include "VPlan.h"
19
20namespace llvm {
22
23template <typename Val, typename Pattern> bool match(Val *V, const Pattern &P) {
24 return P.match(V);
25}
26
27template <typename Pattern> bool match(VPUser *U, const Pattern &P) {
28 auto *R = dyn_cast<VPRecipeBase>(U);
29 return R && match(R, P);
30}
31
32template <typename Val, typename Pattern> struct VPMatchFunctor {
33 const Pattern &P;
34 VPMatchFunctor(const Pattern &P) : P(P) {}
35 bool operator()(Val *V) const { return match(V, P); }
36};
37
38/// A match functor that can be used as a UnaryPredicate in functional
39/// algorithms like all_of.
40template <typename Val = VPUser, typename Pattern>
44
45template <typename Class> struct class_match {
46 template <typename ITy> bool match(ITy *V) const { return isa<Class>(V); }
47};
48
49/// Match an arbitrary VPValue and ignore it.
51
52template <typename Class> struct bind_ty {
53 Class *&VR;
54
55 bind_ty(Class *&V) : VR(V) {}
56
57 template <typename ITy> bool match(ITy *V) const {
58 if (auto *CV = dyn_cast<Class>(V)) {
59 VR = CV;
60 return true;
61 }
62 return false;
63 }
64};
65
66/// Match a specified VPValue.
68 const VPValue *Val;
69
70 specificval_ty(const VPValue *V) : Val(V) {}
71
72 bool match(VPValue *VPV) const { return VPV == Val; }
73};
74
75inline specificval_ty m_Specific(const VPValue *VPV) { return VPV; }
76
77/// Stores a reference to the VPValue *, not the VPValue * itself,
78/// thus can be used in commutative matchers.
80 VPValue *const &Val;
81
82 deferredval_ty(VPValue *const &V) : Val(V) {}
83
84 bool match(VPValue *const V) const { return V == Val; }
85};
86
87/// Like m_Specific(), but works if the specific value to match is determined
88/// as part of the same match() expression. For example:
89/// m_Mul(m_VPValue(X), m_Specific(X)) is incorrect, because m_Specific() will
90/// bind X before the pattern match starts.
91/// m_Mul(m_VPValue(X), m_Deferred(X)) is correct, and will check against
92/// whichever value m_VPValue(X) populated.
93inline deferredval_ty m_Deferred(VPValue *const &V) { return V; }
94
95/// Match an integer constant or vector of constants if Pred::isValue returns
96/// true for the APInt. \p BitWidth optionally specifies the bitwidth the
97/// matched constant must have. If it is 0, the matched constant can have any
98/// bitwidth.
99template <typename Pred, unsigned BitWidth = 0> struct int_pred_ty {
100 Pred P;
101
102 int_pred_ty(Pred P) : P(std::move(P)) {}
103 int_pred_ty() : P() {}
104
105 bool match(VPValue *VPV) const {
106 if (!VPV->isLiveIn())
107 return false;
108 Value *V = VPV->getLiveInIRValue();
109 if (!V)
110 return false;
111 assert(!V->getType()->isVectorTy() && "Unexpected vector live-in");
112 const auto *CI = dyn_cast<ConstantInt>(V);
113 if (!CI)
114 return false;
115
116 if (BitWidth != 0 && CI->getBitWidth() != BitWidth)
117 return false;
118 return P.isValue(CI->getValue());
119 }
120};
121
122/// Match a specified integer value or vector of all elements of that
123/// value. \p BitWidth optionally specifies the bitwidth the matched constant
124/// must have. If it is 0, the matched constant can have any bitwidth.
127
129
130 bool isValue(const APInt &C) const { return APInt::isSameValue(Val, C); }
131};
132
133template <unsigned Bitwidth = 0>
135
139
143
147
149 bool isValue(const APInt &C) const { return C.isAllOnes(); }
150};
151
152/// Match an integer or vector with all bits set.
153/// For vectors, this includes constants with undefined elements.
157
159 bool isValue(const APInt &C) const { return C.isZero(); }
160};
161
162struct is_one {
163 bool isValue(const APInt &C) const { return C.isOne(); }
164};
165
166/// Match an integer 0 or a vector with all elements equal to 0.
167/// For vectors, this includes constants with undefined elements.
171
172/// Match an integer 1 or a vector with all elements equal to 1.
173/// For vectors, this includes constants with undefined elements.
175
178
180
181 bool match(VPValue *VPV) const {
182 if (!VPV->isLiveIn())
183 return false;
184 Value *V = VPV->getLiveInIRValue();
185 if (!V)
186 return false;
187 assert(!V->getType()->isVectorTy() && "Unexpected vector live-in");
188 const auto *CI = dyn_cast<ConstantInt>(V);
189 if (!CI)
190 return false;
191 if (auto C = CI->getValue().tryZExtValue()) {
192 Res = *C;
193 return true;
194 }
195 return false;
196 }
197};
198
199/// Match a plain integer constant no wider than 64-bits, capturing it if we
200/// match.
202
203/// Matching combinators
204template <typename LTy, typename RTy> struct match_combine_or {
205 LTy L;
206 RTy R;
207
208 match_combine_or(const LTy &Left, const RTy &Right) : L(Left), R(Right) {}
209
210 template <typename ITy> bool match(ITy *V) const {
211 return L.match(V) || R.match(V);
212 }
213};
214
215template <typename LTy, typename RTy> struct match_combine_and {
216 LTy L;
217 RTy R;
218
219 match_combine_and(const LTy &Left, const RTy &Right) : L(Left), R(Right) {}
220
221 template <typename ITy> bool match(ITy *V) const {
222 return L.match(V) && R.match(V);
223 }
224};
225
226/// Combine two pattern matchers matching L || R
227template <typename LTy, typename RTy>
228inline match_combine_or<LTy, RTy> m_CombineOr(const LTy &L, const RTy &R) {
229 return match_combine_or<LTy, RTy>(L, R);
230}
231
232/// Combine two pattern matchers matching L && R
233template <typename LTy, typename RTy>
234inline match_combine_and<LTy, RTy> m_CombineAnd(const LTy &L, const RTy &R) {
235 return match_combine_and<LTy, RTy>(L, R);
236}
237
238/// Match a VPValue, capturing it if we match.
239inline bind_ty<VPValue> m_VPValue(VPValue *&V) { return V; }
240
241/// Match a VPInstruction, capturing if we match.
243
244template <typename Ops_t, unsigned Opcode, bool Commutative,
245 typename... RecipeTys>
247 Ops_t Ops;
248
249 template <typename... OpTy> Recipe_match(OpTy... Ops) : Ops(Ops...) {
250 static_assert(std::tuple_size<Ops_t>::value == sizeof...(Ops) &&
251 "number of operands in constructor doesn't match Ops_t");
252 static_assert((!Commutative || std::tuple_size<Ops_t>::value == 2) &&
253 "only binary ops can be commutative");
254 }
255
256 bool match(const VPValue *V) const {
257 auto *DefR = V->getDefiningRecipe();
258 return DefR && match(DefR);
259 }
260
261 bool match(const VPSingleDefRecipe *R) const {
262 return match(static_cast<const VPRecipeBase *>(R));
263 }
264
265 bool match(const VPRecipeBase *R) const {
266 if (std::tuple_size_v<Ops_t> == 0) {
267 auto *VPI = dyn_cast<VPInstruction>(R);
268 return VPI && VPI->getOpcode() == Opcode;
269 }
270
271 if ((!matchRecipeAndOpcode<RecipeTys>(R) && ...))
272 return false;
273
274 if (R->getNumOperands() != std::tuple_size<Ops_t>::value) {
275 assert(Opcode == Instruction::PHI &&
276 "non-variadic recipe with matched opcode does not have the "
277 "expected number of operands");
278 return false;
279 }
280
281 auto IdxSeq = std::make_index_sequence<std::tuple_size<Ops_t>::value>();
282 if (all_of_tuple_elements(IdxSeq, [R](auto Op, unsigned Idx) {
283 return Op.match(R->getOperand(Idx));
284 }))
285 return true;
286
287 return Commutative &&
288 all_of_tuple_elements(IdxSeq, [R](auto Op, unsigned Idx) {
289 return Op.match(R->getOperand(R->getNumOperands() - Idx - 1));
290 });
291 }
292
293private:
294 template <typename RecipeTy>
295 static bool matchRecipeAndOpcode(const VPRecipeBase *R) {
296 auto *DefR = dyn_cast<RecipeTy>(R);
297 // Check for recipes that do not have opcodes.
298 if constexpr (std::is_same_v<RecipeTy, VPScalarIVStepsRecipe> ||
299 std::is_same_v<RecipeTy, VPCanonicalIVPHIRecipe> ||
300 std::is_same_v<RecipeTy, VPDerivedIVRecipe>)
301 return DefR;
302 else
303 return DefR && DefR->getOpcode() == Opcode;
304 }
305
306 /// Helper to check if predicate \p P holds on all tuple elements in Ops using
307 /// the provided index sequence.
308 template <typename Fn, std::size_t... Is>
309 bool all_of_tuple_elements(std::index_sequence<Is...>, Fn P) const {
310 return (P(std::get<Is>(Ops), Is) && ...);
311 }
312};
313
314template <unsigned Opcode, typename... OpTys>
316 Recipe_match<std::tuple<OpTys...>, Opcode, /*Commutative*/ false,
319
320template <unsigned Opcode, typename... OpTys>
322 Recipe_match<std::tuple<OpTys...>, Opcode, /*Commutative*/ true,
324
325template <unsigned Opcode, typename... OpTys>
326using VPInstruction_match = Recipe_match<std::tuple<OpTys...>, Opcode,
327 /*Commutative*/ false, VPInstruction>;
328
329template <unsigned Opcode, typename... OpTys>
330inline VPInstruction_match<Opcode, OpTys...>
331m_VPInstruction(const OpTys &...Ops) {
332 return VPInstruction_match<Opcode, OpTys...>(Ops...);
333}
334
335/// BuildVector is matches only its opcode, w/o matching its operands as the
336/// number of operands is not fixed.
340
341template <typename Op0_t>
343m_Freeze(const Op0_t &Op0) {
345}
346
347template <typename Op0_t>
352
353template <typename Op0_t>
358
359template <typename Op0_t>
364
365template <typename Op0_t>
370
371template <typename Op0_t, typename Op1_t, typename Op2_t>
373m_ActiveLaneMask(const Op0_t &Op0, const Op1_t &Op1, const Op2_t &Op2) {
375}
376
377template <typename Op0_t, typename Op1_t>
379m_BranchOnCount(const Op0_t &Op0, const Op1_t &Op1) {
381}
382
383template <typename Op0_t>
385m_AnyOf(const Op0_t &Op0) {
387}
388
389template <unsigned Opcode, typename Op0_t>
393
394template <typename Op0_t>
398
399template <typename Op0_t>
403
404template <typename Op0_t>
408
409template <typename Op0_t>
412m_ZExtOrSExt(const Op0_t &Op0) {
413 return m_CombineOr(m_ZExt(Op0), m_SExt(Op0));
414}
415
416template <typename Op0_t>
418m_ZExtOrSelf(const Op0_t &Op0) {
419 return m_CombineOr(m_ZExt(Op0), Op0);
420}
421
422template <unsigned Opcode, typename Op0_t, typename Op1_t>
424 const Op1_t &Op1) {
426}
427
428template <unsigned Opcode, typename Op0_t, typename Op1_t>
430m_c_Binary(const Op0_t &Op0, const Op1_t &Op1) {
432}
433
434template <typename Op0_t, typename Op1_t>
436m_c_Add(const Op0_t &Op0, const Op1_t &Op1) {
438}
439
440template <typename Op0_t, typename Op1_t>
445
446template <typename Op0_t, typename Op1_t>
451
452template <typename Op0_t, typename Op1_t>
454m_c_Mul(const Op0_t &Op0, const Op1_t &Op1) {
456}
457
458/// Match a binary AND operation.
459template <typename Op0_t, typename Op1_t>
461m_c_BinaryAnd(const Op0_t &Op0, const Op1_t &Op1) {
463}
464
465/// Match a binary OR operation. Note that while conceptually the operands can
466/// be matched commutatively, \p Commutative defaults to false in line with the
467/// IR-based pattern matching infrastructure. Use m_c_BinaryOr for a commutative
468/// version of the matcher.
469template <typename Op0_t, typename Op1_t>
471m_BinaryOr(const Op0_t &Op0, const Op1_t &Op1) {
473}
474
475template <typename Op0_t, typename Op1_t>
477m_c_BinaryOr(const Op0_t &Op0, const Op1_t &Op1) {
479}
480
481/// Cmp_match is a variant of BinaryRecipe_match that also binds the comparison
482/// predicate. Opcodes must either be Instruction::ICmp or Instruction::FCmp, or
483/// both.
484template <typename Op0_t, typename Op1_t, unsigned... Opcodes>
485struct Cmp_match {
486 static_assert((sizeof...(Opcodes) == 1 || sizeof...(Opcodes) == 2) &&
487 "Expected one or two opcodes");
488 static_assert(
489 ((Opcodes == Instruction::ICmp || Opcodes == Instruction::FCmp) && ...) &&
490 "Expected a compare instruction opcode");
491
495
496 Cmp_match(CmpPredicate &Pred, const Op0_t &Op0, const Op1_t &Op1)
497 : Predicate(&Pred), Op0(Op0), Op1(Op1) {}
498 Cmp_match(const Op0_t &Op0, const Op1_t &Op1) : Op0(Op0), Op1(Op1) {}
499
500 bool match(const VPValue *V) const {
501 auto *DefR = V->getDefiningRecipe();
502 return DefR && match(DefR);
503 }
504
505 bool match(const VPRecipeBase *V) const {
506 if ((m_Binary<Opcodes>(Op0, Op1).match(V) || ...)) {
507 if (Predicate)
508 *Predicate = cast<VPRecipeWithIRFlags>(V)->getPredicate();
509 return true;
510 }
511 return false;
512 }
513};
514
515/// SpecificCmp_match is a variant of Cmp_match that matches the comparison
516/// predicate, instead of binding it.
517template <typename Op0_t, typename Op1_t, unsigned... Opcodes>
522
524 : Predicate(Pred), Op0(LHS), Op1(RHS) {}
525
526 bool match(const VPValue *V) const {
527 CmpPredicate CurrentPred;
528 return Cmp_match<Op0_t, Op1_t, Opcodes...>(CurrentPred, Op0, Op1)
529 .match(V) &&
531 }
532};
533
534template <typename Op0_t, typename Op1_t>
539
540template <typename Op0_t, typename Op1_t>
541inline Cmp_match<Op0_t, Op1_t, Instruction::ICmp>
542m_ICmp(CmpPredicate &Pred, const Op0_t &Op0, const Op1_t &Op1) {
543 return Cmp_match<Op0_t, Op1_t, Instruction::ICmp>(Pred, Op0, Op1);
544}
545
546template <typename Op0_t, typename Op1_t>
547inline SpecificCmp_match<Op0_t, Op1_t, Instruction::ICmp>
548m_SpecificICmp(CmpPredicate MatchPred, const Op0_t &Op0, const Op1_t &Op1) {
550 Op1);
551}
552
553template <typename Op0_t, typename Op1_t>
554inline Cmp_match<Op0_t, Op1_t, Instruction::ICmp, Instruction::FCmp>
555m_Cmp(const Op0_t &Op0, const Op1_t &Op1) {
557 Op1);
558}
559
560template <typename Op0_t, typename Op1_t>
561inline Cmp_match<Op0_t, Op1_t, Instruction::ICmp, Instruction::FCmp>
562m_Cmp(CmpPredicate &Pred, const Op0_t &Op0, const Op1_t &Op1) {
564 Pred, Op0, Op1);
565}
566
567template <typename Op0_t, typename Op1_t>
568inline SpecificCmp_match<Op0_t, Op1_t, Instruction::ICmp, Instruction::FCmp>
569m_SpecificCmp(CmpPredicate MatchPred, const Op0_t &Op0, const Op1_t &Op1) {
571 MatchPred, Op0, Op1);
572}
573
574template <typename Op0_t, typename Op1_t>
576 Recipe_match<std::tuple<Op0_t, Op1_t>, Instruction::GetElementPtr,
577 /*Commutative*/ false, VPReplicateRecipe, VPWidenGEPRecipe>,
581
582template <typename Op0_t, typename Op1_t>
584 const Op1_t &Op1) {
585 return m_CombineOr(
586 Recipe_match<std::tuple<Op0_t, Op1_t>, Instruction::GetElementPtr,
587 /*Commutative*/ false, VPReplicateRecipe, VPWidenGEPRecipe>(
588 Op0, Op1),
592 Op1)));
593}
594
595template <typename Op0_t, typename Op1_t, typename Op2_t>
597m_Select(const Op0_t &Op0, const Op1_t &Op1, const Op2_t &Op2) {
599 {Op0, Op1, Op2});
600}
601
602template <typename Op0_t>
605 Instruction::Xor, int_pred_ty<is_all_ones>, Op0_t>>
610
611template <typename Op0_t, typename Op1_t>
612inline match_combine_or<
615m_LogicalAnd(const Op0_t &Op0, const Op1_t &Op1) {
616 return m_CombineOr(
618 m_Select(Op0, Op1, m_False()));
619}
620
621template <typename Op0_t, typename Op1_t>
623m_LogicalOr(const Op0_t &Op0, const Op1_t &Op1) {
624 return m_Select(Op0, m_True(), Op1);
625}
626
627template <typename Op0_t, typename Op1_t, typename Op2_t>
629 false, VPScalarIVStepsRecipe>;
630
631template <typename Op0_t, typename Op1_t, typename Op2_t>
633m_ScalarIVSteps(const Op0_t &Op0, const Op1_t &Op1, const Op2_t &Op2) {
634 return VPScalarIVSteps_match<Op0_t, Op1_t, Op2_t>({Op0, Op1, Op2});
635}
636
637template <typename Op0_t, typename Op1_t, typename Op2_t>
640
641template <typename Op0_t, typename Op1_t, typename Op2_t>
643m_DerivedIV(const Op0_t &Op0, const Op1_t &Op1, const Op2_t &Op2) {
644 return VPDerivedIV_match<Op0_t, Op1_t, Op2_t>({Op0, Op1, Op2});
645}
646
647/// Match a call argument at a given argument index.
648template <typename Opnd_t> struct Argument_match {
649 /// Call argument index to match.
650 unsigned OpI;
651 Opnd_t Val;
652
653 Argument_match(unsigned OpIdx, const Opnd_t &V) : OpI(OpIdx), Val(V) {}
654
655 template <typename OpTy> bool match(OpTy *V) const {
656 if (const auto *R = dyn_cast<VPWidenIntrinsicRecipe>(V))
657 return Val.match(R->getOperand(OpI));
658 if (const auto *R = dyn_cast<VPWidenCallRecipe>(V))
659 return Val.match(R->getOperand(OpI));
660 if (const auto *R = dyn_cast<VPReplicateRecipe>(V))
661 if (isa<CallInst>(R->getUnderlyingInstr()))
662 return Val.match(R->getOperand(OpI + 1));
663 return false;
664 }
665};
666
667/// Match a call argument.
668template <unsigned OpI, typename Opnd_t>
669inline Argument_match<Opnd_t> m_Argument(const Opnd_t &Op) {
670 return Argument_match<Opnd_t>(OpI, Op);
671}
672
673/// Intrinsic matchers.
675 unsigned ID;
676
677 IntrinsicID_match(Intrinsic::ID IntrID) : ID(IntrID) {}
678
679 template <typename OpTy> bool match(OpTy *V) const {
680 if (const auto *R = dyn_cast<VPWidenIntrinsicRecipe>(V))
681 return R->getVectorIntrinsicID() == ID;
682 if (const auto *R = dyn_cast<VPWidenCallRecipe>(V))
683 return R->getCalledScalarFunction()->getIntrinsicID() == ID;
684 if (const auto *R = dyn_cast<VPReplicateRecipe>(V))
685 if (const auto *CI = dyn_cast<CallInst>(R->getUnderlyingInstr()))
686 if (const auto *F = CI->getCalledFunction())
687 return F->getIntrinsicID() == ID;
688 return false;
689 }
690};
691
692/// Intrinsic matches are combinations of ID matchers, and argument
693/// matchers. Higher arity matcher are defined recursively in terms of and-ing
694/// them with lower arity matchers. Here's some convenient typedefs for up to
695/// several arguments, and more can be added as needed
696template <typename T0 = void, typename T1 = void, typename T2 = void,
697 typename T3 = void>
698struct m_Intrinsic_Ty;
699template <typename T0> struct m_Intrinsic_Ty<T0> {
701};
702template <typename T0, typename T1> struct m_Intrinsic_Ty<T0, T1> {
703 using Ty =
705};
706template <typename T0, typename T1, typename T2>
711template <typename T0, typename T1, typename T2, typename T3>
716
717/// Match intrinsic calls like this:
718/// m_Intrinsic<Intrinsic::fabs>(m_VPValue(X), ...)
719template <Intrinsic::ID IntrID> inline IntrinsicID_match m_Intrinsic() {
720 return IntrinsicID_match(IntrID);
721}
722
723template <Intrinsic::ID IntrID, typename T0>
724inline typename m_Intrinsic_Ty<T0>::Ty m_Intrinsic(const T0 &Op0) {
726}
727
728template <Intrinsic::ID IntrID, typename T0, typename T1>
729inline typename m_Intrinsic_Ty<T0, T1>::Ty m_Intrinsic(const T0 &Op0,
730 const T1 &Op1) {
732}
733
734template <Intrinsic::ID IntrID, typename T0, typename T1, typename T2>
735inline typename m_Intrinsic_Ty<T0, T1, T2>::Ty
736m_Intrinsic(const T0 &Op0, const T1 &Op1, const T2 &Op2) {
737 return m_CombineAnd(m_Intrinsic<IntrID>(Op0, Op1), m_Argument<2>(Op2));
738}
739
740template <Intrinsic::ID IntrID, typename T0, typename T1, typename T2,
741 typename T3>
743m_Intrinsic(const T0 &Op0, const T1 &Op1, const T2 &Op2, const T3 &Op3) {
744 return m_CombineAnd(m_Intrinsic<IntrID>(Op0, Op1, Op2), m_Argument<3>(Op3));
745}
746
748 template <typename ITy> bool match(ITy *V) const {
749 VPValue *Val = dyn_cast<VPValue>(V);
750 return Val && Val->isLiveIn();
751 }
752};
753
755
756template <typename SubPattern_t> struct OneUse_match {
757 SubPattern_t SubPattern;
758
759 OneUse_match(const SubPattern_t &SP) : SubPattern(SP) {}
760
761 template <typename OpTy> bool match(OpTy *V) {
762 return V->hasOneUse() && SubPattern.match(V);
763 }
764};
765
766template <typename T> inline OneUse_match<T> m_OneUse(const T &SubPattern) {
767 return SubPattern;
768}
769
770} // namespace VPlanPatternMatch
771} // namespace llvm
772
773#endif
assert(UImm &&(UImm !=~static_cast< T >(0)) &&"Invalid immediate!")
const AbstractManglingParser< Derived, Alloc >::OperatorInfo AbstractManglingParser< Derived, Alloc >::Ops[]
#define F(x, y, z)
Definition MD5.cpp:55
#define T
#define T1
MachineInstr unsigned OpIdx
#define P(N)
This file contains the declarations of the Vectorization Plan base classes:
Value * RHS
Value * LHS
Class for arbitrary precision integers.
Definition APInt.h:78
static bool isSameValue(const APInt &I1, const APInt &I2)
Determine if two APInts have the same value, after zero-extending one of them (if needed!...
Definition APInt.h:553
An abstraction over a floating-point predicate, and a pack of an integer predicate with samesign info...
static LLVM_ABI std::optional< CmpPredicate > getMatching(CmpPredicate A, CmpPredicate B)
Compares two CmpPredicates taking samesign into account and returns the canonicalized CmpPredicate if...
A recipe for converting the input value IV value to the corresponding value of an IV with different s...
Definition VPlan.h:3576
This is a concrete Recipe that models a single VPlan-level instruction.
Definition VPlan.h:980
VPRecipeBase is a base class modeling a sequence of one or more output IR instructions.
Definition VPlan.h:394
VPReplicateRecipe replicates a given instruction producing multiple scalar copies of the original sca...
Definition VPlan.h:2847
A recipe for handling phi nodes of integer and floating-point inductions, producing their scalar valu...
Definition VPlan.h:3645
VPSingleDef is a base class for recipes for modeling a sequence of one or more output IR that define ...
Definition VPlan.h:521
This class augments VPValue with operands which provide the inverse def-use edges from VPValue's user...
Definition VPlanValue.h:199
Value * getLiveInIRValue() const
Returns the underlying IR value, if this VPValue is defined outside the scope of VPlan.
Definition VPlanValue.h:176
bool isLiveIn() const
Returns true if this VPValue is a live-in, i.e. defined outside the VPlan.
Definition VPlanValue.h:171
VPWidenCastRecipe is a recipe to create vector cast instructions.
Definition VPlan.h:1479
A recipe for handling GEP instructions.
Definition VPlan.h:1765
VPWidenRecipe is a recipe for producing a widened instruction using the opcode and operands of the re...
Definition VPlan.h:1436
LLVM Value Representation.
Definition Value.h:75
@ C
The default llvm calling convention, compatible with C.
Definition CallingConv.h:34
class_match< ConstantInt > m_ConstantInt()
Match an arbitrary ConstantInt and ignore it.
auto m_LogicalOr()
Matches L || R where L and R are arbitrary values.
class_match< CmpInst > m_Cmp()
Matches any compare instruction and ignore it.
auto m_LogicalAnd()
Matches L && R where L and R are arbitrary values.
AllRecipe_match< Instruction::Select, Op0_t, Op1_t, Op2_t > m_Select(const Op0_t &Op0, const Op1_t &Op1, const Op2_t &Op2)
VPInstruction_match< Instruction::Freeze, Op0_t > m_Freeze(const Op0_t &Op0)
AllRecipe_commutative_match< Instruction::And, Op0_t, Op1_t > m_c_BinaryAnd(const Op0_t &Op0, const Op1_t &Op1)
Match a binary AND operation.
AllRecipe_match< Instruction::ZExt, Op0_t > m_ZExt(const Op0_t &Op0)
AllRecipe_match< Instruction::Or, Op0_t, Op1_t > m_BinaryOr(const Op0_t &Op0, const Op1_t &Op1)
Match a binary OR operation.
int_pred_ty< is_specific_int, Bitwidth > specific_intval
int_pred_ty< is_zero_int > m_ZeroInt()
Match an integer 0 or a vector with all elements equal to 0.
SpecificCmp_match< Op0_t, Op1_t, Instruction::ICmp, Instruction::FCmp > m_SpecificCmp(CmpPredicate MatchPred, const Op0_t &Op0, const Op1_t &Op1)
match_combine_or< VPInstruction_match< VPInstruction::Not, Op0_t >, AllRecipe_commutative_match< Instruction::Xor, int_pred_ty< is_all_ones >, Op0_t > > m_Not(const Op0_t &Op0)
int_pred_ty< is_all_ones > m_AllOnes()
Match an integer or vector with all bits set.
AllRecipe_commutative_match< Opcode, Op0_t, Op1_t > m_c_Binary(const Op0_t &Op0, const Op1_t &Op1)
AllRecipe_commutative_match< Instruction::Add, Op0_t, Op1_t > m_c_Add(const Op0_t &Op0, const Op1_t &Op1)
AllRecipe_commutative_match< Instruction::Or, Op0_t, Op1_t > m_c_BinaryOr(const Op0_t &Op0, const Op1_t &Op1)
match_combine_or< AllRecipe_match< Instruction::ZExt, Op0_t >, AllRecipe_match< Instruction::SExt, Op0_t > > m_ZExtOrSExt(const Op0_t &Op0)
match_combine_and< LTy, RTy > m_CombineAnd(const LTy &L, const RTy &R)
Combine two pattern matchers matching L && R.
SpecificCmp_match< Op0_t, Op1_t, Instruction::ICmp > m_SpecificICmp(CmpPredicate MatchPred, const Op0_t &Op0, const Op1_t &Op1)
VPInstruction_match< VPInstruction::AnyOf, Op0_t > m_AnyOf(const Op0_t &Op0)
VPScalarIVSteps_match< Op0_t, Op1_t, Op2_t > m_ScalarIVSteps(const Op0_t &Op0, const Op1_t &Op1, const Op2_t &Op2)
match_combine_or< LTy, RTy > m_CombineOr(const LTy &L, const RTy &R)
Combine two pattern matchers matching L || R.
GEPLikeRecipe_match< Op0_t, Op1_t > m_GetElementPtr(const Op0_t &Op0, const Op1_t &Op1)
Recipe_match< std::tuple< OpTys... >, Opcode, false, VPInstruction > VPInstruction_match
VPInstruction_match< VPInstruction::ExtractLastElement, Op0_t > m_ExtractLastElement(const Op0_t &Op0)
AllRecipe_match< Opcode, Op0_t, Op1_t > m_Binary(const Op0_t &Op0, const Op1_t &Op1)
AllRecipe_match< Opcode, Op0_t > m_Unary(const Op0_t &Op0)
AllRecipe_commutative_match< Instruction::Mul, Op0_t, Op1_t > m_c_Mul(const Op0_t &Op0, const Op1_t &Op1)
Cmp_match< Op0_t, Op1_t, Instruction::ICmp > m_ICmp(const Op0_t &Op0, const Op1_t &Op1)
AllRecipe_match< Instruction::Mul, Op0_t, Op1_t > m_Mul(const Op0_t &Op0, const Op1_t &Op1)
specificval_ty m_Specific(const VPValue *VPV)
match_combine_or< Recipe_match< std::tuple< Op0_t, Op1_t >, Instruction::GetElementPtr, false, VPReplicateRecipe, VPWidenGEPRecipe >, match_combine_or< VPInstruction_match< VPInstruction::PtrAdd, Op0_t, Op1_t >, VPInstruction_match< VPInstruction::WidePtrAdd, Op0_t, Op1_t > > > GEPLikeRecipe_match
specific_intval< 1 > m_False()
VPDerivedIV_match< Op0_t, Op1_t, Op2_t > m_DerivedIV(const Op0_t &Op0, const Op1_t &Op1, const Op2_t &Op2)
VPMatchFunctor< Val, Pattern > match_fn(const Pattern &P)
A match functor that can be used as a UnaryPredicate in functional algorithms like all_of.
specific_intval< 0 > m_SpecificInt(uint64_t V)
VPInstruction_match< VPInstruction::ActiveLaneMask, Op0_t, Op1_t, Op2_t > m_ActiveLaneMask(const Op0_t &Op0, const Op1_t &Op1, const Op2_t &Op2)
VPInstruction_match< VPInstruction::BranchOnCount, Op0_t, Op1_t > m_BranchOnCount(const Op0_t &Op0, const Op1_t &Op1)
Recipe_match< std::tuple< Op0_t, Op1_t, Op2_t >, 0, false, VPDerivedIVRecipe > VPDerivedIV_match
AllRecipe_match< Instruction::Sub, Op0_t, Op1_t > m_Sub(const Op0_t &Op0, const Op1_t &Op1)
AllRecipe_match< Instruction::SExt, Op0_t > m_SExt(const Op0_t &Op0)
specific_intval< 1 > m_True()
Recipe_match< std::tuple< OpTys... >, Opcode, false, VPWidenRecipe, VPReplicateRecipe, VPWidenCastRecipe, VPInstruction, VPWidenSelectRecipe > AllRecipe_match
IntrinsicID_match m_Intrinsic()
Match intrinsic calls like this: m_Intrinsic<Intrinsic::fabs>(m_VPValue(X), ...)
Recipe_match< std::tuple< OpTys... >, Opcode, true, VPWidenRecipe, VPReplicateRecipe, VPInstruction > AllRecipe_commutative_match
deferredval_ty m_Deferred(VPValue *const &V)
Like m_Specific(), but works if the specific value to match is determined as part of the same match()...
VPInstruction_match< VPInstruction::Broadcast, Op0_t > m_Broadcast(const Op0_t &Op0)
bool match(Val *V, const Pattern &P)
class_match< VPValue > m_VPValue()
Match an arbitrary VPValue and ignore it.
OneUse_match< T > m_OneUse(const T &SubPattern)
VPInstruction_match< VPInstruction::ExplicitVectorLength, Op0_t > m_EVL(const Op0_t &Op0)
VPInstruction_match< VPInstruction::BuildVector > m_BuildVector()
BuildVector is matches only its opcode, w/o matching its operands as the number of operands is not fi...
AllRecipe_match< Instruction::Trunc, Op0_t > m_Trunc(const Op0_t &Op0)
match_combine_or< AllRecipe_match< Instruction::ZExt, Op0_t >, Op0_t > m_ZExtOrSelf(const Op0_t &Op0)
VPInstruction_match< VPInstruction::BranchOnCond, Op0_t > m_BranchOnCond(const Op0_t &Op0)
Argument_match< Opnd_t > m_Argument(const Opnd_t &Op)
Match a call argument.
bind_ty< VPInstruction > m_VPInstruction(VPInstruction *&V)
Match a VPInstruction, capturing if we match.
Recipe_match< std::tuple< Op0_t, Op1_t, Op2_t >, 0, false, VPScalarIVStepsRecipe > VPScalarIVSteps_match
int_pred_ty< is_one > m_One()
Match an integer 1 or a vector with all elements equal to 1.
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 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
DWARFExpression::Operation Op
constexpr unsigned BitWidth
OutputIt move(R &&Range, OutputIt Out)
Provide wrappers to std::move which take ranges instead of having to pass begin/end explicitly.
Definition STLExtras.h:1847
decltype(auto) cast(const From &Val)
cast<X> - Return the argument parameter cast to the specified type.
Definition Casting.h:565
Implement std::hash so that hash_code can be used in STL containers.
Definition BitVector.h:851
Intrinsic matches are combinations of ID matchers, and argument matchers.
A recipe for widening select instructions.
Definition VPlan.h:1719
Match a call argument at a given argument index.
unsigned OpI
Call argument index to match.
Argument_match(unsigned OpIdx, const Opnd_t &V)
Cmp_match is a variant of BinaryRecipe_match that also binds the comparison predicate.
Cmp_match(CmpPredicate &Pred, const Op0_t &Op0, const Op1_t &Op1)
Cmp_match(const Op0_t &Op0, const Op1_t &Op1)
bool match(const VPValue *V) const
bool match(const VPRecipeBase *V) const
bool match(const VPSingleDefRecipe *R) const
bool match(const VPValue *V) const
bool match(const VPRecipeBase *R) const
SpecificCmp_match is a variant of Cmp_match that matches the comparison predicate,...
SpecificCmp_match(CmpPredicate Pred, const Op0_t &LHS, const Op1_t &RHS)
Stores a reference to the VPValue *, not the VPValue * itself, thus can be used in commutative matche...
Match an integer constant or vector of constants if Pred::isValue returns true for the APInt.
bool isValue(const APInt &C) const
Match a specified integer value or vector of all elements of that value.
match_combine_and< typename m_Intrinsic_Ty< T0, T1 >::Ty, Argument_match< T2 > > Ty
match_combine_and< typename m_Intrinsic_Ty< T0 >::Ty, Argument_match< T1 > > Ty
match_combine_and< IntrinsicID_match, Argument_match< T0 > > Ty
Intrinsic matches are combinations of ID matchers, and argument matchers.
match_combine_and< typename m_Intrinsic_Ty< T0, T1, T2 >::Ty, Argument_match< T3 > > Ty
match_combine_and(const LTy &Left, const RTy &Right)
match_combine_or(const LTy &Left, const RTy &Right)