diff options
author | zeertzjq <zeertzjq@outlook.com> | 2024-02-16 07:34:02 +0800 |
---|---|---|
committer | GitHub <noreply@github.com> | 2024-02-16 07:34:02 +0800 |
commit | 04dfa2eea914086a9f42a5a00a33e9524f9fded4 (patch) | |
tree | c6502c68406be5ae3ef88e292f81cff9f9d71ac2 | |
parent | d60eeacae4ce4aa780636b90199ac20af609e91b (diff) | |
parent | 163add40b8b98b91dfb8eff589f49dc75f1032ea (diff) | |
download | rneovim-04dfa2eea914086a9f42a5a00a33e9524f9fded4.tar.gz rneovim-04dfa2eea914086a9f42a5a00a33e9524f9fded4.tar.bz2 rneovim-04dfa2eea914086a9f42a5a00a33e9524f9fded4.zip |
Merge pull request #27485 from zeertzjq/vim-9.1.0112
vim-patch:9.1.{0112,0113}
-rw-r--r-- | src/nvim/buffer.c | 5 | ||||
-rw-r--r-- | src/nvim/fileio.c | 3 | ||||
-rw-r--r-- | src/nvim/quickfix.c | 10 | ||||
-rw-r--r-- | src/nvim/undo.c | 36 | ||||
-rw-r--r-- | test/functional/legacy/autocmd_spec.lua | 40 | ||||
-rw-r--r-- | test/old/testdir/test_autocmd.vim | 28 |
6 files changed, 104 insertions, 18 deletions
diff --git a/src/nvim/buffer.c b/src/nvim/buffer.c index 38c3ee13aa..bf1d2ac6dd 100644 --- a/src/nvim/buffer.c +++ b/src/nvim/buffer.c @@ -843,8 +843,9 @@ void buf_freeall(buf_T *buf, int flags) ml_close(buf, true); // close and delete the memline/memfile buf->b_ml.ml_line_count = 0; // no lines in buffer if ((flags & BFA_KEEP_UNDO) == 0) { - u_blockfree(buf); // free the memory allocated for undo - u_clearall(buf); // reset all undo information + // free the memory allocated for undo + // and reset all undo information + u_clearallandblockfree(buf); } syntax_clear(&buf->b_s); // reset syntax info buf->b_flags &= ~BF_READERR; // a read error is no longer relevant diff --git a/src/nvim/fileio.c b/src/nvim/fileio.c index a06a773f47..2c96e4bd87 100644 --- a/src/nvim/fileio.c +++ b/src/nvim/fileio.c @@ -3171,8 +3171,7 @@ void buf_reload(buf_T *buf, int orig_mode, bool reload_options) // Mark the buffer as unmodified and free undo info. unchanged(buf, true, true); if ((flags & READ_KEEP_UNDO) == 0) { - u_blockfree(buf); - u_clearall(buf); + u_clearallandblockfree(buf); } else { // Mark all undo states as changed. u_unchanged(curbuf); diff --git a/src/nvim/quickfix.c b/src/nvim/quickfix.c index 14758c8cea..651ebc9f93 100644 --- a/src/nvim/quickfix.c +++ b/src/nvim/quickfix.c @@ -68,6 +68,7 @@ #include "nvim/strings.h" #include "nvim/types_defs.h" #include "nvim/ui.h" +#include "nvim/undo.h" #include "nvim/vim_defs.h" #include "nvim/window.h" @@ -4142,6 +4143,12 @@ static void qf_fill_buffer(qf_list_T *qfl, buf_T *buf, qfline_T *old_last, int q } // delete all existing lines + // + // Note: we cannot store undo information, because + // qf buffer is usually not allowed to be modified. + // + // So we need to clean up undo information + // otherwise autocommands may invalidate the undo stack while ((curbuf->b_ml.ml_flags & ML_EMPTY) == 0) { // If deletion fails, this loop may run forever, so // signal error and return. @@ -4150,6 +4157,9 @@ static void qf_fill_buffer(qf_list_T *qfl, buf_T *buf, qfline_T *old_last, int q return; } } + + // Remove all undo information + u_clearallandblockfree(curbuf); } // Check if there is anything to display diff --git a/src/nvim/undo.c b/src/nvim/undo.c index 6081268e53..e9170ba858 100644 --- a/src/nvim/undo.c +++ b/src/nvim/undo.c @@ -2988,6 +2988,28 @@ void u_clearall(buf_T *buf) buf->b_u_line_lnum = 0; } +/// Free all allocated memory blocks for the buffer 'buf'. +void u_blockfree(buf_T *buf) +{ + while (buf->b_u_oldhead != NULL) { +#ifndef NDEBUG + u_header_T *previous_oldhead = buf->b_u_oldhead; +#endif + + u_freeheader(buf, buf->b_u_oldhead, NULL); + assert(buf->b_u_oldhead != previous_oldhead); + } + xfree(buf->b_u_line_ptr); +} + +/// Free all allocated memory blocks for the buffer 'buf'. +/// and invalidate the undo buffer +void u_clearallandblockfree(buf_T *buf) +{ + u_blockfree(buf); + u_clearall(buf); +} + /// Save the line "lnum" for the "U" command. void u_saveline(buf_T *buf, linenr_T lnum) { @@ -3054,20 +3076,6 @@ void u_undoline(void) check_cursor_col(); } -/// Free all allocated memory blocks for the buffer 'buf'. -void u_blockfree(buf_T *buf) -{ - while (buf->b_u_oldhead != NULL) { -#ifndef NDEBUG - u_header_T *previous_oldhead = buf->b_u_oldhead; -#endif - - u_freeheader(buf, buf->b_u_oldhead, NULL); - assert(buf->b_u_oldhead != previous_oldhead); - } - xfree(buf->b_u_line_ptr); -} - /// Allocate memory and copy curbuf line into it. /// /// @param lnum the line to copy diff --git a/test/functional/legacy/autocmd_spec.lua b/test/functional/legacy/autocmd_spec.lua new file mode 100644 index 0000000000..97051f3d3f --- /dev/null +++ b/test/functional/legacy/autocmd_spec.lua @@ -0,0 +1,40 @@ +local helpers = require('test.functional.helpers')(after_each) +local clear = helpers.clear +local write_file = helpers.write_file +local command = helpers.command +local feed = helpers.feed +local api = helpers.api +local eq = helpers.eq + +before_each(clear) + +-- oldtest: Test_autocmd_invalidates_undo_on_textchanged() +it('no E440 in quickfix window when autocommand invalidates undo', function() + write_file( + 'XTest_autocmd_invalidates_undo_on_textchanged', + [[ + set hidden + " create quickfix list (at least 2 lines to move line) + vimgrep /u/j % + + " enter quickfix window + cwindow + + " set modifiable + setlocal modifiable + + " set autocmd to clear quickfix list + + autocmd! TextChanged <buffer> call setqflist([]) + " move line + move+1 + ]] + ) + finally(function() + os.remove('XTest_autocmd_invalidates_undo_on_textchanged') + end) + command('edit XTest_autocmd_invalidates_undo_on_textchanged') + command('so %') + feed('G') + eq('', api.nvim_get_vvar('errmsg')) +end) diff --git a/test/old/testdir/test_autocmd.vim b/test/old/testdir/test_autocmd.vim index 0a28ae6147..6901627c17 100644 --- a/test/old/testdir/test_autocmd.vim +++ b/test/old/testdir/test_autocmd.vim @@ -3840,4 +3840,32 @@ func Test_autocmd_shortmess() delfunc SetupVimTest_shm endfunc +func Test_autocmd_invalidates_undo_on_textchanged() + CheckRunVimInTerminal + let script =<< trim END + set hidden + " create quickfix list (at least 2 lines to move line) + vimgrep /u/j % + + " enter quickfix window + cwindow + + " set modifiable + setlocal modifiable + + " set autocmd to clear quickfix list + + autocmd! TextChanged <buffer> call setqflist([]) + " move line + move+1 + END + call writefile(script, 'XTest_autocmd_invalidates_undo_on_textchanged', 'D') + let buf = RunVimInTerminal('XTest_autocmd_invalidates_undo_on_textchanged', {'rows': 20}) + call term_sendkeys(buf, ":so %\<cr>") + call term_sendkeys(buf, "G") + call WaitForAssert({-> assert_match('^XTest_autocmd_invalidates_undo_on_textchanged\s*$', term_getline(buf, 20))}, 1000) + + call StopVimInTerminal(buf) +endfunc + " vim: shiftwidth=2 sts=2 expandtab |