Fix weakref callback for wrap (V8) #118
Merged
Add this suggestion to a batch that can be applied as a single commit.
This suggestion is invalid because no changes were made to the code.
Suggestions cannot be applied while the pull request is closed.
Suggestions cannot be applied while viewing a subset of changes.
Only one suggestion per line can be applied in a batch.
Add this suggestion to a batch that can be applied as a single commit.
Applying suggestions on deleted lines is not supported.
You must change the existing code in this line in order to create a valid suggestion.
Outdated suggestions cannot be applied.
This suggestion has been applied or marked resolved.
Suggestions cannot be applied from pending reviews.
Suggestions cannot be applied on multi-line comments.
Suggestions cannot be applied while the pull request is queued to merge.
Suggestion cannot be applied right now. Please check back later.
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:
napi_wrap
is called. It creates twov8impl::Reference
instances, which each wrap a weakv8::Persistent
. The Reference wrapping Persistent A is for internal tracking of the caller's finalize callback. The Reference wrapping Persistent B is returned bynapi_wrap
, to be held as a field in the ObjectWrap instance and used for refcounting across native async code.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.