summaryrefslogtreecommitdiff
path: root/src/ir
diff options
context:
space:
mode:
authorThomas Lively <tlively@google.com>2024-09-16 14:12:42 -0700
committerGitHub <noreply@github.com>2024-09-16 14:12:42 -0700
commit106f84b4f2dc373b540ace29139f850576f22b8a (patch)
treebef25740c15e35f561da89fedc751eddb1648bcb /src/ir
parent2a4096512d514b2319ff1c9360f0fbc9ecc94e5d (diff)
downloadbinaryen-106f84b4f2dc373b540ace29139f850576f22b8a.tar.gz
binaryen-106f84b4f2dc373b540ace29139f850576f22b8a.tar.bz2
binaryen-106f84b4f2dc373b540ace29139f850576f22b8a.zip
[wasm-split] Add an option to skip importing placeholders (#6942)
Wasm-split generally assumes that calls to secondary functions made before the secondary module has been loaded and instantiated should go to imported placeholder functions that can be responsible for loading the secondary module and forwarding the call to the loaded function. That scheme makes the loading entirely transparent from the application's point of view, which is not always a good thing. Other schemes would make it impossible for a secondary function to be called before the secondary module has been explicitly loaded, in which case the placeholder functions would never be called. To improve code size and simplify instantiation under these schemes, add a new `--no-placeholders` option that skips adding imported placeholder functions.
Diffstat (limited to 'src/ir')
-rw-r--r--src/ir/module-splitting.cpp55
-rw-r--r--src/ir/module-splitting.h6
2 files changed, 39 insertions, 22 deletions
diff --git a/src/ir/module-splitting.cpp b/src/ir/module-splitting.cpp
index 22f8b301f..adda1f4a6 100644
--- a/src/ir/module-splitting.cpp
+++ b/src/ir/module-splitting.cpp
@@ -94,10 +94,9 @@ template<class F> void forEachElement(Module& module, F f) {
} else if (auto* g = segment->offset->dynCast<GlobalGet>()) {
base = g->name;
}
- ElementUtils::iterElementSegmentFunctionNames(
- segment, [&](Name& entry, Index i) {
- f(segment->table, base, offset + i, entry);
- });
+ for (Index i = 0; i < segment->data.size(); ++i) {
+ f(segment->table, base, offset + i, segment->data[i]);
+ }
});
}
@@ -209,9 +208,12 @@ TableSlotManager::TableSlotManager(Module& module) : module(module) {
}
// Initialize funcIndices with the functions already in the table.
- forEachElement(module, [&](Name table, Name base, Index offset, Name func) {
- addSlot(func, {table, base, offset});
- });
+ forEachElement(module,
+ [&](Name table, Name base, Index offset, Expression* elem) {
+ if (auto* func = elem->dynCast<RefFunc>()) {
+ addSlot(func->func, {table, base, offset});
+ }
+ });
}
Table* TableSlotManager::makeTable() {
@@ -693,21 +695,32 @@ void ModuleSplitter::setupTablePatching() {
// Replace table references to secondary functions with an imported
// placeholder that encodes the table index in its name:
// `importNamespace`.`index`.
- forEachElement(primary, [&](Name, Name, Index index, Name& elem) {
- if (secondaryFuncs.count(elem)) {
- placeholderMap[index] = elem;
- auto* secondaryFunc = secondary.getFunction(elem);
- replacedElems[index] = secondaryFunc;
- auto placeholder = std::make_unique<Function>();
- placeholder->module = config.placeholderNamespace;
- placeholder->base = std::to_string(index);
- placeholder->name = Names::getValidFunctionName(
- primary, std::string("placeholder_") + placeholder->base.toString());
- placeholder->hasExplicitName = true;
- placeholder->type = secondaryFunc->type;
- elem = placeholder->name;
- primary.addFunction(std::move(placeholder));
+ forEachElement(primary, [&](Name, Name, Index index, Expression*& elem) {
+ auto* ref = elem->dynCast<RefFunc>();
+ if (!ref) {
+ return;
+ }
+ if (!secondaryFuncs.count(ref->func)) {
+ return;
+ }
+ placeholderMap[index] = ref->func;
+ auto* secondaryFunc = secondary.getFunction(ref->func);
+ replacedElems[index] = secondaryFunc;
+ if (!config.usePlaceholders) {
+ // TODO: This can create active element segments with lots of nulls. We
+ // should optimize them like we do data segments with zeros.
+ elem = Builder(primary).makeRefNull(HeapType::nofunc);
+ return;
}
+ auto placeholder = std::make_unique<Function>();
+ placeholder->module = config.placeholderNamespace;
+ placeholder->base = std::to_string(index);
+ placeholder->name = Names::getValidFunctionName(
+ primary, std::string("placeholder_") + placeholder->base.toString());
+ placeholder->hasExplicitName = true;
+ placeholder->type = secondaryFunc->type;
+ elem = Builder(primary).makeRefFunc(placeholder->name, placeholder->type);
+ primary.addFunction(std::move(placeholder));
});
if (replacedElems.size() == 0) {
diff --git a/src/ir/module-splitting.h b/src/ir/module-splitting.h
index dc5bb1998..620993d2d 100644
--- a/src/ir/module-splitting.h
+++ b/src/ir/module-splitting.h
@@ -52,11 +52,15 @@ struct Config {
// exists. May or may not include imported functions, which are always kept in
// the primary module regardless.
std::set<Name> primaryFuncs;
+ // Whether to import placeholder functions into the primary module that will
+ // be called when a secondary function is called before the secondary module
+ // has been loaded.
+ bool usePlaceholders = true;
// The namespace from which to import primary functions into the secondary
// module.
Name importNamespace = "primary";
// The namespace from which to import placeholder functions into the primary
- // module.
+ // module. Ignored if `usePlaceholders` is false.
Name placeholderNamespace = "placeholder";
// The prefix to attach to the name of any newly created exports. This can be
// used to differentiate between "real" exports of the module and exports that