From: Bram Moolenaar Date: Tue, 13 Sep 2022 20:10:45 +0000 (+0100) Subject: patch 9.0.0459: Vim9: block in for loop doesn't behave like a code block X-Git-Tag: v9.0.0459 X-Git-Url: https://granicus.if.org/sourcecode?a=commitdiff_plain;h=353b68a99189875a8460124d44fc33eae6def74e;p=vim patch 9.0.0459: Vim9: block in for loop doesn't behave like a code block Problem: Vim9: block in for loop doesn't behave like a code block. Solution: Use a new block ID for each loop at the script level. --- diff --git a/src/ex_eval.c b/src/ex_eval.c index 2f31b3e1c..4315c2a62 100644 --- a/src/ex_eval.c +++ b/src/ex_eval.c @@ -1230,15 +1230,18 @@ ex_while(exarg_T *eap) { scriptitem_T *si = SCRIPT_ITEM(current_sctx.sc_sid); int i; + int first; int func_defined = cstack->cs_flags[cstack->cs_idx] & CSF_FUNC_DEF; // Any variables defined in the previous round are no longer // visible. Keep the first one for ":for", it is the loop // variable that we reuse every time around. - for (i = cstack->cs_script_var_len[cstack->cs_idx] + // Do this backwards, so that vars defined in a later round are + // found first. + first = cstack->cs_script_var_len[cstack->cs_idx] + (eap->cmdidx == CMD_while ? 0 : 1); - i < si->sn_var_vals.ga_len; ++i) + for (i = si->sn_var_vals.ga_len - 1; i >= first; --i) { svar_T *sv = ((svar_T *)si->sn_var_vals.ga_data) + i; @@ -1250,6 +1253,12 @@ ex_while(exarg_T *eap) // still exists, from sn_vars. hide_script_var(si, i, func_defined); } + + // Start a new block ID, so that variables defined inside the + // loop are created new and not shared with the previous loop. + // Matters when used in a closure. + cstack->cs_block_id[cstack->cs_idx] = ++si->sn_last_block_id; + si->sn_current_block_id = si->sn_last_block_id; } } cstack->cs_flags[cstack->cs_idx] = diff --git a/src/testdir/test_vim9_script.vim b/src/testdir/test_vim9_script.vim index f746ca178..fa9bb0cc2 100644 --- a/src/testdir/test_vim9_script.vim +++ b/src/testdir/test_vim9_script.vim @@ -2266,10 +2266,12 @@ def Test_for_loop_with_closure() flist[i] = () => inloop endfor for i in range(5) - assert_equal(4, flist[i]()) + assert_equal(i, flist[i]()) endfor END - v9.CheckDefAndScriptSuccess(lines) + # FIXME + # v9.CheckDefAndScriptSuccess(lines) + v9.CheckScriptSuccess(['vim9script'] + lines) lines =<< trim END var flist: list @@ -2280,10 +2282,12 @@ def Test_for_loop_with_closure() } endfor for i in range(5) - assert_equal(4, flist[i]()) + assert_equal(i, flist[i]()) endfor END - v9.CheckDefAndScriptSuccess(lines) + # FIXME + # v9.CheckDefAndScriptSuccess(lines) + v9.CheckScriptSuccess(['vim9script'] + lines) enddef def Test_for_loop_fails() diff --git a/src/version.c b/src/version.c index 25c28f847..dede808bb 100644 --- a/src/version.c +++ b/src/version.c @@ -703,6 +703,8 @@ static char *(features[]) = static int included_patches[] = { /* Add new patch number below this line */ +/**/ + 459, /**/ 458, /**/