Skip to content

Commit ec745c0

Browse files
author
Tim Foley
authored
Represent global shader parameters explicitly in the IR (shader-slang#756)
Before this change, global shader parameters were represented in the IR as just being ordinary global variables. The only indication that a particular global represented a parameter was when it got a layotu attached to it (as part of back-end processing), and we've had a number of bugs related to layouts being dropped so that what should have been a shader parameter turned into an ordinary global variable in the output. This change is more strongly motivated by the fact that making shader parameters look like globals means that we cannot easily reason about their value when doing IR transformations. If we see two `load`s from the same global variable can we assume they yield the same value? In the general case we cannot, and this means that any transformation that wants to rely on the fact that an input `Texture2D` shader parameter can't actually change over the life of the program needs to do extra work. The fix here is to introduce a new kind of IR instruction that represents a global shader parameter directly (not a pointer to it as a global would), at which point there isn't even such a notion as a "load" from the parameter, since it represents the value directly. In several cases logic that used to apply to global variables in case they were shader parameters (by looking for a layout) is now moved to apply to these global parameters. The biggest source of issues in this change was that switching from pointers to plain values to represent these shader parameters stresses different cases in type legalization. I also had to deal with the case of legalization for GLSL where we actually *do* need global shader parameters that are writable (since varying output goes in the global scope), but in that case I borrowed the use of pointer-like `Out<...>` and `InOut<...>` types to represent that intent, which we were already using for function parameters representing outputs. A few tests started failing because the changes lead to a slightly different order of code emission, which in some HLSL tests resulted in a function parameter named `s` getting emitted before a global parameter named `s`, leading to the latter getting the name `s_1` instead of `s_0`. A few SPIR-V tests started failing because the new approach means that we no longer end up performing a load from all varying input parameters at the start of `main` and instead reference the varying inputs directly. The resulting code is more idomatic, but it differed from the baselines for those tests.
1 parent 11793ed commit ec745c0

14 files changed

+496
-176
lines changed

source/slang/check.h

+7
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,7 @@
1+
// check.h
2+
#pragma once
3+
4+
namespace Slang
5+
{
6+
bool isGlobalShaderParameter(VarDeclBase* decl);
7+
}

source/slang/emit.cpp

+77-33
Original file line numberDiff line numberDiff line change
@@ -2355,6 +2355,7 @@ struct EmitVisitor
23552355
case kIROp_Var:
23562356
case kIROp_GlobalVar:
23572357
case kIROp_GlobalConstant:
2358+
case kIROp_GlobalParam:
23582359
case kIROp_Param:
23592360
return false;
23602361

@@ -5558,7 +5559,7 @@ struct EmitVisitor
55585559

55595560
void emitHLSLParameterGroup(
55605561
EmitContext* ctx,
5561-
IRGlobalVar* varDecl,
5562+
IRGlobalParam* varDecl,
55625563
IRUniformParameterGroupType* type)
55635564
{
55645565
if(as<IRTextureBufferType>(type))
@@ -5623,7 +5624,7 @@ struct EmitVisitor
56235624

56245625
void emitGLSLParameterGroup(
56255626
EmitContext* ctx,
5626-
IRGlobalVar* varDecl,
5627+
IRGlobalParam* varDecl,
56275628
IRUniformParameterGroupType* type)
56285629
{
56295630
auto varLayout = getVarLayout(ctx, varDecl);
@@ -5680,14 +5681,14 @@ struct EmitVisitor
56805681

56815682
// If the underlying variable was an array (or array of arrays, etc.)
56825683
// we need to emit all those array brackets here.
5683-
emitArrayBrackets(ctx, varDecl->getDataType()->getValueType());
5684+
emitArrayBrackets(ctx, varDecl->getDataType());
56845685

56855686
emit(";\n");
56865687
}
56875688

56885689
void emitIRParameterGroup(
56895690
EmitContext* ctx,
5690-
IRGlobalVar* varDecl,
5691+
IRGlobalParam* varDecl,
56915692
IRUniformParameterGroupType* type)
56925693
{
56935694
switch (ctx->shared->target)
@@ -5763,7 +5764,7 @@ struct EmitVisitor
57635764

57645765
void emitIRStructuredBuffer_GLSL(
57655766
EmitContext* ctx,
5766-
IRGlobalVar* varDecl,
5767+
IRGlobalParam* varDecl,
57675768
IRHLSLStructuredBufferTypeBase* structuredBufferType)
57685769
{
57695770
// Shader storage buffer is an OpenGL 430 feature
@@ -5809,14 +5810,14 @@ struct EmitVisitor
58095810
emit("} ");
58105811

58115812
emit(getIRName(varDecl));
5812-
emitArrayBrackets(ctx, varDecl->getDataType()->getValueType());
5813+
emitArrayBrackets(ctx, varDecl->getDataType());
58135814

58145815
emit(";\n");
58155816
}
58165817

58175818
void emitIRByteAddressBuffer_GLSL(
58185819
EmitContext* ctx,
5819-
IRGlobalVar* varDecl,
5820+
IRGlobalParam* varDecl,
58205821
IRByteAddressBufferTypeBase* /* byteAddressBufferType */)
58215822
{
58225823
// TODO: A lot of this logic is copy-pasted from `emitIRStructuredBuffer_GLSL`.
@@ -5862,7 +5863,7 @@ struct EmitVisitor
58625863
emit("} ");
58635864

58645865
emit(getIRName(varDecl));
5865-
emitArrayBrackets(ctx, varDecl->getDataType()->getValueType());
5866+
emitArrayBrackets(ctx, varDecl->getDataType());
58665867

58675868
emit(";\n");
58685869
}
@@ -5894,6 +5895,63 @@ struct EmitVisitor
58945895
Emit("}\n");
58955896
}
58965897

5898+
// An ordinary global variable won't have a layout
5899+
// associated with it, since it is not a shader
5900+
// parameter.
5901+
//
5902+
SLANG_ASSERT(!getVarLayout(ctx, varDecl));
5903+
VarLayout* layout = nullptr;
5904+
5905+
// An ordinary global variable (which is not a
5906+
// shader parameter) may need special
5907+
// modifiers to indicate it as such.
5908+
//
5909+
switch (getTarget(ctx))
5910+
{
5911+
case CodeGenTarget::HLSL:
5912+
// HLSL requires the `static` modifier on any
5913+
// global variables; otherwise they are assumed
5914+
// to be uniforms.
5915+
Emit("static ");
5916+
break;
5917+
5918+
default:
5919+
break;
5920+
}
5921+
5922+
emitIRVarModifiers(ctx, layout, varDecl, varType);
5923+
5924+
emitIRRateQualifiers(ctx, varDecl);
5925+
emitIRType(ctx, varType, getIRName(varDecl));
5926+
5927+
// TODO: These shouldn't be needed for ordinary
5928+
// global variables.
5929+
//
5930+
emitIRSemantics(ctx, varDecl);
5931+
emitIRLayoutSemantics(ctx, varDecl);
5932+
5933+
if (varDecl->getFirstBlock())
5934+
{
5935+
Emit(" = ");
5936+
emit(initFuncName);
5937+
Emit("()");
5938+
}
5939+
5940+
emit(";\n\n");
5941+
}
5942+
5943+
void emitIRGlobalParam(
5944+
EmitContext* ctx,
5945+
IRGlobalParam* varDecl)
5946+
{
5947+
auto rawType = varDecl->getDataType();
5948+
5949+
auto varType = rawType;
5950+
if( auto outType = as<IROutTypeBase>(varType) )
5951+
{
5952+
varType = outType->getValueType();
5953+
}
5954+
58975955
// When a global shader parameter represents a "parameter group"
58985956
// (either a constant buffer or a parameter block with non-resource
58995957
// data in it), we will prefer to emit it as an ordinary `cbuffer`
@@ -5985,26 +6043,11 @@ struct EmitVisitor
59856043

59866044
// Need to emit appropriate modifiers here.
59876045

6046+
// We expect/require all shader parameters to
6047+
// have some kind of layout information associted with them.
6048+
//
59886049
auto layout = getVarLayout(ctx, varDecl);
5989-
5990-
if (!layout)
5991-
{
5992-
// A global variable without a layout is just an
5993-
// ordinary global variable, and may need special
5994-
// modifiers to indicate it as such.
5995-
switch (getTarget(ctx))
5996-
{
5997-
case CodeGenTarget::HLSL:
5998-
// HLSL requires the `static` modifier on any
5999-
// global variables; otherwise they are assumed
6000-
// to be uniforms.
6001-
Emit("static ");
6002-
break;
6003-
6004-
default:
6005-
break;
6006-
}
6007-
}
6050+
SLANG_ASSERT(layout);
60086051

60096052
emitIRVarModifiers(ctx, layout, varDecl, varType);
60106053

@@ -6015,16 +6058,13 @@ struct EmitVisitor
60156058

60166059
emitIRLayoutSemantics(ctx, varDecl);
60176060

6018-
if (varDecl->getFirstBlock())
6019-
{
6020-
Emit(" = ");
6021-
emit(initFuncName);
6022-
Emit("()");
6023-
}
6061+
// A shader parameter cannot have an initializer,
6062+
// so we do need to consider emitting one here.
60246063

60256064
emit(";\n\n");
60266065
}
60276066

6067+
60286068
void emitIRGlobalConstantInitializer(
60296069
EmitContext* ctx,
60306070
IRGlobalConstant* valDecl)
@@ -6098,6 +6138,10 @@ struct EmitVisitor
60986138
emitIRGlobalVar(ctx, (IRGlobalVar*) inst);
60996139
break;
61006140

6141+
case kIROp_GlobalParam:
6142+
emitIRGlobalParam(ctx, (IRGlobalParam*) inst);
6143+
break;
6144+
61016145
case kIROp_GlobalConstant:
61026146
emitIRGlobalConstant(ctx, (IRGlobalConstant*) inst);
61036147
break;

source/slang/ir-inst-defs.h

+2
Original file line numberDiff line numberDiff line change
@@ -163,6 +163,8 @@ INST_RANGE(Type, VoidType, StructType)
163163
INST(GlobalConstant, global_constant, 0, 0)
164164
INST_RANGE(GlobalValueWithCode, Func, GlobalConstant)
165165

166+
INST(GlobalParam, global_param, 0, 0)
167+
166168
INST(StructKey, key, 0, 0)
167169
INST(GlobalGenericParam, global_generic_param, 0, 0)
168170
INST(WitnessTable, witness_table, 0, 0)

source/slang/ir-insts.h

+8
Original file line numberDiff line numberDiff line change
@@ -558,6 +558,12 @@ struct IRGlobalConstant : IRGlobalValueWithCode
558558
IR_LEAF_ISA(GlobalConstant)
559559
};
560560

561+
struct IRGlobalParam : IRInst
562+
{
563+
IR_LEAF_ISA(GlobalParam)
564+
};
565+
566+
561567
// An entry in a witness table (see below)
562568
struct IRWitnessTableEntry : IRInst
563569
{
@@ -798,6 +804,8 @@ struct IRBuilder
798804
IRType* valueType);
799805
IRGlobalConstant* createGlobalConstant(
800806
IRType* valueType);
807+
IRGlobalParam* createGlobalParam(
808+
IRType* valueType);
801809
IRWitnessTable* createWitnessTable();
802810
IRWitnessTableEntry* createWitnessTableEntry(
803811
IRWitnessTable* witnessTable,

0 commit comments

Comments
 (0)