diff options
Diffstat (limited to 'source')
| -rw-r--r-- | source/slang/hlsl.meta.slang | 110 | ||||
| -rw-r--r-- | source/slang/slang-ir-lower-dynamic-resource-heap.cpp | 16 |
2 files changed, 119 insertions, 7 deletions
diff --git a/source/slang/hlsl.meta.slang b/source/slang/hlsl.meta.slang index 99f83776e..57cb188bb 100644 --- a/source/slang/hlsl.meta.slang +++ b/source/slang/hlsl.meta.slang @@ -22251,6 +22251,7 @@ enum DescriptorKind Buffer, /// A buffer descriptor. Sampler, /// A sampler state descriptor. AccelerationStructure, /// A ray tracing acceleration structure descriptor. + TexelBuffer /// A texel buffer descriptor. } enum DescriptorAccess @@ -22303,7 +22304,9 @@ extension _Texture<T, Shape, isArray, isMS, sampleCount, access, isShadow, isCom typealias Handle = DescriptorHandle<This>; - static const DescriptorKind kind = isCombined != 0 ? DescriptorKind.CombinedTextureSampler : DescriptorKind.Texture; + static const DescriptorKind kind = (Shape.flavor == __ShapeBuffer.flavor ? + DescriptorKind.TexelBuffer : (isCombined != 0 ? DescriptorKind.CombinedTextureSampler : DescriptorKind.Texture) +); static const DescriptorAccess descriptorAccess = (DescriptorAccess)access; __implicit_conversion($(kConversionCost_ImplicitDereference)) @@ -22422,8 +22425,36 @@ __prefix T operator*(DescriptorHandle<T> value) return getDescriptorFromHandle(value); } +// https://registry.khronos.org/vulkan/specs/latest/man/html/VkDescriptorType.html +enum DefaultVkBindlessBindings : uint +{ + Sampler = 0, /// SAMPLER + CombinedTextureSampler = 1, /// COMBINED_IMAGE_SAMPLER + Texture_Read = 2, /// SAMPLED_IMAGE + Texture_ReadWrite = 3, /// STORAGE_IMAGE + TexelBuffer_Read = 4, /// UNIFORM_TEXEL_BUFFER + TexelBuffer_ReadWrite = 5, /// STORAGE_TEXEL_BUFFER + Buffer_Read = 6, /// UNIFORM_BUFFER + Buffer_ReadWrite = 7, /// STORAGE_BUFFER + Unknown = 8, /// Other +} + +// https://github.com/KhronosGroup/Vulkan-Docs/blob/main/proposals/VK_EXT_mutable_descriptor_type.adoc +enum VkMutableBindlessBindings : uint +{ + Sampler = 0, /// SAMPLER + CombinedTextureSampler = 1, /// COMBINED_IMAGE_SAMPLER + Texture_Read = 2, /// SAMPLED_IMAGE + Texture_ReadWrite = 2, /// STORAGE_IMAGE + TexelBuffer_Read = 2, /// UNIFORM_TEXEL_BUFFER + TexelBuffer_ReadWrite = 2, /// STORAGE_TEXEL_BUFFER + Buffer_Read = 2, /// UNIFORM_BUFFER + Buffer_ReadWrite = 2, /// STORAGE_BUFFER, + Unknown = 3, /// Other +} + __intrinsic_op($(kIROp_GetDynamicResourceHeap)) -T[] __getDynamicResourceHeap<T:IOpaqueDescriptor>(); +T[] __getDynamicResourceHeap<T:IOpaqueDescriptor>(constexpr uint bindingIndex = 0); // Used by the HLSL backend only, load a sampler state handle from SamplerDescriptorHeap. @@ -22442,10 +22473,16 @@ T __makeCombinedTextureSamplerFromHandle<T, U>(U handle); __intrinsic_op($(kIROp_CastDescriptorHandleToResource)) T __castDescriptorHandleToResource<T:IOpaqueDescriptor>(DescriptorHandle<T> ptr); +public enum BindlessDescriptorOptions +{ + None = 0, /// Bind assuming regular binding model rules. + VkMutable = 1, /// Bind assuming `VK_EXT_mutable_descriptor_type` without mutable `AccelerationStructure` binding support. +} + /// The default implementation of `getDescriptorFromHandle`, which converts from a descriptor handle /// to a descriptor object. [ForceInline] -T defaultGetDescriptorFromHandle<T:IOpaqueDescriptor>(DescriptorHandle<T> handleValue) +T defaultGetDescriptorFromHandle<T:IOpaqueDescriptor>(DescriptorHandle<T> handleValue, constexpr BindlessDescriptorOptions bindlessOptions = BindlessDescriptorOptions.None) { __target_switch { @@ -22458,6 +22495,73 @@ T defaultGetDescriptorFromHandle<T:IOpaqueDescriptor>(DescriptorHandle<T> handle return __loadResourceDescriptorFromHeap<T>(((uint2)handleValue).x); case spirv: case glsl: + + switch(bindlessOptions) + { +${{{{ +{ + static const struct + { + char const* option; + char const* enumType; + }kBindlessOptions[] = + { + {"None", "DefaultVkBindlessBindings"}, + {"VkMutable", "VkMutableBindlessBindings"}, + }; + + for(auto bindlessOption : kBindlessOptions) + { + StringBuilder bindlessOptionIfElsePre; + StringBuilder bindlessOptionIfElsePost; + + bindlessOptionIfElsePre << "case BindlessDescriptorOptions." << bindlessOption.option <<":\n{\n"; + bindlessOptionIfElsePost << "}\n"; +}}}} + $(bindlessOptionIfElsePre.toString()) + switch(T.kind) + { + case DescriptorKind.Sampler: + return __getDynamicResourceHeap<T>($(bindlessOption.enumType).Sampler)[((uint2)handleValue).x]; + case DescriptorKind.CombinedTextureSampler: + return __getDynamicResourceHeap<T>($(bindlessOption.enumType).CombinedTextureSampler)[((uint2)handleValue).x]; + case DescriptorKind.Texture: + { + if(DescriptorAccess.Read == T.descriptorAccess) + return __getDynamicResourceHeap<T>($(bindlessOption.enumType).Texture_Read)[((uint2)handleValue).x]; + else + return __getDynamicResourceHeap<T>($(bindlessOption.enumType).Texture_ReadWrite)[((uint2)handleValue).x]; + } + case DescriptorKind.TexelBuffer: + { + if(DescriptorAccess.Read == T.descriptorAccess) + return __getDynamicResourceHeap<T>($(bindlessOption.enumType).TexelBuffer_Read)[((uint2)handleValue).x]; + else + return __getDynamicResourceHeap<T>($(bindlessOption.enumType).TexelBuffer_ReadWrite)[((uint2)handleValue).x]; + } + case DescriptorKind.Buffer: + { + if(DescriptorAccess.Read == T.descriptorAccess) + return __getDynamicResourceHeap<T>($(bindlessOption.enumType).Buffer_Read)[((uint2)handleValue).x]; + else + return __getDynamicResourceHeap<T>($(bindlessOption.enumType).Buffer_ReadWrite)[((uint2)handleValue).x]; + } + case DescriptorKind.AccelerationStructure: + return __slang_noop_cast<T>(RaytracingAccelerationStructure(__asuint64((uint2)handleValue))); + default: + return __getDynamicResourceHeap<T>($(bindlessOption.enumType).Unknown)[((uint2)handleValue).x]; + } + $(bindlessOptionIfElsePost.toString()) +${{{{ + } +} +}}}} + default: + { + static_assert(false, "Impossible to end up here unless something went very wrong"); + return __getDynamicResourceHeap<T>()[((uint2)handleValue).x]; + } + } case wgsl: return __getDynamicResourceHeap<T>()[((uint2)handleValue).x]; default: diff --git a/source/slang/slang-ir-lower-dynamic-resource-heap.cpp b/source/slang/slang-ir-lower-dynamic-resource-heap.cpp index 60ec688a0..b2199a174 100644 --- a/source/slang/slang-ir-lower-dynamic-resource-heap.cpp +++ b/source/slang/slang-ir-lower-dynamic-resource-heap.cpp @@ -58,10 +58,11 @@ UInt findUnusedSpaceIndex(TargetProgram* targetProgram, IRModule* module) return index; } -IRVarLayout* createResourceHeapVarLayoutWithSpace( +IRVarLayout* createResourceHeapVarLayoutWithSpaceAndBinding( IRBuilder& builder, IRInst* param, - UInt spaceIndex) + UInt spaceIndex, + UInt bindingIndex) { SLANG_UNUSED(param); IRTypeLayout::Builder typeLayoutBuilder(&builder); @@ -71,7 +72,8 @@ IRVarLayout* createResourceHeapVarLayoutWithSpace( auto typeLayout = typeLayoutBuilder.build(); IRVarLayout::Builder varLayoutBuilder(&builder, typeLayout); varLayoutBuilder.findOrAddResourceInfo(LayoutResourceKind::RegisterSpace)->offset = spaceIndex; - varLayoutBuilder.findOrAddResourceInfo(LayoutResourceKind::DescriptorTableSlot)->offset = 0; + varLayoutBuilder.findOrAddResourceInfo(LayoutResourceKind::DescriptorTableSlot)->offset = + bindingIndex; return varLayoutBuilder.build(); } @@ -93,8 +95,14 @@ void lowerDynamicResourceHeap(TargetProgram* targetProgram, IRModule* module, Di IRBuilder builder(inst); builder.setInsertBefore(inst); + auto bindingIndex = (UInt)as<IRIntLit>(inst->getOperand(0))->getValue(); + auto param = builder.createGlobalParam(arrayType); - auto varLayout = createResourceHeapVarLayoutWithSpace(builder, param, unusedSpaceIndex); + auto varLayout = createResourceHeapVarLayoutWithSpaceAndBinding( + builder, + param, + unusedSpaceIndex, + bindingIndex); builder.addLayoutDecoration(param, varLayout); builder.addNameHintDecoration(param, toSlice("__slang_resource_heap")); inst->replaceUsesWith(param); |
