Skip to content

Commit ff681e7

Browse files
committed
add test to crash on GC state access and update docs
1 parent c9b4b6f commit ff681e7

9 files changed

+153
-25
lines changed

doc/api/n-api.md

+36
Original file line numberDiff line numberDiff line change
@@ -5468,6 +5468,42 @@ invocation. If it is deleted before then, then the finalize callback may never
54685468
be invoked. Therefore, when obtaining a reference a finalize callback is also
54695469
required in order to enable correct disposal of the reference.
54705470

5471+
#### `node_api_post_finalizer`
5472+
5473+
<!-- YAML
5474+
added: REPLACEME
5475+
-->
5476+
5477+
> Stability: 1 - Experimental
5478+
5479+
```c
5480+
napi_status node_api_post_finalizer(napi_env env,
5481+
napi_finalize finalize_cb,
5482+
void* finalize_data,
5483+
void* finalize_hint);
5484+
```
5485+
5486+
* `[in] env`: The environment that the API is invoked under.
5487+
* `[in] finalize_cb`: Native callback that will be used to free the
5488+
native data when the JavaScript object has been garbage-collected.
5489+
[`napi_finalize`][] provides more details.
5490+
* `[in] finalize_data`: Optional data to be passed to `finalize_cb`.
5491+
* `[in] finalize_hint`: Optional contextual hint that is passed to the
5492+
finalize callback.
5493+
5494+
Returns `napi_ok` if the API succeeded.
5495+
5496+
Schedules `napi_finalize` callback to be called asynchronously in the
5497+
event loop.
5498+
5499+
This API must be called inside of a finalizer if it must call any code
5500+
that may affect the state of GC (garbage collector).
5501+
5502+
The finalizers are called while GC collects objects. At that point of time
5503+
calling any API that may cause changes in GC state will cause unpredictable
5504+
behavior and crashes. The `node_api_post_finalizer` helps to work around
5505+
this limitation by running code outside of the GC finalization.
5506+
54715507
## Simple asynchronous operations
54725508

54735509
Addon modules often need to leverage async helpers from libuv as part of their

src/js_native_api_v8.cc

+24-6
Original file line numberDiff line numberDiff line change
@@ -62,14 +62,14 @@ void napi_env__::InvokeFinalizerFromGC(v8impl::RefTracker* finalizer) {
6262
EnqueueFinalizer(finalizer);
6363
} else {
6464
// The experimental code calls finalizers immediately to release native
65-
// objects as soon as possible, but it suspends use of JS from finalizer.
66-
// If JS calls are needed, then the finalizer code must call
67-
// node_api_post_finalizer.
65+
// objects as soon as possible. In that state any code that may affect GC
66+
// state causes a fatal error. To work around this issue the finalizer code
67+
// must call node_api_post_finalizer.
6868
if (last_error.error_code == napi_ok && last_exception.IsEmpty()) {
69-
bool saved_suspend_call_into_js = suspend_call_into_js;
70-
suspend_call_into_js = true;
69+
auto restore_state = node::OnScopeLeave(
70+
[this, saved = in_gc_finalizer] { in_gc_finalizer = saved; });
71+
in_gc_finalizer = true;
7172
finalizer->Finalize();
72-
suspend_call_into_js = saved_suspend_call_into_js;
7373
} else {
7474
// The finalizers can be run in the middle of JS or C++ code.
7575
// That code may be in an error state. In that case use the asynchronous
@@ -93,6 +93,7 @@ napi_status NewString(napi_env env,
9393
CHECK_ARG(env, result);
9494
RETURN_STATUS_IF_FALSE(
9595
env, (length == NAPI_AUTO_LENGTH) || length <= INT_MAX, napi_invalid_arg);
96+
env->CheckGCAccess();
9697

9798
auto isolate = env->isolate;
9899
auto str_maybe = string_maker(isolate);
@@ -1539,6 +1540,7 @@ napi_status NAPI_CDECL napi_get_prototype(napi_env env,
15391540
napi_status NAPI_CDECL napi_create_object(napi_env env, napi_value* result) {
15401541
CHECK_ENV(env);
15411542
CHECK_ARG(env, result);
1543+
env->CheckGCAccess();
15421544

15431545
*result = v8impl::JsValueFromV8LocalValue(v8::Object::New(env->isolate));
15441546

@@ -1548,6 +1550,7 @@ napi_status NAPI_CDECL napi_create_object(napi_env env, napi_value* result) {
15481550
napi_status NAPI_CDECL napi_create_array(napi_env env, napi_value* result) {
15491551
CHECK_ENV(env);
15501552
CHECK_ARG(env, result);
1553+
env->CheckGCAccess();
15511554

15521555
*result = v8impl::JsValueFromV8LocalValue(v8::Array::New(env->isolate));
15531556

@@ -1559,6 +1562,7 @@ napi_status NAPI_CDECL napi_create_array_with_length(napi_env env,
15591562
napi_value* result) {
15601563
CHECK_ENV(env);
15611564
CHECK_ARG(env, result);
1565+
env->CheckGCAccess();
15621566

15631567
*result =
15641568
v8impl::JsValueFromV8LocalValue(v8::Array::New(env->isolate, length));
@@ -1659,6 +1663,7 @@ napi_status NAPI_CDECL napi_create_double(napi_env env,
16591663
napi_value* result) {
16601664
CHECK_ENV(env);
16611665
CHECK_ARG(env, result);
1666+
env->CheckGCAccess();
16621667

16631668
*result =
16641669
v8impl::JsValueFromV8LocalValue(v8::Number::New(env->isolate, value));
@@ -1671,6 +1676,7 @@ napi_status NAPI_CDECL napi_create_int32(napi_env env,
16711676
napi_value* result) {
16721677
CHECK_ENV(env);
16731678
CHECK_ARG(env, result);
1679+
env->CheckGCAccess();
16741680

16751681
*result =
16761682
v8impl::JsValueFromV8LocalValue(v8::Integer::New(env->isolate, value));
@@ -1683,6 +1689,7 @@ napi_status NAPI_CDECL napi_create_uint32(napi_env env,
16831689
napi_value* result) {
16841690
CHECK_ENV(env);
16851691
CHECK_ARG(env, result);
1692+
env->CheckGCAccess();
16861693

16871694
*result = v8impl::JsValueFromV8LocalValue(
16881695
v8::Integer::NewFromUnsigned(env->isolate, value));
@@ -1695,6 +1702,7 @@ napi_status NAPI_CDECL napi_create_int64(napi_env env,
16951702
napi_value* result) {
16961703
CHECK_ENV(env);
16971704
CHECK_ARG(env, result);
1705+
env->CheckGCAccess();
16981706

16991707
*result = v8impl::JsValueFromV8LocalValue(
17001708
v8::Number::New(env->isolate, static_cast<double>(value)));
@@ -1707,6 +1715,7 @@ napi_status NAPI_CDECL napi_create_bigint_int64(napi_env env,
17071715
napi_value* result) {
17081716
CHECK_ENV(env);
17091717
CHECK_ARG(env, result);
1718+
env->CheckGCAccess();
17101719

17111720
*result =
17121721
v8impl::JsValueFromV8LocalValue(v8::BigInt::New(env->isolate, value));
@@ -1719,6 +1728,7 @@ napi_status NAPI_CDECL napi_create_bigint_uint64(napi_env env,
17191728
napi_value* result) {
17201729
CHECK_ENV(env);
17211730
CHECK_ARG(env, result);
1731+
env->CheckGCAccess();
17221732

17231733
*result = v8impl::JsValueFromV8LocalValue(
17241734
v8::BigInt::NewFromUnsigned(env->isolate, value));
@@ -1734,6 +1744,7 @@ napi_status NAPI_CDECL napi_create_bigint_words(napi_env env,
17341744
NAPI_PREAMBLE(env);
17351745
CHECK_ARG(env, words);
17361746
CHECK_ARG(env, result);
1747+
env->CheckGCAccess();
17371748

17381749
v8::Local<v8::Context> context = env->context();
17391750

@@ -1753,6 +1764,7 @@ napi_status NAPI_CDECL napi_get_boolean(napi_env env,
17531764
napi_value* result) {
17541765
CHECK_ENV(env);
17551766
CHECK_ARG(env, result);
1767+
env->CheckGCAccess();
17561768

17571769
v8::Isolate* isolate = env->isolate;
17581770

@@ -1770,6 +1782,7 @@ napi_status NAPI_CDECL napi_create_symbol(napi_env env,
17701782
napi_value* result) {
17711783
CHECK_ENV(env);
17721784
CHECK_ARG(env, result);
1785+
env->CheckGCAccess();
17731786

17741787
v8::Isolate* isolate = env->isolate;
17751788

@@ -1792,6 +1805,7 @@ napi_status NAPI_CDECL node_api_symbol_for(napi_env env,
17921805
napi_value* result) {
17931806
CHECK_ENV(env);
17941807
CHECK_ARG(env, result);
1808+
env->CheckGCAccess();
17951809

17961810
napi_value js_description_string;
17971811
STATUS_CALL(napi_create_string_utf8(
@@ -1838,6 +1852,7 @@ napi_status NAPI_CDECL napi_create_error(napi_env env,
18381852
CHECK_ENV(env);
18391853
CHECK_ARG(env, msg);
18401854
CHECK_ARG(env, result);
1855+
env->CheckGCAccess();
18411856

18421857
v8::Local<v8::Value> message_value = v8impl::V8LocalValueFromJsValue(msg);
18431858
RETURN_STATUS_IF_FALSE(env, message_value->IsString(), napi_string_expected);
@@ -1858,6 +1873,7 @@ napi_status NAPI_CDECL napi_create_type_error(napi_env env,
18581873
CHECK_ENV(env);
18591874
CHECK_ARG(env, msg);
18601875
CHECK_ARG(env, result);
1876+
env->CheckGCAccess();
18611877

18621878
v8::Local<v8::Value> message_value = v8impl::V8LocalValueFromJsValue(msg);
18631879
RETURN_STATUS_IF_FALSE(env, message_value->IsString(), napi_string_expected);
@@ -1878,6 +1894,7 @@ napi_status NAPI_CDECL napi_create_range_error(napi_env env,
18781894
CHECK_ENV(env);
18791895
CHECK_ARG(env, msg);
18801896
CHECK_ARG(env, result);
1897+
env->CheckGCAccess();
18811898

18821899
v8::Local<v8::Value> message_value = v8impl::V8LocalValueFromJsValue(msg);
18831900
RETURN_STATUS_IF_FALSE(env, message_value->IsString(), napi_string_expected);
@@ -1898,6 +1915,7 @@ napi_status NAPI_CDECL node_api_create_syntax_error(napi_env env,
18981915
CHECK_ENV(env);
18991916
CHECK_ARG(env, msg);
19001917
CHECK_ARG(env, result);
1918+
env->CheckGCAccess();
19011919

19021920
v8::Local<v8::Value> message_value = v8impl::V8LocalValueFromJsValue(msg);
19031921
RETURN_STATUS_IF_FALSE(env, message_value->IsString(), napi_string_expected);

src/js_native_api_v8.h

+16-3
Original file line numberDiff line numberDiff line change
@@ -68,7 +68,7 @@ struct napi_env__ {
6868
if (--refs == 0) DeleteMe();
6969
}
7070

71-
virtual bool can_call_into_js() const { return !suspend_call_into_js; }
71+
virtual bool can_call_into_js() const { return true; }
7272

7373
static inline void HandleThrow(napi_env env, v8::Local<v8::Value> value) {
7474
if (env->terminatedOrTerminating()) {
@@ -102,7 +102,6 @@ struct napi_env__ {
102102
// Call finalizer immediately.
103103
virtual void CallFinalizer(napi_finalize cb, void* data, void* hint) {
104104
v8::HandleScope handle_scope(isolate);
105-
v8::Context::Scope context_scope(context());
106105
CallIntoModule([&](napi_env env) { cb(env, data, hint); });
107106
}
108107

@@ -134,6 +133,19 @@ struct napi_env__ {
134133
delete this;
135134
}
136135

136+
void CheckGCAccess() {
137+
if (module_api_version == NAPI_VERSION_EXPERIMENTAL && in_gc_finalizer) {
138+
v8impl::OnFatalError(
139+
nullptr,
140+
"Finalizer is calling a function that may affect GC state.\n"
141+
"The finalizers are run directly from GC and must not affect GC "
142+
"state.\n"
143+
"Use `node_api_post_finalizer` from inside of the finalizer to work "
144+
"around this issue.\n"
145+
"It schedules the call as a new task in the event loop.");
146+
}
147+
}
148+
137149
v8::Isolate* const isolate; // Shortcut for context()->GetIsolate()
138150
v8impl::Persistent<v8::Context> context_persistent;
139151

@@ -152,7 +164,7 @@ struct napi_env__ {
152164
int refs = 1;
153165
void* instance_data = nullptr;
154166
int32_t module_api_version = NODE_API_DEFAULT_MODULE_API_VERSION;
155-
bool suspend_call_into_js = false;
167+
bool in_gc_finalizer = false;
156168

157169
protected:
158170
// Should not be deleted directly. Delete with `napi_env__::DeleteMe()`
@@ -218,6 +230,7 @@ inline napi_status napi_set_last_error(napi_env env,
218230
CHECK_ENV((env)); \
219231
RETURN_STATUS_IF_FALSE( \
220232
(env), (env)->last_exception.IsEmpty(), napi_pending_exception); \
233+
(env)->CheckGCAccess(); \
221234
RETURN_STATUS_IF_FALSE((env), \
222235
(env)->can_call_into_js(), \
223236
(env->module_api_version == NAPI_VERSION_EXPERIMENTAL \

src/js_native_api_v8_internals.h

+6
Original file line numberDiff line numberDiff line change
@@ -17,6 +17,7 @@
1717

1818
#include "env.h"
1919
#include "gtest/gtest_prod.h"
20+
#include "node_errors.h"
2021
#include "node_internals.h"
2122

2223
#define NAPI_ARRAYSIZE(array) node::arraysize((array))
@@ -34,6 +35,11 @@ using Persistent = v8::Global<T>;
3435

3536
using PersistentToLocal = node::PersistentToLocal;
3637

38+
[[noreturn]] inline void OnFatalError(const char* location,
39+
const char* message) {
40+
node::OnFatalError(location, message);
41+
}
42+
3743
} // end of namespace v8impl
3844

3945
#endif // SRC_JS_NATIVE_API_V8_INTERNALS_H_

src/node_api.cc

+1-1
Original file line numberDiff line numberDiff line change
@@ -33,7 +33,7 @@ void node_napi_env__::DeleteMe() {
3333
}
3434

3535
bool node_napi_env__::can_call_into_js() const {
36-
return Super::can_call_into_js() && node_env()->can_call_into_js();
36+
return node_env()->can_call_into_js();
3737
}
3838

3939
void node_napi_env__::CallFinalizer(napi_finalize cb, void* data, void* hint) {

src/node_api_internals.h

-2
Original file line numberDiff line numberDiff line change
@@ -9,8 +9,6 @@
99
#include "util-inl.h"
1010

1111
struct node_napi_env__ : public napi_env__ {
12-
using Super = napi_env__;
13-
1412
node_napi_env__(v8::Local<v8::Context> context,
1513
const std::string& module_filename,
1614
int32_t module_api_version);

test/js-native-api/test_finalizer/test.js

+4-7
Original file line numberDiff line numberDiff line change
@@ -5,15 +5,12 @@ const common = require('../../common');
55
const test_finalizer = require(`./build/${common.buildType}/test_finalizer`);
66
const assert = require('assert');
77

8-
function addFinalizer() {
9-
// Define obj in a function context to let it GC-collected.
8+
(() => {
109
const obj = {};
1110
test_finalizer.addFinalizer(obj);
12-
}
13-
14-
addFinalizer();
11+
})();
1512

16-
for (let i = 0; i < 1000; ++i) {
13+
for (let i = 0; i < 10; ++i) {
1714
global.gc();
1815
if (test_finalizer.getFinalizerCallCount() === 1) {
1916
break;
@@ -28,7 +25,7 @@ async function runAsyncTests() {
2825
const obj = {};
2926
test_finalizer.addFinalizerWithJS(obj, () => { js_is_called = true; });
3027
})();
31-
await common.gcUntil('test JS finalizer',
28+
await common.gcUntil('ensure JS finalizer called',
3229
() => (test_finalizer.getFinalizerCallCount() === 2));
3330
assert(js_is_called);
3431
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,31 @@
1+
'use strict';
2+
3+
if (process.argv[2] === 'child') {
4+
const common = require('../../common');
5+
const test_finalizer = require(`./build/${common.buildType}/test_finalizer`);
6+
7+
(() => {
8+
const obj = {};
9+
test_finalizer.addFinalizerFailOnJS(obj);
10+
})();
11+
12+
// Collect garbage 10 times. At least one of those should throw the exception
13+
// and cause the whole process to bail with it, its text printed to stderr and
14+
// asserted by the parent process to match expectations.
15+
let gcCount = 10;
16+
(function gcLoop() {
17+
global.gc();
18+
if (--gcCount > 0) {
19+
setImmediate(() => gcLoop());
20+
}
21+
})();
22+
return;
23+
}
24+
25+
const assert = require('assert');
26+
const { spawnSync } = require('child_process');
27+
const child = spawnSync(process.execPath, [
28+
'--expose-gc', __filename, 'child',
29+
]);
30+
assert.strictEqual(child.signal, null);
31+
assert.match(child.stderr.toString(), /Finalizer is calling a function that may affect GC state/);

0 commit comments

Comments
 (0)