mirror of
https://github.com/nodejs/node.git
synced 2025-05-02 22:16:31 +00:00

Throwing an exception from a finalizer can cause the following fatal error: Error: async hook stack has become corrupted (actual: 2, expected: 0) 1: 0x970b5a node::InternalCallbackScope::~InternalCallbackScope() [./node] 2: 0x99dda0 node::Environment::RunTimers(uv_timer_s*) [./node] 3: 0x13d8b22 [./node] 4: 0x13dbe42 uv_run [./node] 5: 0xa57974 node::NodeMainInstance::Run() [./node] 6: 0x9dbc17 node::Start(int, char**) [./node] 7: 0x7f4965417f43 __libc_start_main [/lib64/libc.so.6] 8: 0x96f4ae _start [./node] By https://github.com/nodejs/node/issues/34341#issuecomment-658426281, calling into JS from a finalizer and/or throwing exceptions from there is not advised, because the stack may or may not be set up for JS execution. The best solution is to run the user's finalizer from a `SetImmediate()` callback. Signed-off-by: Gabriel Schulhof <gabriel.schulhof@intel.com> Fixes: https://github.com/nodejs/node/issues/34341 PR-URL: https://github.com/nodejs/node/pull/34386 Reviewed-By: James M Snell <jasnell@gmail.com> Reviewed-By: Anna Henningsen <anna@addaleax.net> Reviewed-By: Michael Dawson <michael_dawson@ca.ibm.com> Reviewed-By: Juan José Arboleda <soyjuanarbol@gmail.com>
33 lines
1.1 KiB
JavaScript
33 lines
1.1 KiB
JavaScript
'use strict';
|
|
if (process.argv[2] === 'child') {
|
|
const common = require('../../common');
|
|
// Trying, catching the exception, and finding the bindings at the `Error`'s
|
|
// `binding` property is done intentionally, because we're also testing what
|
|
// happens when the add-on entry point throws. See test.js.
|
|
try {
|
|
require(`./build/${common.buildType}/test_exception`);
|
|
} catch (anException) {
|
|
anException.binding.createExternal();
|
|
}
|
|
|
|
// Collect garbage 10 times. At least one of those should throw the exception
|
|
// and cause the whole process to bail with it, its text printed to stderr and
|
|
// asserted by the parent process to match expectations.
|
|
let gcCount = 10;
|
|
(function gcLoop() {
|
|
global.gc();
|
|
if (--gcCount > 0) {
|
|
setImmediate(() => gcLoop());
|
|
}
|
|
})();
|
|
return;
|
|
}
|
|
|
|
const assert = require('assert');
|
|
const { spawnSync } = require('child_process');
|
|
const child = spawnSync(process.execPath, [
|
|
'--expose-gc', __filename, 'child'
|
|
]);
|
|
assert.strictEqual(child.signal, null);
|
|
assert.match(child.stderr.toString(), /Error during Finalize/m);
|