mirror of
https://github.com/nodejs/node.git
synced 2025-05-05 11:29:34 +00:00

* Get rid of recursive `make` when building the node binary. An earlier commit makes GYP write out rules that we can use for proper dependency tracking. * Use module name 'binding' in addons.md and addons-napi/*/binding.gyp. This massively simplifies the logic for generating the build rules. * Check in auto-generated add-on tests from `doc/api/addons.md`. The files change rarely and generating them dynamically causes no end of race conditions and special-casing during the build. PR-URL: https://github.com/nodejs/node/pull/17407 Reviewed-By: Richard Lau <riclau@uk.ibm.com>
121 lines
3.7 KiB
JavaScript
121 lines
3.7 KiB
JavaScript
'use strict';
|
|
// Flags: --expose-gc
|
|
|
|
const common = require('../../common');
|
|
const assert = require('assert');
|
|
|
|
const test_reference = require(`./build/${common.buildType}/binding`);
|
|
|
|
// This test script uses external values with finalizer callbacks
|
|
// in order to track when values get garbage-collected. Each invocation
|
|
// of a finalizer callback increments the finalizeCount property.
|
|
assert.strictEqual(test_reference.finalizeCount, 0);
|
|
|
|
// Run each test function in sequence,
|
|
// with an async delay and GC call between each.
|
|
function runTests(i, title, tests) {
|
|
if (tests[i]) {
|
|
if (typeof tests[i] === 'string') {
|
|
title = tests[i];
|
|
runTests(i + 1, title, tests);
|
|
} else {
|
|
try {
|
|
tests[i]();
|
|
} catch (e) {
|
|
console.error(`Test failed: ${title}`);
|
|
throw e;
|
|
}
|
|
setImmediate(() => {
|
|
global.gc();
|
|
runTests(i + 1, title, tests);
|
|
});
|
|
}
|
|
}
|
|
}
|
|
runTests(0, undefined, [
|
|
|
|
'External value without a finalizer',
|
|
() => {
|
|
const value = test_reference.createExternal();
|
|
assert.strictEqual(test_reference.finalizeCount, 0);
|
|
assert.strictEqual(typeof value, 'object');
|
|
test_reference.checkExternal(value);
|
|
},
|
|
() => {
|
|
assert.strictEqual(test_reference.finalizeCount, 0);
|
|
},
|
|
|
|
'External value with a finalizer',
|
|
() => {
|
|
const value = test_reference.createExternalWithFinalize();
|
|
assert.strictEqual(test_reference.finalizeCount, 0);
|
|
assert.strictEqual(typeof value, 'object');
|
|
test_reference.checkExternal(value);
|
|
},
|
|
() => {
|
|
assert.strictEqual(test_reference.finalizeCount, 1);
|
|
},
|
|
|
|
'Weak reference',
|
|
() => {
|
|
const value = test_reference.createExternalWithFinalize();
|
|
assert.strictEqual(test_reference.finalizeCount, 0);
|
|
test_reference.createReference(value, 0);
|
|
assert.strictEqual(test_reference.referenceValue, value);
|
|
},
|
|
() => {
|
|
// Value should be GC'd because there is only a weak ref
|
|
assert.strictEqual(test_reference.referenceValue, undefined);
|
|
assert.strictEqual(test_reference.finalizeCount, 1);
|
|
test_reference.deleteReference();
|
|
},
|
|
|
|
'Strong reference',
|
|
() => {
|
|
const value = test_reference.createExternalWithFinalize();
|
|
assert.strictEqual(test_reference.finalizeCount, 0);
|
|
test_reference.createReference(value, 1);
|
|
assert.strictEqual(test_reference.referenceValue, value);
|
|
},
|
|
() => {
|
|
// Value should NOT be GC'd because there is a strong ref
|
|
assert.strictEqual(test_reference.finalizeCount, 0);
|
|
test_reference.deleteReference();
|
|
},
|
|
() => {
|
|
// Value should be GC'd because the strong ref was deleted
|
|
assert.strictEqual(test_reference.finalizeCount, 1);
|
|
},
|
|
|
|
'Strong reference, increment then decrement to weak reference',
|
|
() => {
|
|
const value = test_reference.createExternalWithFinalize();
|
|
assert.strictEqual(test_reference.finalizeCount, 0);
|
|
test_reference.createReference(value, 1);
|
|
},
|
|
() => {
|
|
// Value should NOT be GC'd because there is a strong ref
|
|
assert.strictEqual(test_reference.finalizeCount, 0);
|
|
assert.strictEqual(test_reference.incrementRefcount(), 2);
|
|
},
|
|
() => {
|
|
// Value should NOT be GC'd because there is a strong ref
|
|
assert.strictEqual(test_reference.finalizeCount, 0);
|
|
assert.strictEqual(test_reference.decrementRefcount(), 1);
|
|
},
|
|
() => {
|
|
// Value should NOT be GC'd because there is a strong ref
|
|
assert.strictEqual(test_reference.finalizeCount, 0);
|
|
assert.strictEqual(test_reference.decrementRefcount(), 0);
|
|
},
|
|
() => {
|
|
// Value should be GC'd because the ref is now weak!
|
|
assert.strictEqual(test_reference.finalizeCount, 1);
|
|
test_reference.deleteReference();
|
|
},
|
|
() => {
|
|
// Value was already GC'd
|
|
assert.strictEqual(test_reference.finalizeCount, 1);
|
|
},
|
|
]);
|