mirror of
https://github.com/neovim/neovim.git
synced 2025-09-30 15:08:35 +00:00
vim-patch:8.0.0537: illegal memory access with :z and large count (#8592)
Problem: Illegal memory access with :z and large count.
Solution: Check for number overflow, using long instead of int. (Dominique
Pelle, closes vim/vim#1612)
fa0ad0bb0b
This commit is contained in:

committed by
Justin M. Keyes

parent
943e607442
commit
a10938ae31
@@ -1306,10 +1306,10 @@ filterend:
|
|||||||
* Call a shell to execute a command.
|
* Call a shell to execute a command.
|
||||||
* When "cmd" is NULL start an interactive shell.
|
* When "cmd" is NULL start an interactive shell.
|
||||||
*/
|
*/
|
||||||
void
|
void
|
||||||
do_shell (
|
do_shell(
|
||||||
char_u *cmd,
|
char_u *cmd,
|
||||||
int flags /* may be SHELL_DOOUT when output is redirected */
|
int flags // may be SHELL_DOOUT when output is redirected
|
||||||
)
|
)
|
||||||
{
|
{
|
||||||
int save_nwr;
|
int save_nwr;
|
||||||
@@ -1789,14 +1789,14 @@ theend:
|
|||||||
* May set eap->forceit if a dialog says it's OK to overwrite.
|
* May set eap->forceit if a dialog says it's OK to overwrite.
|
||||||
* Return OK if it's OK, FAIL if it is not.
|
* Return OK if it's OK, FAIL if it is not.
|
||||||
*/
|
*/
|
||||||
int
|
int
|
||||||
check_overwrite (
|
check_overwrite(
|
||||||
exarg_T *eap,
|
exarg_T *eap,
|
||||||
buf_T *buf,
|
buf_T *buf,
|
||||||
char_u *fname, /* file name to be used (can differ from
|
char_u *fname, // file name to be used (can differ from
|
||||||
buf->ffname) */
|
// buf->ffname)
|
||||||
char_u *ffname, /* full path version of fname */
|
char_u *ffname, // full path version of fname
|
||||||
int other /* writing under other name */
|
int other // writing under other name
|
||||||
)
|
)
|
||||||
{
|
{
|
||||||
/*
|
/*
|
||||||
@@ -2823,7 +2823,7 @@ void ex_change(exarg_T *eap)
|
|||||||
void ex_z(exarg_T *eap)
|
void ex_z(exarg_T *eap)
|
||||||
{
|
{
|
||||||
char_u *x;
|
char_u *x;
|
||||||
int bigness;
|
int64_t bigness;
|
||||||
char_u *kind;
|
char_u *kind;
|
||||||
int minus = 0;
|
int minus = 0;
|
||||||
linenr_T start, end, curs, i;
|
linenr_T start, end, curs, i;
|
||||||
@@ -2856,10 +2856,17 @@ void ex_z(exarg_T *eap)
|
|||||||
EMSG(_("E144: non-numeric argument to :z"));
|
EMSG(_("E144: non-numeric argument to :z"));
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
bigness = atoi((char *)x);
|
bigness = atol((char *)x);
|
||||||
|
|
||||||
|
// bigness could be < 0 if atol(x) overflows.
|
||||||
|
if (bigness > 2 * curbuf->b_ml.ml_line_count || bigness < 0) {
|
||||||
|
bigness = 2 * curbuf->b_ml.ml_line_count;
|
||||||
|
}
|
||||||
|
|
||||||
p_window = bigness;
|
p_window = bigness;
|
||||||
if (*kind == '=')
|
if (*kind == '=') {
|
||||||
bigness += 2;
|
bigness += 2;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* the number of '-' and '+' multiplies the distance */
|
/* the number of '-' and '+' multiplies the distance */
|
||||||
@@ -4587,11 +4594,11 @@ char_u *check_help_lang(char_u *arg)
|
|||||||
* Assumption is made that the matched_string passed has already been found to
|
* Assumption is made that the matched_string passed has already been found to
|
||||||
* match some string for which help is requested. webb.
|
* match some string for which help is requested. webb.
|
||||||
*/
|
*/
|
||||||
int
|
int
|
||||||
help_heuristic (
|
help_heuristic(
|
||||||
char_u *matched_string,
|
char_u *matched_string,
|
||||||
int offset, /* offset for match */
|
int offset, // offset for match
|
||||||
int wrong_case /* no matching case */
|
int wrong_case // no matching case
|
||||||
)
|
)
|
||||||
{
|
{
|
||||||
int num_letters;
|
int num_letters;
|
||||||
|
@@ -5,6 +5,7 @@ source test_assign.vim
|
|||||||
source test_changedtick.vim
|
source test_changedtick.vim
|
||||||
source test_cursor_func.vim
|
source test_cursor_func.vim
|
||||||
source test_ex_undo.vim
|
source test_ex_undo.vim
|
||||||
|
source test_ex_z.vim
|
||||||
source test_execute_func.vim
|
source test_execute_func.vim
|
||||||
source test_expr.vim
|
source test_expr.vim
|
||||||
source test_feedkeys.vim
|
source test_feedkeys.vim
|
||||||
@@ -19,8 +20,8 @@ source test_jumps.vim
|
|||||||
source test_fileformat.vim
|
source test_fileformat.vim
|
||||||
source test_filetype.vim
|
source test_filetype.vim
|
||||||
source test_lambda.vim
|
source test_lambda.vim
|
||||||
source test_menu.vim
|
|
||||||
source test_mapping.vim
|
source test_mapping.vim
|
||||||
|
source test_menu.vim
|
||||||
source test_messages.vim
|
source test_messages.vim
|
||||||
source test_partial.vim
|
source test_partial.vim
|
||||||
source test_popup.vim
|
source test_popup.vim
|
||||||
|
78
src/nvim/testdir/test_ex_z.vim
Normal file
78
src/nvim/testdir/test_ex_z.vim
Normal file
@@ -0,0 +1,78 @@
|
|||||||
|
" Test :z
|
||||||
|
|
||||||
|
func Test_z()
|
||||||
|
call setline(1, range(1, 100))
|
||||||
|
|
||||||
|
let a = execute('20z3')
|
||||||
|
call assert_equal("\n20\n21\n22", a)
|
||||||
|
call assert_equal(22, line('.'))
|
||||||
|
" 'window' should be set to the {count} value.
|
||||||
|
call assert_equal(3, &window)
|
||||||
|
|
||||||
|
" If there is only one window, then twice the amount of 'scroll' is used.
|
||||||
|
set scroll=2
|
||||||
|
let a = execute('20z')
|
||||||
|
call assert_equal("\n20\n21\n22\n23", a)
|
||||||
|
call assert_equal(23, line('.'))
|
||||||
|
|
||||||
|
let a = execute('20z+3')
|
||||||
|
" FIXME: I would expect the same result as '20z3' but it
|
||||||
|
" gives "\n21\n22\n23" instead. Bug in Vim or in ":help :z"?
|
||||||
|
"call assert_equal("\n20\n21\n22", a)
|
||||||
|
"call assert_equal(22, line('.'))
|
||||||
|
|
||||||
|
let a = execute('20z-3')
|
||||||
|
call assert_equal("\n18\n19\n20", a)
|
||||||
|
call assert_equal(20, line('.'))
|
||||||
|
|
||||||
|
let a = execute('20z=3')
|
||||||
|
call assert_match("^\n18\n19\n-\\+\n20\n-\\+\n21\n22$", a)
|
||||||
|
call assert_equal(20, line('.'))
|
||||||
|
|
||||||
|
let a = execute('20z^3')
|
||||||
|
call assert_equal("\n14\n15\n16\n17", a)
|
||||||
|
call assert_equal(17, line('.'))
|
||||||
|
|
||||||
|
let a = execute('20z.3')
|
||||||
|
call assert_equal("\n19\n20\n21", a)
|
||||||
|
call assert_equal(21, line('.'))
|
||||||
|
|
||||||
|
let a = execute('20z#3')
|
||||||
|
call assert_equal("\n 20 20\n 21 21\n 22 22", a)
|
||||||
|
call assert_equal(22, line('.'))
|
||||||
|
|
||||||
|
let a = execute('20z#-3')
|
||||||
|
call assert_equal("\n 18 18\n 19 19\n 20 20", a)
|
||||||
|
call assert_equal(20, line('.'))
|
||||||
|
|
||||||
|
let a = execute('20z#=3')
|
||||||
|
call assert_match("^\n 18 18\n 19 19\n-\\+\n 20 20\n-\\+\n 21 21\n 22 22$", a)
|
||||||
|
call assert_equal(20, line('.'))
|
||||||
|
|
||||||
|
" Test with {count} bigger than the number of lines in buffer.
|
||||||
|
let a = execute('20z1000')
|
||||||
|
call assert_match("^\n20\n21\n.*\n99\n100$", a)
|
||||||
|
call assert_equal(100, line('.'))
|
||||||
|
|
||||||
|
let a = execute('20z-1000')
|
||||||
|
call assert_match("^\n1\n2\n.*\n19\n20$", a)
|
||||||
|
call assert_equal(20, line('.'))
|
||||||
|
|
||||||
|
let a = execute('20z=1000')
|
||||||
|
call assert_match("^\n1\n.*\n-\\+\n20\n-\\\+\n.*\n100$", a)
|
||||||
|
call assert_equal(20, line('.'))
|
||||||
|
|
||||||
|
call assert_fails('20z=a', 'E144:')
|
||||||
|
|
||||||
|
set window& scroll&
|
||||||
|
bw!
|
||||||
|
endfunc
|
||||||
|
|
||||||
|
func Test_z_bug()
|
||||||
|
" This used to access invalid memory as a result of an integer overflow
|
||||||
|
" and freeze vim.
|
||||||
|
normal ox
|
||||||
|
normal Heat
|
||||||
|
z777777776666666
|
||||||
|
')
|
||||||
|
endfunc
|
Reference in New Issue
Block a user