From b12211eeca292a01bf27a2c88632f77b78555489 Mon Sep 17 00:00:00 2001 From: Anna Henningsen Date: Wed, 22 Jul 2020 00:59:41 +0200 Subject: [PATCH] src: prefer internal fields in ModuleWrap MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Use internal fields instead of `v8::Global`s where possible, since they generally come with lower overhead and it’s much harder to introduce memory leaks with them. PR-URL: https://github.com/nodejs/node/pull/34470 Reviewed-By: James M Snell Reviewed-By: Guy Bedford Reviewed-By: Joyee Cheung Reviewed-By: Gus Caplan Reviewed-By: David Carlier Reviewed-By: Franziska Hinkelmann --- src/module_wrap.cc | 33 ++++++++++++++++++++++++--------- src/module_wrap.h | 13 +++++++++---- 2 files changed, 33 insertions(+), 13 deletions(-) diff --git a/src/module_wrap.cc b/src/module_wrap.cc index 825afea7b12c8b..f778b089dc4009 100644 --- a/src/module_wrap.cc +++ b/src/module_wrap.cc @@ -55,9 +55,13 @@ ModuleWrap::ModuleWrap(Environment* env, Local url) : BaseObject(env, object), module_(env->isolate(), module), - url_(env->isolate(), url), id_(env->get_next_module_id()) { env->id_to_module_map.emplace(id_, this); + + Local undefined = Undefined(env->isolate()); + object->SetInternalField(kURLSlot, url); + object->SetInternalField(kSyntheticEvaluationStepsSlot, undefined); + object->SetInternalField(kContextObjectSlot, undefined); } ModuleWrap::~ModuleWrap() { @@ -73,6 +77,12 @@ ModuleWrap::~ModuleWrap() { } } +Local ModuleWrap::context() const { + Local obj = object()->GetInternalField(kContextObjectSlot); + if (obj.IsEmpty()) return {}; + return obj.As()->CreationContext(); +} + ModuleWrap* ModuleWrap::GetFromModule(Environment* env, Local module) { auto range = env->hash_to_module_map.equal_range(module->GetIdentityHash()); @@ -220,11 +230,14 @@ void ModuleWrap::New(const FunctionCallbackInfo& args) { if (synthetic) { obj->synthetic_ = true; - obj->synthetic_evaluation_steps_.Reset( - env->isolate(), args[3].As()); + obj->object()->SetInternalField(kSyntheticEvaluationStepsSlot, args[3]); } - obj->context_.Reset(isolate, context); + // Use the extras object as an object whose CreationContext() will be the + // original `context`, since the `Context` itself strictly speaking cannot + // be stored in an internal field. + obj->object()->SetInternalField(kContextObjectSlot, + context->GetExtrasBindingObject()); obj->contextify_context_ = contextify_context; env->hash_to_module_map.emplace(module->GetIdentityHash(), obj); @@ -254,7 +267,7 @@ void ModuleWrap::Link(const FunctionCallbackInfo& args) { Local resolver_arg = args[0].As(); - Local mod_context = obj->context_.Get(isolate); + Local mod_context = obj->context(); Local module = obj->module_.Get(isolate); const int module_requests_length = module->GetModuleRequestsLength(); @@ -295,7 +308,7 @@ void ModuleWrap::Instantiate(const FunctionCallbackInfo& args) { Isolate* isolate = args.GetIsolate(); ModuleWrap* obj; ASSIGN_OR_RETURN_UNWRAP(&obj, args.This()); - Local context = obj->context_.Get(isolate); + Local context = obj->context(); Local module = obj->module_.Get(isolate); TryCatchScope try_catch(env); USE(module->InstantiateModule(context, ResolveCallback)); @@ -318,7 +331,7 @@ void ModuleWrap::Evaluate(const FunctionCallbackInfo& args) { Isolate* isolate = env->isolate(); ModuleWrap* obj; ASSIGN_OR_RETURN_UNWRAP(&obj, args.This()); - Local context = obj->context_.Get(isolate); + Local context = obj->context(); Local module = obj->module_.Get(isolate); ContextifyContext* contextify_context = obj->contextify_context_; @@ -636,8 +649,10 @@ MaybeLocal ModuleWrap::SyntheticModuleEvaluationStepsCallback( TryCatchScope try_catch(env); Local synthetic_evaluation_steps = - obj->synthetic_evaluation_steps_.Get(isolate); - obj->synthetic_evaluation_steps_.Reset(); + obj->object()->GetInternalField(kSyntheticEvaluationStepsSlot) + .As(); + obj->object()->SetInternalField( + kSyntheticEvaluationStepsSlot, Undefined(isolate)); MaybeLocal ret = synthetic_evaluation_steps->Call(context, obj->object(), 0, nullptr); if (ret.IsEmpty()) { diff --git a/src/module_wrap.h b/src/module_wrap.h index 2b490fc41374b6..3d8bb57750397f 100644 --- a/src/module_wrap.h +++ b/src/module_wrap.h @@ -32,6 +32,14 @@ enum HostDefinedOptions : int { class ModuleWrap : public BaseObject { public: + enum InternalFields { + kModuleWrapBaseField = BaseObject::kInternalFieldCount, + kURLSlot, + kSyntheticEvaluationStepsSlot, + kContextObjectSlot, // Object whose creation context is the target Context + kInternalFieldCount + }; + static void Initialize(v8::Local target, v8::Local unused, v8::Local context, @@ -42,11 +50,11 @@ class ModuleWrap : public BaseObject { v8::Local meta); void MemoryInfo(MemoryTracker* tracker) const override { - tracker->TrackField("url", url_); tracker->TrackField("resolve_cache", resolve_cache_); } inline uint32_t id() { return id_; } + v8::Local context() const; static ModuleWrap* GetFromID(node::Environment*, uint32_t id); SET_MEMORY_INFO_NAME(ModuleWrap) @@ -85,11 +93,8 @@ class ModuleWrap : public BaseObject { v8::Local referrer); static ModuleWrap* GetFromModule(node::Environment*, v8::Local); - v8::Global synthetic_evaluation_steps_; v8::Global module_; - v8::Global url_; std::unordered_map> resolve_cache_; - v8::Global context_; contextify::ContextifyContext* contextify_context_ = nullptr; bool synthetic_ = false; bool linked_ = false;