enhance evaluate & reduce_vars (#3873)
This commit is contained in:
@@ -492,8 +492,7 @@ merge(Compressor.prototype, {
|
|||||||
function ref_once(tw, compressor, def) {
|
function ref_once(tw, compressor, def) {
|
||||||
return compressor.option("unused")
|
return compressor.option("unused")
|
||||||
&& !def.scope.pinned()
|
&& !def.scope.pinned()
|
||||||
&& def.references.length - def.recursive_refs == 1
|
&& def.references.length - def.recursive_refs == 1;
|
||||||
&& tw.loop_ids[def.id] === tw.in_loop;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function is_immutable(value) {
|
function is_immutable(value) {
|
||||||
@@ -797,8 +796,9 @@ merge(Compressor.prototype, {
|
|||||||
if (recursive) {
|
if (recursive) {
|
||||||
d.recursive_refs++;
|
d.recursive_refs++;
|
||||||
} else if (value && ref_once(tw, compressor, d)) {
|
} else if (value && ref_once(tw, compressor, d)) {
|
||||||
|
d.in_loop = tw.loop_ids[d.id] !== tw.in_loop;
|
||||||
d.single_use = value instanceof AST_Lambda && !value.pinned()
|
d.single_use = value instanceof AST_Lambda && !value.pinned()
|
||||||
|| d.scope === this.scope && value.is_constant_expression();
|
|| !d.in_loop && d.scope === this.scope && value.is_constant_expression();
|
||||||
} else {
|
} else {
|
||||||
d.single_use = false;
|
d.single_use = false;
|
||||||
}
|
}
|
||||||
@@ -3501,7 +3501,21 @@ merge(Compressor.prototype, {
|
|||||||
if (fn.name && fn.name.definition().recursive_refs > 0) return this;
|
if (fn.name && fn.name.definition().recursive_refs > 0) return this;
|
||||||
if (this.is_expr_pure(compressor)) return this;
|
if (this.is_expr_pure(compressor)) return this;
|
||||||
var stat = fn.first_statement();
|
var stat = fn.first_statement();
|
||||||
if (!(stat instanceof AST_Return)) return this;
|
if (!(stat instanceof AST_Return)) {
|
||||||
|
if (ignore_side_effects) {
|
||||||
|
var found = false;
|
||||||
|
fn.walk(new TreeWalker(function(node) {
|
||||||
|
if (found) return true;
|
||||||
|
if (node instanceof AST_Return) {
|
||||||
|
if (node.value && node.value.evaluate(compressor, true) !== undefined) found = true;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
if (node instanceof AST_Scope && node !== fn) return true;
|
||||||
|
}));
|
||||||
|
if (!found) return void 0;
|
||||||
|
}
|
||||||
|
return this;
|
||||||
|
}
|
||||||
var args = eval_args(this.args);
|
var args = eval_args(this.args);
|
||||||
if (!args) return this;
|
if (!args) return this;
|
||||||
if (!stat.value) return undefined;
|
if (!stat.value) return undefined;
|
||||||
@@ -7427,7 +7441,7 @@ merge(Compressor.prototype, {
|
|||||||
var single_use = def.single_use && !(parent instanceof AST_Call && parent.is_expr_pure(compressor));
|
var single_use = def.single_use && !(parent instanceof AST_Call && parent.is_expr_pure(compressor));
|
||||||
if (single_use) {
|
if (single_use) {
|
||||||
if (fixed instanceof AST_Lambda) {
|
if (fixed instanceof AST_Lambda) {
|
||||||
if (def.scope !== self.scope
|
if ((def.scope !== self.scope || def.in_loop)
|
||||||
&& (!compressor.option("reduce_funcs") || def.escaped.depth == 1 || fixed.inlined)) {
|
&& (!compressor.option("reduce_funcs") || def.escaped.depth == 1 || fixed.inlined)) {
|
||||||
single_use = false;
|
single_use = false;
|
||||||
} else if (recursive_ref(compressor, def)) {
|
} else if (recursive_ref(compressor, def)) {
|
||||||
@@ -7771,7 +7785,7 @@ merge(Compressor.prototype, {
|
|||||||
expressions.push(self);
|
expressions.push(self);
|
||||||
return make_sequence(self, expressions);
|
return make_sequence(self, expressions);
|
||||||
}
|
}
|
||||||
var condition = self.condition.is_truthy() || self.condition.evaluate(compressor);
|
var condition = self.condition.is_truthy() || self.condition.evaluate(compressor, true);
|
||||||
if (!condition) {
|
if (!condition) {
|
||||||
AST_Node.warn("Condition always false [{file}:{line},{col}]", self.start);
|
AST_Node.warn("Condition always false [{file}:{line},{col}]", self.start);
|
||||||
return make_sequence(self, [ self.condition, self.alternative ]).optimize(compressor);
|
return make_sequence(self, [ self.condition, self.alternative ]).optimize(compressor);
|
||||||
|
|||||||
@@ -2190,3 +2190,139 @@ issue_3755: {
|
|||||||
}
|
}
|
||||||
expect_stdout: "undefined"
|
expect_stdout: "undefined"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void_side_effects: {
|
||||||
|
options = {
|
||||||
|
evaluate: true,
|
||||||
|
reduce_vars: true,
|
||||||
|
toplevel: true,
|
||||||
|
unused: true,
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
var a = void console.log("PASS");
|
||||||
|
console.log(a);
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
console.log("PASS");
|
||||||
|
console.log(void 0);
|
||||||
|
}
|
||||||
|
expect_stdout: [
|
||||||
|
"PASS",
|
||||||
|
"undefined",
|
||||||
|
]
|
||||||
|
}
|
||||||
|
|
||||||
|
no_returns: {
|
||||||
|
options = {
|
||||||
|
evaluate: true,
|
||||||
|
reduce_vars: true,
|
||||||
|
toplevel: true,
|
||||||
|
unused: true,
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
var a = function() {
|
||||||
|
console.log("PASS");
|
||||||
|
}();
|
||||||
|
console.log(a);
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
(function() {
|
||||||
|
console.log("PASS");
|
||||||
|
})();
|
||||||
|
console.log(void 0);
|
||||||
|
}
|
||||||
|
expect_stdout: [
|
||||||
|
"PASS",
|
||||||
|
"undefined",
|
||||||
|
]
|
||||||
|
}
|
||||||
|
|
||||||
|
void_returns: {
|
||||||
|
options = {
|
||||||
|
evaluate: true,
|
||||||
|
reduce_vars: true,
|
||||||
|
toplevel: true,
|
||||||
|
unused: true,
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
var a = function f() {
|
||||||
|
function g(b) {
|
||||||
|
if (b) console.log("FAIL");
|
||||||
|
}
|
||||||
|
while (1) {
|
||||||
|
console.log("PASS");
|
||||||
|
try {
|
||||||
|
if (console) return;
|
||||||
|
} catch (e) {
|
||||||
|
return g(e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}();
|
||||||
|
console.log(a);
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
(function() {
|
||||||
|
function g(b) {
|
||||||
|
if (b) console.log("FAIL");
|
||||||
|
}
|
||||||
|
while (1) {
|
||||||
|
console.log("PASS");
|
||||||
|
try {
|
||||||
|
if (console) return;
|
||||||
|
} catch (e) {
|
||||||
|
return g(e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})();
|
||||||
|
console.log(void 0);
|
||||||
|
}
|
||||||
|
expect_stdout: [
|
||||||
|
"PASS",
|
||||||
|
"undefined",
|
||||||
|
]
|
||||||
|
}
|
||||||
|
|
||||||
|
void_returns_recursive: {
|
||||||
|
options = {
|
||||||
|
evaluate: true,
|
||||||
|
reduce_vars: true,
|
||||||
|
toplevel: true,
|
||||||
|
unused: true,
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
var a = function f() {
|
||||||
|
function g(b) {
|
||||||
|
return f();
|
||||||
|
}
|
||||||
|
while (1) {
|
||||||
|
console.log("PASS");
|
||||||
|
try {
|
||||||
|
if (console) return;
|
||||||
|
} catch (e) {
|
||||||
|
return g(e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}();
|
||||||
|
console.log(a);
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
var a = function f() {
|
||||||
|
function g(b) {
|
||||||
|
return f();
|
||||||
|
}
|
||||||
|
while (1) {
|
||||||
|
console.log("PASS");
|
||||||
|
try {
|
||||||
|
if (console) return;
|
||||||
|
} catch (e) {
|
||||||
|
return g();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}();
|
||||||
|
console.log(a);
|
||||||
|
}
|
||||||
|
expect_stdout: [
|
||||||
|
"PASS",
|
||||||
|
"undefined",
|
||||||
|
]
|
||||||
|
}
|
||||||
|
|||||||
@@ -3514,10 +3514,9 @@ hoisted_single_use: {
|
|||||||
}
|
}
|
||||||
expect: {
|
expect: {
|
||||||
function f(a) {
|
function f(a) {
|
||||||
for (var r in a) g(r);
|
for (var r in a) (function(a) {
|
||||||
}
|
console.log(a);
|
||||||
function g(a) {
|
})(r);
|
||||||
console.log(a);
|
|
||||||
}
|
}
|
||||||
(function(a) {
|
(function(a) {
|
||||||
var g = a.bar;
|
var g = a.bar;
|
||||||
|
|||||||
@@ -1197,11 +1197,10 @@ toplevel_on_loops_1: {
|
|||||||
while (x);
|
while (x);
|
||||||
}
|
}
|
||||||
expect: {
|
expect: {
|
||||||
function bar() {
|
|
||||||
console.log("bar:", --x);
|
|
||||||
}
|
|
||||||
var x = 3;
|
var x = 3;
|
||||||
for (;bar(), x;);
|
for (;function() {
|
||||||
|
console.log("bar:", --x);
|
||||||
|
}(), x;);
|
||||||
}
|
}
|
||||||
expect_stdout: true
|
expect_stdout: true
|
||||||
}
|
}
|
||||||
@@ -1254,10 +1253,9 @@ toplevel_on_loops_2: {
|
|||||||
while (x);
|
while (x);
|
||||||
}
|
}
|
||||||
expect: {
|
expect: {
|
||||||
function bar() {
|
for (;;) (function() {
|
||||||
console.log("bar:");
|
console.log("bar:");
|
||||||
}
|
})();
|
||||||
for (;;) bar();
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -4231,13 +4229,12 @@ issue_2450_4: {
|
|||||||
}
|
}
|
||||||
expect: {
|
expect: {
|
||||||
var a;
|
var a;
|
||||||
function f(b) {
|
|
||||||
console.log(a === b);
|
|
||||||
a = b;
|
|
||||||
}
|
|
||||||
function g() {}
|
function g() {}
|
||||||
for (var i = 3; --i >= 0;)
|
for (var i = 3; --i >= 0;)
|
||||||
f(g);
|
(function(b) {
|
||||||
|
console.log(a === b);
|
||||||
|
a = b;
|
||||||
|
})(g);
|
||||||
}
|
}
|
||||||
expect_stdout: [
|
expect_stdout: [
|
||||||
"false",
|
"false",
|
||||||
@@ -4338,14 +4335,13 @@ perf_1: {
|
|||||||
console.log(sum);
|
console.log(sum);
|
||||||
}
|
}
|
||||||
expect: {
|
expect: {
|
||||||
function indirect_foo(x, y, z) {
|
|
||||||
return function(x, y, z) {
|
|
||||||
return x < y ? x * y + z : x * z - y;
|
|
||||||
}(x, y, z);
|
|
||||||
}
|
|
||||||
var sum = 0;
|
var sum = 0;
|
||||||
for (var i = 0; i < 100; ++i)
|
for (var i = 0; i < 100; ++i)
|
||||||
sum += indirect_foo(i, i + 1, 3 * i);
|
sum += function(x, y, z) {
|
||||||
|
return function(x, y, z) {
|
||||||
|
return x < y ? x * y + z : x * z - y;
|
||||||
|
}(x, y, z);
|
||||||
|
}(i, i + 1, 3 * i);
|
||||||
console.log(sum);
|
console.log(sum);
|
||||||
}
|
}
|
||||||
expect_stdout: "348150"
|
expect_stdout: "348150"
|
||||||
@@ -4406,14 +4402,13 @@ perf_3: {
|
|||||||
console.log(sum);
|
console.log(sum);
|
||||||
}
|
}
|
||||||
expect: {
|
expect: {
|
||||||
var indirect_foo = function(x, y, z) {
|
|
||||||
return function(x, y, z) {
|
|
||||||
return x < y ? x * y + z : x * z - y;
|
|
||||||
}(x, y, z);
|
|
||||||
}
|
|
||||||
var sum = 0;
|
var sum = 0;
|
||||||
for (var i = 0; i < 100; ++i)
|
for (var i = 0; i < 100; ++i)
|
||||||
sum += indirect_foo(i, i + 1, 3 * i);
|
sum += function(x, y, z) {
|
||||||
|
return function(x, y, z) {
|
||||||
|
return x < y ? x * y + z : x * z - y;
|
||||||
|
}(x, y, z);
|
||||||
|
}(i, i + 1, 3 * i);
|
||||||
console.log(sum);
|
console.log(sum);
|
||||||
}
|
}
|
||||||
expect_stdout: "348150"
|
expect_stdout: "348150"
|
||||||
@@ -4475,14 +4470,13 @@ perf_5: {
|
|||||||
console.log(sum);
|
console.log(sum);
|
||||||
}
|
}
|
||||||
expect: {
|
expect: {
|
||||||
function indirect_foo(x, y, z) {
|
|
||||||
return function(x, y, z) {
|
|
||||||
return x < y ? x * y + z : x * z - y;
|
|
||||||
}(x, y, z);
|
|
||||||
}
|
|
||||||
var sum = 0;
|
var sum = 0;
|
||||||
for (var i = 0; i < 100; ++i)
|
for (var i = 0; i < 100; ++i)
|
||||||
sum += indirect_foo(i, i + 1, 3 * i);
|
sum += function(x, y, z) {
|
||||||
|
return function(x, y, z) {
|
||||||
|
return x < y ? x * y + z : x * z - y;
|
||||||
|
}(x, y, z);
|
||||||
|
}(i, i + 1, 3 * i);
|
||||||
console.log(sum);
|
console.log(sum);
|
||||||
}
|
}
|
||||||
expect_stdout: "348150"
|
expect_stdout: "348150"
|
||||||
@@ -4543,14 +4537,13 @@ perf_7: {
|
|||||||
console.log(sum);
|
console.log(sum);
|
||||||
}
|
}
|
||||||
expect: {
|
expect: {
|
||||||
var indirect_foo = function(x, y, z) {
|
|
||||||
return function(x, y, z) {
|
|
||||||
return x < y ? x * y + z : x * z - y;
|
|
||||||
}(x, y, z);
|
|
||||||
}
|
|
||||||
var sum = 0;
|
var sum = 0;
|
||||||
for (var i = 0; i < 100; ++i)
|
for (var i = 0; i < 100; ++i)
|
||||||
sum += indirect_foo(i, i + 1, 3 * i);
|
sum += function(x, y, z) {
|
||||||
|
return function(x, y, z) {
|
||||||
|
return x < y ? x * y + z : x * z - y;
|
||||||
|
}(x, y, z);
|
||||||
|
}(i, i + 1, 3 * i);
|
||||||
console.log(sum);
|
console.log(sum);
|
||||||
}
|
}
|
||||||
expect_stdout: "348150"
|
expect_stdout: "348150"
|
||||||
@@ -7054,24 +7047,3 @@ issue_3866: {
|
|||||||
}
|
}
|
||||||
expect_stdout: "PASS"
|
expect_stdout: "PASS"
|
||||||
}
|
}
|
||||||
|
|
||||||
void_side_effects: {
|
|
||||||
options = {
|
|
||||||
evaluate: true,
|
|
||||||
reduce_vars: true,
|
|
||||||
toplevel: true,
|
|
||||||
unused: true,
|
|
||||||
}
|
|
||||||
input: {
|
|
||||||
var a = void console.log("PASS");
|
|
||||||
console.log(a);
|
|
||||||
}
|
|
||||||
expect: {
|
|
||||||
console.log("PASS");
|
|
||||||
console.log(void 0);
|
|
||||||
}
|
|
||||||
expect_stdout: [
|
|
||||||
"PASS",
|
|
||||||
"undefined",
|
|
||||||
]
|
|
||||||
}
|
|
||||||
|
|||||||
Reference in New Issue
Block a user