15#ifndef LLVM_TRANSFORM_VECTORIZE_VPLANPATTERNMATCH_H
16#define LLVM_TRANSFORM_VECTORIZE_VPLANPATTERNMATCH_H
23template <
typename Val,
typename Pattern>
bool match(Val *V,
const Pattern &
P) {
44 template <
typename ITy>
bool match(ITy *V)
const {
86template <
typename Pred,
unsigned BitW
idth = 0>
struct int_pred_ty {
99 if (!CI && V->getType()->isVectorTy())
102 C->getSplatValue(
false));
108 return P.isValue(CI->getValue());
123template <
unsigned Bitw
idth = 0>
173 template <
typename ITy>
bool match(ITy *V)
const {
174 return L.match(V) ||
R.match(V);
184 template <
typename ITy>
bool match(ITy *V)
const {
185 return L.match(V) &&
R.match(V);
190template <
typename LTy,
typename RTy>
196template <
typename LTy,
typename RTy>
207template <
typename Ops_t,
unsigned Opcode,
bool Commutative,
208 typename... RecipeTys>
213 static_assert(std::tuple_size<Ops_t>::value ==
sizeof...(Ops) &&
214 "number of operands in constructor doesn't match Ops_t");
215 static_assert((!Commutative || std::tuple_size<Ops_t>::value == 2) &&
216 "only binary ops can be commutative");
220 auto *DefR = V->getDefiningRecipe();
221 return DefR &&
match(DefR);
229 if (std::tuple_size<Ops_t>::value == 0) {
231 "can only match BuildVector with empty ops");
236 if ((!matchRecipeAndOpcode<RecipeTys>(R) && ...))
239 if (R->getNumOperands() != std::tuple_size<Ops_t>::value) {
240 assert(Opcode == Instruction::PHI &&
241 "non-variadic recipe with matched opcode does not have the "
242 "expected number of operands");
246 auto IdxSeq = std::make_index_sequence<std::tuple_size<Ops_t>::value>();
247 if (all_of_tuple_elements(IdxSeq, [R](
auto Op,
unsigned Idx) {
248 return Op.match(R->getOperand(Idx));
252 return Commutative &&
253 all_of_tuple_elements(IdxSeq, [R](
auto Op,
unsigned Idx) {
254 return Op.match(R->getOperand(R->getNumOperands() - Idx - 1));
259 template <
typename RecipeTy>
260 static bool matchRecipeAndOpcode(
const VPRecipeBase *R) {
263 if constexpr (std::is_same_v<RecipeTy, VPScalarIVStepsRecipe> ||
264 std::is_same_v<RecipeTy, VPCanonicalIVPHIRecipe> ||
265 std::is_same_v<RecipeTy, VPDerivedIVRecipe>)
268 return DefR && DefR->getOpcode() == Opcode;
273 template <
typename Fn, std::size_t... Is>
274 bool all_of_tuple_elements(std::index_sequence<Is...>, Fn
P)
const {
275 return (
P(std::get<Is>(
Ops), Is) && ...);
279template <
unsigned Opcode,
typename... OpTys>
285template <
unsigned Opcode,
typename... OpTys>
290template <
unsigned Opcode,
typename... OpTys>
294template <
unsigned Opcode,
typename... OpTys>
306template <
typename Op0_t>
312template <
typename Op0_t>
318template <
typename Op0_t>
324template <
typename Op0_t>
330template <
typename Op0_t>
336template <
typename Op0_t,
typename Op1_t,
typename Op2_t>
342template <
typename Op0_t,
typename Op1_t>
348template <
unsigned Opcode,
typename Op0_t>
353template <
typename Op0_t>
358template <
typename Op0_t>
363template <
typename Op0_t>
368template <
typename Op0_t>
375template <
typename Op0_t>
381template <
unsigned Opcode,
typename Op0_t,
typename Op1_t>
387template <
unsigned Opcode,
typename Op0_t,
typename Op1_t>
393template <
typename Op0_t,
typename Op1_t>
399template <
typename Op0_t,
typename Op1_t>
405template <
typename Op0_t,
typename Op1_t>
411template <
typename Op0_t,
typename Op1_t>
418template <
typename Op0_t,
typename Op1_t>
428template <
typename Op0_t,
typename Op1_t>
434template <
typename Op0_t,
typename Op1_t>
443template <
typename Op0_t,
typename Op1_t,
unsigned... Opcodes>
445 static_assert((
sizeof...(Opcodes) == 1 ||
sizeof...(Opcodes) == 2) &&
446 "Expected one or two opcodes");
448 ((Opcodes == Instruction::ICmp || Opcodes == Instruction::FCmp) && ...) &&
449 "Expected a compare instruction opcode");
460 auto *DefR = V->getDefiningRecipe();
461 return DefR &&
match(DefR);
476template <
typename Op0_t,
typename Op1_t,
unsigned... Opcodes>
493template <
typename Op0_t,
typename Op1_t>
499template <
typename Op0_t,
typename Op1_t>
500inline Cmp_match<Op0_t, Op1_t, Instruction::ICmp>
505template <
typename Op0_t,
typename Op1_t>
506inline SpecificCmp_match<Op0_t, Op1_t, Instruction::ICmp>
512template <
typename Op0_t,
typename Op1_t>
513inline Cmp_match<Op0_t, Op1_t, Instruction::ICmp, Instruction::FCmp>
519template <
typename Op0_t,
typename Op1_t>
520inline Cmp_match<Op0_t, Op1_t, Instruction::ICmp, Instruction::FCmp>
526template <
typename Op0_t,
typename Op1_t>
527inline SpecificCmp_match<Op0_t, Op1_t, Instruction::ICmp, Instruction::FCmp>
530 MatchPred, Op0, Op1);
533template <
typename Op0_t,
typename Op1_t>
541template <
typename Op0_t,
typename Op1_t>
545 Recipe_match<std::tuple<Op0_t, Op1_t>, Instruction::GetElementPtr,
554template <
typename Op0_t,
typename Op1_t,
typename Op2_t>
561template <
typename Op0_t>
564 Instruction::Xor, int_pred_ty<is_all_ones>,
Op0_t>>
570template <
typename Op0_t,
typename Op1_t>
580template <
typename Op0_t,
typename Op1_t>
586template <
typename Op0_t,
typename Op1_t,
typename Op2_t>
590template <
typename Op0_t,
typename Op1_t,
typename Op2_t>
596template <
typename Op0_t,
typename Op1_t,
typename Op2_t>
600template <
typename Op0_t,
typename Op1_t,
typename Op2_t>
614 template <
typename OpTy>
bool match(OpTy *V)
const {
616 return Val.match(R->getOperand(
OpI));
618 return Val.match(R->getOperand(
OpI));
621 return Val.match(R->getOperand(
OpI + 1));
627template <
unsigned OpI,
typename Opnd_t>
638 template <
typename OpTy>
bool match(OpTy *V)
const {
640 return R->getVectorIntrinsicID() ==
ID;
642 return R->getCalledScalarFunction()->getIntrinsicID() ==
ID;
645 if (
const auto *
F = CI->getCalledFunction())
646 return F->getIntrinsicID() ==
ID;
655template <
typename T0 = void,
typename T1 = void,
typename T2 = void,
665template <
typename T0,
typename T1,
typename T2>
670template <
typename T0,
typename T1,
typename T2,
typename T3>
682template <Intrinsic::ID IntrID,
typename T0>
687template <Intrinsic::ID IntrID,
typename T0,
typename T1>
693template <Intrinsic::ID IntrID,
typename T0,
typename T1,
typename T2>
assert(UImm &&(UImm !=~static_cast< T >(0)) &&"Invalid immediate!")
const AbstractManglingParser< Derived, Alloc >::OperatorInfo AbstractManglingParser< Derived, Alloc >::Ops[]
MachineInstr unsigned OpIdx
This file contains the declarations of the Vectorization Plan base classes:
Class for arbitrary precision integers.
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!...
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...
This is a concrete Recipe that models a single VPlan-level instruction.
@ BuildVector
Creates a fixed-width vector containing all operands.
VPRecipeBase is a base class modeling a sequence of one or more output IR instructions.
VPReplicateRecipe replicates a given instruction producing multiple scalar copies of the original sca...
A recipe for handling phi nodes of integer and floating-point inductions, producing their scalar valu...
VPSingleDef is a base class for recipes for modeling a sequence of one or more output IR that define ...
This class augments VPValue with operands which provide the inverse def-use edges from VPValue's user...
Value * getLiveInIRValue() const
Returns the underlying IR value, if this VPValue is defined outside the scope of VPlan.
bool isLiveIn() const
Returns true if this VPValue is a live-in, i.e. defined outside the VPlan.
VPWidenCastRecipe is a recipe to create vector cast instructions.
A recipe for handling GEP instructions.
VPWidenRecipe is a recipe for producing a widened instruction using the opcode and operands of the re...
LLVM Value Representation.
@ C
The default llvm calling convention, compatible with C.
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)
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)
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.
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.
auto dyn_cast_or_null(const Y &Val)
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...
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.
decltype(auto) cast(const From &Val)
cast<X> - Return the argument parameter cast to the specified type.
Implement std::hash so that hash_code can be used in STL containers.
Intrinsic matches are combinations of ID matchers, and argument matchers.
A recipe for widening select instructions.
Match a call argument at a given argument index.
unsigned OpI
Call argument index to match.
Argument_match(unsigned OpIdx, const Opnd_t &V)
bool match(OpTy *V) const
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
IntrinsicID_match(Intrinsic::ID IntrID)
bool match(OpTy *V) const
bool match(const VPSingleDefRecipe *R) const
Recipe_match(OpTy... Ops)
std::tuple< OpTys... > Ops
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)
const CmpPredicate Predicate
bool match(const VPValue *V) const
Stores a reference to the VPValue *, not the VPValue * itself, thus can be used in commutative matche...
deferredval_ty(VPValue *const &V)
bool match(VPValue *const V) const
Match an integer constant or vector of constants if Pred::isValue returns true for the APInt.
bool match(VPValue *VPV) const
bool isValue(const APInt &C) const
bool isValue(const APInt &C) const
Match a specified integer value or vector of all elements of that value.
bool isValue(const APInt &C) const
is_specific_int(APInt Val)
bool isValue(const APInt &C) const
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)
llvm::VPlanPatternMatch::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 > > >::L Recipe_match< std::tuple< Op0_t, Op1_t >, Instruction::GetElementPtr, false, VPReplicateRecipe, VPWidenGEPRecipe > L
llvm::VPlanPatternMatch::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 > > >::R match_combine_or< VPInstruction_match< VPInstruction::PtrAdd, Op0_t, Op1_t >, VPInstruction_match< VPInstruction::WidePtrAdd, Op0_t, Op1_t > > R
match_combine_or(const LTy &Left, const RTy &Right)
Match a specified VPValue.
specificval_ty(const VPValue *V)
bool match(VPValue *VPV) const