From f6cbb81e1c0080518185294ee94705f5e93aa849 Mon Sep 17 00:00:00 2001 From: Sai Praveen Bangaru <31557731+saipraveenb25@users.noreply.github.com> Date: Wed, 5 Feb 2025 13:49:19 -0800 Subject: Fix DCE for calls to functions that have associations (#6272) * Fix DCE for calls to functions that have associations * Update slang-ir-util.cpp * Update slang-ir-util.cpp --- source/slang/slang-ir-util.cpp | 82 ++++++++++++++++++++++++++++++++++++++++-- 1 file changed, 80 insertions(+), 2 deletions(-) (limited to 'source') diff --git a/source/slang/slang-ir-util.cpp b/source/slang/slang-ir-util.cpp index dd7819e1a..a3cf28a68 100644 --- a/source/slang/slang-ir-util.cpp +++ b/source/slang/slang-ir-util.cpp @@ -1215,8 +1215,65 @@ bool isSideEffectFreeFunctionalCall(IRCall* call, SideEffectAnalysisOptions opti return false; } +// Enumerate any associated functions of 'func' +// that might be used by a pass (e.g. auto-diff) +// +template +void forEachAssociatedFunction(IRInst* func, TFunc callback) +{ + // Resolve the function to get all its decorations + auto resolvedFunc = getResolvedInstForDecorations(func); + if (!resolvedFunc) + return; + + // We'll scan for appropriate decorations and return + // the function references. + // + // TODO: In the future, as we get more function transformation + // passes, we might want to create a parent class for such + // decorations that associate functions with each other. + // + for (auto decor : resolvedFunc->getDecorations()) + { + switch (decor->getOp()) + { + case kIROp_UserDefinedBackwardDerivativeDecoration: + if (as(decor)) + { + auto associatedCallee = as(decor) + ->getBackwardDerivativeFunc(); + callback(associatedCallee); + } + break; + + case kIROp_ForwardDerivativeDecoration: + if (as(decor)) + { + auto associatedCallee = + as(decor)->getForwardDerivativeFunc(); + callback(associatedCallee); + } + break; + + case kIROp_PrimalSubstituteDecoration: + if (as(decor)) + { + auto associatedCallee = + as(decor)->getPrimalSubstituteFunc(); + callback(associatedCallee); + } + break; + + default: + break; + } + } +} + bool doesCalleeHaveSideEffect(IRInst* callee) { + bool sideEffect = true; + for (auto decor : getResolvedInstForDecorations(callee)->getDecorations()) { switch (decor->getOp()) @@ -1224,10 +1281,31 @@ bool doesCalleeHaveSideEffect(IRInst* callee) case kIROp_NoSideEffectDecoration: case kIROp_ReadNoneDecoration: case kIROp_IgnoreSideEffectsDecoration: - return false; + sideEffect = false; + break; + default: + break; } } - return true; + + // If the callee has no side effect, check if any of its associated functions have side effect. + // If so, we want to keep the callee around. + // + // Typically, once the relevant pass has completed, the association is removed, + // and at that point we can remove the function. + // + if (!sideEffect) + { + forEachAssociatedFunction( + callee, + [&](IRInst* associatedCallee) + { + sideEffect |= doesCalleeHaveSideEffect(associatedCallee); + return; + }); + } + + return sideEffect; } IRInst* findInterfaceRequirement(IRInterfaceType* type, IRInst* key) -- cgit v1.2.3