Skip to content

Commit

Permalink
fixup! verify dangling napi_ref in napi_wrap
Browse files Browse the repository at this point in the history
  • Loading branch information
legendecas committed Nov 18, 2022
1 parent 57a58d1 commit 344493a
Show file tree
Hide file tree
Showing 4 changed files with 141 additions and 29 deletions.
72 changes: 48 additions & 24 deletions src/js_native_api_v8.cc
Original file line number Diff line number Diff line change
Expand Up @@ -218,7 +218,12 @@ inline napi_status Unwrap(napi_env env,
if (action == RemoveWrap) {
CHECK(obj->DeletePrivate(context, NAPI_PRIVATE_KEY(context, wrapper))
.FromJust());
delete reference;
if (reference->ownership() == Ownership::kUserland) {
// When the wrap is been removed, the finalizer should be reset.
reference->ResetFinalizer();
} else {
delete reference;
}
}

return GET_RETURN_STATUS(env);
Expand All @@ -245,7 +250,8 @@ class CallbackBundle {
bundle->env = env;

v8::Local<v8::Value> cbdata = v8::External::New(env->isolate, bundle);
Reference::New(env, cbdata, 0, true, Delete, bundle, nullptr);
Reference::New(
env, cbdata, 0, Ownership::kRuntime, Delete, bundle, nullptr);
return cbdata;
}
napi_env env; // Necessary to invoke C++ NAPI callback
Expand Down Expand Up @@ -429,16 +435,21 @@ inline napi_status Wrap(napi_env env,
// before then, then the finalize callback will never be invoked.)
// Therefore a finalize callback is required when returning a reference.
CHECK_ARG(env, finalize_cb);
reference = v8impl::Reference::New(
env, obj, 0, false, finalize_cb, native_object, finalize_hint);
reference = v8impl::Reference::New(env,
obj,
0,
v8impl::Ownership::kUserland,
finalize_cb,
native_object,
finalize_hint);
*result = reinterpret_cast<napi_ref>(reference);
} else {
// Create a self-deleting reference.
reference = v8impl::Reference::New(
env,
obj,
0,
true,
v8impl::Ownership::kRuntime,
finalize_cb,
native_object,
finalize_cb == nullptr ? nullptr : finalize_hint);
Expand All @@ -456,16 +467,22 @@ inline napi_status Wrap(napi_env env,

} // end of anonymous namespace

void Finalizer::ResetFinalizer() {
finalize_callback_ = nullptr;
finalize_data_ = nullptr;
finalize_hint_ = nullptr;
}

// Wrapper around v8impl::Persistent that implements reference counting.
RefBase::RefBase(napi_env env,
uint32_t initial_refcount,
bool delete_self,
Ownership ownership,
napi_finalize finalize_callback,
void* finalize_data,
void* finalize_hint)
: Finalizer(env, finalize_callback, finalize_data, finalize_hint),
refcount_(initial_refcount),
delete_self_(delete_self) {
ownership_(ownership) {
Link(finalize_callback == nullptr ? &env->reflist : &env->finalizing_reflist);
}

Expand All @@ -480,13 +497,13 @@ RefBase::~RefBase() {

RefBase* RefBase::New(napi_env env,
uint32_t initial_refcount,
bool delete_self,
Ownership ownership,
napi_finalize finalize_callback,
void* finalize_data,
void* finalize_hint) {
return new RefBase(env,
initial_refcount,
delete_self,
ownership,
finalize_callback,
finalize_data,
finalize_hint);
Expand All @@ -512,27 +529,29 @@ uint32_t RefBase::RefCount() {
}

void RefBase::Finalize() {
bool delete_self = delete_self_;
Ownership ownership = ownership_;
// Swap out the field finalize_callback so that it can not be accidentally
// called more than once.
napi_finalize finalize_callback = finalize_callback_;
finalize_callback_ = nullptr;
void* finalize_data = finalize_data_;
void* finalize_hint = finalize_hint_;
ResetFinalizer();

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

// If the RefBase is not self-destructive, userland code should delete it.
// Now delete it if it is self-destructive.
if (delete_self) {
// If the RefBase is not Ownership::kRuntime, userland code should delete it.
// Now delete it if it is Ownership::kRuntime.
if (ownership == Ownership::kRuntime) {
delete this;
}
}
Expand All @@ -554,14 +573,14 @@ Reference::~Reference() {
Reference* Reference::New(napi_env env,
v8::Local<v8::Value> value,
uint32_t initial_refcount,
bool delete_self,
Ownership ownership,
napi_finalize finalize_callback,
void* finalize_data,
void* finalize_hint) {
return new Reference(env,
value,
initial_refcount,
delete_self,
ownership,
finalize_callback,
finalize_data,
finalize_hint);
Expand Down Expand Up @@ -2298,8 +2317,13 @@ napi_status NAPI_CDECL napi_create_external(napi_env env,
if (finalize_cb) {
// The Reference object will delete itself after invoking the finalizer
// callback.
v8impl::Reference::New(
env, external_value, 0, true, finalize_cb, data, finalize_hint);
v8impl::Reference::New(env,
external_value,
0,
v8impl::Ownership::kRuntime,
finalize_cb,
data,
finalize_hint);
}

*result = v8impl::JsValueFromV8LocalValue(external_value);
Expand Down Expand Up @@ -2408,8 +2432,8 @@ napi_status NAPI_CDECL napi_create_reference(napi_env env,
return napi_set_last_error(env, napi_invalid_arg);
}

v8impl::Reference* reference =
v8impl::Reference::New(env, v8_value, initial_refcount, false);
v8impl::Reference* reference = v8impl::Reference::New(
env, v8_value, initial_refcount, v8impl::Ownership::kUserland);

*result = reinterpret_cast<napi_ref>(reference);
return napi_clear_last_error(env);
Expand Down Expand Up @@ -3116,8 +3140,8 @@ napi_status NAPI_CDECL napi_set_instance_data(napi_env env,
delete old_data;
}

env->instance_data =
v8impl::RefBase::New(env, 0, true, finalize_cb, data, finalize_hint);
env->instance_data = v8impl::RefBase::New(
env, 0, v8impl::Ownership::kRuntime, finalize_cb, data, finalize_hint);

return napi_clear_last_error(env);
}
Expand Down
22 changes: 18 additions & 4 deletions src/js_native_api_v8.h
Original file line number Diff line number Diff line change
Expand Up @@ -316,6 +316,8 @@ class Finalizer {
void* data() { return finalize_data_; }
void* hint() { return finalize_hint_; }

void ResetFinalizer();

protected:
napi_env env_;
napi_finalize finalize_callback_;
Expand All @@ -337,20 +339,30 @@ class TryCatch : public v8::TryCatch {
napi_env _env;
};

// Ownership of a reference.
enum class Ownership {
// The reference is owned by the runtime. No userland call is needed to
// destruct the reference.
kRuntime,
// The reference is owned by the userland. User code is responsible to delete
// the reference with appropriate node-api calls.
kUserland,
};

// Wrapper around Finalizer that implements reference counting.
class RefBase : public Finalizer, public RefTracker {
protected:
RefBase(napi_env env,
uint32_t initial_refcount,
bool delete_self,
Ownership ownership,
napi_finalize finalize_callback,
void* finalize_data,
void* finalize_hint);

public:
static RefBase* New(napi_env env,
uint32_t initial_refcount,
bool delete_self,
Ownership ownership,
napi_finalize finalize_callback,
void* finalize_data,
void* finalize_hint);
Expand All @@ -361,12 +373,14 @@ class RefBase : public Finalizer, public RefTracker {
uint32_t Unref();
uint32_t RefCount();

Ownership ownership() { return ownership_; }

protected:
void Finalize() override;

private:
uint32_t refcount_;
bool delete_self_;
Ownership ownership_;
};

// Wrapper around v8impl::Persistent.
Expand All @@ -379,7 +393,7 @@ class Reference : public RefBase {
static Reference* New(napi_env env,
v8::Local<v8::Value> value,
uint32_t initial_refcount,
bool delete_self,
Ownership ownership,
napi_finalize finalize_callback = nullptr,
void* finalize_data = nullptr,
void* finalize_hint = nullptr);
Expand Down
63 changes: 62 additions & 1 deletion test/js-native-api/6_object_wrap/6_object_wrap.cc
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
#include "myobject.h"
#include "../common.h"
#include "assert.h"
#include "myobject.h"

napi_ref MyObject::constructor;

Expand Down Expand Up @@ -151,9 +152,69 @@ napi_value MyObject::Multiply(napi_env env, napi_callback_info info) {
return instance;
}

// This finalizer should never be invoked.
void ObjectWrapDanglingReferenceFinalizer(napi_env env,
void* finalize_data,
void* finalize_hint) {
assert(0 && "unreachable");
}

napi_ref dangling_ref;
napi_value ObjectWrapDanglingReference(napi_env env, napi_callback_info info) {
size_t argc = 1;
napi_value args[1];
NODE_API_CALL(env,
napi_get_cb_info(env, info, &argc, args, nullptr, nullptr));

// Create a napi_wrap and remove it immediately, whilst leaving the out-param
// ref dangling (not deleted).
NODE_API_CALL(env,
napi_wrap(env,
args[0],
nullptr,
ObjectWrapDanglingReferenceFinalizer,
nullptr,
&dangling_ref));
NODE_API_CALL(env, napi_remove_wrap(env, args[0], nullptr));

return args[0];
}

napi_value ObjectWrapDanglingReferenceTest(napi_env env,
napi_callback_info info) {
napi_value out;
napi_value ret;
NODE_API_CALL(env, napi_get_reference_value(env, dangling_ref, &out));

if (out == nullptr) {
// If the napi_ref has been invalidated, delete it.
NODE_API_CALL(env, napi_delete_reference(env, dangling_ref));
NODE_API_CALL(env, napi_get_boolean(env, true, &ret));
} else {
// The dangling napi_ref is still valid.
NODE_API_CALL(env, napi_get_boolean(env, false, &ret));
}
return ret;
}

EXTERN_C_START
napi_value Init(napi_env env, napi_value exports) {
MyObject::Init(env, exports);

napi_property_descriptor descriptors[] = {
DECLARE_NODE_API_PROPERTY("objectWrapDanglingReference",
ObjectWrapDanglingReference),
DECLARE_NODE_API_PROPERTY("objectWrapDanglingReferenceTest",
ObjectWrapDanglingReferenceTest),
};

NODE_API_CALL(
env,
napi_define_properties(env,
exports,
sizeof(descriptors) / sizeof(*descriptors),
descriptors));

return exports;
}
EXTERN_C_END
13 changes: 13 additions & 0 deletions test/js-native-api/6_object_wrap/test-object-wrap-ref.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,13 @@
// Flags: --expose-gc

'use strict';
const common = require('../../common');
const addon = require(`./build/${common.buildType}/6_object_wrap`);

(function scope() {
addon.objectWrapDanglingReference({});
})();

common.gcUntil('object-wrap-ref', () => {
return addon.objectWrapDanglingReferenceTest();
});

0 comments on commit 344493a

Please sign in to comment.