diff options
author | Thomas Lively <tlively@google.com> | 2024-09-16 14:12:42 -0700 |
---|---|---|
committer | GitHub <noreply@github.com> | 2024-09-16 14:12:42 -0700 |
commit | 106f84b4f2dc373b540ace29139f850576f22b8a (patch) | |
tree | bef25740c15e35f561da89fedc751eddb1648bcb /src/ir | |
parent | 2a4096512d514b2319ff1c9360f0fbc9ecc94e5d (diff) | |
download | binaryen-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.cpp | 55 | ||||
-rw-r--r-- | src/ir/module-splitting.h | 6 |
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 |