Merge #27930 nvim_create_buf fixes

This commit is contained in:
Justin M. Keyes 2024-03-20 06:15:38 -07:00 committed by GitHub
commit 7549845e8d
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
2 changed files with 75 additions and 9 deletions

View File

@ -958,21 +958,21 @@ Buffer nvim_create_buf(Boolean listed, Boolean scratch, Error *err)
FUNC_API_SINCE(6) FUNC_API_SINCE(6)
{ {
try_start(); try_start();
// Block autocommands for now so they don't mess with the buffer before we
// finish configuring it.
block_autocmds();
buf_T *buf = buflist_new(NULL, NULL, 0, buf_T *buf = buflist_new(NULL, NULL, 0,
BLN_NOOPT | BLN_NEW | (listed ? BLN_LISTED : 0)); BLN_NOOPT | BLN_NEW | (listed ? BLN_LISTED : 0));
try_end(err);
if (buf == NULL) { if (buf == NULL) {
unblock_autocmds();
goto fail; goto fail;
} }
// Open the memline for the buffer. This will avoid spurious autocmds when // Open the memline for the buffer. This will avoid spurious autocmds when
// a later nvim_buf_set_lines call would have needed to "open" the buffer. // a later nvim_buf_set_lines call would have needed to "open" the buffer.
try_start(); if (ml_open(buf) == FAIL) {
block_autocmds(); unblock_autocmds();
int status = ml_open(buf);
unblock_autocmds();
try_end(err);
if (status == FAIL) {
goto fail; goto fail;
} }
@ -983,7 +983,7 @@ Buffer nvim_create_buf(Boolean listed, Boolean scratch, Error *err)
buf->b_last_changedtick_pum = buf_get_changedtick(buf); buf->b_last_changedtick_pum = buf_get_changedtick(buf);
// Only strictly needed for scratch, but could just as well be consistent // Only strictly needed for scratch, but could just as well be consistent
// and do this now. buffer is created NOW, not when it latter first happen // and do this now. Buffer is created NOW, not when it later first happens
// to reach a window or aucmd_prepbuf() .. // to reach a window or aucmd_prepbuf() ..
buf_copy_options(buf, BCO_ENTER | BCO_NOHELP); buf_copy_options(buf, BCO_ENTER | BCO_NOHELP);
@ -994,10 +994,26 @@ Buffer nvim_create_buf(Boolean listed, Boolean scratch, Error *err)
buf->b_p_swf = false; buf->b_p_swf = false;
buf->b_p_ml = false; buf->b_p_ml = false;
} }
unblock_autocmds();
bufref_T bufref;
set_bufref(&bufref, buf);
if (apply_autocmds(EVENT_BUFNEW, NULL, NULL, false, buf)
&& !bufref_valid(&bufref)) {
goto fail;
}
if (listed
&& apply_autocmds(EVENT_BUFADD, NULL, NULL, false, buf)
&& !bufref_valid(&bufref)) {
goto fail;
}
try_end(err);
return buf->b_fnum; return buf->b_fnum;
fail: fail:
if (!ERROR_SET(err)) { if (!try_end(err)) {
api_set_error(err, kErrorTypeException, "Failed to create buffer"); api_set_error(err, kErrorTypeException, "Failed to create buffer");
} }
return 0; return 0;

View File

@ -3135,6 +3135,56 @@ describe('API', function()
-- nowadays this works because we don't execute any spurious autocmds at all #24824 -- nowadays this works because we don't execute any spurious autocmds at all #24824
assert_alive() assert_alive()
end) end)
it('no memory leak when autocommands load the buffer immediately', function()
exec([[
autocmd BufNew * ++once call bufload(expand("<abuf>")->str2nr())
\| let loaded = bufloaded(expand("<abuf>")->str2nr())
]])
api.nvim_create_buf(false, true)
eq(1, eval('g:loaded'))
end)
it('creating scratch buffer where autocommands set &swapfile works', function()
exec([[
autocmd BufNew * ++once execute expand("<abuf>") "buffer"
\| file foobar
\| setlocal swapfile
]])
local new_buf = api.nvim_create_buf(false, true)
neq('', fn.swapname(new_buf))
end)
it('fires expected autocommands', function()
exec([=[
" Append the &buftype to check autocommands trigger *after* the buffer was configured to be
" scratch, if applicable.
autocmd BufNew * let fired += [["BufNew", expand("<abuf>")->str2nr(),
\ getbufvar(expand("<abuf>")->str2nr(), "&buftype")]]
autocmd BufAdd * let fired += [["BufAdd", expand("<abuf>")->str2nr(),
\ getbufvar(expand("<abuf>")->str2nr(), "&buftype")]]
" Don't want to see OptionSet; buffer options set from passing true for "scratch", etc.
" should be configured invisibly, and before autocommands.
autocmd OptionSet * let fired += [["OptionSet", expand("<amatch>")]]
let fired = []
]=])
local new_buf = api.nvim_create_buf(false, false)
eq({ { 'BufNew', new_buf, '' } }, eval('g:fired'))
command('let fired = []')
new_buf = api.nvim_create_buf(false, true)
eq({ { 'BufNew', new_buf, 'nofile' } }, eval('g:fired'))
command('let fired = []')
new_buf = api.nvim_create_buf(true, false)
eq({ { 'BufNew', new_buf, '' }, { 'BufAdd', new_buf, '' } }, eval('g:fired'))
command('let fired = []')
new_buf = api.nvim_create_buf(true, true)
eq({ { 'BufNew', new_buf, 'nofile' }, { 'BufAdd', new_buf, 'nofile' } }, eval('g:fired'))
end)
end) end)
describe('nvim_get_runtime_file', function() describe('nvim_get_runtime_file', function()