summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorEllie Hermaszewska <ellieh@nvidia.com>2024-06-02 10:27:13 +0800
committerGitHub <noreply@github.com>2024-06-01 19:27:13 -0700
commit0bc89bc13251fedc9ed90cf473d2e6eb7fda3abf (patch)
tree564e5160660c8b65e9f7b9b15c1e7d3c0d863f11
parentc5a453e56985022deb820cbbb2ff5cd6a8347e34 (diff)
Prevent hoisting of non-hoistable instructions in non-function values with code (#4250)
-rw-r--r--source/slang/slang-ir-redundancy-removal.cpp4
-rw-r--r--source/slang/slang-ir.cpp12
-rw-r--r--source/slang/slang-ir.h3
-rw-r--r--tests/bugs/generic-initializer-inlining.slang33
4 files changed, 50 insertions, 2 deletions
diff --git a/source/slang/slang-ir-redundancy-removal.cpp b/source/slang/slang-ir-redundancy-removal.cpp
index c7986cfbc..b679d8438 100644
--- a/source/slang/slang-ir-redundancy-removal.cpp
+++ b/source/slang/slang-ir-redundancy-removal.cpp
@@ -25,9 +25,9 @@ struct RedundancyRemovalContext
for (UInt i = 0; i < inst->getOperandCount(); i++)
{
auto operand = inst->getOperand(i);
- if (getParentFunc(operand) != func)
+ if (!hasDescendent(func, operand))
{
- // Global value won't prevent hoisting.
+ // Only prevent hoisting from operands local to this function
continue;
}
auto operandParent = as<IRBlock>(operand->getParent());
diff --git a/source/slang/slang-ir.cpp b/source/slang/slang-ir.cpp
index 3e96d61eb..b6d000d20 100644
--- a/source/slang/slang-ir.cpp
+++ b/source/slang/slang-ir.cpp
@@ -8530,6 +8530,18 @@ namespace Slang
return nullptr;
}
+ bool hasDescendent(IRInst* inst, IRInst* child)
+ {
+ auto parent = child->getParent();
+ while (parent)
+ {
+ if (inst == parent)
+ return true;
+ parent = parent->getParent();
+ }
+ return false;
+ }
+
IRInst* getGenericReturnVal(IRInst* inst)
{
if (auto gen = as<IRGeneric>(inst))
diff --git a/source/slang/slang-ir.h b/source/slang/slang-ir.h
index 3a5384e8d..9e8be7fc5 100644
--- a/source/slang/slang-ir.h
+++ b/source/slang/slang-ir.h
@@ -2538,6 +2538,9 @@ bool isBuiltin(IRInst* inst);
// Get the enclosuing function of an instruction.
IRFunc* getParentFunc(IRInst* inst);
+ // Is child a descendent of inst
+bool hasDescendent(IRInst* inst, IRInst* child);
+
// True if moving this inst will not change the semantics of the program
bool isMovableInst(IRInst* inst);
diff --git a/tests/bugs/generic-initializer-inlining.slang b/tests/bugs/generic-initializer-inlining.slang
new file mode 100644
index 000000000..8f126e43a
--- /dev/null
+++ b/tests/bugs/generic-initializer-inlining.slang
@@ -0,0 +1,33 @@
+//TEST(compute):COMPARE_COMPUTE(filecheck-buffer=CHECK):-cpu
+
+// CHECK: 1
+// CHECK-NEXT: 0
+// CHECK-NEXT: 0
+// CHECK-NEXT: 0
+
+//TEST_INPUT:ubuffer(data=[0 0 0 0], stride=4):out,name=outputBuffer
+RWStructuredBuffer<uint> outputBuffer;
+
+struct C<T>
+{
+ [ForceInline]
+ __init<U>(U value)
+ {
+ for (int i = 0; i < 1; ++i)
+ {}
+ }
+}
+
+static C<int> c = C<int>(1);
+
+int use<T>(T x)
+{
+ return 1;
+}
+
+[numthreads(1, 1, 1)]
+void computeMain(uint i : SV_GroupIndex)
+{
+ outputBuffer[0] = use(c);
+}
+