summaryrefslogtreecommitdiffstats
path: root/source
diff options
context:
space:
mode:
authorDario Mylonopoulos <32958057+ramenguy99@users.noreply.github.com>2025-03-21 18:21:04 +0100
committerGitHub <noreply@github.com>2025-03-22 01:21:04 +0800
commit969d101aff074675de32bdbe6b97baf744634f78 (patch)
treed82e01e35cdf373564c5ff5c19f149a7b04bf142 /source
parent844d8d2212d11f3d28a55c81f234c99db2c26250 (diff)
Support spirv ops added by multiple extensions (#6615)
* spirv: add support for ops added by multiple extensions Some spirv ops are added by multiple extensions and capabilities. This commit adds support to avoid emitting unnecessary extensions and capabilities if one of the options is already required by some other op. * spirv: allow OpRaytracingAccelerationStructure to use multiple extensions This Op is provided by both SPV_KHR_ray_tracing and SPV_KHR_ray_query and the respective capabilities. Use one if already available and otherwise fall back to SPV_KHR_ray_tracing. * tests/vkray: add negative checks for RayTracingKHR and RayQueryKHR - Add new rayquery-compute.slang to test that only RayQueryKHR is needed in compute shaders. - Add checks for RayTracingKHR and RayQueryKHR capabilities and extensions in raygen.slang --------- Co-authored-by: Ellie Hermaszewska <ellieh@nvidia.com>
Diffstat (limited to 'source')
-rw-r--r--source/slang/slang-emit-spirv.cpp72
1 files changed, 70 insertions, 2 deletions
diff --git a/source/slang/slang-emit-spirv.cpp b/source/slang/slang-emit-spirv.cpp
index 8ecbe1bc7..fda7b098d 100644
--- a/source/slang/slang-emit-spirv.cpp
+++ b/source/slang/slang-emit-spirv.cpp
@@ -1382,6 +1382,36 @@ struct SPIRVEmitContext : public SourceEmitterBase, public SPIRVEmitSharedContex
return result;
}
+ List<List<UnownedStringSlice>> m_anyExtension;
+ void ensureAnyExtensionDeclaration(List<UnownedStringSlice> extensions)
+ {
+ if (!m_anyExtension.contains(extensions))
+ {
+ m_anyExtension.add(extensions);
+ }
+ }
+
+ void emitSPIRVAnyExtension()
+ {
+ for (const auto& options : m_anyExtension)
+ {
+ bool found = false;
+ for (UnownedStringSlice option : options)
+ {
+ if (m_extensionInsts.tryGetValue(option))
+ {
+ found = true;
+ break;
+ }
+ }
+
+ if (!found)
+ {
+ ensureExtensionDeclaration(options[0]);
+ }
+ }
+ }
+
SpvInst* ensureExtensionDeclarationBeforeSpv14(UnownedStringSlice name)
{
if (isSpirv14OrLater())
@@ -1710,8 +1740,10 @@ struct SPIRVEmitContext : public SourceEmitterBase, public SPIRVEmitSharedContex
return emitOpTypeSampler(inst);
case kIROp_RaytracingAccelerationStructureType:
- requireSPIRVCapability(SpvCapabilityRayTracingKHR);
- ensureExtensionDeclaration(UnownedStringSlice("SPV_KHR_ray_tracing"));
+ requireSPIRVAnyCapability({SpvCapabilityRayTracingKHR, SpvCapabilityRayQueryKHR});
+ ensureAnyExtensionDeclaration(
+ {UnownedStringSlice("SPV_KHR_ray_tracing"),
+ UnownedStringSlice("SPV_KHR_ray_query")});
return emitOpTypeAccelerationStructure(inst);
case kIROp_RayQueryType:
@@ -8224,6 +8256,36 @@ struct SPIRVEmitContext : public SourceEmitterBase, public SPIRVEmitSharedContex
}
}
+ List<List<SpvCapability>> m_anyCapability;
+ void requireSPIRVAnyCapability(List<SpvCapability> capabilities)
+ {
+ if (!m_anyCapability.contains(capabilities))
+ {
+ m_anyCapability.add(capabilities);
+ }
+ }
+
+ void emitSPIRVAnyCapabilities()
+ {
+ for (const auto& options : m_anyCapability)
+ {
+ bool found = false;
+ for (SpvCapability option : options)
+ {
+ if (m_capabilities.contains(option))
+ {
+ found = true;
+ break;
+ }
+ }
+
+ if (!found)
+ {
+ requireSPIRVCapability(options[0]);
+ }
+ }
+ }
+
void requireVariableBufferCapabilityIfNeeded(IRInst* type)
{
if (auto ptrType = as<IRPtrTypeBase>(type))
@@ -8416,6 +8478,12 @@ SlangResult emitSPIRVFromIR(
}
} while (context.m_forwardDeclaredPointers.getCount() != 0);
+ // Emit extensions and capabilities for which there are multiple options available.
+ // This is delayed to avoid emitting unnecessary extensions and capabilities if
+ // one of the options is already required by some other op.
+ context.emitSPIRVAnyExtension();
+ context.emitSPIRVAnyCapabilities();
+
context.emitFrontMatter();
context.emitPhysicalLayout();