improve boolean compression (#2548)

fixes #2535
This commit is contained in:
Alex Lam S.L
2017-12-01 22:41:35 +08:00
committed by GitHub
parent 7ac6fdcc99
commit 9a6b11f8e6
3 changed files with 186 additions and 24 deletions

View File

@@ -158,6 +158,7 @@ merge(Compressor.prototype, {
return true; return true;
} }
if (p instanceof AST_Binary && (p.operator == "&&" || p.operator == "||") if (p instanceof AST_Binary && (p.operator == "&&" || p.operator == "||")
|| p instanceof AST_Conditional
|| p.tail_node() === self) { || p.tail_node() === self) {
self = p; self = p;
} else { } else {
@@ -4090,49 +4091,72 @@ merge(Compressor.prototype, {
if (compressor.option("evaluate")) { if (compressor.option("evaluate")) {
switch (self.operator) { switch (self.operator) {
case "&&": case "&&":
var ll = self.left.evaluate(compressor); var ll = self.left.truthy ? true : self.left.falsy ? false : self.left.evaluate(compressor);
if (!ll) { if (!ll) {
compressor.warn("Condition left of && always false [{file}:{line},{col}]", self.start); compressor.warn("Condition left of && always false [{file}:{line},{col}]", self.start);
return maintain_this_binding(compressor.parent(), compressor.self(), self.left).optimize(compressor); return maintain_this_binding(compressor.parent(), compressor.self(), self.left).optimize(compressor);
} else if (ll !== self.left) { } else if (!(ll instanceof AST_Node)) {
compressor.warn("Condition left of && always true [{file}:{line},{col}]", self.start); compressor.warn("Condition left of && always true [{file}:{line},{col}]", self.start);
return maintain_this_binding(compressor.parent(), compressor.self(), self.right).optimize(compressor); return make_sequence(self, [ self.left, self.right ]).optimize(compressor);
} }
if (compressor.in_boolean_context()) { var rr = self.right.evaluate(compressor);
var rr = self.right.evaluate(compressor); if (!rr) {
if (!rr) { if (compressor.in_boolean_context()) {
compressor.warn("Boolean && always false [{file}:{line},{col}]", self.start); compressor.warn("Boolean && always false [{file}:{line},{col}]", self.start);
return make_sequence(self, [ return make_sequence(self, [
self.left, self.left,
make_node(AST_False, self) make_node(AST_False, self)
]).optimize(compressor); ]).optimize(compressor);
} else if (rr !== self.right) { } else self.falsy = true;
compressor.warn("Dropping side-effect-free && in boolean context [{file}:{line},{col}]", self.start); } else if (!(rr instanceof AST_Node)) {
var parent = compressor.parent();
if (parent.operator == "&&" && parent.left === compressor.self() || compressor.in_boolean_context()) {
compressor.warn("Dropping side-effect-free && [{file}:{line},{col}]", self.start);
return self.left.optimize(compressor); return self.left.optimize(compressor);
} }
} }
// x || false && y ---> x ? y : false
if (self.left.operator == "||") {
var lr = self.left.right.evaluate(compressor);
if (!lr) return make_node(AST_Conditional, self, {
condition: self.left.left,
consequent: self.right,
alternative: self.left.right
}).optimize(compressor);
}
break; break;
case "||": case "||":
var ll = self.left.evaluate(compressor); var ll = self.left.truthy ? true : self.left.falsy ? false : self.left.evaluate(compressor);
if (!ll) { if (!ll) {
compressor.warn("Condition left of || always false [{file}:{line},{col}]", self.start); compressor.warn("Condition left of || always false [{file}:{line},{col}]", self.start);
return maintain_this_binding(compressor.parent(), compressor.self(), self.right).optimize(compressor); return make_sequence(self, [ self.left, self.right ]).optimize(compressor);
} else if (ll !== self.left) { } else if (!(ll instanceof AST_Node)) {
compressor.warn("Condition left of || always true [{file}:{line},{col}]", self.start); compressor.warn("Condition left of || always true [{file}:{line},{col}]", self.start);
return maintain_this_binding(compressor.parent(), compressor.self(), self.left).optimize(compressor); return maintain_this_binding(compressor.parent(), compressor.self(), self.left).optimize(compressor);
} }
if (compressor.in_boolean_context()) { var rr = self.right.evaluate(compressor);
var rr = self.right.evaluate(compressor); if (!rr) {
if (!rr) { var parent = compressor.parent();
compressor.warn("Dropping side-effect-free || in boolean context [{file}:{line},{col}]", self.start); if (parent.operator == "||" && parent.left === compressor.self() || compressor.in_boolean_context()) {
compressor.warn("Dropping side-effect-free || [{file}:{line},{col}]", self.start);
return self.left.optimize(compressor); return self.left.optimize(compressor);
} else if (rr !== self.right) { }
} else if (!(rr instanceof AST_Node)) {
if (compressor.in_boolean_context()) {
compressor.warn("Boolean || always true [{file}:{line},{col}]", self.start); compressor.warn("Boolean || always true [{file}:{line},{col}]", self.start);
return make_sequence(self, [ return make_sequence(self, [
self.left, self.left,
make_node(AST_True, self) make_node(AST_True, self)
]).optimize(compressor); ]).optimize(compressor);
} } else self.truthy = true;
}
if (self.left.operator == "&&") {
var lr = self.left.right.evaluate(compressor);
if (lr && !(lr instanceof AST_Node)) return make_node(AST_Conditional, self, {
condition: self.left.left,
consequent: self.left.right,
alternative: self.right
}).optimize(compressor);
} }
break; break;
} }
@@ -4640,7 +4664,7 @@ merge(Compressor.prototype, {
consequent consequent
]).optimize(compressor); ]).optimize(compressor);
} }
var in_bool = compressor.in_boolean_context();
if (is_true(self.consequent)) { if (is_true(self.consequent)) {
if (is_false(self.alternative)) { if (is_false(self.alternative)) {
// c ? true : false ---> !!c // c ? true : false ---> !!c
@@ -4696,18 +4720,24 @@ merge(Compressor.prototype, {
// AST_True or !0 // AST_True or !0
function is_true(node) { function is_true(node) {
return node instanceof AST_True return node instanceof AST_True
|| in_bool
&& node instanceof AST_Constant
&& node.getValue()
|| (node instanceof AST_UnaryPrefix || (node instanceof AST_UnaryPrefix
&& node.operator == "!" && node.operator == "!"
&& node.expression instanceof AST_Constant && node.expression instanceof AST_Constant
&& !node.expression.value); && !node.expression.getValue());
} }
// AST_False or !1 // AST_False or !1
function is_false(node) { function is_false(node) {
return node instanceof AST_False return node instanceof AST_False
|| in_bool
&& node instanceof AST_Constant
&& !node.getValue()
|| (node instanceof AST_UnaryPrefix || (node instanceof AST_UnaryPrefix
&& node.operator == "!" && node.operator == "!"
&& node.expression instanceof AST_Constant && node.expression instanceof AST_Constant
&& !!node.expression.value); && node.expression.getValue());
} }
}); });

View File

@@ -1016,7 +1016,7 @@ delete_conditional_2: {
expect_stdout: true expect_stdout: true
} }
issue_2535: { issue_2535_1: {
options = { options = {
booleans: true, booleans: true,
conditionals: true, conditionals: true,
@@ -1044,3 +1044,74 @@ issue_2535: {
(x(), 0) && y(); (x(), 0) && y();
} }
} }
issue_2535_2: {
options = {
booleans: true,
conditionals: true,
evaluate: true,
side_effects: true,
}
input: {
function x() {}
function y() {
return "foo";
}
console.log((x() || true) || y());
console.log((y() || true) || x());
console.log((x() || true) && y());
console.log((y() || true) && x());
console.log((x() && true) || y());
console.log((y() && true) || x());
console.log((x() && true) && y());
console.log((y() && true) && x());
console.log((x() || false) || y());
console.log((y() || false) || x());
console.log((x() || false) && y());
console.log((y() || false) && x());
console.log((x() && false) || y());
console.log((y() && false) || x());
console.log((x() && false) && y());
console.log((y() && false) && x());
}
expect: {
function x() {}
function y() {
return "foo";
}
console.log(x() || !0);
console.log(y() || !0);
console.log((x(), y()));
console.log((y(), x()));
console.log(!!x() || y());
console.log(!!y() || x());
console.log(x() && y());
console.log(y() && x());
console.log(x() || y());
console.log(y() || x());
console.log(!!x() && y());
console.log(!!y() && x());
console.log((x(), y()));
console.log((y(), x()));
console.log(x() && !1);
console.log(y() && !1);
}
expect_stdout: [
"true",
"foo",
"foo",
"undefined",
"foo",
"true",
"undefined",
"undefined",
"foo",
"foo",
"false",
"undefined",
"foo",
"undefined",
"undefined",
"false",
]
}

View File

@@ -1,6 +1,7 @@
and: { and: {
options = { options = {
evaluate: true evaluate: true,
side_effects: true,
} }
input: { input: {
var a; var a;
@@ -76,7 +77,8 @@ and: {
or: { or: {
options = { options = {
evaluate: true evaluate: true,
side_effects: true,
} }
input: { input: {
var a; var a;
@@ -158,7 +160,8 @@ or: {
unary_prefix: { unary_prefix: {
options = { options = {
evaluate: true evaluate: true,
side_effects: true,
} }
input: { input: {
a = !0 && b; a = !0 && b;
@@ -1245,3 +1248,61 @@ self_comparison_2: {
} }
expect_stdout: "false false true true 'number'" expect_stdout: "false false true true 'number'"
} }
issue_2535_1: {
options = {
booleans: true,
evaluate: true,
sequences: true,
side_effects: true,
}
input: {
if ((x() || true) || y()) z();
if ((x() || true) && y()) z();
if ((x() && true) || y()) z();
if ((x() && true) && y()) z();
if ((x() || false) || y()) z();
if ((x() || false) && y()) z();
if ((x() && false) || y()) z();
if ((x() && false) && y()) z();
}
expect: {
if (x(), 1) z();
if (x(), y()) z();
if (x() || y()) z();
if (x() && y()) z();
if (x() || y()) z();
if (x() && y()) z();
if (x(), y()) z();
if (x(), 0) z();
}
}
issue_2535_2: {
options = {
booleans: true,
evaluate: true,
sequences: true,
side_effects: true,
}
input: {
(x() || true) || y();
(x() || true) && y();
(x() && true) || y();
(x() && true) && y();
(x() || false) || y();
(x() || false) && y();
(x() && false) || y();
(x() && false) && y();
}
expect: {
x(),
x(), y(),
x() || y(),
x() && y(),
x() || y(),
x() && y(),
x(), y(),
x();
}
}