@@ -638,15 +638,30 @@ Compressor.prototype.compress = function(node) {
638
638
}
639
639
640
640
function push(tw, sequential) {
641
+ var defined_ids = Object.create(tw.defined_ids);
641
642
var safe_ids = Object.create(tw.safe_ids);
642
- if (!sequential) safe_ids.seq = {};
643
+ if (!sequential) {
644
+ defined_ids.seq = {};
645
+ safe_ids.seq = {};
646
+ }
647
+ tw.defined_ids = defined_ids;
643
648
tw.safe_ids = safe_ids;
644
649
}
645
650
646
651
function pop(tw) {
652
+ tw.defined_ids = Object.getPrototypeOf(tw.defined_ids);
647
653
tw.safe_ids = Object.getPrototypeOf(tw.safe_ids);
648
654
}
649
655
656
+ function access(tw, def) {
657
+ tw.defined_ids[def.id] = [ tw.defined_ids.seq ];
658
+ }
659
+
660
+ function assign(tw, def) {
661
+ var defined = tw.defined_ids[def.id];
662
+ if (defined) defined[0] = false;
663
+ }
664
+
650
665
function mark(tw, def) {
651
666
tw.safe_ids[def.id] = {};
652
667
}
@@ -939,9 +954,13 @@ Compressor.prototype.compress = function(node) {
939
954
return fixed_node;
940
955
}, visit);
941
956
walk_lambda(fn, tw);
957
+ var defined_ids = tw.defined_ids;
942
958
var safe_ids = tw.safe_ids;
943
959
pop_scope(tw, fn);
944
- if (!aborts) tw.safe_ids = safe_ids;
960
+ if (!aborts) {
961
+ tw.defined_ids = defined_ids;
962
+ tw.safe_ids = safe_ids;
963
+ }
945
964
return true;
946
965
947
966
function visit(node, fixed) {
@@ -966,10 +985,10 @@ Compressor.prototype.compress = function(node) {
966
985
var scan = ld || left instanceof AST_Destructured;
967
986
switch (node.operator) {
968
987
case "=":
988
+ if (ld) assign(tw, ld);
969
989
if (left.equals(right) && !left.has_side_effects(compressor)) {
970
990
right.walk(tw);
971
991
walk_prop(left);
972
- node.redundant = true;
973
992
return true;
974
993
}
975
994
if (ld && right instanceof AST_LambdaExpression) {
@@ -990,6 +1009,7 @@ Compressor.prototype.compress = function(node) {
990
1009
case "||=":
991
1010
case "??=":
992
1011
var lazy = true;
1012
+ if (ld) assign(tw, ld);
993
1013
default:
994
1014
if (!scan) {
995
1015
mark_assignment_to_arguments(left);
@@ -1103,7 +1123,7 @@ Compressor.prototype.compress = function(node) {
1103
1123
def(AST_BlockScope, function(tw, descend, compressor) {
1104
1124
reset_block_variables(tw, compressor, this);
1105
1125
});
1106
- def(AST_Call, function(tw, descend ) {
1126
+ def(AST_Call, function(tw) {
1107
1127
var node = this;
1108
1128
var exp = node.expression;
1109
1129
if (exp instanceof AST_LambdaExpression) {
@@ -1134,6 +1154,7 @@ Compressor.prototype.compress = function(node) {
1134
1154
if (fixed instanceof AST_Lambda) {
1135
1155
mark_fn_def(tw, exp.definition(), fixed);
1136
1156
} else {
1157
+ tw.defined_ids.seq = {};
1137
1158
tw.find_parent(AST_Scope).may_call_this();
1138
1159
}
1139
1160
return true;
@@ -1238,6 +1259,12 @@ Compressor.prototype.compress = function(node) {
1238
1259
tw.in_loop = save_loop;
1239
1260
return true;
1240
1261
});
1262
+ def(AST_Dot, function(tw, descend) {
1263
+ descend();
1264
+ var expr = this.expression;
1265
+ if (expr instanceof AST_SymbolRef) access(tw, expr.definition());
1266
+ return true;
1267
+ });
1241
1268
def(AST_For, function(tw, descend, compressor) {
1242
1269
var node = this;
1243
1270
reset_block_variables(tw, compressor, node);
@@ -1336,10 +1363,13 @@ Compressor.prototype.compress = function(node) {
1336
1363
return true;
1337
1364
});
1338
1365
def(AST_Sub, function(tw) {
1339
- if (!this.optional) return;
1340
- this.expression.walk(tw);
1366
+ var node = this;
1367
+ if (!node.optional) return;
1368
+ var expr = node.expression;
1369
+ expr.walk(tw);
1370
+ if (expr instanceof AST_SymbolRef) access(tw, expr.definition());
1341
1371
push(tw, true);
1342
- this .property.walk(tw);
1372
+ node .property.walk(tw);
1343
1373
pop(tw);
1344
1374
return true;
1345
1375
});
@@ -1390,6 +1420,8 @@ Compressor.prototype.compress = function(node) {
1390
1420
var d = ref.definition();
1391
1421
var fixed = d.fixed || d.last_ref && d.last_ref.fixed;
1392
1422
push_ref(d, ref);
1423
+ var defined = tw.defined_ids[d.id];
1424
+ if (defined && defined[0] === tw.defined_ids.seq) ref.defined = true;
1393
1425
if (d.references.length == 1 && !d.fixed && d.orig[0] instanceof AST_SymbolDefun) {
1394
1426
tw.loop_ids[d.id] = tw.in_loop;
1395
1427
}
@@ -1618,6 +1650,9 @@ Compressor.prototype.compress = function(node) {
1618
1650
reset_flags(node);
1619
1651
return node.reduce_vars(tw, descend, compressor);
1620
1652
} : reset_flags);
1653
+ // Side-effect tracking on sequential property access
1654
+ tw.defined_ids = Object.create(null);
1655
+ tw.defined_ids.seq = {};
1621
1656
// Flow control for visiting lambda definitions
1622
1657
tw.fn_scanning = null;
1623
1658
tw.fn_visited = [];
@@ -4712,6 +4747,7 @@ Compressor.prototype.compress = function(node) {
4712
4747
return this.tail_node()._dot_throw(compressor);
4713
4748
});
4714
4749
def(AST_SymbolRef, function(compressor, force) {
4750
+ if (this.defined) return false;
4715
4751
if (this.is_undefined) return true;
4716
4752
if (!is_strict(compressor, force)) return false;
4717
4753
if (is_undeclared_ref(this) && this.is_declared(compressor)) return false;
@@ -12955,16 +12991,14 @@ Compressor.prototype.compress = function(node) {
12955
12991
if (compressor.option("dead_code")) {
12956
12992
if (self.left instanceof AST_PropAccess) {
12957
12993
if (self.operator == "=") {
12958
- if (self.redundant) {
12959
- var exprs = [ self.left.expression ];
12960
- if (self.left instanceof AST_Sub) exprs.push(self.left.property);
12961
- exprs.push(self.right);
12962
- return make_sequence(self, exprs).optimize(compressor);
12963
- }
12964
- if (self.left.equals(self.right) && !self.left.has_side_effects(compressor)) {
12965
- return self.right;
12966
- }
12967
12994
var exp = self.left.expression;
12995
+ if (self.left.equals(self.right)) {
12996
+ var defined = exp.defined;
12997
+ exp.defined = false;
12998
+ var drop_lhs = !self.left.has_side_effects(compressor);
12999
+ exp.defined = defined;
13000
+ if (drop_lhs) return self.right;
13001
+ }
12968
13002
if (exp instanceof AST_Lambda
12969
13003
|| !compressor.has_directive("use strict")
12970
13004
&& exp instanceof AST_Constant
0 commit comments