diff options
| author | Michael Ennen <mike.ennen@gmail.com> | 2016-12-17 22:33:03 -0700 | 
|---|---|---|
| committer | Michael Ennen <mike.ennen@gmail.com> | 2017-02-14 17:38:18 -0700 | 
| commit | 1f715ac1c1a1eee43360be911636020ed855e12c (patch) | |
| tree | b7e7fdb6df01a0e32bba4f9defa2e701168b2661 /src | |
| parent | 1e3c0efa0f3d41563f91097d04b3616848d5eb62 (diff) | |
| download | rneovim-1f715ac1c1a1eee43360be911636020ed855e12c.tar.gz rneovim-1f715ac1c1a1eee43360be911636020ed855e12c.tar.bz2 rneovim-1f715ac1c1a1eee43360be911636020ed855e12c.zip | |
vim-patch:7.4.2136
Problem:    Closure function fails.
Solution:   Don't reset uf_scoped when it points to another funccal.
https://github.com/vim/vim/commit/580164481924ed8611eb79f0247a0eb1ca0b3b9a
Diffstat (limited to 'src')
| -rw-r--r-- | src/nvim/eval.c | 70 | ||||
| -rw-r--r-- | src/nvim/testdir/test_lambda.vim | 4 | ||||
| -rw-r--r-- | src/nvim/version.c | 2 | 
3 files changed, 36 insertions, 40 deletions
| diff --git a/src/nvim/eval.c b/src/nvim/eval.c index d1b60d67b6..192afa0708 100644 --- a/src/nvim/eval.c +++ b/src/nvim/eval.c @@ -7018,6 +7018,18 @@ err_ret:    return FAIL;  } +/// Register function "fp" as using "current_funccal" as its scope. +static int register_closure(ufunc_T *fp) { +  funccal_unref(fp->uf_scoped, NULL); +  fp->uf_scoped = current_funccal; +  current_funccal->fc_refcount++; +  ga_grow(¤t_funccal->fc_funcs, 1); +  ((ufunc_T **)current_funccal->fc_funcs.ga_data) +    [current_funccal->fc_funcs.ga_len++] = fp; +  func_ref(current_funccal->func->uf_name); +  return OK; +} +  /// Parse a lambda expression and get a Funcref from "*arg".  ///  /// @return OK or FAIL.  Returns NOTDONE for dict or {expr}. @@ -7083,7 +7095,7 @@ static int get_lambda_tv(char_u **arg, typval_T *rettv, int evaluate)      snprintf((char *)name, sizeof(name), "<lambda>%d", lambda_no++); -    fp = (ufunc_T *)xmalloc((unsigned)(sizeof(ufunc_T) + STRLEN(name))); +    fp = (ufunc_T *)xcalloc(1, (unsigned)(sizeof(ufunc_T) + STRLEN(name)));      if (fp == NULL) {        goto errret;      } @@ -7109,12 +7121,7 @@ static int get_lambda_tv(char_u **arg, typval_T *rettv, int evaluate)      fp->uf_lines = newlines;      if (current_funccal != NULL && eval_lavars) {        flags |= FC_CLOSURE; -      fp->uf_scoped = current_funccal; -      current_funccal->fc_refcount++; -      ga_grow(¤t_funccal->fc_funcs, 1); -      ((ufunc_T **)current_funccal->fc_funcs.ga_data) -       [current_funccal->fc_funcs.ga_len++] = fp; -      func_ref(current_funccal->func->uf_name); +      register_closure(fp);      } else {        fp->uf_scoped = NULL;      } @@ -21076,6 +21083,12 @@ void ex_function(exarg_T *eap)      } else if (STRNCMP(p, "closure", 7) == 0) {        flags |= FC_CLOSURE;        p += 7; +      if (current_funccal == NULL) { +        emsg_funcname(N_ +                      ("E932 Closure function should not be at top level: %s"), +                      name == NULL ? (char_u *)"" : name); +        goto erret; +      }      } else {        break;      } @@ -21329,7 +21342,7 @@ void ex_function(exarg_T *eap)        }      } -    fp = xmalloc(sizeof(ufunc_T) + STRLEN(name)); +    fp = xcalloc(1, sizeof(ufunc_T) + STRLEN(name));      if (fudi.fd_dict != NULL) {        if (fudi.fd_di == NULL) { @@ -21362,17 +21375,7 @@ void ex_function(exarg_T *eap)    fp->uf_args = newargs;    fp->uf_lines = newlines;    if ((flags & FC_CLOSURE) != 0) { -    if (current_funccal == NULL) { -      emsg_funcname(N_("E932 Closure function should not be at top level: %s"), -                    name); -      goto erret; -    } -    fp->uf_scoped = current_funccal; -    current_funccal->fc_refcount++; -    ga_grow(¤t_funccal->fc_funcs, 1); -    ((ufunc_T **)current_funccal->fc_funcs.ga_data) -      [current_funccal->fc_funcs.ga_len++] = fp; -    func_ref(current_funccal->func->uf_name); +    register_closure(fp);    } else {      fp->uf_scoped = NULL;    } @@ -22543,8 +22546,8 @@ call_user_func (    current_funccal = fc->caller;    --depth; -  /* If the a:000 list and the l: and a: dicts are not referenced we can -   * free the funccall_T and what's in it. */ +  // If the a:000 list and the l: and a: dicts are not referenced and there +  // is no closure using it, we can free the funccall_T and what's in it.    if (fc->l_varlist.lv_refcount == DO_NOT_FREE_CNT        && fc->l_vars.dv_refcount == DO_NOT_FREE_CNT        && fc->l_avars.dv_refcount == DO_NOT_FREE_CNT @@ -22555,9 +22558,9 @@ call_user_func (      listitem_T      *li;      int todo; -    /* "fc" is still in use.  This can happen when returning "a:000" or -     * assigning "l:" to a global variable. -     * Link "fc" in the list for garbage collection later. */ +    // "fc" is still in use.  This can happen when returning "a:000", +    // assigning "l:" to a global variable or defining a closure. +    // Link "fc" in the list for garbage collection later.      fc->caller = previous_funccal;      previous_funccal = fc; @@ -22653,9 +22656,15 @@ free_funccal (      ufunc_T *fp = ((ufunc_T **)(fc->fc_funcs.ga_data))[i];      if (fp != NULL) { -      fp->uf_scoped = NULL; +      // Function may have been redefined and point to another +      // funccall_T, don't clear it then. +      if (fp->uf_scoped == fc) { +        fp->uf_scoped = NULL; +      } +      func_unref(fc->func->uf_name);      }    } +  ga_clear(&fc->fc_funcs):    // The a: variables typevals may not have been allocated, only free the    // allocated variables. @@ -22671,15 +22680,6 @@ free_funccal (      }    } -  for (int i = 0; i < fc->fc_funcs.ga_len; i++) { -    ufunc_T *fp = ((ufunc_T **)(fc->fc_funcs.ga_data))[i]; - -    if (fp != NULL) { -      func_unref(fc->func->uf_name); -    } -  } -  ga_clear(&fc->fc_funcs); -    func_unref(fc->func->uf_name);    xfree(fc);  } @@ -23013,7 +23013,7 @@ hashitem_T *find_hi_in_scoped_ht(char_u *name, char_u **varname,    // Search in parent scope which is possible to reference from lambda    current_funccal = current_funccal->func->uf_scoped; -  while (current_funccal) { +  while (current_funccal != NULL) {      ht = find_var_ht(name, varname);      if (ht != NULL && **varname != NUL) {        hi = hash_find(ht, *varname); diff --git a/src/nvim/testdir/test_lambda.vim b/src/nvim/testdir/test_lambda.vim index 98d8ffab40..d51b6f7c5a 100644 --- a/src/nvim/testdir/test_lambda.vim +++ b/src/nvim/testdir/test_lambda.vim @@ -284,7 +284,3 @@ func Test_named_function_closure()    call garbagecollect()    call assert_equal(14, s:Abar())  endfunc -======= ->>>>>>> 42b34811... vim-patch:7.4.2119 -======= ->>>>>>> 789a3319... vim-patch:7.4.2120 diff --git a/src/nvim/version.c b/src/nvim/version.c index 9134ee04ff..781753112e 100644 --- a/src/nvim/version.c +++ b/src/nvim/version.c @@ -304,7 +304,7 @@ static int included_patches[] = {    // 2139,    // 2138 NA    // 2137, -  // 2136, +  2136,    // 2135,    2134,    // 2133 NA | 
