aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorBjörn Linse <bjorn.linse@gmail.com>2021-11-11 10:57:16 +0100
committerGitHub <noreply@github.com>2021-11-11 10:57:16 +0100
commit5d653a134464fe06c488dd67071359b04e90949e (patch)
tree0ea641168deeac2a84f424a0fafcd5cb0ad90673
parent0ecc58c2775ae576a3325aec163aa1018aad9e0a (diff)
parent8b5c32c8cd9b76aaefa0977a69789fa9bd43f2fd (diff)
downloadrneovim-5d653a134464fe06c488dd67071359b04e90949e.tar.gz
rneovim-5d653a134464fe06c488dd67071359b04e90949e.tar.bz2
rneovim-5d653a134464fe06c488dd67071359b04e90949e.zip
Merge pull request #16276 from zeertzjq/channel-closed-term-error
Fixes and tests for sending to terminal channel whose terminal has been deleted
-rw-r--r--src/nvim/api/vim.c3
-rw-r--r--src/nvim/channel.c6
-rw-r--r--test/functional/terminal/channel_spec.lua49
3 files changed, 56 insertions, 2 deletions
diff --git a/src/nvim/api/vim.c b/src/nvim/api/vim.c
index 2fbafb9889..c1374ff00e 100644
--- a/src/nvim/api/vim.c
+++ b/src/nvim/api/vim.c
@@ -1006,7 +1006,6 @@ Integer nvim_open_term(Buffer buffer, DictionaryOf(LuaRef) opts, Error *err)
Terminal *term = terminal_open(buf, topts);
terminal_check_size(term);
chan->term = term;
- channel_incref(chan);
return (Integer)chan->id;
}
@@ -1036,6 +1035,8 @@ static void term_close(void *data)
Channel *chan = data;
terminal_destroy(chan->term);
chan->term = NULL;
+ api_free_luaref(chan->stream.internal.cb);
+ chan->stream.internal.cb = LUA_NOREF;
channel_decref(chan);
}
diff --git a/src/nvim/channel.c b/src/nvim/channel.c
index 0eae87c1f8..9662f6205f 100644
--- a/src/nvim/channel.c
+++ b/src/nvim/channel.c
@@ -535,7 +535,11 @@ size_t channel_send(uint64_t id, char *data, size_t len, bool data_owned, const
goto retfree;
}
- if (chan->streamtype == kChannelStreamInternal && chan->term) {
+ if (chan->streamtype == kChannelStreamInternal) {
+ if (!chan->term) {
+ *error = _("Can't send data to closed stream");
+ goto retfree;
+ }
terminal_receive(chan->term, data, len);
written = len;
goto retfree;
diff --git a/test/functional/terminal/channel_spec.lua b/test/functional/terminal/channel_spec.lua
new file mode 100644
index 0000000000..7d37dcccc8
--- /dev/null
+++ b/test/functional/terminal/channel_spec.lua
@@ -0,0 +1,49 @@
+local helpers = require('test.functional.helpers')(after_each)
+local clear = helpers.clear
+local eq = helpers.eq
+local command = helpers.command
+local exc_exec = helpers.exc_exec
+local feed = helpers.feed
+local sleep = helpers.sleep
+local poke_eventloop = helpers.poke_eventloop
+
+describe('associated channel is closed and later freed for terminal', function()
+ before_each(clear)
+
+ it('opened by nvim_open_term() and deleted by :bdelete!', function()
+ command([[let id = nvim_open_term(0, {})]])
+ -- channel hasn't been freed yet
+ eq("Vim(call):Can't send data to closed stream", exc_exec([[bdelete! | call chansend(id, 'test')]]))
+ -- process free_channel_event
+ poke_eventloop()
+ -- channel has been freed
+ eq("Vim(call):E900: Invalid channel id", exc_exec([[call chansend(id, 'test')]]))
+ end)
+
+ it('opened by termopen(), exited, and deleted by pressing a key', function()
+ command([[let id = termopen('echo')]])
+ sleep(500)
+ -- process has exited
+ eq("Vim(call):Can't send data to closed stream", exc_exec([[call chansend(id, 'test')]]))
+ -- delete terminal
+ feed('i<CR>')
+ -- process term_delayed_free and free_channel_event
+ poke_eventloop()
+ -- channel has been freed
+ eq("Vim(call):E900: Invalid channel id", exc_exec([[call chansend(id, 'test')]]))
+ end)
+
+ -- This indirectly covers #16264
+ it('opened by termopen(), exited, and deleted by :bdelete', function()
+ command([[let id = termopen('echo')]])
+ sleep(500)
+ -- process has exited
+ eq("Vim(call):Can't send data to closed stream", exc_exec([[call chansend(id, 'test')]]))
+ -- channel hasn't been freed yet
+ eq("Vim(call):Can't send data to closed stream", exc_exec([[bdelete | call chansend(id, 'test')]]))
+ -- process term_delayed_free and free_channel_event
+ poke_eventloop()
+ -- channel has been freed
+ eq("Vim(call):E900: Invalid channel id", exc_exec([[call chansend(id, 'test')]]))
+ end)
+end)