diff options
author | Ashley Nelson <nashley@google.com> | 2023-01-03 12:37:35 -0800 |
---|---|---|
committer | GitHub <noreply@github.com> | 2023-01-03 20:37:35 +0000 |
commit | 5fefa9361a0d958dd7977907eaf1ae8facf3ba48 (patch) | |
tree | bdc3f2973213f34233f57a92e60b2cc9e9d98199 /src/passes/MultiMemoryLowering.cpp | |
parent | f36229b509ddb59e7ab20cd6fd3ccd962e9990a5 (diff) | |
download | binaryen-5fefa9361a0d958dd7977907eaf1ae8facf3ba48.tar.gz binaryen-5fefa9361a0d958dd7977907eaf1ae8facf3ba48.tar.bz2 binaryen-5fefa9361a0d958dd7977907eaf1ae8facf3ba48.zip |
Maintain first memory import/export in Multi-Memory Lowering Pass (#5363)
This PR maintains the first memory's import/export in the single combined memory after multi-memories are lowered.
Diffstat (limited to 'src/passes/MultiMemoryLowering.cpp')
-rw-r--r-- | src/passes/MultiMemoryLowering.cpp | 52 |
1 files changed, 50 insertions, 2 deletions
diff --git a/src/passes/MultiMemoryLowering.cpp b/src/passes/MultiMemoryLowering.cpp index ab0a1376b..5414f2f99 100644 --- a/src/passes/MultiMemoryLowering.cpp +++ b/src/passes/MultiMemoryLowering.cpp @@ -59,6 +59,14 @@ struct MultiMemoryLowering : public Pass { Builder::MemoryInfo memoryInfo; // If the combined memory is shared bool isShared; + // If the combined memory is imported + bool isImported; + // If the combined memory is exported + bool isExported = false; + // If the combined memory should be imported, the following two + // properties will be set + Name module; + Name base; // The initial page size of the combined memory Address totalInitialPages; // The max page size of the combined memory @@ -385,6 +393,9 @@ struct MultiMemoryLowering : public Pass { createMemoryGrowFunctions(); removeExistingMemories(); addCombinedMemory(); + if (isExported) { + updateMemoryExports(); + } Replacer(*this, *wasm).run(getPassRunner(), wasm); } @@ -407,17 +418,25 @@ struct MultiMemoryLowering : public Pass { // offsetGlobalNames bool isLastMemory(Index idx) { return idx == offsetGlobalNames.size(); } + Memory& getFirstMemory() { return *wasm->memories[0]; } + void prepCombinedMemory() { - pointerType = wasm->memories[0]->indexType; + pointerType = getFirstMemory().indexType; memoryInfo = pointerType == Type::i32 ? Builder::MemoryInfo::Memory32 : Builder::MemoryInfo::Memory64; - isShared = wasm->memories[0]->shared; + isShared = getFirstMemory().shared; + isImported = getFirstMemory().imported(); for (auto& memory : wasm->memories) { // We are assuming that each memory is configured the same as the first // and assert if any of the memories does not match this configuration assert(memory->shared == isShared); assert(memory->indexType == pointerType); + // TODO: handle memory import for memories other than the first + if (memory->name != getFirstMemory().name && memory->imported()) { + Fatal() << "MultiMemoryLowering: only the first memory can be imported"; + } + // Calculating the total initial and max page size for the combined memory // by totaling the initial and max page sizes for the memories in the // module @@ -437,6 +456,20 @@ struct MultiMemoryLowering : public Pass { totalInitialPages = totalMaxPages; } + // Save the module and base to set on the combinedMemory + if (isImported) { + module = getFirstMemory().module; + base = getFirstMemory().base; + } + // Ensuring only the first memory is an exported memory + for (auto& exp : wasm->exports) { + if (exp->kind == ExternalKind::Memory && + exp->value == getFirstMemory().name) { + isExported = true; + } else if (exp->kind == ExternalKind::Memory) { + Fatal() << "MultiMemoryLowering: only the first memory can be exported"; + } + } // Creating the combined memory name so we can reference the combined memory // in subsequent instructions before it is added to the module combinedMemory = Names::getValidMemoryName(*wasm, "combined_memory"); @@ -657,8 +690,23 @@ struct MultiMemoryLowering : public Pass { memory->indexType = pointerType; memory->initial = totalInitialPages; memory->max = totalMaxPages; + if (isImported) { + memory->base = base; + memory->module = module; + } wasm->addMemory(std::move(memory)); } + + void updateMemoryExports() { + for (auto& exp : wasm->exports) { + if (exp->kind == ExternalKind::Memory) { + // We checked in prepCombinedMemory that any memory exports are for + // the first memory, so setting the exports to the combinedMemory means + // calling JS will not have to worry about offsets + exp->value = combinedMemory; + } + } + } }; Pass* createMultiMemoryLoweringPass() { return new MultiMemoryLowering(false); } |