Skip to content

Commit eadbf52

Browse files
committed
src: make realm binding data store weak
The binding data must be weak so that it won't keep the realm reachable from strong GC roots indefinitely. The wrapper object of binding data should be referenced from JavaScript, thus the binding data should be reachable throughout the lifetime of the realm.
1 parent cd7a958 commit eadbf52

15 files changed

+116
-49
lines changed

src/base_object-inl.h

+6
Original file line numberDiff line numberDiff line change
@@ -289,6 +289,12 @@ template <typename T, typename... Args>
289289
BaseObjectPtr<T> MakeBaseObject(Args&&... args) {
290290
return BaseObjectPtr<T>(new T(std::forward<Args>(args)...));
291291
}
292+
template <typename T, typename... Args>
293+
BaseObjectWeakPtr<T> MakeWeakBaseObject(Args&&... args) {
294+
T* target = new T(std::forward<Args>(args)...);
295+
target->MakeWeak();
296+
return BaseObjectWeakPtr<T>(target);
297+
}
292298

293299
template <typename T, typename... Args>
294300
BaseObjectPtr<T> MakeDetachedBaseObject(Args&&... args) {

src/base_object.h

+4
Original file line numberDiff line numberDiff line change
@@ -302,6 +302,10 @@ using BaseObjectWeakPtr = BaseObjectPtrImpl<T, true>;
302302
template <typename T, typename... Args>
303303
inline BaseObjectPtr<T> MakeBaseObject(Args&&... args);
304304
// Create a BaseObject instance and return a pointer to it.
305+
// This variant makes the object a weak GC root by default.
306+
template <typename T, typename... Args>
307+
inline BaseObjectWeakPtr<T> MakeWeakBaseObject(Args&&... args);
308+
// Create a BaseObject instance and return a pointer to it.
305309
// This variant detaches the object by default, meaning that the caller fully
306310
// owns it, and once the last BaseObjectPtr to it is destroyed, the object
307311
// itself is also destroyed.

src/env.cc

+20-10
Original file line numberDiff line numberDiff line change
@@ -595,6 +595,20 @@ void Environment::AssignToContext(Local<v8::Context> context,
595595
TrackContext(context);
596596
}
597597

598+
void Environment::UnassignToContext(Local<v8::Context> context) {
599+
if (!context.IsEmpty()) {
600+
context->SetAlignedPointerInEmbedderData(ContextEmbedderIndex::kEnvironment,
601+
nullptr);
602+
context->SetAlignedPointerInEmbedderData(ContextEmbedderIndex::kRealm,
603+
nullptr);
604+
context->SetAlignedPointerInEmbedderData(
605+
ContextEmbedderIndex::kBindingDataStoreIndex, nullptr);
606+
context->SetAlignedPointerInEmbedderData(
607+
ContextEmbedderIndex::kContextifyContext, nullptr);
608+
}
609+
UntrackContext(context);
610+
}
611+
598612
void Environment::TryLoadAddon(
599613
const char* filename,
600614
int flags,
@@ -820,7 +834,6 @@ void Environment::InitializeMainContext(Local<Context> context,
820834
const EnvSerializeInfo* env_info) {
821835
principal_realm_ = std::make_unique<PrincipalRealm>(
822836
this, context, MAYBE_FIELD_PTR(env_info, principal_realm));
823-
AssignToContext(context, principal_realm_.get(), ContextInfo(""));
824837
if (env_info != nullptr) {
825838
DeserializeProperties(env_info);
826839
}
@@ -890,9 +903,9 @@ Environment::~Environment() {
890903
inspector_agent_.reset();
891904
#endif
892905

893-
ctx->SetAlignedPointerInEmbedderData(ContextEmbedderIndex::kEnvironment,
894-
nullptr);
895-
ctx->SetAlignedPointerInEmbedderData(ContextEmbedderIndex::kRealm, nullptr);
906+
// Sub-realms should have been cleared with Environment's cleanup.
907+
DCHECK_EQ(shadow_realms_.size(), 0);
908+
principal_realm_.reset();
896909

897910
if (trace_state_observer_) {
898911
tracing::AgentWriterHandle* writer = GetTracingAgentWriter();
@@ -915,10 +928,6 @@ Environment::~Environment() {
915928
addon.Close();
916929
}
917930
}
918-
919-
for (auto realm : shadow_realms_) {
920-
realm->OnEnvironmentDestruct();
921-
}
922931
}
923932

924933
void Environment::InitializeLibuv() {
@@ -1714,6 +1723,9 @@ void Environment::DeserializeProperties(const EnvSerializeInfo* info) {
17141723
std::cerr << *info << "\n";
17151724
}
17161725

1726+
// Deserialize the realm's properties before running the deserialize
1727+
// requests as the requests may need to access the realm's properties.
1728+
principal_realm_->DeserializeProperties(&info->principal_realm);
17171729
RunDeserializeRequests();
17181730

17191731
async_hooks_.Deserialize(ctx);
@@ -1724,8 +1736,6 @@ void Environment::DeserializeProperties(const EnvSerializeInfo* info) {
17241736
exit_info_.Deserialize(ctx);
17251737
stream_base_state_.Deserialize(ctx);
17261738
should_abort_on_uncaught_toggle_.Deserialize(ctx);
1727-
1728-
principal_realm_->DeserializeProperties(&info->principal_realm);
17291739
}
17301740

17311741
uint64_t GuessMemoryAvailableToTheProcess() {

src/env.h

+3-2
Original file line numberDiff line numberDiff line change
@@ -645,8 +645,7 @@ class Environment : public MemoryRetainer {
645645
void AssignToContext(v8::Local<v8::Context> context,
646646
Realm* realm,
647647
const ContextInfo& info);
648-
void TrackContext(v8::Local<v8::Context> context);
649-
void UntrackContext(v8::Local<v8::Context> context);
648+
void UnassignToContext(v8::Local<v8::Context> context);
650649
void TrackShadowRealm(shadow_realm::ShadowRealm* realm);
651650
void UntrackShadowRealm(shadow_realm::ShadowRealm* realm);
652651

@@ -998,6 +997,8 @@ class Environment : public MemoryRetainer {
998997
private:
999998
inline void ThrowError(v8::Local<v8::Value> (*fun)(v8::Local<v8::String>),
1000999
const char* errmsg);
1000+
void TrackContext(v8::Local<v8::Context> context);
1001+
void UntrackContext(v8::Local<v8::Context> context);
10011002

10021003
std::list<binding::DLib> loaded_addons_;
10031004
v8::Isolate* const isolate_;

src/inspector_js_api.cc

+2-2
Original file line numberDiff line numberDiff line change
@@ -172,12 +172,12 @@ static bool InspectorEnabled(Environment* env) {
172172
}
173173

174174
void SetConsoleExtensionInstaller(const FunctionCallbackInfo<Value>& info) {
175-
auto env = Environment::GetCurrent(info);
175+
Realm* realm = Realm::GetCurrent(info);
176176

177177
CHECK_EQ(info.Length(), 1);
178178
CHECK(info[0]->IsFunction());
179179

180-
env->set_inspector_console_extension_installer(info[0].As<Function>());
180+
realm->set_inspector_console_extension_installer(info[0].As<Function>());
181181
}
182182

183183
void CallAndPauseOnStart(const FunctionCallbackInfo<v8::Value>& args) {

src/node_contextify.cc

+1-1
Original file line numberDiff line numberDiff line change
@@ -163,7 +163,7 @@ ContextifyContext::~ContextifyContext() {
163163
Isolate* isolate = env()->isolate();
164164
HandleScope scope(isolate);
165165

166-
env()->UntrackContext(PersistentToLocal::Weak(isolate, context_));
166+
env()->UnassignToContext(PersistentToLocal::Weak(isolate, context_));
167167
context_.Reset();
168168
}
169169

src/node_realm-inl.h

+7-2
Original file line numberDiff line numberDiff line change
@@ -86,8 +86,13 @@ inline T* Realm::AddBindingData(v8::Local<v8::Context> context,
8686
Args&&... args) {
8787
DCHECK_EQ(GetCurrent(context), this);
8888
// This won't compile if T is not a BaseObject subclass.
89-
BaseObjectPtr<T> item =
90-
MakeDetachedBaseObject<T>(this, target, std::forward<Args>(args)...);
89+
static_assert(std::is_base_of_v<BaseObject, T>);
90+
// The binding data must be weak so that it won't keep the realm reachable
91+
// from strong GC roots indefinitely. The wrapper object of binding data
92+
// should be referenced from JavaScript, thus the binding data should be
93+
// reachable throughout the lifetime of the realm.
94+
BaseObjectWeakPtr<T> item =
95+
MakeWeakBaseObject<T>(this, target, std::forward<Args>(args)...);
9196
DCHECK_EQ(context->GetAlignedPointerFromEmbedderData(
9297
ContextEmbedderIndex::kBindingDataStoreIndex),
9398
&binding_data_store_);

src/node_realm.cc

+12
Original file line numberDiff line numberDiff line change
@@ -21,6 +21,7 @@ using v8::Value;
2121
Realm::Realm(Environment* env, v8::Local<v8::Context> context, Kind kind)
2222
: env_(env), isolate_(context->GetIsolate()), kind_(kind) {
2323
context_.Reset(isolate_, context);
24+
env->AssignToContext(context, this, ContextInfo(""));
2425
}
2526

2627
Realm::~Realm() {
@@ -278,11 +279,15 @@ v8::Local<v8::Context> Realm::context() const {
278279
return PersistentToLocal::Strong(context_);
279280
}
280281

282+
// Per-realm strong value accessors. The per-realm values should avoid being
283+
// accessed across realms.
281284
#define V(PropertyName, TypeName) \
282285
v8::Local<TypeName> PrincipalRealm::PropertyName() const { \
283286
return PersistentToLocal::Strong(PropertyName##_); \
284287
} \
285288
void PrincipalRealm::set_##PropertyName(v8::Local<TypeName> value) { \
289+
DCHECK_IMPLIES(!value.IsEmpty(), \
290+
isolate()->GetCurrentContext() == context()); \
286291
PropertyName##_.Reset(isolate(), value); \
287292
}
288293
PER_REALM_STRONG_PERSISTENT_VALUES(V)
@@ -300,6 +305,13 @@ PrincipalRealm::PrincipalRealm(Environment* env,
300305
}
301306
}
302307

308+
PrincipalRealm::~PrincipalRealm() {
309+
DCHECK(!context_.IsEmpty());
310+
311+
HandleScope handle_scope(isolate());
312+
env_->UnassignToContext(context());
313+
}
314+
303315
MaybeLocal<Value> PrincipalRealm::BootstrapRealm() {
304316
HandleScope scope(isolate_);
305317

src/node_realm.h

+4-4
Original file line numberDiff line numberDiff line change
@@ -21,9 +21,9 @@ struct RealmSerializeInfo {
2121
friend std::ostream& operator<<(std::ostream& o, const RealmSerializeInfo& i);
2222
};
2323

24-
using BindingDataStore = std::array<BaseObjectPtr<BaseObject>,
25-
static_cast<size_t>(
26-
BindingDataType::kBindingDataTypeCount)>;
24+
using BindingDataStore =
25+
std::array<BaseObjectWeakPtr<BaseObject>,
26+
static_cast<size_t>(BindingDataType::kBindingDataTypeCount)>;
2727

2828
/**
2929
* node::Realm is a container for a set of JavaScript objects and functions
@@ -162,7 +162,7 @@ class PrincipalRealm : public Realm {
162162
PrincipalRealm(Environment* env,
163163
v8::Local<v8::Context> context,
164164
const RealmSerializeInfo* realm_info);
165-
~PrincipalRealm() = default;
165+
~PrincipalRealm();
166166

167167
SET_MEMORY_INFO_NAME(PrincipalRealm)
168168
SET_SELF_SIZE(PrincipalRealm)

src/node_shadow_realm.cc

+33-10
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,7 @@
55
namespace node {
66
namespace shadow_realm {
77
using v8::Context;
8+
using v8::EscapableHandleScope;
89
using v8::HandleScope;
910
using v8::Local;
1011
using v8::MaybeLocal;
@@ -15,7 +16,6 @@ using TryCatchScope = node::errors::TryCatchScope;
1516
// static
1617
ShadowRealm* ShadowRealm::New(Environment* env) {
1718
ShadowRealm* realm = new ShadowRealm(env);
18-
env->AssignToContext(realm->context(), realm, ContextInfo(""));
1919

2020
// We do not expect the realm bootstrapping to throw any
2121
// exceptions. If it does, exit the current Node.js instance.
@@ -31,39 +31,62 @@ ShadowRealm* ShadowRealm::New(Environment* env) {
3131
MaybeLocal<Context> HostCreateShadowRealmContextCallback(
3232
Local<Context> initiator_context) {
3333
Environment* env = Environment::GetCurrent(initiator_context);
34+
EscapableHandleScope scope(env->isolate());
3435
ShadowRealm* realm = ShadowRealm::New(env);
3536
if (realm != nullptr) {
36-
return realm->context();
37+
return scope.Escape(realm->context());
3738
}
3839
return MaybeLocal<Context>();
3940
}
4041

4142
// static
4243
void ShadowRealm::WeakCallback(const v8::WeakCallbackInfo<ShadowRealm>& data) {
4344
ShadowRealm* realm = data.GetParameter();
45+
realm->context_.Reset();
46+
47+
// Yield to pending weak callbacks before deleting the realm.
48+
// This is necessary to avoid cleaning up base objects before their scheduled
49+
// weak callbacks are invoked, which can lead to accessing to v8 apis during
50+
// the first pass of the weak callback.
51+
realm->env()->SetImmediate([realm](Environment* env) { delete realm; });
52+
// Remove the cleanup hook to avoid deleting the realm again.
53+
realm->env()->RemoveCleanupHook(DeleteMe, realm);
54+
}
55+
56+
// static
57+
void ShadowRealm::DeleteMe(void* data) {
58+
ShadowRealm* realm = static_cast<ShadowRealm*>(data);
59+
// Clear the context handle to avoid invoking the weak callback again.
60+
// Also, the context internal slots are cleared and the context is no longer
61+
// reference to the realm.
4462
delete realm;
4563
}
4664

4765
ShadowRealm::ShadowRealm(Environment* env)
4866
: Realm(env, NewContext(env->isolate()), kShadowRealm) {
49-
env->TrackShadowRealm(this);
5067
context_.SetWeak(this, WeakCallback, v8::WeakCallbackType::kParameter);
5168
CreateProperties();
69+
70+
env->TrackShadowRealm(this);
71+
env->AddCleanupHook(DeleteMe, this);
5272
}
5373

5474
ShadowRealm::~ShadowRealm() {
5575
while (HasCleanupHooks()) {
5676
RunCleanup();
5777
}
58-
if (env_ != nullptr) {
59-
env_->UntrackShadowRealm(this);
78+
79+
env_->UntrackShadowRealm(this);
80+
81+
if (context_.IsEmpty()) {
82+
// This most likely happened because the weak callback cleared it.
83+
return;
6084
}
61-
}
6285

63-
void ShadowRealm::OnEnvironmentDestruct() {
64-
CHECK_NOT_NULL(env_);
65-
env_ = nullptr; // This means that the shadow realm has out-lived the
66-
// environment.
86+
{
87+
HandleScope handle_scope(isolate());
88+
env_->UnassignToContext(context());
89+
}
6790
}
6891

6992
v8::Local<v8::Context> ShadowRealm::context() const {

src/node_shadow_realm.h

+1-2
Original file line numberDiff line numberDiff line change
@@ -24,13 +24,12 @@ class ShadowRealm : public Realm {
2424
PER_REALM_STRONG_PERSISTENT_VALUES(V)
2525
#undef V
2626

27-
void OnEnvironmentDestruct();
28-
2927
protected:
3028
v8::MaybeLocal<v8::Value> BootstrapRealm() override;
3129

3230
private:
3331
static void WeakCallback(const v8::WeakCallbackInfo<ShadowRealm>& data);
32+
static void DeleteMe(void* data);
3433

3534
explicit ShadowRealm(Environment* env);
3635
~ShadowRealm();

src/node_url.cc

+1
Original file line numberDiff line numberDiff line change
@@ -41,6 +41,7 @@ BindingData::BindingData(Realm* realm, v8::Local<v8::Object> object)
4141
FIXED_ONE_BYTE_STRING(realm->isolate(), "urlComponents"),
4242
url_components_buffer_.GetJSArray())
4343
.Check();
44+
url_components_buffer_.MakeWeak();
4445
}
4546

4647
bool BindingData::PrepareForSerialization(v8::Local<v8::Context> context,

test/known_issues/known_issues.status

-3
Original file line numberDiff line numberDiff line change
@@ -11,9 +11,6 @@ prefix known_issues
1111
# foreseeable future. The test itself is flaky and skipped. It
1212
# serves as a demonstration of the issue only.
1313
test-vm-timeout-escape-queuemicrotask: SKIP
14-
# Skipping it because it crashes out of OOM instead of exiting.
15-
# https://github.com/nodejs/node/issues/47353
16-
test-shadow-realm-gc: SKIP
1714

1815
[$system==win32]
1916

test/known_issues/test-shadow-realm-gc.js

-13
This file was deleted.

test/parallel/test-shadow-realm-gc.js

+22
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,22 @@
1+
// Flags: --experimental-shadow-realm --max-old-space-size=20
2+
'use strict';
3+
4+
/**
5+
* Verifying ShadowRealm instances can be correctly garbage collected.
6+
*/
7+
8+
const common = require('../common');
9+
const assert = require('assert');
10+
const { isMainThread, Worker } = require('worker_threads');
11+
12+
for (let i = 0; i < 100; i++) {
13+
const realm = new ShadowRealm();
14+
realm.evaluate('new TextEncoder(); 1;');
15+
}
16+
17+
if (isMainThread) {
18+
const worker = new Worker(__filename);
19+
worker.on('exit', common.mustCall((code) => {
20+
assert.strictEqual(code, 0);
21+
}));
22+
}

0 commit comments

Comments
 (0)