summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorYong He <yonghe@outlook.com>2025-02-10 23:48:07 -0800
committerGitHub <noreply@github.com>2025-02-11 15:48:07 +0800
commit0bc18d233966fc80cf2c482922d0b773d58394ca (patch)
tree5c18ca0f3d9bc81bed3bfb79f710c1536bd26f00
parent3c2d46aa1c8575dc046d7457793e77c7a4789093 (diff)
Emit missing atomic64 capability, fix int-typed builtin used in both vs and fs. (#6314)
Co-authored-by: Ellie Hermaszewska <ellieh@nvidia.com>
-rw-r--r--source/slang/slang-emit-spirv.cpp47
-rw-r--r--tests/spirv/atomic-64bit.slang13
-rw-r--r--tests/spirv/view-id.slang19
3 files changed, 66 insertions, 13 deletions
diff --git a/source/slang/slang-emit-spirv.cpp b/source/slang/slang-emit-spirv.cpp
index 4a66cb4ef..ef015df7f 100644
--- a/source/slang/slang-emit-spirv.cpp
+++ b/source/slang/slang-emit-spirv.cpp
@@ -3192,6 +3192,17 @@ struct SPIRVEmitContext : public SourceEmitterBase, public SPIRVEmitSharedContex
void ensureAtomicCapability(IRInst* atomicInst, SpvOp op)
{
auto typeOp = atomicInst->getDataType()->getOp();
+ if (typeOp == kIROp_VoidType)
+ {
+ auto ptrType = atomicInst->getOperand(0)->getDataType();
+ IRBuilder builder(atomicInst);
+ if (auto valType = tryGetPointedToType(&builder, ptrType))
+ {
+ if (auto atomicType = as<IRAtomicType>(valType))
+ valType = atomicType->getElementType();
+ typeOp = valType->getOp();
+ }
+ }
switch (op)
{
case SpvOpAtomicFAddEXT:
@@ -5094,18 +5105,23 @@ struct SPIRVEmitContext : public SourceEmitterBase, public SPIRVEmitSharedContex
{
SpvBuiltIn builtinName;
SpvStorageClass storageClass = SpvStorageClassInput;
+ bool flat = false;
BuiltinSpvVarKey() = default;
- BuiltinSpvVarKey(SpvBuiltIn builtin, SpvStorageClass storageClass)
- : builtinName(builtin), storageClass(storageClass)
+ BuiltinSpvVarKey(SpvBuiltIn builtin, SpvStorageClass storageClass, bool isFlat)
+ : builtinName(builtin), storageClass(storageClass), flat(isFlat)
{
}
bool operator==(const BuiltinSpvVarKey& other) const
{
- return builtinName == other.builtinName && storageClass == other.storageClass;
+ return builtinName == other.builtinName && storageClass == other.storageClass &&
+ flat == other.flat;
}
HashCode getHashCode() const
{
- return combineHash(Slang::getHashCode(builtinName), Slang::getHashCode(storageClass));
+ return combineHash(
+ Slang::getHashCode(builtinName),
+ Slang::getHashCode(storageClass),
+ Slang::getHashCode(flat));
}
};
Dictionary<BuiltinSpvVarKey, SpvInst*> m_builtinGlobalVars;
@@ -5127,26 +5143,25 @@ struct SPIRVEmitContext : public SourceEmitterBase, public SPIRVEmitSharedContex
return false;
}
- void maybeEmitFlatDecorationForBuiltinVar(IRInst* irInst, SpvInst* spvInst)
+ bool needFlatDecorationForBuiltinVar(IRInst* irInst)
{
if (!irInst)
- return;
+ return false;
if (irInst->getOp() != kIROp_GlobalVar && irInst->getOp() != kIROp_GlobalParam)
- return;
+ return false;
auto ptrType = as<IRPtrType>(irInst->getDataType());
if (!ptrType)
- return;
+ return false;
auto addrSpace = ptrType->getAddressSpace();
if (addrSpace == AddressSpace::Input || addrSpace == AddressSpace::BuiltinInput)
{
if (isIntegralScalarOrCompositeType(ptrType->getValueType()))
{
if (isInstUsedInStage(irInst, Stage::Fragment))
- _maybeEmitInterpolationModifierDecoration(
- IRInterpolationMode::NoInterpolation,
- getID(spvInst));
+ return true;
}
}
+ return false;
}
SpvInst* getBuiltinGlobalVar(IRType* type, SpvBuiltIn builtinVal, IRInst* irInst)
@@ -5155,7 +5170,8 @@ struct SPIRVEmitContext : public SourceEmitterBase, public SPIRVEmitSharedContex
auto ptrType = as<IRPtrTypeBase>(type);
SLANG_ASSERT(ptrType && "`getBuiltinGlobalVar`: `type` must be ptr type.");
auto storageClass = addressSpaceToStorageClass(ptrType->getAddressSpace());
- auto key = BuiltinSpvVarKey(builtinVal, storageClass);
+ bool isFlat = needFlatDecorationForBuiltinVar(irInst);
+ auto key = BuiltinSpvVarKey(builtinVal, storageClass, isFlat);
if (m_builtinGlobalVars.tryGetValue(key, result))
{
return result;
@@ -5185,7 +5201,12 @@ struct SPIRVEmitContext : public SourceEmitterBase, public SPIRVEmitSharedContex
}
m_builtinGlobalVars[key] = varInst;
- maybeEmitFlatDecorationForBuiltinVar(irInst, varInst);
+ if (isFlat)
+ {
+ _maybeEmitInterpolationModifierDecoration(
+ IRInterpolationMode::NoInterpolation,
+ getID(varInst));
+ }
return varInst;
}
diff --git a/tests/spirv/atomic-64bit.slang b/tests/spirv/atomic-64bit.slang
new file mode 100644
index 000000000..2b8220dd0
--- /dev/null
+++ b/tests/spirv/atomic-64bit.slang
@@ -0,0 +1,13 @@
+//TEST:SIMPLE(filecheck=CHECK):-target spirv
+
+// CHECK: OpCapability Int64Atomics
+
+groupshared Atomic<uint64_t> atomicVariable;
+
+[shader("compute")]
+[numthreads(1, 1, 1)]
+void main(uint3 threadId : SV_DispatchThreadID)
+{
+ uint64_t value = 4ll;
+ atomicVariable.store(value);
+} \ No newline at end of file
diff --git a/tests/spirv/view-id.slang b/tests/spirv/view-id.slang
new file mode 100644
index 000000000..9df3168a4
--- /dev/null
+++ b/tests/spirv/view-id.slang
@@ -0,0 +1,19 @@
+//TEST:SIMPLE(filecheck=CHECK): -target spirv
+
+// ViewIndex builtin should be declared twice, once for vertex and once for fragment shader.
+// Because the fragment shader builtin needs Flat, and vertex shader does not.
+
+// CHECK: OpDecorate %{{.*}} BuiltIn ViewIndex
+// CHECK: OpDecorate %{{.*}} BuiltIn ViewIndex
+
+[shader("vertex")]
+float4 vert(int _viewportIndex : SV_ViewID):SV_Position
+{
+ return float4(_viewportIndex);
+}
+
+[shader("fragment")]
+float4 frag(int _viewportIndex : SV_ViewID) : SV_Target
+{
+ return float4(_viewportIndex, 0, 0, 1);
+} \ No newline at end of file