diff options
author | Alon Zakai <azakai@google.com> | 2021-07-26 13:04:09 -0700 |
---|---|---|
committer | GitHub <noreply@github.com> | 2021-07-26 20:04:09 +0000 |
commit | 8b606524401bf6fc9e0f46f51a66c5e37ec6e707 (patch) | |
tree | 995731a27ac3a03e0085517896d6dcea314e75b5 /src/passes/DeadArgumentElimination.cpp | |
parent | f09bb989a15451960c1078426b61dcc50f232a0a (diff) | |
download | binaryen-8b606524401bf6fc9e0f46f51a66c5e37ec6e707.tar.gz binaryen-8b606524401bf6fc9e0f46f51a66c5e37ec6e707.tar.bz2 binaryen-8b606524401bf6fc9e0f46f51a66c5e37ec6e707.zip |
[Wasm GC] Refine return types (#4020)
Corresponds to #4014 which did the same for parameter types. This sees
whether the return types actually returned from a function allow us to use
a more specific type for the function's return. If so, we update that type, as
well as calls to the function.
Diffstat (limited to 'src/passes/DeadArgumentElimination.cpp')
-rw-r--r-- | src/passes/DeadArgumentElimination.cpp | 84 |
1 files changed, 83 insertions, 1 deletions
diff --git a/src/passes/DeadArgumentElimination.cpp b/src/passes/DeadArgumentElimination.cpp index a2231469d..ddba3bee8 100644 --- a/src/passes/DeadArgumentElimination.cpp +++ b/src/passes/DeadArgumentElimination.cpp @@ -43,6 +43,7 @@ #include "ir/find_all.h" #include "ir/module-utils.h" #include "ir/type-updating.h" +#include "ir/utils.h" #include "pass.h" #include "passes/opt-utils.h" #include "support/sorted_vector.h" @@ -308,6 +309,9 @@ struct DAE : public Pass { allDroppedCalls[pair.first] = pair.second; } } + // If we refine return types then we will need to do more type updating + // at the end. + bool refinedReturnTypes = false; // We now have a mapping of all call sites for each function, and can look // for optimization opportunities. for (auto& pair : allCalls) { @@ -323,6 +327,10 @@ struct DAE : public Pass { // affect whether an argument is used or not, it just refines the type // where possible. refineArgumentTypes(func, calls, module); + // Refine return types as well. + if (refineReturnTypes(func, calls, module)) { + refinedReturnTypes = true; + } // Check if all calls pass the same constant for a particular argument. for (Index i = 0; i < numParams; i++) { Literal value; @@ -357,6 +365,12 @@ struct DAE : public Pass { } } } + if (refinedReturnTypes) { + // Changing a call expression's return type can propagate out to its + // parents, and so we must refinalize. + // TODO: We could track in which functions we actually make changes. + ReFinalize().run(runner, module); + } // Track which functions we changed, and optimize them later if necessary. std::unordered_set<Function*> changed; // We now know which parameters are unused, and can potentially remove them. @@ -444,7 +458,7 @@ struct DAE : public Pass { if (optimize && !changed.empty()) { OptUtils::optimizeAfterInlining(changed, module, runner); } - return !changed.empty(); + return !changed.empty() || refinedReturnTypes; } private: @@ -591,6 +605,74 @@ private: } } } + + // See if the types returned from a function allow us to define a more refined + // return type for it. If so, we can update it and all calls going to it. + // + // This assumes that the function has no calls aside from |calls|, that is, it + // is not exported or called from the table or by reference. Exports should be + // fine, as should indirect calls in principle, but VMs will need to support + // function subtyping in indirect calls. TODO: relax this when possible + // + // Returns whether we optimized. + // + // TODO: We may be missing a global optimum here, as e.g. if a function calls + // itself and returns that value, then we would not do any change here, + // as one of the return values is exactly what it already is. Similar + // unoptimality can happen with multiple functions, more local code in + // the middle, etc. + bool refineReturnTypes(Function* func, + const std::vector<Call*>& calls, + Module* module) { + if (!module->features.hasGC()) { + return false; + } + + Type originalType = func->getResults(); + if (!originalType.hasRef()) { + // Nothing to refine. + return false; + } + + // Before we do anything, we must refinalize the function, because otherwise + // its body may contain a block with a forced type, + // + // (func (result X) + // (block (result X) + // (..content with more specific type Y..) + // ) + ReFinalize().walkFunctionInModule(func, module); + + Type refinedType = func->body->type; + if (refinedType == originalType) { + return false; + } + + // Scan the body and look at the returns. + for (auto* ret : FindAll<Return>(func->body).list) { + refinedType = Type::getLeastUpperBound(refinedType, ret->value->type); + if (refinedType == originalType) { + return false; + } + } + assert(refinedType != originalType); + + // If the refined type is unreachable then nothing actually returns from + // this function. + // TODO: We can propagate that to the outside, and not just for GC. + if (refinedType == Type::unreachable) { + return false; + } + + // Success. Update the type, and the calls. + func->setResults(refinedType); + for (auto* call : calls) { + if (call->type != Type::unreachable) { + call->type = refinedType; + } + } + return true; + } }; Pass* createDAEPass() { return new DAE(); } |