forked from aniani/vim
patch 7.4.1860
Problem: Using a partial for timer_start() may cause a crash. Solution: Set the copyID in timer objects. (Ozaki Kiichi)
This commit is contained in:
@@ -7046,6 +7046,10 @@ garbage_collect(int testing)
|
|||||||
abort = abort || set_ref_in_nb_channel(copyID);
|
abort = abort || set_ref_in_nb_channel(copyID);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#ifdef FEAT_TIMERS
|
||||||
|
abort = abort || set_ref_in_timer(copyID);
|
||||||
|
#endif
|
||||||
|
|
||||||
if (!abort)
|
if (!abort)
|
||||||
{
|
{
|
||||||
/*
|
/*
|
||||||
|
@@ -1252,6 +1252,25 @@ stop_timer(timer_T *timer)
|
|||||||
remove_timer(timer);
|
remove_timer(timer);
|
||||||
free_timer(timer);
|
free_timer(timer);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Mark references in partials of timers.
|
||||||
|
*/
|
||||||
|
int
|
||||||
|
set_ref_in_timer(int copyID)
|
||||||
|
{
|
||||||
|
int abort = FALSE;
|
||||||
|
timer_T *timer;
|
||||||
|
typval_T tv;
|
||||||
|
|
||||||
|
for (timer = first_timer; timer != NULL; timer = timer->tr_next)
|
||||||
|
{
|
||||||
|
tv.v_type = VAR_PARTIAL;
|
||||||
|
tv.vval.v_partial = timer->tr_partial;
|
||||||
|
abort = abort || set_ref_in_item(&tv, copyID, NULL, NULL);
|
||||||
|
}
|
||||||
|
return abort;
|
||||||
|
}
|
||||||
# endif
|
# endif
|
||||||
|
|
||||||
#if defined(FEAT_SYN_HL) && defined(FEAT_RELTIME) && defined(FEAT_FLOAT)
|
#if defined(FEAT_SYN_HL) && defined(FEAT_RELTIME) && defined(FEAT_FLOAT)
|
||||||
|
@@ -22,6 +22,7 @@ timer_T *create_timer(long msec, int repeats);
|
|||||||
long check_due_timer(void);
|
long check_due_timer(void);
|
||||||
timer_T *find_timer(int id);
|
timer_T *find_timer(int id);
|
||||||
void stop_timer(timer_T *timer);
|
void stop_timer(timer_T *timer);
|
||||||
|
int set_ref_in_timer(int copyID);
|
||||||
void profile_divide(proftime_T *tm, int count, proftime_T *tm2);
|
void profile_divide(proftime_T *tm, int count, proftime_T *tm2);
|
||||||
void profile_add(proftime_T *tm, proftime_T *tm2);
|
void profile_add(proftime_T *tm, proftime_T *tm2);
|
||||||
void profile_self(proftime_T *self, proftime_T *total, proftime_T *children);
|
void profile_self(proftime_T *self, proftime_T *total, proftime_T *children);
|
||||||
|
@@ -8,6 +8,10 @@ func MyHandler(timer)
|
|||||||
let s:val += 1
|
let s:val += 1
|
||||||
endfunc
|
endfunc
|
||||||
|
|
||||||
|
func MyHandlerWithLists(lists, timer)
|
||||||
|
let x = string(a:lists)
|
||||||
|
endfunc
|
||||||
|
|
||||||
func Test_oneshot()
|
func Test_oneshot()
|
||||||
let s:val = 0
|
let s:val = 0
|
||||||
let timer = timer_start(50, 'MyHandler')
|
let timer = timer_start(50, 'MyHandler')
|
||||||
@@ -42,4 +46,10 @@ func Test_with_partial_callback()
|
|||||||
sleep 200m
|
sleep 200m
|
||||||
call assert_equal(1, s:val)
|
call assert_equal(1, s:val)
|
||||||
endfunc
|
endfunc
|
||||||
|
|
||||||
|
func Test_retain_partial()
|
||||||
|
call timer_start(100, function('MyHandlerWithLists', [['a']]))
|
||||||
|
call test_garbagecollect_now()
|
||||||
|
sleep 200m
|
||||||
|
endfunc
|
||||||
" vim: ts=2 sw=0 et
|
" vim: ts=2 sw=0 et
|
||||||
|
@@ -753,6 +753,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 */
|
||||||
|
/**/
|
||||||
|
1860,
|
||||||
/**/
|
/**/
|
||||||
1859,
|
1859,
|
||||||
/**/
|
/**/
|
||||||
|
Reference in New Issue
Block a user