neovim

Neovim text editor
git clone https://git.dasho.dev/neovim.git
Log | Files | Refs | README

commit e38c9f734a262e3e7595434ca009f961afcccef2
parent 4d46d040c3215373f8cf64378d03d004e095d646
Author: zeertzjq <zeertzjq@outlook.com>
Date:   Tue,  6 Jan 2026 07:17:37 +0800

Merge pull request #37251 from zeertzjq/termclose

fix(terminal): crash when TermClose deletes other buffers
Diffstat:
Msrc/nvim/buffer.c | 16++++++++++++----
Mtest/functional/autocmd/termxx_spec.lua | 30+++++++++++++++++++++++++++---
2 files changed, 39 insertions(+), 7 deletions(-)

diff --git a/src/nvim/buffer.c b/src/nvim/buffer.c @@ -648,10 +648,6 @@ bool close_buffer(win_T *win, buf_T *buf, int action, bool abort_if_last, bool i return true; } - if (buf->terminal) { - buf_close_terminal(buf); - } - // Always remove the buffer when there is no file name. if (buf->b_ffname == NULL) { del_buf = true; @@ -675,6 +671,18 @@ bool close_buffer(win_T *win, buf_T *buf, int action, bool abort_if_last, bool i buf->b_nwindows = nwindows; + if (buf->terminal) { + buf->b_locked_split++; + buf_close_terminal(buf); + buf->b_locked_split--; + + // Must check this before calling buf_freeall(), otherwise is_curbuf will be true + // in buf_freeall() but still false here, leading to a 0-line buffer. + if (buf == curbuf && !is_curbuf) { + return false; + } + } + buf_freeall(buf, ((del_buf ? BFA_DEL : 0) + (wipe_buf ? BFA_WIPE : 0) + (ignore_abort ? BFA_IGNORE_ABORT : 0))); diff --git a/test/functional/autocmd/termxx_spec.lua b/test/functional/autocmd/termxx_spec.lua @@ -20,13 +20,15 @@ describe('autocmd TermClose', function() clear() api.nvim_set_option_value('shell', testprg('shell-test'), {}) command('set shellcmdflag=EXE shellredir= shellpipe= shellquote= shellxquote=') + command('autocmd! nvim.terminal TermClose') end) local function test_termclose_delete_own_buf() -- The terminal process needs to keep running so that TermClose isn't triggered immediately. api.nvim_set_option_value('shell', string.format('"%s" INTERACT', testprg('shell-test')), {}) - command('autocmd TermClose * bdelete!') command('terminal') + local termbuf = api.nvim_get_current_buf() + command(('autocmd TermClose * bdelete! %d'):format(termbuf)) matches( '^TermClose Autocommands for "%*": Vim%(bdelete%):E937: Attempt to delete a buffer that is in use: term://', pcall_err(command, 'bdelete!') @@ -34,8 +36,7 @@ describe('autocmd TermClose', function() assert_alive() end - -- TODO: fixed after merging patches for `can_unload_buffer`? - pending('TermClose deleting its own buffer, altbuf = buffer 1 #10386', function() + it('TermClose deleting its own buffer, altbuf = buffer 1 #10386', function() test_termclose_delete_own_buf() end) @@ -44,6 +45,29 @@ describe('autocmd TermClose', function() test_termclose_delete_own_buf() end) + it('TermClose deleting all other buffers', function() + local oldbuf = api.nvim_get_current_buf() + -- The terminal process needs to keep running so that TermClose isn't triggered immediately. + api.nvim_set_option_value('shell', string.format('"%s" INTERACT', testprg('shell-test')), {}) + command(('autocmd TermClose * bdelete! %d'):format(oldbuf)) + command('horizontal terminal') + neq(oldbuf, api.nvim_get_current_buf()) + command('bdelete!') + feed('<C-G>') -- This shouldn't crash due to having a 0-line buffer. + assert_alive() + end) + + it('TermClose switching back to terminal buffer', function() + local buf = api.nvim_get_current_buf() + api.nvim_open_term(buf, {}) + command(('autocmd TermClose * buffer %d | new'):format(buf)) + eq( + 'TermClose Autocommands for "*": Vim(buffer):E1546: Cannot switch to a closing buffer', + pcall_err(command, 'bwipe!') + ) + assert_alive() + end) + it('triggers when fast-exiting terminal job stops', function() command('autocmd TermClose * let g:test_termclose = 23') command('terminal')