From bb2fafb6242c2a4af05e139b4b6e44cdc32ff508 Mon Sep 17 00:00:00 2001 From: alexlamsl Date: Wed, 20 Nov 2024 05:09:57 +0200 Subject: [PATCH] fix corner case in `reduce_vars` fixes #5963 --- lib/compress.js | 45 ++++++------ test/compress/default-values.js | 23 ++++++ test/compress/destructured.js | 46 ++++++++++++ test/compress/properties.js | 122 ++++++++++++++++++++++++++++++++ 4 files changed, 213 insertions(+), 23 deletions(-) diff --git a/lib/compress.js b/lib/compress.js index 7b1f6c552ec..5dcab298289 100644 --- a/lib/compress.js +++ b/lib/compress.js @@ -691,6 +691,15 @@ Compressor.prototype.compress = function(node) { return !assigned || assigned === seq; } + function dot(expr, tw) { + while (expr instanceof AST_Assign && expr.operator == "=") { + var lhs = expr.left; + if (lhs instanceof AST_SymbolRef) access(tw, lhs.definition()); + expr = expr.right; + } + if (expr instanceof AST_SymbolRef) access(tw, expr.definition()); + } + function mark(tw, def) { tw.safe_ids[def.id] = {}; } @@ -1032,7 +1041,9 @@ Compressor.prototype.compress = function(node) { return true; } mark_assignment_to_arguments(left); - return; + descend(); + if (left instanceof AST_PropAccess) dot(left.expression, tw); + return true; case "&&=": case "||=": case "??=": @@ -1108,6 +1119,7 @@ Compressor.prototype.compress = function(node) { if (!(sym instanceof AST_SymbolRef)) { mark_assignment_to_arguments(sym); walk(); + if (sym instanceof AST_PropAccess) dot(sym.expression, tw); return; } var d = sym.definition(); @@ -1291,15 +1303,7 @@ Compressor.prototype.compress = function(node) { def(AST_Dot, function(tw, descend) { descend(); var node = this; - var expr = node.expression; - if (!node.optional) { - while (expr instanceof AST_Assign && expr.operator == "=") { - var lhs = expr.left; - if (lhs instanceof AST_SymbolRef) access(tw, lhs.definition()); - expr = expr.right; - } - if (expr instanceof AST_SymbolRef) access(tw, expr.definition()); - } + if (!node.optional && !is_direct_assignment(node, tw.parent())) dot(node.expression, tw); return true; }); def(AST_For, function(tw, descend, compressor) { @@ -1410,12 +1414,7 @@ Compressor.prototype.compress = function(node) { prop.walk(tw); pop(tw); } else { - while (expr instanceof AST_Assign && expr.operator == "=") { - var lhs = expr.left; - if (lhs instanceof AST_SymbolRef) access(tw, lhs.definition()); - expr = expr.right; - } - if (expr instanceof AST_SymbolRef) access(tw, expr.definition()); + if (!is_direct_assignment(node, tw.parent())) dot(expr, tw); prop.walk(tw); } return true; @@ -1974,6 +1973,13 @@ Compressor.prototype.compress = function(node) { return array; } + function is_direct_assignment(node, parent) { + if (parent instanceof AST_Assign) return parent.operator == "=" && parent.left === node; + if (parent instanceof AST_DefaultValue) return parent.name === node; + if (parent instanceof AST_DestructuredArray) return true; + if (parent instanceof AST_DestructuredKeyVal) return parent.value === node; + } + function is_lexical_definition(stat) { return stat instanceof AST_Const || stat instanceof AST_DefClass || stat instanceof AST_Let; } @@ -2614,13 +2620,6 @@ Compressor.prototype.compress = function(node) { } } - function is_direct_assignment(node, parent) { - if (parent instanceof AST_Assign) return parent.operator == "=" && parent.left === node; - if (parent instanceof AST_DefaultValue) return parent.name === node; - if (parent instanceof AST_DestructuredArray) return true; - if (parent instanceof AST_DestructuredKeyVal) return parent.value === node; - } - function should_stop(node, parent) { if (node === rvalue) return true; if (parent instanceof AST_For) { diff --git a/test/compress/default-values.js b/test/compress/default-values.js index 72044ab3e08..a1b288a902a 100644 --- a/test/compress/default-values.js +++ b/test/compress/default-values.js @@ -3135,3 +3135,26 @@ issue_5863: { expect_stdout: "function" node_version: ">=6" } + +issue_5963: { + options = { + pure_getters: "strict", + reduce_vars: true, + side_effects: true, + } + input: { + var a = Object.create(null); + [ a.PASS = 42 ] = []; + a.FAIL; + for (var p in a) + console.log(p); + } + expect: { + var a = Object.create(null); + [ a.PASS = 42 ] = []; + for (var p in a) + console.log(p); + } + expect_stdout: "PASS" + node_version: ">=6" +} diff --git a/test/compress/destructured.js b/test/compress/destructured.js index 83e7392e002..f4326ef3ab6 100644 --- a/test/compress/destructured.js +++ b/test/compress/destructured.js @@ -4289,3 +4289,49 @@ issue_5899_2: { ] node_version: ">=6" } + +issue_5963_array: { + options = { + pure_getters: "strict", + reduce_vars: true, + side_effects: true, + } + input: { + var a = Object.create(null); + [ a.PASS ] = [ 42 ]; + a.FAIL; + for (var p in a) + console.log(p); + } + expect: { + var a = Object.create(null); + [ a.PASS ] = [ 42 ]; + for (var p in a) + console.log(p); + } + expect_stdout: "PASS" + node_version: ">=6" +} + +issue_5963_object: { + options = { + pure_getters: "strict", + reduce_vars: true, + side_effects: true, + } + input: { + var a = Object.create(null); + ({ p: a.PASS } = { p: 42 }); + a.FAIL; + for (var p in a) + console.log(p); + } + expect: { + var a = Object.create(null); + ({ p: a.PASS } = { p: 42 }); + for (var p in a) + console.log(p); + } + expect_stdout: "PASS" + node_version: ">=6" +} diff --git a/test/compress/properties.js b/test/compress/properties.js index 618ec3146e9..a2175a08809 100644 --- a/test/compress/properties.js +++ b/test/compress/properties.js @@ -2046,3 +2046,125 @@ issue_5949_2: { } expect_stdout: "PASS" } + +issue_5963_dot: { + options = { + collapse_vars: true, + evaluate: true, + pure_getters: "strict", + reduce_vars: true, + } + input: { + var a = "PASS", b; + try { + b.p = (b.q = null, a = "FAIL 1", !0); + console.log("FAIL 2") + } catch (e) { + console.log(a); + } + } + expect: { + var a = "PASS", b; + try { + b.p = (b.q = null, a = "FAIL 1", !0); + console.log("FAIL 2") + } catch (e) { + console.log(a); + } + } + expect_stdout: "PASS" +} + +issue_5963_sub: { + options = { + collapse_vars: true, + evaluate: true, + pure_getters: "strict", + reduce_vars: true, + } + input: { + var a = "PASS", b; + try { + b[42] = (b.q = null, a = "FAIL 1", !0); + console.log("FAIL 2") + } catch (e) { + console.log(a); + } + } + expect: { + var a = "PASS", b; + try { + b[42] = (b.q = null, a = "FAIL 1", !0); + console.log("FAIL 2") + } catch (e) { + console.log(a); + } + } + expect_stdout: "PASS" +} + +issue_5963_assign: { + options = { + pure_getters: "strict", + reduce_vars: true, + side_effects: true, + } + input: { + var a = Object.create(null); + a.PASS = 42; + a.FAIL; + for (var p in a) + console.log(p); + } + expect: { + var a = Object.create(null); + a.PASS = 42; + for (var p in a) + console.log(p); + } + expect_stdout: "PASS" +} + +issue_5963_compound_assign: { + options = { + pure_getters: "strict", + reduce_vars: true, + side_effects: true, + } + input: { + var a = Object.create(null); + a.PASS ^= 42; + a.FAIL; + for (var p in a) + console.log(p); + } + expect: { + var a = Object.create(null); + a.PASS ^= 42; + for (var p in a) + console.log(p); + } + expect_stdout: "PASS" +} + +issue_5963_unary: { + options = { + pure_getters: "strict", + reduce_vars: true, + side_effects: true, + } + input: { + var a = Object.create(null); + a.PASS++; + a.FAIL; + for (var p in a) + console.log(p); + } + expect: { + var a = Object.create(null); + a.PASS++; + for (var p in a) + console.log(p); + } + expect_stdout: "PASS" +}