diff options
author | Alon Zakai <azakai@google.com> | 2023-12-06 15:25:16 -0800 |
---|---|---|
committer | GitHub <noreply@github.com> | 2023-12-06 15:25:16 -0800 |
commit | f722171f73732c6852cc4d283cc006ca56142ebf (patch) | |
tree | fa0c80c4488c4fcde03e63772d1d057bf965ae8d /src/passes/Inlining.cpp | |
parent | 66277f9b767cb1b45ce12b77109c2538b6cb9c12 (diff) | |
download | binaryen-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.cpp | 22 |
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; } |