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

src: define NAPI_HAS_THREADS to make TSFN available on Emscripten #1283

Closed
wants to merge 5 commits into from

Conversation

toyobayashi
Copy link
Contributor

@toyobayashi toyobayashi commented Feb 8, 2023

Hi, Node-API team.

I created emnapi, that is a Node-API implementation for Emscripten/wasi-sdk/wasm32. This project aims to help users port their or existing Node-API native addons to wasm, and run it on browser! It is well tested by using Node.js official test cases. Recently napi-rs is also plan to integrate emnapi so that napi-rs can target wasm as well. See real world use cases and discussions here.

Currently emnapi shipped a modified copy of node-addon-api in @tybys/emnapi npm package, due to the original source disables threadsafe function API when building for wasm, though tsfn can work well by relying on emscripten pthreads. And CallbackScope can not be implemented, neither on browser nor on Node.js. Needless to say in the browser environment, even on Node.js, the reason is that native node::CallbackScope* can not be returned to JavaScript.

This PR defines NAPI_HAS_THREADS to make TSFN available on Emscripten, and disable CallbackScope in wasm. Thus emnapi no longer need to maintain a copy, users can just do npm install node-addon-api to work with emnapi.

@toyobayashi
Copy link
Contributor Author

Sorry, I have no idea why npm run lint not work on my local machine. Manually do Alt + Shift + F use the clang-format in vscode, request run CI again.

Copy link
Member

@legendecas legendecas left a comment

Choose a reason for hiding this comment

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

Hey Toyo, thanks for the work!

I'm wondering if classes like AsyncWorker should be scoped with NAPI_HAS_THREADS too?

@toyobayashi
Copy link
Contributor Author

Because emnapi napi_*_async_work implementation relies on pthread too. I have ported libuv's threadpool to emnapi.

@legendecas
Copy link
Member

I'm still curious about the reason that CallbackScope can not be implemented for missing JavaScript equivalent representatives. IMO, classes like HandleScope, AsyncWorker, and Reference are native structures, not JavaScript values.

On the other hand, CallbackScope and TSFN themselves don't create any threads or require threaded works. Instead, AsyncWorker and its subclasses are executed in the thread pool. Would it be more reasonable to only scope AsyncWorker in NAPI_HAS_THREADS instead?

@toyobayashi
Copy link
Contributor Author

toyobayashi commented Feb 10, 2023

I'm still curious about the reason that CallbackScope can not be implemented for missing JavaScript equivalent representatives.

Imagine this, you call napi_open_callback_scope() in wasm, then wasm need to tell the host (Node.js) open a real node::CallbackScope or call real native napi_open_callback_scope, then return the pointer address to wasm memory to save.

wasm napi_open_callback_scope() -> Node.js JavaScript -> call napi_open_callback_scope in native node binding 

-> return pointer -> JavaScript -> set the pointer address to wasm memory

However, both of them failed, Node.js doesn't allow leaving an opening CallbackScope before returning to JavaScript from C++, it just crashed, saying something like callback scope mismatch. In addition, when native callback scope destructing, the pending process.nextTick() will be called before returning back to JavaScript, and there is no such async context in browser, so it can be not implemented on browser.

classes like HandleScope, AsyncWorker, and Reference are native structures, not JavaScript values

emnapi's napi_value / napi_reference / napi_handle_scope is not the native, they are implemented in JavaScript, just JavaScript numbers.

On the other hand, CallbackScope and TSFN themselves don't create any threads or require threaded works. Instead, AsyncWorker and its subclasses are executed in the thread pool. Would it be more reasonable to only scope AsyncWorker in NAPI_HAS_THREADS instead?

I understand your point, but without threads, TSFN and AsyncWorker never work, they are broken. TSFN also rely on pthread mutex. Currently, if no pthread support, napi_create_async_work and such APIs in emnapi will always return napi_generic_failure. Scope them in NAPI_HAS_THREADS can warn user at compile time.

image

@toyobayashi
Copy link
Contributor Author

And this PR doesn't affect native platforms, NAPI_HAS_THREADS is always 1 in native.

toyobayashi added a commit to toyobayashi/emnapi that referenced this pull request Feb 11, 2023
@legendecas
Copy link
Member

legendecas commented Feb 13, 2023

Imagine this, you call napi_open_callback_scope() in wasm, then wasm need to tell the host (Node.js) open a real node::CallbackScope or call real native napi_open_callback_scope, then return the pointer address to wasm memory to save.

Thanks for the explanation! However, HandleScope has the same limitation as CallbackScope: https://github.com/nodejs/node/blob/main/src/js_native_api_v8.h#L95-L96. How did you circumvent the issue for HandleScope, as it is essential for managing javascript value handles?

I think CallbackScope can be used without threads. It is essential for any async callbacks to build proper async context tracking. So I'd find it would be great to ensure it is available unconditionally too.

@toyobayashi
Copy link
Contributor Author

toyobayashi commented Feb 13, 2023

How did you circumvent the issue for HandleScope, as it is essential for managing javascript value handles?

As I explained above,

emnapi's napi_value / napi_reference / napi_handle_scope is not the native, they are implemented in JavaScript, just JavaScript numbers.

HandleScope in emnapi is completely implemented in JavaScript, so is napi_value / napi_reference / napi_deferred, they are independent of native Node.js runtime, all these things are available on browser!

I think CallbackScope can be used without threads.

Yeah, I'm scoping Napi::CallbackScope in !defined(__wasm__), not NAPI_HAS_THREADS.

#if (NAPI_VERSION > 2 && !defined(__wasm__))

@legendecas
Copy link
Member

legendecas commented Feb 13, 2023

HandleScope in emnapi is completely implemented in JavaScript, so is napi_value / napi_reference / napi_deferred, they are independent of native Node.js runtime, all these things are available on browser!
Yeah, I'm scoping Napi::CallbackInfo in !wasm, not NAPI_HAS_THREADS.

In this case, I'd find it is an implementation detail of emnapi.

@legendecas
Copy link
Member

legendecas commented Feb 13, 2023

napi_async_init/napi_async_destroy and napi_open_callback_scope/napi_close_callback_scope should be able to be implemented with JS API AsyncResource.

Edit: probably there is a problem with asyncResource.runInAsyncScope. Sorry.

@toyobayashi
Copy link
Contributor Author

Now I have removed !defined(__wasm__) on CallbackScope.

napi_async_init, napi_async_destroy and napi_make_callback have been implemented via node binding as a bridge.

https://github.com/toyobayashi/emnapi/blob/main/packages/emnapi/src/node.ts

napi-inl.h Outdated Show resolved Hide resolved
napi.h Outdated Show resolved Hide resolved
@toyobayashi
Copy link
Contributor Author

In this case, I'd find it is an implementation detail of emnapi.

CallbackScope cannot be implemented in wasm, this thing itself is not strongly related to emnapi, so I personally prefer to put it in !defined(__wasm__). Or maybe there is probability to implement this in the future if Node.js add some ability to workaround?

@legendecas
Copy link
Member

legendecas commented Feb 13, 2023

Yeah, I'm thinking of providing a built-in wasm node-api module in Node.js since we already allowed declaring a wasm addon in node_api.h. I'll create a tracking issue for this.

Previous discussion: nodejs/abi-stable-node#375

@toyobayashi
Copy link
Contributor Author

By the way, would it be possible to consider mentioning emnapi in the official documentation Node-API chapter? I have confidence to say now emnapi is the only project which has so complete Node-API implementation for wasm, official test cases guarantees that the runtime behavior on browser is almost exactly the same as on native Node.js. Implementation detail also follows official source code as far as possible. I believe this project can make contribution to community :)

@legendecas
Copy link
Member

toyobayashi added a commit to toyobayashi/emnapi that referenced this pull request Feb 13, 2023
nodejs-github-bot pushed a commit to nodejs/node that referenced this pull request Feb 24, 2023
PR-URL: #46633
Refs: #33597
Refs: nodejs/node-addon-api#1283
Reviewed-By: Tobias Nießen <[email protected]>
Reviewed-By: Colin Ihrig <[email protected]>
Reviewed-By: Michael Dawson <[email protected]>
Reviewed-By: James M Snell <[email protected]>
Copy link
Member

@mhdawson mhdawson left a comment

Choose a reason for hiding this comment

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

LGTM

mhdawson pushed a commit that referenced this pull request Mar 3, 2023
PR-URL: #1283
Reviewed-By: Michael Dawson <[email protected]
Reviewed-By: Chengzhong Wu <[email protected]>
@mhdawson
Copy link
Member

mhdawson commented Mar 3, 2023

Landed in 0b53d88

@mhdawson mhdawson closed this Mar 3, 2023
@mhdawson mhdawson mentioned this pull request Mar 3, 2023
targos pushed a commit to nodejs/node that referenced this pull request Mar 13, 2023
PR-URL: #46633
Refs: #33597
Refs: nodejs/node-addon-api#1283
Reviewed-By: Tobias Nießen <[email protected]>
Reviewed-By: Colin Ihrig <[email protected]>
Reviewed-By: Michael Dawson <[email protected]>
Reviewed-By: James M Snell <[email protected]>
danielleadams pushed a commit to nodejs/node that referenced this pull request Apr 11, 2023
PR-URL: #46633
Refs: #33597
Refs: nodejs/node-addon-api#1283
Reviewed-By: Tobias Nießen <[email protected]>
Reviewed-By: Colin Ihrig <[email protected]>
Reviewed-By: Michael Dawson <[email protected]>
Reviewed-By: James M Snell <[email protected]>
johnfrench3 pushed a commit to johnfrench3/node-addon-api-git that referenced this pull request Aug 11, 2023
PR-URL: nodejs/node-addon-api#1283
Reviewed-By: Michael Dawson <[email protected]
Reviewed-By: Chengzhong Wu <[email protected]>
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.

3 participants