Skip to content

Commit 9c80456

Browse files
authoredJun 18, 2024··
enhance dead_code & side_effects (#5840)
closes #5794
1 parent dc7aa32 commit 9c80456

File tree

5 files changed

+159
-24
lines changed

5 files changed

+159
-24
lines changed
 

‎README.md

+5-5
Original file line numberDiff line numberDiff line change
@@ -779,11 +779,11 @@ to be `false` and all symbol names will be omitted.
779779
overhead (compression will be slower). Make sure symbols under `pure_funcs`
780780
are also under `mangle.reserved` to avoid mangling.
781781

782-
- `pure_getters` (default: `"strict"`) — If you pass `true` for
783-
this, UglifyJS will assume that object property access
784-
(e.g. `foo.bar` or `foo["bar"]`) doesn't have any side effects.
785-
Specify `"strict"` to treat `foo.bar` as side-effect-free only when
786-
`foo` is certain to not throw, i.e. not `null` or `undefined`.
782+
- `pure_getters` (default: `"strict"`) — Pass `true` for UglifyJS to assume that
783+
object property access (e.g. `foo.bar` or `a[42]`) does not throw exception or
784+
alter program states via getter function. Pass `"strict"` to allow dropping or
785+
reordering `foo.bar` only if `foo` is not `null` or `undefined` and is safe to
786+
access as a variable. Pass `false` to retain all property accesses.
787787

788788
- `reduce_funcs` (default: `true`) — Allows single-use functions to be
789789
inlined as function expressions when permissible allowing further

‎lib/ast.js

-2
Original file line numberDiff line numberDiff line change
@@ -169,8 +169,6 @@ DEF_BITPROPS(AST_Node, [
169169
"private",
170170
// AST_Call
171171
"pure",
172-
// AST_Assign
173-
"redundant",
174172
// AST_Node
175173
"single_use",
176174
// AST_ClassProperty

‎lib/compress.js

+50-16
Original file line numberDiff line numberDiff line change
@@ -638,15 +638,30 @@ Compressor.prototype.compress = function(node) {
638638
}
639639

640640
function push(tw, sequential) {
641+
var defined_ids = Object.create(tw.defined_ids);
641642
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;
643648
tw.safe_ids = safe_ids;
644649
}
645650

646651
function pop(tw) {
652+
tw.defined_ids = Object.getPrototypeOf(tw.defined_ids);
647653
tw.safe_ids = Object.getPrototypeOf(tw.safe_ids);
648654
}
649655

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+
650665
function mark(tw, def) {
651666
tw.safe_ids[def.id] = {};
652667
}
@@ -939,9 +954,13 @@ Compressor.prototype.compress = function(node) {
939954
return fixed_node;
940955
}, visit);
941956
walk_lambda(fn, tw);
957+
var defined_ids = tw.defined_ids;
942958
var safe_ids = tw.safe_ids;
943959
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+
}
945964
return true;
946965

947966
function visit(node, fixed) {
@@ -966,10 +985,10 @@ Compressor.prototype.compress = function(node) {
966985
var scan = ld || left instanceof AST_Destructured;
967986
switch (node.operator) {
968987
case "=":
988+
if (ld) assign(tw, ld);
969989
if (left.equals(right) && !left.has_side_effects(compressor)) {
970990
right.walk(tw);
971991
walk_prop(left);
972-
node.redundant = true;
973992
return true;
974993
}
975994
if (ld && right instanceof AST_LambdaExpression) {
@@ -990,6 +1009,7 @@ Compressor.prototype.compress = function(node) {
9901009
case "||=":
9911010
case "??=":
9921011
var lazy = true;
1012+
if (ld) assign(tw, ld);
9931013
default:
9941014
if (!scan) {
9951015
mark_assignment_to_arguments(left);
@@ -1103,7 +1123,7 @@ Compressor.prototype.compress = function(node) {
11031123
def(AST_BlockScope, function(tw, descend, compressor) {
11041124
reset_block_variables(tw, compressor, this);
11051125
});
1106-
def(AST_Call, function(tw, descend) {
1126+
def(AST_Call, function(tw) {
11071127
var node = this;
11081128
var exp = node.expression;
11091129
if (exp instanceof AST_LambdaExpression) {
@@ -1134,6 +1154,7 @@ Compressor.prototype.compress = function(node) {
11341154
if (fixed instanceof AST_Lambda) {
11351155
mark_fn_def(tw, exp.definition(), fixed);
11361156
} else {
1157+
tw.defined_ids.seq = {};
11371158
tw.find_parent(AST_Scope).may_call_this();
11381159
}
11391160
return true;
@@ -1238,6 +1259,12 @@ Compressor.prototype.compress = function(node) {
12381259
tw.in_loop = save_loop;
12391260
return true;
12401261
});
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+
});
12411268
def(AST_For, function(tw, descend, compressor) {
12421269
var node = this;
12431270
reset_block_variables(tw, compressor, node);
@@ -1336,10 +1363,13 @@ Compressor.prototype.compress = function(node) {
13361363
return true;
13371364
});
13381365
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());
13411371
push(tw, true);
1342-
this.property.walk(tw);
1372+
node.property.walk(tw);
13431373
pop(tw);
13441374
return true;
13451375
});
@@ -1390,6 +1420,8 @@ Compressor.prototype.compress = function(node) {
13901420
var d = ref.definition();
13911421
var fixed = d.fixed || d.last_ref && d.last_ref.fixed;
13921422
push_ref(d, ref);
1423+
var defined = tw.defined_ids[d.id];
1424+
if (defined && defined[0] === tw.defined_ids.seq) ref.defined = true;
13931425
if (d.references.length == 1 && !d.fixed && d.orig[0] instanceof AST_SymbolDefun) {
13941426
tw.loop_ids[d.id] = tw.in_loop;
13951427
}
@@ -1618,6 +1650,9 @@ Compressor.prototype.compress = function(node) {
16181650
reset_flags(node);
16191651
return node.reduce_vars(tw, descend, compressor);
16201652
} : reset_flags);
1653+
// Side-effect tracking on sequential property access
1654+
tw.defined_ids = Object.create(null);
1655+
tw.defined_ids.seq = {};
16211656
// Flow control for visiting lambda definitions
16221657
tw.fn_scanning = null;
16231658
tw.fn_visited = [];
@@ -4712,6 +4747,7 @@ Compressor.prototype.compress = function(node) {
47124747
return this.tail_node()._dot_throw(compressor);
47134748
});
47144749
def(AST_SymbolRef, function(compressor, force) {
4750+
if (this.defined) return false;
47154751
if (this.is_undefined) return true;
47164752
if (!is_strict(compressor, force)) return false;
47174753
if (is_undeclared_ref(this) && this.is_declared(compressor)) return false;
@@ -12955,16 +12991,14 @@ Compressor.prototype.compress = function(node) {
1295512991
if (compressor.option("dead_code")) {
1295612992
if (self.left instanceof AST_PropAccess) {
1295712993
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-
}
1296712994
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+
}
1296813002
if (exp instanceof AST_Lambda
1296913003
|| !compressor.has_directive("use strict")
1297013004
&& exp instanceof AST_Constant

‎test/compress/arrows.js

+1-1
Original file line numberDiff line numberDiff line change
@@ -1312,7 +1312,7 @@ issue_5653: {
13121312
}
13131313
expect: {
13141314
console.log((a => {
1315-
return console, +{};
1315+
return +{};
13161316
})());
13171317
}
13181318
expect_stdout: "NaN"

‎test/compress/side_effects.js

+103
Original file line numberDiff line numberDiff line change
@@ -724,3 +724,106 @@ retain_instanceof: {
724724
}
725725
expect_stdout: "PASS"
726726
}
727+
728+
drop_access: {
729+
options = {
730+
pure_getters: "strict",
731+
reduce_vars: true,
732+
side_effects: true,
733+
}
734+
input: {
735+
var o = {};
736+
o.p;
737+
try {
738+
(function() {
739+
o.q;
740+
})();
741+
console.log("PASS");
742+
} catch (e) {
743+
console.log("FAIL");
744+
}
745+
}
746+
expect: {
747+
var o = {};
748+
o.p;
749+
try {
750+
console.log("PASS");
751+
} catch (e) {
752+
console.log("FAIL");
753+
}
754+
}
755+
expect_stdout: "PASS"
756+
}
757+
758+
keep_access: {
759+
options = {
760+
pure_getters: "strict",
761+
reduce_vars: true,
762+
side_effects: true,
763+
}
764+
input: {
765+
var o = {};
766+
o.p;
767+
o = null;
768+
try {
769+
(function() {
770+
o.q;
771+
})();
772+
console.log("FAIL");
773+
} catch (e) {
774+
console.log("PASS");
775+
}
776+
}
777+
expect: {
778+
var o = {};
779+
o.p;
780+
o = null;
781+
try {
782+
(function() {
783+
o.q;
784+
})();
785+
console.log("FAIL");
786+
} catch (e) {
787+
console.log("PASS");
788+
}
789+
}
790+
expect_stdout: "PASS"
791+
}
792+
793+
keep_access_after_call: {
794+
options = {
795+
pure_getters: "strict",
796+
reduce_vars: true,
797+
side_effects: true,
798+
}
799+
input: {
800+
var o = {};
801+
o.p;
802+
o.q;
803+
f();
804+
try {
805+
o.r;
806+
console.log("FAIL");
807+
} catch (e) {
808+
console.log("PASS");
809+
}
810+
function f() {
811+
o = null;
812+
}
813+
}
814+
expect: {
815+
var o = {};
816+
o.p;
817+
f();
818+
try {
819+
o.r;
820+
console.log("FAIL");
821+
} catch (e) {
822+
console.log("PASS");
823+
}
824+
function f() {
825+
o = null;
826+
}
827+
}
828+
expect_stdout: "PASS"
829+
}

0 commit comments

Comments
 (0)
Please sign in to comment.