Skip to content

Commit a79c903

Browse files
committed
lib,src: iterate module requests of a module wrap in JS
Avoid repetitively calling into JS callback from C++ in `ModuleWrap::Link`. This removes the convoluted callback style of the internal `ModuleWrap` link step.
1 parent 6f4d601 commit a79c903

File tree

7 files changed

+247
-248
lines changed

7 files changed

+247
-248
lines changed

lib/internal/modules/esm/module_job.js

+53-31
Original file line numberDiff line numberDiff line change
@@ -1,8 +1,8 @@
11
'use strict';
22

33
const {
4+
Array,
45
ArrayPrototypeJoin,
5-
ArrayPrototypePush,
66
ArrayPrototypeSome,
77
FunctionPrototype,
88
ObjectSetPrototypeOf,
@@ -83,30 +83,8 @@ class ModuleJob extends ModuleJobBase {
8383
this.modulePromise = PromiseResolve(this.modulePromise);
8484
}
8585

86-
// Wait for the ModuleWrap instance being linked with all dependencies.
87-
const link = async () => {
88-
this.module = await this.modulePromise;
89-
assert(this.module instanceof ModuleWrap);
90-
91-
// Explicitly keeping track of dependency jobs is needed in order
92-
// to flatten out the dependency graph below in `_instantiate()`,
93-
// so that circular dependencies can't cause a deadlock by two of
94-
// these `link` callbacks depending on each other.
95-
const dependencyJobs = [];
96-
const promises = this.module.link(async (specifier, attributes) => {
97-
const job = await this.loader.getModuleJob(specifier, url, attributes);
98-
ArrayPrototypePush(dependencyJobs, job);
99-
return job.modulePromise;
100-
});
101-
102-
if (promises !== undefined) {
103-
await SafePromiseAllReturnVoid(promises);
104-
}
105-
106-
return SafePromiseAllReturnArrayLike(dependencyJobs);
107-
};
10886
// Promise for the list of all dependencyJobs.
109-
this.linked = link();
87+
this.linked = this._link();
11088
// This promise is awaited later anyway, so silence
11189
// 'unhandled rejection' warnings.
11290
PromisePrototypeThen(this.linked, undefined, noop);
@@ -116,6 +94,48 @@ class ModuleJob extends ModuleJobBase {
11694
this.instantiated = undefined;
11795
}
11896

97+
/**
98+
* Iterates the module requests and links with the loader.
99+
* @returns {Promise<ModuleJob[]>} Dependency module jobs.
100+
*/
101+
async _link() {
102+
this.module = await this.modulePromise;
103+
assert(this.module instanceof ModuleWrap);
104+
105+
const moduleRequestsLength = this.module.moduleRequests.length;
106+
// Explicitly keeping track of dependency jobs is needed in order
107+
// to flatten out the dependency graph below in `_instantiate()`,
108+
// so that circular dependencies can't cause a deadlock by two of
109+
// these `link` callbacks depending on each other.
110+
// Create an ArrayLike to avoid calling into userspace with `.then`
111+
// when returned from the async function.
112+
const dependencyJobs = Array(moduleRequestsLength);
113+
ObjectSetPrototypeOf(dependencyJobs, null);
114+
115+
// Specifiers should be aligned with the moduleRequests array in order.
116+
const specifiers = Array(moduleRequestsLength);
117+
const modulePromises = Array(moduleRequestsLength);
118+
// Iterate with index to avoid calling into userspace with `Symbol.iterator`.
119+
for (let idx = 0; idx < moduleRequestsLength; idx++) {
120+
const { specifier, attributes } = this.module.moduleRequests[idx];
121+
122+
const dependencyJobPromise = this.loader.getModuleJob(
123+
specifier, this.url, attributes,
124+
);
125+
const modulePromise = PromisePrototypeThen(dependencyJobPromise, (job) => {
126+
dependencyJobs[idx] = job;
127+
return job.modulePromise;
128+
});
129+
modulePromises[idx] = modulePromise;
130+
specifiers[idx] = specifier;
131+
}
132+
133+
const modules = await SafePromiseAllReturnArrayLike(modulePromises);
134+
this.module.link(specifiers, modules);
135+
136+
return dependencyJobs;
137+
}
138+
119139
instantiate() {
120140
if (this.instantiated === undefined) {
121141
this.instantiated = this._instantiate();
@@ -268,15 +288,17 @@ class ModuleJobSync extends ModuleJobBase {
268288
constructor(loader, url, importAttributes, moduleWrap, isMain, inspectBrk) {
269289
super(loader, url, importAttributes, moduleWrap, isMain, inspectBrk, true);
270290
assert(this.module instanceof ModuleWrap);
271-
const moduleRequests = this.module.getModuleRequestsSync();
272-
for (let i = 0; i < moduleRequests.length; ++i) {
273-
const { 0: specifier, 1: attributes } = moduleRequests[i];
291+
const moduleRequestsLength = this.module.moduleRequests.length;
292+
// Specifiers should be aligned with the moduleRequests array in order.
293+
const specifiers = Array(moduleRequestsLength);
294+
const modules = Array(moduleRequestsLength);
295+
for (let i = 0; i < moduleRequestsLength; ++i) {
296+
const { specifier, attributes } = this.module.moduleRequests[i];
274297
const wrap = this.loader.getModuleWrapForRequire(specifier, url, attributes);
275-
const isLast = (i === moduleRequests.length - 1);
276-
// TODO(joyeecheung): make the resolution callback deal with both promisified
277-
// an raw module wraps, then we don't need to wrap it with a promise here.
278-
this.module.cacheResolvedWrapsSync(specifier, PromiseResolve(wrap), isLast);
298+
specifiers[i] = specifier;
299+
modules[i] = wrap;
279300
}
301+
this.module.link(specifiers, modules);
280302
}
281303

282304
async run() {

lib/internal/vm/module.js

+56-36
Original file line numberDiff line numberDiff line change
@@ -2,16 +2,20 @@
22

33
const assert = require('internal/assert');
44
const {
5+
Array,
56
ArrayIsArray,
67
ArrayPrototypeForEach,
78
ArrayPrototypeIndexOf,
9+
ArrayPrototypeMap,
810
ArrayPrototypeSome,
911
ObjectDefineProperty,
1012
ObjectFreeze,
1113
ObjectGetPrototypeOf,
1214
ObjectSetPrototypeOf,
15+
PromiseResolve,
16+
PromisePrototypeThen,
1317
ReflectApply,
14-
SafePromiseAllReturnVoid,
18+
SafePromiseAllReturnArrayLike,
1519
Symbol,
1620
SymbolToStringTag,
1721
TypeError,
@@ -303,46 +307,62 @@ class SourceTextModule extends Module {
303307
importModuleDynamically,
304308
});
305309

306-
this[kLink] = async (linker) => {
307-
this.#statusOverride = 'linking';
310+
this[kDependencySpecifiers] = undefined;
311+
}
308312

309-
const promises = this[kWrap].link(async (identifier, attributes) => {
310-
const module = await linker(identifier, this, { attributes, assert: attributes });
311-
if (module[kWrap] === undefined) {
312-
throw new ERR_VM_MODULE_NOT_MODULE();
313-
}
314-
if (module.context !== this.context) {
315-
throw new ERR_VM_MODULE_DIFFERENT_CONTEXT();
316-
}
317-
if (module.status === 'errored') {
318-
throw new ERR_VM_MODULE_LINK_FAILURE(`request for '${identifier}' resolved to an errored module`, module.error);
319-
}
320-
if (module.status === 'unlinked') {
321-
await module[kLink](linker);
322-
}
323-
return module[kWrap];
313+
async [kLink](linker) {
314+
this.#statusOverride = 'linking';
315+
316+
const moduleRequestsLength = this[kWrap].moduleRequests.length;
317+
// Iterates the module requests and links with the linker.
318+
// Specifiers should be aligned with the moduleRequests array in order.
319+
const specifiers = Array(moduleRequestsLength);
320+
const modulePromises = Array(moduleRequestsLength);
321+
// Iterates with index to avoid calling into userspace with `Symbol.iterator`.
322+
for (let idx = 0; idx < moduleRequestsLength; idx++) {
323+
const { specifier, attributes } = this[kWrap].moduleRequests[idx];
324+
325+
const linkerResult = linker(specifier, this, {
326+
attributes,
327+
assert: attributes,
324328
});
325-
326-
try {
327-
if (promises !== undefined) {
328-
await SafePromiseAllReturnVoid(promises);
329-
}
330-
} catch (e) {
331-
this.#error = e;
332-
throw e;
333-
} finally {
334-
this.#statusOverride = undefined;
335-
}
336-
};
337-
338-
this[kDependencySpecifiers] = undefined;
329+
const modulePromise = PromisePrototypeThen(
330+
PromiseResolve(linkerResult), async (module) => {
331+
if (module[kWrap] === undefined) {
332+
throw new ERR_VM_MODULE_NOT_MODULE();
333+
}
334+
if (module.context !== this.context) {
335+
throw new ERR_VM_MODULE_DIFFERENT_CONTEXT();
336+
}
337+
if (module.status === 'errored') {
338+
throw new ERR_VM_MODULE_LINK_FAILURE(`request for '${specifier}' resolved to an errored module`, module.error);
339+
}
340+
if (module.status === 'unlinked') {
341+
await module[kLink](linker);
342+
}
343+
return module[kWrap];
344+
});
345+
modulePromises[idx] = modulePromise;
346+
specifiers[idx] = specifier;
347+
}
348+
349+
try {
350+
const modules = await SafePromiseAllReturnArrayLike(modulePromises);
351+
this[kWrap].link(specifiers, modules);
352+
} catch (e) {
353+
this.#error = e;
354+
throw e;
355+
} finally {
356+
this.#statusOverride = undefined;
357+
}
339358
}
340359

341360
get dependencySpecifiers() {
342361
if (this[kWrap] === undefined) {
343362
throw new ERR_VM_MODULE_NOT_MODULE();
344363
}
345-
this[kDependencySpecifiers] ??= ObjectFreeze(this[kWrap].getStaticDependencySpecifiers());
364+
this[kDependencySpecifiers] ??= ObjectFreeze(
365+
ArrayPrototypeMap(this[kWrap].moduleRequests, (request) => request.specifier));
346366
return this[kDependencySpecifiers];
347367
}
348368

@@ -408,10 +428,10 @@ class SyntheticModule extends Module {
408428
context,
409429
identifier,
410430
});
431+
}
411432

412-
this[kLink] = () => this[kWrap].link(() => {
413-
assert.fail('link callback should not be called');
414-
});
433+
[kLink]() {
434+
/** nothing to do for synthetic modules */
415435
}
416436

417437
setExport(name, value) {

src/env_properties.h

+4
Original file line numberDiff line numberDiff line change
@@ -67,6 +67,7 @@
6767
V(args_string, "args") \
6868
V(asn1curve_string, "asn1Curve") \
6969
V(async_ids_stack_string, "async_ids_stack") \
70+
V(attributes_string, "attributes") \
7071
V(base_string, "base") \
7172
V(bits_string, "bits") \
7273
V(block_list_string, "blockList") \
@@ -213,6 +214,7 @@
213214
V(mgf1_hash_algorithm_string, "mgf1HashAlgorithm") \
214215
V(minttl_string, "minttl") \
215216
V(module_string, "module") \
217+
V(module_requests_string, "moduleRequests") \
216218
V(modulus_string, "modulus") \
217219
V(modulus_length_string, "modulusLength") \
218220
V(name_string, "name") \
@@ -300,6 +302,7 @@
300302
V(sni_context_string, "sni_context") \
301303
V(source_string, "source") \
302304
V(source_map_url_string, "sourceMapURL") \
305+
V(specifier_string, "specifier") \
303306
V(stack_string, "stack") \
304307
V(standard_name_string, "standardName") \
305308
V(start_time_string, "startTime") \
@@ -377,6 +380,7 @@
377380
V(js_transferable_constructor_template, v8::FunctionTemplate) \
378381
V(libuv_stream_wrap_ctor_template, v8::FunctionTemplate) \
379382
V(message_port_constructor_template, v8::FunctionTemplate) \
383+
V(module_wrap_constructor_template, v8::FunctionTemplate) \
380384
V(microtask_queue_ctor_template, v8::FunctionTemplate) \
381385
V(pipe_constructor_template, v8::FunctionTemplate) \
382386
V(promise_wrap_template, v8::ObjectTemplate) \

0 commit comments

Comments
 (0)