# HG changeset patch # User Bram Moolenaar # Date 1627229702 -7200 # Node ID d52504ef26edc6216d9cd284dd4f15fb3e3967dc # Parent 1624e9d9a4d95f9808c66a5619dd9168c29e0954 patch 8.2.3222: Vim9: cannot used loop variable later as lambda argument Commit: https://github.com/vim/vim/commit/3c77b6a1ce1d4a06c60bb9fae7eec2775f547d55 Author: Bram Moolenaar Date: Sun Jul 25 18:07:00 2021 +0200 patch 8.2.3222: Vim9: cannot used loop variable later as lambda argument Problem: Vim9: cannot used loop variable later as lambda argument. Solution: When not in function context check the current block ID. (closes #8637) diff --git a/src/testdir/test_vim9_func.vim b/src/testdir/test_vim9_func.vim --- a/src/testdir/test_vim9_func.vim +++ b/src/testdir/test_vim9_func.vim @@ -2352,7 +2352,7 @@ def Test_list_lambda() assert_match('def \d\+(_: any): number\n1 return 0\n enddef', body) enddef -def Test_lamba_block_variable() +def Test_lambda_block_variable() var lines =<< trim END vim9script var flist: list @@ -2386,6 +2386,15 @@ def Test_lamba_block_variable() endfor END CheckScriptFailure(lines, 'E1001: Variable not found: outloop', 1) + + lines =<< trim END + vim9script + for i in range(10) + var Ref = () => 0 + endfor + assert_equal(0, ((i) => 0)(0)) + END + CheckScriptSuccess(lines) enddef def Test_legacy_lambda() diff --git a/src/version.c b/src/version.c --- a/src/version.c +++ b/src/version.c @@ -756,6 +756,8 @@ static char *(features[]) = static int included_patches[] = { /* Add new patch number below this line */ /**/ + 3222, +/**/ 3221, /**/ 3220, diff --git a/src/vim9compile.c b/src/vim9compile.c --- a/src/vim9compile.c +++ b/src/vim9compile.c @@ -355,10 +355,23 @@ find_script_var(char_u *name, size_t len return NULL; sav = HI2SAV(hi); - if (sav->sav_block_id == 0 || cctx == NULL) - // variable defined in the script scope or not in a function. + if (sav->sav_block_id == 0) + // variable defined in the top script scope is always visible return sav; + if (cctx == NULL) + { + // Not in a function scope, find variable with block id equal to or + // smaller than the current block id. + while (sav != NULL) + { + if (sav->sav_block_id <= si->sn_current_block_id) + break; + sav = sav->sav_next; + } + return sav; + } + // Go over the variables with this name and find one that was visible // from the function. ufunc = cctx->ctx_ufunc;