12#include "llvm/IR/IntrinsicsSPIRV.h"
33 if (
MI.getOpcode() != TargetOpcode::G_INTRINSIC ||
40 if (SubInstr->
getOpcode() != TargetOpcode::G_FSUB)
48 Register SubDestReg =
MI.getOperand(2).getReg();
52 Register ResultReg =
MI.getOperand(0).getReg();
55 Builder.buildIntrinsic(Intrinsic::spv_distance, ResultReg)
99 if (DotInstr->
getOpcode() != TargetOpcode::G_INTRINSIC ||
105 !
MRI.getType(DotOperand1).isScalar() ||
106 !
MRI.getType(DotOperand2).isScalar())
115 auto AreNegatedConstantsOrSplats = [&](
Register TrueReg,
Register FalseReg) {
116 std::optional<FPValueAndVReg> TrueVal, FalseVal;
120 APFloat TrueValNegated = TrueVal->Value;
127 std::optional<FPValueAndVReg> MulConstant;
130 if (TrueInstr->
getOpcode() == TargetOpcode::G_BUILD_VECTOR &&
131 FalseInstr->
getOpcode() == TargetOpcode::G_BUILD_VECTOR &&
149 if (!MulConstant || !MulConstant->Value.isExactlyValue(-1.0))
151 }
else if (!AreNegatedConstantsOrSplats(TrueReg, FalseReg))
168 if (DotInstr->
getOpcode() == TargetOpcode::G_FMUL) {
176 Register FalseReg =
MI.getOperand(3).getReg();
178 if (TrueInstr->
getOpcode() == TargetOpcode::G_FNEG ||
179 TrueInstr->
getOpcode() == TargetOpcode::G_FMUL)
183 Register ResultReg =
MI.getOperand(0).getReg();
185 Builder.buildIntrinsic(Intrinsic::spv_faceforward, ResultReg)
188 .addUse(DotOperand2);
192 auto RemoveAllUses = [&](
Register Reg) {
194 for (
auto &
UseMI :
MRI.use_instructions(Reg))
198 for (
auto *MIToErase : UsesToErase)
199 MIToErase->eraseFromParent();
202 RemoveAllUses(CondReg);
205 RemoveAllUses(DotReg);
208 RemoveAllUses(FalseReg);
MachineInstrBuilder & UseMI
static GCRegistry::Add< OcamlGC > B("ocaml", "ocaml 3.10-compatible GC")
Declares convenience wrapper classes for interpreting MachineInstr instances as specific generic oper...
Contains matchers for matching SSA Machine Instructions.
Predicate
This enumeration lists the possible predicates for CmpInst subclasses.
@ FCMP_OLT
0 1 0 0 True if ordered and less than
@ FCMP_OGT
0 0 1 0 True if ordered and greater than
@ FCMP_ULT
1 1 0 0 True if unordered or less than
@ FCMP_UGT
1 0 1 0 True if unordered or greater than
MachineRegisterInfo & MRI
MachineDominatorTree * MDT
GISelChangeObserver & Observer
MachineIRBuilder & Builder
ConstantFP - Floating Point Values [float, double].
bool isZero() const
Return true if the value is positive or negative zero.
Abstract class that contains various methods for clients to notify about changes.
DominatorTree Class - Concrete subclass of DominatorTreeBase that is used to compute a normal dominat...
Helper class to build MachineInstr.
Representation of each machine instruction.
unsigned getOpcode() const
Returns the opcode of this MachineInstr.
unsigned getNumOperands() const
Retuns the total number of operands.
LLVM_ABI void eraseFromParent()
Unlink 'this' from the containing basic block and delete it.
const MachineOperand & getOperand(unsigned i) const
Register getReg() const
getReg - Returns the register number.
Wrapper class representing virtual and physical registers.
bool matchSelectToFaceForward(MachineInstr &MI) const
This match is part of a combine that rewrites select(fcmp(dot(I, Ng), 0), N, -N) to faceforward(N,...
CombinerHelper(GISelChangeObserver &Observer, MachineIRBuilder &B, bool IsPreLegalize, GISelValueTracking *VT=nullptr, MachineDominatorTree *MDT=nullptr, const LegalizerInfo *LI=nullptr)
void applySPIRVFaceForward(MachineInstr &MI) const
SPIRVCombinerHelper(GISelChangeObserver &Observer, MachineIRBuilder &B, bool IsPreLegalize, GISelValueTracking *VT, MachineDominatorTree *MDT, const LegalizerInfo *LI, const SPIRVSubtarget &STI)
const SPIRVSubtarget & STI
void applySPIRVDistance(MachineInstr &MI) const
bool matchLengthToDistance(MachineInstr &MI) const
This match is part of a combine that rewrites length(X - Y) to distance(X, Y) (f32 (g_intrinsic lengt...
void invalidateMachineInstr(MachineInstr *MI)
void push_back(const T &Elt)
This is a 'vector' (really, a variable-sized array), optimized for the case when the array is small.
operand_type_match m_Reg()
operand_type_match m_Pred()
TernaryOp_match< Src0Ty, Src1Ty, Src2Ty, TargetOpcode::G_SELECT > m_GISelect(const Src0Ty &Src0, const Src1Ty &Src1, const Src2Ty &Src2)
bool mi_match(Reg R, const MachineRegisterInfo &MRI, Pattern &&P)
SpecificRegisterMatch m_SpecificReg(Register RequestedReg)
Matches a register only if it is equal to RequestedReg.
UnaryOp_match< SrcTy, TargetOpcode::G_FNEG > m_GFNeg(const SrcTy &Src)
GFCstAndRegMatch m_GFCst(std::optional< FPValueAndVReg > &FPValReg)
GFCstOrSplatGFCstMatch m_GFCstOrSplat(std::optional< FPValueAndVReg > &FPValReg)
BinaryOp_match< LHS, RHS, TargetOpcode::G_FMUL, true > m_GFMul(const LHS &L, const RHS &R)
CompareOp_match< Pred, LHS, RHS, TargetOpcode::G_FCMP > m_GFCmp(const Pred &P, const LHS &L, const RHS &R)
This is an optimization pass for GlobalISel generic memory operations.
decltype(auto) cast(const From &Val)
cast<X> - Return the argument parameter cast to the specified type.
void swap(llvm::BitVector &LHS, llvm::BitVector &RHS)
Implement std::swap in terms of BitVector swap.