Skip to content

Commit 2fd97ce

Browse files
guybedfordruyadorno
authored andcommitted
deps: v8 backport 9689b17687b
[top-level-await] Implement spec fix for cycle root detection Refs: http://github.com/v8/v8/commit/9689b17687b21c800c3f7400df4255c55b9c6ec0 PR-URL: #37865 Backport-PR-URL: #37985 Reviewed-By: Michaël Zasso <[email protected]>
1 parent f2cef54 commit 2fd97ce

13 files changed

+109
-62
lines changed

common.gypi

+1-1
Original file line numberDiff line numberDiff line change
@@ -36,7 +36,7 @@
3636

3737
# Reset this number to 0 on major V8 upgrades.
3838
# Increment by one for each non-official patch applied to deps/v8.
39-
'v8_embedder_string': '-node.27',
39+
'v8_embedder_string': '-node.28',
4040

4141
##### V8 defaults for Node.js #####
4242

deps/v8/src/diagnostics/objects-printer.cc

+1
Original file line numberDiff line numberDiff line change
@@ -1601,6 +1601,7 @@ void SourceTextModule::SourceTextModulePrint(std::ostream& os) { // NOLINT
16011601
os << "\n - requested_modules: " << Brief(requested_modules());
16021602
os << "\n - script: " << Brief(script());
16031603
os << "\n - import_meta: " << Brief(import_meta());
1604+
os << "\n - cycle_root: " << Brief(cycle_root());
16041605
os << "\n";
16051606
}
16061607

deps/v8/src/heap/factory.cc

+1
Original file line numberDiff line numberDiff line change
@@ -2505,6 +2505,7 @@ Handle<SourceTextModule> Factory::NewSourceTextModule(
25052505
module->set_flags(0);
25062506
module->set_async(IsAsyncModule(code->kind()));
25072507
module->set_async_evaluating(false);
2508+
module->set_cycle_root(roots.the_hole_value());
25082509
module->set_async_parent_modules(*async_parent_modules);
25092510
module->set_pending_async_dependencies(0);
25102511
return module;

deps/v8/src/objects/module-inl.h

+8
Original file line numberDiff line numberDiff line change
@@ -112,6 +112,14 @@ class UnorderedModuleSet
112112
ZoneAllocator<Handle<Module>>(zone)) {}
113113
};
114114

115+
Handle<SourceTextModule> SourceTextModule::GetCycleRoot(
116+
Isolate* isolate) const {
117+
CHECK_GE(status(), kEvaluated);
118+
DCHECK(!cycle_root().IsTheHole(isolate));
119+
Handle<SourceTextModule> root(SourceTextModule::cast(cycle_root()), isolate);
120+
return root;
121+
}
122+
115123
void SourceTextModule::AddAsyncParentModule(Isolate* isolate,
116124
Handle<SourceTextModule> module,
117125
Handle<SourceTextModule> parent) {

deps/v8/src/objects/source-text-module.cc

+14-57
Original file line numberDiff line numberDiff line change
@@ -397,6 +397,7 @@ bool SourceTextModule::MaybeTransitionComponent(
397397
DCHECK_LE(module->dfs_ancestor_index(), module->dfs_index());
398398
if (module->dfs_ancestor_index() == module->dfs_index()) {
399399
// This is the root of its strongly connected component.
400+
Handle<SourceTextModule> cycle_root = module;
400401
Handle<SourceTextModule> ancestor;
401402
do {
402403
ancestor = stack->front();
@@ -406,6 +407,9 @@ bool SourceTextModule::MaybeTransitionComponent(
406407
if (new_status == kInstantiated) {
407408
if (!SourceTextModule::RunInitializationCode(isolate, ancestor))
408409
return false;
410+
} else if (new_status == kEvaluated) {
411+
DCHECK(ancestor->cycle_root().IsTheHole(isolate));
412+
ancestor->set_cycle_root(*cycle_root);
409413
}
410414
ancestor->SetStatus(new_status);
411415
} while (*ancestor != *module);
@@ -619,9 +623,9 @@ MaybeHandle<Object> SourceTextModule::EvaluateMaybeAsync(
619623
CHECK(module->status() == kInstantiated || module->status() == kEvaluated);
620624

621625
// 3. If module.[[Status]] is "evaluated", set module to
622-
// GetAsyncCycleRoot(module).
626+
// module.[[CycleRoot]].
623627
if (module->status() == kEvaluated) {
624-
module = GetAsyncCycleRoot(isolate, module);
628+
module = module->GetCycleRoot(isolate);
625629
}
626630

627631
// 4. If module.[[TopLevelCapability]] is not undefined, then
@@ -736,37 +740,27 @@ void SourceTextModule::AsyncModuleExecutionFulfilled(
736740
for (int i = 0; i < module->AsyncParentModuleCount(); i++) {
737741
Handle<SourceTextModule> m = module->GetAsyncParentModule(isolate, i);
738742

739-
// a. If module.[[DFSIndex]] is not equal to module.[[DFSAncestorIndex]],
740-
// then
741-
if (module->dfs_index() != module->dfs_ancestor_index()) {
742-
// i. Assert: m.[[DFSAncestorIndex]] is equal to
743-
// module.[[DFSAncestorIndex]].
744-
DCHECK_LE(m->dfs_ancestor_index(), module->dfs_ancestor_index());
745-
}
746-
// b. Decrement m.[[PendingAsyncDependencies]] by 1.
743+
// a. Decrement m.[[PendingAsyncDependencies]] by 1.
747744
m->DecrementPendingAsyncDependencies();
748745

749-
// c. If m.[[PendingAsyncDependencies]] is 0 and m.[[EvaluationError]] is
746+
// b. If m.[[PendingAsyncDependencies]] is 0 and m.[[EvaluationError]] is
750747
// undefined, then
751748
if (!m->HasPendingAsyncDependencies() && m->status() == kEvaluated) {
752749
// i. Assert: m.[[AsyncEvaluating]] is true.
753750
DCHECK(m->async_evaluating());
754751

755-
// ii. Let cycleRoot be ! GetAsyncCycleRoot(m).
756-
auto cycle_root = GetAsyncCycleRoot(isolate, m);
757-
758-
// iii. If cycleRoot.[[EvaluationError]] is not undefined,
752+
// ii. If m.[[CycleRoot]].[[EvaluationError]] is not undefined,
759753
// return undefined.
760-
if (cycle_root->status() == kErrored) {
754+
if (m->GetCycleRoot(isolate)->status() == kErrored) {
761755
return;
762756
}
763757

764-
// iv. If m.[[Async]] is true, then
758+
// iii. If m.[[Async]] is true, then
765759
if (m->async()) {
766760
// 1. Perform ! ExecuteAsyncModule(m).
767761
ExecuteAsyncModule(isolate, m);
768762
} else {
769-
// v. Otherwise,
763+
// iv. Otherwise,
770764
// 1. Let result be m.ExecuteModule().
771765
// 2. If result is a normal completion,
772766
Handle<Object> unused_result;
@@ -1046,8 +1040,8 @@ MaybeHandle<Object> SourceTextModule::InnerModuleEvaluation(
10461040
required_module->dfs_ancestor_index()));
10471041
} else {
10481042
// iv. Otherwise,
1049-
// 1. Set requiredModule to GetAsyncCycleRoot(requiredModule).
1050-
required_module = GetAsyncCycleRoot(isolate, required_module);
1043+
// 1. Set requiredModule to requiredModule.[[CycleRoot]].
1044+
required_module = required_module->GetCycleRoot(isolate);
10511045

10521046
// 2. Assert: requiredModule.[[Status]] is "evaluated".
10531047
CHECK_GE(required_module->status(), kEvaluated);
@@ -1105,43 +1099,6 @@ MaybeHandle<Object> SourceTextModule::InnerModuleEvaluation(
11051099
return result;
11061100
}
11071101

1108-
Handle<SourceTextModule> SourceTextModule::GetAsyncCycleRoot(
1109-
Isolate* isolate, Handle<SourceTextModule> module) {
1110-
// 1. Assert: module.[[Status]] is "evaluated".
1111-
CHECK_GE(module->status(), kEvaluated);
1112-
1113-
// 2. If module.[[AsyncParentModules]] is an empty List, return module.
1114-
if (module->AsyncParentModuleCount() == 0) {
1115-
return module;
1116-
}
1117-
1118-
// 3. Repeat, while module.[[DFSIndex]] is greater than
1119-
// module.[[DFSAncestorIndex]],
1120-
while (module->dfs_index() > module->dfs_ancestor_index()) {
1121-
// a. Assert: module.[[AsyncParentModules]] is a non-empty List.
1122-
DCHECK_GT(module->AsyncParentModuleCount(), 0);
1123-
1124-
// b. Let nextCycleModule be the first element of
1125-
// module.[[AsyncParentModules]].
1126-
Handle<SourceTextModule> next_cycle_module =
1127-
module->GetAsyncParentModule(isolate, 0);
1128-
1129-
// c. Assert: nextCycleModule.[[DFSAncestorIndex]] is less than or equal
1130-
// to module.[[DFSAncestorIndex]].
1131-
DCHECK_LE(next_cycle_module->dfs_ancestor_index(),
1132-
module->dfs_ancestor_index());
1133-
1134-
// d. Set module to nextCycleModule
1135-
module = next_cycle_module;
1136-
}
1137-
1138-
// 4. Assert: module.[[DFSIndex]] is equal to module.[[DFSAncestorIndex]].
1139-
DCHECK_EQ(module->dfs_index(), module->dfs_ancestor_index());
1140-
1141-
// 5. Return module.
1142-
return module;
1143-
}
1144-
11451102
void SourceTextModule::Reset(Isolate* isolate,
11461103
Handle<SourceTextModule> module) {
11471104
Factory* factory = isolate->factory();

deps/v8/src/objects/source-text-module.h

+3-4
Original file line numberDiff line numberDiff line change
@@ -79,6 +79,9 @@ class SourceTextModule
7979
Handle<SourceTextModule> module,
8080
Handle<SourceTextModule> parent);
8181

82+
// Get the non-hole cycle root. Only valid when status >= kEvaluated.
83+
inline Handle<SourceTextModule> GetCycleRoot(Isolate* isolate) const;
84+
8285
// Returns a SourceTextModule, the
8386
// ith parent in depth first traversal order of a given async child.
8487
inline Handle<SourceTextModule> GetAsyncParentModule(Isolate* isolate,
@@ -163,10 +166,6 @@ class SourceTextModule
163166
Isolate* isolate, Handle<SourceTextModule> module,
164167
ZoneForwardList<Handle<SourceTextModule>>* stack, Status new_status);
165168

166-
// Implementation of spec GetAsyncCycleRoot.
167-
static V8_WARN_UNUSED_RESULT Handle<SourceTextModule> GetAsyncCycleRoot(
168-
Isolate* isolate, Handle<SourceTextModule> module);
169-
170169
// Implementation of spec ExecuteModule is broken up into
171170
// InnerExecuteAsyncModule for asynchronous modules and ExecuteModule
172171
// for synchronous modules.

deps/v8/src/objects/source-text-module.tq

+5
Original file line numberDiff line numberDiff line change
@@ -34,6 +34,11 @@ extern class SourceTextModule extends Module {
3434
// a JSObject afterwards.
3535
import_meta: TheHole|JSObject;
3636

37+
// The first visited module of a cycle. For modules not in a cycle, this is
38+
// the module itself. It's the hole before the module state transitions to
39+
// kEvaluated.
40+
cycle_root: SourceTextModule|TheHole;
41+
3742
async_parent_modules: ArrayList;
3843
top_level_capability: JSPromise|Undefined;
3944

Original file line numberDiff line numberDiff line change
@@ -0,0 +1,12 @@
1+
// Copyright 2021 the V8 project authors. All rights reserved.
2+
// Use of this source code is governed by a BSD-style license that can be
3+
// found in the LICENSE file.
4+
5+
// Flags: --harmony-top-level-await
6+
7+
import "modules-skip-async-cycle-start.mjs"
8+
9+
assertEquals(globalThis.test_order, [
10+
'2', 'async before', 'async after', '1',
11+
'3', 'start',
12+
]);
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,14 @@
1+
// Copyright 2021 the V8 project authors. All rights reserved.
2+
// Use of this source code is governed by a BSD-style license that can be
3+
// found in the LICENSE file.
4+
5+
// Flags: --harmony-top-level-await
6+
7+
import "modules-skip-async-cycle-2.mjs";
8+
import "modules-skip-async-cycle-leaf.mjs";
9+
10+
if (globalThis.test_order === undefined) {
11+
globalThis.test_order = [];
12+
}
13+
globalThis.test_order.push('1');
14+
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,12 @@
1+
// Copyright 2021 the V8 project authors. All rights reserved.
2+
// Use of this source code is governed by a BSD-style license that can be
3+
// found in the LICENSE file.
4+
5+
// Flags: --harmony-top-level-await
6+
7+
import "modules-skip-async-cycle-1.mjs";
8+
9+
if (globalThis.test_order === undefined) {
10+
globalThis.test_order = [];
11+
}
12+
globalThis.test_order.push('2');
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,12 @@
1+
// Copyright 2021 the V8 project authors. All rights reserved.
2+
// Use of this source code is governed by a BSD-style license that can be
3+
// found in the LICENSE file.
4+
5+
// Flags: --harmony-top-level-await
6+
7+
import "modules-skip-async-cycle-2.mjs";
8+
9+
if (globalThis.test_order === undefined) {
10+
globalThis.test_order = [];
11+
}
12+
globalThis.test_order.push('3');
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,13 @@
1+
// Copyright 2021 the V8 project authors. All rights reserved.
2+
// Use of this source code is governed by a BSD-style license that can be
3+
// found in the LICENSE file.
4+
5+
// Flags: --harmony-top-level-await
6+
7+
if (globalThis.test_order === undefined) {
8+
globalThis.test_order = [];
9+
}
10+
11+
globalThis.test_order.push('async before');
12+
await 0;
13+
globalThis.test_order.push('async after');
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,13 @@
1+
// Copyright 2021 the V8 project authors. All rights reserved.
2+
// Use of this source code is governed by a BSD-style license that can be
3+
// found in the LICENSE file.
4+
5+
// Flags: --harmony-top-level-await
6+
7+
import "modules-skip-async-cycle-1.mjs";
8+
import "modules-skip-async-cycle-3.mjs";
9+
10+
if (globalThis.test_order === undefined) {
11+
globalThis.test_order = [];
12+
}
13+
globalThis.test_order.push('start');

0 commit comments

Comments
 (0)