15#ifndef LLVM_TRANSFORM_VECTORIZE_VPLANPATTERNMATCH_H
16#define LLVM_TRANSFORM_VECTORIZE_VPLANPATTERNMATCH_H
26template <
typename Val,
typename Pattern>
bool match(Val *V,
const Pattern &
P) {
75template <
typename Pred,
unsigned BitW
idth = 0>
struct int_pred_ty {
84 VPV = VPI->getOperand(0);
91 return P.isValue(CI->getAPInt());
108template <
unsigned Bitw
idth = 0>
165 Res = &CI->getAPInt();
178 const APInt *APConst;
220template <
typename Ops_t,
unsigned Opcode,
bool Commutative,
221 typename... RecipeTys>
226 static_assert(std::tuple_size<Ops_t>::value ==
sizeof...(Ops) &&
227 "number of operands in constructor doesn't match Ops_t");
228 static_assert((!Commutative || std::tuple_size<Ops_t>::value == 2) &&
229 "only binary ops can be commutative");
233 auto *DefR = V->getDefiningRecipe();
234 return DefR &&
match(DefR);
242 if (std::tuple_size_v<Ops_t> == 0) {
244 return VPI && VPI->getOpcode() == Opcode;
247 if ((!matchRecipeAndOpcode<RecipeTys>(R) && ...))
250 if (R->getNumOperands() < std::tuple_size<Ops_t>::value) {
254 (RepR && std::tuple_size_v<Ops_t> ==
255 RepR->getNumOperands() - RepR->isPredicated())) &&
256 "non-variadic recipe with matched opcode does not have the "
257 "expected number of operands");
264 if (R->getNumOperands() > std::tuple_size<Ops_t>::value) {
266 if (!VPI || !VPI->isMasked() ||
267 VPI->getNumOperandsWithoutMask() != std::tuple_size<Ops_t>::value)
271 auto IdxSeq = std::make_index_sequence<std::tuple_size<Ops_t>::value>();
272 if (all_of_tuple_elements(IdxSeq, [R](
auto Op,
unsigned Idx) {
273 return Op.match(R->getOperand(Idx));
277 return Commutative &&
278 all_of_tuple_elements(IdxSeq, [R](
auto Op,
unsigned Idx) {
279 return Op.match(R->getOperand(R->getNumOperands() - Idx - 1));
284 template <
typename RecipeTy>
285 static bool matchRecipeAndOpcode(
const VPRecipeBase *R) {
288 if constexpr (std::is_same_v<RecipeTy, VPScalarIVStepsRecipe> ||
289 std::is_same_v<RecipeTy, VPDerivedIVRecipe> ||
290 std::is_same_v<RecipeTy, VPVectorEndPointerRecipe>)
293 return DefR && DefR->getOpcode() == Opcode;
298 template <
typename Fn, std::size_t... Is>
299 bool all_of_tuple_elements(std::index_sequence<Is...>,
300 [[maybe_unused]] Fn
P)
const {
301 return (
P(std::get<Is>(
Ops), Is) && ...);
305template <
unsigned Opcode,
typename... OpTys>
311template <
unsigned Opcode,
typename... OpTys>
316template <
unsigned Opcode,
typename... OpTys>
320template <
unsigned Opcode,
typename... OpTys>
325template <
unsigned Opcode,
typename... OpTys>
331template <
unsigned Opcode,
typename Op0_t,
typename Op1_t>
350template <
typename Op0_t>
360template <
typename Op0_t>
371template <
typename Op0_t,
typename Op1_t>
377template <
typename Op0_t>
383template <
typename Op0_t>
389template <
typename Op0_t>
395template <
typename Op0_t,
typename Op1_t>
401template <
typename Op0_t,
typename Op1_t,
typename Op2_t>
407template <
typename Op0_t,
typename Op1_t>
413template <
typename Op0_t>
419template <
typename Op0_t>
427template <
typename Op0_t>
433template <
typename Op0_t,
typename Op1_t,
typename Op2_t>
443template <
typename Op0_t,
typename Op1_t>
453template <
typename Op0_t>
459template <
typename Op0_t>
465template <
typename Op0_t>
471template <
typename Op0_t,
typename Op1_t,
typename Op2_t>
478template <
typename Op0_t>
487template <
typename Op0_t,
typename Op1_t>
495template <
typename Op0_t>
505template <
typename Op0_t>
511template <
unsigned Opcode,
typename Op0_t>
516template <
typename Op0_t>
521template <
typename Op0_t>
527template <
typename Op0_t>
532template <
typename Op0_t>
537template <
typename Op0_t>
542template <
typename Op0_t>
547template <
typename Op0_t>
558template <
typename Op0_t>
inline auto m_AnyNeg(
const Op0_t &Op0) {
562template <
typename Op0_t>
572template <
unsigned Opcode,
typename Op0_t,
typename Op1_t>
578template <
unsigned Opcode,
typename Op0_t,
typename Op1_t>
584template <
typename Op0_t,
typename Op1_t>
590template <
typename Op0_t,
typename Op1_t>
596template <
typename Op0_t,
typename Op1_t>
602template <
typename Op0_t,
typename Op1_t>
608template <
typename Op0_t,
typename Op1_t>
614template <
typename Op0_t,
typename Op1_t>
620template <
typename Op0_t,
typename Op1_t>
626template <
typename Op0_t,
typename Op1_t>
632template <
typename Op0_t,
typename Op1_t>
638template <
typename Op0_t,
typename Op1_t>
644template <
typename Op0_t,
typename Op1_t>
651template <
typename Op0_t,
typename Op1_t>
661template <
typename Op0_t,
typename Op1_t>
667template <
typename Op0_t,
typename Op1_t>
676template <
typename Op0_t,
typename Op1_t,
unsigned... Opcodes>
678 static_assert((
sizeof...(Opcodes) == 1 ||
sizeof...(Opcodes) == 2) &&
679 "Expected one or two opcodes");
681 ((Opcodes == Instruction::ICmp || Opcodes == Instruction::FCmp) && ...) &&
682 "Expected a compare instruction opcode");
693 auto *DefR = V->getDefiningRecipe();
694 return DefR &&
match(DefR);
709template <
typename Op0_t,
typename Op1_t,
unsigned... Opcodes>
719 auto *DefR = V->getDefiningRecipe();
720 return DefR &&
match(DefR);
731template <
typename Op0_t,
typename Op1_t>
737template <
typename Op0_t,
typename Op1_t>
738inline Cmp_match<Op0_t, Op1_t, Instruction::ICmp>
743template <
typename Op0_t,
typename Op1_t>
744inline SpecificCmp_match<Op0_t, Op1_t, Instruction::ICmp>
750template <
typename Op0_t,
typename Op1_t>
751inline Cmp_match<Op0_t, Op1_t, Instruction::ICmp, Instruction::FCmp>
757template <
typename Op0_t,
typename Op1_t>
758inline Cmp_match<Op0_t, Op1_t, Instruction::ICmp, Instruction::FCmp>
764template <
typename Op0_t,
typename Op1_t>
765inline SpecificCmp_match<Op0_t, Op1_t, Instruction::ICmp, Instruction::FCmp>
768 MatchPred, Op0, Op1);
771template <
typename Op0_t,
typename Op1_t>
774 Recipe_match<std::tuple<Op0_t, Op1_t>, Instruction::GetElementPtr,
781template <
typename Op0_t,
typename Op1_t,
typename Op2_t>
788template <
typename Op0_t>
inline auto m_Not(
const Op0_t &Op0) {
793template <
typename Op0_t,
typename Op1_t,
typename Op2_t>
798template <
typename Op0_t,
typename Op1_t>
805template <
typename Op0_t,
typename Op1_t>
812template <
typename Op0_t,
typename Op1_t>
819template <
typename Op0_t,
typename Op1_t>
826 template <
typename ArgTy>
bool match(
const ArgTy *V)
const {
828 return RV && RV->getDefiningRegion()->getCanonicalIV() == RV;
841 template <
typename ArgTy>
bool match(ArgTy *V)
const {
843 if (!WidenIV || !WidenIV->isCanonical())
854inline canonical_widen_iv_match
859template <
typename Op0_t,
typename Op1_t,
typename Op2_t>
866template <
typename Op0_t,
typename Op1_t,
typename Op2_t>
872template <
typename Addr_t,
typename Mask_t>
struct Load_match {
878 template <
typename OpTy>
bool match(
const OpTy *V)
const {
880 if (!Load || !
Addr.match(Load->getAddr()) || !Load->isMasked() ||
881 !
Mask.match(Load->getMask()))
888template <
typename Addr_t,
typename Mask_t>
890 const Mask_t &Mask) {
894template <
typename Addr_t,
typename Val_t,
typename Mask_t>
struct Store_match {
902 template <
typename OpTy>
bool match(
const OpTy *V)
const {
904 if (!Store || !
Addr.match(Store->getAddr()) ||
905 !
Val.match(Store->getStoredValue()) || !Store->isMasked() ||
906 !
Mask.match(Store->getMask()))
913template <
typename Addr_t,
typename Val_t,
typename Mask_t>
914inline Store_match<Addr_t, Val_t, Mask_t>
919template <
typename Op0_t,
typename Op1_t>
924template <
typename Op0_t,
typename Op1_t>
938 template <
typename OpTy>
bool match(OpTy *V)
const {
940 return Val.match(R->getOperand(
OpI));
942 return Val.match(R->getOperand(
OpI));
944 if (R->getOpcode() == Instruction::Call)
945 return Val.match(R->getOperand(
OpI));
947 if (R->getOpcode() == Instruction::Call)
948 return Val.match(R->getOperand(
OpI));
954template <
unsigned OpI,
typename Opnd_t>
965 template <
typename OpTy>
bool match(OpTy *V)
const {
974template <
typename T0 = void,
typename T1 = void,
typename T2 = void,
984template <
typename T0,
typename T1,
typename T2>
989template <
typename T0,
typename T1,
typename T2,
typename T3>
1006template <Intrinsic::ID IntrID,
typename T0>
1011template <Intrinsic::ID IntrID,
typename T0,
typename T1>
1017template <Intrinsic::ID IntrID,
typename T0,
typename T1,
typename T2>
1023template <
Intrinsic::ID IntrID,
typename T0,
typename T1,
typename T2,
1026m_Intrinsic(
const T0 &Op0,
const T1 &Op1,
const T2 &Op2,
const T3 &Op3) {
1046 template <
typename ITy>
bool match(ITy *V)
const {
1047 return matchRecipeAndBind<VPWidenGEPRecipe>(V) ||
1048 matchRecipeAndBind<VPInstruction>(V) ||
1049 matchRecipeAndBind<VPReplicateRecipe>(V);
1053 template <
typename RecipeTy>
bool matchRecipeAndBind(
const VPValue *V)
const {
1058 if constexpr (std::is_same_v<RecipeTy, VPWidenGEPRecipe>) {
1060 }
else if (DefR->getOpcode() == Instruction::GetElementPtr) {
1062 ->getSourceElementType();
1063 }
else if constexpr (std::is_same_v<RecipeTy, VPInstruction>) {
1066 LLVMContext &Ctx = DefR->getParent()->getPlan()->getContext();
1092 template <
typename OpTy>
bool match(OpTy *V)
const {
1093 return V->hasOneUse() &&
SubPattern.match(V);
1106template <
typename Op0_t,
typename Op1_t>
1114template <
typename MatchT>
assert(UImm &&(UImm !=~static_cast< T >(0)) &&"Invalid immediate!")
static constexpr Value * getValue(Ty &ValueOrUse)
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.
std::optional< uint64_t > tryZExtValue() const
Get zero extended value if possible.
static bool isSameValue(const APInt &I1, const APInt &I2, bool SignedCompare=false)
Determine if two APInts have the same value, after zero-extending or sign-extending (if SignedCompare...
Represent a constant reference to an array (0 or more elements consecutively in memory),...
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...
This is an important class for using LLVM in a threaded context.
The instances of the Type class are immutable: once they are created, they are never changed.
static LLVM_ABI IntegerType * getInt8Ty(LLVMContext &C)
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.
@ ExtractLastActive
Extracts the last active lane from a set of vectors.
VPRecipeBase is a base class modeling a sequence of one or more output IR instructions.
A recipe for handling reduction phis.
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...
VPSingleDefRecipe is a base class for recipes that model a sequence of one or more output IR that def...
This class augments VPValue with operands which provide the inverse def-use edges from VPValue's user...
This is the base class of the VPlan Def/Use graph, used for modeling the data flow into,...
A recipe to compute a pointer to the last element of each part of a widened memory access for widened...
VPWidenCastRecipe is a recipe to create vector cast instructions.
A recipe for handling GEP instructions.
A recipe for handling phi nodes of integer and floating-point inductions, producing their vector valu...
VPWidenRecipe is a recipe for producing a widened instruction using the opcode and operands of the re...
@ C
The default llvm calling convention, compatible with C.
match_isa< To... > m_Isa()
match_combine_or< Ty... > m_CombineOr(const Ty &...Ps)
Combine pattern matchers matching any of Ps patterns.
match_combine_and< Ty... > m_CombineAnd(const Ty &...Ps)
Combine pattern matchers matching all of Ps patterns.
auto m_Cmp()
Matches any compare instruction and ignore it.
auto m_LogicalOr()
Matches L || R where L and R are arbitrary values.
auto m_LogicalAnd()
Matches L && R where L and R are arbitrary values.
auto m_ConstantInt()
Match an arbitrary ConstantInt and ignore it.
VPInstruction_match< VPInstruction::ExtractLastLane, VPInstruction_match< VPInstruction::ExtractLastPart, Op0_t > > m_ExtractLastLaneOfLastPart(const Op0_t &Op0)
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::URem, Op0_t, Op1_t > m_URem(const Op0_t &Op0, const Op1_t &Op1)
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
Store_match< Addr_t, Val_t, Mask_t > m_MaskedStore(const Addr_t &Addr, const Val_t &Val, const Mask_t &Mask)
Match a (possibly reversed) masked store.
int_pred_ty< is_zero_int > m_ZeroInt()
Match an integer 0 or a vector with all elements equal to 0.
AllRecipe_match< Instruction::FMul, Op0_t, Op1_t > m_FMul(const Op0_t &Op0, const Op1_t &Op1)
SpecificCmp_match< Op0_t, Op1_t, Instruction::ICmp, Instruction::FCmp > m_SpecificCmp(CmpPredicate MatchPred, const Op0_t &Op0, const Op1_t &Op1)
VPInstruction_match< VPInstruction::AnyOf > m_AnyOf()
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)
bool matchFindIVResult(VPInstruction *VPI, Op0_t ReducedIV, Op1_t Start)
Match FindIV result pattern: select(icmp ne ComputeReductionResult(ReducedIV), Sentinel),...
VPInstruction_match< VPInstruction::ComputeReductionResult, Op0_t > m_ComputeReductionResult(const Op0_t &Op0)
auto m_WidenAnyExtend(const Op0_t &Op0)
match_bind< VPIRValue > m_VPIRValue(VPIRValue *&V)
Match a VPIRValue.
VPInstruction_match< VPInstruction::StepVector > m_StepVector()
auto m_c_LogicalOr(const Op0_t &Op0, const Op1_t &Op1)
match_deferred< VPValue > m_Deferred(VPValue *const &V)
Like m_Specific(), but works if the specific value to match is determined as part of the same match()...
match_combine_or< AllRecipe_match< Instruction::ZExt, Op0_t >, AllRecipe_match< Instruction::SExt, Op0_t > > m_ZExtOrSExt(const Op0_t &Op0)
auto m_VPPhi(const Op0_t &Op0, const Op1_t &Op1)
SpecificCmp_match< Op0_t, Op1_t, Instruction::ICmp > m_SpecificICmp(CmpPredicate MatchPred, const Op0_t &Op0, const Op1_t &Op1)
AllRecipe_match< Instruction::Add, Op0_t, Op1_t > m_Add(const Op0_t &Op0, const Op1_t &Op1)
match_poison m_Poison()
Match a VPIRValue that's poison.
auto m_c_Select(const Op0_t &Op0, const Op1_t &Op1, const Op2_t &Op2)
Recipe_match< std::tuple< OpTys... >, Opcode, false, VPInstruction > VPInstruction_match
VPInstruction_match< VPInstruction::BranchOnTwoConds > m_BranchOnTwoConds()
VPInstruction_match< Instruction::InsertElement, Op0_t, Op1_t, Op2_t > m_InsertElement(const Op0_t &Op0, const Op1_t &Op1, const Op2_t &Op2)
AllRecipe_match< Opcode, Op0_t, Op1_t > m_Binary(const Op0_t &Op0, const Op1_t &Op1)
auto match_fn(const Pattern &P)
A match functor that can be used as a UnaryPredicate in functional algorithms like all_of.
VPInstruction_match< VPInstruction::LastActiveLane, Op0_t > m_LastActiveLane(const Op0_t &Op0)
AllRecipe_match< Opcode, Op0_t > m_Unary(const Op0_t &Op0)
auto m_WidenIntrinsic(const T &...Ops)
Recipe_match< std::tuple< OpTys... >, Opcode, true, VPInstruction > VPInstruction_commutative_match
AllRecipe_commutative_match< Instruction::FAdd, Op0_t, Op1_t > m_c_FAdd(const Op0_t &Op0, const Op1_t &Op1)
Load_match< Addr_t, Mask_t > m_MaskedLoad(const Addr_t &Addr, const Mask_t &Mask)
Match a (possibly reversed) masked load.
VPInstruction_match< VPInstruction::ExtractLastActive, Op0_t, Op1_t, Op2_t > m_ExtractLastActive(const Op0_t &Op0, const Op1_t &Op1, const Op2_t &Op2)
match_combine_or< AllRecipe_match< Instruction::Trunc, Op0_t >, Op0_t > m_TruncOrSelf(const Op0_t &Op0)
AllRecipe_match< Instruction::FPExt, Op0_t > m_FPExt(const Op0_t &Op0)
AllRecipe_commutative_match< Instruction::Mul, Op0_t, Op1_t > m_c_Mul(const Op0_t &Op0, const Op1_t &Op1)
canonical_widen_iv_match m_CanonicalWidenIV()
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)
VPInstruction_match< VPInstruction::ExitingIVValue, Op0_t > m_ExitingIVValue(const Op0_t &Op0)
VPInstruction_match< Instruction::ExtractElement, Op0_t, Op1_t > m_ExtractElement(const Op0_t &Op0, const Op1_t &Op1)
specific_intval< 1 > m_False()
VPInstruction_match< VPInstruction::ExtractLastLane, Op0_t > m_ExtractLastLane(const Op0_t &Op0)
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)
match_bind< VPSingleDefRecipe > m_VPSingleDefRecipe(VPSingleDefRecipe *&V)
Match a VPSingleDefRecipe, capturing if we match.
VPInstruction_match< VPInstruction::BranchOnCount > m_BranchOnCount()
auto m_GetElementPtr(const Op0_t &Op0, const Op1_t &Op1)
auto m_ZExtOrTruncOrSelf(const Op0_t &Op0)
AllRecipe_match< Instruction::Sub, Op0_t, Op1_t > m_Sub(const Op0_t &Op0, const Op1_t &Op1)
canonical_iv_match m_CanonicalIV()
AllRecipe_match< Instruction::SExt, Op0_t > m_SExt(const Op0_t &Op0)
VPInstruction_commutative_match< Opcode, Op0_t, Op1_t > m_c_VPInstruction(const Op0_t &Op0, const Op1_t &Op1)
specific_intval< 1 > m_True()
auto m_VPValue()
Match an arbitrary VPValue and ignore it.
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
specific_intval< 0 > m_SpecificSInt(int64_t V)
AllRecipe_match< Instruction::FAdd, Op0_t, Op1_t > m_FAdd(const Op0_t &Op0, const Op1_t &Op1)
VectorEndPointerRecipe_match< Op0_t, Op1_t > m_VecEndPtr(const Op0_t &Op0, const Op1_t &Op1)
VPInstruction_match< VPInstruction::ExtractLastPart, Op0_t > m_ExtractLastPart(const Op0_t &Op0)
VPInstruction_match< VPInstruction::Broadcast, Op0_t > m_Broadcast(const Op0_t &Op0)
bool match(Val *V, const Pattern &P)
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)
VPInstruction_match< VPInstruction::ExtractPenultimateElement, Op0_t > m_ExtractPenultimateElement(const Op0_t &Op0)
AllRecipe_match< Instruction::Shl, Op0_t, Op1_t > m_Shl(const Op0_t &Op0, const Op1_t &Op1)
Recipe_match< std::tuple< Op0_t, Op1_t >, 0, false, VPVectorEndPointerRecipe > VectorEndPointerRecipe_match
match_bind< VPInstruction > m_VPInstruction(VPInstruction *&V)
Match a VPInstruction, capturing if we match.
match_combine_or< AllRecipe_match< Instruction::ZExt, Op0_t >, Op0_t > m_ZExtOrSelf(const Op0_t &Op0)
VPInstruction_match< VPInstruction::FirstActiveLane, Op0_t > m_FirstActiveLane(const Op0_t &Op0)
Argument_match< Opnd_t > m_Argument(const Opnd_t &Op)
Match a call argument.
AllRecipe_match< Instruction::FNeg, Op0_t > m_FNeg(const Op0_t &Op0)
AllRecipe_match< Instruction::UDiv, Op0_t, Op1_t > m_UDiv(const Op0_t &Op0, const Op1_t &Op1)
auto m_Not(const Op0_t &Op0)
auto m_DerivedIV(const Op0_t &Op0, const Op1_t &Op1, const Op2_t &Op2)
auto m_c_LogicalAnd(const Op0_t &Op0, const Op1_t &Op1)
int_pred_ty< is_one > m_One()
Match an integer 1 or a vector with all elements equal to 1.
Recipe_match< std::tuple< OpTys... >, Opcode, false, VPWidenRecipe, VPReplicateRecipe, VPWidenCastRecipe, VPInstruction > AllRecipe_match
VPInstruction_match< VPInstruction::BranchOnCond > m_BranchOnCond()
match_bind< VPReductionPHIRecipe > m_ReductionPhi(VPReductionPHIRecipe *&V)
static VPRecipeBase * findUserOf(VPValue *V, const MatchT &P)
If V is used by a recipe matching pattern P, return it.
auto m_ScalarIVSteps(const Op0_t &Op0, const Op1_t &Op1, const Op2_t &Op2)
VPInstruction_match< VPInstruction::ExtractLane, Op0_t, Op1_t > m_ExtractLane(const Op0_t &Op0, const Op1_t &Op1)
VPInstruction_match< VPInstruction::BuildStructVector > m_BuildStructVector()
BuildStructVector matches only its opcode, w/o matching its operands as the number of operands is not...
bind_apint m_APInt(const APInt *&C)
auto m_AnyNeg(const Op0_t &Op0)
VPInstruction_match< VPInstruction::Reverse, Op0_t > m_Reverse(const Op0_t &Op0)
Intrinsic::ID getIntrinsicID(const Ty *R)
Return the intrinsic ID underlying a call.
decltype(auto) dyn_cast(const From &Val)
dyn_cast<X> - Return the argument parameter cast to the specified type.
auto cast_or_null(const Y &Val)
constexpr auto bind_back(FnT &&Fn, BindArgsT &&...BindArgs)
C++23 bind_back.
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
ArrayRef(const T &OneElt) -> ArrayRef< T >
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.
auto find_if(R &&Range, UnaryPredicate P)
Provide wrappers to std::find_if which take ranges instead of having to pass begin/end explicitly.
Implement std::hash so that hash_code can be used in STL containers.
Matcher to bind the captured value.
Matching and combinator leaf case.
Matching or combinator leaf case.
Matcher for a specific value, but stores a reference to the value, not the value itself.
Intrinsic matches are combinations of ID matchers, and argument matchers.
A VPValue representing a live-in from the input IR or a constant.
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
Match a GEP recipe (VPWidenGEPRecipe, VPInstruction, or VPReplicateRecipe) and bind the source elemen...
GetElementPtr_match(Type *&SourceElementType, ArrayRef< VPValue * > &Operands)
ArrayRef< VPValue * > & Operands
Type *& SourceElementType
IntrinsicID_match(Intrinsic::ID IntrID)
bool match(OpTy *V) const
Load_match(Addr_t Addr, Mask_t Mask)
bool match(const OpTy *V) const
OneUse_match(const SubPattern_t &SP)
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
bool match(const VPRecipeBase *V) const
Store_match(Addr_t Addr, Val_t Val, Mask_t Mask)
bool match(const OpTy *V) const
bind_apint(const APInt *&Res)
bool match(const VPValue *VPV) const
bool match(const VPValue *VPV) const
bind_const_int(uint64_t &Res)
Match the canonical induction variable (IV) of any loop region.
bool match(const ArgTy *V) const
Match a canonical VPWidenIntOrFpInductionRecipe optionally capturing it.
bool match(ArgTy *V) const
canonical_widen_iv_match()=default
VPWidenIntOrFpInductionRecipe ** Capture
canonical_widen_iv_match(VPWidenIntOrFpInductionRecipe *&V)
Match an integer constant if Pred::isValue returns true for the APInt.
bool match(const VPValue *VPV) const
bool isValue(const APInt &C) const
bool isValue(const APInt &C) const
Match a specified signed or unsigned integer value.
bool isValue(const APInt &C) const
is_specific_int(APInt Val, bool IsSigned=false)
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
bool match(const VPValue *V) const
Match a specified VPValue.
specificval_ty(const VPValue *V)
bool match(const VPValue *VPV) const