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

Fix weakref callback for wrap (V8) #118

Merged
merged 1 commit into from
Mar 1, 2017

Conversation

jasongin
Copy link
Member

@jasongin jasongin commented Mar 1, 2017

This fixes a subtle bug related to V8 Persistent weakref callbacks that NAPI uses for object wrapping. The bug was caused by a late change I made in the references PR that I neglected to stress-test. I found the bug today when trying to re-run the Canvas benchmarks after that change.

The feedback in the review was valid, and I'm keeping the change that was made there, just with an extra check to fix the bug. This is a complex issue that is difficult to explain. Here's an overview of the sequence that leads to the bug:

  1. napi_wrap is called. It creates two v8impl::Reference instances, which each wrap a weak v8::Persistent. The Reference wrapping Persistent A is for internal tracking of the caller's finalize callback. The Reference wrapping Persistent B is returned by napi_wrap, to be held as a field in the ObjectWrap instance and used for refcounting across native async code.
  2. (Some work is done by the wrapped object. All references to the object are eventually released and the object is ready to be GC'd.)
  3. A and B weakref callbacks are added to V8's finalize queue.
  4. Persistent A weakref callback is invoked.
  5. That invokes ObjectWrap's finalizer callback.
  6. That deletes the ObjectWrap instance.
  7. The ObjectWrap instance destructor deletes its field Reference B
  8. Reference B's desctructor resets its Persistent B.
  9. The ObjectWrap finalizer callback returns.
  10. The Persistent A weakref callback continues, and deletes Reference A.
  11. Reference A's destructor resets its Persistent A.
  12. Meanwhile Persistent B's weakref callback is still on the finalize queue.
  13. Persistent B's weakref callback CRASHES when trying to dereference its data pointer because Reference B was already deleted.

The fix prevents registration of a weakref callback when the callback doesn't need to do anything anyway, as is the case for Reference B. Therefore it isn't a problem to delete Reference B from Persistent A's weakref callback.

@jasongin jasongin changed the title napi: Fix weakref callback for wrap Fix weakref callback for wrap (V8) Mar 1, 2017
@digitalinfinity
Copy link
Contributor

:shipit:

boingoing

This comment was marked as off-topic.

@jasongin jasongin merged commit 75e28c5 into nodejs:api-prototype-8.x Mar 1, 2017
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