inline.js 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629
  1. /***********************************************************************
  2. A JavaScript tokenizer / parser / beautifier / compressor.
  3. https://github.com/mishoo/UglifyJS2
  4. -------------------------------- (C) ---------------------------------
  5. Author: Mihai Bazon
  6. <mihai.bazon@gmail.com>
  7. http://mihai.bazon.net/blog
  8. Distributed under the BSD license:
  9. Copyright 2012 (c) Mihai Bazon <mihai.bazon@gmail.com>
  10. Redistribution and use in source and binary forms, with or without
  11. modification, are permitted provided that the following conditions
  12. are met:
  13. * Redistributions of source code must retain the above
  14. copyright notice, this list of conditions and the following
  15. disclaimer.
  16. * Redistributions in binary form must reproduce the above
  17. copyright notice, this list of conditions and the following
  18. disclaimer in the documentation and/or other materials
  19. provided with the distribution.
  20. THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDER “AS IS” AND ANY
  21. EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
  22. IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
  23. PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER BE
  24. LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY,
  25. OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
  26. PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
  27. PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
  28. THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR
  29. TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF
  30. THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
  31. SUCH DAMAGE.
  32. ***********************************************************************/
  33. import {
  34. AST_Array,
  35. AST_Assign,
  36. AST_Block,
  37. AST_Call,
  38. AST_Catch,
  39. AST_Class,
  40. AST_ClassExpression,
  41. AST_DefaultAssign,
  42. AST_DefClass,
  43. AST_Defun,
  44. AST_Destructuring,
  45. AST_EmptyStatement,
  46. AST_Expansion,
  47. AST_Export,
  48. AST_Function,
  49. AST_IterationStatement,
  50. AST_Lambda,
  51. AST_Node,
  52. AST_Number,
  53. AST_Object,
  54. AST_ObjectKeyVal,
  55. AST_PropAccess,
  56. AST_Return,
  57. AST_Scope,
  58. AST_SimpleStatement,
  59. AST_Statement,
  60. AST_SymbolDefun,
  61. AST_SymbolFunarg,
  62. AST_SymbolLambda,
  63. AST_SymbolRef,
  64. AST_SymbolVar,
  65. AST_This,
  66. AST_Toplevel,
  67. AST_UnaryPrefix,
  68. AST_Undefined,
  69. AST_Var,
  70. AST_VarDef,
  71. walk,
  72. _INLINE,
  73. _NOINLINE,
  74. _PURE
  75. } from "../ast.js";
  76. import { make_node, has_annotation } from "../utils/index.js";
  77. import "../size.js";
  78. import "./evaluate.js";
  79. import "./drop-side-effect-free.js";
  80. import "./reduce-vars.js";
  81. import {
  82. SQUEEZED,
  83. INLINED,
  84. UNUSED,
  85. has_flag,
  86. set_flag,
  87. } from "./compressor-flags.js";
  88. import {
  89. make_sequence,
  90. best_of,
  91. make_node_from_constant,
  92. identifier_atom,
  93. is_empty,
  94. is_func_expr,
  95. is_iife_call,
  96. is_reachable,
  97. is_recursive_ref,
  98. retain_top_func,
  99. } from "./common.js";
  100. /**
  101. * Module that contains the inlining logic.
  102. *
  103. * @module
  104. *
  105. * The stars of the show are `inline_into_symbolref` and `inline_into_call`.
  106. */
  107. function within_array_or_object_literal(compressor) {
  108. var node, level = 0;
  109. while (node = compressor.parent(level++)) {
  110. if (node instanceof AST_Statement) return false;
  111. if (node instanceof AST_Array
  112. || node instanceof AST_ObjectKeyVal
  113. || node instanceof AST_Object) {
  114. return true;
  115. }
  116. }
  117. return false;
  118. }
  119. function scope_encloses_variables_in_this_scope(scope, pulled_scope) {
  120. for (const enclosed of pulled_scope.enclosed) {
  121. if (pulled_scope.variables.has(enclosed.name)) {
  122. continue;
  123. }
  124. const looked_up = scope.find_variable(enclosed.name);
  125. if (looked_up) {
  126. if (looked_up === enclosed) continue;
  127. return true;
  128. }
  129. }
  130. return false;
  131. }
  132. export function inline_into_symbolref(self, compressor) {
  133. const parent = compressor.parent();
  134. const def = self.definition();
  135. const nearest_scope = compressor.find_scope();
  136. if (compressor.top_retain && def.global && compressor.top_retain(def)) {
  137. def.fixed = false;
  138. def.single_use = false;
  139. return self;
  140. }
  141. let fixed = self.fixed_value();
  142. let single_use = def.single_use
  143. && !(parent instanceof AST_Call
  144. && (parent.is_callee_pure(compressor))
  145. || has_annotation(parent, _NOINLINE))
  146. && !(parent instanceof AST_Export
  147. && fixed instanceof AST_Lambda
  148. && fixed.name);
  149. if (single_use && fixed instanceof AST_Node) {
  150. single_use =
  151. !fixed.has_side_effects(compressor)
  152. && !fixed.may_throw(compressor);
  153. }
  154. if (single_use && (fixed instanceof AST_Lambda || fixed instanceof AST_Class)) {
  155. if (retain_top_func(fixed, compressor)) {
  156. single_use = false;
  157. } else if (def.scope !== self.scope
  158. && (def.escaped == 1
  159. || has_flag(fixed, INLINED)
  160. || within_array_or_object_literal(compressor)
  161. || !compressor.option("reduce_funcs"))) {
  162. single_use = false;
  163. } else if (is_recursive_ref(compressor, def)) {
  164. single_use = false;
  165. } else if (def.scope !== self.scope || def.orig[0] instanceof AST_SymbolFunarg) {
  166. single_use = fixed.is_constant_expression(self.scope);
  167. if (single_use == "f") {
  168. var scope = self.scope;
  169. do {
  170. if (scope instanceof AST_Defun || is_func_expr(scope)) {
  171. set_flag(scope, INLINED);
  172. }
  173. } while (scope = scope.parent_scope);
  174. }
  175. }
  176. }
  177. if (single_use && (fixed instanceof AST_Lambda || fixed instanceof AST_Class)) {
  178. single_use =
  179. def.scope === self.scope
  180. && !scope_encloses_variables_in_this_scope(nearest_scope, fixed)
  181. || parent instanceof AST_Call
  182. && parent.expression === self
  183. && !scope_encloses_variables_in_this_scope(nearest_scope, fixed)
  184. && !(fixed.name && fixed.name.definition().recursive_refs > 0);
  185. }
  186. if (single_use && fixed) {
  187. if (fixed instanceof AST_DefClass) {
  188. set_flag(fixed, SQUEEZED);
  189. fixed = make_node(AST_ClassExpression, fixed, fixed);
  190. }
  191. if (fixed instanceof AST_Defun) {
  192. set_flag(fixed, SQUEEZED);
  193. fixed = make_node(AST_Function, fixed, fixed);
  194. }
  195. if (def.recursive_refs > 0 && fixed.name instanceof AST_SymbolDefun) {
  196. const defun_def = fixed.name.definition();
  197. let lambda_def = fixed.variables.get(fixed.name.name);
  198. let name = lambda_def && lambda_def.orig[0];
  199. if (!(name instanceof AST_SymbolLambda)) {
  200. name = make_node(AST_SymbolLambda, fixed.name, fixed.name);
  201. name.scope = fixed;
  202. fixed.name = name;
  203. lambda_def = fixed.def_function(name);
  204. }
  205. walk(fixed, node => {
  206. if (node instanceof AST_SymbolRef && node.definition() === defun_def) {
  207. node.thedef = lambda_def;
  208. lambda_def.references.push(node);
  209. }
  210. });
  211. }
  212. if (
  213. (fixed instanceof AST_Lambda || fixed instanceof AST_Class)
  214. && fixed.parent_scope !== nearest_scope
  215. ) {
  216. fixed = fixed.clone(true, compressor.get_toplevel());
  217. nearest_scope.add_child_scope(fixed);
  218. }
  219. return fixed.optimize(compressor);
  220. }
  221. // multiple uses
  222. if (fixed) {
  223. let replace;
  224. if (fixed instanceof AST_This) {
  225. if (!(def.orig[0] instanceof AST_SymbolFunarg)
  226. && def.references.every((ref) =>
  227. def.scope === ref.scope
  228. )) {
  229. replace = fixed;
  230. }
  231. } else {
  232. var ev = fixed.evaluate(compressor);
  233. if (
  234. ev !== fixed
  235. && (compressor.option("unsafe_regexp") || !(ev instanceof RegExp))
  236. ) {
  237. replace = make_node_from_constant(ev, fixed);
  238. }
  239. }
  240. if (replace) {
  241. const name_length = self.size(compressor);
  242. const replace_size = replace.size(compressor);
  243. let overhead = 0;
  244. if (compressor.option("unused") && !compressor.exposed(def)) {
  245. overhead =
  246. (name_length + 2 + replace_size) /
  247. (def.references.length - def.assignments);
  248. }
  249. if (replace_size <= name_length + overhead) {
  250. return replace;
  251. }
  252. }
  253. }
  254. return self;
  255. }
  256. export function inline_into_call(self, fn, compressor) {
  257. var exp = self.expression;
  258. var simple_args = self.args.every((arg) => !(arg instanceof AST_Expansion));
  259. if (compressor.option("reduce_vars")
  260. && fn instanceof AST_SymbolRef
  261. && !has_annotation(self, _NOINLINE)
  262. ) {
  263. const fixed = fn.fixed_value();
  264. if (!retain_top_func(fixed, compressor)) {
  265. fn = fixed;
  266. }
  267. }
  268. var is_func = fn instanceof AST_Lambda;
  269. var stat = is_func && fn.body[0];
  270. var is_regular_func = is_func && !fn.is_generator && !fn.async;
  271. var can_inline = is_regular_func && compressor.option("inline") && !self.is_callee_pure(compressor);
  272. if (can_inline && stat instanceof AST_Return) {
  273. let returned = stat.value;
  274. if (!returned || returned.is_constant_expression()) {
  275. if (returned) {
  276. returned = returned.clone(true);
  277. } else {
  278. returned = make_node(AST_Undefined, self);
  279. }
  280. const args = self.args.concat(returned);
  281. return make_sequence(self, args).optimize(compressor);
  282. }
  283. // optimize identity function
  284. if (
  285. fn.argnames.length === 1
  286. && (fn.argnames[0] instanceof AST_SymbolFunarg)
  287. && self.args.length < 2
  288. && !(self.args[0] instanceof AST_Expansion)
  289. && returned instanceof AST_SymbolRef
  290. && returned.name === fn.argnames[0].name
  291. ) {
  292. const replacement =
  293. (self.args[0] || make_node(AST_Undefined)).optimize(compressor);
  294. let parent;
  295. if (
  296. replacement instanceof AST_PropAccess
  297. && (parent = compressor.parent()) instanceof AST_Call
  298. && parent.expression === self
  299. ) {
  300. // identity function was being used to remove `this`, like in
  301. //
  302. // id(bag.no_this)(...)
  303. //
  304. // Replace with a larger but more effish (0, bag.no_this) wrapper.
  305. return make_sequence(self, [
  306. make_node(AST_Number, self, { value: 0 }),
  307. replacement
  308. ]);
  309. }
  310. // replace call with first argument or undefined if none passed
  311. return replacement;
  312. }
  313. }
  314. if (can_inline) {
  315. var scope, in_loop, level = -1;
  316. let def;
  317. let returned_value;
  318. let nearest_scope;
  319. if (simple_args
  320. && !fn.uses_arguments
  321. && !(compressor.parent() instanceof AST_Class)
  322. && !(fn.name && fn instanceof AST_Function)
  323. && (returned_value = can_flatten_body(stat))
  324. && (exp === fn
  325. || has_annotation(self, _INLINE)
  326. || compressor.option("unused")
  327. && (def = exp.definition()).references.length == 1
  328. && !is_recursive_ref(compressor, def)
  329. && fn.is_constant_expression(exp.scope))
  330. && !has_annotation(self, _PURE | _NOINLINE)
  331. && !fn.contains_this()
  332. && can_inject_symbols()
  333. && (nearest_scope = compressor.find_scope())
  334. && !scope_encloses_variables_in_this_scope(nearest_scope, fn)
  335. && !(function in_default_assign() {
  336. // Due to the fact function parameters have their own scope
  337. // which can't use `var something` in the function body within,
  338. // we simply don't inline into DefaultAssign.
  339. let i = 0;
  340. let p;
  341. while ((p = compressor.parent(i++))) {
  342. if (p instanceof AST_DefaultAssign) return true;
  343. if (p instanceof AST_Block) break;
  344. }
  345. return false;
  346. })()
  347. && !(scope instanceof AST_Class)
  348. ) {
  349. set_flag(fn, SQUEEZED);
  350. nearest_scope.add_child_scope(fn);
  351. return make_sequence(self, flatten_fn(returned_value)).optimize(compressor);
  352. }
  353. }
  354. if (can_inline && has_annotation(self, _INLINE)) {
  355. set_flag(fn, SQUEEZED);
  356. fn = make_node(fn.CTOR === AST_Defun ? AST_Function : fn.CTOR, fn, fn);
  357. fn = fn.clone(true);
  358. fn.figure_out_scope({}, {
  359. parent_scope: compressor.find_scope(),
  360. toplevel: compressor.get_toplevel()
  361. });
  362. return make_node(AST_Call, self, {
  363. expression: fn,
  364. args: self.args,
  365. }).optimize(compressor);
  366. }
  367. const can_drop_this_call = is_regular_func && compressor.option("side_effects") && fn.body.every(is_empty);
  368. if (can_drop_this_call) {
  369. var args = self.args.concat(make_node(AST_Undefined, self));
  370. return make_sequence(self, args).optimize(compressor);
  371. }
  372. if (compressor.option("negate_iife")
  373. && compressor.parent() instanceof AST_SimpleStatement
  374. && is_iife_call(self)) {
  375. return self.negate(compressor, true);
  376. }
  377. var ev = self.evaluate(compressor);
  378. if (ev !== self) {
  379. ev = make_node_from_constant(ev, self).optimize(compressor);
  380. return best_of(compressor, ev, self);
  381. }
  382. return self;
  383. function return_value(stat) {
  384. if (!stat) return make_node(AST_Undefined, self);
  385. if (stat instanceof AST_Return) {
  386. if (!stat.value) return make_node(AST_Undefined, self);
  387. return stat.value.clone(true);
  388. }
  389. if (stat instanceof AST_SimpleStatement) {
  390. return make_node(AST_UnaryPrefix, stat, {
  391. operator: "void",
  392. expression: stat.body.clone(true)
  393. });
  394. }
  395. }
  396. function can_flatten_body(stat) {
  397. var body = fn.body;
  398. var len = body.length;
  399. if (compressor.option("inline") < 3) {
  400. return len == 1 && return_value(stat);
  401. }
  402. stat = null;
  403. for (var i = 0; i < len; i++) {
  404. var line = body[i];
  405. if (line instanceof AST_Var) {
  406. if (stat && !line.definitions.every((var_def) =>
  407. !var_def.value
  408. )) {
  409. return false;
  410. }
  411. } else if (stat) {
  412. return false;
  413. } else if (!(line instanceof AST_EmptyStatement)) {
  414. stat = line;
  415. }
  416. }
  417. return return_value(stat);
  418. }
  419. function can_inject_args(block_scoped, safe_to_inject) {
  420. for (var i = 0, len = fn.argnames.length; i < len; i++) {
  421. var arg = fn.argnames[i];
  422. if (arg instanceof AST_DefaultAssign) {
  423. if (has_flag(arg.left, UNUSED)) continue;
  424. return false;
  425. }
  426. if (arg instanceof AST_Destructuring) return false;
  427. if (arg instanceof AST_Expansion) {
  428. if (has_flag(arg.expression, UNUSED)) continue;
  429. return false;
  430. }
  431. if (has_flag(arg, UNUSED)) continue;
  432. if (!safe_to_inject
  433. || block_scoped.has(arg.name)
  434. || identifier_atom.has(arg.name)
  435. || scope.conflicting_def(arg.name)) {
  436. return false;
  437. }
  438. if (in_loop) in_loop.push(arg.definition());
  439. }
  440. return true;
  441. }
  442. function can_inject_vars(block_scoped, safe_to_inject) {
  443. var len = fn.body.length;
  444. for (var i = 0; i < len; i++) {
  445. var stat = fn.body[i];
  446. if (!(stat instanceof AST_Var)) continue;
  447. if (!safe_to_inject) return false;
  448. for (var j = stat.definitions.length; --j >= 0;) {
  449. var name = stat.definitions[j].name;
  450. if (name instanceof AST_Destructuring
  451. || block_scoped.has(name.name)
  452. || identifier_atom.has(name.name)
  453. || scope.conflicting_def(name.name)) {
  454. return false;
  455. }
  456. if (in_loop) in_loop.push(name.definition());
  457. }
  458. }
  459. return true;
  460. }
  461. function can_inject_symbols() {
  462. var block_scoped = new Set();
  463. do {
  464. scope = compressor.parent(++level);
  465. if (scope.is_block_scope() && scope.block_scope) {
  466. // TODO this is sometimes undefined during compression.
  467. // But it should always have a value!
  468. scope.block_scope.variables.forEach(function (variable) {
  469. block_scoped.add(variable.name);
  470. });
  471. }
  472. if (scope instanceof AST_Catch) {
  473. // TODO can we delete? AST_Catch is a block scope.
  474. if (scope.argname) {
  475. block_scoped.add(scope.argname.name);
  476. }
  477. } else if (scope instanceof AST_IterationStatement) {
  478. in_loop = [];
  479. } else if (scope instanceof AST_SymbolRef) {
  480. if (scope.fixed_value() instanceof AST_Scope) return false;
  481. }
  482. } while (!(scope instanceof AST_Scope));
  483. var safe_to_inject = !(scope instanceof AST_Toplevel) || compressor.toplevel.vars;
  484. var inline = compressor.option("inline");
  485. if (!can_inject_vars(block_scoped, inline >= 3 && safe_to_inject)) return false;
  486. if (!can_inject_args(block_scoped, inline >= 2 && safe_to_inject)) return false;
  487. return !in_loop || in_loop.length == 0 || !is_reachable(fn, in_loop);
  488. }
  489. function append_var(decls, expressions, name, value) {
  490. var def = name.definition();
  491. // Name already exists, only when a function argument had the same name
  492. const already_appended = scope.variables.has(name.name);
  493. if (!already_appended) {
  494. scope.variables.set(name.name, def);
  495. scope.enclosed.push(def);
  496. decls.push(make_node(AST_VarDef, name, {
  497. name: name,
  498. value: null
  499. }));
  500. }
  501. var sym = make_node(AST_SymbolRef, name, name);
  502. def.references.push(sym);
  503. if (value) expressions.push(make_node(AST_Assign, self, {
  504. operator: "=",
  505. logical: false,
  506. left: sym,
  507. right: value.clone()
  508. }));
  509. }
  510. function flatten_args(decls, expressions) {
  511. var len = fn.argnames.length;
  512. for (var i = self.args.length; --i >= len;) {
  513. expressions.push(self.args[i]);
  514. }
  515. for (i = len; --i >= 0;) {
  516. var name = fn.argnames[i];
  517. var value = self.args[i];
  518. if (has_flag(name, UNUSED) || !name.name || scope.conflicting_def(name.name)) {
  519. if (value) expressions.push(value);
  520. } else {
  521. var symbol = make_node(AST_SymbolVar, name, name);
  522. name.definition().orig.push(symbol);
  523. if (!value && in_loop) value = make_node(AST_Undefined, self);
  524. append_var(decls, expressions, symbol, value);
  525. }
  526. }
  527. decls.reverse();
  528. expressions.reverse();
  529. }
  530. function flatten_vars(decls, expressions) {
  531. var pos = expressions.length;
  532. for (var i = 0, lines = fn.body.length; i < lines; i++) {
  533. var stat = fn.body[i];
  534. if (!(stat instanceof AST_Var)) continue;
  535. for (var j = 0, defs = stat.definitions.length; j < defs; j++) {
  536. var var_def = stat.definitions[j];
  537. var name = var_def.name;
  538. append_var(decls, expressions, name, var_def.value);
  539. if (in_loop && fn.argnames.every((argname) =>
  540. argname.name != name.name
  541. )) {
  542. var def = fn.variables.get(name.name);
  543. var sym = make_node(AST_SymbolRef, name, name);
  544. def.references.push(sym);
  545. expressions.splice(pos++, 0, make_node(AST_Assign, var_def, {
  546. operator: "=",
  547. logical: false,
  548. left: sym,
  549. right: make_node(AST_Undefined, name)
  550. }));
  551. }
  552. }
  553. }
  554. }
  555. function flatten_fn(returned_value) {
  556. var decls = [];
  557. var expressions = [];
  558. flatten_args(decls, expressions);
  559. flatten_vars(decls, expressions);
  560. expressions.push(returned_value);
  561. if (decls.length) {
  562. const i = scope.body.indexOf(compressor.parent(level - 1)) + 1;
  563. scope.body.splice(i, 0, make_node(AST_Var, fn, {
  564. definitions: decls
  565. }));
  566. }
  567. return expressions.map(exp => exp.clone(true));
  568. }
  569. }