summaryrefslogtreecommitdiff
path: root/source/slang/slang-emit-spirv.cpp
diff options
context:
space:
mode:
authorEllie Hermaszewska <ellieh@nvidia.com>2023-11-03 04:00:16 +0800
committerGitHub <noreply@github.com>2023-11-02 13:00:16 -0700
commit72e95f2c62b39ef1ddb6c169a9452a3b4fcb22a5 (patch)
tree878d8b747b74179530b7164dc639ce82814b8dee /source/slang/slang-emit-spirv.cpp
parente712ebd93aa9a2845bde3ea541aaa7cd415548b7 (diff)
Several spirv intrinsics and small fix (#3307)
* spirv intrinsic for faceforward * spirv intrinsic for fwidth * spirv intrinsic for modf * spirv intrinsic for nonuniformresourceindex * spirv intrinsic for transpose * Make sure address space matches for OpAccessChain * Correct placement for OpDecorate instructions in spirv asm blocks
Diffstat (limited to 'source/slang/slang-emit-spirv.cpp')
-rw-r--r--source/slang/slang-emit-spirv.cpp48
1 files changed, 40 insertions, 8 deletions
diff --git a/source/slang/slang-emit-spirv.cpp b/source/slang/slang-emit-spirv.cpp
index 0eae87a10..f26340969 100644
--- a/source/slang/slang-emit-spirv.cpp
+++ b/source/slang/slang-emit-spirv.cpp
@@ -3732,19 +3732,25 @@ struct SPIRVEmitContext
SpvInst* emitGetElementPtr(SpvInstParent* parent, IRGetElementPtr* inst)
{
+ IRBuilder builder(m_irModule);
auto base = inst->getBase();
- SpvWord baseId = 0;
- // Only used in debug build, but we don't want a warning/error for an unused initialized variable
+ const SpvWord baseId = getID(ensureInst(base));
+
+ // We might replace resultType with a different storage class equivalent
+ auto resultType = as<IRPtrTypeBase>(inst->getFullType());
+ SLANG_ASSERT(resultType);
- if (as<IRPointerLikeType>(base->getDataType()) || as<IRPtrTypeBase>(base->getDataType()))
+ if(const auto basePtrType = as<IRPtrTypeBase>(base->getDataType()))
{
- baseId = getID(ensureInst(base));
+ // If the base pointer has a specific address space and the
+ // expected result type doesn't, then make sure they match.
+ // It's invalid spir-v if they don't match
+ resultType = getPtrTypeWithAddressSpace(cast<IRPtrTypeBase>(inst->getDataType()), basePtrType->getAddressSpace());
}
else
{
- SLANG_ASSERT(!"invalid IR: base of getElementPtr must be a pointer.");
+ SLANG_ASSERT(as<IRPointerLikeType>(base->getDataType()) || !"invalid IR: base of getElementPtr must be a pointer.");
}
- SLANG_ASSERT(as<IRPtrTypeBase>(inst->getFullType()));
return emitOpAccessChain(
parent,
inst,
@@ -3835,12 +3841,36 @@ struct SPIRVEmitContext
return emitOpVectorShuffle(parent, inst, inst->getFullType(), inst->getBase(), inst->getSource(), shuffleIndices.getArrayView());
}
+ IRPtrTypeBase* getPtrTypeWithAddressSpace(IRPtrTypeBase* ptrTypeWithNoAddressSpace, IRIntegerValue addressSpace)
+ {
+ // If it's already ok, return as is
+ if(ptrTypeWithNoAddressSpace->getAddressSpace() == addressSpace)
+ return ptrTypeWithNoAddressSpace;
+
+ // It has an address space, but it doesn't match fail, this indicates a
+ // problem with whatever's creating these types
+ SLANG_ASSERT(!ptrTypeWithNoAddressSpace->hasAddressSpace());
+
+ IRBuilder builder(ptrTypeWithNoAddressSpace);
+ return builder.getPtrType(
+ ptrTypeWithNoAddressSpace->getOp(),
+ ptrTypeWithNoAddressSpace->getValueType(),
+ addressSpace
+ );
+ }
+
SpvInst* emitStructuredBufferGetElementPtr(SpvInstParent* parent, IRInst* inst)
{
//"%addr = OpAccessChain resultType*StorageBuffer resultId _0 const(int, 0) _1;"
IRBuilder builder(inst);
- auto addr = emitInst(parent, inst, SpvOpAccessChain, inst->getDataType(), kResultID, inst->getOperand(0), emitIntConstant(0, builder.getIntType()), inst->getOperand(1));
- return addr;
+ return emitOpAccessChain(
+ parent,
+ inst,
+ // Make sure the resulting pointer has the correct storage class
+ getPtrTypeWithAddressSpace(cast<IRPtrTypeBase>(inst->getDataType()), SpvStorageClassStorageBuffer),
+ inst->getOperand(0),
+ makeArray(emitIntConstant(0, builder.getIntType()), ensureInst(inst->getOperand(1)))
+ );
}
SpvInst* emitStructuredBufferGetDimensions(SpvInstParent* parent, IRInst* inst)
@@ -4442,6 +4472,8 @@ struct SPIRVEmitContext
return getSection(SpvLogicalSectionID::Extensions);
case SpvOpExecutionMode:
return getSection(SpvLogicalSectionID::ExecutionModes);
+ case SpvOpDecorate:
+ return getSection(SpvLogicalSectionID::Annotations);
default:
return defaultParent;