summaryrefslogtreecommitdiff
path: root/src/passes/JSPI.cpp
diff options
context:
space:
mode:
authorBrendan Dahl <brendan.dahl@gmail.com>2022-11-08 12:24:15 -0800
committerGitHub <noreply@github.com>2022-11-08 12:24:15 -0800
commiteb8481c6dc499a5b109f3ca60bd79f1803ee8ebe (patch)
treeb6e25045b2f221d50c2f658f2c920d8f20ef3099 /src/passes/JSPI.cpp
parentdcf19345b9825dc579334f27092aa0f7eff9b506 (diff)
downloadbinaryen-eb8481c6dc499a5b109f3ca60bd79f1803ee8ebe.tar.gz
binaryen-eb8481c6dc499a5b109f3ca60bd79f1803ee8ebe.tar.bz2
binaryen-eb8481c6dc499a5b109f3ca60bd79f1803ee8ebe.zip
Add arguments to control which imports/exports are JSPI'd. (#5217)
Instead of automatically determining which exports will be async they will be explicitly set by the user. We'll rely on the runtime trapping if they are incorrectly set. Two new arguments that behave similar to asyncify-imports: - jspi-imports - jspi-exports
Diffstat (limited to 'src/passes/JSPI.cpp')
-rw-r--r--src/passes/JSPI.cpp60
1 files changed, 54 insertions, 6 deletions
diff --git a/src/passes/JSPI.cpp b/src/passes/JSPI.cpp
index 48f6207ed..aa1546984 100644
--- a/src/passes/JSPI.cpp
+++ b/src/passes/JSPI.cpp
@@ -22,26 +22,71 @@
#include "ir/utils.h"
#include "pass.h"
#include "shared-constants.h"
+#include "support/file.h"
+#include "support/string.h"
#include "wasm-builder.h"
#include "wasm.h"
#include <utility>
//
// Convert a module to be compatible with JavaScript promise integration (JSPI).
-// All exports will be wrapped with a function that will handle storing
+// Promising exports will be wrapped with a function that will handle storing
// the suspsender that is passed in as the first param from a "promising"
-// `WebAssembly.Function`. All imports will also be wrapped, but they will take
-// the stored suspender and pass it as the first param to the imported function
-// that should be created from a "suspending" `WebAssembly.Function`.
+// `WebAssembly.Function`. Suspending imports will also be wrapped, but they
+// will take the stored suspender and pass it as the first param to the imported
+// function that should be created from a "suspending" `WebAssembly.Function`.
//
+// By default all imports and exports will be wrapped, but this can be
+// controlled with the following options:
+//
+// --pass-arg=jspi-imports@module1.base1,module2.base2,module3.base3
+//
+// Wrap each import in the comma-separated list. Wildcards and a separate
+// files are supported. See `asyncify-imports` for more details.
+//
+// --pass-arg=jspi-exports@function_one,function_two,function_three
+//
+// Wrap each export in the comma-separated list. Similar to jspi-imports,
+// wildcards and separate files are supported.
+//
+
namespace wasm {
+static std::string getFullFunctionName(Name module, Name base) {
+ return std::string(module.str) + '.' + base.toString();
+}
+
+static bool canChangeState(std::string name, String::Split stateChangers) {
+ // When no state changers are given default to everything changes state.
+ if (stateChangers.empty()) {
+ return true;
+ }
+ for (auto& stateChanger : stateChangers) {
+ if (String::wildcardMatch(stateChanger, name)) {
+ return true;
+ }
+ }
+ return false;
+}
+
struct JSPI : public Pass {
Type externref = Type(HeapType::ext, Nullable);
void run(Module* module) override {
Builder builder(*module);
+
+ auto& options = getPassOptions();
+ // Find which imports can suspend.
+ auto stateChangingImports = String::trim(read_possible_response_file(
+ options.getArgumentOrDefault("jspi-imports", "")));
+ String::Split listedImports(stateChangingImports, ",");
+
+ // Find which exports should create a promise.
+ auto stateChangingExports = String::trim(read_possible_response_file(
+ options.getArgumentOrDefault("jspi-exports", "")));
+ String::Split listedExports(stateChangingExports, ",");
+
// Create a global to store the suspender that is passed into exported
// functions and will then need to be passed out to the imported functions.
Name suspender = Names::getValidGlobalName(*module, "suspender");
@@ -57,7 +102,8 @@ struct JSPI : public Pass {
// Wrap each exported function in a function that stores the suspender
// and calls the original export.
for (auto& ex : module->exports) {
- if (ex->kind == ExternalKind::Function) {
+ if (ex->kind == ExternalKind::Function &&
+ canChangeState(ex->name.toString(), listedExports)) {
auto* func = module->getFunction(ex->value);
Name wrapperName;
auto iter = wrappedExports.find(func->name);
@@ -79,7 +125,9 @@ struct JSPI : public Pass {
// Wrap each imported function in a function that gets the global suspender
// and passes it on to the imported function.
for (auto* im : originalFunctions) {
- if (im->imported()) {
+ if (im->imported() &&
+ canChangeState(getFullFunctionName(im->module, im->base),
+ listedImports)) {
makeWrapperForImport(im, module, suspender);
}
}