Skip to content

Commit 9416d37

Browse files
committed
fixup! verify dangling napi_ref in napi_wrap
1 parent 689879a commit 9416d37

File tree

4 files changed

+145
-29
lines changed

4 files changed

+145
-29
lines changed

src/js_native_api_v8.cc

+52-24
Original file line numberDiff line numberDiff line change
@@ -218,7 +218,11 @@ inline napi_status Unwrap(napi_env env,
218218
if (action == RemoveWrap) {
219219
CHECK(obj->DeletePrivate(context, NAPI_PRIVATE_KEY(context, wrapper))
220220
.FromJust());
221-
delete reference;
221+
if (reference->ownership() == Ownership::kUserland) {
222+
reference->Reset();
223+
} else {
224+
delete reference;
225+
}
222226
}
223227

224228
return GET_RETURN_STATUS(env);
@@ -245,7 +249,8 @@ class CallbackBundle {
245249
bundle->env = env;
246250

247251
v8::Local<v8::Value> cbdata = v8::External::New(env->isolate, bundle);
248-
Reference::New(env, cbdata, 0, true, Delete, bundle, nullptr);
252+
Reference::New(
253+
env, cbdata, 0, Ownership::kRuntime, Delete, bundle, nullptr);
249254
return cbdata;
250255
}
251256
napi_env env; // Necessary to invoke C++ NAPI callback
@@ -429,16 +434,21 @@ inline napi_status Wrap(napi_env env,
429434
// before then, then the finalize callback will never be invoked.)
430435
// Therefore a finalize callback is required when returning a reference.
431436
CHECK_ARG(env, finalize_cb);
432-
reference = v8impl::Reference::New(
433-
env, obj, 0, false, finalize_cb, native_object, finalize_hint);
437+
reference = v8impl::Reference::New(env,
438+
obj,
439+
0,
440+
v8impl::Ownership::kUserland,
441+
finalize_cb,
442+
native_object,
443+
finalize_hint);
434444
*result = reinterpret_cast<napi_ref>(reference);
435445
} else {
436446
// Create a self-deleting reference.
437447
reference = v8impl::Reference::New(
438448
env,
439449
obj,
440450
0,
441-
true,
451+
v8impl::Ownership::kRuntime,
442452
finalize_cb,
443453
native_object,
444454
finalize_cb == nullptr ? nullptr : finalize_hint);
@@ -456,16 +466,22 @@ inline napi_status Wrap(napi_env env,
456466

457467
} // end of anonymous namespace
458468

469+
void Finalizer::Reset() {
470+
finalize_callback_ = nullptr;
471+
finalize_data_ = nullptr;
472+
finalize_hint_ = nullptr;
473+
}
474+
459475
// Wrapper around v8impl::Persistent that implements reference counting.
460476
RefBase::RefBase(napi_env env,
461477
uint32_t initial_refcount,
462-
bool delete_self,
478+
Ownership ownership,
463479
napi_finalize finalize_callback,
464480
void* finalize_data,
465481
void* finalize_hint)
466482
: Finalizer(env, finalize_callback, finalize_data, finalize_hint),
467483
refcount_(initial_refcount),
468-
delete_self_(delete_self) {
484+
ownership_(ownership) {
469485
Link(finalize_callback == nullptr ? &env->reflist : &env->finalizing_reflist);
470486
}
471487

@@ -480,13 +496,13 @@ RefBase::~RefBase() {
480496

481497
RefBase* RefBase::New(napi_env env,
482498
uint32_t initial_refcount,
483-
bool delete_self,
499+
Ownership ownership,
484500
napi_finalize finalize_callback,
485501
void* finalize_data,
486502
void* finalize_hint) {
487503
return new RefBase(env,
488504
initial_refcount,
489-
delete_self,
505+
ownership,
490506
finalize_callback,
491507
finalize_data,
492508
finalize_hint);
@@ -512,27 +528,29 @@ uint32_t RefBase::RefCount() {
512528
}
513529

514530
void RefBase::Finalize() {
515-
bool delete_self = delete_self_;
531+
Ownership ownership = ownership_;
516532
// Swap out the field finalize_callback so that it can not be accidentally
517533
// called more than once.
518534
napi_finalize finalize_callback = finalize_callback_;
519-
finalize_callback_ = nullptr;
535+
void* finalize_data = finalize_data_;
536+
void* finalize_hint = finalize_hint_;
537+
Reset();
520538

521539
// Either the RefBase is going to be deleted in the finalize_callback or not,
522540
// it should be removed from the tracked list.
523541
Unlink();
524542
// 1. If the finalize_callback is present, it should either delete the
525-
// RefBase, or set the flag `delete_self`.
543+
// RefBase, or set ownership with Ownership::kRuntime.
526544
// 2. If the finalizer is not present, the RefBase can be deleted after the
527545
// call.
528546
if (finalize_callback != nullptr) {
529-
env_->CallFinalizer(finalize_callback, finalize_data_, finalize_hint_);
547+
env_->CallFinalizer(finalize_callback, finalize_data, finalize_hint);
530548
// No access to `this` after finalize_callback is called.
531549
}
532550

533-
// If the RefBase is not self-destructive, userland code should delete it.
534-
// Now delete it if it is self-destructive.
535-
if (delete_self) {
551+
// If the RefBase is not Ownership::kRuntime, userland code should delete it.
552+
// Now delete it if it is Ownership::kRuntime.
553+
if (ownership == Ownership::kRuntime) {
536554
delete this;
537555
}
538556
}
@@ -554,14 +572,14 @@ Reference::~Reference() {
554572
Reference* Reference::New(napi_env env,
555573
v8::Local<v8::Value> value,
556574
uint32_t initial_refcount,
557-
bool delete_self,
575+
Ownership ownership,
558576
napi_finalize finalize_callback,
559577
void* finalize_data,
560578
void* finalize_hint) {
561579
return new Reference(env,
562580
value,
563581
initial_refcount,
564-
delete_self,
582+
ownership,
565583
finalize_callback,
566584
finalize_data,
567585
finalize_hint);
@@ -602,6 +620,11 @@ v8::Local<v8::Value> Reference::Get() {
602620
}
603621
}
604622

623+
void Reference::Reset() {
624+
persistent_.Reset();
625+
Finalizer::Reset();
626+
}
627+
605628
void Reference::Finalize() {
606629
// Unconditionally reset the persistent handle so that no weak callback will
607630
// be invoked again.
@@ -2297,8 +2320,13 @@ napi_status NAPI_CDECL napi_create_external(napi_env env,
22972320
if (finalize_cb) {
22982321
// The Reference object will delete itself after invoking the finalizer
22992322
// callback.
2300-
v8impl::Reference::New(
2301-
env, external_value, 0, true, finalize_cb, data, finalize_hint);
2323+
v8impl::Reference::New(env,
2324+
external_value,
2325+
0,
2326+
v8impl::Ownership::kRuntime,
2327+
finalize_cb,
2328+
data,
2329+
finalize_hint);
23022330
}
23032331

23042332
*result = v8impl::JsValueFromV8LocalValue(external_value);
@@ -2407,8 +2435,8 @@ napi_status NAPI_CDECL napi_create_reference(napi_env env,
24072435
return napi_set_last_error(env, napi_invalid_arg);
24082436
}
24092437

2410-
v8impl::Reference* reference =
2411-
v8impl::Reference::New(env, v8_value, initial_refcount, false);
2438+
v8impl::Reference* reference = v8impl::Reference::New(
2439+
env, v8_value, initial_refcount, v8impl::Ownership::kUserland);
24122440

24132441
*result = reinterpret_cast<napi_ref>(reference);
24142442
return napi_clear_last_error(env);
@@ -3115,8 +3143,8 @@ napi_status NAPI_CDECL napi_set_instance_data(napi_env env,
31153143
delete old_data;
31163144
}
31173145

3118-
env->instance_data =
3119-
v8impl::RefBase::New(env, 0, true, finalize_cb, data, finalize_hint);
3146+
env->instance_data = v8impl::RefBase::New(
3147+
env, 0, v8impl::Ownership::kRuntime, finalize_cb, data, finalize_hint);
31203148

31213149
return napi_clear_last_error(env);
31223150
}

src/js_native_api_v8.h

+20-4
Original file line numberDiff line numberDiff line change
@@ -316,6 +316,8 @@ class Finalizer {
316316
void* data() { return finalize_data_; }
317317
void* hint() { return finalize_hint_; }
318318

319+
virtual void Reset();
320+
319321
protected:
320322
napi_env env_;
321323
napi_finalize finalize_callback_;
@@ -337,20 +339,30 @@ class TryCatch : public v8::TryCatch {
337339
napi_env _env;
338340
};
339341

342+
// Ownership of a reference.
343+
enum class Ownership {
344+
// The reference is owned by the runtime. No userland call is needed to
345+
// destruct the reference.
346+
kRuntime,
347+
// The reference is owned by the userland. User code is responsible to delete
348+
// the reference with appropriate node-api calls.
349+
kUserland,
350+
};
351+
340352
// Wrapper around Finalizer that implements reference counting.
341353
class RefBase : public Finalizer, public RefTracker {
342354
protected:
343355
RefBase(napi_env env,
344356
uint32_t initial_refcount,
345-
bool delete_self,
357+
Ownership ownership,
346358
napi_finalize finalize_callback,
347359
void* finalize_data,
348360
void* finalize_hint);
349361

350362
public:
351363
static RefBase* New(napi_env env,
352364
uint32_t initial_refcount,
353-
bool delete_self,
365+
Ownership ownership,
354366
napi_finalize finalize_callback,
355367
void* finalize_data,
356368
void* finalize_hint);
@@ -361,12 +373,14 @@ class RefBase : public Finalizer, public RefTracker {
361373
uint32_t Unref();
362374
uint32_t RefCount();
363375

376+
Ownership ownership() { return ownership_; }
377+
364378
protected:
365379
void Finalize() override;
366380

367381
private:
368382
uint32_t refcount_;
369-
bool delete_self_;
383+
Ownership ownership_;
370384
};
371385

372386
// Wrapper around v8impl::Persistent.
@@ -379,7 +393,7 @@ class Reference : public RefBase {
379393
static Reference* New(napi_env env,
380394
v8::Local<v8::Value> value,
381395
uint32_t initial_refcount,
382-
bool delete_self,
396+
Ownership ownership,
383397
napi_finalize finalize_callback = nullptr,
384398
void* finalize_data = nullptr,
385399
void* finalize_hint = nullptr);
@@ -389,6 +403,8 @@ class Reference : public RefBase {
389403
uint32_t Unref();
390404
v8::Local<v8::Value> Get();
391405

406+
void Reset() override;
407+
392408
protected:
393409
void Finalize() override;
394410

test/js-native-api/6_object_wrap/6_object_wrap.cc

+62-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,6 @@
1-
#include "myobject.h"
21
#include "../common.h"
2+
#include "assert.h"
3+
#include "myobject.h"
34

45
napi_ref MyObject::constructor;
56

@@ -151,9 +152,69 @@ napi_value MyObject::Multiply(napi_env env, napi_callback_info info) {
151152
return instance;
152153
}
153154

155+
// This finalizer should never be invoked.
156+
void ObjectWrapDanglingReferenceFinalizer(napi_env env,
157+
void* finalize_data,
158+
void* finalize_hint) {
159+
assert(0 && "unreachable");
160+
}
161+
162+
napi_ref dangling_ref;
163+
napi_value ObjectWrapDanglingReference(napi_env env, napi_callback_info info) {
164+
size_t argc = 1;
165+
napi_value args[1];
166+
NODE_API_CALL(env,
167+
napi_get_cb_info(env, info, &argc, args, nullptr, nullptr));
168+
169+
// Create a napi_wrap and remove it immediately, whilst leaving the out-param
170+
// ref dangling (not deleted).
171+
NODE_API_CALL(env,
172+
napi_wrap(env,
173+
args[0],
174+
nullptr,
175+
ObjectWrapDanglingReferenceFinalizer,
176+
nullptr,
177+
&dangling_ref));
178+
NODE_API_CALL(env, napi_remove_wrap(env, args[0], nullptr));
179+
180+
return args[0];
181+
}
182+
183+
napi_value ObjectWrapDanglingReferenceTest(napi_env env,
184+
napi_callback_info info) {
185+
napi_value out;
186+
napi_value ret;
187+
NODE_API_CALL(env, napi_get_reference_value(env, dangling_ref, &out));
188+
189+
if (out == nullptr) {
190+
// If the napi_ref has been invalidated, delete it.
191+
NODE_API_CALL(env, napi_delete_reference(env, dangling_ref));
192+
NODE_API_CALL(env, napi_get_boolean(env, true, &ret));
193+
} else {
194+
// The dangling napi_ref is still valid.
195+
NODE_API_CALL(env, napi_get_boolean(env, false, &ret));
196+
}
197+
return ret;
198+
}
199+
154200
EXTERN_C_START
155201
napi_value Init(napi_env env, napi_value exports) {
156202
MyObject::Init(env, exports);
203+
204+
napi_property_descriptor descriptors[] = {
205+
DECLARE_NODE_API_PROPERTY("objectWrapDanglingReference",
206+
ObjectWrapDanglingReference),
207+
DECLARE_NODE_API_PROPERTY("objectWrapDanglingReferenceTest",
208+
ObjectWrapDanglingReferenceTest),
209+
};
210+
211+
NODE_API_CALL(
212+
env,
213+
napi_define_properties(env,
214+
exports,
215+
sizeof(descriptors) / sizeof(*descriptors),
216+
descriptors));
217+
157218
return exports;
158219
}
159220
EXTERN_C_END
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,11 @@
1+
'use strict';
2+
const common = require('../../common');
3+
const addon = require(`./build/${common.buildType}/6_object_wrap`);
4+
5+
(function scope() {
6+
addon.objectWrapDanglingReference({});
7+
})();
8+
9+
common.gcUntil('object-wrap-ref', () => {
10+
return addon.objectWrapDanglingReferenceTest();
11+
});

0 commit comments

Comments
 (0)