You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
|
|
/* MIT License http://www.opensource.org/licenses/mit-license.php
Author Tobias Koppers @sokra */
"use strict";
const { compareIds } = require("../util/comparators");
/** @typedef {import("../Chunk")} Chunk */ /** @typedef {import("../Chunk").ChunkId} ChunkId */ /** @typedef {import("../Compiler")} Compiler */ /** @typedef {import("../Module")} Module */
class FlagIncludedChunksPlugin { /** * Apply the plugin * @param {Compiler} compiler the compiler instance * @returns {void} */ apply(compiler) { compiler.hooks.compilation.tap("FlagIncludedChunksPlugin", compilation => { compilation.hooks.optimizeChunkIds.tap( "FlagIncludedChunksPlugin", chunks => { const chunkGraph = compilation.chunkGraph;
// prepare two bit integers for each module
// 2^31 is the max number represented as SMI in v8
// we want the bits distributed this way:
// the bit 2^31 is pretty rar and only one module should get it
// so it has a probability of 1 / modulesCount
// the first bit (2^0) is the easiest and every module could get it
// if it doesn't get a better bit
// from bit 2^n to 2^(n+1) there is a probability of p
// so 1 / modulesCount == p^31
// <=> p = sqrt31(1 / modulesCount)
// so we use a modulo of 1 / sqrt31(1 / modulesCount)
/** @type {WeakMap<Module, number>} */ const moduleBits = new WeakMap(); const modulesCount = compilation.modules.size;
// precalculate the modulo values for each bit
const modulo = 1 / (1 / modulesCount) ** (1 / 31); const modulos = Array.from( { length: 31 }, (x, i) => (modulo ** i) | 0 );
// iterate all modules to generate bit values
let i = 0; for (const module of compilation.modules) { let bit = 30; while (i % modulos[bit] !== 0) { bit--; } moduleBits.set(module, 1 << bit); i++; }
// iterate all chunks to generate bitmaps
/** @type {WeakMap<Chunk, number>} */ const chunkModulesHash = new WeakMap(); for (const chunk of chunks) { let hash = 0; for (const module of chunkGraph.getChunkModulesIterable(chunk)) { hash |= /** @type {number} */ (moduleBits.get(module)); } chunkModulesHash.set(chunk, hash); }
for (const chunkA of chunks) { const chunkAHash = /** @type {number} */ (chunkModulesHash.get(chunkA)); const chunkAModulesCount = chunkGraph.getNumberOfChunkModules(chunkA); if (chunkAModulesCount === 0) continue; let bestModule; for (const module of chunkGraph.getChunkModulesIterable(chunkA)) { if ( bestModule === undefined || chunkGraph.getNumberOfModuleChunks(bestModule) > chunkGraph.getNumberOfModuleChunks(module) ) bestModule = module; } loopB: for (const chunkB of chunkGraph.getModuleChunksIterable( /** @type {Module} */ (bestModule) )) { // as we iterate the same iterables twice
// skip if we find ourselves
if (chunkA === chunkB) continue;
const chunkBModulesCount = chunkGraph.getNumberOfChunkModules(chunkB);
// ids for empty chunks are not included
if (chunkBModulesCount === 0) continue;
// instead of swapping A and B just bail
// as we loop twice the current A will be B and B then A
if (chunkAModulesCount > chunkBModulesCount) continue;
// is chunkA in chunkB?
// we do a cheap check for the hash value
const chunkBHash = /** @type {number} */ (chunkModulesHash.get(chunkB)); if ((chunkBHash & chunkAHash) !== chunkAHash) continue;
// compare all modules
for (const m of chunkGraph.getChunkModulesIterable(chunkA)) { if (!chunkGraph.isModuleInChunk(m, chunkB)) continue loopB; }
/** @type {ChunkId[]} */ (chunkB.ids).push(/** @type {ChunkId} */ (chunkA.id)); // https://github.com/webpack/webpack/issues/18837
/** @type {ChunkId[]} */ (chunkB.ids).sort(compareIds); } } } ); }); } } module.exports = FlagIncludedChunksPlugin;
|