mirror of
https://github.com/nodejs/node.git
synced 2025-04-28 13:40:37 +00:00

Previously when managing the importModuleDynamically callback of vm.compileFunction(), we use an ID number as the host defined option and maintain a per-Environment ID -> CompiledFnEntry map to retain the top-level referrer function returned by vm.compileFunction() in order to pass it back to the callback, but it would leak because with how we used v8::Persistent to maintain this reference, V8 would not be able to understand the cycle and would just think that the CompiledFnEntry was supposed to live forever. We made an attempt to make that reference known to V8 by making the CompiledFnEntry weak and using a private symbol to make CompiledFnEntry strongly references the top-level referrer function in https://github.com/nodejs/node/pull/46785, but that turned out to be unsound, because the there's no guarantee that the top-level function must be alive while import() can still be initiated from that function, since V8 could discard the top-level function and only keep inner functions alive, so relying on the top-level function to keep the CompiledFnEntry alive could result in use-after-free which caused a revert of that fix. With this patch we use a symbol in the host defined options instead of a number, because with the stage-3 symbol-as-weakmap-keys proposal we could directly use that symbol to keep the referrer alive using a WeakMap. As a bonus this also keeps the other kinds of referrers alive as long as import() can still be initiated from that Script/Module, so this also fixes the long-standing crash caused by vm.Script being GC'ed too early when its importModuleDynamically callback still needs it. PR-URL: https://github.com/nodejs/node/pull/48510 Refs: https://github.com/nodejs/node/issues/44211 Refs: https://github.com/nodejs/node/issues/42080 Refs: https://github.com/nodejs/node/issues/47096 Refs: https://github.com/nodejs/node/issues/43205 Refs: https://github.com/nodejs/node/issues/38695 Reviewed-By: Antoine du Hamel <duhamelantoine1995@gmail.com> Reviewed-By: Benjamin Gruenbaum <benjamingr@gmail.com> Reviewed-By: Stephen Belanger <admin@stephenbelanger.com>
342 lines
10 KiB
JavaScript
342 lines
10 KiB
JavaScript
// Copyright Joyent, Inc. and other Node contributors.
|
|
//
|
|
// Permission is hereby granted, free of charge, to any person obtaining a
|
|
// copy of this software and associated documentation files (the
|
|
// "Software"), to deal in the Software without restriction, including
|
|
// without limitation the rights to use, copy, modify, merge, publish,
|
|
// distribute, sublicense, and/or sell copies of the Software, and to permit
|
|
// persons to whom the Software is furnished to do so, subject to the
|
|
// following conditions:
|
|
//
|
|
// The above copyright notice and this permission notice shall be included
|
|
// in all copies or substantial portions of the Software.
|
|
//
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
|
|
// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN
|
|
// NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,
|
|
// DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
|
|
// OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE
|
|
// USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
|
|
'use strict';
|
|
|
|
const {
|
|
ArrayPrototypeForEach,
|
|
Symbol,
|
|
PromiseReject,
|
|
ReflectApply,
|
|
} = primordials;
|
|
|
|
const {
|
|
ContextifyScript,
|
|
makeContext,
|
|
constants,
|
|
measureMemory: _measureMemory,
|
|
} = internalBinding('contextify');
|
|
const {
|
|
ERR_CONTEXT_NOT_INITIALIZED,
|
|
ERR_INVALID_ARG_TYPE,
|
|
} = require('internal/errors').codes;
|
|
const {
|
|
validateBoolean,
|
|
validateBuffer,
|
|
validateFunction,
|
|
validateInt32,
|
|
validateObject,
|
|
validateOneOf,
|
|
validateString,
|
|
validateUint32,
|
|
} = require('internal/validators');
|
|
const {
|
|
emitExperimentalWarning,
|
|
kEmptyObject,
|
|
kVmBreakFirstLineSymbol,
|
|
} = require('internal/util');
|
|
const {
|
|
internalCompileFunction,
|
|
isContext,
|
|
} = require('internal/vm');
|
|
const kParsingContext = Symbol('script parsing context');
|
|
|
|
class Script extends ContextifyScript {
|
|
constructor(code, options = kEmptyObject) {
|
|
code = `${code}`;
|
|
if (typeof options === 'string') {
|
|
options = { filename: options };
|
|
} else {
|
|
validateObject(options, 'options');
|
|
}
|
|
|
|
const {
|
|
filename = 'evalmachine.<anonymous>',
|
|
lineOffset = 0,
|
|
columnOffset = 0,
|
|
cachedData,
|
|
produceCachedData = false,
|
|
importModuleDynamically,
|
|
[kParsingContext]: parsingContext,
|
|
} = options;
|
|
|
|
validateString(filename, 'options.filename');
|
|
validateInt32(lineOffset, 'options.lineOffset');
|
|
validateInt32(columnOffset, 'options.columnOffset');
|
|
if (cachedData !== undefined) {
|
|
validateBuffer(cachedData, 'options.cachedData');
|
|
}
|
|
validateBoolean(produceCachedData, 'options.produceCachedData');
|
|
|
|
// Calling `ReThrow()` on a native TryCatch does not generate a new
|
|
// abort-on-uncaught-exception check. A dummy try/catch in JS land
|
|
// protects against that.
|
|
try { // eslint-disable-line no-useless-catch
|
|
super(code,
|
|
filename,
|
|
lineOffset,
|
|
columnOffset,
|
|
cachedData,
|
|
produceCachedData,
|
|
parsingContext);
|
|
} catch (e) {
|
|
throw e; /* node-do-not-add-exception-line */
|
|
}
|
|
|
|
if (importModuleDynamically !== undefined) {
|
|
validateFunction(importModuleDynamically,
|
|
'options.importModuleDynamically');
|
|
const { importModuleDynamicallyWrap } = require('internal/vm/module');
|
|
const { registerModule } = require('internal/modules/esm/utils');
|
|
registerModule(this, {
|
|
__proto__: null,
|
|
importModuleDynamically:
|
|
importModuleDynamicallyWrap(importModuleDynamically),
|
|
});
|
|
}
|
|
}
|
|
|
|
runInThisContext(options) {
|
|
const { breakOnSigint, args } = getRunInContextArgs(null, options);
|
|
if (breakOnSigint && process.listenerCount('SIGINT') > 0) {
|
|
return sigintHandlersWrap(super.runInContext, this, args);
|
|
}
|
|
return ReflectApply(super.runInContext, this, args);
|
|
}
|
|
|
|
runInContext(contextifiedObject, options) {
|
|
validateContext(contextifiedObject);
|
|
const { breakOnSigint, args } = getRunInContextArgs(
|
|
contextifiedObject,
|
|
options,
|
|
);
|
|
if (breakOnSigint && process.listenerCount('SIGINT') > 0) {
|
|
return sigintHandlersWrap(super.runInContext, this, args);
|
|
}
|
|
return ReflectApply(super.runInContext, this, args);
|
|
}
|
|
|
|
runInNewContext(contextObject, options) {
|
|
const context = createContext(contextObject, getContextOptions(options));
|
|
return this.runInContext(context, options);
|
|
}
|
|
}
|
|
|
|
function validateContext(contextifiedObject) {
|
|
if (!isContext(contextifiedObject)) {
|
|
throw new ERR_INVALID_ARG_TYPE('contextifiedObject', 'vm.Context',
|
|
contextifiedObject);
|
|
}
|
|
}
|
|
|
|
function getRunInContextArgs(contextifiedObject, options = kEmptyObject) {
|
|
validateObject(options, 'options');
|
|
|
|
let timeout = options.timeout;
|
|
if (timeout === undefined) {
|
|
timeout = -1;
|
|
} else {
|
|
validateUint32(timeout, 'options.timeout', true);
|
|
}
|
|
|
|
const {
|
|
displayErrors = true,
|
|
breakOnSigint = false,
|
|
[kVmBreakFirstLineSymbol]: breakFirstLine = false,
|
|
} = options;
|
|
|
|
validateBoolean(displayErrors, 'options.displayErrors');
|
|
validateBoolean(breakOnSigint, 'options.breakOnSigint');
|
|
|
|
return {
|
|
breakOnSigint,
|
|
args: [
|
|
contextifiedObject,
|
|
timeout,
|
|
displayErrors,
|
|
breakOnSigint,
|
|
breakFirstLine,
|
|
],
|
|
};
|
|
}
|
|
|
|
function getContextOptions(options) {
|
|
if (!options)
|
|
return {};
|
|
const contextOptions = {
|
|
name: options.contextName,
|
|
origin: options.contextOrigin,
|
|
codeGeneration: undefined,
|
|
microtaskMode: options.microtaskMode,
|
|
};
|
|
if (contextOptions.name !== undefined)
|
|
validateString(contextOptions.name, 'options.contextName');
|
|
if (contextOptions.origin !== undefined)
|
|
validateString(contextOptions.origin, 'options.contextOrigin');
|
|
if (options.contextCodeGeneration !== undefined) {
|
|
validateObject(options.contextCodeGeneration,
|
|
'options.contextCodeGeneration');
|
|
const { strings, wasm } = options.contextCodeGeneration;
|
|
if (strings !== undefined)
|
|
validateBoolean(strings, 'options.contextCodeGeneration.strings');
|
|
if (wasm !== undefined)
|
|
validateBoolean(wasm, 'options.contextCodeGeneration.wasm');
|
|
contextOptions.codeGeneration = { strings, wasm };
|
|
}
|
|
if (options.microtaskMode !== undefined)
|
|
validateString(options.microtaskMode, 'options.microtaskMode');
|
|
return contextOptions;
|
|
}
|
|
|
|
let defaultContextNameIndex = 1;
|
|
function createContext(contextObject = {}, options = kEmptyObject) {
|
|
if (isContext(contextObject)) {
|
|
return contextObject;
|
|
}
|
|
|
|
validateObject(options, 'options');
|
|
|
|
const {
|
|
name = `VM Context ${defaultContextNameIndex++}`,
|
|
origin,
|
|
codeGeneration,
|
|
microtaskMode,
|
|
} = options;
|
|
|
|
validateString(name, 'options.name');
|
|
if (origin !== undefined)
|
|
validateString(origin, 'options.origin');
|
|
if (codeGeneration !== undefined)
|
|
validateObject(codeGeneration, 'options.codeGeneration');
|
|
|
|
let strings = true;
|
|
let wasm = true;
|
|
if (codeGeneration !== undefined) {
|
|
({ strings = true, wasm = true } = codeGeneration);
|
|
validateBoolean(strings, 'options.codeGeneration.strings');
|
|
validateBoolean(wasm, 'options.codeGeneration.wasm');
|
|
}
|
|
|
|
validateOneOf(microtaskMode,
|
|
'options.microtaskMode',
|
|
['afterEvaluate', undefined]);
|
|
const microtaskQueue = (microtaskMode === 'afterEvaluate');
|
|
|
|
makeContext(contextObject, name, origin, strings, wasm, microtaskQueue);
|
|
return contextObject;
|
|
}
|
|
|
|
function createScript(code, options) {
|
|
return new Script(code, options);
|
|
}
|
|
|
|
// Remove all SIGINT listeners and re-attach them after the wrapped function
|
|
// has executed, so that caught SIGINT are handled by the listeners again.
|
|
function sigintHandlersWrap(fn, thisArg, argsArray) {
|
|
const sigintListeners = process.rawListeners('SIGINT');
|
|
|
|
process.removeAllListeners('SIGINT');
|
|
|
|
try {
|
|
return ReflectApply(fn, thisArg, argsArray);
|
|
} finally {
|
|
// Add using the public methods so that the `newListener` handler of
|
|
// process can re-attach the listeners.
|
|
ArrayPrototypeForEach(sigintListeners, (listener) => {
|
|
process.addListener('SIGINT', listener);
|
|
});
|
|
}
|
|
}
|
|
|
|
function runInContext(code, contextifiedObject, options) {
|
|
validateContext(contextifiedObject);
|
|
if (typeof options === 'string') {
|
|
options = {
|
|
filename: options,
|
|
[kParsingContext]: contextifiedObject,
|
|
};
|
|
} else {
|
|
options = { ...options, [kParsingContext]: contextifiedObject };
|
|
}
|
|
return createScript(code, options)
|
|
.runInContext(contextifiedObject, options);
|
|
}
|
|
|
|
function runInNewContext(code, contextObject, options) {
|
|
if (typeof options === 'string') {
|
|
options = { filename: options };
|
|
}
|
|
contextObject = createContext(contextObject, getContextOptions(options));
|
|
options = { ...options, [kParsingContext]: contextObject };
|
|
return createScript(code, options).runInNewContext(contextObject, options);
|
|
}
|
|
|
|
function runInThisContext(code, options) {
|
|
if (typeof options === 'string') {
|
|
options = { filename: options };
|
|
}
|
|
return createScript(code, options).runInThisContext(options);
|
|
}
|
|
|
|
function compileFunction(code, params, options = kEmptyObject) {
|
|
return internalCompileFunction(code, params, options).function;
|
|
}
|
|
|
|
const measureMemoryModes = {
|
|
summary: constants.measureMemory.mode.SUMMARY,
|
|
detailed: constants.measureMemory.mode.DETAILED,
|
|
};
|
|
|
|
const measureMemoryExecutions = {
|
|
default: constants.measureMemory.execution.DEFAULT,
|
|
eager: constants.measureMemory.execution.EAGER,
|
|
};
|
|
|
|
function measureMemory(options = kEmptyObject) {
|
|
emitExperimentalWarning('vm.measureMemory');
|
|
validateObject(options, 'options');
|
|
const { mode = 'summary', execution = 'default' } = options;
|
|
validateOneOf(mode, 'options.mode', ['summary', 'detailed']);
|
|
validateOneOf(execution, 'options.execution', ['default', 'eager']);
|
|
const result = _measureMemory(measureMemoryModes[mode],
|
|
measureMemoryExecutions[execution]);
|
|
if (result === undefined) {
|
|
return PromiseReject(new ERR_CONTEXT_NOT_INITIALIZED());
|
|
}
|
|
return result;
|
|
}
|
|
|
|
module.exports = {
|
|
Script,
|
|
createContext,
|
|
createScript,
|
|
runInContext,
|
|
runInNewContext,
|
|
runInThisContext,
|
|
isContext,
|
|
compileFunction,
|
|
measureMemory,
|
|
};
|
|
|
|
// The vm module is patched to include vm.Module, vm.SourceTextModule
|
|
// and vm.SyntheticModule in the pre-execution phase when
|
|
// --experimental-vm-modules is on.
|