123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217 |
- /*
- MIT License http://www.opensource.org/licenses/mit-license.php
- Author Tobias Koppers @sokra
- */
- "use strict";
- const { RawSource } = require("webpack-sources");
- const { UsageState } = require("../ExportsInfo");
- const Generator = require("../Generator");
- const InitFragment = require("../InitFragment");
- const RuntimeGlobals = require("../RuntimeGlobals");
- const Template = require("../Template");
- const ModuleDependency = require("../dependencies/ModuleDependency");
- const WebAssemblyExportImportedDependency = require("../dependencies/WebAssemblyExportImportedDependency");
- const WebAssemblyImportDependency = require("../dependencies/WebAssemblyImportDependency");
- /** @typedef {import("webpack-sources").Source} Source */
- /** @typedef {import("../Dependency")} Dependency */
- /** @typedef {import("../DependencyTemplates")} DependencyTemplates */
- /** @typedef {import("../Generator").GenerateContext} GenerateContext */
- /** @typedef {import("../Module")} Module */
- /** @typedef {import("../NormalModule")} NormalModule */
- /** @typedef {import("../RuntimeTemplate")} RuntimeTemplate */
- const TYPES = new Set(["webassembly"]);
- class WebAssemblyJavascriptGenerator extends Generator {
- /**
- * @param {NormalModule} module fresh module
- * @returns {Set<string>} available types (do not mutate)
- */
- getTypes(module) {
- return TYPES;
- }
- /**
- * @param {NormalModule} module the module
- * @param {string=} type source type
- * @returns {number} estimate size of the module
- */
- getSize(module, type) {
- return 95 + module.dependencies.length * 5;
- }
- /**
- * @param {NormalModule} module module for which the code should be generated
- * @param {GenerateContext} generateContext context for generate
- * @returns {Source} generated code
- */
- generate(module, generateContext) {
- const {
- runtimeTemplate,
- moduleGraph,
- chunkGraph,
- runtimeRequirements,
- runtime
- } = generateContext;
- /** @type {InitFragment<InitFragment<string>>[]} */
- const initFragments = [];
- const exportsInfo = moduleGraph.getExportsInfo(module);
- let needExportsCopy = false;
- const importedModules = new Map();
- const initParams = [];
- let index = 0;
- for (const dep of module.dependencies) {
- const moduleDep =
- dep && dep instanceof ModuleDependency ? dep : undefined;
- if (moduleGraph.getModule(dep)) {
- let importData = importedModules.get(moduleGraph.getModule(dep));
- if (importData === undefined) {
- importedModules.set(
- moduleGraph.getModule(dep),
- (importData = {
- importVar: `m${index}`,
- index,
- request: (moduleDep && moduleDep.userRequest) || undefined,
- names: new Set(),
- reexports: []
- })
- );
- index++;
- }
- if (dep instanceof WebAssemblyImportDependency) {
- importData.names.add(dep.name);
- if (dep.description.type === "GlobalType") {
- const exportName = dep.name;
- const importedModule = moduleGraph.getModule(dep);
- if (importedModule) {
- const usedName = moduleGraph
- .getExportsInfo(importedModule)
- .getUsedName(exportName, runtime);
- if (usedName) {
- initParams.push(
- runtimeTemplate.exportFromImport({
- moduleGraph,
- module: importedModule,
- request: dep.request,
- importVar: importData.importVar,
- originModule: module,
- exportName: dep.name,
- asiSafe: true,
- isCall: false,
- callContext: null,
- defaultInterop: true,
- initFragments,
- runtime,
- runtimeRequirements
- })
- );
- }
- }
- }
- }
- if (dep instanceof WebAssemblyExportImportedDependency) {
- importData.names.add(dep.name);
- const usedName = moduleGraph
- .getExportsInfo(module)
- .getUsedName(dep.exportName, runtime);
- if (usedName) {
- runtimeRequirements.add(RuntimeGlobals.exports);
- const exportProp = `${module.exportsArgument}[${JSON.stringify(
- usedName
- )}]`;
- const defineStatement = Template.asString([
- `${exportProp} = ${runtimeTemplate.exportFromImport({
- moduleGraph,
- module: /** @type {Module} */ (moduleGraph.getModule(dep)),
- request: dep.request,
- importVar: importData.importVar,
- originModule: module,
- exportName: dep.name,
- asiSafe: true,
- isCall: false,
- callContext: null,
- defaultInterop: true,
- initFragments,
- runtime,
- runtimeRequirements
- })};`,
- `if(WebAssembly.Global) ${exportProp} = ` +
- `new WebAssembly.Global({ value: ${JSON.stringify(
- dep.valueType
- )} }, ${exportProp});`
- ]);
- importData.reexports.push(defineStatement);
- needExportsCopy = true;
- }
- }
- }
- }
- const importsCode = Template.asString(
- Array.from(
- importedModules,
- ([module, { importVar, request, reexports }]) => {
- const importStatement = runtimeTemplate.importStatement({
- module,
- chunkGraph,
- request,
- importVar,
- originModule: module,
- runtimeRequirements
- });
- return importStatement[0] + importStatement[1] + reexports.join("\n");
- }
- )
- );
- const copyAllExports =
- exportsInfo.otherExportsInfo.getUsed(runtime) === UsageState.Unused &&
- !needExportsCopy;
- // need these globals
- runtimeRequirements.add(RuntimeGlobals.module);
- runtimeRequirements.add(RuntimeGlobals.moduleId);
- runtimeRequirements.add(RuntimeGlobals.wasmInstances);
- if (exportsInfo.otherExportsInfo.getUsed(runtime) !== UsageState.Unused) {
- runtimeRequirements.add(RuntimeGlobals.makeNamespaceObject);
- runtimeRequirements.add(RuntimeGlobals.exports);
- }
- if (!copyAllExports) {
- runtimeRequirements.add(RuntimeGlobals.exports);
- }
- // create source
- const source = new RawSource(
- [
- '"use strict";',
- "// Instantiate WebAssembly module",
- `var wasmExports = ${RuntimeGlobals.wasmInstances}[${module.moduleArgument}.id];`,
- exportsInfo.otherExportsInfo.getUsed(runtime) !== UsageState.Unused
- ? `${RuntimeGlobals.makeNamespaceObject}(${module.exportsArgument});`
- : "",
- // this must be before import for circular dependencies
- "// export exports from WebAssembly module",
- copyAllExports
- ? `${module.moduleArgument}.exports = wasmExports;`
- : "for(var name in wasmExports) " +
- `if(name) ` +
- `${module.exportsArgument}[name] = wasmExports[name];`,
- "// exec imports from WebAssembly module (for esm order)",
- importsCode,
- "",
- "// exec wasm module",
- `wasmExports[""](${initParams.join(", ")})`
- ].join("\n")
- );
- return InitFragment.addToSource(source, initFragments, generateContext);
- }
- }
- module.exports = WebAssemblyJavascriptGenerator;
|