]> granicus.if.org Git - vim/commitdiff
patch 9.0.0495: closure doesn't work properly in nested loop v9.0.0495
authorBram Moolenaar <Bram@vim.org>
Sun, 18 Sep 2022 11:00:21 +0000 (12:00 +0100)
committerBram Moolenaar <Bram@vim.org>
Sun, 18 Sep 2022 11:00:21 +0000 (12:00 +0100)
Problem:    Closure doesn't work properly in nested loop.
Solution:   Save variables up to the outer loop.

src/testdir/test_vim9_script.vim
src/version.c
src/vim9execute.c

index 85700b6c36de0f7b4d0ffc23a98500cac3cee73b..c4f02ab28afbf5321cefd9bb954efd6169c5e424 100644 (file)
@@ -2300,6 +2300,32 @@ def Test_for_loop_with_closure()
       endfor
   END
   v9.CheckDefAndScriptSuccess(lines)
+
+  # Also works for a nested loop
+  lines =<< trim END
+      var flist: list<func>
+      var n = 0
+      for i in range(3)
+        var ii = i
+        for a in ['a', 'b', 'c']
+          var aa = a
+          flist[n] = () => ii .. aa
+          ++n
+        endfor
+      endfor
+
+      n = 0
+      for i in range(3)
+        for a in ['a', 'b', 'c']
+          assert_equal(i .. a, flist[n]())
+          ++n
+        endfor
+      endfor
+  END
+  v9.CheckScriptSuccess(['vim9script'] + lines)
+  # FIXME: not yet right for :def
+  lines[14] = 'assert_equal(2 .. a, flist[n]())'
+  v9.CheckDefSuccess(lines)
 enddef
 
 def Test_for_loop_fails()
index 88888d4307de7b1d3b267fd4da09d5d2fe12605f..c361f97b6bb795d51fcc39396c3986ee51a746d2 100644 (file)
@@ -699,6 +699,8 @@ static char *(features[]) =
 
 static int included_patches[] =
 {   /* Add new patch number below this line */
+/**/
+    495,
 /**/
     494,
 /**/
index b6252612a0c06c815cc944f827af0757dbb1e658..22dfaaa52deab00e81f48a1a87d60732ba038fb2 100644 (file)
@@ -2671,7 +2671,7 @@ execute_endloop(isn_T *iptr, ectx_T *ectx)
     {
        partial_T   *pt = ((partial_T **)gap->ga_data)[idx];
 
-       if (pt->pt_refcount > 1)
+       if (pt->pt_refcount > 1 && pt->pt_loopvars == NULL)
        {
            int refcount = pt->pt_refcount;
            int i;
@@ -2727,7 +2727,7 @@ execute_endloop(isn_T *iptr, ectx_T *ectx)
     {
        partial_T   *pt = ((partial_T **)gap->ga_data)[idx];
 
-       if (pt->pt_refcount > 1)
+       if (pt->pt_refcount > 1 && pt->pt_loopvars == NULL)
        {
            ++loopvars->lvs_refcount;
            pt->pt_loopvars = loopvars;