From afb9113d4565179f4d64001ce1806648e8693ee4 Mon Sep 17 00:00:00 2001 From: legendecas Date: Sat, 30 Jul 2022 22:15:24 +0800 Subject: [PATCH] src: introduce node::Realm To distinguish per-context values from the node::Environment, split those values to a new node::Realm structure and consolidate bootstrapping methods with it. --- node.gyp | 4 + src/api/environment.cc | 11 +- src/env-inl.h | 32 +-- src/env.cc | 201 +++------------- src/env.h | 471 ++----------------------------------- src/env_properties.h | 432 ++++++++++++++++++++++++++++++++++ src/node.cc | 153 +----------- src/node_context_data.h | 43 +++- src/node_internals.h | 4 - src/node_main_instance.cc | 3 +- src/node_process.h | 3 +- src/node_process_object.cc | 22 +- src/node_realm-inl.h | 66 ++++++ src/node_realm.cc | 301 ++++++++++++++++++++++++ src/node_realm.h | 79 +++++++ src/node_snapshotable.cc | 95 +++++++- src/node_snapshotable.h | 15 ++ 17 files changed, 1124 insertions(+), 811 deletions(-) create mode 100644 src/env_properties.h create mode 100644 src/node_realm-inl.h create mode 100644 src/node_realm.cc create mode 100644 src/node_realm.h diff --git a/node.gyp b/node.gyp index 9563073fd6c8985..5479f204ccf61df 100644 --- a/node.gyp +++ b/node.gyp @@ -512,6 +512,7 @@ 'src/node_process_events.cc', 'src/node_process_methods.cc', 'src/node_process_object.cc', + 'src/node_realm.cc', 'src/node_report.cc', 'src/node_report_module.cc', 'src/node_report_utils.cc', @@ -570,6 +571,7 @@ 'src/connection_wrap.h', 'src/debug_utils.h', 'src/debug_utils-inl.h', + 'src/env_properties.h', 'src/env.h', 'src/env-inl.h', 'src/handle_wrap.h', @@ -617,6 +619,8 @@ 'src/node_platform.h', 'src/node_process.h', 'src/node_process-inl.h', + 'src/node_realm.h', + 'src/node_realm-inl.h', 'src/node_report.h', 'src/node_revert.h', 'src/node_root_certs.h', diff --git a/src/api/environment.cc b/src/api/environment.cc index bdf67521cbf80c7..4f2dacf329a116c 100644 --- a/src/api/environment.cc +++ b/src/api/environment.cc @@ -5,6 +5,7 @@ #include "node_internals.h" #include "node_options-inl.h" #include "node_platform.h" +#include "node_realm-inl.h" #include "node_shadow_realm.h" #include "node_v8_platform-inl.h" #include "node_wasm_web_api.h" @@ -377,7 +378,7 @@ Environment* CreateEnvironment( } #endif - if (env->RunBootstrapping().IsEmpty()) { + if (env->principal_realm()->RunBootstrapping().IsEmpty()) { FreeEnvironment(env); return nullptr; } @@ -452,11 +453,13 @@ MaybeLocal LoadEnvironment( builtins::BuiltinLoader::Add( name.c_str(), UnionBytes(**main_utf16, main_utf16->length())); env->set_main_utf16(std::move(main_utf16)); + Realm* realm = env->principal_realm(); + // Arguments must match the parameters specified in // BuiltinLoader::LookupAndCompile(). - std::vector> args = {env->process_object(), - env->builtin_module_require()}; - return ExecuteBootstrapper(env, name.c_str(), &args); + std::vector> args = {realm->process_object(), + realm->builtin_module_require()}; + return realm->ExecuteBootstrapper(name.c_str(), &args); }); } diff --git a/src/env-inl.h b/src/env-inl.h index f6b340d8149b88a..7341b19c78875d4 100644 --- a/src/env-inl.h +++ b/src/env-inl.h @@ -31,6 +31,7 @@ #include "node_context_data.h" #include "node_internals.h" #include "node_perf_common.h" +#include "node_realm-inl.h" #include "util-inl.h" #include "uv.h" #include "v8.h" @@ -177,16 +178,7 @@ inline Environment* Environment::GetCurrent(v8::Isolate* isolate) { } inline Environment* Environment::GetCurrent(v8::Local context) { - if (UNLIKELY(context.IsEmpty())) { - return nullptr; - } - if (UNLIKELY(context->GetNumberOfEmbedderDataFields() <= - ContextEmbedderIndex::kContextTag)) { - return nullptr; - } - if (UNLIKELY(context->GetAlignedPointerFromEmbedderData( - ContextEmbedderIndex::kContextTag) != - Environment::kNodeContextTagPtr)) { + if (UNLIKELY(!ContextEmbedderTag::IsNodeContext(context))) { return nullptr; } return static_cast( @@ -623,11 +615,13 @@ inline void Environment::set_can_call_into_js(bool can_call_into_js) { } inline bool Environment::has_run_bootstrapping_code() const { - return has_run_bootstrapping_code_; + return principal_realm_->has_run_bootstrapping_code(); } inline void Environment::DoneBootstrapping() { - has_run_bootstrapping_code_ = true; + CHECK(has_run_bootstrapping_code()); + // TODO(legendecas): distinguish base objects with realms. + // This adjusts the return value of base_object_created_after_bootstrap() so // that tests that check the count do not have to account for internally // created BaseObjects. @@ -922,16 +916,22 @@ void Environment::set_process_exit_handler( #define V(PropertyName, TypeName) \ inline v8::Local Environment::PropertyName() const { \ - return PersistentToLocal::Strong(PropertyName##_); \ + DCHECK_NOT_NULL(principal_realm_); \ + return principal_realm_->PropertyName(); \ } \ inline void Environment::set_##PropertyName(v8::Local value) { \ - PropertyName##_.Reset(isolate(), value); \ + DCHECK_NOT_NULL(principal_realm_); \ + principal_realm_->set_##PropertyName(value); \ } - ENVIRONMENT_STRONG_PERSISTENT_VALUES(V) + PER_REALM_STRONG_PERSISTENT_VALUES(V) #undef V v8::Local Environment::context() const { - return PersistentToLocal::Strong(context_); + return principal_realm()->context(); +} + +Realm* Environment::principal_realm() const { + return principal_realm_.get(); } } // namespace node diff --git a/src/env.cc b/src/env.cc index 83f65f2ad20dba5..0ee4d8df85e9a67 100644 --- a/src/env.cc +++ b/src/env.cc @@ -61,9 +61,9 @@ using v8::WeakCallbackInfo; using v8::WeakCallbackType; using worker::Worker; -int const Environment::kNodeContextTag = 0x6e6f64; -void* const Environment::kNodeContextTagPtr = const_cast( - static_cast(&Environment::kNodeContextTag)); +int const ContextEmbedderTag::kNodeContextTag = 0x6e6f64; +void* const ContextEmbedderTag::kNodeContextTagPtr = const_cast( + static_cast(&ContextEmbedderTag::kNodeContextTag)); void AsyncHooks::SetJSPromiseHooks(Local init, Local before, @@ -481,6 +481,10 @@ void TrackingTraceStateObserver::UpdateTraceCategoryState() { return; } + if (env_->principal_realm() == nullptr) { + return; + } + bool async_hooks_enabled = (*(TRACE_EVENT_API_GET_CATEGORY_GROUP_ENABLED( TRACING_CATEGORY_NODE1(async_hooks)))) != 0; @@ -497,11 +501,16 @@ void TrackingTraceStateObserver::UpdateTraceCategoryState() { void Environment::AssignToContext(Local context, const ContextInfo& info) { + AssignToContext(context, nullptr, info); +} + +void Environment::AssignToContext(Local context, + Realm* realm, + const ContextInfo& info) { + ContextEmbedderTag::TagNodeContext(context); context->SetAlignedPointerInEmbedderData(ContextEmbedderIndex::kEnvironment, this); - // Used by Environment::GetCurrent to know that we are on a node context. - context->SetAlignedPointerInEmbedderData(ContextEmbedderIndex::kContextTag, - Environment::kNodeContextTagPtr); + context->SetAlignedPointerInEmbedderData(ContextEmbedderIndex::kRealm, realm); // Used to retrieve bindings context->SetAlignedPointerInEmbedderData( ContextEmbedderIndex::kBindingListIndex, &(this->bindings_)); @@ -571,52 +580,13 @@ std::unique_ptr Environment::release_managed_buffer( void Environment::CreateProperties() { HandleScope handle_scope(isolate_); - Local ctx = context(); - { - Context::Scope context_scope(ctx); - Local templ = FunctionTemplate::New(isolate()); - templ->InstanceTemplate()->SetInternalFieldCount( - BaseObject::kInternalFieldCount); - templ->Inherit(BaseObject::GetConstructorTemplate(this)); - - set_binding_data_ctor_template(templ); - } - - // Store primordials setup by the per-context script in the environment. - Local per_context_bindings = - GetPerContextExports(ctx).ToLocalChecked(); - Local primordials = - per_context_bindings->Get(ctx, primordials_string()).ToLocalChecked(); - CHECK(primordials->IsObject()); - set_primordials(primordials.As()); - - Local prototype_string = - FIXED_ONE_BYTE_STRING(isolate(), "prototype"); - -#define V(EnvPropertyName, PrimordialsPropertyName) \ - { \ - Local ctor = \ - primordials.As() \ - ->Get(ctx, \ - FIXED_ONE_BYTE_STRING(isolate(), PrimordialsPropertyName)) \ - .ToLocalChecked(); \ - CHECK(ctor->IsObject()); \ - Local prototype = \ - ctor.As()->Get(ctx, prototype_string).ToLocalChecked(); \ - CHECK(prototype->IsObject()); \ - set_##EnvPropertyName(prototype.As()); \ - } - - V(primordials_safe_map_prototype_object, "SafeMap"); - V(primordials_safe_set_prototype_object, "SafeSet"); - V(primordials_safe_weak_map_prototype_object, "SafeWeakMap"); - V(primordials_safe_weak_set_prototype_object, "SafeWeakSet"); -#undef V + Local templ = FunctionTemplate::New(isolate()); + templ->InstanceTemplate()->SetInternalFieldCount( + BaseObject::kInternalFieldCount); + templ->Inherit(BaseObject::GetConstructorTemplate(this)); - Local process_object = - node::CreateProcessObject(this).FromMaybe(Local()); - set_process_object(process_object); + set_binding_data_ctor_template(templ); } std::string GetExecPath(const std::vector& argv) { @@ -754,7 +724,11 @@ Environment::Environment(IsolateData* isolate_data, void Environment::InitializeMainContext(Local context, const EnvSerializeInfo* env_info) { - context_.Reset(context->GetIsolate(), context); + principal_realm_ = std::make_unique( + isolate_data_, + this, + context, + env_info == nullptr ? nullptr : &env_info->principal_realm); AssignToContext(context, ContextInfo("")); if (env_info != nullptr) { DeserializeProperties(env_info); @@ -784,6 +758,9 @@ void Environment::InitializeMainContext(Local context, } Environment::~Environment() { + HandleScope handle_scope(isolate()); + Local ctx = context(); + if (Environment** interrupt_data = interrupt_data_.load()) { // There are pending RequestInterrupt() callbacks. Tell them not to run, // then force V8 to run interrupts by compiling and running an empty script @@ -791,9 +768,8 @@ Environment::~Environment() { *interrupt_data = nullptr; Isolate::AllowJavascriptExecutionScope allow_js_here(isolate()); - HandleScope handle_scope(isolate()); TryCatch try_catch(isolate()); - Context::Scope context_scope(context()); + Context::Scope context_scope(ctx); #ifdef DEBUG bool consistency_check = false; @@ -803,8 +779,8 @@ Environment::~Environment() { #endif Local