patch 9.1.0062: Internal error when :luado/perldo/pydo etc delete lines
Problem:  Internal error when :luado/perldo/pydo etc delete lines
Solution: Test that the line is still valid line number
          (zeertzjq)
closes: #13931
Signed-off-by: zeertzjq <zeertzjq@outlook.com>
Signed-off-by: Christian Brabandt <cb@256bit.org>
			
			
This commit is contained in:
		
				
					committed by
					
						 Christian Brabandt
						Christian Brabandt
					
				
			
			
				
	
			
			
			
						parent
						
							e6d8b4662d
						
					
				
				
					commit
					e99f068878
				
			| @ -2692,9 +2692,12 @@ ex_luado(exarg_T *eap) | |||||||
| 	    luaV_emsg(L); | 	    luaV_emsg(L); | ||||||
| 	    break; | 	    break; | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	// Catch the command switching to another buffer. | 	// Catch the command switching to another buffer. | ||||||
| 	if (curbuf != was_curbuf) | 	// Check the line number, the command may have deleted lines. | ||||||
|  | 	if (curbuf != was_curbuf || l > curbuf->b_ml.ml_line_count) | ||||||
| 	    break; | 	    break; | ||||||
|  |  | ||||||
| 	if (lua_isstring(L, -1)) // update line? | 	if (lua_isstring(L, -1)) // update line? | ||||||
| 	{ | 	{ | ||||||
| #ifdef HAVE_SANDBOX | #ifdef HAVE_SANDBOX | ||||||
|  | |||||||
| @ -1368,7 +1368,7 @@ ex_perldo(exarg_T *eap) | |||||||
| 	PUSHMARK(sp); | 	PUSHMARK(sp); | ||||||
| 	perl_call_pv("VIM::perldo", G_SCALAR | G_EVAL); | 	perl_call_pv("VIM::perldo", G_SCALAR | G_EVAL); | ||||||
| 	str = SvPV(GvSV(PL_errgv), length); | 	str = SvPV(GvSV(PL_errgv), length); | ||||||
| 	if (length || curbuf != was_curbuf) | 	if (length || curbuf != was_curbuf || i > curbuf->b_ml.ml_line_count) | ||||||
| 	    break; | 	    break; | ||||||
| 	SPAGAIN; | 	SPAGAIN; | ||||||
| 	if (SvTRUEx(POPs)) | 	if (SvTRUEx(POPs)) | ||||||
|  | |||||||
| @ -6136,7 +6136,8 @@ run_do(const char *cmd, void *arg UNUSED | |||||||
| 	    goto err; | 	    goto err; | ||||||
|  |  | ||||||
| 	// Check that the command didn't switch to another buffer. | 	// Check that the command didn't switch to another buffer. | ||||||
| 	if (curbuf != was_curbuf) | 	// Check the line number, the command my have deleted lines. | ||||||
|  | 	if (curbuf != was_curbuf || lnum > curbuf->b_ml.ml_line_count) | ||||||
| 	{ | 	{ | ||||||
| 	    Py_XDECREF(ret); | 	    Py_XDECREF(ret); | ||||||
| 	    goto err; | 	    goto err; | ||||||
|  | |||||||
| @ -885,7 +885,7 @@ ex_rubydo(exarg_T *eap) | |||||||
| 	    error_print(state); | 	    error_print(state); | ||||||
| 	    break; | 	    break; | ||||||
| 	} | 	} | ||||||
| 	if (was_curbuf != curbuf) | 	if (was_curbuf != curbuf || i > curbuf->b_ml.ml_line_count) | ||||||
| 	    break; | 	    break; | ||||||
| 	line = rb_lastline_get(); | 	line = rb_lastline_get(); | ||||||
| 	if (!NIL_P(line)) | 	if (!NIL_P(line)) | ||||||
|  | |||||||
| @ -2012,7 +2012,8 @@ ex_tcldo(exarg_T *eap) | |||||||
| #if (TCL_MAJOR_VERSION == 8 && TCL_MINOR_VERSION >= 5) || TCL_MAJOR_VERSION > 8 | #if (TCL_MAJOR_VERSION == 8 && TCL_MINOR_VERSION >= 5) || TCL_MAJOR_VERSION > 8 | ||||||
| 	    || Tcl_LimitExceeded(tclinfo.interp) | 	    || Tcl_LimitExceeded(tclinfo.interp) | ||||||
| #endif | #endif | ||||||
| 	    || curbuf != was_curbuf) | 	    || curbuf != was_curbuf | ||||||
|  | 	    || (linenr_T)rs > curbuf->b_ml.ml_line_count) | ||||||
| 	    break; | 	    break; | ||||||
| 	line = (char *)Tcl_GetVar(tclinfo.interp, var_line, 0); | 	line = (char *)Tcl_GetVar(tclinfo.interp, var_line, 0); | ||||||
| 	if (line) | 	if (line) | ||||||
|  | |||||||
| @ -28,21 +28,37 @@ func TearDown() | |||||||
| endfunc | endfunc | ||||||
|  |  | ||||||
| " Check that switching to another buffer does not trigger ml_get error. | " Check that switching to another buffer does not trigger ml_get error. | ||||||
| func Test_lua_command_new_no_ml_get_error() | func Test_lua_luado_change_buffer() | ||||||
|   new |   new | ||||||
|  |  | ||||||
|   let wincount = winnr('$') |   let wincount = winnr('$') | ||||||
|   call setline(1, ['one', 'two', 'three']) |   call setline(1, ['one', 'two', 'three']) | ||||||
|   luado vim.command("new") |   luado vim.command("new") | ||||||
|   call assert_equal(wincount + 1, winnr('$')) |   call assert_equal(wincount + 1, winnr('$')) | ||||||
|  |  | ||||||
|   %bwipe! |   %bwipe! | ||||||
| endfunc | endfunc | ||||||
|  |  | ||||||
| " Test vim.command() | " Check that :luado deleting lines does not trigger ml_get error. | ||||||
| func Test_lua_command() | func Test_lua_luado_delete_lines() | ||||||
|   new |   new | ||||||
|  |  | ||||||
|  |   call setline(1, ['one', 'two', 'three']) | ||||||
|  |   luado vim.command("%d_") | ||||||
|  |   call assert_equal([''], getline(1, '$')) | ||||||
|  |  | ||||||
|   call setline(1, ['one', 'two', 'three']) |   call setline(1, ['one', 'two', 'three']) | ||||||
|   luado vim.command("1,2d_") |   luado vim.command("1,2d_") | ||||||
|   call assert_equal(['three'], getline(1, '$')) |   call assert_equal(['three'], getline(1, '$')) | ||||||
|  |  | ||||||
|  |   call setline(1, ['one', 'two', 'three']) | ||||||
|  |   luado vim.command("2,3d_"); return "REPLACED" | ||||||
|  |   call assert_equal(['REPLACED'], getline(1, '$')) | ||||||
|  |  | ||||||
|  |   call setline(1, ['one', 'two', 'three']) | ||||||
|  |   2,3luado vim.command("1,2d_"); return "REPLACED" | ||||||
|  |   call assert_equal(['three'], getline(1, '$')) | ||||||
|  |  | ||||||
|   bwipe! |   bwipe! | ||||||
| endfunc | endfunc | ||||||
|  |  | ||||||
|  | |||||||
| @ -211,10 +211,25 @@ func Test_perldo() | |||||||
|   call assert_false(search('\Cperl')) |   call assert_false(search('\Cperl')) | ||||||
|   bw! |   bw! | ||||||
|  |  | ||||||
|   " Check deleting lines does not trigger ml_get error. |  | ||||||
|   new |   new | ||||||
|  |  | ||||||
|  |   " Check deleting lines does not trigger ml_get error. | ||||||
|   call setline(1, ['one', 'two', 'three']) |   call setline(1, ['one', 'two', 'three']) | ||||||
|   perldo VIM::DoCommand("%d_") |   perldo VIM::DoCommand("%d_") | ||||||
|  |   call assert_equal([''], getline(1, '$')) | ||||||
|  |  | ||||||
|  |   call setline(1, ['one', 'two', 'three']) | ||||||
|  |   perldo VIM::DoCommand("1,2d_") | ||||||
|  |   call assert_equal(['three'], getline(1, '$')) | ||||||
|  |  | ||||||
|  |   call setline(1, ['one', 'two', 'three']) | ||||||
|  |   perldo VIM::DoCommand("2,3d_"); $_ = "REPLACED" | ||||||
|  |   call assert_equal(['REPLACED'], getline(1, '$')) | ||||||
|  |  | ||||||
|  |   call setline(1, ['one', 'two', 'three']) | ||||||
|  |   2,3perldo VIM::DoCommand("1,2d_"); $_ = "REPLACED" | ||||||
|  |   call assert_equal(['three'], getline(1, '$')) | ||||||
|  |  | ||||||
|   bwipe! |   bwipe! | ||||||
|  |  | ||||||
|   " Check a Perl expression which gives an error. |   " Check a Perl expression which gives an error. | ||||||
|  | |||||||
| @ -56,10 +56,25 @@ func Test_AAA_python_setup() | |||||||
| endfunc | endfunc | ||||||
|  |  | ||||||
| func Test_pydo() | func Test_pydo() | ||||||
|   " Check deleting lines does not trigger an ml_get error. |  | ||||||
|   new |   new | ||||||
|  |  | ||||||
|  |   " Check deleting lines does not trigger an ml_get error. | ||||||
|   call setline(1, ['one', 'two', 'three']) |   call setline(1, ['one', 'two', 'three']) | ||||||
|   pydo vim.command("%d_") |   pydo vim.command("%d_") | ||||||
|  |   call assert_equal([''], getline(1, '$')) | ||||||
|  |  | ||||||
|  |   call setline(1, ['one', 'two', 'three']) | ||||||
|  |   pydo vim.command("1,2d_") | ||||||
|  |   call assert_equal(['three'], getline(1, '$')) | ||||||
|  |  | ||||||
|  |   call setline(1, ['one', 'two', 'three']) | ||||||
|  |   pydo vim.command("2,3d_"); return "REPLACED" | ||||||
|  |   call assert_equal(['REPLACED'], getline(1, '$')) | ||||||
|  |  | ||||||
|  |   call setline(1, ['one', 'two', 'three']) | ||||||
|  |   2,3pydo vim.command("1,2d_"); return "REPLACED" | ||||||
|  |   call assert_equal(['three'], getline(1, '$')) | ||||||
|  |  | ||||||
|   bwipe! |   bwipe! | ||||||
|  |  | ||||||
|   " Check switching to another buffer does not trigger an ml_get error. |   " Check switching to another buffer does not trigger an ml_get error. | ||||||
|  | |||||||
| @ -88,10 +88,25 @@ func Test_AAA_python3_setup() | |||||||
| endfunc | endfunc | ||||||
|  |  | ||||||
| func Test_py3do() | func Test_py3do() | ||||||
|   " Check deleting lines does not trigger an ml_get error. |  | ||||||
|   new |   new | ||||||
|  |  | ||||||
|  |   " Check deleting lines does not trigger an ml_get error. | ||||||
|   call setline(1, ['one', 'two', 'three']) |   call setline(1, ['one', 'two', 'three']) | ||||||
|   py3do vim.command("%d_") |   py3do vim.command("%d_") | ||||||
|  |   call assert_equal([''], getline(1, '$')) | ||||||
|  |  | ||||||
|  |   call setline(1, ['one', 'two', 'three']) | ||||||
|  |   py3do vim.command("1,2d_") | ||||||
|  |   call assert_equal(['three'], getline(1, '$')) | ||||||
|  |  | ||||||
|  |   call setline(1, ['one', 'two', 'three']) | ||||||
|  |   py3do vim.command("2,3d_"); return "REPLACED" | ||||||
|  |   call assert_equal(['REPLACED'], getline(1, '$')) | ||||||
|  |  | ||||||
|  |   call setline(1, ['one', 'two', 'three']) | ||||||
|  |   2,3py3do vim.command("1,2d_"); return "REPLACED" | ||||||
|  |   call assert_equal(['three'], getline(1, '$')) | ||||||
|  |  | ||||||
|   bwipe! |   bwipe! | ||||||
|  |  | ||||||
|   " Check switching to another buffer does not trigger an ml_get error. |   " Check switching to another buffer does not trigger an ml_get error. | ||||||
|  | |||||||
| @ -11,10 +11,25 @@ func Test_ruby_change_buffer() | |||||||
| endfunc | endfunc | ||||||
|  |  | ||||||
| func Test_rubydo() | func Test_rubydo() | ||||||
|   " Check deleting lines does not trigger ml_get error. |  | ||||||
|   new |   new | ||||||
|  |  | ||||||
|  |   " Check deleting lines does not trigger ml_get error. | ||||||
|   call setline(1, ['one', 'two', 'three']) |   call setline(1, ['one', 'two', 'three']) | ||||||
|   rubydo Vim.command("%d_") |   rubydo Vim.command("%d_") | ||||||
|  |   call assert_equal(['one'], getline(1, '$')) | ||||||
|  |  | ||||||
|  |   call setline(1, ['one', 'two', 'three']) | ||||||
|  |   rubydo Vim.command("1,2d_") | ||||||
|  |   call assert_equal(['one'], getline(1, '$')) | ||||||
|  |  | ||||||
|  |   call setline(1, ['one', 'two', 'three']) | ||||||
|  |   rubydo Vim.command("2,3d_"); $_ = "REPLACED" | ||||||
|  |   call assert_equal(['REPLACED'], getline(1, '$')) | ||||||
|  |  | ||||||
|  |   call setline(1, ['one', 'two', 'three']) | ||||||
|  |   2,3rubydo Vim.command("1,2d_"); $_ = "REPLACED" | ||||||
|  |   call assert_equal(['three'], getline(1, '$')) | ||||||
|  |  | ||||||
|   bwipe! |   bwipe! | ||||||
|  |  | ||||||
|   " Check switching to another buffer does not trigger ml_get error. |   " Check switching to another buffer does not trigger ml_get error. | ||||||
|  | |||||||
| @ -11,10 +11,25 @@ func TclEval(tcl_expr) | |||||||
| endfunc | endfunc | ||||||
|  |  | ||||||
| func Test_tcldo() | func Test_tcldo() | ||||||
|   " Check deleting lines does not trigger ml_get error. |  | ||||||
|   new |   new | ||||||
|  |  | ||||||
|  |   " Check deleting lines does not trigger ml_get error. | ||||||
|   call setline(1, ['one', 'two', 'three']) |   call setline(1, ['one', 'two', 'three']) | ||||||
|   tcldo ::vim::command %d_ |   tcldo ::vim::command %d_ | ||||||
|  |   call assert_equal(['one'], getline(1, '$')) | ||||||
|  |  | ||||||
|  |   call setline(1, ['one', 'two', 'three']) | ||||||
|  |   tcldo ::vim::command 1,2d_ | ||||||
|  |   call assert_equal(['one'], getline(1, '$')) | ||||||
|  |  | ||||||
|  |   call setline(1, ['one', 'two', 'three']) | ||||||
|  |   tcldo ::vim::command 2,3d_ ; set line REPLACED | ||||||
|  |   call assert_equal(['REPLACED'], getline(1, '$')) | ||||||
|  |  | ||||||
|  |   call setline(1, ['one', 'two', 'three']) | ||||||
|  |   2,3tcldo ::vim::command 1,2d_ ; set line REPLACED | ||||||
|  |   call assert_equal(['three'], getline(1, '$')) | ||||||
|  |  | ||||||
|   bwipe! |   bwipe! | ||||||
|  |  | ||||||
|   " Check that switching to another buffer does not trigger ml_get error. |   " Check that switching to another buffer does not trigger ml_get error. | ||||||
|  | |||||||
| @ -704,6 +704,8 @@ static char *(features[]) = | |||||||
|  |  | ||||||
| static int included_patches[] = | static int included_patches[] = | ||||||
| {   /* Add new patch number below this line */ | {   /* Add new patch number below this line */ | ||||||
|  | /**/ | ||||||
|  |     62, | ||||||
| /**/ | /**/ | ||||||
|     61, |     61, | ||||||
| /**/ | /**/ | ||||||
|  | |||||||
		Reference in New Issue
	
	Block a user