summaryrefslogtreecommitdiffstats
path: root/source/slang/slang-ir-array-reg-to-mem.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'source/slang/slang-ir-array-reg-to-mem.cpp')
-rw-r--r--source/slang/slang-ir-array-reg-to-mem.cpp87
1 files changed, 87 insertions, 0 deletions
diff --git a/source/slang/slang-ir-array-reg-to-mem.cpp b/source/slang/slang-ir-array-reg-to-mem.cpp
new file mode 100644
index 000000000..6f749f242
--- /dev/null
+++ b/source/slang/slang-ir-array-reg-to-mem.cpp
@@ -0,0 +1,87 @@
+#include "slang-ir-array-reg-to-mem.h"
+
+#include "slang-ir.h"
+#include "slang-ir-insts.h"
+#include "slang-ir-util.h"
+
+namespace Slang
+{
+ bool eliminateArrayTypeParameters(IRFunc* func)
+ {
+ IRBuilder builder(func);
+ bool changed = false;
+ List<UInt> arrayParamIds;
+ UInt idx = 0;
+ List<IRParam*> paramWorkList;
+ for (auto param : func->getParams())
+ {
+ if (auto arrayType = as<IRArrayTypeBase>(param->getFullType()))
+ {
+ paramWorkList.add(param);
+ arrayParamIds.add(idx);
+ }
+ idx++;
+ }
+ for (auto param : paramWorkList)
+ {
+ // We have an array type parameter, so we need to replace it with a pointer to the array
+ // type.
+ //
+ // We will also need to insert a `load` instruction at the start of the function body
+ // to load the actual pointer value from the parameter.
+ //
+ if (auto arrayType = as<IRArrayTypeBase>(param->getFullType()))
+ {
+ changed = true;
+ builder.setInsertBefore(param);
+ auto ptrArrayType = builder.getPtrType(arrayType);
+ auto newParam = builder.emitParam(ptrArrayType);
+ setInsertAfterOrdinaryInst(&builder, param);
+ auto regVal = builder.emitLoad(newParam);
+ param->replaceUsesWith(regVal);
+ param->removeAndDeallocate();
+ }
+ }
+ if (changed)
+ {
+ // The function is modified, we need to also update its type.
+ List<IRType*> paramTypes;
+ for (auto param : func->getParams())
+ {
+ paramTypes.add(param->getFullType());
+ }
+ auto newFuncType = builder.getFuncType((UInt)paramTypes.getCount(), paramTypes.getBuffer(), func->getResultType());
+ func->setFullType(newFuncType);
+
+ // Update all the call sites to pass the arrays by pointer.
+ traverseUses(func, [&](IRUse* use)
+ {
+ if (const auto call = as<IRCall>(use->getUser()))
+ {
+ builder.setInsertBefore(call);
+ for (auto paramId : arrayParamIds)
+ {
+ auto arg = call->getArg(paramId);
+ auto var = builder.emitVar(as<IRPtrTypeBase>(paramTypes[paramId])->getValueType());
+ builder.emitStore(var, arg);
+ call->setArg(paramId, var);
+ }
+ }
+ });
+ }
+ return changed;
+ }
+
+ bool eliminateArrayTypeSSARegisters(IRModule* module)
+ {
+ bool changed = false;
+ for (auto inst : module->getGlobalInsts())
+ {
+ if (auto func = as<IRFunc>(inst))
+ {
+ changed |= eliminateArrayTypeParameters(func);
+ }
+ }
+ return changed;
+ }
+}