diff --git a/src/env_properties.h b/src/env_properties.h index 52cd06ed92699e9..9abcb8d9d72d0e3 100644 --- a/src/env_properties.h +++ b/src/env_properties.h @@ -20,7 +20,6 @@ #define PER_ISOLATE_PRIVATE_SYMBOL_PROPERTIES(V) \ V(arrow_message_private_symbol, "node:arrowMessage") \ V(contextify_context_private_symbol, "node:contextify:context") \ - V(compiled_function_entry, "node:compiled_function_entry") \ V(decorated_private_symbol, "node:decorated") \ V(napi_type_tag, "node:napi:type_tag") \ V(napi_wrapper, "node:napi:wrapper") \ diff --git a/src/node_contextify.cc b/src/node_contextify.cc index bd46a4a1e50e864..5aadefd84d6af7d 100644 --- a/src/node_contextify.cc +++ b/src/node_contextify.cc @@ -77,6 +77,7 @@ using v8::Uint32; using v8::UnboundScript; using v8::Value; using v8::WeakCallbackInfo; +using v8::WeakCallbackType; // The vm module executes code in a sandboxed environment with a different // global object than the rest of the code. This is achieved by applying @@ -1262,7 +1263,8 @@ void ContextifyContext::CompileFunction( context).ToLocal(&cache_key)) { return; } - new CompiledFnEntry(env, cache_key, id, fn); + CompiledFnEntry* entry = new CompiledFnEntry(env, cache_key, id, fn); + env->id_to_function_map.emplace(id, entry); Local result = Object::New(isolate); if (result->Set(parsing_context, env->function_string(), fn).IsNothing()) @@ -1294,18 +1296,23 @@ void ContextifyContext::CompileFunction( args.GetReturnValue().Set(result); } +void CompiledFnEntry::WeakCallback( + const WeakCallbackInfo& data) { + CompiledFnEntry* entry = data.GetParameter(); + delete entry; +} + CompiledFnEntry::CompiledFnEntry(Environment* env, Local object, uint32_t id, Local fn) - : BaseObject(env, object), id_(id) { - MakeWeak(); - fn->SetPrivate(env->context(), env->compiled_function_entry(), object); - env->id_to_function_map.emplace(id, this); + : BaseObject(env, object), id_(id), fn_(env->isolate(), fn) { + fn_.SetWeak(this, WeakCallback, v8::WeakCallbackType::kParameter); } CompiledFnEntry::~CompiledFnEntry() { env()->id_to_function_map.erase(id_); + fn_.ClearWeak(); } static void StartSigintWatchdog(const FunctionCallbackInfo& args) { diff --git a/src/node_contextify.h b/src/node_contextify.h index a9b7741e61b2c25..76c89318bb6cbf6 100644 --- a/src/node_contextify.h +++ b/src/node_contextify.h @@ -194,6 +194,9 @@ class CompiledFnEntry final : public BaseObject { private: uint32_t id_; + v8::Global fn_; + + static void WeakCallback(const v8::WeakCallbackInfo& data); }; v8::Maybe StoreCodeCacheResult( diff --git a/test/pummel/test-vm-compile-function-leak.js b/test/pummel/test-vm-compile-function-leak.js deleted file mode 100644 index 465f300d4310d1a..000000000000000 --- a/test/pummel/test-vm-compile-function-leak.js +++ /dev/null @@ -1,14 +0,0 @@ -'use strict'; - -// Flags: --max-old-space-size=10 - -require('../common'); -const vm = require('vm'); - -const code = `console.log("${'hello world '.repeat(1e5)}");`; - -for (let i = 0; i < 10000; i++) { - vm.compileFunction(code, [], { - importModuleDynamically: () => {}, - }); -}