From 012046b8b158bcdecfd3ed6af854fd4441003a1a Mon Sep 17 00:00:00 2001 From: alexlamsl Date: Tue, 7 Nov 2017 13:47:49 +0800 Subject: [PATCH] preserve function identity in `reduce_vars` fixes #2450 --- lib/compress.js | 35 +++++++++-------- test/compress/reduce_vars.js | 75 ++++++++++++++++++++++++++++++++++++ 2 files changed, 95 insertions(+), 15 deletions(-) diff --git a/lib/compress.js b/lib/compress.js index 274ab604..cd8b9414 100644 --- a/lib/compress.js +++ b/lib/compress.js @@ -623,7 +623,8 @@ merge(Compressor.prototype, { function mark_escaped(d, node, value, level) { var parent = tw.parent(level); - if (value instanceof AST_Constant || value instanceof AST_Function) return; + if (value instanceof AST_Constant) return; + if (level > 0 && value instanceof AST_Function) return; if (parent instanceof AST_Assign && parent.operator == "=" && node === parent.right || parent instanceof AST_Call && node !== parent.expression || parent instanceof AST_Return && node === parent.value && node.scope !== d.scope @@ -4216,6 +4217,17 @@ merge(Compressor.prototype, { return self; }); + function recursive_ref(compressor, def) { + var node; + for (var i = 0; node = compressor.parent(i); i++) { + if (node instanceof AST_Lambda) { + var name = node.name; + if (name && name.definition() === def) break; + } + } + return node; + } + OPT(AST_SymbolRef, function(self, compressor){ var def = self.resolve_defines(compressor); if (def) { @@ -4241,20 +4253,13 @@ merge(Compressor.prototype, { if (fixed instanceof AST_Defun) { d.fixed = fixed = make_node(AST_Function, fixed, fixed); } - if (fixed && d.single_use) { - var recurse; - if (fixed instanceof AST_Function) { - for (var i = 0; recurse = compressor.parent(i); i++) { - if (recurse instanceof AST_Lambda) { - var name = recurse.name; - if (name && name.definition() === d) break; - } - } - } - if (!recurse) { - var value = fixed.optimize(compressor); - return value === fixed ? fixed.clone(true) : value; - } + if (fixed + && d.single_use + && !(fixed instanceof AST_Function + && (d.escaped && d.scope !== self.scope + || recursive_ref(compressor, d)))) { + var value = fixed.optimize(compressor); + return value === fixed ? fixed.clone(true) : value; } if (fixed && d.should_replace === undefined) { var init; diff --git a/test/compress/reduce_vars.js b/test/compress/reduce_vars.js index f1a27ff9..a005044f 100644 --- a/test/compress/reduce_vars.js +++ b/test/compress/reduce_vars.js @@ -3845,3 +3845,78 @@ recursive_inlining_5: { "foo 0", ] } + +issue_2450_1: { + options = { + reduce_vars: true, + toplevel: true, + unused: true, + } + input: { + function f() {} + function g() { + return f; + } + console.log(g() === g()); + } + expect: { + function f() {} + function g() { + return f; + } + console.log(g() === g()); + } + expect_stdout: "true" +} + +issue_2450_2: { + options = { + reduce_vars: true, + toplevel: true, + unused: true, + } + input: { + function g() { + function f() {} + return f; + } + console.log(g() === g()); + } + expect: { + function g() { + return function() {}; + } + console.log(g() === g()); + } + expect_stdout: "false" +} + +issue_2450_3: { + options = { + reduce_vars: true, + unused: true, + } + input: { + var x = (function() { + function test() { + return "foo"; + } + return function b() { + return [1, test]; + } + })(); + console.log(x()[1] === x()[1]); + } + expect: { + var x = (function() { + function test() { + return "foo"; + } + return function() { + return [1, test]; + } + })(); + console.log(x()[1] === x()[1]); + } + expect_stdout: "true" +}