mirror of
https://github.com/neovim/neovim.git
synced 2025-09-06 03:18:16 +00:00
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:
@@ -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
|
||||||
|
@@ -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;
|
||||||
|
@@ -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)
|
||||||
|
Reference in New Issue
Block a user