mirror of
https://github.com/vim/vim.git
synced 2025-09-26 04:04:07 -04:00
patch 8.1.1579: dict and list could be GC'ed while displaying error
Problem: Dict and list could be GC'ed while displaying error in a timer. (Yasuhiro Matsumoto) Solution: Block garbage collection when executing a timer. Add test_garbagecollect_soon(). Add "no_wait_return" to test_override(). (closes #4571)
This commit is contained in:
@@ -2741,6 +2741,7 @@ test_alloc_fail({id}, {countdown}, {repeat})
|
|||||||
test_autochdir() none enable 'autochdir' during startup
|
test_autochdir() none enable 'autochdir' during startup
|
||||||
test_feedinput({string}) none add key sequence to input buffer
|
test_feedinput({string}) none add key sequence to input buffer
|
||||||
test_garbagecollect_now() none free memory right now for testing
|
test_garbagecollect_now() none free memory right now for testing
|
||||||
|
test_garbagecollect_soon() none free memory soon for testing
|
||||||
test_getvalue({string}) any get value of an internal variable
|
test_getvalue({string}) any get value of an internal variable
|
||||||
test_ignore_error({expr}) none ignore a specific error
|
test_ignore_error({expr}) none ignore a specific error
|
||||||
test_null_blob() Blob null value for testing
|
test_null_blob() Blob null value for testing
|
||||||
@@ -10009,6 +10010,10 @@ test_garbagecollect_now() *test_garbagecollect_now()*
|
|||||||
internally, and |v:testing| must have been set before calling
|
internally, and |v:testing| must have been set before calling
|
||||||
any function.
|
any function.
|
||||||
|
|
||||||
|
test_garbagecollect_soon() *test_garbagecollect_soon()*
|
||||||
|
Set the flag to call the garbagecollector as if in the main
|
||||||
|
loop. Only to be used in tests.
|
||||||
|
|
||||||
test_getvalue({name}) *test_getvalue()*
|
test_getvalue({name}) *test_getvalue()*
|
||||||
Get the value of an internal variable. These values for
|
Get the value of an internal variable. These values for
|
||||||
{name} are supported:
|
{name} are supported:
|
||||||
@@ -10072,6 +10077,8 @@ test_override({name}, {val}) *test_override()*
|
|||||||
fallback to the old engine
|
fallback to the old engine
|
||||||
no_query_mouse do not query the mouse position for "dec"
|
no_query_mouse do not query the mouse position for "dec"
|
||||||
terminals
|
terminals
|
||||||
|
no_wait_return set the "no_wait_return" flag. Not restored
|
||||||
|
with "ALL".
|
||||||
ALL clear all overrides ({val} is not used)
|
ALL clear all overrides ({val} is not used)
|
||||||
|
|
||||||
"starting" is to be used when a test should behave like
|
"starting" is to be used when a test should behave like
|
||||||
|
@@ -28,7 +28,7 @@ dict_alloc(void)
|
|||||||
{
|
{
|
||||||
dict_T *d;
|
dict_T *d;
|
||||||
|
|
||||||
d = ALLOC_ONE(dict_T);
|
d = ALLOC_CLEAR_ONE(dict_T);
|
||||||
if (d != NULL)
|
if (d != NULL)
|
||||||
{
|
{
|
||||||
/* Add the dict to the list of dicts for garbage collection. */
|
/* Add the dict to the list of dicts for garbage collection. */
|
||||||
@@ -811,7 +811,7 @@ dict_get_tv(char_u **arg, typval_T *rettv, int evaluate)
|
|||||||
{
|
{
|
||||||
semsg(_("E723: Missing end of Dictionary '}': %s"), *arg);
|
semsg(_("E723: Missing end of Dictionary '}': %s"), *arg);
|
||||||
failret:
|
failret:
|
||||||
if (evaluate)
|
if (d != NULL)
|
||||||
dict_free(d);
|
dict_free(d);
|
||||||
return FAIL;
|
return FAIL;
|
||||||
}
|
}
|
||||||
|
@@ -448,6 +448,7 @@ static void f_test_option_not_set(typval_T *argvars, typval_T *rettv);
|
|||||||
static void f_test_override(typval_T *argvars, typval_T *rettv);
|
static void f_test_override(typval_T *argvars, typval_T *rettv);
|
||||||
static void f_test_refcount(typval_T *argvars, typval_T *rettv);
|
static void f_test_refcount(typval_T *argvars, typval_T *rettv);
|
||||||
static void f_test_garbagecollect_now(typval_T *argvars, typval_T *rettv);
|
static void f_test_garbagecollect_now(typval_T *argvars, typval_T *rettv);
|
||||||
|
static void f_test_garbagecollect_soon(typval_T *argvars, typval_T *rettv);
|
||||||
static void f_test_ignore_error(typval_T *argvars, typval_T *rettv);
|
static void f_test_ignore_error(typval_T *argvars, typval_T *rettv);
|
||||||
static void f_test_null_blob(typval_T *argvars, typval_T *rettv);
|
static void f_test_null_blob(typval_T *argvars, typval_T *rettv);
|
||||||
#ifdef FEAT_JOB_CHANNEL
|
#ifdef FEAT_JOB_CHANNEL
|
||||||
@@ -1019,6 +1020,7 @@ static struct fst
|
|||||||
{"test_autochdir", 0, 0, f_test_autochdir},
|
{"test_autochdir", 0, 0, f_test_autochdir},
|
||||||
{"test_feedinput", 1, 1, f_test_feedinput},
|
{"test_feedinput", 1, 1, f_test_feedinput},
|
||||||
{"test_garbagecollect_now", 0, 0, f_test_garbagecollect_now},
|
{"test_garbagecollect_now", 0, 0, f_test_garbagecollect_now},
|
||||||
|
{"test_garbagecollect_soon", 0, 0, f_test_garbagecollect_soon},
|
||||||
{"test_getvalue", 1, 1, f_test_getvalue},
|
{"test_getvalue", 1, 1, f_test_getvalue},
|
||||||
{"test_ignore_error", 1, 1, f_test_ignore_error},
|
{"test_ignore_error", 1, 1, f_test_ignore_error},
|
||||||
{"test_null_blob", 0, 0, f_test_null_blob},
|
{"test_null_blob", 0, 0, f_test_null_blob},
|
||||||
@@ -14460,6 +14462,8 @@ f_test_override(typval_T *argvars, typval_T *rettv UNUSED)
|
|||||||
nfa_fail_for_testing = val;
|
nfa_fail_for_testing = val;
|
||||||
else if (STRCMP(name, (char_u *)"no_query_mouse") == 0)
|
else if (STRCMP(name, (char_u *)"no_query_mouse") == 0)
|
||||||
no_query_mouse_for_testing = val;
|
no_query_mouse_for_testing = val;
|
||||||
|
else if (STRCMP(name, (char_u *)"no_wait_return") == 0)
|
||||||
|
no_wait_return = val;
|
||||||
else if (STRCMP(name, (char_u *)"ALL") == 0)
|
else if (STRCMP(name, (char_u *)"ALL") == 0)
|
||||||
{
|
{
|
||||||
disable_char_avail_for_testing = FALSE;
|
disable_char_avail_for_testing = FALSE;
|
||||||
@@ -14550,6 +14554,15 @@ f_test_garbagecollect_now(typval_T *argvars UNUSED, typval_T *rettv UNUSED)
|
|||||||
garbage_collect(TRUE);
|
garbage_collect(TRUE);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* "test_garbagecollect_soon()" function
|
||||||
|
*/
|
||||||
|
static void
|
||||||
|
f_test_garbagecollect_soon(typval_T *argvars UNUSED, typval_T *rettv UNUSED)
|
||||||
|
{
|
||||||
|
may_garbage_collect = TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* "test_ignore_error()" function
|
* "test_ignore_error()" function
|
||||||
*/
|
*/
|
||||||
|
@@ -309,4 +309,28 @@ func Test_restore_count()
|
|||||||
call delete('Xtrctext')
|
call delete('Xtrctext')
|
||||||
endfunc
|
endfunc
|
||||||
|
|
||||||
|
" Test that the garbage collector isn't triggered if a timer callback invokes
|
||||||
|
" vgetc().
|
||||||
|
func Test_nocatch_garbage_collect()
|
||||||
|
" 'uptimetime. must be bigger than the timer timeout
|
||||||
|
set ut=200
|
||||||
|
call test_garbagecollect_soon()
|
||||||
|
call test_override('no_wait_return', 0)
|
||||||
|
func CauseAnError(id)
|
||||||
|
" This will show an error and wait for Enter.
|
||||||
|
let a = {'foo', 'bar'}
|
||||||
|
endfunc
|
||||||
|
func FeedChar(id)
|
||||||
|
call feedkeys('x', 't')
|
||||||
|
endfunc
|
||||||
|
call timer_start(300, 'FeedChar')
|
||||||
|
call timer_start(100, 'CauseAnError')
|
||||||
|
let x = getchar()
|
||||||
|
|
||||||
|
set ut&
|
||||||
|
call test_override('no_wait_return', 1)
|
||||||
|
delfunc CauseAnError
|
||||||
|
delfunc FeedChar
|
||||||
|
endfunc
|
||||||
|
|
||||||
" vim: shiftwidth=2 sts=2 expandtab
|
" vim: shiftwidth=2 sts=2 expandtab
|
||||||
|
@@ -777,6 +777,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 */
|
||||||
|
/**/
|
||||||
|
1579,
|
||||||
/**/
|
/**/
|
||||||
1578,
|
1578,
|
||||||
/**/
|
/**/
|
||||||
|
Reference in New Issue
Block a user