Skip to content

Commit 8d77db3

Browse files
author
Tim Foley
authored
Add options to control matrix layout rules (shader-slang#583)
* Add options to control matrix layout rules Up to this point, the Slang compiler has assumed that the default matrix layout conventions for the target API will be used. This means column-major layout for D3D, and *row major* layout for GL/Vulkan (note that while GL/Vulkan describe the default as "column major" there is an implicit swap of "row" and "column" when mapping HLSL conventions to GLSL). This commit introduces two main changes: 1. The default layout convention is switched to column-major on all targets, to ensure that D3D and GL/Vulkan can easily be driven by the same application logic. I would prefer to make the default be row-major (because this is the "obvious" convention for matrices), but I don't want to deviate from the defaults in existing HLSL compilers. 2. Command-line and API options are introduced for setting the matrix layout convention to use (by default) for each code generation target. It is still possible for explicit qualifiers like `row_major` to change the layout from within shader code. I also added an API to query the matrix layout convention that was used for a type layout (which should be of the `SLANG_TYPE_KIND_MATRIX` kind), but this isn't yet exercised. I added a reflection test case to make sure that the offsets/sizes we compute for matrix-type fields are appropriately modified by the flag that gets passed in. In a future change we could possibly switch the default convention to row-major, if we also changed our testing to match, since there are currently not many clients to be adversely impacted by the change. * Fixup: silence 64-bit build warning
1 parent 8c593ae commit 8d77db3

10 files changed

+355
-47
lines changed

slang.h

+21
Original file line numberDiff line numberDiff line change
@@ -170,6 +170,14 @@ extern "C"
170170
SLANG_PROFILE_UNKNOWN,
171171
};
172172

173+
typedef unsigned int SlangMatrixLayoutMode;
174+
enum
175+
{
176+
SLANG_MATRIX_LAYOUT_MODE_UNKNOWN = 0,
177+
SLANG_MATRIX_LAYOUT_ROW_MAJOR,
178+
SLANG_MATRIX_LAYOUT_COLUMN_MAJOR,
179+
};
180+
173181
//#define SLANG_LAYOUT_UNIFORM 0
174182
//#define SLANG_LAYOUT_PACKED 1
175183
//#define SLANG_LAYOUT_STORAGE 2
@@ -271,6 +279,11 @@ extern "C"
271279
int targetIndex,
272280
SlangTargetFlags flags);
273281

282+
SLANG_API void spSetTargetMatrixLayoutMode(
283+
SlangCompileRequest* request,
284+
int targetIndex,
285+
SlangMatrixLayoutMode mode);
286+
274287
/*!
275288
@brief Set the container format to be used for binary output.
276289
*/
@@ -683,6 +696,8 @@ extern "C"
683696
SLANG_API unsigned spReflectionTypeLayout_GetCategoryCount(SlangReflectionTypeLayout* type);
684697
SLANG_API SlangParameterCategory spReflectionTypeLayout_GetCategoryByIndex(SlangReflectionTypeLayout* type, unsigned index);
685698

699+
SLANG_API SlangMatrixLayoutMode spReflectionTypeLayout_GetMatrixLayoutMode(SlangReflectionTypeLayout* type);
700+
686701
// Variable Reflection
687702

688703
SLANG_API char const* spReflectionVariable_GetName(SlangReflectionVariable* var);
@@ -1039,6 +1054,12 @@ namespace slang
10391054
{
10401055
return getType()->getName();
10411056
}
1057+
1058+
SlangMatrixLayoutMode getMatrixLayoutMode()
1059+
{
1060+
return spReflectionTypeLayout_GetMatrixLayoutMode((SlangReflectionTypeLayout*) this);
1061+
}
1062+
10421063
};
10431064

10441065
struct Modifier

source/slang/compiler.h

+16
Original file line numberDiff line numberDiff line change
@@ -73,6 +73,18 @@ namespace Slang
7373
Binary
7474
};
7575

76+
// When storing the layout for a matrix-type
77+
// value, we need to know whether it has been
78+
// laid ot with row-major or column-major
79+
// storage.
80+
//
81+
enum MatrixLayoutMode
82+
{
83+
kMatrixLayoutMode_RowMajor = SLANG_MATRIX_LAYOUT_ROW_MAJOR,
84+
kMatrixLayoutMode_ColumnMajor = SLANG_MATRIX_LAYOUT_COLUMN_MAJOR,
85+
};
86+
87+
7688
class CompileRequest;
7789
class TranslationUnitRequest;
7890

@@ -221,6 +233,10 @@ namespace Slang
221233

222234
// TypeLayouts created on the fly by reflection API
223235
Dictionary<Type*, RefPtr<TypeLayout>> typeLayouts;
236+
237+
/// The layout to use for matrices by default (row/column major)
238+
MatrixLayoutMode defaultMatrixLayoutMode = kMatrixLayoutMode_ColumnMajor;
239+
MatrixLayoutMode getDefaultMatrixLayoutMode() { return defaultMatrixLayoutMode; }
224240
};
225241

226242
// Compute the "effective" profile to use when outputting the given entry point

source/slang/options.cpp

+19
Original file line numberDiff line numberDiff line change
@@ -247,6 +247,8 @@ struct OptionsParser
247247

248248
//
249249

250+
SlangMatrixLayoutMode defaultMatrixLayoutMode = SLANG_MATRIX_LAYOUT_MODE_UNKNOWN;
251+
250252
char const* const* argCursor = &argv[0];
251253
char const* const* argEnd = &argv[argc];
252254
while (argCursor != argEnd)
@@ -488,6 +490,14 @@ struct OptionsParser
488490

489491
addOutputPath(outputPath);
490492
}
493+
else if(argStr == "-matrix-layout-row-major")
494+
{
495+
defaultMatrixLayoutMode = kMatrixLayoutMode_RowMajor;
496+
}
497+
else if(argStr == "-matrix-layout-column-major")
498+
{
499+
defaultMatrixLayoutMode = kMatrixLayoutMode_ColumnMajor;
500+
}
491501
else if (argStr == "--")
492502
{
493503
// The `--` option causes us to stop trying to parse options,
@@ -745,6 +755,15 @@ struct OptionsParser
745755
spSetTargetFlags(compileRequest, 0, targetFlags);
746756
}
747757

758+
if(defaultMatrixLayoutMode != SLANG_MATRIX_LAYOUT_MODE_UNKNOWN)
759+
{
760+
UInt targetCount = requestImpl->targets.Count();
761+
for(UInt tt = 0; tt < targetCount; ++tt)
762+
{
763+
spSetTargetMatrixLayoutMode(compileRequest, int(tt), defaultMatrixLayoutMode);
764+
}
765+
}
766+
748767
// Next, we want to make sure that entry points get attached to the appropriate translation
749768
// unit that will provide them.
750769
{

source/slang/reflection.cpp

+17
Original file line numberDiff line numberDiff line change
@@ -658,6 +658,23 @@ SLANG_API SlangParameterCategory spReflectionTypeLayout_GetCategoryByIndex(Slang
658658
return typeLayout->resourceInfos[index].kind;
659659
}
660660

661+
SLANG_API SlangMatrixLayoutMode spReflectionTypeLayout_GetMatrixLayoutMode(SlangReflectionTypeLayout* inTypeLayout)
662+
{
663+
auto typeLayout = convert(inTypeLayout);
664+
if(!typeLayout) return SLANG_MATRIX_LAYOUT_MODE_UNKNOWN;
665+
666+
if( auto matrixLayout = dynamic_cast<MatrixTypeLayout*>(typeLayout) )
667+
{
668+
return matrixLayout->mode;
669+
}
670+
else
671+
{
672+
return SLANG_MATRIX_LAYOUT_MODE_UNKNOWN;
673+
}
674+
675+
}
676+
677+
661678
// Variable Reflection
662679

663680
SLANG_API char const* spReflectionVariable_GetName(SlangReflectionVariable* inVar)

source/slang/slang.cpp

+10
Original file line numberDiff line numberDiff line change
@@ -952,6 +952,16 @@ SLANG_API void spSetTargetFlags(
952952
req->targets[targetIndex]->targetFlags = flags;
953953
}
954954

955+
SLANG_API void spSetTargetMatrixLayoutMode(
956+
SlangCompileRequest* request,
957+
int targetIndex,
958+
SlangMatrixLayoutMode mode)
959+
{
960+
auto req = REQ(request);
961+
req->targets[targetIndex]->defaultMatrixLayoutMode = Slang::MatrixLayoutMode(mode);
962+
}
963+
964+
955965
SLANG_API void spSetOutputContainerFormat(
956966
SlangCompileRequest* request,
957967
SlangContainerFormat format)

source/slang/type-layout.cpp

+2-33
Original file line numberDiff line numberDiff line change
@@ -27,12 +27,6 @@ static size_t RoundUpToPowerOfTwo( size_t value )
2727

2828
//
2929

30-
MatrixLayoutMode LayoutRulesImpl::getDefaultMatrixLayoutMode() {
31-
return family->getDefaultMatrixLayoutMode();
32-
}
33-
34-
//
35-
3630
struct DefaultLayoutRulesImpl : SimpleLayoutRulesImpl
3731
{
3832
// Get size and alignment for a single value of base type.
@@ -428,25 +422,6 @@ struct GLSLLayoutRulesFamilyImpl : LayoutRulesFamilyImpl
428422
virtual LayoutRulesImpl* getSpecializationConstantRules() override;
429423
virtual LayoutRulesImpl* getShaderStorageBufferRules() override;
430424
virtual LayoutRulesImpl* getParameterBlockRules() override;
431-
432-
virtual MatrixLayoutMode getDefaultMatrixLayoutMode() override
433-
{
434-
// The default matrix layout mode in GLSL is specified
435-
// to be "column major" but what GLSL calls a "column"
436-
// is actually what HLSL (and hence Slang) calls a row.
437-
//
438-
// That is, an HLSL `float3x4` has 3 rows and 4 columns,
439-
// and indexing yields a `float4`.
440-
//
441-
// A GLSL `mat3x4` has 3 "columns" and 4 "rows", and
442-
// indexing into it yields a `vec4`.
443-
//
444-
// The Slang compiler needs to be consistent about this mess,
445-
// and so when the GLSL spec says that "column"-major is
446-
// the default, we know that they actually mean what we
447-
// call row-major.
448-
return kMatrixLayoutMode_RowMajor;
449-
}
450425
};
451426

452427
struct HLSLLayoutRulesFamilyImpl : LayoutRulesFamilyImpl
@@ -459,11 +434,6 @@ struct HLSLLayoutRulesFamilyImpl : LayoutRulesFamilyImpl
459434
virtual LayoutRulesImpl* getSpecializationConstantRules() override;
460435
virtual LayoutRulesImpl* getShaderStorageBufferRules() override;
461436
virtual LayoutRulesImpl* getParameterBlockRules() override;
462-
463-
virtual MatrixLayoutMode getDefaultMatrixLayoutMode() override
464-
{
465-
return kMatrixLayoutMode_ColumnMajor;
466-
}
467437
};
468438

469439
GLSLLayoutRulesFamilyImpl kGLSLLayoutRulesFamilyImpl;
@@ -644,12 +614,11 @@ TypeLayoutContext getInitialLayoutContextForTarget(TargetRequest* targetReq)
644614
TypeLayoutContext context;
645615
context.targetReq = targetReq;
646616
context.rules = nullptr;
647-
context.matrixLayoutMode = MatrixLayoutMode::kMatrixLayoutMode_RowMajor;
617+
context.matrixLayoutMode = targetReq->getDefaultMatrixLayoutMode();
648618

649619
if( rulesFamily )
650620
{
651621
context.rules = rulesFamily->getConstantBufferRules();
652-
context.matrixLayoutMode = rulesFamily->getDefaultMatrixLayoutMode();
653622
}
654623

655624
return context;
@@ -1150,7 +1119,7 @@ createParameterGroupTypeLayout(
11501119
RefPtr<TypeLayout> elementTypeLayout)
11511120
{
11521121
return createParameterGroupTypeLayout(
1153-
context.with(parameterGroupRules).with(parameterGroupRules->getDefaultMatrixLayoutMode()),
1122+
context.with(parameterGroupRules).with(context.targetReq->getDefaultMatrixLayoutMode()),
11541123
parameterGroupType,
11551124
parameterGroupInfo,
11561125
elementTypeLayout);

source/slang/type-layout.h

-14
Original file line numberDiff line numberDiff line change
@@ -361,16 +361,6 @@ class StreamOutputTypeLayout : public TypeLayout
361361
};
362362

363363

364-
// When storing the layout for a matrix-type
365-
// value, we need to know whether it has been
366-
// laid ot with row-major or column-major
367-
// storage.
368-
//
369-
enum MatrixLayoutMode
370-
{
371-
kMatrixLayoutMode_RowMajor,
372-
kMatrixLayoutMode_ColumnMajor,
373-
};
374364
class MatrixTypeLayout : public TypeLayout
375365
{
376366
public:
@@ -603,8 +593,6 @@ struct LayoutRulesImpl
603593
//
604594

605595
LayoutRulesFamilyImpl* getLayoutRulesFamily() { return family; }
606-
607-
MatrixLayoutMode getDefaultMatrixLayoutMode();
608596
};
609597

610598
struct LayoutRulesFamilyImpl
@@ -617,8 +605,6 @@ struct LayoutRulesFamilyImpl
617605
virtual LayoutRulesImpl* getSpecializationConstantRules()= 0;
618606
virtual LayoutRulesImpl* getShaderStorageBufferRules() = 0;
619607
virtual LayoutRulesImpl* getParameterBlockRules() = 0;
620-
621-
virtual MatrixLayoutMode getDefaultMatrixLayoutMode() = 0;
622608
};
623609

624610
struct TypeLayoutContext

tests/reflection/matrix-layout.slang

+28
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,28 @@
1+
//TEST:REFLECTION:-profile ps_4_0 -target hlsl
2+
//TEST:REFLECTION:-profile ps_4_0 -target hlsl -matrix-layout-row-major
3+
4+
// Test that we apply matrix layout rules correctly.
5+
6+
cbuffer A
7+
{
8+
float3x4 aa;
9+
row_major float3x4 ab;
10+
column_major float3x4 ac;
11+
}
12+
13+
struct SB
14+
{
15+
float3x4 ba;
16+
row_major float3x4 bb;
17+
column_major float3x4 bc;
18+
};
19+
20+
cbuffer B
21+
{
22+
SB b;
23+
}
24+
25+
float4 main() : SV_Target
26+
{
27+
return 0.0;
28+
}

0 commit comments

Comments
 (0)