From 969d101aff074675de32bdbe6b97baf744634f78 Mon Sep 17 00:00:00 2001 From: Dario Mylonopoulos <32958057+ramenguy99@users.noreply.github.com> Date: Fri, 21 Mar 2025 18:21:04 +0100 Subject: 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 --- source/slang/slang-emit-spirv.cpp | 72 +++++++++++++++++++++++++++++++++++++-- 1 file changed, 70 insertions(+), 2 deletions(-) (limited to 'source') 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> m_anyExtension; + void ensureAnyExtensionDeclaration(List 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> m_anyCapability; + void requireSPIRVAnyCapability(List 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(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(); -- cgit v1.2.3