27#include "llvm/IR/IntrinsicsSPIRV.h"
41 if (MIRBuilder.
getMF()
57 *STI.getRegBankInfo());
71 if (
F.hasFnAttribute(Attribute::AttrKind::NoInline))
72 FuncControl |=
static_cast<uint32_t>(SPIRV::FunctionControl::DontInline);
73 else if (
F.hasFnAttribute(Attribute::AttrKind::AlwaysInline))
74 FuncControl |=
static_cast<uint32_t>(SPIRV::FunctionControl::Inline);
77 FuncControl |=
static_cast<uint32_t>(SPIRV::FunctionControl::Pure);
79 FuncControl |=
static_cast<uint32_t>(SPIRV::FunctionControl::Const);
81 if (ST->canUseExtension(SPIRV::Extension::SPV_INTEL_optnone) ||
82 ST->canUseExtension(SPIRV::Extension::SPV_EXT_optnone))
83 if (
F.hasFnAttribute(Attribute::OptimizeNone))
84 FuncControl |=
static_cast<uint32_t>(SPIRV::FunctionControl::OptNoneEXT);
109 return Arg.getType()->isPointerTy();
112 Type *RetTy = FTy->getReturnType();
122 for (
auto SArgTy : SArgTys)
127static SPIRV::AccessQualifier::AccessQualifier
130 return SPIRV::AccessQualifier::ReadWrite;
134 return SPIRV::AccessQualifier::ReadWrite;
136 if (ArgAttribute->
getString() ==
"read_only")
137 return SPIRV::AccessQualifier::ReadOnly;
138 if (ArgAttribute->
getString() ==
"write_only")
139 return SPIRV::AccessQualifier::WriteOnly;
140 return SPIRV::AccessQualifier::ReadWrite;
143static std::vector<SPIRV::Decoration::Decoration>
146 if (ArgAttribute && ArgAttribute->
getString() ==
"volatile")
147 return {SPIRV::Decoration::Volatile};
156 SPIRV::AccessQualifier::AccessQualifier ArgAccessQual =
159 Type *OriginalArgType =
173 VTy->getNumElements(), MIRBuilder,
true);
205 if (
II &&
II->getIntrinsicID() == Intrinsic::spv_assign_type) {
209 assert(BuiltinType->isTargetExtTy() &&
"Expected TargetExtType");
215 if (!
II ||
II->getIntrinsicID() != Intrinsic::spv_assign_ptr_type)
222 ElementTy, MIRBuilder,
230 ArgAccessQual,
true);
233static SPIRV::ExecutionModel::ExecutionModel
236 "Environment must be resolved before lowering entry points.");
239 return SPIRV::ExecutionModel::Kernel;
241 auto attribute =
F.getFnAttribute(
"hlsl.shader");
242 if (!attribute.isValid()) {
244 "This entry point lacks mandatory hlsl.shader attribute.");
247 const auto value = attribute.getValueAsString();
248 if (value ==
"compute")
249 return SPIRV::ExecutionModel::GLCompute;
250 if (value ==
"vertex")
251 return SPIRV::ExecutionModel::Vertex;
252 if (value ==
"pixel")
253 return SPIRV::ExecutionModel::Fragment;
266 assert(GR &&
"Must initialize the SPIRV type registry before lowering args.");
267 GR->setCurrentFunc(MIRBuilder.
getMF());
275 if (VRegs.size() > 0) {
277 for (
const auto &Arg :
F.args()) {
280 if (VRegs[i].
size() > 1)
283 GR->assignSPIRVTypeToVReg(SpirvTy, VRegs[i][0], MIRBuilder.
getMF());
287 buildOpName(VRegs[i][0], Arg.getName(), MIRBuilder);
289 auto DerefBytes =
static_cast<unsigned>(Arg.getDereferenceableBytes());
292 SPIRV::Decoration::MaxByteOffset, {DerefBytes});
294 if (Arg.hasAttribute(Attribute::Alignment) && !ST->isShader()) {
295 auto Alignment =
static_cast<unsigned>(
296 Arg.getAttribute(Attribute::Alignment).getValueAsInt());
300 if (!ST->isShader()) {
301 if (Arg.hasAttribute(Attribute::ReadOnly)) {
303 static_cast<unsigned>(SPIRV::FunctionParameterAttribute::NoWrite);
305 SPIRV::Decoration::FuncParamAttr, {Attr});
307 if (Arg.hasAttribute(Attribute::ZExt)) {
309 static_cast<unsigned>(SPIRV::FunctionParameterAttribute::Zext);
311 SPIRV::Decoration::FuncParamAttr, {Attr});
313 if (Arg.hasAttribute(Attribute::SExt)) {
315 static_cast<unsigned>(SPIRV::FunctionParameterAttribute::Sext);
317 SPIRV::Decoration::FuncParamAttr, {Attr});
319 if (Arg.hasAttribute(Attribute::NoAlias)) {
321 static_cast<unsigned>(SPIRV::FunctionParameterAttribute::NoAlias);
323 SPIRV::Decoration::FuncParamAttr, {Attr});
330 if (Arg.hasAttribute(Attribute::ByVal) ||
331 (Arg.hasAttribute(Attribute::ByRef) &&
332 F.getParent()->getTargetTriple().getVendor() ==
335 static_cast<unsigned>(SPIRV::FunctionParameterAttribute::ByVal);
337 SPIRV::Decoration::FuncParamAttr, {Attr});
339 if (Arg.hasAttribute(Attribute::StructRet)) {
341 static_cast<unsigned>(SPIRV::FunctionParameterAttribute::Sret);
343 SPIRV::Decoration::FuncParamAttr, {Attr});
348 std::vector<SPIRV::Decoration::Decoration> ArgTypeQualDecs =
350 for (SPIRV::Decoration::Decoration Decoration : ArgTypeQualDecs)
354 MDNode *
Node =
F.getMetadata(
"spirv.ParameterDecorations");
355 if (
Node && i < Node->getNumOperands() &&
360 assert(MD2 &&
"Metadata operand is expected");
362 assert(Const &&
"MDOperand should be ConstantInt");
364 static_cast<SPIRV::Decoration::Decoration
>(Const->getZExtValue());
365 std::vector<uint32_t> DecVec;
368 assert(Const &&
"MDOperand should be ConstantInt");
369 DecVec.push_back(
static_cast<uint32_t>(Const->getZExtValue()));
378 auto MRI = MIRBuilder.
getMRI();
380 MRI->setRegClass(FuncVReg, &SPIRV::iIDRegClass);
384 if (
Type *FRetElemTy = GR->findDeducedElementType(&
F)) {
387 GR->addReturnType(&
F, DerivedTy);
392 FRetTy, MIRBuilder, SPIRV::AccessQualifier::ReadWrite,
true);
394 SPIRVTypeInst FuncTy = GR->getOrCreateOpTypeFunctionWithArgs(
395 FTy, RetTy, ArgTypeVRegs, MIRBuilder);
401 .
addUse(GR->getSPIRVTypeID(RetTy))
403 .
addUse(GR->getSPIRVTypeID(FuncTy));
405 GR->addGlobalObject(&
F, &MIRBuilder.
getMF(), FuncVReg);
406 if (
F.isDeclaration())
411 for (
const auto &Arg :
F.args()) {
412 assert(VRegs[i].
size() == 1 &&
"Formal arg has multiple vregs");
414 MRI->setRegClass(ArgReg, GR->getRegClass(ArgTypeVRegs[i]));
415 auto MIB = MIRBuilder.
buildInstr(SPIRV::OpFunctionParameter)
417 .
addUse(GR->getSPIRVTypeID(ArgTypeVRegs[i]));
418 if (
F.isDeclaration())
420 GR->addGlobalObject(&Arg, &MIRBuilder.
getMF(), ArgReg);
429 auto MIB = MIRBuilder.
buildInstr(SPIRV::OpEntryPoint)
434 buildOpDecorate(FuncVReg, MIRBuilder, SPIRV::Decoration::LinkageAttributes,
435 {
static_cast<uint32_t>(*LnkTy)},
F.getName());
439 bool hasFunctionPointers =
440 ST->canUseExtension(SPIRV::Extension::SPV_INTEL_function_pointers);
441 if (hasFunctionPointers) {
442 if (
F.hasFnAttribute(
"referenced-indirectly")) {
444 "Unexpected 'referenced-indirectly' attribute of the kernel "
447 SPIRV::Decoration::ReferencedIndirectlyINTEL, {});
462void SPIRVCallLowering::produceIndirectPtrType(
464 const SPIRVCallLowering::SPIRVIndirectCall &IC)
const {
469 IC.RetTy, MIRBuilder, SPIRV::AccessQualifier::ReadWrite,
true);
471 for (
size_t i = 0; i < IC.ArgTys.size(); ++i) {
473 IC.ArgTys[i], MIRBuilder, SPIRV::AccessQualifier::ReadWrite,
true);
482 FTy, SpirvRetTy, SpirvArgTypes, MIRBuilder);
484 auto SC = ST.canUseExtension(SPIRV::Extension::SPV_INTEL_function_pointers)
485 ? SPIRV::StorageClass::CodeSectionINTEL
486 : SPIRV::StorageClass::Function;
497 if (Info.OrigRet.Regs.size() > 1)
500 GR->setCurrentFunc(MF);
502 std::string DemangledName;
503 const Type *OrigRetTy = Info.OrigRet.Ty;
508 if (Info.Callee.isGlobal()) {
509 std::string FuncName = Info.Callee.getGlobal()->getName().str();
519 if (
auto *DerivedRetTy = GR->findReturnType(CF))
520 OrigRetTy = DerivedRetTy;
526 Info.OrigRet.Regs.empty() ?
Register(0) : Info.OrigRet.Regs[0];
530 if (isFunctionDecl && !DemangledName.empty()) {
532 if (!GR->getSPIRVTypeForVReg(ResVReg)) {
533 const Type *RetTy = OrigRetTy;
535 const Value *OrigValue = Info.OrigRet.OrigValue;
539 if (
Type *ElemTy = GR->findDeducedElementType(OrigValue))
544 SPIRV::AccessQualifier::ReadWrite,
true);
548 SPIRV::AccessQualifier::ReadWrite,
true);
551 for (
auto Arg : Info.OrigArgs) {
552 assert(Arg.Regs.size() == 1 &&
"Call arg has multiple VRegs");
557 Type *ArgTy =
nullptr;
565 if (
Type *ElemTy = GR->findDeducedElementType(Arg.OrigValue))
572 SpvType = GR->getOrCreateSPIRVType(
573 ArgTy, MIRBuilder, SPIRV::AccessQualifier::ReadWrite,
true);
574 GR->assignSPIRVTypeToVReg(SpvType, ArgReg, MF);
581 MRI->
setRegClass(ArgReg, SpvType ? GR->getRegClass(SpvType)
582 : &SPIRV::pIDRegClass);
585 SpvType ? GR->getRegType(SpvType)
587 GR->getPointerSize()));
591 DemangledName, ST->getPreferredInstructionSet(), MIRBuilder,
592 ResVReg, OrigRetTy, ArgVRegs, GR, *Info.CB))
596 if (isFunctionDecl && !GR->find(CF, &MF).isValid()) {
600 FirstBlockBuilder.
setMF(MF);
619 if (MIRBuilder.
getMF()
629 if (Info.CB->isIndirectCall()) {
630 if (!ST->canUseExtension(SPIRV::Extension::SPV_INTEL_function_pointers))
632 "extensions does not support it",
635 CallOp = SPIRV::OpFunctionPointerCallINTEL;
637 Register CalleeReg = Info.Callee.getReg();
644 "Function types mismatch");
645 for (
unsigned I = 0;
I != Info.OrigArgs.size(); ++
I) {
646 assert(Info.OrigArgs[
I].Regs.size() == 1 &&
647 "Call arg has multiple VRegs");
655 CallOp = SPIRV::OpFunctionCall;
662 OrigRetTy, ResVReg, MIRBuilder, SPIRV::AccessQualifier::ReadWrite,
true);
667 .
addUse(GR->getSPIRVTypeID(RetType))
670 for (
const auto &Arg : Info.OrigArgs) {
672 if (Arg.Regs.size() > 1)
674 MIB.addUse(Arg.Regs[0]);
677 if (ST->canUseExtension(SPIRV::Extension::SPV_INTEL_memory_access_aliasing)) {
682 GR->buildMemAliasingOpDecorate(ResVReg, MIRBuilder,
683 SPIRV::Decoration::AliasScopeINTEL, MD);
685 GR->buildMemAliasingOpDecorate(ResVReg, MIRBuilder,
686 SPIRV::Decoration::NoAliasINTEL, MD);
691 *ST->getRegBankInfo());
assert(UImm &&(UImm !=~static_cast< T >(0)) &&"Invalid immediate!")
Promote Memory to Register
uint64_t IntrinsicInst * II
static ConstantInt * getConstInt(MDNode *MD, unsigned NumOp)
static SPIRVTypeInst getArgSPIRVType(const Function &F, unsigned ArgIdx, SPIRVGlobalRegistry *GR, MachineIRBuilder &MIRBuilder, const SPIRVSubtarget &ST)
static SPIRV::ExecutionModel::ExecutionModel getExecutionModel(const SPIRVSubtarget &STI, const Function &F)
static uint32_t getFunctionControl(const Function &F, const SPIRVSubtarget *ST)
static SPIRV::AccessQualifier::AccessQualifier getArgAccessQual(const Function &F, unsigned ArgIdx)
static FunctionType * fixFunctionTypeIfPtrArgs(SPIRVGlobalRegistry *GR, const Function &F, FunctionType *FTy, SPIRVTypeInst SRetTy, const SmallVector< SPIRVTypeInst, 4 > &SArgTys)
static std::vector< SPIRV::Decoration::Decoration > getKernelArgTypeQual(const Function &F, unsigned ArgIdx)
#define SPIRV_BACKEND_SERVICE_FUN_NAME
This class represents an incoming formal argument to a Function.
Represent a constant reference to an array (0 or more elements consecutively in memory),...
size_t size() const
Get the array size.
bool isValid() const
Return true if the attribute is any kind of attribute.
Base class for all callable instructions (InvokeInst and CallInst) Holds everything related to callin...
CallLowering(const TargetLowering *TLI)
This is the shared class of boolean and integer constants.
TypeSize getTypeStoreSize(Type *Ty) const
Returns the maximum number of bytes that may be overwritten by storing the specified type.
FunctionLoweringInfo - This contains information that is global to a function that is used when lower...
Class to represent function types.
unsigned getNumParams() const
Return the number of fixed parameters this function type requires.
Type * getParamType(unsigned i) const
Parameter type accessors.
Type * getReturnType() const
static LLVM_ABI FunctionType * get(Type *Result, ArrayRef< Type * > Params, bool isVarArg)
This static method is the primary way of constructing a FunctionType.
iterator_range< arg_iterator > args()
Attribute getFnAttribute(Attribute::AttrKind Kind) const
Return the attribute for the given attribute kind.
LLVM_ABI bool isDeclaration() const
Return true if the primary definition of this global value is outside of the current translation unit...
bool hasMetadata() const
Return true if this instruction has any metadata attached to it.
MDNode * getMetadata(unsigned KindID) const
Get the metadata of given kind attached to this Instruction.
static constexpr LLT scalar(unsigned SizeInBits)
Get a low-level scalar or aggregate "bag of bits".
static constexpr LLT pointer(unsigned AddressSpace, unsigned SizeInBits)
Get a low-level pointer in the given address space.
const MDOperand & getOperand(unsigned I) const
ArrayRef< MDOperand > operands() const
unsigned getNumOperands() const
Return number of MDNode operands.
Tracking metadata reference owned by Metadata.
LLVM_ABI StringRef getString() const
const TargetSubtargetInfo & getSubtarget() const
getSubtarget - Return the subtarget for which this machine code is being compiled.
MachineBasicBlock * getBlockNumbered(unsigned N) const
getBlockNumbered - MachineBasicBlocks are automatically numbered when they are inserted into the mach...
Function & getFunction()
Return the LLVM function that this machine code represents.
Helper class to build MachineInstr.
const TargetInstrInfo & getTII()
MachineInstrBuilder buildInstr(unsigned Opcode)
Build and insert <empty> = Opcode <empty>.
MachineFunction & getMF()
Getter for the function we currently build.
void setMBB(MachineBasicBlock &MBB)
Set the insertion point to the end of MBB.
MachineInstrBuilder buildTrap(bool Debug=false)
Build and insert G_TRAP or G_DEBUGTRAP.
MachineRegisterInfo * getMRI()
Getter for MRI.
const DataLayout & getDataLayout() const
void setMF(MachineFunction &MF)
void constrainAllUses(const TargetInstrInfo &TII, const TargetRegisterInfo &TRI, const RegisterBankInfo &RBI) const
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 & add(const MachineOperand &MO) const
const MachineInstrBuilder & addDef(Register RegNo, RegState Flags={}, unsigned SubReg=0) const
Add a virtual register definition operand.
MachineInstr * getInstr() const
If conversion operators fail, use this method to get the MachineInstr explicitly.
const MachineOperand & getOperand(unsigned i) const
MachineRegisterInfo - Keep track of information for virtual and physical registers,...
LLVM_ABI Register createVirtualRegister(const TargetRegisterClass *RegClass, StringRef Name="")
createVirtualRegister - Create and return a new virtual register in the function with the specified r...
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.
bool doesNotAccessMemory() const
Whether this function accesses no memory.
bool onlyReadsMemory() const
Whether this function only (at most) reads memory.
Wrapper class representing virtual and physical registers.
constexpr bool isValid() const
bool lowerCall(MachineIRBuilder &MIRBuilder, CallLoweringInfo &Info) const override
This hook must be implemented to lower the given call instruction, including argument and return valu...
bool lowerReturn(MachineIRBuilder &MIRBuiler, const Value *Val, ArrayRef< Register > VRegs, FunctionLoweringInfo &FLI, Register SwiftErrorVReg) const override
This hook must be implemented to lower outgoing return values, described by Val, into the specified v...
SPIRVCallLowering(const SPIRVTargetLowering &TLI, SPIRVGlobalRegistry *GR)
bool lowerFormalArguments(MachineIRBuilder &MIRBuilder, const Function &F, ArrayRef< ArrayRef< Register > > VRegs, FunctionLoweringInfo &FLI) const override
This hook must be implemented to lower the incoming (formal) arguments, described by VRegs,...
void assignSPIRVTypeToVReg(SPIRVTypeInst Type, Register VReg, const MachineFunction &MF)
SPIRVTypeInst getOrCreateOpTypeFunctionWithArgs(const Type *Ty, SPIRVTypeInst RetType, const SmallVectorImpl< SPIRVTypeInst > &ArgTypes, MachineIRBuilder &MIRBuilder)
SPIRVTypeInst getOrCreateSPIRVVectorType(SPIRVTypeInst BaseType, unsigned NumElements, MachineIRBuilder &MIRBuilder, bool EmitIR)
const Type * getTypeForSPIRVType(SPIRVTypeInst Ty) const
SPIRVTypeInst getOrCreateSPIRVPointerType(const Type *BaseType, MachineIRBuilder &MIRBuilder, SPIRV::StorageClass::StorageClass SC)
SPIRVTypeInst getOrCreateSPIRVType(const Type *Type, MachineInstr &I, SPIRV::AccessQualifier::AccessQualifier AQ, bool EmitIR)
SPIRVTypeInst getSPIRVTypeForVReg(Register VReg, const MachineFunction *MF=nullptr) const
Type * findDeducedElementType(const Value *Val)
SPIRVEnvType getEnv() const
void push_back(const T &Elt)
This is a 'vector' (really, a variable-sized array), optimized for the case when the array is small.
const TargetRegisterInfo & getRegisterInfo() const
The instances of the Type class are immutable: once they are created, they are never changed.
bool isPointerTy() const
True if this is an instance of PointerType.
A few GPU targets, such as DXIL and SPIR-V, have typed pointers.
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.
LLVM Value Representation.
Type * getType() const
All values are typed, get the type of this value.
iterator_range< user_iterator > users()
constexpr bool isZero() const
@ SPIR_KERNEL
Used for SPIR kernel functions.
std::optional< bool > lowerBuiltin(const StringRef DemangledCall, SPIRV::InstructionSet::InstructionSet Set, MachineIRBuilder &MIRBuilder, const Register OrigRet, const Type *OrigRetTy, const SmallVectorImpl< Register > &Args, SPIRVGlobalRegistry *GR, const CallBase &CB)
FunctionType * getOriginalFunctionType(const Function &F)
This is an optimization pass for GlobalISel generic memory operations.
void buildOpName(Register Target, const StringRef &Name, MachineIRBuilder &MIRBuilder)
auto size(R &&Range, std::enable_if_t< std::is_base_of< std::random_access_iterator_tag, typename std::iterator_traits< decltype(Range.begin())>::iterator_category >::value, void > *=nullptr)
Get the size of a range.
unsigned getPointerAddressSpace(const Type *T)
decltype(auto) dyn_cast(const From &Val)
dyn_cast<X> - Return the argument parameter cast to the specified type.
Register createVirtualRegister(SPIRVTypeInst SpvType, SPIRVGlobalRegistry *GR, MachineRegisterInfo *MRI, const MachineFunction &MF)
MemoryEffectsBase< IRMemLocation > MemoryEffects
Summary of how a function affects memory in the program.
void buildOpDecorate(Register Reg, MachineIRBuilder &MIRBuilder, SPIRV::Decoration::Decoration Dec, ArrayRef< uint32_t > DecArgs, StringRef StrImm)
MDString * getOCLKernelArgAccessQual(const Function &F, unsigned ArgIdx)
std::string getOclOrSpirvBuiltinDemangledName(StringRef Name)
bool isTypedPointerTy(const Type *T)
auto dyn_cast_or_null(const Y &Val)
bool any_of(R &&range, UnaryPredicate P)
Provide wrappers to std::any_of which take ranges instead of having to pass begin/end explicitly.
Type * toTypedPointer(Type *Ty)
bool isPointerTy(const Type *T)
LLVM_ABI void report_fatal_error(Error Err, bool gen_crash_diag=true)
void setRegClassType(Register Reg, SPIRVTypeInst SpvType, SPIRVGlobalRegistry *GR, MachineRegisterInfo *MRI, const MachineFunction &MF, bool Force)
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...
std::optional< SPIRV::LinkageType::LinkageType > getSpirvLinkageTypeFor(const SPIRVSubtarget &ST, const GlobalValue &GV)
bool isEntryPoint(const Function &F)
SPIRV::StorageClass::StorageClass addressSpaceToStorageClass(unsigned AddrSpace, const SPIRVSubtarget &STI)
MDString * getOCLKernelArgTypeQual(const Function &F, unsigned ArgIdx)
Type * getPointeeTypeByAttr(Argument *Arg)
bool hasPointeeTypeAttr(Argument *Arg)
decltype(auto) cast(const From &Val)
cast<X> - Return the argument parameter cast to the specified type.
bool isPointerTyOrWrapper(const Type *Ty)
void addStringImm(const StringRef &Str, MCInst &Inst)
bool isUntypedPointerTy(const Type *T)