Skip to content

Commit fdf061e

Browse files
authored
Add datalayout for constant buffers. (shader-slang#5608)
* Add datalayout for constant buffers. * Fixes. * Fix test. * Fix glsl codegen. * Update spirv-specific doc. * Fix test. * Fix binding in the presense of specialization constants. * address comments. * Add a test for constant buffer layout.
1 parent dcc7c6f commit fdf061e

34 files changed

+515
-152
lines changed

docs/user-guide/a2-01-spirv-target-specific.md

+74-73
Original file line numberDiff line numberDiff line change
@@ -68,7 +68,7 @@ The system-value semantics are translated to the following SPIR-V code.
6868
| `SV_ViewID` | `BuiltIn ViewIndex` |
6969
| `SV_ViewportArrayIndex` | `BuiltIn ViewportIndex` |
7070

71-
*Note* that `SV_PointSize` is a unique keyword that HLSL doesn't have.
71+
*Note* that `SV_PointSize` is a Slang-specific semantic that is not defined in HLSL.
7272

7373

7474
Behavior of `discard` after SPIR-V 1.6
@@ -131,27 +131,29 @@ SPIR-V 1.5 with [SPV_EXT_shader_atomic_float16_add](https://github.com/KhronosGr
131131
| SPIR-V | Yes | Yes | Yes (SPV1.5+ext) | Yes (SPV1.5+ext) | Yes (SPV1.5+ext) |
132132

133133

134-
ConstantBuffer, (RW/RasterizerOrdered)StructuredBuffer, (RW/RasterizerOrdered)ByteAddressBuffer
134+
ConstantBuffer, StructuredBuffer and ByteAddressBuffer
135135
-----------------------------------------------------------------------------------------------
136136

137137
Each member in a `ConstantBuffer` will be emitted as `uniform` parameter in a uniform block.
138-
StructuredBuffer and ByteAddressBuffer are translated to a shader storage buffer with `readonly` layout.
139-
RWStructuredBuffer and RWByteAddressBuffer are translated to a shader storage buffer with `read-write` layout.
138+
StructuredBuffer and ByteAddressBuffer are translated to a shader storage buffer with `readonly` access.
139+
RWStructuredBuffer and RWByteAddressBuffer are translated to a shader storage buffer with `read-write` access.
140140
RasterizerOrderedStructuredBuffer and RasterizerOrderedByteAddressBuffer will use an extension, `SPV_EXT_fragment_shader_interlock`.
141141

142-
If you need to apply a different buffer layout for indivisual StructuredBuffer, you can specify the layout as a second generic argument to StructuredBuffer. E.g., StructuredBuffer<T, Std140Layout>, StructuredBuffer<T, Std430Layout> or StructuredBuffer<T, ScalarLayout>.
142+
If you need to apply a different buffer layout for indivisual `ConstantBuffer` or `StructuredBuffer`, you can specify the layout as a second generic argument. E.g., `ConstantBuffer<T, Std430DataLayout>`, `StructuredBuffer<T, Std140DataLayout>`, `StructuredBuffer<T, Std430DataLayout>` or `StructuredBuffer<T, ScalarDataLayout>`.
143143

144-
Note that there are compiler options, "-fvk-use-scalar-layout" and "-force-glsl-scalar-layout".
144+
Note that there are compiler options, "-fvk-use-scalar-layout" / "-force-glsl-scalar-layout" and "-fvk-use-dx-layout".
145145
These options do the same but they are applied globally.
146146

147147

148148
ParameterBlock for SPIR-V target
149149
--------------------------------
150150

151151
`ParameterBlock` is a Slang generic type for binding uniform parameters.
152-
It is similar to `ConstantBuffer` in HLSL, and `ParameterBlock` can include not only constant parameters but also descriptors such as Texture2D or StructuredBuffer.
152+
In contrast to `ConstantBuffer`, a `ParameterBlock<T>` introduces a new descriptor set ID for resource/sampler handles defined in the element type `T`.
153153

154-
`ParameterBlock` is designed specifically for d3d/vulkan/metal, so that parameters are laid out more naturally on these platforms. For Vulkan, when a ParameterBlock doesn't contain nested parameter block fields, it always maps to a single descriptor set, with a dedicated set number and every resources is placed into the set with binding index starting from 0.
154+
`ParameterBlock` is designed specifically for D3D12/Vulkan/Metal/WebGPU, so that parameters defined in `T` can be placed into an independent descriptor table/descriptor set/argument buffer/binding group.
155+
156+
For example, when targeting Vulkan, when a ParameterBlock doesn't contain nested parameter block fields, it will always map to a single descriptor set, with a dedicated set number and every resources is placed into the set with binding index starting from 0. This allows the user application to create and pre-populate the descriptor set and reuse it during command encoding, without explicilty specifying the binding index for each individual parameter.
155157

156158
When both ordinary data fields and resource typed fields exist in a parameter block, all ordinary data fields will be grouped together into a uniform buffer and appear as a binding 0 of the resulting descriptor set.
157159

@@ -160,8 +162,8 @@ Push Constants
160162
---------------------
161163

162164
By default, a `uniform` parameter defined in the parameter list of an entrypoint function is translated to a push constant in SPIRV, if the type of the parameter is ordinary data type (no resources/textures).
163-
All `uniform` parameter defined in global scope are grouped together and placed in a default constant bbuffer. You can make a global uniform parameter laid out as a push constant by using the `[vk::push_constant]` attribute
164-
on the uniform parameter.
165+
All `uniform` parameters defined in global scope are grouped together and placed in a default constant buffer. You can make a global uniform parameter laid out as a push constant by using the `[vk::push_constant]` attribute
166+
on the uniform parameter. All push constants follow the std430 layout by default.
165167

166168
Specialization Constants
167169
------------------------
@@ -184,65 +186,6 @@ Alternatively, the GLSL `layout` syntax is also supported by Slang:
184186
layout(constant_id = 1) const int MyConst = 1;
185187
```
186188

187-
SPIR-V specific Compiler options
188-
--------------------------------
189-
190-
The following compiler options are specific to SPIR-V.
191-
192-
### -emit-spirv-directly
193-
Generate SPIR-V output directly (default)
194-
It cannot be used with -emit-spirv-via-glsl
195-
196-
### -emit-spirv-via-glsl
197-
Generate SPIR-V output by compiling to glsl source first, then use glslang compiler to produce SPIRV from the glsl.
198-
It cannot be used with -emit-spirv-directly
199-
200-
### -g
201-
Include debug information in the generated code, where possible.
202-
When targeting SPIR-V, this option emits [SPIR-V NonSemantic Shader DebugInfo Instructions](https://github.com/KhronosGroup/SPIRV-Registry/blob/main/nonsemantic/NonSemantic.Shader.DebugInfo.100.asciidoc).
203-
204-
### -O<optimization-level>
205-
Set the optimization level.
206-
Under `-O0` option, Slang will not perform extensive inlining for all functions calls, instead it will preserve the call graph as much as possible to help with understanding the SPIRV structure and diagnosing any downstream toolchain issues.
207-
208-
### -fvk-{b|s|t|u}-shift <N> <space>
209-
For example '-fvk-b-shift <N> <space>' shifts by N the inferred binding
210-
numbers for all resources in 'b' registers of space <space>. For a resource attached with :register(bX, <space>)
211-
but not [vk::binding(...)], sets its Vulkan descriptor set to <space> and binding number to X + N. If you need to
212-
shift the inferred binding numbers for more than one space, provide more than one such option. If more than one
213-
such option is provided for the same space, the last one takes effect. If you need to shift the inferred binding
214-
numbers for all sets, use 'all' as <space>.
215-
216-
For more information, see the following pages:
217-
- [DXC description](https://github.com/Microsoft/DirectXShaderCompiler/blob/main/docs/SPIR-V.rst#implicit-binding-number-assignment)
218-
- [GLSL wiki](https://github.com/KhronosGroup/glslang/wiki/HLSL-FAQ#auto-mapped-binding-numbers)
219-
220-
### -fvk-bind-globals <N> <descriptor-set>
221-
Places the $Globals cbuffer at descriptor set <descriptor-set> and binding <N>.
222-
It lets you specify the descriptor for the source at a certain register.
223-
224-
For more information, see the following pages:
225-
- [DXC description](https://github.com/Microsoft/DirectXShaderCompiler/blob/main/docs/SPIR-V.rst#hlsl-global-variables-and-vulkan-binding)
226-
227-
### -fvk-use-scalar-layout, -force-glsl-scalar-layout
228-
Make data accessed through ConstantBuffer, ParameterBlock, StructuredBuffer, ByteAddressBuffer and general pointers follow the 'scalar' layout when targeting GLSL or SPIRV.
229-
230-
### -fvk-use-gl-layout
231-
Use std430 layout instead of D3D buffer layout for raw buffer load/stores.
232-
233-
### -fvk-use-dx-layout
234-
Pack members using FXCs member packing rules when targeting GLSL or SPIRV.
235-
236-
### -fvk-use-entrypoint-name
237-
Uses the entrypoint name from the source instead of 'main' in the spirv output.
238-
239-
### -fspv-reflect
240-
Include reflection decorations in the resulting SPIRV for shader parameters.
241-
242-
### -spirv-core-grammar
243-
A path to a specific spirv.core.grammar.json to use when generating SPIR-V output
244-
245-
246189
SPIR-V specific Attributes
247190
--------------------------
248191

@@ -291,13 +234,13 @@ When there are more than one entry point, the default behavior will prevent a sh
291234
To generate a valid SPIR-V with multiple entry points, use `-fvk-use-entrypoint-name` compiler option to disable the renaming behavior and preserve the entry point names.
292235

293236

294-
Memory pointer is experimental
237+
Global memory pointers
295238
------------------------------
296239

297-
Slang supports memory pointers when targetting SPIRV. See [an example and explanation](convenience-features.html#pointers-limited).
240+
Slang supports global memory pointers when targetting SPIRV. See [an example and explanation](convenience-features.html#pointers-limited).
298241

299-
When a memory pointer points to a physical memory location, the pointer will be translated to a PhysicalStorageBuffer storage class in SPIRV.
300-
When a slang module uses a pointer, the resulting SPIRV will be using the SpvAddressingModelPhysicalStorageBuffer64 addressing mode. Modules with pointers but they don't point to a physical memory location will use SpvAddressingModelLogical addressing mode.
242+
`float4*` in user code will be translated to a pointer in PhysicalStorageBuffer storage class in SPIRV.
243+
When a slang module uses a pointer type, the resulting SPIRV will be using the SpvAddressingModelPhysicalStorageBuffer64 addressing mode. Modules without use of pointers will use SpvAddressingModelLogical addressing mode.
301244

302245

303246
Matrix type translation
@@ -409,4 +352,62 @@ void main() {
409352
This behavior is same to [how DXC translates Hull shader from HLSL to SPIR-V](https://github.com/Microsoft/DirectXShaderCompiler/blob/main/docs/SPIR-V.rst#patch-constant-function).
410353

411354

355+
SPIR-V specific Compiler options
356+
--------------------------------
357+
358+
The following compiler options are specific to SPIR-V.
359+
360+
### -emit-spirv-directly
361+
Generate SPIR-V output directly (default)
362+
It cannot be used with -emit-spirv-via-glsl
363+
364+
### -emit-spirv-via-glsl
365+
Generate SPIR-V output by compiling to glsl source first, then use glslang compiler to produce SPIRV from the glsl.
366+
It cannot be used with -emit-spirv-directly
367+
368+
### -g
369+
Include debug information in the generated code, where possible.
370+
When targeting SPIR-V, this option emits [SPIR-V NonSemantic Shader DebugInfo Instructions](https://github.com/KhronosGroup/SPIRV-Registry/blob/main/nonsemantic/NonSemantic.Shader.DebugInfo.100.asciidoc).
371+
372+
### -O<optimization-level>
373+
Set the optimization level.
374+
Under `-O0` option, Slang will not perform extensive inlining for all functions calls, instead it will preserve the call graph as much as possible to help with understanding the SPIRV structure and diagnosing any downstream toolchain issues.
375+
376+
### -fvk-{b|s|t|u}-shift <N> <space>
377+
For example '-fvk-b-shift <N> <space>' shifts by N the inferred binding
378+
numbers for all resources in 'b' registers of space <space>. For a resource attached with :register(bX, <space>)
379+
but not [vk::binding(...)], sets its Vulkan descriptor set to <space> and binding number to X + N. If you need to
380+
shift the inferred binding numbers for more than one space, provide more than one such option. If more than one
381+
such option is provided for the same space, the last one takes effect. If you need to shift the inferred binding
382+
numbers for all sets, use 'all' as <space>.
383+
384+
For more information, see the following pages:
385+
- [DXC description](https://github.com/Microsoft/DirectXShaderCompiler/blob/main/docs/SPIR-V.rst#implicit-binding-number-assignment)
386+
- [GLSL wiki](https://github.com/KhronosGroup/glslang/wiki/HLSL-FAQ#auto-mapped-binding-numbers)
387+
388+
### -fvk-bind-globals <N> <descriptor-set>
389+
Places the $Globals cbuffer at descriptor set <descriptor-set> and binding <N>.
390+
It lets you specify the descriptor for the source at a certain register.
391+
392+
For more information, see the following pages:
393+
- [DXC description](https://github.com/Microsoft/DirectXShaderCompiler/blob/main/docs/SPIR-V.rst#hlsl-global-variables-and-vulkan-binding)
394+
395+
### -fvk-use-scalar-layout, -force-glsl-scalar-layout
396+
Make data accessed through ConstantBuffer, ParameterBlock, StructuredBuffer, ByteAddressBuffer and general pointers follow the 'scalar' layout when targeting GLSL or SPIRV.
397+
398+
### -fvk-use-gl-layout
399+
Use std430 layout instead of D3D buffer layout for raw buffer load/stores.
400+
401+
### -fvk-use-dx-layout
402+
Pack members using FXCs member packing rules when targeting GLSL or SPIRV.
403+
404+
### -fvk-use-entrypoint-name
405+
Uses the entrypoint name from the source instead of 'main' in the spirv output.
406+
407+
### -fspv-reflect
408+
Include reflection decorations in the resulting SPIRV for shader parameters.
409+
410+
### -spirv-core-grammar
411+
A path to a specific spirv.core.grammar.json to use when generating SPIR-V output
412+
412413

source/slang/core.meta.slang

+1-1
Original file line numberDiff line numberDiff line change
@@ -1871,7 +1871,7 @@ for (int tt = 0; tt < kBaseTypeCount; ++tt)
18711871

18721872
//@ public:
18731873

1874-
__generic<T>
1874+
__generic<T, L:IBufferDataLayout = DefaultDataLayout>
18751875
__intrinsic_type($(kIROp_ConstantBufferType))
18761876
__magic_type(ConstantBufferType)
18771877
struct ConstantBuffer {}

source/slang/hlsl.meta.slang

+8-1
Original file line numberDiff line numberDiff line change
@@ -10,31 +10,36 @@ void __requireGLSLExtension(String extensionName);
1010
/// Represents an interface for buffer data layout.
1111
/// This interface is used as a base for defining specific data layouts for buffers.
1212
[sealed]
13+
__magic_type(IBufferDataLayoutType)
1314
interface IBufferDataLayout
1415
{
1516
}
1617

1718
/// @category misc_types
1819
__intrinsic_type($(kIROp_DefaultBufferLayoutType))
20+
__magic_type(DefaultDataLayoutType)
1921
struct DefaultDataLayout : IBufferDataLayout
2022
{};
2123

2224
/// @category misc_types
2325
__intrinsic_type($(kIROp_Std140BufferLayoutType))
2426
[require(spirv)]
2527
[require(glsl)]
28+
__magic_type(Std140DataLayoutType)
2629
struct Std140DataLayout : IBufferDataLayout
2730
{};
2831

2932
/// @category misc_types
3033
__intrinsic_type($(kIROp_Std430BufferLayoutType))
3134
[require(spirv)]
3235
[require(glsl)]
36+
__magic_type(Std430DataLayoutType)
3337
struct Std430DataLayout : IBufferDataLayout
3438
{};
3539

3640
/// @category misc_types
3741
__intrinsic_type($(kIROp_ScalarBufferLayoutType))
42+
__magic_type(ScalarDataLayoutType)
3843
struct ScalarDataLayout : IBufferDataLayout
3944
{};
4045

@@ -20588,14 +20593,16 @@ const char* kDynamicResourceCastableTypes[] = {
2058820593

2058920594
"SamplerState", "SamplerComparisonState",
2059020595

20591-
"ConstantBuffer<T>", "TextureBuffer<T>",
20596+
"ConstantBuffer<T, L>", "TextureBuffer<T>",
2059220597
};
2059320598

2059420599
for (auto typeName : kDynamicResourceCastableTypes) {
2059520600
auto kind = strstr(typeName, "Sampler") ? "Sampler" : "General";
2059620601

2059720602
if (strstr(typeName, "StructuredBuffer<T, L>"))
2059820603
sb << "__generic<T, L : IBufferDataLayout = DefaultDataLayout>\n";
20604+
else if (strstr(typeName, "ConstantBuffer<T, L>"))
20605+
sb << "__generic<T, L : IBufferDataLayout = DefaultDataLayout>\n";
2059920606
else if (strstr(typeName, "Buffer<T>"))
2060020607
sb << "__generic<T>\n";
2060120608
}}}}

source/slang/slang-ast-builder.cpp

+35-2
Original file line numberDiff line numberDiff line change
@@ -146,6 +146,16 @@ Type* SharedASTBuilder::getDiffInterfaceType()
146146
return m_diffInterfaceType;
147147
}
148148

149+
Type* SharedASTBuilder::getIBufferDataLayoutType()
150+
{
151+
if (!m_IBufferDataLayoutType)
152+
{
153+
auto decl = findMagicDecl("IBufferDataLayoutType");
154+
m_IBufferDataLayoutType = DeclRefType::create(m_astBuilder, makeDeclRef<Decl>(decl));
155+
}
156+
return m_IBufferDataLayoutType;
157+
}
158+
149159
Type* SharedASTBuilder::getErrorType()
150160
{
151161
if (!m_errorType)
@@ -296,6 +306,23 @@ PtrType* ASTBuilder::getPtrType(Type* valueType, AddressSpace addrSpace)
296306
return dynamicCast<PtrType>(getPtrType(valueType, addrSpace, "PtrType"));
297307
}
298308

309+
Type* ASTBuilder::getDefaultLayoutType()
310+
{
311+
return getSpecializedBuiltinType({}, "DefaultDataLayoutType");
312+
}
313+
Type* ASTBuilder::getStd140LayoutType()
314+
{
315+
return getSpecializedBuiltinType({}, "Std140DataLayoutType");
316+
}
317+
Type* ASTBuilder::getStd430LayoutType()
318+
{
319+
return getSpecializedBuiltinType({}, "Std430DataLayoutType");
320+
}
321+
Type* ASTBuilder::getScalarLayoutType()
322+
{
323+
return getSpecializedBuiltinType({}, "ScalarDataLayoutType");
324+
}
325+
299326
// Construct the type `Out<valueType>`
300327
OutType* ASTBuilder::getOutType(Type* valueType)
301328
{
@@ -358,9 +385,15 @@ ArrayExpressionType* ASTBuilder::getArrayType(Type* elementType, IntVal* element
358385
getSpecializedBuiltinType(makeArrayView(args), "ArrayExpressionType"));
359386
}
360387

361-
ConstantBufferType* ASTBuilder::getConstantBufferType(Type* elementType)
388+
ConstantBufferType* ASTBuilder::getConstantBufferType(
389+
Type* elementType,
390+
Type* layoutType,
391+
Val* layoutWitness)
362392
{
363-
return as<ConstantBufferType>(getSpecializedBuiltinType(elementType, "ConstantBufferType"));
393+
Val* args[] = {elementType, layoutType, layoutWitness};
394+
395+
return as<ConstantBufferType>(
396+
getSpecializedBuiltinType(makeArrayView(args), "ConstantBufferType"));
364397
}
365398

366399
ParameterBlockType* ASTBuilder::getParameterBlockType(Type* elementType)

source/slang/slang-ast-builder.h

+12-1
Original file line numberDiff line numberDiff line change
@@ -39,6 +39,8 @@ class SharedASTBuilder : public RefObject
3939
/// Get the `IDifferentiable` type
4040
Type* getDiffInterfaceType();
4141

42+
Type* getIBufferDataLayoutType();
43+
4244
Type* getErrorType();
4345
Type* getBottomType();
4446
Type* getInitializerListType();
@@ -89,6 +91,7 @@ class SharedASTBuilder : public RefObject
8991
Type* m_bottomType = nullptr;
9092
Type* m_initializerListType = nullptr;
9193
Type* m_overloadedType = nullptr;
94+
Type* m_IBufferDataLayoutType = nullptr;
9295

9396
// The following types are created lazily, such that part of their definition
9497
// can be in the core module.
@@ -482,6 +485,11 @@ class ASTBuilder : public RefObject
482485
Type* getSpecializedBuiltinType(Type* typeParam, const char* magicTypeName);
483486
Type* getSpecializedBuiltinType(ArrayView<Val*> genericArgs, const char* magicTypeName);
484487

488+
Type* getDefaultLayoutType();
489+
Type* getStd140LayoutType();
490+
Type* getStd430LayoutType();
491+
Type* getScalarLayoutType();
492+
485493
Type* getInitializerListType() { return m_sharedASTBuilder->getInitializerListType(); }
486494
Type* getOverloadedType() { return m_sharedASTBuilder->getOverloadedType(); }
487495
Type* getErrorType() { return m_sharedASTBuilder->getErrorType(); }
@@ -525,7 +533,10 @@ class ASTBuilder : public RefObject
525533
IntVal* colCount,
526534
IntVal* layout);
527535

528-
ConstantBufferType* getConstantBufferType(Type* elementType);
536+
ConstantBufferType* getConstantBufferType(
537+
Type* elementType,
538+
Type* layoutType,
539+
Val* layoutIsILayout);
529540

530541
ParameterBlockType* getParameterBlockType(Type* elementType);
531542

0 commit comments

Comments
 (0)