Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Support direct global bindings #104

Open
wants to merge 13 commits into
base: main
Choose a base branch
from
Open

Support direct global bindings #104

wants to merge 13 commits into from

Conversation

guybedford
Copy link
Collaborator

@guybedford guybedford commented Mar 13, 2025

This updates the specification to provide the unwrapping of WebAssembly.Global values in the exports of WebAssembly module instances, while fully retaining the existing support of Wasm to Wasm global binding and ensuring constant and mutable globals are only ever fulfilled by their same corresponding type.

This was originally discussed in the early stages of this proposal, and then brought up recently again in #103.

With this change, import { exportVal } from './mod.wasm' where mod.wasm exports a mutable global, can reflect the live direct JS value for exportVal in JS of that mutable global without requiring the exportVal.value accessor normally needed for instances of WebAssembly.Global. We effectively get live direct bindings on exports, and also for imports to other Wasm modules, while continuing to have snapshotted bindings on non-Wasm imports.

Live bindings for JS modules are supported for Wasm mutable globals due the infallible nature of ToJSValue() allowing us to create an association between the JS environment record and the Wasm global.

The way we handle Wasm to Wasm imports is to store the [[Instance]] on the WebAssembly module record, so that we can access the original global address from the instance when resolving to another WebAssembly Module record. All other types retain their existing handling. Since Wasm to Wasm imports no longer have the wrapping and unwrapping through JS which relied on implicit subtyping, we instead directly perform the same extern subtype checks of the WebAssembly value imports, using the corresponding core definition for this.

It would be great to discuss this one one last time before we more widely ship the current integration.

@guybedford
Copy link
Collaborator Author

I've significantly simplified the approach here by removing the extra spec work added for resolving cyclic bindings, to instead discuss any possible cycle support separately.

@guybedford guybedford force-pushed the module-bindings branch 2 times, most recently from 1834063 to e46cfcc Compare March 16, 2025 20:47
1. Let |resolutionInstance| be |resolution|.\[[Module]].\[[Instance]].
1. If |resolutionInstance| is ~empty~ then,
1. Throw a {LinkError} exception.
1. Let |resolutionModule| be |resolution|.\[[Module]].\[[ModuleSource]].\[[Module]].
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

(this is confusing, I thought [[Module]] was the Module Record corresponding to [[ModuleSource]] rather than a WebAssembly module, but I don't think we can improve it anyway)

1. [=list/Append=] |externval| to |imports|.
1. Otherwise,
1. Let |env| be |resolution|.\[[Module]].\[[Environment]].
1. Let |v| be [=?=] |env|.GetBindingValue(|resolution|.\[[BindingName]], true).
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

What if [[BindingName]] is NAMESPACE? e.g. if I import foo from a module that does export * as foo from "./bar.js".

1. Note: The condition above leaves unsupported JS values as uninitialized in TDZ and therefore as a reference error on
access. When integrating with shared globals, they may be excluded here similarly to v128 above.
1. Perform [=!=] |record|.\[[Environment]].InitializeBinding(|name|, [=ToJSValue=](|global_value|)).
1. If |mut| is [=var=], then associate all future mutations of |globaladdr| with the ECMA-262 binding record for |name| in
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Comment on lines +1536 to +1537
1. Note: The condition above leaves unsupported JS values as uninitialized in TDZ and therefore as a reference error on
access. When integrating with shared globals, they may be excluded here similarly to v128 above.
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Why do we do this instead of simply not exposing them, making it a linking error?

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

2 participants