Skip to content

Commit 935b629

Browse files
author
Tim Foley
authored
Fix IR emit logic for methods in struct types (shader-slang#791)
There was a bug in the logic for emitting initial IR, such that it was neglecting to emit "methods" (member functions) unless they were also referenced by a non-member (global) function, or were needed to satisfy an interface requirement. This would only matter for `import`ed modules, since for non-`import`ed code, anything relevant would be referenced by the entry point so that the problem would never surface. This change fixes the underlying problem by adding a step to the IR lowering pass called `ensureAllDeclsRec` that makes sure that not only global-scope declarations, but also anything nested under a `struct` type gets emitted to the initial IR module. There are also a few unrelated fixes in this PR, which are things I ran into while making the fix: * Deleted support for the (long gone) `IRDeclRef` type in our `slang.natvis` file * Added support for visualizing the value of IR string and integer literals when they appear in the debugger * Fixed IR dumping logic to not skip emitting `struct` and `interface` instructions. Switching those to inherit from `IRType` accidentally affected how they get printed in IR dumps by default. * Fixed up the IR linking logic so that it correctly takes `[export]` decorations into account, so that an exported definition will always be taken over any other (unless the latter is more specialized for the target). I initially implemented this in an attempt to fix the original issue, but found it wasn't a fix for the root cause. It is still a better approach than what was implemented previously, so I'm leaving it in place.
1 parent a08a314 commit 935b629

File tree

7 files changed

+99
-17
lines changed

7 files changed

+99
-17
lines changed

source/slang/ir-link.cpp

+9-1
Original file line numberDiff line numberDiff line change
@@ -855,7 +855,15 @@ bool isBetterForTarget(
855855
if(newLevel != oldLevel)
856856
return UInt(newLevel) > UInt(oldLevel);
857857

858-
// All other factors being equal, a definition is
858+
// All preceding factors being equal, an `[export]` is better
859+
// than an `[import]`.
860+
//
861+
bool newIsExport = newVal->findDecoration<IRExportDecoration>() != nullptr;
862+
bool oldIsExport = oldVal->findDecoration<IRExportDecoration>() != nullptr;
863+
if(newIsExport != oldIsExport)
864+
return newIsExport;
865+
866+
// All preceding factors being equal, a definition is
859867
// better than a declaration.
860868
auto newIsDef = isDefinition(newVal);
861869
auto oldIsDef = isDefinition(oldVal);

source/slang/ir.cpp

+16
Original file line numberDiff line numberDiff line change
@@ -3118,6 +3118,22 @@ namespace Slang
31183118
if(as<IRConstant>(inst))
31193119
return true;
31203120

3121+
// We are going to have a general rule that
3122+
// a type should be folded into its use site,
3123+
// which improves output in most cases, but
3124+
// we would like to not apply that rule to
3125+
// "nominal" types like `struct`s.
3126+
//
3127+
switch( inst->op )
3128+
{
3129+
case kIROp_StructType:
3130+
case kIROp_InterfaceType:
3131+
return false;
3132+
3133+
default:
3134+
break;
3135+
}
3136+
31213137
if(as<IRType>(inst))
31223138
return true;
31233139

source/slang/lower-to-ir.cpp

+32-7
Original file line numberDiff line numberDiff line change
@@ -4922,7 +4922,6 @@ struct DeclLoweringVisitor : DeclVisitor<DeclLoweringVisitor, LoweredValInfo>
49224922

49234923
IRStructType* irStruct = subBuilder->createStructType();
49244924
addNameHint(context, irStruct, decl);
4925-
49264925
addLinkageDecoration(context, irStruct, decl);
49274926

49284927
subBuilder->setInsertInto(irStruct);
@@ -4959,15 +4958,16 @@ struct DeclLoweringVisitor : DeclVisitor<DeclLoweringVisitor, LoweredValInfo>
49594958
fieldType);
49604959
}
49614960

4962-
// TODO: we should enumerate the non-field members of the type
4963-
// as well, and ensure those have been emitted (e.g., any
4964-
// member functions).
4961+
// There may be members not handled by the above logic (e.g.,
4962+
// member functions), but we will not immediately force them
4963+
// to be emitted here, so as not to risk a circular dependency.
4964+
//
4965+
// Instead we will force emission of all children of aggregate
4966+
// type declarations later, from the top-level emit logic.
49654967

49664968
irStruct->moveToEnd();
4967-
49684969
addTargetIntrinsicDecorations(irStruct, decl);
49694970

4970-
49714971
return LoweredValInfo::simple(finishOuterGenerics(subBuilder, irStruct));
49724972
}
49734973

@@ -6147,6 +6147,31 @@ static void lowerEntryPointToIR(
61476147
}
61486148
}
61496149

6150+
/// Ensure that `decl` and all relevant declarations under it get emitted.
6151+
static void ensureAllDeclsRec(
6152+
IRGenContext* context,
6153+
Decl* decl)
6154+
{
6155+
ensureDecl(context, decl);
6156+
6157+
// Note: We are checking here for aggregate type declarations, and
6158+
// not for `ContainerDecl`s in general. This is because many kinds
6159+
// of container declarations will already take responsibility for emitting
6160+
// their children directly (e.g., a function declaration is responsible
6161+
// for emitting its own parameters).
6162+
//
6163+
// Aggregate types are the main case where we can emit an outer declaration
6164+
// and not the stuff nested inside of it.
6165+
//
6166+
if(auto containerDecl = dynamic_cast<AggTypeDecl*>(decl))
6167+
{
6168+
for (auto memberDecl : containerDecl->Members)
6169+
{
6170+
ensureAllDeclsRec(context, memberDecl);
6171+
}
6172+
}
6173+
}
6174+
61506175
IRModule* generateIRForTranslationUnit(
61516176
TranslationUnitRequest* translationUnit)
61526177
{
@@ -6192,7 +6217,7 @@ IRModule* generateIRForTranslationUnit(
61926217
// been emitted.
61936218
for (auto decl : translationUnit->SyntaxNode->Members)
61946219
{
6195-
ensureDecl(context, decl);
6220+
ensureAllDeclsRec(context, decl);
61966221
}
61976222

61986223
#if 0

source/slang/slang.natvis

+2-9
Original file line numberDiff line numberDiff line change
@@ -82,6 +82,8 @@
8282
<Expand>
8383
<Item Name="[op]">op</Item>
8484
<Item Name="[type]">typeUse.usedValue</Item>
85+
<Item Name="[value]" Condition="op == Slang::kIROp_StringLit">((IRStringLit*)this)->value.stringVal.chars,[((IRStringLit*)this)->value.stringVal.numChars]s8</Item>
86+
<Item Name="[value]" Condition="op == Slang::kIROp_IntLit">((IRIntLit*)this)->value.intVal</Item>
8587
<Synthetic Name="[operands]">
8688
<DisplayString>{{count = {operandCount}}}</DisplayString>
8789
<Expand>
@@ -113,15 +115,6 @@
113115
</Synthetic>
114116
</Expand>
115117
</Type>
116-
<Type Name="Slang::IRDeclRef">
117-
<DisplayString>{{IRDeclRef {declRef}}}</DisplayString>
118-
<Expand>
119-
<Item Name="[op]">op</Item>
120-
<Item Name="[type]">type</Item>
121-
<Item Name="[declRef]">declRef</Item>
122-
<Item Name="[parent]">parent</Item>
123-
</Expand>
124-
</Type>
125118
<Type Name="Slang::IRUse">
126119
<DisplayString>{{IRUse {usedValue}}}</DisplayString>
127120
<Expand>

tests/bugs/gl-33-ext.slang

+8
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
// gl-33-ext.slang
2+
//TEST_IGNORE_FILE:
3+
4+
struct A
5+
{
6+
int state;
7+
[mutating] int next() { return state; }
8+
};

tests/bugs/gl-33.slang

+28
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,28 @@
1+
// gl-33.slang
2+
//TEST(compute):COMPARE_COMPUTE:
3+
4+
// Test for GitLab issue # 33
5+
6+
import gl_33_ext;
7+
8+
typedef A B;
9+
10+
int test(int val)
11+
{
12+
B b = { val };
13+
return b.next();
14+
}
15+
16+
//TEST_INPUT:ubuffer(data=[0 0 0 0], stride=4):dxbinding(0),glbinding(0),out
17+
RWStructuredBuffer<int> gBuffer;
18+
19+
[numthreads(4, 1, 1)]
20+
void computeMain(uint3 dispatchThreadID : SV_DispatchThreadID)
21+
{
22+
uint tid = dispatchThreadID.x;
23+
24+
int val = tid;
25+
val = test(val);
26+
27+
gBuffer[tid] = val;
28+
}

tests/bugs/gl-33.slang.expected.txt

+4
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,4 @@
1+
0
2+
1
3+
2
4+
3

0 commit comments

Comments
 (0)