fix corner case in `inilne` (#4204)
authorAlex Lam S.L <alexlamsl@gmail.com>
Mon, 12 Oct 2020 15:10:32 +0000 (16:10 +0100)
committerGitHub <noreply@github.com>
Mon, 12 Oct 2020 15:10:32 +0000 (23:10 +0800)
fixes #4202

lib/compress.js
test/compress/const.js

index 3464a23..f6de865 100644 (file)
@@ -7198,24 +7198,24 @@ merge(Compressor.prototype, {
             return defined[name] || identifier_atom[name] || scope.var_names()[name];
         }
 
-        function can_inject_args(catches, used, safe_to_inject) {
+        function can_inject_args(defined, used, safe_to_inject) {
             for (var i = 0; i < fn.argnames.length; i++) {
                 var arg = fn.argnames[i];
                 if (arg.__unused) continue;
-                if (!safe_to_inject || var_exists(catches, arg.name)) return false;
+                if (!safe_to_inject || var_exists(defined, arg.name)) return false;
                 used[arg.name] = true;
                 if (in_loop) in_loop.push(arg.definition());
             }
             return true;
         }
 
-        function can_inject_vars(catches, used, safe_to_inject) {
+        function can_inject_vars(defined, used, safe_to_inject) {
             for (var i = 0; i < fn.body.length; i++) {
                 var stat = fn.body[i];
                 if (stat instanceof AST_Defun) {
                     if (!safe_to_inject || var_exists(used, stat.name.name)) return false;
                     if (!all(stat.enclosed, function(def) {
-                        return def.scope === stat || !catches[def.name];
+                        return def.scope === stat || !defined[def.name];
                     })) return false;
                     continue;
                 }
@@ -7223,7 +7223,7 @@ merge(Compressor.prototype, {
                 if (!safe_to_inject) return false;
                 for (var j = stat.definitions.length; --j >= 0;) {
                     var name = stat.definitions[j].name;
-                    if (var_exists(catches, name.name)) return false;
+                    if (var_exists(defined, name.name)) return false;
                     if (in_loop) in_loop.push(name.definition());
                 }
             }
@@ -7231,15 +7231,16 @@ merge(Compressor.prototype, {
         }
 
         function can_inject_symbols() {
-            var catches = Object.create(null);
+            var defined = Object.create(null);
             var child;
             scope = compressor.self();
             do {
                 child = scope;
                 scope = compressor.parent(++level);
-                if (scope instanceof AST_Catch) {
-                    if (scope.argname) catches[scope.argname.name] = true;
-                } else if (scope instanceof AST_DWLoop) {
+                if (scope.variables) scope.variables.each(function(def) {
+                    defined[def.name] = true;
+                });
+                if (scope instanceof AST_DWLoop) {
                     in_loop = [];
                 } else if (scope instanceof AST_For) {
                     if (scope.init === child) continue;
@@ -7255,9 +7256,9 @@ merge(Compressor.prototype, {
             var safe_to_inject = (!(scope instanceof AST_Toplevel) || compressor.toplevel.vars)
                 && (exp !== fn || fn.parent_scope.resolve() === compressor.find_parent(AST_Scope));
             var inline = compressor.option("inline");
-            var used = Object.create(catches);
-            if (!can_inject_args(catches, used, inline >= 2 && safe_to_inject)) return false;
-            if (!can_inject_vars(catches, used, inline >= 3 && safe_to_inject)) return false;
+            var used = Object.create(defined);
+            if (!can_inject_args(defined, used, inline >= 2 && safe_to_inject)) return false;
+            if (!can_inject_vars(defined, used, inline >= 3 && safe_to_inject)) return false;
             return !in_loop || in_loop.length == 0 || !is_reachable(fn, in_loop);
         }
 
index 72033f4..28dda01 100644 (file)
@@ -840,3 +840,35 @@ issue_4198: {
     }
     expect_stdout: "PASS"
 }
+
+issue_4202: {
+    options = {
+        inline: true,
+        toplevel: true,
+    }
+    input: {
+        {
+            const o = {};
+            (function() {
+                function f() {
+                    o.p = 42;
+                }
+                f(f);
+            })();
+            console.log(o.p++);
+        }
+    }
+    expect: {
+        {
+            const o = {};
+            (function() {
+                function f() {
+                    o.p = 42;
+                }
+                f(f);
+            })();
+            console.log(o.p++);
+        }
+    }
+    expect_stdout: "42"
+}