Skip to content

Commit b3f9b38

Browse files
committed
n-api: check against invalid handle scope usage
Fixes: #16175 PR-URL: #16201 Reviewed-By: Michael Dawson <[email protected]> Reviewed-By: James M Snell <[email protected]> Reviewed-By: Ben Noordhuis <[email protected]> Reviewed-By: Colin Ihrig <[email protected]>
1 parent 19ab261 commit b3f9b38

File tree

4 files changed

+22
-8
lines changed

4 files changed

+22
-8
lines changed

src/node_api.cc

+15
Original file line numberDiff line numberDiff line change
@@ -42,6 +42,7 @@ struct napi_env__ {
4242
v8::Persistent<v8::ObjectTemplate> function_data_template;
4343
v8::Persistent<v8::ObjectTemplate> accessor_data_template;
4444
napi_extended_error_info last_error;
45+
int open_handle_scopes = 0;
4546
};
4647

4748
#define ENV_OBJECT_TEMPLATE(env, prefix, destination, field_count) \
@@ -499,12 +500,16 @@ class CallbackWrapperBase : public CallbackWrapper {
499500
// Make sure any errors encountered last time we were in N-API are gone.
500501
napi_clear_last_error(env);
501502

503+
int open_handle_scopes = env->open_handle_scopes;
504+
502505
napi_value result = cb(env, cbinfo_wrapper);
503506

504507
if (result != nullptr) {
505508
this->SetReturnValue(result);
506509
}
507510

511+
CHECK_EQ(env->open_handle_scopes, open_handle_scopes);
512+
508513
if (!env->last_exception.IsEmpty()) {
509514
isolate->ThrowException(
510515
v8::Local<v8::Value>::New(isolate, env->last_exception));
@@ -2580,6 +2585,7 @@ napi_status napi_open_handle_scope(napi_env env, napi_handle_scope* result) {
25802585

25812586
*result = v8impl::JsHandleScopeFromV8HandleScope(
25822587
new v8impl::HandleScopeWrapper(env->isolate));
2588+
env->open_handle_scopes++;
25832589
return napi_clear_last_error(env);
25842590
}
25852591

@@ -2588,7 +2594,11 @@ napi_status napi_close_handle_scope(napi_env env, napi_handle_scope scope) {
25882594
// JS exceptions.
25892595
CHECK_ENV(env);
25902596
CHECK_ARG(env, scope);
2597+
if (env->open_handle_scopes == 0) {
2598+
return napi_handle_scope_mismatch;
2599+
}
25912600

2601+
env->open_handle_scopes--;
25922602
delete v8impl::V8HandleScopeFromJsHandleScope(scope);
25932603
return napi_clear_last_error(env);
25942604
}
@@ -2603,6 +2613,7 @@ napi_status napi_open_escapable_handle_scope(
26032613

26042614
*result = v8impl::JsEscapableHandleScopeFromV8EscapableHandleScope(
26052615
new v8impl::EscapableHandleScopeWrapper(env->isolate));
2616+
env->open_handle_scopes++;
26062617
return napi_clear_last_error(env);
26072618
}
26082619

@@ -2613,8 +2624,12 @@ napi_status napi_close_escapable_handle_scope(
26132624
// JS exceptions.
26142625
CHECK_ENV(env);
26152626
CHECK_ARG(env, scope);
2627+
if (env->open_handle_scopes == 0) {
2628+
return napi_handle_scope_mismatch;
2629+
}
26162630

26172631
delete v8impl::V8EscapableHandleScopeFromJsEscapableHandleScope(scope);
2632+
env->open_handle_scopes--;
26182633
return napi_clear_last_error(env);
26192634
}
26202635

src/node_api_types.h

+2-1
Original file line numberDiff line numberDiff line change
@@ -69,7 +69,8 @@ typedef enum {
6969
napi_generic_failure,
7070
napi_pending_exception,
7171
napi_cancelled,
72-
napi_escape_called_twice
72+
napi_escape_called_twice,
73+
napi_handle_scope_mismatch
7374
} napi_status;
7475

7576
typedef napi_value (*napi_callback)(napi_env env,

test/addons-napi/test_handle_scope/test.js

+1-5
Original file line numberDiff line numberDiff line change
@@ -10,11 +10,7 @@ testHandleScope.NewScope();
1010

1111
assert.ok(testHandleScope.NewScopeEscape() instanceof Object);
1212

13-
assert.throws(
14-
() => {
15-
testHandleScope.NewScopeEscapeTwice();
16-
},
17-
Error);
13+
testHandleScope.NewScopeEscapeTwice();
1814

1915
assert.throws(
2016
() => {

test/addons-napi/test_handle_scope/test_handle_scope.c

+4-2
Original file line numberDiff line numberDiff line change
@@ -33,13 +33,15 @@ napi_value NewScopeEscapeTwice(napi_env env, napi_callback_info info) {
3333
napi_escapable_handle_scope scope;
3434
napi_value output = NULL;
3535
napi_value escapee = NULL;
36+
napi_status status;
3637

3738
NAPI_CALL(env, napi_open_escapable_handle_scope(env, &scope));
3839
NAPI_CALL(env, napi_create_object(env, &output));
3940
NAPI_CALL(env, napi_escape_handle(env, scope, output, &escapee));
40-
NAPI_CALL(env, napi_escape_handle(env, scope, output, &escapee));
41+
status = napi_escape_handle(env, scope, output, &escapee);
42+
NAPI_ASSERT(env, status == napi_escape_called_twice, "Escaping twice fails");
4143
NAPI_CALL(env, napi_close_escapable_handle_scope(env, scope));
42-
return escapee;
44+
return NULL;
4345
}
4446

4547
napi_value NewScopeWithException(napi_env env, napi_callback_info info) {

0 commit comments

Comments
 (0)