1
0
forked from aniani/vim

patch 7.4.2136

Problem:    Closure function fails.
Solution:   Don't reset uf_scoped when it points to another funccal.
This commit is contained in:
Bram Moolenaar
2016-07-31 18:30:22 +02:00
parent 89eaa4185e
commit 5801644819
3 changed files with 81 additions and 50 deletions

View File

@@ -247,3 +247,27 @@ function! Test_closure_unlet()
call assert_false(has_key(s:foo(), 'x')) call assert_false(has_key(s:foo(), 'x'))
call test_garbagecollect_now() call test_garbagecollect_now()
endfunction endfunction
function! LambdaFoo()
let x = 0
function! LambdaBar() closure
let x += 1
return x
endfunction
return function('LambdaBar')
endfunction
func Test_closure_refcount()
let g:Count = LambdaFoo()
call test_garbagecollect_now()
call assert_equal(1, g:Count())
let g:Count2 = LambdaFoo()
call test_garbagecollect_now()
call assert_equal(1, g:Count2())
call assert_equal(2, g:Count())
call assert_equal(3, g:Count2())
" This causes memory access errors.
" delfunc LambdaFoo
" delfunc LambdaBar
endfunc

View File

@@ -150,6 +150,7 @@ static int
# endif # endif
prof_self_cmp(const void *s1, const void *s2); prof_self_cmp(const void *s1, const void *s2);
#endif #endif
static void funccal_unref(funccall_T *fc, ufunc_T *fp);
void void
func_init() func_init()
@@ -257,6 +258,23 @@ err_ret:
return FAIL; 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++;
if (ga_grow(&current_funccal->fc_funcs, 1) == FAIL)
return FAIL;
((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". * Parse a lambda expression and get a Funcref from "*arg".
* Return OK or FAIL. Returns NOTDONE for dict or {expr}. * Return OK or FAIL. Returns NOTDONE for dict or {expr}.
@@ -318,7 +336,7 @@ get_lambda_tv(char_u **arg, typval_T *rettv, int evaluate)
sprintf((char*)name, "<lambda>%d", ++lambda_no); sprintf((char*)name, "<lambda>%d", ++lambda_no);
fp = (ufunc_T *)alloc((unsigned)(sizeof(ufunc_T) + STRLEN(name))); fp = (ufunc_T *)alloc_clear((unsigned)(sizeof(ufunc_T) + STRLEN(name)));
if (fp == NULL) if (fp == NULL)
goto errret; goto errret;
@@ -343,13 +361,8 @@ get_lambda_tv(char_u **arg, typval_T *rettv, int evaluate)
if (current_funccal != NULL && eval_lavars) if (current_funccal != NULL && eval_lavars)
{ {
flags |= FC_CLOSURE; flags |= FC_CLOSURE;
fp->uf_scoped = current_funccal; if (register_closure(fp) == FAIL)
current_funccal->fc_refcount++;
if (ga_grow(&current_funccal->fc_funcs, 1) == FAIL)
goto errret; goto errret;
((ufunc_T **)current_funccal->fc_funcs.ga_data)
[current_funccal->fc_funcs.ga_len++] = fp;
func_ref(current_funccal->func->uf_name);
} }
else else
fp->uf_scoped = NULL; fp->uf_scoped = NULL;
@@ -660,8 +673,15 @@ free_funccal(
ufunc_T *fp = ((ufunc_T **)(fc->fc_funcs.ga_data))[i]; ufunc_T *fp = ((ufunc_T **)(fc->fc_funcs.ga_data))[i];
if (fp != NULL) if (fp != 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; 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 /* The a: variables typevals may not have been allocated, only free the
* allocated variables. */ * allocated variables. */
@@ -675,15 +695,6 @@ free_funccal(
for (li = fc->l_varlist.lv_first; li != NULL; li = li->li_next) for (li = fc->l_varlist.lv_first; li != NULL; li = li->li_next)
clear_tv(&li->li_tv); clear_tv(&li->li_tv);
for (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); func_unref(fc->func->uf_name);
vim_free(fc); vim_free(fc);
} }
@@ -1046,8 +1057,8 @@ call_user_func(
current_funccal = fc->caller; current_funccal = fc->caller;
--depth; --depth;
/* If the a:000 list and the l: and a: dicts are not referenced we can /* If the a:000 list and the l: and a: dicts are not referenced and there
* free the funccall_T and what's in it. */ * 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 if (fc->l_varlist.lv_refcount == DO_NOT_FREE_CNT
&& fc->l_vars.dv_refcount == DO_NOT_FREE_CNT && fc->l_vars.dv_refcount == DO_NOT_FREE_CNT
&& fc->l_avars.dv_refcount == DO_NOT_FREE_CNT && fc->l_avars.dv_refcount == DO_NOT_FREE_CNT
@@ -1061,8 +1072,8 @@ call_user_func(
listitem_T *li; listitem_T *li;
int todo; int todo;
/* "fc" is still in use. This can happen when returning "a:000" or /* "fc" is still in use. This can happen when returning "a:000",
* assigning "l:" to a global variable. * assigning "l:" to a global variable or defining a closure.
* Link "fc" in the list for garbage collection later. */ * Link "fc" in the list for garbage collection later. */
fc->caller = previous_funccal; fc->caller = previous_funccal;
previous_funccal = fc; previous_funccal = fc;
@@ -1121,14 +1132,12 @@ funccal_unref(funccall_T *fc, ufunc_T *fp)
func_unref(fc->func->uf_name); func_unref(fc->func->uf_name);
if (fp != NULL) if (fp != NULL)
{
for (i = 0; i < fc->fc_funcs.ga_len; ++i) for (i = 0; i < fc->fc_funcs.ga_len; ++i)
{ {
if (((ufunc_T **)(fc->fc_funcs.ga_data))[i] == fp) if (((ufunc_T **)(fc->fc_funcs.ga_data))[i] == fp)
((ufunc_T **)(fc->fc_funcs.ga_data))[i] = NULL; ((ufunc_T **)(fc->fc_funcs.ga_data))[i] = NULL;
} }
} }
}
} }
/* /*
@@ -1976,6 +1985,12 @@ ex_function(exarg_T *eap)
{ {
flags |= FC_CLOSURE; flags |= FC_CLOSURE;
p += 7; 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 else
break; break;
@@ -2265,7 +2280,7 @@ ex_function(exarg_T *eap)
} }
} }
fp = (ufunc_T *)alloc((unsigned)(sizeof(ufunc_T) + STRLEN(name))); fp = (ufunc_T *)alloc_clear((unsigned)(sizeof(ufunc_T) + STRLEN(name)));
if (fp == NULL) if (fp == NULL)
goto erret; goto erret;
@@ -2311,20 +2326,10 @@ ex_function(exarg_T *eap)
fp->uf_lines = newlines; fp->uf_lines = newlines;
if ((flags & FC_CLOSURE) != 0) if ((flags & FC_CLOSURE) != 0)
{ {
if (current_funccal == NULL) ++fp->uf_refcount;
{ if (register_closure(fp) == FAIL)
emsg_funcname(N_("E932 Closure function should not be at top level: %s"),
name);
goto erret; goto erret;
} }
fp->uf_scoped = current_funccal;
current_funccal->fc_refcount++;
if (ga_grow(&current_funccal->fc_funcs, 1) == FAIL)
goto erret;
((ufunc_T **)current_funccal->fc_funcs.ga_data)
[current_funccal->fc_funcs.ga_len++] = fp;
func_ref(current_funccal->func->uf_name);
}
else else
fp->uf_scoped = NULL; fp->uf_scoped = NULL;
@@ -3582,7 +3587,7 @@ find_hi_in_scoped_ht(char_u *name, char_u **varname, hashtab_T **pht)
/* Search in parent scope which is possible to reference from lambda */ /* Search in parent scope which is possible to reference from lambda */
current_funccal = current_funccal->func->uf_scoped; current_funccal = current_funccal->func->uf_scoped;
while (current_funccal) while (current_funccal != NULL)
{ {
ht = find_var_ht(name, varname); ht = find_var_ht(name, varname);
if (ht != NULL && **varname != NUL) if (ht != NULL && **varname != NUL)

View File

@@ -763,6 +763,8 @@ static char *(features[]) =
static int included_patches[] = static int included_patches[] =
{ /* Add new patch number below this line */ { /* Add new patch number below this line */
/**/
2136,
/**/ /**/
2135, 2135,
/**/ /**/