summaryrefslogtreecommitdiff
path: root/src/passes/Inlining.cpp
diff options
context:
space:
mode:
authorAlon Zakai <azakai@google.com>2023-12-06 15:25:16 -0800
committerGitHub <noreply@github.com>2023-12-06 15:25:16 -0800
commitf722171f73732c6852cc4d283cc006ca56142ebf (patch)
treefa0c80c4488c4fcde03e63772d1d057bf965ae8d /src/passes/Inlining.cpp
parent66277f9b767cb1b45ce12b77109c2538b6cb9c12 (diff)
downloadbinaryen-f722171f73732c6852cc4d283cc006ca56142ebf.tar.gz
binaryen-f722171f73732c6852cc4d283cc006ca56142ebf.tar.bz2
binaryen-f722171f73732c6852cc4d283cc006ca56142ebf.zip
Add no-inline IR annotation, and passes to set it based on function name (#6146)
Any function can now be annotated as not to be inlined fully (normally) or not to be inlined partially. In the future we'll want to read those annotations from the proposed wasm metadata section on code hints, and from wat text as well, but for now add trivial passes that set those fields based on function name wildcards, e.g.: --no-inline=*leave-alone* --inlining That will not inline any function whose name contains "leave-alone" in the name. --no-inline disables all inlining (full or partial) while --no-full-inline and --no-partial-inline affect only full or partial inlining.
Diffstat (limited to 'src/passes/Inlining.cpp')
-rw-r--r--src/passes/Inlining.cpp22
1 files changed, 14 insertions, 8 deletions
diff --git a/src/passes/Inlining.cpp b/src/passes/Inlining.cpp
index 20172d88d..e1e2dd022 100644
--- a/src/passes/Inlining.cpp
+++ b/src/passes/Inlining.cpp
@@ -686,8 +686,10 @@ struct FunctionSplitter {
auto outlinedFunctionSize = info.size - Measurer::measure(iff);
// If outlined function will be worth normal inline, skip the intermediate
- // state and inline fully now.
- if (outlinedFunctionWorthInlining(info, outlinedFunctionSize)) {
+ // state and inline fully now. Note that if full inlining is disabled we
+ // will not do this, and instead inline partially.
+ if (!func->noFullInline &&
+ outlinedFunctionWorthInlining(info, outlinedFunctionSize)) {
return InliningMode::Full;
}
@@ -771,10 +773,12 @@ struct FunctionSplitter {
// Success, this matches the pattern.
// If the outlined function will be worth inlining normally, skip the
- // intermediate state and inline fully now.
+ // intermediate state and inline fully now. (As above, if full inlining is
+ // disabled, we only partially inline.)
if (numIfs == 1) {
auto outlinedFunctionSize = Measurer::measure(iff->ifTrue);
- if (outlinedFunctionWorthInlining(info, outlinedFunctionSize)) {
+ if (!func->noFullInline &&
+ outlinedFunctionWorthInlining(info, outlinedFunctionSize)) {
return InliningMode::Full;
}
}
@@ -1197,7 +1201,9 @@ struct Inlining : public Pass {
// See explanation in doInlining() for the parameter nameHint.
Index inlinedNameHint = 0;
+ // Decide for a given function whether to inline, and if so in what mode.
InliningMode getInliningMode(Name name) {
+ auto* func = module->getFunction(name);
auto& info = infos[name];
if (info.inliningMode != InliningMode::Unknown) {
@@ -1205,19 +1211,19 @@ struct Inlining : public Pass {
}
// Check if the function itself is worth inlining as it is.
- if (info.worthFullInlining(getPassOptions())) {
- info.inliningMode = InliningMode::Full;
- return info.inliningMode;
+ if (!func->noFullInline && info.worthFullInlining(getPassOptions())) {
+ return info.inliningMode = InliningMode::Full;
}
// Otherwise, check if we can at least inline part of it, if we are
// interested in such things.
- if (functionSplitter) {
+ if (!func->noPartialInline && functionSplitter) {
info.inliningMode = functionSplitter->getSplitDrivenInliningMode(
module->getFunction(name), info);
return info.inliningMode;
}
+ // Cannot be fully or partially inlined => uninlineable.
info.inliningMode = InliningMode::Uninlineable;
return info.inliningMode;
}