mirror of
https://github.com/nodejs/node.git
synced 2025-04-29 22:40:57 +00:00

Currently, a reference is being unlinked from the list of references tracked by the environment when `v8impl::Reference::Delete` is called. This causes a leak when deletion must be deferred because the finalizer hasn't yet run, but the finalizer does not run because environment teardown is in progress, and so no more gc runs will happen, and the `FinalizeAll` run that happens during environment teardown does not catch the reference because it's no longer in the list. The test below will fail when running with ASAN: ``` ./node ./test/node-api/test_worker_terminate_finalization/test.js ``` OTOH if, to address the above leak, we make a special case to not unlink a reference during environment teardown, we run into a situation where the reference gets deleted by `v8impl::Reference::Delete` but does not get unlinked because it's environment teardown time. This leaves a stale pointer in the linked list which will result in a use-after-free in `FinalizeAll` during environment teardown. The test below will fail if we make the above change: ``` ./node -e "require('./test/node-api/test_instance_data/build/Release/test_ref_then_set.node');" ``` Thus, we unlink a reference precisely when we destroy it – in its destructor. Refs: https://github.com/nodejs/node/issues/34731 Refs: https://github.com/nodejs/node/pull/34839 Refs: https://github.com/nodejs/node/issues/35620 Refs: https://github.com/nodejs/node/issues/35777 Fixes: https://github.com/nodejs/node/issues/35778 Signed-off-by: Gabriel Schulhof <gabriel.schulhof@intel.com> PR-URL: https://github.com/nodejs/node/pull/35933 Reviewed-By: Rich Trott <rtrott@gmail.com> Reviewed-By: Michael Dawson <midawson@redhat.com> Reviewed-By: Zeyu Yang <himself65@outlook.com>
24 lines
779 B
JavaScript
24 lines
779 B
JavaScript
'use strict';
|
|
const common = require('../../common');
|
|
|
|
// Refs: https://github.com/nodejs/node/issues/34731
|
|
// Refs: https://github.com/nodejs/node/pull/35777
|
|
// Refs: https://github.com/nodejs/node/issues/35778
|
|
|
|
const { Worker, isMainThread } = require('worker_threads');
|
|
|
|
if (isMainThread) {
|
|
const worker = new Worker(__filename);
|
|
worker.on('error', common.mustNotCall());
|
|
} else {
|
|
const { Test } =
|
|
require(`./build/${common.buildType}/test_worker_terminate_finalization`);
|
|
|
|
// Spin up thread and call add-on create the right sequence
|
|
// of rerences to hit the case reported in
|
|
// https://github.com/nodejs/node-addon-api/issues/722
|
|
// will crash if run under debug and its not possible to
|
|
// create object in the specific finalizer
|
|
Test(new Object());
|
|
}
|