summaryrefslogtreecommitdiff
path: root/src/tools/wasm-split/wasm-split.cpp
diff options
context:
space:
mode:
authorBenjamin Ling <serenade.mint@gmail.com>2024-07-11 02:35:27 +0800
committerGitHub <noreply@github.com>2024-07-10 11:35:27 -0700
commit76f661203f98820ebc6840ecf627a5eafc038403 (patch)
tree5da3797654f8a4da1fced8adf8b268a7dd81f7a5 /src/tools/wasm-split/wasm-split.cpp
parent0750bdbc1f7c356a160493acdb8dc314a68f1f12 (diff)
downloadbinaryen-76f661203f98820ebc6840ecf627a5eafc038403.tar.gz
binaryen-76f661203f98820ebc6840ecf627a5eafc038403.tar.bz2
binaryen-76f661203f98820ebc6840ecf627a5eafc038403.zip
Allow --keepfuncs and --splitfuncs to be use alongside a profile data (#6322)
There are times after collecting a profile, we wish to manually include specific functions into the primary module. It could be due to non-deterministic profiling or functions for error scenarios (e.g. _trap). This PR helps to unlock this workflow by honoring both the `--keep-funcs` flag as well as the `--profile` flag
Diffstat (limited to 'src/tools/wasm-split/wasm-split.cpp')
-rw-r--r--src/tools/wasm-split/wasm-split.cpp44
1 files changed, 34 insertions, 10 deletions
diff --git a/src/tools/wasm-split/wasm-split.cpp b/src/tools/wasm-split/wasm-split.cpp
index d7dc19d67..abb1646c1 100644
--- a/src/tools/wasm-split/wasm-split.cpp
+++ b/src/tools/wasm-split/wasm-split.cpp
@@ -212,30 +212,35 @@ void splitModule(const WasmSplitOptions& options) {
parseInput(wasm, options);
std::set<Name> keepFuncs;
+ std::set<Name> splitFuncs;
if (options.profileFile.size()) {
// Use the profile to set `keepFuncs`.
uint64_t hash = hashFile(options.inputFiles[0]);
- std::set<Name> splitFuncs;
getFunctionsToKeepAndSplit(
wasm, hash, options.profileFile, keepFuncs, splitFuncs);
- } else if (options.keepFuncs.size()) {
+ }
+
+ if (options.keepFuncs.size()) {
// Use the explicitly provided `keepFuncs`.
for (auto& func : options.keepFuncs) {
if (!options.quiet && wasm.getFunctionOrNull(func) == nullptr) {
std::cerr << "warning: function " << func << " does not exist\n";
+ continue;
}
+
keepFuncs.insert(func);
+ splitFuncs.erase(func);
}
- } else if (options.splitFuncs.size()) {
+ }
+
+ if (options.splitFuncs.size()) {
// Use the explicitly provided `splitFuncs`.
- for (auto& func : wasm.functions) {
- keepFuncs.insert(func->name);
- }
for (auto& func : options.splitFuncs) {
auto* function = wasm.getFunctionOrNull(func);
if (!options.quiet && function == nullptr) {
std::cerr << "warning: function " << func << " does not exist\n";
+ continue;
}
if (function && function->imported()) {
if (!options.quiet) {
@@ -243,20 +248,39 @@ void splitModule(const WasmSplitOptions& options) {
<< "\n";
}
} else {
+ if (!options.quiet && keepFuncs.count(func) > 0) {
+ std::cerr
+ << "warning: function " << func
+ << " was to be kept in primary module. "
+ << "However it will now be split out into secondary module.\n";
+ }
+
+ splitFuncs.insert(func);
keepFuncs.erase(func);
}
}
- }
- if (options.jspi) {
- // The load secondary module function must be kept in the main module.
- keepFuncs.insert(ModuleSplitting::LOAD_SECONDARY_MODULE);
+ if (keepFuncs.empty()) {
+ // could be the case where every function has been split out
+ // or when `splitFuncs` is used standalone, which is the case we'll cover
+ // here
+ for (auto& func : wasm.functions) {
+ if (splitFuncs.count(func->name) == 0) {
+ keepFuncs.insert(func->name);
+ }
+ }
+ }
}
if (!options.quiet && keepFuncs.size() == 0) {
std::cerr << "warning: not keeping any functions in the primary module\n";
}
+ if (options.jspi) {
+ // The load secondary module function must be kept in the main module.
+ keepFuncs.insert(ModuleSplitting::LOAD_SECONDARY_MODULE);
+ }
+
// If warnings are enabled, check that any functions are being split out.
if (!options.quiet) {
std::set<Name> splitFuncs;