Skip to content

Commit ab3ea69

Browse files
csyonghedjohansson
authored andcommitted
Parameter layout and reflection for Metal bindings. (shader-slang#4022)
1 parent 0a45172 commit ab3ea69

14 files changed

+374
-103
lines changed

slang.h

+11
Original file line numberDiff line numberDiff line change
@@ -2259,9 +2259,16 @@ extern "C"
22592259
// The input_attachment_index subpass occupancy tracker
22602260
SLANG_PARAMETER_CATEGORY_SUBPASS,
22612261

2262+
// Metal resource binding points.
2263+
SLANG_PARAMETER_CATEGORY_METAL_ARGUMENT_BUFFER_ELEMENT,
2264+
22622265
//
22632266
SLANG_PARAMETER_CATEGORY_COUNT,
22642267

2268+
// Aliases for Metal-specific categories.
2269+
SLANG_PARAMETER_CATEGORY_METAL_BUFFER = SLANG_PARAMETER_CATEGORY_CONSTANT_BUFFER,
2270+
SLANG_PARAMETER_CATEGORY_METAL_TEXTURE = SLANG_PARAMETER_CATEGORY_SHADER_RESOURCE,
2271+
SLANG_PARAMETER_CATEGORY_METAL_SAMPLER = SLANG_PARAMETER_CATEGORY_SAMPLER_STATE,
22652272

22662273
// DEPRECATED:
22672274
SLANG_PARAMETER_CATEGORY_VERTEX_INPUT = SLANG_PARAMETER_CATEGORY_VARYING_INPUT,
@@ -2825,6 +2832,10 @@ namespace slang
28252832

28262833
InputAttachmentIndex = SLANG_PARAMETER_CATEGORY_SUBPASS,
28272834

2835+
MetalBuffer = SLANG_PARAMETER_CATEGORY_CONSTANT_BUFFER,
2836+
MetalTexture = SLANG_PARAMETER_CATEGORY_METAL_TEXTURE,
2837+
MetalArgumentBufferElement = SLANG_PARAMETER_CATEGORY_METAL_ARGUMENT_BUFFER_ELEMENT,
2838+
28282839
// DEPRECATED:
28292840
VertexInput = SLANG_PARAMETER_CATEGORY_VERTEX_INPUT,
28302841
FragmentOutput = SLANG_PARAMETER_CATEGORY_FRAGMENT_OUTPUT,

source/slang/slang-compiler.h

+3
Original file line numberDiff line numberDiff line change
@@ -1688,6 +1688,9 @@ namespace Slang
16881688
/// Are we generating code for a D3D API?
16891689
bool isD3DTarget(TargetRequest* targetReq);
16901690

1691+
// Are we generating code for Metal?
1692+
bool isMetalTarget(TargetRequest* targetReq);
1693+
16911694
/// Are we generating code for a Khronos API (OpenGL or Vulkan)?
16921695
bool isKhronosTarget(TargetRequest* targetReq);
16931696

source/slang/slang-emit-metal.cpp

+27-47
Original file line numberDiff line numberDiff line change
@@ -37,44 +37,6 @@ void MetalSourceEmitter::_emitHLSLDecorationSingleInt(const char* name, IRFunc*
3737
m_writer->emit(")]]\n");
3838
}
3939

40-
void MetalSourceEmitter::_emitHLSLRegisterSemantic(LayoutResourceKind kind, EmitVarChain* chain, IRInst* inst, char const* uniformSemanticSpelling)
41-
{
42-
// Metal does not use explicit binding.
43-
SLANG_UNUSED(kind);
44-
SLANG_UNUSED(chain);
45-
SLANG_UNUSED(inst);
46-
SLANG_UNUSED(uniformSemanticSpelling);
47-
}
48-
49-
void MetalSourceEmitter::_emitHLSLRegisterSemantics(EmitVarChain* chain, IRInst* inst, char const* uniformSemanticSpelling)
50-
{
51-
// TODO: implement.
52-
SLANG_UNUSED(chain);
53-
SLANG_UNUSED(inst);
54-
SLANG_UNUSED(uniformSemanticSpelling);
55-
}
56-
57-
void MetalSourceEmitter::_emitHLSLRegisterSemantics(IRVarLayout* varLayout, IRInst* inst, char const* uniformSemanticSpelling)
58-
{
59-
// TODO: implement.
60-
SLANG_UNUSED(varLayout);
61-
SLANG_UNUSED(inst);
62-
SLANG_UNUSED(uniformSemanticSpelling);
63-
}
64-
65-
void MetalSourceEmitter::_emitHLSLParameterGroupFieldLayoutSemantics(EmitVarChain* chain)
66-
{
67-
// TODO: implement.
68-
SLANG_UNUSED(chain);
69-
}
70-
71-
void MetalSourceEmitter::_emitHLSLParameterGroupFieldLayoutSemantics(IRVarLayout* fieldLayout, EmitVarChain* inChain)
72-
{
73-
// TODO: implement.
74-
SLANG_UNUSED(fieldLayout);
75-
SLANG_UNUSED(inChain);
76-
}
77-
7840
void MetalSourceEmitter::_emitHLSLParameterGroup(IRGlobalParam* varDecl, IRUniformParameterGroupType* type)
7941
{
8042
// Metal does not allow shader parameters declared as global variables, so we shouldn't see this.
@@ -139,17 +101,34 @@ void MetalSourceEmitter::_emitHLSLTextureType(IRTextureTypeBase* texType)
139101
m_writer->emit(">");
140102
}
141103

142-
void MetalSourceEmitter::_emitHLSLSubpassInputType(IRSubpassInputType* subpassType)
143-
{
144-
SLANG_UNUSED(subpassType);
145-
}
146-
147-
void MetalSourceEmitter::emitLayoutSemanticsImpl(IRInst* inst, char const* uniformSemanticSpelling)
104+
void MetalSourceEmitter::emitFuncParamLayoutImpl(IRInst* param)
148105
{
149-
auto layout = getVarLayout(inst);
150-
if (layout)
106+
auto layoutDecoration = param->findDecoration<IRLayoutDecoration>();
107+
if (!layoutDecoration)
108+
return;
109+
auto layout = as<IRVarLayout>(layoutDecoration->getLayout());
110+
if (!layout)
111+
return;
112+
for (auto rr : layout->getOffsetAttrs())
151113
{
152-
_emitHLSLRegisterSemantics(layout, inst, uniformSemanticSpelling);
114+
switch (rr->getResourceKind())
115+
{
116+
case LayoutResourceKind::MetalTexture:
117+
m_writer->emit(" [[texture(");
118+
m_writer->emit(rr->getOffset());
119+
m_writer->emit(")]]");
120+
break;
121+
case LayoutResourceKind::MetalBuffer:
122+
m_writer->emit(" [[buffer(");
123+
m_writer->emit(rr->getOffset());
124+
m_writer->emit(")]]");
125+
break;
126+
case LayoutResourceKind::SamplerState:
127+
m_writer->emit(" [[sampler(");
128+
m_writer->emit(rr->getOffset());
129+
m_writer->emit(")]]");
130+
break;
131+
}
153132
}
154133
}
155134

@@ -647,6 +626,7 @@ void MetalSourceEmitter::_emitStageAccessSemantic(IRStageAccessDecoration* decor
647626
void MetalSourceEmitter::emitSimpleFuncParamImpl(IRParam* param)
648627
{
649628
Super::emitSimpleFuncParamImpl(param);
629+
emitFuncParamLayoutImpl(param);
650630
}
651631

652632
static UnownedStringSlice _getInterpolationModifierText(IRInterpolationMode mode)

source/slang/slang-emit-metal.h

+2-11
Original file line numberDiff line numberDiff line change
@@ -23,7 +23,6 @@ class MetalSourceEmitter : public CLikeSourceEmitter
2323
protected:
2424
RefPtr<MetalExtensionTracker> m_extensionTracker;
2525

26-
virtual void emitLayoutSemanticsImpl(IRInst* inst, char const* uniformSemanticSpelling) SLANG_OVERRIDE;
2726
virtual void emitParameterGroupImpl(IRGlobalParam* varDecl, IRUniformParameterGroupType* type) SLANG_OVERRIDE;
2827
virtual void emitEntryPointAttributesImpl(IRFunc* irFunc, IREntryPointDecoration* entryPointDecor) SLANG_OVERRIDE;
2928

@@ -32,6 +31,7 @@ class MetalSourceEmitter : public CLikeSourceEmitter
3231
virtual void emitRateQualifiersAndAddressSpaceImpl(IRRate* rate, IRIntegerValue addressSpace) SLANG_OVERRIDE;
3332
virtual void emitSemanticsImpl(IRInst* inst, bool allowOffsets) SLANG_OVERRIDE;
3433
virtual void emitSimpleFuncParamImpl(IRParam* param) SLANG_OVERRIDE;
34+
3535
virtual void emitInterpolationModifiersImpl(IRInst* varInst, IRType* valueType, IRVarLayout* layout) SLANG_OVERRIDE;
3636
virtual void emitPackOffsetModifier(IRInst* varInst, IRType* valueType, IRPackOffsetDecoration* decoration) SLANG_OVERRIDE;
3737

@@ -56,16 +56,7 @@ class MetalSourceEmitter : public CLikeSourceEmitter
5656

5757
virtual bool doesTargetSupportPtrTypes() SLANG_OVERRIDE { return true; }
5858

59-
// Emit a single `register` semantic, as appropriate for a given resource-type-specific layout info
60-
// Keyword to use in the uniform case (`register` for globals, `packoffset` inside a `cbuffer`)
61-
void _emitHLSLRegisterSemantic(LayoutResourceKind kind, EmitVarChain* chain, IRInst* inst, char const* uniformSemanticSpelling = "register");
62-
63-
// Emit all the `register` semantics that are appropriate for a particular variable layout
64-
void _emitHLSLRegisterSemantics(EmitVarChain* chain, IRInst* inst, char const* uniformSemanticSpelling = "register");
65-
void _emitHLSLRegisterSemantics(IRVarLayout* varLayout, IRInst* inst, char const* uniformSemanticSpelling = "register");
66-
67-
void _emitHLSLParameterGroupFieldLayoutSemantics(EmitVarChain* chain);
68-
void _emitHLSLParameterGroupFieldLayoutSemantics(IRVarLayout* fieldLayout, EmitVarChain* inChain);
59+
void emitFuncParamLayoutImpl(IRInst* param);
6960

7061
void _emitHLSLParameterGroup(IRGlobalParam* varDecl, IRUniformParameterGroupType* type);
7162

source/slang/slang-emit.cpp

+10-3
Original file line numberDiff line numberDiff line change
@@ -583,6 +583,7 @@ Result linkAndOptimizeIR(
583583
// will have (more) legal shader code.
584584
//
585585
legalizeExistentialTypeLayout(
586+
targetProgram,
586587
irModule,
587588
sink);
588589

@@ -603,6 +604,7 @@ Result linkAndOptimizeIR(
603604
// then become multiple variables/parameters/arguments/etc.
604605
//
605606
legalizeResourceTypes(
607+
targetProgram,
606608
irModule,
607609
sink);
608610

@@ -617,6 +619,7 @@ Result linkAndOptimizeIR(
617619
// On CPU/CUDA targets, we simply elminate any empty types if
618620
// they are not part of public interface.
619621
legalizeEmptyTypes(
622+
targetProgram,
620623
irModule,
621624
sink);
622625
}
@@ -876,6 +879,7 @@ Result linkAndOptimizeIR(
876879
case CodeGenTarget::GLSL:
877880
case CodeGenTarget::SPIRV:
878881
case CodeGenTarget::SPIRVAssembly:
882+
case CodeGenTarget::Metal:
879883
moveGlobalVarInitializationToEntryPoints(irModule);
880884
break;
881885
case CodeGenTarget::CPPSource:
@@ -941,9 +945,12 @@ Result linkAndOptimizeIR(
941945

942946
if (options.shouldLegalizeExistentialAndResourceTypes)
943947
{
944-
// We need to lower any types used in a buffer resource (e.g. ContantBuffer or StructuredBuffer) into
945-
// a simple storage type that has target independent layout based on the kind of buffer resource.
946-
lowerBufferElementTypeToStorageType(targetProgram, irModule);
948+
if (!isMetalTarget(targetRequest))
949+
{
950+
// We need to lower any types used in a buffer resource (e.g. ContantBuffer or StructuredBuffer) into
951+
// a simple storage type that has target independent layout based on the kind of buffer resource.
952+
lowerBufferElementTypeToStorageType(targetProgram, irModule);
953+
}
947954
}
948955

949956
// Rewrite functions that return arrays to return them via `out` parameter,

source/slang/slang-ir-legalize-types.cpp

+15-10
Original file line numberDiff line numberDiff line change
@@ -95,8 +95,11 @@ LegalVal LegalVal::wrappedBuffer(
9595
//
9696

9797
IRTypeLegalizationContext::IRTypeLegalizationContext(
98+
TargetProgram* target,
9899
IRModule* inModule)
99100
{
101+
targetProgram = target;
102+
100103
session = inModule->getSession();
101104
module = inModule;
102105

@@ -3868,8 +3871,8 @@ static void legalizeTypes(
38683871
//
38693872
struct IRResourceTypeLegalizationContext : IRTypeLegalizationContext
38703873
{
3871-
IRResourceTypeLegalizationContext(IRModule* module)
3872-
: IRTypeLegalizationContext(module)
3874+
IRResourceTypeLegalizationContext(TargetProgram* target, IRModule* module)
3875+
: IRTypeLegalizationContext(target, module)
38733876
{}
38743877

38753878
bool isSpecialType(IRType* type) override
@@ -3903,8 +3906,8 @@ struct IRResourceTypeLegalizationContext : IRTypeLegalizationContext
39033906
//
39043907
struct IRExistentialTypeLegalizationContext : IRTypeLegalizationContext
39053908
{
3906-
IRExistentialTypeLegalizationContext(IRModule* module)
3907-
: IRTypeLegalizationContext(module)
3909+
IRExistentialTypeLegalizationContext(TargetProgram* target, IRModule* module)
3910+
: IRTypeLegalizationContext(target, module)
39083911
{}
39093912

39103913
bool isSpecialType(IRType* inType) override
@@ -3944,8 +3947,8 @@ struct IRExistentialTypeLegalizationContext : IRTypeLegalizationContext
39443947
// a public function signature.
39453948
struct IREmptyTypeLegalizationContext : IRTypeLegalizationContext
39463949
{
3947-
IREmptyTypeLegalizationContext(IRModule* module)
3948-
: IRTypeLegalizationContext(module)
3950+
IREmptyTypeLegalizationContext(TargetProgram* target, IRModule* module)
3951+
: IRTypeLegalizationContext(target, module)
39493952
{}
39503953

39513954
bool isSpecialType(IRType*) override
@@ -3985,18 +3988,20 @@ struct IREmptyTypeLegalizationContext : IRTypeLegalizationContext
39853988
// specialized context type to use to get the job done.
39863989

39873990
void legalizeResourceTypes(
3991+
TargetProgram* target,
39883992
IRModule* module,
39893993
DiagnosticSink* sink)
39903994
{
39913995
SLANG_PROFILE;
39923996

39933997
SLANG_UNUSED(sink);
39943998

3995-
IRResourceTypeLegalizationContext context(module);
3999+
IRResourceTypeLegalizationContext context(target, module);
39964000
legalizeTypes(&context);
39974001
}
39984002

39994003
void legalizeExistentialTypeLayout(
4004+
TargetProgram* target,
40004005
IRModule* module,
40014006
DiagnosticSink* sink)
40024007
{
@@ -4005,15 +4010,15 @@ void legalizeExistentialTypeLayout(
40054010
SLANG_UNUSED(module);
40064011
SLANG_UNUSED(sink);
40074012

4008-
IRExistentialTypeLegalizationContext context(module);
4013+
IRExistentialTypeLegalizationContext context(target, module);
40094014
legalizeTypes(&context);
40104015
}
40114016

4012-
void legalizeEmptyTypes(IRModule* module, DiagnosticSink* sink)
4017+
void legalizeEmptyTypes(TargetProgram* target, IRModule* module, DiagnosticSink* sink)
40134018
{
40144019
SLANG_UNUSED(sink);
40154020

4016-
IREmptyTypeLegalizationContext context(module);
4021+
IREmptyTypeLegalizationContext context(target, module);
40174022
legalizeTypes(&context);
40184023
}
40194024

source/slang/slang-ir-wrap-global-context.cpp

+1-2
Original file line numberDiff line numberDiff line change
@@ -65,8 +65,7 @@ namespace Slang
6565
{
6666
auto newParam = builder.createParam(globalParam.first->getFullType());
6767
newParam->insertBefore(paramInsertPoint);
68-
if (auto name = findNameHint(globalParam.first))
69-
builder.addNameHintDecoration(newParam, name);
68+
globalParam.first->transferDecorationsTo(newParam);
7069
newParams.add({newParam, globalParam.second});
7170
}
7271

source/slang/slang-legalize-types.cpp

+26-15
Original file line numberDiff line numberDiff line change
@@ -1156,23 +1156,34 @@ LegalType legalizeTypeImpl(
11561156
auto originalElementType = uniformBufferType->getElementType();
11571157

11581158
// Legalize the element type to see what we are working with.
1159-
auto legalElementType = legalizeType(context,
1160-
originalElementType);
1161-
1162-
// As a bit of a corner case, if the user requested something
1163-
// like `ConstantBuffer<Texture2D>` the element type would
1164-
// legalize to a "simple" type, and that would be interpreted
1165-
// as an *ordinary* type, but we really need to notice the
1166-
// case when the element type is simple, but *special*.
1167-
//
1168-
if( context->isSpecialType(originalElementType) )
1159+
LegalType legalElementType;
1160+
1161+
if (isMetalTarget(context->targetProgram->getTargetReq()) &&
1162+
as<IRParameterBlockType>(uniformBufferType))
1163+
{
1164+
// On Metal, we do not need to legalize the element type of
1165+
// a parameter block because we can translate it directly into
1166+
// an argument buffer.
1167+
legalElementType = LegalType::simple(originalElementType);
1168+
}
1169+
else
11691170
{
1170-
// Anything that has a special element type needs to
1171-
// be handled by the pass-specific logic in the context.
1171+
legalElementType = legalizeType(context, originalElementType);
1172+
// As a bit of a corner case, if the user requested something
1173+
// like `ConstantBuffer<Texture2D>` the element type would
1174+
// legalize to a "simple" type, and that would be interpreted
1175+
// as an *ordinary* type, but we really need to notice the
1176+
// case when the element type is simple, but *special*.
11721177
//
1173-
return context->createLegalUniformBufferType(
1174-
uniformBufferType->getOp(),
1175-
legalElementType);
1178+
if( context->isSpecialType(originalElementType) )
1179+
{
1180+
// Anything that has a special element type needs to
1181+
// be handled by the pass-specific logic in the context.
1182+
//
1183+
return context->createLegalUniformBufferType(
1184+
uniformBufferType->getOp(),
1185+
legalElementType);
1186+
}
11761187
}
11771188

11781189
// Note that even when legalElementType.flavor == Simple

source/slang/slang-legalize-types.h

+5-1
Original file line numberDiff line numberDiff line change
@@ -602,10 +602,11 @@ struct IRTypeLegalizationContext
602602
Session* session;
603603
IRModule* module;
604604
IRBuilder* builder;
605-
605+
TargetProgram* targetProgram;
606606
IRBuilder builderStorage;
607607

608608
IRTypeLegalizationContext(
609+
TargetProgram* target,
609610
IRModule* inModule);
610611

611612
// When inserting new globals, put them before this one.
@@ -710,14 +711,17 @@ LegalType createLegalUniformBufferTypeForExistentials(
710711

711712

712713
void legalizeExistentialTypeLayout(
714+
TargetProgram* target,
713715
IRModule* module,
714716
DiagnosticSink* sink);
715717

716718
void legalizeResourceTypes(
719+
TargetProgram* target,
717720
IRModule* module,
718721
DiagnosticSink* sink);
719722

720723
void legalizeEmptyTypes(
724+
TargetProgram* target,
721725
IRModule* module,
722726
DiagnosticSink* sink);
723727

0 commit comments

Comments
 (0)