Merge pull request #21262 from zeertzjq/vim-8.2.1748

vim-patch:8.2.1748: closing split window in other tab may cause a crash
This commit is contained in:
zeertzjq
2022-12-02 11:15:47 +08:00
committed by GitHub
3 changed files with 46 additions and 14 deletions

View File

@@ -192,7 +192,23 @@ func Test_tabwin_close()
call win_execute(l:wid, 'close') call win_execute(l:wid, 'close')
" Should not crash. " Should not crash.
call assert_true(v:true) call assert_true(v:true)
%bwipe!
" This tests closing a window in another tab, while leaving the tab open
" i.e. two windows in another tab.
tabedit
let w:this_win = 42
new
let othertab_wid = win_getid()
tabprevious
call win_execute(othertab_wid, 'q')
" drawing the tabline helps check that the other tab's windows and buffers
" are still valid
redrawtabline
" but to be certain, ensure we can focus the other tab too
tabnext
call assert_equal(42, w:this_win)
bwipe!
endfunc endfunc
" Test when closing a split window (above/below) restores space to the window " Test when closing a split window (above/below) restores space to the window

View File

@@ -1643,12 +1643,21 @@ bool win_valid_floating(const win_T *win)
/// ///
/// @param win window to check /// @param win window to check
bool win_valid(const win_T *win) FUNC_ATTR_PURE FUNC_ATTR_WARN_UNUSED_RESULT bool win_valid(const win_T *win) FUNC_ATTR_PURE FUNC_ATTR_WARN_UNUSED_RESULT
{
return tabpage_win_valid(curtab, win);
}
/// Check if "win" is a pointer to an existing window in tabpage "tp".
///
/// @param win window to check
static bool tabpage_win_valid(const tabpage_T *tp, const win_T *win)
FUNC_ATTR_PURE FUNC_ATTR_WARN_UNUSED_RESULT
{ {
if (win == NULL) { if (win == NULL) {
return false; return false;
} }
FOR_ALL_WINDOWS_IN_TAB(wp, curtab) { FOR_ALL_WINDOWS_IN_TAB(wp, tp) {
if (wp == win) { if (wp == win) {
return true; return true;
} }
@@ -3049,6 +3058,7 @@ void win_close_othertab(win_T *win, int free_buf, tabpage_T *tp)
static win_T *win_free_mem(win_T *win, int *dirp, tabpage_T *tp) static win_T *win_free_mem(win_T *win, int *dirp, tabpage_T *tp)
{ {
win_T *wp; win_T *wp;
tabpage_T *win_tp = tp == NULL ? curtab : tp;
if (!win->w_floating) { if (!win->w_floating) {
// Remove the window and its frame from the tree of frames. // Remove the window and its frame from the tree of frames.
@@ -3057,22 +3067,26 @@ static win_T *win_free_mem(win_T *win, int *dirp, tabpage_T *tp)
xfree(frp); xfree(frp);
} else { } else {
*dirp = 'h'; // Dummy value. *dirp = 'h'; // Dummy value.
if (win_valid(prevwin) && prevwin != win) { if (tp == NULL) {
wp = prevwin; if (win_valid(prevwin) && prevwin != win) {
wp = prevwin;
} else {
wp = firstwin;
}
} else { } else {
wp = firstwin; if (tabpage_win_valid(tp, tp->tp_prevwin) && tp->tp_prevwin != win) {
wp = tp->tp_prevwin;
} else {
wp = tp->tp_firstwin;
}
} }
} }
win_free(win, tp); win_free(win, tp);
// When deleting the current window of another tab page select a new // When deleting the current window in the tab, select a new current
// current window. // window.
if (tp != NULL && win == tp->tp_curwin) { if (win == win_tp->tp_curwin) {
if (win_valid(tp->tp_prevwin) && tp->tp_prevwin != win) { win_tp->tp_curwin = wp;
tp->tp_curwin = tp->tp_prevwin;
} else {
tp->tp_curwin = tp->tp_firstwin;
}
} }
return wp; return wp;

View File

@@ -490,6 +490,8 @@ describe('API/win', function()
it('closing current (float) window of another tabpage #15313', function() it('closing current (float) window of another tabpage #15313', function()
command('tabedit') command('tabedit')
command('botright split')
local prevwin = curwin().id
eq(2, eval('tabpagenr()')) eq(2, eval('tabpagenr()'))
local win = meths.open_win(0, true, { local win = meths.open_win(0, true, {
relative='editor', row=10, col=10, width=50, height=10 relative='editor', row=10, col=10, width=50, height=10
@@ -499,7 +501,7 @@ describe('API/win', function()
eq(1, eval('tabpagenr()')) eq(1, eval('tabpagenr()'))
meths.win_close(win, false) meths.win_close(win, false)
eq(1001, meths.tabpage_get_win(tab).id) eq(prevwin, meths.tabpage_get_win(tab).id)
assert_alive() assert_alive()
end) end)
end) end)