mirror of
https://github.com/vim/vim.git
synced 2025-09-28 04:24:06 -04:00
patch 7.4.1331
Problem: Crash when closing the channel in a callback. (Christian J. Robinson) Solution: Take the callback out of the list before invoking it.
This commit is contained in:
@@ -888,8 +888,7 @@ channel_parse_json(channel_T *channel)
|
|||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Remove "node" from the queue that it is in and free it.
|
* Remove "node" from the queue that it is in. Does not free it.
|
||||||
* Also frees the contained callback name.
|
|
||||||
*/
|
*/
|
||||||
static void
|
static void
|
||||||
remove_cb_node(cbq_T *head, cbq_T *node)
|
remove_cb_node(cbq_T *head, cbq_T *node)
|
||||||
@@ -902,8 +901,6 @@ remove_cb_node(cbq_T *head, cbq_T *node)
|
|||||||
head->cq_prev = node->cq_prev;
|
head->cq_prev = node->cq_prev;
|
||||||
else
|
else
|
||||||
node->cq_next->cq_prev = node->cq_prev;
|
node->cq_next->cq_prev = node->cq_prev;
|
||||||
vim_free(node->cq_callback);
|
|
||||||
vim_free(node);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@@ -1144,8 +1141,12 @@ may_invoke_callback(channel_T *channel)
|
|||||||
if (item->cq_seq_nr == seq_nr)
|
if (item->cq_seq_nr == seq_nr)
|
||||||
{
|
{
|
||||||
ch_log(channel, "Invoking one-time callback\n");
|
ch_log(channel, "Invoking one-time callback\n");
|
||||||
invoke_callback(channel, item->cq_callback, argv);
|
/* Remove the item from the list first, if the callback
|
||||||
|
* invokes ch_close() the list will be cleared. */
|
||||||
remove_cb_node(head, item);
|
remove_cb_node(head, item);
|
||||||
|
invoke_callback(channel, item->cq_callback, argv);
|
||||||
|
vim_free(item->cq_callback);
|
||||||
|
vim_free(item);
|
||||||
done = TRUE;
|
done = TRUE;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
@@ -1329,7 +1330,13 @@ channel_clear(channel_T *channel)
|
|||||||
vim_free(channel_get(channel));
|
vim_free(channel_get(channel));
|
||||||
|
|
||||||
while (cb_head->cq_next != NULL)
|
while (cb_head->cq_next != NULL)
|
||||||
remove_cb_node(cb_head, cb_head->cq_next);
|
{
|
||||||
|
cbq_T *node = cb_head->cq_next;
|
||||||
|
|
||||||
|
remove_cb_node(cb_head, node);
|
||||||
|
vim_free(node->cq_callback);
|
||||||
|
vim_free(node);
|
||||||
|
}
|
||||||
|
|
||||||
while (json_head->jq_next != NULL)
|
while (json_head->jq_next != NULL)
|
||||||
{
|
{
|
||||||
|
@@ -301,6 +301,8 @@ func Test_pipe()
|
|||||||
endtry
|
endtry
|
||||||
endfunc
|
endfunc
|
||||||
|
|
||||||
|
""""""""""
|
||||||
|
|
||||||
let s:unletResponse = ''
|
let s:unletResponse = ''
|
||||||
func s:UnletHandler(handle, msg)
|
func s:UnletHandler(handle, msg)
|
||||||
let s:unletResponse = a:msg
|
let s:unletResponse = a:msg
|
||||||
@@ -319,6 +321,28 @@ func Test_unlet_handle()
|
|||||||
call s:run_server('s:unlet_handle')
|
call s:run_server('s:unlet_handle')
|
||||||
endfunc
|
endfunc
|
||||||
|
|
||||||
|
""""""""""
|
||||||
|
|
||||||
|
let s:unletResponse = ''
|
||||||
|
func s:CloseHandler(handle, msg)
|
||||||
|
let s:unletResponse = a:msg
|
||||||
|
call ch_close(s:channelfd)
|
||||||
|
endfunc
|
||||||
|
|
||||||
|
" Test that "unlet handle" in a handler doesn't crash Vim.
|
||||||
|
func s:close_handle(port)
|
||||||
|
let s:channelfd = ch_open('localhost:' . a:port, s:chopt)
|
||||||
|
call ch_sendexpr(s:channelfd, "test", function('s:CloseHandler'))
|
||||||
|
sleep 10m
|
||||||
|
call assert_equal('what?', s:unletResponse)
|
||||||
|
endfunc
|
||||||
|
|
||||||
|
func Test_close_handle()
|
||||||
|
call s:run_server('s:close_handle')
|
||||||
|
endfunc
|
||||||
|
|
||||||
|
""""""""""
|
||||||
|
|
||||||
func Test_open_fail()
|
func Test_open_fail()
|
||||||
silent! let ch = ch_open("noserver")
|
silent! let ch = ch_open("noserver")
|
||||||
echo ch
|
echo ch
|
||||||
|
@@ -747,6 +747,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 */
|
||||||
|
/**/
|
||||||
|
1331,
|
||||||
/**/
|
/**/
|
||||||
1330,
|
1330,
|
||||||
/**/
|
/**/
|
||||||
|
Reference in New Issue
Block a user