23#include "llvm/IR/IntrinsicsSPIRV.h"
26#define DEBUG_TYPE "spirv-prelegalizer"
40void SPIRVPreLegalizer::getAnalysisUsage(
AnalysisUsage &AU)
const {
48 MI->eraseFromParent();
66 MI.getOperand(3).getMetadata()->getOperand(0))
74 RegsAlreadyAddedToDT[&
MI] =
Reg;
81 BuildVec->getOpcode() == TargetOpcode::G_BUILD_VECTOR);
82 GR->
add(Const, BuildVec);
83 for (
unsigned i = 0; i < ConstVec->getNumElements(); ++i) {
86 Constant *ElemConst = ConstVec->getElementAsConstant(i);
90 MRI.
getVRegDef(BuildVec->getOperand(1 + i).getReg()));
92 BuildVec->getOperand(1 + i).setReg(ElemReg);
95 if (Const->getType()->isTargetExtTy()) {
99 GR->
add(Const, SrcMI);
100 if (SrcMI && (SrcMI->
getOpcode() == TargetOpcode::G_CONSTANT ||
101 SrcMI->
getOpcode() == TargetOpcode::G_IMPLICIT_DEF))
102 TargetExtConstTypes[SrcMI] = Const->getType();
103 if (Const->isNullValue()) {
107 Const->getType(), MIB, SPIRV::AccessQualifier::ReadWrite,
109 assert(SrcMI &&
"Expected source instruction to be valid");
116 RegsAlreadyAddedToDT[&
MI] =
Reg;
119 assert(
MI.getOperand(2).isReg() &&
"Reg operand is expected");
121 if (SrcMI &&
isSpvIntrinsic(*SrcMI, Intrinsic::spv_const_composite))
129 auto It = RegsAlreadyAddedToDT.
find(
MI);
130 if (It != RegsAlreadyAddedToDT.
end())
150 const MDNode *MD =
MI.getOperand(2).getMetadata();
172 UseMI->getOperand(1).getReg() ==
Reg)
182 assert(ResType && OpType &&
"Operand types are expected");
188 if (ResType == OpType)
219 "Expected destination SPIR-V type to have been assigned already.");
222 "Expected source SPIR-V type to have been assigned already.");
223 if (DstType == SrcType) {
231 if (
MI.getOpcode() != TargetOpcode::G_BITCAST)
236 MI.getOperand(1).getReg());
262 ? SPIRV::StorageClass::CodeSectionINTEL
305 assert(
MI &&
"Machine instr is expected");
306 if (
MI->getOperand(0).isReg()) {
310 switch (
MI->getOpcode()) {
311 case TargetOpcode::G_FCONSTANT:
312 case TargetOpcode::G_CONSTANT: {
314 Type *Ty =
MI->getOperand(1).getCImm()->getType();
316 Ty, MIB, SPIRV::AccessQualifier::ReadWrite,
true);
319 case TargetOpcode::G_GLOBAL_VALUE: {
324 Global->getType()->getAddressSpace());
326 Ty, MIB, SPIRV::AccessQualifier::ReadWrite,
true);
329 case TargetOpcode::G_ANYEXT:
330 case TargetOpcode::G_SEXT:
331 case TargetOpcode::G_ZEXT: {
332 if (
MI->getOperand(1).isReg()) {
338 unsigned ExpectedBW =
350 case TargetOpcode::G_PTRTOINT:
354 case TargetOpcode::G_TRUNC:
355 case TargetOpcode::G_ADDRSPACE_CAST:
356 case TargetOpcode::G_PTR_ADD:
357 case TargetOpcode::COPY: {
370 if (SpvType->
getOpcode() == SPIRV::OpTypePointer &&
371 RegType.isPointer() &&
373 RegType.getAddressSpace()) {
383 : &SPIRV::iIDRegClass);
400 if (!RegType.isScalar())
404 if (NewWidth != CurrentWidth)
412 if (NewWidth != CurrentWidth) {
422 Def->getNextNode() ? Def->getNextNode()->getIterator() :
MBB.end();
424 while (DefIt !=
MBB.end() &&
425 (DefIt->isPHI() || DefIt->isDebugOrPseudoInstr()))
426 DefIt = std::next(DefIt);
434 assert((Ty || SpvType) &&
"Either LLVM or SPIRV type is expected.");
439 SPIRV::AccessQualifier::ReadWrite,
true);
451 for (
auto &
Op :
MI.operands()) {
452 if (!
Op.isReg() ||
Op.isDef())
456 if (!SpvType && KnownResType) {
457 SpvType = KnownResType;
481 bool IsExtendedInts =
483 SPIRV::Extension::SPV_ALTERA_arbitrary_precision_integers) ||
484 ST->canUseExtension(SPIRV::Extension::SPV_KHR_bit_instructions) ||
485 ST->canUseExtension(SPIRV::Extension::SPV_INTEL_int4);
487 if (!IsExtendedInts) {
508 unsigned MIOp =
MI.getOpcode();
509 if (MIOp != TargetOpcode::G_TRUNC)
521 (SrcTy.isScalar() || SrcTy.isVector()) &&
522 "Expected scalar or vector G_TRUNC types");
524 "Expected matching scalar/vector G_TRUNC types");
527 "Expected equal vector element counts");
530 unsigned OriginalSrcWidth = SrcTy.getScalarSizeInBits();
539 if (OriginalDstWidth == NewDstWidth) {
556 MIB.
buildAnd(MaskedReg, SrcReg, MaskReg);
558 if (NewSrcWidth == NewDstWidth) {
562 MI.getOperand(1).setReg(MaskedReg);
567 MI->eraseFromParent();
574 bool ReachedBegin =
false;
575 for (
auto MII = std::prev(
MBB->end()), Begin =
MBB->begin();
578 unsigned MIOp =
MI.getOpcode();
580 if (!IsExtendedInts) {
582 for (
auto &MOP :
MI.operands()) {
585 else if (MOP.isCImm())
605 assert(Def &&
"Expecting an instruction that defines the register");
607 if (Def->getOpcode() != TargetOpcode::G_GLOBAL_VALUE)
615 assert(Def &&
"Expecting an instruction that defines the register");
617 if (Def->getOpcode() != TargetOpcode::G_GLOBAL_VALUE)
620 }
else if (MIOp == TargetOpcode::FAKE_USE &&
MI.getNumOperands() > 0) {
625 for (
unsigned I = 1,
E =
MI.getNumOperands();
I !=
E && Def; ++
I)
639 }
else if (MIOp == TargetOpcode::G_CONSTANT ||
640 MIOp == TargetOpcode::G_FCONSTANT ||
641 MIOp == TargetOpcode::G_BUILD_VECTOR) {
647 if (MIOp == TargetOpcode::G_CONSTANT) {
648 auto TargetExtIt = TargetExtConstTypes.
find(&
MI);
649 Ty = TargetExtIt == TargetExtConstTypes.
end()
650 ?
MI.getOperand(1).getCImm()->getType()
651 : TargetExtIt->second;
664 }
else if (PrimaryReg !=
Reg &&
668 if (!RCReg || RCPrimary == RCReg) {
669 RegsAlreadyAddedToDT[&
MI] = PrimaryReg;
671 NeedAssignType =
false;
674 }
else if (MIOp == TargetOpcode::G_FCONSTANT) {
675 Ty =
MI.getOperand(1).getFPImm()->getType();
677 assert(MIOp == TargetOpcode::G_BUILD_VECTOR);
678 Type *ElemTy =
nullptr;
682 if (ElemMI->
getOpcode() == TargetOpcode::G_CONSTANT) {
684 }
else if (ElemMI->
getOpcode() == TargetOpcode::G_FCONSTANT) {
693 ElemTy,
MI.getNumExplicitOperands() -
MI.getNumExplicitDefs(),
696 NeedAssignType =
false;
700 }
else if (MIOp == TargetOpcode::G_GLOBAL_VALUE) {
711 auto It = RegsAlreadyAddedToDT.
find(
MI);
712 if (It != RegsAlreadyAddedToDT.
end())
721 switch (
MI.getOpcode()) {
722 case TargetOpcode::G_TRUNC:
723 case TargetOpcode::G_ANYEXT:
724 case TargetOpcode::G_SEXT:
725 case TargetOpcode::G_ZEXT:
726 case TargetOpcode::G_PTRTOINT:
727 case TargetOpcode::COPY:
728 case TargetOpcode::G_ADDRSPACE_CAST:
752 for (
unsigned Idx = StartOp, MISz =
MI->getNumOperands(); Idx != MISz;
757 if (Idx == AsmDescOp && MO.
isImm()) {
760 AsmDescOp += 1 +
F.getNumOperandRegisters();
780 for (
unsigned i = 0, Sz = ToProcess.
size(); i + 1 < Sz; i += 2) {
781 MachineInstr *I1 = ToProcess[i], *I2 = ToProcess[i + 1];
788 MRI.
setRegClass(AsmTargetReg, &SPIRV::iIDRegClass);
792 GR->
add(AsmTargetMIB.getInstr(), AsmTargetMIB);
796 const MDNode *IAMD = I1->getOperand(1).getMetadata();
799 for (
const auto &ArgTy : FTy->params())
801 ArgTy, MIRBuilder, SPIRV::AccessQualifier::ReadWrite,
true));
804 SPIRV::AccessQualifier::ReadWrite,
true);
806 FTy, RetType, ArgTypes, MIRBuilder);
811 auto AsmMIB = MIRBuilder.
buildInstr(SPIRV::OpAsmINTEL)
823 GR->
add(AsmMIB.getInstr(), AsmMIB);
830 .
addImm(
static_cast<uint32_t>(SPIRV::Decoration::SideEffectsINTEL));
838 SPIRV::AccessQualifier::ReadWrite,
true);
842 auto AsmCall = MIRBuilder.
buildInstr(SPIRV::OpAsmCallINTEL)
846 for (
unsigned IntrIdx = 3; IntrIdx < I1->getNumOperands(); ++IntrIdx)
847 AsmCall.
addUse(I1->getOperand(IntrIdx).getReg());
856 if (CopyMI.
getOpcode() == TargetOpcode::COPY) {
860 if (TruncMI.
getOpcode() == TargetOpcode::G_TRUNC) {
882 MI.getOpcode() == TargetOpcode::INLINEASM)
886 if (ToProcess.
size() == 0)
889 if (!ST.canUseExtension(SPIRV::Extension::SPV_INTEL_inline_assembly))
891 "following SPIR-V extension: SPV_INTEL_inline_assembly",
910 MI.getOperand(2).getMetadata(), ST);
912 Intrinsic::spv_assign_fpmaxerror_decoration)) {
914 MI.getOperand(2).getMetadata()->getOperand(0));
918 SPIRV::Decoration::FPMaxErrorDecorationINTEL,
922 MI.getOperand(2).getImm(),
923 MI.getOperand(3).getMetadata());
949 for (
unsigned i = 3; i <
MI.getNumOperands(); i += 2) {
959 while (
MI.getNumOperands() > 0)
961 for (
auto &MO : NewOperands)
975 MI.getOpcode() == TargetOpcode::G_BRINDIRECT)
991 BB2MBB[
MBB.getBasicBlock()] = &
MBB;
1013 for (
unsigned i = 0; i <
MI->getNumOperands(); ++i) {
1015 if (!
MI->getOperand(i).isReg()) {
1023 if (!BuildMBB || BuildMBB->
getOpcode() != TargetOpcode::G_BLOCK_ADDR) {
1028 assert(BuildMBB && BuildMBB->
getOpcode() == TargetOpcode::G_BLOCK_ADDR &&
1033 auto It = BB2MBB.
find(BB);
1034 if (It == BB2MBB.
end())
1036 "in a switch statement");
1040 ClearAddressTaken.
insert(ReferencedBlock);
1041 ToEraseMI.
insert(BuildMBB);
1046 while (
MI->getNumOperands() > 0)
1047 MI->removeOperand(0);
1048 for (
auto &MO : NewOps)
1054 Next =
MI->getNextNode();
1056 if (
Next &&
Next->getOpcode() == TargetOpcode::G_BRINDIRECT)
1065 Succ->setAddressTakenIRBlock(
nullptr);
1076 if (BlockAddrI->getOpcode() == TargetOpcode::G_BLOCK_ADDR) {
1078 BlockAddrI->getOperand(1).getBlockAddress());
1089 return MBB.getNextNode() !=
nullptr;
1093 if (!
MBB.canFallThrough())
1120bool SPIRVPreLegalizer::runOnMachineFunction(MachineFunction &MF) {
1123 SPIRVGlobalRegistry *GR =
ST.getSPIRVGlobalRegistry();
1124 GR->setCurrentFunc(MF);
1125 MachineIRBuilder MIB(MF);
1127 DenseMap<MachineInstr *, Type *> TargetExtConstTypes;
1150char SPIRVPreLegalizer::
ID = 0;
1153 return new SPIRVPreLegalizer();
MachineInstrBuilder & UseMI
assert(UImm &&(UImm !=~static_cast< T >(0)) &&"Invalid immediate!")
This file contains the simple types necessary to represent the attributes associated with functions a...
static GCRegistry::Add< CoreCLRGC > E("coreclr", "CoreCLR-compatible GC")
Provides analysis for continuously CSEing during GISel passes.
This file contains the declarations for the subclasses of Constant, which represent the different fla...
Provides analysis for querying information about KnownBits during GISel passes.
const AbstractManglingParser< Derived, Alloc >::OperatorInfo AbstractManglingParser< Derived, Alloc >::Ops[]
Promote Memory to Register
#define INITIALIZE_PASS(passName, arg, name, cfg, analysis)
This file builds on the ADT/GraphTraits.h file to build a generic graph post order iterator.
static Register collectInlineAsmInstrOperands(MachineInstr *MI, SmallVector< unsigned, 4 > *Ops=nullptr)
static void insertInlineAsm(MachineFunction &MF, SPIRVGlobalRegistry *GR, const SPIRVSubtarget &ST, MachineIRBuilder MIRBuilder)
static void cleanupHelperInstructions(MachineFunction &MF, SPIRVGlobalRegistry *GR)
static void insertInlineAsmProcess(MachineFunction &MF, SPIRVGlobalRegistry *GR, const SPIRVSubtarget &ST, MachineIRBuilder MIRBuilder, const SmallVector< MachineInstr * > &ToProcess)
static void removeImplicitFallthroughs(MachineFunction &MF, MachineIRBuilder MIB)
static unsigned widenBitWidthToNextPow2(unsigned BitWidth)
static void setInsertPtAfterDef(MachineIRBuilder &MIB, MachineInstr *Def)
static bool isImplicitFallthrough(MachineBasicBlock &MBB)
static void insertSpirvDecorations(MachineFunction &MF, SPIRVGlobalRegistry *GR, MachineIRBuilder MIB)
static void insertBitcasts(MachineFunction &MF, SPIRVGlobalRegistry *GR, MachineIRBuilder MIB)
static void processInstrsWithTypeFolding(MachineFunction &MF, SPIRVGlobalRegistry *GR, MachineIRBuilder MIB)
static void processSwitchesConstants(MachineFunction &MF, SPIRVGlobalRegistry *GR, MachineIRBuilder MIB)
static void lowerBitcasts(MachineFunction &MF, SPIRVGlobalRegistry *GR, MachineIRBuilder MIB)
static MachineInstr * findAssignTypeInstr(Register Reg, MachineRegisterInfo *MRI)
static void widenCImmType(MachineOperand &MOP)
static void buildOpBitcast(SPIRVGlobalRegistry *GR, MachineIRBuilder &MIB, Register ResVReg, Register OpReg)
static void processBlockAddr(MachineFunction &MF, SPIRVGlobalRegistry *GR, MachineIRBuilder MIB)
static void widenScalarType(Register Reg, MachineRegisterInfo &MRI)
static void foldConstantsIntoIntrinsics(MachineFunction &MF, SPIRVGlobalRegistry *GR, MachineIRBuilder MIB)
static void addConstantsToTrack(MachineFunction &MF, SPIRVGlobalRegistry *GR, const SPIRVSubtarget &STI, DenseMap< MachineInstr *, Type * > &TargetExtConstTypes)
static SPIRVTypeInst propagateSPIRVType(MachineInstr *MI, SPIRVGlobalRegistry *GR, MachineRegisterInfo &MRI, MachineIRBuilder &MIB)
static void invalidateAndEraseMI(SPIRVGlobalRegistry *GR, MachineInstr *MI)
static void generateAssignInstrs(MachineFunction &MF, SPIRVGlobalRegistry *GR, MachineIRBuilder MIB, DenseMap< MachineInstr *, Type * > &TargetExtConstTypes)
APInt bitcastToAPInt() const
Class for arbitrary precision integers.
uint64_t getZExtValue() const
Get zero extended value.
LLVM_ABI APInt zextOrTrunc(unsigned width) const
Zero extend or truncate to width.
static APInt getLowBitsSet(unsigned numBits, unsigned loBitsSet)
Constructs an APInt value that has the bottom loBitsSet bits set.
Represent the analysis usage information of a pass.
AnalysisUsage & addPreserved()
Add the specified Pass class to the set of analyses preserved by this pass.
LLVM Basic Block Representation.
The address of a basic block.
BasicBlock * getBasicBlock() const
static LLVM_ABI Constant * getIntToPtr(Constant *C, Type *Ty, bool OnlyIfReduced=false)
ConstantFP - Floating Point Values [float, double].
const APFloat & getValueAPF() const
This is the shared class of boolean and integer constants.
unsigned getBitWidth() const
getBitWidth - Return the scalar bitwidth of this constant.
const APInt & getValue() const
Return the constant as an APInt value reference.
This is an important base class in LLVM.
LLVM_ABI void destroyConstant()
Called if some element of this constant is no longer valid.
iterator find(const_arg_type_t< KeyT > Val)
LLVMContext & getContext() const
getContext - Return a reference to the LLVMContext associated with this function.
constexpr unsigned getScalarSizeInBits() const
constexpr bool isScalar() const
static constexpr LLT scalar(unsigned SizeInBits)
Get a low-level scalar or aggregate "bag of bits".
constexpr bool isValid() const
constexpr uint16_t getNumElements() const
Returns the number of elements in a vector LLT.
constexpr bool isVector() const
constexpr ElementCount getElementCount() const
LLT changeElementSize(unsigned NewEltSize) const
If this type is a vector, return a vector with the same number of elements but the new element size.
const MDOperand & getOperand(unsigned I) const
LLVM_ABI iterator getFirstNonPHI()
Returns a pointer to the first instruction in this block that is not a PHINode instruction.
MachineInstrBundleIterator< MachineInstr, true > reverse_iterator
MachineInstrBundleIterator< MachineInstr > iterator
MachineFunctionPass - This class adapts the FunctionPass interface to allow convenient creation of pa...
void getAnalysisUsage(AnalysisUsage &AU) const override
getAnalysisUsage - Subclasses that override getAnalysisUsage must call this.
const TargetSubtargetInfo & getSubtarget() const
getSubtarget - Return the subtarget for which this machine code is being compiled.
MachineRegisterInfo & getRegInfo()
getRegInfo - Return information about the registers currently in use.
Function & getFunction()
Return the LLVM function that this machine code represents.
const MachineBasicBlock & front() const
Helper class to build MachineInstr.
MachineInstrBuilder buildBr(MachineBasicBlock &Dest)
Build and insert G_BR Dest.
void setInsertPt(MachineBasicBlock &MBB, MachineBasicBlock::iterator II)
Set the insertion point before the specified position.
MachineInstrBuilder buildAnd(const DstOp &Dst, const SrcOp &Src0, const SrcOp &Src1)
Build and insert Res = G_AND Op0, Op1.
MachineInstrBuilder buildInstr(unsigned Opcode)
Build and insert <empty> = Opcode <empty>.
MachineInstrBuilder buildBuildVectorConstant(const DstOp &Res, ArrayRef< APInt > Ops)
Build and insert Res = G_BUILD_VECTOR Op0, ... where each OpN is built with G_CONSTANT.
MachineFunction & getMF()
Getter for the function we currently build.
MachineInstrBuilder buildBitcast(const DstOp &Dst, const SrcOp &Src)
Build and insert Dst = G_BITCAST Src.
MachineRegisterInfo * getMRI()
Getter for MRI.
MachineInstrBuilder buildCopy(const DstOp &Res, const SrcOp &Op)
Build and insert Res = COPY Op.
virtual MachineInstrBuilder buildConstant(const DstOp &Res, const ConstantInt &Val)
Build and insert Res = G_CONSTANT Val.
const MachineInstrBuilder & addUse(Register RegNo, RegState Flags={}, unsigned SubReg=0) const
Add a virtual register use operand.
const MachineInstrBuilder & addImm(int64_t Val) const
Add a new immediate operand.
const MachineInstrBuilder & addDef(Register RegNo, RegState Flags={}, unsigned SubReg=0) const
Add a virtual register definition operand.
Representation of each machine instruction.
mop_range defs()
Returns all explicit operands that are register definitions.
unsigned getOpcode() const
Returns the opcode of this MachineInstr.
LLVM_ABI void addOperand(MachineFunction &MF, const MachineOperand &Op)
Add the specified operand to the instruction.
LLVM_ABI void setDesc(const MCInstrDesc &TID)
Replace the instruction descriptor (thus opcode) of the current instruction with a new one.
const MachineOperand & getOperand(unsigned i) const
MachineOperand class - Representation of each machine instruction operand.
const ConstantInt * getCImm() const
bool isReg() const
isReg - Tests if this is a MO_Register operand.
const MDNode * getMetadata() const
static MachineOperand CreateCImm(const ConstantInt *CI)
bool isImm() const
isImm - Tests if this is a MO_Immediate operand.
bool isMetadata() const
isMetadata - Tests if this is a MO_Metadata operand.
const BlockAddress * getBlockAddress() const
void setCImm(const ConstantInt *CI)
bool isBlockAddress() const
isBlockAddress - Tests if this is a MO_BlockAddress operand.
Register getReg() const
getReg - Returns the register number.
const ConstantFP * getFPImm() const
static MachineOperand CreateReg(Register Reg, bool isDef, bool isImp=false, bool isKill=false, bool isDead=false, bool isUndef=false, bool isEarlyClobber=false, unsigned SubReg=0, bool isDebug=false, bool isInternalRead=false, bool isRenamable=false)
static MachineOperand CreateMBB(MachineBasicBlock *MBB, unsigned TargetFlags=0)
MachineRegisterInfo - Keep track of information for virtual and physical registers,...
defusechain_instr_iterator< true, false, false, true > use_instr_iterator
use_instr_iterator/use_instr_begin/use_instr_end - Walk all uses of the specified register,...
LLVM_ABI MachineInstr * getVRegDef(Register Reg) const
getVRegDef - Return the machine instr that defines the specified virtual register or null if none is ...
use_instr_iterator use_instr_begin(Register RegNo) const
LLT getType(Register Reg) const
Get the low-level type of Reg or LLT{} if Reg is not a generic (target independent) virtual register.
bool hasOneUse(Register RegNo) const
hasOneUse - Return true if there is exactly one instruction using the specified register.
static use_instr_iterator use_instr_end()
LLVM_ABI void setType(Register VReg, LLT Ty)
Set the low-level type of VReg to Ty.
LLVM_ABI void setRegClass(Register Reg, const TargetRegisterClass *RC)
setRegClass - Set the register class of the specified virtual register.
LLVM_ABI Register createGenericVirtualRegister(LLT Ty, StringRef Name="")
Create and return a new generic virtual register with low-level type Ty.
const TargetRegisterClass * getRegClassOrNull(Register Reg) const
Return the register class of Reg, or null if Reg has not been assigned a register class yet.
LLVM_ABI void replaceRegWith(Register FromReg, Register ToReg)
replaceRegWith - Replace all instances of FromReg with ToReg in the machine function.
Wrapper class representing virtual and physical registers.
constexpr bool isValid() const
void assignSPIRVTypeToVReg(SPIRVTypeInst Type, Register VReg, const MachineFunction &MF)
SPIRVTypeInst getOrCreateOpTypeFunctionWithArgs(const Type *Ty, SPIRVTypeInst RetType, const SmallVectorImpl< SPIRVTypeInst > &ArgTypes, MachineIRBuilder &MIRBuilder)
const TargetRegisterClass * getRegClass(SPIRVTypeInst SpvType) const
unsigned getScalarOrVectorBitWidth(SPIRVTypeInst Type) const
SPIRVTypeInst getOrCreateSPIRVIntegerType(unsigned BitWidth, MachineIRBuilder &MIRBuilder)
SPIRVTypeInst getOrCreateSPIRVVectorType(SPIRVTypeInst BaseType, unsigned NumElements, MachineIRBuilder &MIRBuilder, bool EmitIR)
unsigned getScalarOrVectorComponentCount(Register VReg) const
const Type * getTypeForSPIRVType(SPIRVTypeInst Ty) const
bool isBitcastCompatible(SPIRVTypeInst Type1, SPIRVTypeInst Type2) const
LLT getRegType(SPIRVTypeInst SpvType) const
void invalidateMachineInstr(MachineInstr *MI)
SPIRVTypeInst getOrCreateSPIRVPointerType(const Type *BaseType, MachineIRBuilder &MIRBuilder, SPIRV::StorageClass::StorageClass SC)
Register getSPIRVTypeID(SPIRVTypeInst SpirvType) const
SPIRVTypeInst changePointerStorageClass(SPIRVTypeInst PtrType, SPIRV::StorageClass::StorageClass SC, MachineInstr &I)
void addGlobalObject(const Value *V, const MachineFunction *MF, Register R)
SPIRVTypeInst getOrCreateSPIRVType(const Type *Type, MachineInstr &I, SPIRV::AccessQualifier::AccessQualifier AQ, bool EmitIR)
SPIRVTypeInst getSPIRVTypeForVReg(Register VReg, const MachineFunction *MF=nullptr) const
Type * getDeducedGlobalValueType(const GlobalValue *Global)
void addValueAttrs(MachineInstr *Key, std::pair< Type *, std::string > Val)
void buildMemAliasingOpDecorate(Register Reg, MachineIRBuilder &MIRBuilder, uint32_t Dec, const MDNode *GVarMD)
SPIRV::StorageClass::StorageClass getPointerStorageClass(Register VReg) const
bool add(SPIRV::IRHandle Handle, const MachineInstr *MI)
Register find(SPIRV::IRHandle Handle, const MachineFunction *MF)
const SPIRVInstrInfo * getInstrInfo() const override
std::pair< iterator, bool > insert(PtrType Ptr)
Inserts Ptr if and only if there is no element in the container equal to Ptr.
SmallPtrSet - This class implements a set which is optimized for holding SmallSize or less elements.
void push_back(const T &Elt)
This is a 'vector' (really, a variable-sized array), optimized for the case when the array is small.
Represent a constant reference to a string, i.e.
The instances of the Type class are immutable: once they are created, they are never changed.
static LLVM_ABI IntegerType * getInt32Ty(LLVMContext &C)
static LLVM_ABI Type * getVoidTy(LLVMContext &C)
LLVMContext & getContext() const
Return the LLVMContext in which this type was uniqued.
static LLVM_ABI TypedPointerType * get(Type *ElementType, unsigned AddressSpace)
This constructs a pointer to an object of the specified type in a numbered address space.
Type * getType() const
All values are typed, get the type of this value.
LLVM_ABI void replaceAllUsesWith(Value *V)
Change all uses of this to point to a new Value.
static LLVM_ABI VectorType * get(Type *ElementType, ElementCount EC)
This static method is the primary way to construct an VectorType.
unsigned ID
LLVM IR allows to use arbitrary numbers as calling convention identifiers.
std::enable_if_t< detail::IsValidPointer< X, Y >::value, X * > dyn_extract(Y &&MD)
Extract a Value from Metadata, if any.
This is an optimization pass for GlobalISel generic memory operations.
void buildOpName(Register Target, const StringRef &Name, MachineIRBuilder &MIRBuilder)
StringMapEntry< Value * > ValueName
bool isTypeFoldingSupported(unsigned Opcode)
decltype(auto) dyn_cast(const From &Val)
dyn_cast<X> - Return the argument parameter cast to the specified type.
FunctionPass * createSPIRVPreLegalizerPass()
void updateRegType(Register Reg, Type *Ty, SPIRVTypeInst SpirvTy, SPIRVGlobalRegistry *GR, MachineIRBuilder &MIB, MachineRegisterInfo &MRI)
Helper external function for assigning a SPIRV type to a register, ensuring the register class and ty...
void buildOpDecorate(Register Reg, MachineIRBuilder &MIRBuilder, SPIRV::Decoration::Decoration Dec, ArrayRef< uint32_t > DecArgs, StringRef StrImm)
constexpr unsigned storageClassToAddressSpace(SPIRV::StorageClass::StorageClass SC)
uint64_t PowerOf2Ceil(uint64_t A)
Returns the power of two which is greater than or equal to the given value.
Type * toTypedPointer(Type *Ty)
LLVM_ABI void report_fatal_error(Error Err, bool gen_crash_diag=true)
auto post_order(const T &G)
Post-order traversal of a graph.
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...
@ Global
Append to llvm.global_dtors.
SPIRV::StorageClass::StorageClass addressSpaceToStorageClass(unsigned AddrSpace, const SPIRVSubtarget &STI)
void buildOpSpirvDecorations(Register Reg, MachineIRBuilder &MIRBuilder, const MDNode *GVarMD, const SPIRVSubtarget &ST)
void processInstr(MachineInstr &MI, MachineIRBuilder &MIB, MachineRegisterInfo &MRI, SPIRVGlobalRegistry *GR, SPIRVTypeInst KnownResType)
DWARFExpression::Operation Op
MachineInstr * getDefInstrMaybeConstant(Register &ConstReg, const MachineRegisterInfo *MRI)
constexpr unsigned BitWidth
decltype(auto) cast(const From &Val)
cast<X> - Return the argument parameter cast to the specified type.
Type * getMDOperandAsType(const MDNode *N, unsigned I)
RelativeUniformCounterPtr ValuesPtrExpr VTableAddr Next
bool isSpvIntrinsic(const MachineInstr &MI, Intrinsic::ID IntrinsicID)
void addStringImm(const StringRef &Str, MCInst &Inst)
MachineInstr * getVRegDef(MachineRegisterInfo &MRI, Register Reg)