tighten-body.js 59 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516
  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_Arrow,
  36. AST_Assign,
  37. AST_Await,
  38. AST_Binary,
  39. AST_Block,
  40. AST_BlockStatement,
  41. AST_Break,
  42. AST_Call,
  43. AST_Case,
  44. AST_Chain,
  45. AST_Class,
  46. AST_Conditional,
  47. AST_Const,
  48. AST_Constant,
  49. AST_Continue,
  50. AST_Debugger,
  51. AST_Default,
  52. AST_Definitions,
  53. AST_Defun,
  54. AST_Destructuring,
  55. AST_Directive,
  56. AST_Dot,
  57. AST_DWLoop,
  58. AST_EmptyStatement,
  59. AST_Exit,
  60. AST_Expansion,
  61. AST_Export,
  62. AST_For,
  63. AST_ForIn,
  64. AST_If,
  65. AST_Import,
  66. AST_IterationStatement,
  67. AST_Lambda,
  68. AST_Let,
  69. AST_LoopControl,
  70. AST_Node,
  71. AST_Number,
  72. AST_Object,
  73. AST_ObjectKeyVal,
  74. AST_ObjectProperty,
  75. AST_PropAccess,
  76. AST_RegExp,
  77. AST_Return,
  78. AST_Scope,
  79. AST_Sequence,
  80. AST_SimpleStatement,
  81. AST_Sub,
  82. AST_Switch,
  83. AST_Symbol,
  84. AST_SymbolConst,
  85. AST_SymbolDeclaration,
  86. AST_SymbolDefun,
  87. AST_SymbolFunarg,
  88. AST_SymbolLambda,
  89. AST_SymbolLet,
  90. AST_SymbolRef,
  91. AST_SymbolVar,
  92. AST_This,
  93. AST_Try,
  94. AST_TryBlock,
  95. AST_Unary,
  96. AST_UnaryPostfix,
  97. AST_UnaryPrefix,
  98. AST_Undefined,
  99. AST_Var,
  100. AST_VarDef,
  101. AST_With,
  102. AST_Yield,
  103. TreeTransformer,
  104. TreeWalker,
  105. walk,
  106. walk_abort,
  107. _NOINLINE,
  108. } from "../ast.js";
  109. import {
  110. make_node,
  111. MAP,
  112. member,
  113. remove,
  114. has_annotation
  115. } from "../utils/index.js";
  116. import { pure_prop_access_globals } from "./native-objects.js";
  117. import {
  118. lazy_op,
  119. unary_side_effects,
  120. is_modified,
  121. is_lhs,
  122. aborts
  123. } from "./inference.js";
  124. import { WRITE_ONLY, clear_flag } from "./compressor-flags.js";
  125. import {
  126. make_sequence,
  127. merge_sequence,
  128. maintain_this_binding,
  129. is_func_expr,
  130. is_identifier_atom,
  131. is_ref_of,
  132. can_be_evicted_from_block,
  133. as_statement_array,
  134. } from "./common.js";
  135. function loop_body(x) {
  136. if (x instanceof AST_IterationStatement) {
  137. return x.body instanceof AST_BlockStatement ? x.body : x;
  138. }
  139. return x;
  140. }
  141. function is_lhs_read_only(lhs) {
  142. if (lhs instanceof AST_This) return true;
  143. if (lhs instanceof AST_SymbolRef) return lhs.definition().orig[0] instanceof AST_SymbolLambda;
  144. if (lhs instanceof AST_PropAccess) {
  145. lhs = lhs.expression;
  146. if (lhs instanceof AST_SymbolRef) {
  147. if (lhs.is_immutable()) return false;
  148. lhs = lhs.fixed_value();
  149. }
  150. if (!lhs) return true;
  151. if (lhs instanceof AST_RegExp) return false;
  152. if (lhs instanceof AST_Constant) return true;
  153. return is_lhs_read_only(lhs);
  154. }
  155. return false;
  156. }
  157. /** var a = 1 --> var a*/
  158. function remove_initializers(var_statement) {
  159. var decls = [];
  160. var_statement.definitions.forEach(function(def) {
  161. if (def.name instanceof AST_SymbolDeclaration) {
  162. def.value = null;
  163. decls.push(def);
  164. } else {
  165. def.declarations_as_names().forEach(name => {
  166. decls.push(make_node(AST_VarDef, def, {
  167. name,
  168. value: null
  169. }));
  170. });
  171. }
  172. });
  173. return decls.length ? make_node(AST_Var, var_statement, { definitions: decls }) : null;
  174. }
  175. /** Called on code which we know is unreachable, to keep elements that affect outside of it. */
  176. export function trim_unreachable_code(compressor, stat, target) {
  177. walk(stat, node => {
  178. if (node instanceof AST_Var) {
  179. const no_initializers = remove_initializers(node);
  180. if (no_initializers) target.push(no_initializers);
  181. return true;
  182. }
  183. if (
  184. node instanceof AST_Defun
  185. && (node === stat || !compressor.has_directive("use strict"))
  186. ) {
  187. target.push(node === stat ? node : make_node(AST_Var, node, {
  188. definitions: [
  189. make_node(AST_VarDef, node, {
  190. name: make_node(AST_SymbolVar, node.name, node.name),
  191. value: null
  192. })
  193. ]
  194. }));
  195. return true;
  196. }
  197. if (node instanceof AST_Export || node instanceof AST_Import) {
  198. target.push(node);
  199. return true;
  200. }
  201. if (node instanceof AST_Scope) {
  202. return true;
  203. }
  204. });
  205. }
  206. /** Tighten a bunch of statements together, and perform statement-level optimization. */
  207. export function tighten_body(statements, compressor) {
  208. const nearest_scope = compressor.find_scope();
  209. const defun_scope = nearest_scope.get_defun_scope();
  210. const { in_loop, in_try } = find_loop_scope_try();
  211. var CHANGED, max_iter = 10;
  212. do {
  213. CHANGED = false;
  214. eliminate_spurious_blocks(statements);
  215. if (compressor.option("dead_code")) {
  216. eliminate_dead_code(statements, compressor);
  217. }
  218. if (compressor.option("if_return")) {
  219. handle_if_return(statements, compressor);
  220. }
  221. if (compressor.sequences_limit > 0) {
  222. sequencesize(statements, compressor);
  223. sequencesize_2(statements, compressor);
  224. }
  225. if (compressor.option("join_vars")) {
  226. join_consecutive_vars(statements);
  227. }
  228. if (compressor.option("collapse_vars")) {
  229. collapse(statements, compressor);
  230. }
  231. } while (CHANGED && max_iter-- > 0);
  232. function find_loop_scope_try() {
  233. var node = compressor.self(), level = 0, in_loop = false, in_try = false;
  234. do {
  235. if (node instanceof AST_IterationStatement) {
  236. in_loop = true;
  237. } else if (node instanceof AST_Scope) {
  238. break;
  239. } else if (node instanceof AST_TryBlock) {
  240. in_try = true;
  241. }
  242. } while (node = compressor.parent(level++));
  243. return { in_loop, in_try };
  244. }
  245. // Search from right to left for assignment-like expressions:
  246. // - `var a = x;`
  247. // - `a = x;`
  248. // - `++a`
  249. // For each candidate, scan from left to right for first usage, then try
  250. // to fold assignment into the site for compression.
  251. // Will not attempt to collapse assignments into or past code blocks
  252. // which are not sequentially executed, e.g. loops and conditionals.
  253. function collapse(statements, compressor) {
  254. if (nearest_scope.pinned() || defun_scope.pinned())
  255. return statements;
  256. var args;
  257. var candidates = [];
  258. var stat_index = statements.length;
  259. var scanner = new TreeTransformer(function (node) {
  260. if (abort)
  261. return node;
  262. // Skip nodes before `candidate` as quickly as possible
  263. if (!hit) {
  264. if (node !== hit_stack[hit_index])
  265. return node;
  266. hit_index++;
  267. if (hit_index < hit_stack.length)
  268. return handle_custom_scan_order(node);
  269. hit = true;
  270. stop_after = find_stop(node, 0);
  271. if (stop_after === node)
  272. abort = true;
  273. return node;
  274. }
  275. // Stop immediately if these node types are encountered
  276. var parent = scanner.parent();
  277. if (node instanceof AST_Assign
  278. && (node.logical || node.operator != "=" && lhs.equivalent_to(node.left))
  279. || node instanceof AST_Await
  280. || node instanceof AST_Call && lhs instanceof AST_PropAccess && lhs.equivalent_to(node.expression)
  281. ||
  282. (node instanceof AST_Call || node instanceof AST_PropAccess)
  283. && node.optional
  284. || node instanceof AST_Debugger
  285. || node instanceof AST_Destructuring
  286. || node instanceof AST_Expansion
  287. && node.expression instanceof AST_Symbol
  288. && (
  289. node.expression instanceof AST_This
  290. || node.expression.definition().references.length > 1
  291. )
  292. || node instanceof AST_IterationStatement && !(node instanceof AST_For)
  293. || node instanceof AST_LoopControl
  294. || node instanceof AST_Try
  295. || node instanceof AST_With
  296. || node instanceof AST_Yield
  297. || node instanceof AST_Export
  298. || node instanceof AST_Class
  299. || parent instanceof AST_For && node !== parent.init
  300. || !replace_all
  301. && (
  302. node instanceof AST_SymbolRef
  303. && !node.is_declared(compressor)
  304. && !pure_prop_access_globals.has(node)
  305. )
  306. || node instanceof AST_SymbolRef
  307. && parent instanceof AST_Call
  308. && has_annotation(parent, _NOINLINE)
  309. || node instanceof AST_ObjectProperty && node.key instanceof AST_Node
  310. ) {
  311. abort = true;
  312. return node;
  313. }
  314. // Stop only if candidate is found within conditional branches
  315. if (!stop_if_hit && (!lhs_local || !replace_all)
  316. && (parent instanceof AST_Binary && lazy_op.has(parent.operator) && parent.left !== node
  317. || parent instanceof AST_Conditional && parent.condition !== node
  318. || parent instanceof AST_If && parent.condition !== node)) {
  319. stop_if_hit = parent;
  320. }
  321. // Replace variable with assignment when found
  322. if (
  323. can_replace
  324. && !(node instanceof AST_SymbolDeclaration)
  325. && lhs.equivalent_to(node)
  326. && !shadows(scanner.find_scope() || nearest_scope, lvalues)
  327. ) {
  328. if (stop_if_hit) {
  329. abort = true;
  330. return node;
  331. }
  332. if (is_lhs(node, parent)) {
  333. if (value_def)
  334. replaced++;
  335. return node;
  336. } else {
  337. replaced++;
  338. if (value_def && candidate instanceof AST_VarDef)
  339. return node;
  340. }
  341. CHANGED = abort = true;
  342. if (candidate instanceof AST_UnaryPostfix) {
  343. return make_node(AST_UnaryPrefix, candidate, candidate);
  344. }
  345. if (candidate instanceof AST_VarDef) {
  346. var def = candidate.name.definition();
  347. var value = candidate.value;
  348. if (def.references.length - def.replaced == 1 && !compressor.exposed(def)) {
  349. def.replaced++;
  350. if (funarg && is_identifier_atom(value)) {
  351. return value.transform(compressor);
  352. } else {
  353. return maintain_this_binding(parent, node, value);
  354. }
  355. }
  356. return make_node(AST_Assign, candidate, {
  357. operator: "=",
  358. logical: false,
  359. left: make_node(AST_SymbolRef, candidate.name, candidate.name),
  360. right: value
  361. });
  362. }
  363. clear_flag(candidate, WRITE_ONLY);
  364. return candidate;
  365. }
  366. // These node types have child nodes that execute sequentially,
  367. // but are otherwise not safe to scan into or beyond them.
  368. var sym;
  369. if (node instanceof AST_Call
  370. || node instanceof AST_Exit
  371. && (side_effects || lhs instanceof AST_PropAccess || may_modify(lhs))
  372. || node instanceof AST_PropAccess
  373. && (side_effects || node.expression.may_throw_on_access(compressor))
  374. || node instanceof AST_SymbolRef
  375. && ((lvalues.has(node.name) && lvalues.get(node.name).modified) || side_effects && may_modify(node))
  376. || node instanceof AST_VarDef && node.value
  377. && (lvalues.has(node.name.name) || side_effects && may_modify(node.name))
  378. || (sym = is_lhs(node.left, node))
  379. && (sym instanceof AST_PropAccess || lvalues.has(sym.name))
  380. || may_throw
  381. && (in_try ? node.has_side_effects(compressor) : side_effects_external(node))) {
  382. stop_after = node;
  383. if (node instanceof AST_Scope)
  384. abort = true;
  385. }
  386. return handle_custom_scan_order(node);
  387. }, function (node) {
  388. if (abort)
  389. return;
  390. if (stop_after === node)
  391. abort = true;
  392. if (stop_if_hit === node)
  393. stop_if_hit = null;
  394. });
  395. var multi_replacer = new TreeTransformer(function (node) {
  396. if (abort)
  397. return node;
  398. // Skip nodes before `candidate` as quickly as possible
  399. if (!hit) {
  400. if (node !== hit_stack[hit_index])
  401. return node;
  402. hit_index++;
  403. if (hit_index < hit_stack.length)
  404. return;
  405. hit = true;
  406. return node;
  407. }
  408. // Replace variable when found
  409. if (node instanceof AST_SymbolRef
  410. && node.name == def.name) {
  411. if (!--replaced)
  412. abort = true;
  413. if (is_lhs(node, multi_replacer.parent()))
  414. return node;
  415. def.replaced++;
  416. value_def.replaced--;
  417. return candidate.value;
  418. }
  419. // Skip (non-executed) functions and (leading) default case in switch statements
  420. if (node instanceof AST_Default || node instanceof AST_Scope)
  421. return node;
  422. });
  423. while (--stat_index >= 0) {
  424. // Treat parameters as collapsible in IIFE, i.e.
  425. // function(a, b){ ... }(x());
  426. // would be translated into equivalent assignments:
  427. // var a = x(), b = undefined;
  428. if (stat_index == 0 && compressor.option("unused"))
  429. extract_args();
  430. // Find collapsible assignments
  431. var hit_stack = [];
  432. extract_candidates(statements[stat_index]);
  433. while (candidates.length > 0) {
  434. hit_stack = candidates.pop();
  435. var hit_index = 0;
  436. var candidate = hit_stack[hit_stack.length - 1];
  437. var value_def = null;
  438. var stop_after = null;
  439. var stop_if_hit = null;
  440. var lhs = get_lhs(candidate);
  441. if (!lhs || is_lhs_read_only(lhs) || lhs.has_side_effects(compressor))
  442. continue;
  443. // Locate symbols which may execute code outside of scanning range
  444. var lvalues = get_lvalues(candidate);
  445. var lhs_local = is_lhs_local(lhs);
  446. if (lhs instanceof AST_SymbolRef) {
  447. lvalues.set(lhs.name, { def: lhs.definition(), modified: false });
  448. }
  449. var side_effects = value_has_side_effects(candidate);
  450. var replace_all = replace_all_symbols();
  451. var may_throw = candidate.may_throw(compressor);
  452. var funarg = candidate.name instanceof AST_SymbolFunarg;
  453. var hit = funarg;
  454. var abort = false, replaced = 0, can_replace = !args || !hit;
  455. if (!can_replace) {
  456. for (
  457. let j = compressor.self().argnames.lastIndexOf(candidate.name) + 1;
  458. !abort && j < args.length;
  459. j++
  460. ) {
  461. args[j].transform(scanner);
  462. }
  463. can_replace = true;
  464. }
  465. for (var i = stat_index; !abort && i < statements.length; i++) {
  466. statements[i].transform(scanner);
  467. }
  468. if (value_def) {
  469. var def = candidate.name.definition();
  470. if (abort && def.references.length - def.replaced > replaced)
  471. replaced = false;
  472. else {
  473. abort = false;
  474. hit_index = 0;
  475. hit = funarg;
  476. for (var i = stat_index; !abort && i < statements.length; i++) {
  477. statements[i].transform(multi_replacer);
  478. }
  479. value_def.single_use = false;
  480. }
  481. }
  482. if (replaced && !remove_candidate(candidate))
  483. statements.splice(stat_index, 1);
  484. }
  485. }
  486. function handle_custom_scan_order(node) {
  487. // Skip (non-executed) functions
  488. if (node instanceof AST_Scope)
  489. return node;
  490. // Scan case expressions first in a switch statement
  491. if (node instanceof AST_Switch) {
  492. node.expression = node.expression.transform(scanner);
  493. for (var i = 0, len = node.body.length; !abort && i < len; i++) {
  494. var branch = node.body[i];
  495. if (branch instanceof AST_Case) {
  496. if (!hit) {
  497. if (branch !== hit_stack[hit_index])
  498. continue;
  499. hit_index++;
  500. }
  501. branch.expression = branch.expression.transform(scanner);
  502. if (!replace_all)
  503. break;
  504. }
  505. }
  506. abort = true;
  507. return node;
  508. }
  509. }
  510. function redefined_within_scope(def, scope) {
  511. if (def.global)
  512. return false;
  513. let cur_scope = def.scope;
  514. while (cur_scope && cur_scope !== scope) {
  515. if (cur_scope.variables.has(def.name)) {
  516. return true;
  517. }
  518. cur_scope = cur_scope.parent_scope;
  519. }
  520. return false;
  521. }
  522. function has_overlapping_symbol(fn, arg, fn_strict) {
  523. var found = false, scan_this = !(fn instanceof AST_Arrow);
  524. arg.walk(new TreeWalker(function (node, descend) {
  525. if (found)
  526. return true;
  527. if (node instanceof AST_SymbolRef && (fn.variables.has(node.name) || redefined_within_scope(node.definition(), fn))) {
  528. var s = node.definition().scope;
  529. if (s !== defun_scope)
  530. while (s = s.parent_scope) {
  531. if (s === defun_scope)
  532. return true;
  533. }
  534. return found = true;
  535. }
  536. if ((fn_strict || scan_this) && node instanceof AST_This) {
  537. return found = true;
  538. }
  539. if (node instanceof AST_Scope && !(node instanceof AST_Arrow)) {
  540. var prev = scan_this;
  541. scan_this = false;
  542. descend();
  543. scan_this = prev;
  544. return true;
  545. }
  546. }));
  547. return found;
  548. }
  549. function arg_is_injectable(arg) {
  550. if (arg instanceof AST_Expansion) return false;
  551. const contains_await = walk(arg, (node) => {
  552. if (node instanceof AST_Await) return walk_abort;
  553. });
  554. if (contains_await) return false;
  555. return true;
  556. }
  557. function extract_args() {
  558. var iife, fn = compressor.self();
  559. if (is_func_expr(fn)
  560. && !fn.name
  561. && !fn.uses_arguments
  562. && !fn.pinned()
  563. && (iife = compressor.parent()) instanceof AST_Call
  564. && iife.expression === fn
  565. && iife.args.every(arg_is_injectable)
  566. ) {
  567. var fn_strict = compressor.has_directive("use strict");
  568. if (fn_strict && !member(fn_strict, fn.body))
  569. fn_strict = false;
  570. var len = fn.argnames.length;
  571. args = iife.args.slice(len);
  572. var names = new Set();
  573. for (var i = len; --i >= 0;) {
  574. var sym = fn.argnames[i];
  575. var arg = iife.args[i];
  576. // The following two line fix is a duplicate of the fix at
  577. // https://github.com/terser/terser/commit/011d3eb08cefe6922c7d1bdfa113fc4aeaca1b75
  578. // This might mean that these two pieces of code (one here in collapse_vars and another in reduce_vars
  579. // Might be doing the exact same thing.
  580. const def = sym.definition && sym.definition();
  581. const is_reassigned = def && def.orig.length > 1;
  582. if (is_reassigned)
  583. continue;
  584. args.unshift(make_node(AST_VarDef, sym, {
  585. name: sym,
  586. value: arg
  587. }));
  588. if (names.has(sym.name))
  589. continue;
  590. names.add(sym.name);
  591. if (sym instanceof AST_Expansion) {
  592. var elements = iife.args.slice(i);
  593. if (elements.every((arg) => !has_overlapping_symbol(fn, arg, fn_strict)
  594. )) {
  595. candidates.unshift([make_node(AST_VarDef, sym, {
  596. name: sym.expression,
  597. value: make_node(AST_Array, iife, {
  598. elements: elements
  599. })
  600. })]);
  601. }
  602. } else {
  603. if (!arg) {
  604. arg = make_node(AST_Undefined, sym).transform(compressor);
  605. } else if (arg instanceof AST_Lambda && arg.pinned()
  606. || has_overlapping_symbol(fn, arg, fn_strict)) {
  607. arg = null;
  608. }
  609. if (arg)
  610. candidates.unshift([make_node(AST_VarDef, sym, {
  611. name: sym,
  612. value: arg
  613. })]);
  614. }
  615. }
  616. }
  617. }
  618. function extract_candidates(expr) {
  619. hit_stack.push(expr);
  620. if (expr instanceof AST_Assign) {
  621. if (!expr.left.has_side_effects(compressor)
  622. && !(expr.right instanceof AST_Chain)) {
  623. candidates.push(hit_stack.slice());
  624. }
  625. extract_candidates(expr.right);
  626. } else if (expr instanceof AST_Binary) {
  627. extract_candidates(expr.left);
  628. extract_candidates(expr.right);
  629. } else if (expr instanceof AST_Call && !has_annotation(expr, _NOINLINE)) {
  630. extract_candidates(expr.expression);
  631. expr.args.forEach(extract_candidates);
  632. } else if (expr instanceof AST_Case) {
  633. extract_candidates(expr.expression);
  634. } else if (expr instanceof AST_Conditional) {
  635. extract_candidates(expr.condition);
  636. extract_candidates(expr.consequent);
  637. extract_candidates(expr.alternative);
  638. } else if (expr instanceof AST_Definitions) {
  639. var len = expr.definitions.length;
  640. // limit number of trailing variable definitions for consideration
  641. var i = len - 200;
  642. if (i < 0)
  643. i = 0;
  644. for (; i < len; i++) {
  645. extract_candidates(expr.definitions[i]);
  646. }
  647. } else if (expr instanceof AST_DWLoop) {
  648. extract_candidates(expr.condition);
  649. if (!(expr.body instanceof AST_Block)) {
  650. extract_candidates(expr.body);
  651. }
  652. } else if (expr instanceof AST_Exit) {
  653. if (expr.value)
  654. extract_candidates(expr.value);
  655. } else if (expr instanceof AST_For) {
  656. if (expr.init)
  657. extract_candidates(expr.init);
  658. if (expr.condition)
  659. extract_candidates(expr.condition);
  660. if (expr.step)
  661. extract_candidates(expr.step);
  662. if (!(expr.body instanceof AST_Block)) {
  663. extract_candidates(expr.body);
  664. }
  665. } else if (expr instanceof AST_ForIn) {
  666. extract_candidates(expr.object);
  667. if (!(expr.body instanceof AST_Block)) {
  668. extract_candidates(expr.body);
  669. }
  670. } else if (expr instanceof AST_If) {
  671. extract_candidates(expr.condition);
  672. if (!(expr.body instanceof AST_Block)) {
  673. extract_candidates(expr.body);
  674. }
  675. if (expr.alternative && !(expr.alternative instanceof AST_Block)) {
  676. extract_candidates(expr.alternative);
  677. }
  678. } else if (expr instanceof AST_Sequence) {
  679. expr.expressions.forEach(extract_candidates);
  680. } else if (expr instanceof AST_SimpleStatement) {
  681. extract_candidates(expr.body);
  682. } else if (expr instanceof AST_Switch) {
  683. extract_candidates(expr.expression);
  684. expr.body.forEach(extract_candidates);
  685. } else if (expr instanceof AST_Unary) {
  686. if (expr.operator == "++" || expr.operator == "--") {
  687. candidates.push(hit_stack.slice());
  688. }
  689. } else if (expr instanceof AST_VarDef) {
  690. if (expr.value && !(expr.value instanceof AST_Chain)) {
  691. candidates.push(hit_stack.slice());
  692. extract_candidates(expr.value);
  693. }
  694. }
  695. hit_stack.pop();
  696. }
  697. function find_stop(node, level, write_only) {
  698. var parent = scanner.parent(level);
  699. if (parent instanceof AST_Assign) {
  700. if (write_only
  701. && !parent.logical
  702. && !(parent.left instanceof AST_PropAccess
  703. || lvalues.has(parent.left.name))) {
  704. return find_stop(parent, level + 1, write_only);
  705. }
  706. return node;
  707. }
  708. if (parent instanceof AST_Binary) {
  709. if (write_only && (!lazy_op.has(parent.operator) || parent.left === node)) {
  710. return find_stop(parent, level + 1, write_only);
  711. }
  712. return node;
  713. }
  714. if (parent instanceof AST_Call)
  715. return node;
  716. if (parent instanceof AST_Case)
  717. return node;
  718. if (parent instanceof AST_Conditional) {
  719. if (write_only && parent.condition === node) {
  720. return find_stop(parent, level + 1, write_only);
  721. }
  722. return node;
  723. }
  724. if (parent instanceof AST_Definitions) {
  725. return find_stop(parent, level + 1, true);
  726. }
  727. if (parent instanceof AST_Exit) {
  728. return write_only ? find_stop(parent, level + 1, write_only) : node;
  729. }
  730. if (parent instanceof AST_If) {
  731. if (write_only && parent.condition === node) {
  732. return find_stop(parent, level + 1, write_only);
  733. }
  734. return node;
  735. }
  736. if (parent instanceof AST_IterationStatement)
  737. return node;
  738. if (parent instanceof AST_Sequence) {
  739. return find_stop(parent, level + 1, parent.tail_node() !== node);
  740. }
  741. if (parent instanceof AST_SimpleStatement) {
  742. return find_stop(parent, level + 1, true);
  743. }
  744. if (parent instanceof AST_Switch)
  745. return node;
  746. if (parent instanceof AST_VarDef)
  747. return node;
  748. return null;
  749. }
  750. function mangleable_var(var_def) {
  751. var value = var_def.value;
  752. if (!(value instanceof AST_SymbolRef))
  753. return;
  754. if (value.name == "arguments")
  755. return;
  756. var def = value.definition();
  757. if (def.undeclared)
  758. return;
  759. return value_def = def;
  760. }
  761. function get_lhs(expr) {
  762. if (expr instanceof AST_Assign && expr.logical) {
  763. return false;
  764. } else if (expr instanceof AST_VarDef && expr.name instanceof AST_SymbolDeclaration) {
  765. var def = expr.name.definition();
  766. if (!member(expr.name, def.orig))
  767. return;
  768. var referenced = def.references.length - def.replaced;
  769. if (!referenced)
  770. return;
  771. var declared = def.orig.length - def.eliminated;
  772. if (declared > 1 && !(expr.name instanceof AST_SymbolFunarg)
  773. || (referenced > 1 ? mangleable_var(expr) : !compressor.exposed(def))) {
  774. return make_node(AST_SymbolRef, expr.name, expr.name);
  775. }
  776. } else {
  777. const lhs = expr instanceof AST_Assign
  778. ? expr.left
  779. : expr.expression;
  780. return !is_ref_of(lhs, AST_SymbolConst)
  781. && !is_ref_of(lhs, AST_SymbolLet) && lhs;
  782. }
  783. }
  784. function get_rvalue(expr) {
  785. if (expr instanceof AST_Assign) {
  786. return expr.right;
  787. } else {
  788. return expr.value;
  789. }
  790. }
  791. function get_lvalues(expr) {
  792. var lvalues = new Map();
  793. if (expr instanceof AST_Unary)
  794. return lvalues;
  795. var tw = new TreeWalker(function (node) {
  796. var sym = node;
  797. while (sym instanceof AST_PropAccess)
  798. sym = sym.expression;
  799. if (sym instanceof AST_SymbolRef) {
  800. const prev = lvalues.get(sym.name);
  801. if (!prev || !prev.modified) {
  802. lvalues.set(sym.name, {
  803. def: sym.definition(),
  804. modified: is_modified(compressor, tw, node, node, 0)
  805. });
  806. }
  807. }
  808. });
  809. get_rvalue(expr).walk(tw);
  810. return lvalues;
  811. }
  812. function remove_candidate(expr) {
  813. if (expr.name instanceof AST_SymbolFunarg) {
  814. var iife = compressor.parent(), argnames = compressor.self().argnames;
  815. var index = argnames.indexOf(expr.name);
  816. if (index < 0) {
  817. iife.args.length = Math.min(iife.args.length, argnames.length - 1);
  818. } else {
  819. var args = iife.args;
  820. if (args[index])
  821. args[index] = make_node(AST_Number, args[index], {
  822. value: 0
  823. });
  824. }
  825. return true;
  826. }
  827. var found = false;
  828. return statements[stat_index].transform(new TreeTransformer(function (node, descend, in_list) {
  829. if (found)
  830. return node;
  831. if (node === expr || node.body === expr) {
  832. found = true;
  833. if (node instanceof AST_VarDef) {
  834. node.value = node.name instanceof AST_SymbolConst
  835. ? make_node(AST_Undefined, node.value) // `const` always needs value.
  836. : null;
  837. return node;
  838. }
  839. return in_list ? MAP.skip : null;
  840. }
  841. }, function (node) {
  842. if (node instanceof AST_Sequence)
  843. switch (node.expressions.length) {
  844. case 0: return null;
  845. case 1: return node.expressions[0];
  846. }
  847. }));
  848. }
  849. function is_lhs_local(lhs) {
  850. while (lhs instanceof AST_PropAccess)
  851. lhs = lhs.expression;
  852. return lhs instanceof AST_SymbolRef
  853. && lhs.definition().scope.get_defun_scope() === defun_scope
  854. && !(in_loop
  855. && (lvalues.has(lhs.name)
  856. || candidate instanceof AST_Unary
  857. || (candidate instanceof AST_Assign
  858. && !candidate.logical
  859. && candidate.operator != "=")));
  860. }
  861. function value_has_side_effects(expr) {
  862. if (expr instanceof AST_Unary)
  863. return unary_side_effects.has(expr.operator);
  864. return get_rvalue(expr).has_side_effects(compressor);
  865. }
  866. function replace_all_symbols() {
  867. if (side_effects)
  868. return false;
  869. if (value_def)
  870. return true;
  871. if (lhs instanceof AST_SymbolRef) {
  872. var def = lhs.definition();
  873. if (def.references.length - def.replaced == (candidate instanceof AST_VarDef ? 1 : 2)) {
  874. return true;
  875. }
  876. }
  877. return false;
  878. }
  879. function may_modify(sym) {
  880. if (!sym.definition)
  881. return true; // AST_Destructuring
  882. var def = sym.definition();
  883. if (def.orig.length == 1 && def.orig[0] instanceof AST_SymbolDefun)
  884. return false;
  885. if (def.scope.get_defun_scope() !== defun_scope)
  886. return true;
  887. return def.references.some((ref) =>
  888. ref.scope.get_defun_scope() !== defun_scope
  889. );
  890. }
  891. function side_effects_external(node, lhs) {
  892. if (node instanceof AST_Assign)
  893. return side_effects_external(node.left, true);
  894. if (node instanceof AST_Unary)
  895. return side_effects_external(node.expression, true);
  896. if (node instanceof AST_VarDef)
  897. return node.value && side_effects_external(node.value);
  898. if (lhs) {
  899. if (node instanceof AST_Dot)
  900. return side_effects_external(node.expression, true);
  901. if (node instanceof AST_Sub)
  902. return side_effects_external(node.expression, true);
  903. if (node instanceof AST_SymbolRef)
  904. return node.definition().scope.get_defun_scope() !== defun_scope;
  905. }
  906. return false;
  907. }
  908. /**
  909. * Will any of the pulled-in lvalues shadow a variable in newScope or parents?
  910. * similar to scope_encloses_variables_in_this_scope */
  911. function shadows(my_scope, lvalues) {
  912. for (const { def } of lvalues.values()) {
  913. const looked_up = my_scope.find_variable(def.name);
  914. if (looked_up) {
  915. if (looked_up === def) continue;
  916. return true;
  917. }
  918. }
  919. return false;
  920. }
  921. }
  922. function eliminate_spurious_blocks(statements) {
  923. var seen_dirs = [];
  924. for (var i = 0; i < statements.length;) {
  925. var stat = statements[i];
  926. if (stat instanceof AST_BlockStatement && stat.body.every(can_be_evicted_from_block)) {
  927. CHANGED = true;
  928. eliminate_spurious_blocks(stat.body);
  929. statements.splice(i, 1, ...stat.body);
  930. i += stat.body.length;
  931. } else if (stat instanceof AST_EmptyStatement) {
  932. CHANGED = true;
  933. statements.splice(i, 1);
  934. } else if (stat instanceof AST_Directive) {
  935. if (seen_dirs.indexOf(stat.value) < 0) {
  936. i++;
  937. seen_dirs.push(stat.value);
  938. } else {
  939. CHANGED = true;
  940. statements.splice(i, 1);
  941. }
  942. } else
  943. i++;
  944. }
  945. }
  946. function handle_if_return(statements, compressor) {
  947. var self = compressor.self();
  948. var multiple_if_returns = has_multiple_if_returns(statements);
  949. var in_lambda = self instanceof AST_Lambda;
  950. // Prevent extremely deep nesting
  951. // https://github.com/terser/terser/issues/1432
  952. // https://github.com/webpack/webpack/issues/17548
  953. const iteration_start = Math.min(statements.length, 500);
  954. for (var i = iteration_start; --i >= 0;) {
  955. var stat = statements[i];
  956. var j = next_index(i);
  957. var next = statements[j];
  958. if (in_lambda && !next && stat instanceof AST_Return) {
  959. if (!stat.value) {
  960. CHANGED = true;
  961. statements.splice(i, 1);
  962. continue;
  963. }
  964. if (stat.value instanceof AST_UnaryPrefix && stat.value.operator == "void") {
  965. CHANGED = true;
  966. statements[i] = make_node(AST_SimpleStatement, stat, {
  967. body: stat.value.expression
  968. });
  969. continue;
  970. }
  971. }
  972. if (stat instanceof AST_If) {
  973. let ab, new_else;
  974. ab = aborts(stat.body);
  975. if (
  976. can_merge_flow(ab)
  977. && (new_else = as_statement_array_with_return(stat.body, ab))
  978. ) {
  979. if (ab.label) {
  980. remove(ab.label.thedef.references, ab);
  981. }
  982. CHANGED = true;
  983. stat = stat.clone();
  984. stat.condition = stat.condition.negate(compressor);
  985. stat.body = make_node(AST_BlockStatement, stat, {
  986. body: as_statement_array(stat.alternative).concat(extract_functions())
  987. });
  988. stat.alternative = make_node(AST_BlockStatement, stat, {
  989. body: new_else
  990. });
  991. statements[i] = stat.transform(compressor);
  992. continue;
  993. }
  994. ab = aborts(stat.alternative);
  995. if (
  996. can_merge_flow(ab)
  997. && (new_else = as_statement_array_with_return(stat.alternative, ab))
  998. ) {
  999. if (ab.label) {
  1000. remove(ab.label.thedef.references, ab);
  1001. }
  1002. CHANGED = true;
  1003. stat = stat.clone();
  1004. stat.body = make_node(AST_BlockStatement, stat.body, {
  1005. body: as_statement_array(stat.body).concat(extract_functions())
  1006. });
  1007. stat.alternative = make_node(AST_BlockStatement, stat.alternative, {
  1008. body: new_else
  1009. });
  1010. statements[i] = stat.transform(compressor);
  1011. continue;
  1012. }
  1013. }
  1014. if (stat instanceof AST_If && stat.body instanceof AST_Return) {
  1015. var value = stat.body.value;
  1016. //---
  1017. // pretty silly case, but:
  1018. // if (foo()) return; return; ==> foo(); return;
  1019. if (!value && !stat.alternative
  1020. && (in_lambda && !next || next instanceof AST_Return && !next.value)) {
  1021. CHANGED = true;
  1022. statements[i] = make_node(AST_SimpleStatement, stat.condition, {
  1023. body: stat.condition
  1024. });
  1025. continue;
  1026. }
  1027. //---
  1028. // if (foo()) return x; return y; ==> return foo() ? x : y;
  1029. if (value && !stat.alternative && next instanceof AST_Return && next.value) {
  1030. CHANGED = true;
  1031. stat = stat.clone();
  1032. stat.alternative = next;
  1033. statements[i] = stat.transform(compressor);
  1034. statements.splice(j, 1);
  1035. continue;
  1036. }
  1037. //---
  1038. // if (foo()) return x; [ return ; ] ==> return foo() ? x : undefined;
  1039. if (value && !stat.alternative
  1040. && (!next && in_lambda && multiple_if_returns
  1041. || next instanceof AST_Return)) {
  1042. CHANGED = true;
  1043. stat = stat.clone();
  1044. stat.alternative = next || make_node(AST_Return, stat, {
  1045. value: null
  1046. });
  1047. statements[i] = stat.transform(compressor);
  1048. if (next)
  1049. statements.splice(j, 1);
  1050. continue;
  1051. }
  1052. //---
  1053. // if (a) return b; if (c) return d; e; ==> return a ? b : c ? d : void e;
  1054. //
  1055. // if sequences is not enabled, this can lead to an endless loop (issue #866).
  1056. // however, with sequences on this helps producing slightly better output for
  1057. // the example code.
  1058. var prev = statements[prev_index(i)];
  1059. if (compressor.option("sequences") && in_lambda && !stat.alternative
  1060. && prev instanceof AST_If && prev.body instanceof AST_Return
  1061. && next_index(j) == statements.length && next instanceof AST_SimpleStatement) {
  1062. CHANGED = true;
  1063. stat = stat.clone();
  1064. stat.alternative = make_node(AST_BlockStatement, next, {
  1065. body: [
  1066. next,
  1067. make_node(AST_Return, next, {
  1068. value: null
  1069. })
  1070. ]
  1071. });
  1072. statements[i] = stat.transform(compressor);
  1073. statements.splice(j, 1);
  1074. continue;
  1075. }
  1076. }
  1077. }
  1078. function has_multiple_if_returns(statements) {
  1079. var n = 0;
  1080. for (var i = statements.length; --i >= 0;) {
  1081. var stat = statements[i];
  1082. if (stat instanceof AST_If && stat.body instanceof AST_Return) {
  1083. if (++n > 1)
  1084. return true;
  1085. }
  1086. }
  1087. return false;
  1088. }
  1089. function is_return_void(value) {
  1090. return !value || value instanceof AST_UnaryPrefix && value.operator == "void";
  1091. }
  1092. function can_merge_flow(ab) {
  1093. if (!ab)
  1094. return false;
  1095. for (var j = i + 1, len = statements.length; j < len; j++) {
  1096. var stat = statements[j];
  1097. if (stat instanceof AST_Const || stat instanceof AST_Let)
  1098. return false;
  1099. }
  1100. var lct = ab instanceof AST_LoopControl ? compressor.loopcontrol_target(ab) : null;
  1101. return ab instanceof AST_Return && in_lambda && is_return_void(ab.value)
  1102. || ab instanceof AST_Continue && self === loop_body(lct)
  1103. || ab instanceof AST_Break && lct instanceof AST_BlockStatement && self === lct;
  1104. }
  1105. function extract_functions() {
  1106. var tail = statements.slice(i + 1);
  1107. statements.length = i + 1;
  1108. return tail.filter(function (stat) {
  1109. if (stat instanceof AST_Defun) {
  1110. statements.push(stat);
  1111. return false;
  1112. }
  1113. return true;
  1114. });
  1115. }
  1116. function as_statement_array_with_return(node, ab) {
  1117. var body = as_statement_array(node);
  1118. if (ab !== body[body.length - 1]) {
  1119. return undefined;
  1120. }
  1121. body = body.slice(0, -1);
  1122. if (ab.value) {
  1123. body.push(make_node(AST_SimpleStatement, ab.value, {
  1124. body: ab.value.expression
  1125. }));
  1126. }
  1127. return body;
  1128. }
  1129. function next_index(i) {
  1130. for (var j = i + 1, len = statements.length; j < len; j++) {
  1131. var stat = statements[j];
  1132. if (!(stat instanceof AST_Var && declarations_only(stat))) {
  1133. break;
  1134. }
  1135. }
  1136. return j;
  1137. }
  1138. function prev_index(i) {
  1139. for (var j = i; --j >= 0;) {
  1140. var stat = statements[j];
  1141. if (!(stat instanceof AST_Var && declarations_only(stat))) {
  1142. break;
  1143. }
  1144. }
  1145. return j;
  1146. }
  1147. }
  1148. function eliminate_dead_code(statements, compressor) {
  1149. var has_quit;
  1150. var self = compressor.self();
  1151. for (var i = 0, n = 0, len = statements.length; i < len; i++) {
  1152. var stat = statements[i];
  1153. if (stat instanceof AST_LoopControl) {
  1154. var lct = compressor.loopcontrol_target(stat);
  1155. if (stat instanceof AST_Break
  1156. && !(lct instanceof AST_IterationStatement)
  1157. && loop_body(lct) === self
  1158. || stat instanceof AST_Continue
  1159. && loop_body(lct) === self) {
  1160. if (stat.label) {
  1161. remove(stat.label.thedef.references, stat);
  1162. }
  1163. } else {
  1164. statements[n++] = stat;
  1165. }
  1166. } else {
  1167. statements[n++] = stat;
  1168. }
  1169. if (aborts(stat)) {
  1170. has_quit = statements.slice(i + 1);
  1171. break;
  1172. }
  1173. }
  1174. statements.length = n;
  1175. CHANGED = n != len;
  1176. if (has_quit)
  1177. has_quit.forEach(function (stat) {
  1178. trim_unreachable_code(compressor, stat, statements);
  1179. });
  1180. }
  1181. function declarations_only(node) {
  1182. return node.definitions.every((var_def) => !var_def.value);
  1183. }
  1184. function sequencesize(statements, compressor) {
  1185. if (statements.length < 2)
  1186. return;
  1187. var seq = [], n = 0;
  1188. function push_seq() {
  1189. if (!seq.length)
  1190. return;
  1191. var body = make_sequence(seq[0], seq);
  1192. statements[n++] = make_node(AST_SimpleStatement, body, { body: body });
  1193. seq = [];
  1194. }
  1195. for (var i = 0, len = statements.length; i < len; i++) {
  1196. var stat = statements[i];
  1197. if (stat instanceof AST_SimpleStatement) {
  1198. if (seq.length >= compressor.sequences_limit)
  1199. push_seq();
  1200. var body = stat.body;
  1201. if (seq.length > 0)
  1202. body = body.drop_side_effect_free(compressor);
  1203. if (body)
  1204. merge_sequence(seq, body);
  1205. } else if (stat instanceof AST_Definitions && declarations_only(stat)
  1206. || stat instanceof AST_Defun) {
  1207. statements[n++] = stat;
  1208. } else {
  1209. push_seq();
  1210. statements[n++] = stat;
  1211. }
  1212. }
  1213. push_seq();
  1214. statements.length = n;
  1215. if (n != len)
  1216. CHANGED = true;
  1217. }
  1218. function to_simple_statement(block, decls) {
  1219. if (!(block instanceof AST_BlockStatement))
  1220. return block;
  1221. var stat = null;
  1222. for (var i = 0, len = block.body.length; i < len; i++) {
  1223. var line = block.body[i];
  1224. if (line instanceof AST_Var && declarations_only(line)) {
  1225. decls.push(line);
  1226. } else if (stat || line instanceof AST_Const || line instanceof AST_Let) {
  1227. return false;
  1228. } else {
  1229. stat = line;
  1230. }
  1231. }
  1232. return stat;
  1233. }
  1234. function sequencesize_2(statements, compressor) {
  1235. function cons_seq(right) {
  1236. n--;
  1237. CHANGED = true;
  1238. var left = prev.body;
  1239. return make_sequence(left, [left, right]).transform(compressor);
  1240. }
  1241. var n = 0, prev;
  1242. for (var i = 0; i < statements.length; i++) {
  1243. var stat = statements[i];
  1244. if (prev) {
  1245. if (stat instanceof AST_Exit) {
  1246. stat.value = cons_seq(stat.value || make_node(AST_Undefined, stat).transform(compressor));
  1247. } else if (stat instanceof AST_For) {
  1248. if (!(stat.init instanceof AST_Definitions)) {
  1249. const abort = walk(prev.body, node => {
  1250. if (node instanceof AST_Scope)
  1251. return true;
  1252. if (node instanceof AST_Binary
  1253. && node.operator === "in") {
  1254. return walk_abort;
  1255. }
  1256. });
  1257. if (!abort) {
  1258. if (stat.init)
  1259. stat.init = cons_seq(stat.init);
  1260. else {
  1261. stat.init = prev.body;
  1262. n--;
  1263. CHANGED = true;
  1264. }
  1265. }
  1266. }
  1267. } else if (stat instanceof AST_ForIn) {
  1268. if (!(stat.init instanceof AST_Const) && !(stat.init instanceof AST_Let)) {
  1269. stat.object = cons_seq(stat.object);
  1270. }
  1271. } else if (stat instanceof AST_If) {
  1272. stat.condition = cons_seq(stat.condition);
  1273. } else if (stat instanceof AST_Switch) {
  1274. stat.expression = cons_seq(stat.expression);
  1275. } else if (stat instanceof AST_With) {
  1276. stat.expression = cons_seq(stat.expression);
  1277. }
  1278. }
  1279. if (compressor.option("conditionals") && stat instanceof AST_If) {
  1280. var decls = [];
  1281. var body = to_simple_statement(stat.body, decls);
  1282. var alt = to_simple_statement(stat.alternative, decls);
  1283. if (body !== false && alt !== false && decls.length > 0) {
  1284. var len = decls.length;
  1285. decls.push(make_node(AST_If, stat, {
  1286. condition: stat.condition,
  1287. body: body || make_node(AST_EmptyStatement, stat.body),
  1288. alternative: alt
  1289. }));
  1290. decls.unshift(n, 1);
  1291. [].splice.apply(statements, decls);
  1292. i += len;
  1293. n += len + 1;
  1294. prev = null;
  1295. CHANGED = true;
  1296. continue;
  1297. }
  1298. }
  1299. statements[n++] = stat;
  1300. prev = stat instanceof AST_SimpleStatement ? stat : null;
  1301. }
  1302. statements.length = n;
  1303. }
  1304. function join_object_assignments(defn, body) {
  1305. if (!(defn instanceof AST_Definitions))
  1306. return;
  1307. var def = defn.definitions[defn.definitions.length - 1];
  1308. if (!(def.value instanceof AST_Object))
  1309. return;
  1310. var exprs;
  1311. if (body instanceof AST_Assign && !body.logical) {
  1312. exprs = [body];
  1313. } else if (body instanceof AST_Sequence) {
  1314. exprs = body.expressions.slice();
  1315. }
  1316. if (!exprs)
  1317. return;
  1318. var trimmed = false;
  1319. do {
  1320. var node = exprs[0];
  1321. if (!(node instanceof AST_Assign))
  1322. break;
  1323. if (node.operator != "=")
  1324. break;
  1325. if (!(node.left instanceof AST_PropAccess))
  1326. break;
  1327. var sym = node.left.expression;
  1328. if (!(sym instanceof AST_SymbolRef))
  1329. break;
  1330. if (def.name.name != sym.name)
  1331. break;
  1332. if (!node.right.is_constant_expression(nearest_scope))
  1333. break;
  1334. var prop = node.left.property;
  1335. if (prop instanceof AST_Node) {
  1336. prop = prop.evaluate(compressor);
  1337. }
  1338. if (prop instanceof AST_Node)
  1339. break;
  1340. prop = "" + prop;
  1341. var diff = compressor.option("ecma") < 2015
  1342. && compressor.has_directive("use strict") ? function (node) {
  1343. return node.key != prop && (node.key && node.key.name != prop);
  1344. } : function (node) {
  1345. return node.key && node.key.name != prop;
  1346. };
  1347. if (!def.value.properties.every(diff))
  1348. break;
  1349. var p = def.value.properties.filter(function (p) { return p.key === prop; })[0];
  1350. if (!p) {
  1351. def.value.properties.push(make_node(AST_ObjectKeyVal, node, {
  1352. key: prop,
  1353. value: node.right
  1354. }));
  1355. } else {
  1356. p.value = new AST_Sequence({
  1357. start: p.start,
  1358. expressions: [p.value.clone(), node.right.clone()],
  1359. end: p.end
  1360. });
  1361. }
  1362. exprs.shift();
  1363. trimmed = true;
  1364. } while (exprs.length);
  1365. return trimmed && exprs;
  1366. }
  1367. function join_consecutive_vars(statements) {
  1368. var defs;
  1369. for (var i = 0, j = -1, len = statements.length; i < len; i++) {
  1370. var stat = statements[i];
  1371. var prev = statements[j];
  1372. if (stat instanceof AST_Definitions) {
  1373. if (prev && prev.TYPE == stat.TYPE) {
  1374. prev.definitions = prev.definitions.concat(stat.definitions);
  1375. CHANGED = true;
  1376. } else if (defs && defs.TYPE == stat.TYPE && declarations_only(stat)) {
  1377. defs.definitions = defs.definitions.concat(stat.definitions);
  1378. CHANGED = true;
  1379. } else {
  1380. statements[++j] = stat;
  1381. defs = stat;
  1382. }
  1383. } else if (stat instanceof AST_Exit) {
  1384. stat.value = extract_object_assignments(stat.value);
  1385. } else if (stat instanceof AST_For) {
  1386. var exprs = join_object_assignments(prev, stat.init);
  1387. if (exprs) {
  1388. CHANGED = true;
  1389. stat.init = exprs.length ? make_sequence(stat.init, exprs) : null;
  1390. statements[++j] = stat;
  1391. } else if (
  1392. prev instanceof AST_Var
  1393. && (!stat.init || stat.init.TYPE == prev.TYPE)
  1394. ) {
  1395. if (stat.init) {
  1396. prev.definitions = prev.definitions.concat(stat.init.definitions);
  1397. }
  1398. stat.init = prev;
  1399. statements[j] = stat;
  1400. CHANGED = true;
  1401. } else if (
  1402. defs instanceof AST_Var
  1403. && stat.init instanceof AST_Var
  1404. && declarations_only(stat.init)
  1405. ) {
  1406. defs.definitions = defs.definitions.concat(stat.init.definitions);
  1407. stat.init = null;
  1408. statements[++j] = stat;
  1409. CHANGED = true;
  1410. } else {
  1411. statements[++j] = stat;
  1412. }
  1413. } else if (stat instanceof AST_ForIn) {
  1414. stat.object = extract_object_assignments(stat.object);
  1415. } else if (stat instanceof AST_If) {
  1416. stat.condition = extract_object_assignments(stat.condition);
  1417. } else if (stat instanceof AST_SimpleStatement) {
  1418. var exprs = join_object_assignments(prev, stat.body);
  1419. if (exprs) {
  1420. CHANGED = true;
  1421. if (!exprs.length)
  1422. continue;
  1423. stat.body = make_sequence(stat.body, exprs);
  1424. }
  1425. statements[++j] = stat;
  1426. } else if (stat instanceof AST_Switch) {
  1427. stat.expression = extract_object_assignments(stat.expression);
  1428. } else if (stat instanceof AST_With) {
  1429. stat.expression = extract_object_assignments(stat.expression);
  1430. } else {
  1431. statements[++j] = stat;
  1432. }
  1433. }
  1434. statements.length = j + 1;
  1435. function extract_object_assignments(value) {
  1436. statements[++j] = stat;
  1437. var exprs = join_object_assignments(prev, value);
  1438. if (exprs) {
  1439. CHANGED = true;
  1440. if (exprs.length) {
  1441. return make_sequence(value, exprs);
  1442. } else if (value instanceof AST_Sequence) {
  1443. return value.tail_node().left;
  1444. } else {
  1445. return value.left;
  1446. }
  1447. }
  1448. return value;
  1449. }
  1450. }
  1451. }