Skip to content

Commit b7081e1

Browse files
committed
patch 8.2.3402: invalid memory access when using :retab with large value
Problem: Invalid memory access when using :retab with large value. Solution: Check the number is positive.
1 parent 10c83dd commit b7081e1

File tree

5 files changed

+34
-21
lines changed

5 files changed

+34
-21
lines changed

src/indent.c

Lines changed: 21 additions & 13 deletions
Original file line numberDiff line numberDiff line change
@@ -18,18 +18,19 @@
1818
/*
1919
* Set the integer values corresponding to the string setting of 'vartabstop'.
2020
* "array" will be set, caller must free it if needed.
21+
* Return FAIL for an error.
2122
*/
2223
int
2324
tabstop_set(char_u *var, int **array)
2425
{
25-
int valcount = 1;
26-
int t;
27-
char_u *cp;
26+
int valcount = 1;
27+
int t;
28+
char_u *cp;
2829

2930
if (var[0] == NUL || (var[0] == '0' && var[1] == NUL))
3031
{
3132
*array = NULL;
32-
return TRUE;
33+
return OK;
3334
}
3435

3536
for (cp = var; *cp != NUL; ++cp)
@@ -43,8 +44,8 @@ tabstop_set(char_u *var, int **array)
4344
if (cp != end)
4445
emsg(_(e_positive));
4546
else
46-
emsg(_(e_invarg));
47-
return FALSE;
47+
semsg(_(e_invarg2), cp);
48+
return FAIL;
4849
}
4950
}
5051

@@ -55,26 +56,33 @@ tabstop_set(char_u *var, int **array)
5556
++valcount;
5657
continue;
5758
}
58-
emsg(_(e_invarg));
59-
return FALSE;
59+
semsg(_(e_invarg2), var);
60+
return FAIL;
6061
}
6162

6263
*array = ALLOC_MULT(int, valcount + 1);
6364
if (*array == NULL)
64-
return FALSE;
65+
return FAIL;
6566
(*array)[0] = valcount;
6667

6768
t = 1;
6869
for (cp = var; *cp != NUL;)
6970
{
70-
(*array)[t++] = atoi((char *)cp);
71-
while (*cp != NUL && *cp != ',')
71+
int n = atoi((char *)cp);
72+
73+
if (n < 0 || n > 9999)
74+
{
75+
semsg(_(e_invarg2), cp);
76+
return FAIL;
77+
}
78+
(*array)[t++] = n;
79+
while (*cp != NUL && *cp != ',')
7280
++cp;
7381
if (*cp != NUL)
7482
++cp;
7583
}
7684

77-
return TRUE;
85+
return OK;
7886
}
7987

8088
/*
@@ -1591,7 +1599,7 @@ ex_retab(exarg_T *eap)
15911599

15921600
#ifdef FEAT_VARTABS
15931601
new_ts_str = eap->arg;
1594-
if (!tabstop_set(eap->arg, &new_vts_array))
1602+
if (tabstop_set(eap->arg, &new_vts_array) == FAIL)
15951603
return;
15961604
while (vim_isdigit(*(eap->arg)) || *(eap->arg) == ',')
15971605
++(eap->arg);

src/option.c

Lines changed: 6 additions & 6 deletions
Original file line numberDiff line numberDiff line change
@@ -2449,9 +2449,9 @@ didset_options2(void)
24492449
#endif
24502450
#ifdef FEAT_VARTABS
24512451
vim_free(curbuf->b_p_vsts_array);
2452-
tabstop_set(curbuf->b_p_vsts, &curbuf->b_p_vsts_array);
2452+
(void)tabstop_set(curbuf->b_p_vsts, &curbuf->b_p_vsts_array);
24532453
vim_free(curbuf->b_p_vts_array);
2454-
tabstop_set(curbuf->b_p_vts, &curbuf->b_p_vts_array);
2454+
(void)tabstop_set(curbuf->b_p_vts, &curbuf->b_p_vts_array);
24552455
#endif
24562456
}
24572457

@@ -5947,7 +5947,7 @@ buf_copy_options(buf_T *buf, int flags)
59475947
buf->b_p_vsts = vim_strsave(p_vsts);
59485948
COPY_OPT_SCTX(buf, BV_VSTS);
59495949
if (p_vsts && p_vsts != empty_option)
5950-
tabstop_set(p_vsts, &buf->b_p_vsts_array);
5950+
(void)tabstop_set(p_vsts, &buf->b_p_vsts_array);
59515951
else
59525952
buf->b_p_vsts_array = 0;
59535953
buf->b_p_vsts_nopaste = p_vsts_nopaste
@@ -6107,7 +6107,7 @@ buf_copy_options(buf_T *buf, int flags)
61076107
buf->b_p_isk = save_p_isk;
61086108
#ifdef FEAT_VARTABS
61096109
if (p_vts && p_vts != empty_option && !buf->b_p_vts_array)
6110-
tabstop_set(p_vts, &buf->b_p_vts_array);
6110+
(void)tabstop_set(p_vts, &buf->b_p_vts_array);
61116111
else
61126112
buf->b_p_vts_array = NULL;
61136113
#endif
@@ -6122,7 +6122,7 @@ buf_copy_options(buf_T *buf, int flags)
61226122
buf->b_p_vts = vim_strsave(p_vts);
61236123
COPY_OPT_SCTX(buf, BV_VTS);
61246124
if (p_vts && p_vts != empty_option && !buf->b_p_vts_array)
6125-
tabstop_set(p_vts, &buf->b_p_vts_array);
6125+
(void)tabstop_set(p_vts, &buf->b_p_vts_array);
61266126
else
61276127
buf->b_p_vts_array = NULL;
61286128
#endif
@@ -6818,7 +6818,7 @@ paste_option_changed(void)
68186818
if (buf->b_p_vsts_array)
68196819
vim_free(buf->b_p_vsts_array);
68206820
if (buf->b_p_vsts && buf->b_p_vsts != empty_option)
6821-
tabstop_set(buf->b_p_vsts, &buf->b_p_vsts_array);
6821+
(void)tabstop_set(buf->b_p_vsts, &buf->b_p_vsts_array);
68226822
else
68236823
buf->b_p_vsts_array = 0;
68246824
#endif

src/optionstr.c

Lines changed: 2 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -2240,7 +2240,7 @@ did_set_string_option(
22402240
if (errmsg == NULL)
22412241
{
22422242
int *oldarray = curbuf->b_p_vsts_array;
2243-
if (tabstop_set(*varp, &(curbuf->b_p_vsts_array)))
2243+
if (tabstop_set(*varp, &(curbuf->b_p_vsts_array)) == OK)
22442244
{
22452245
if (oldarray)
22462246
vim_free(oldarray);
@@ -2279,7 +2279,7 @@ did_set_string_option(
22792279
{
22802280
int *oldarray = curbuf->b_p_vts_array;
22812281

2282-
if (tabstop_set(*varp, &(curbuf->b_p_vts_array)))
2282+
if (tabstop_set(*varp, &(curbuf->b_p_vts_array)) == OK)
22832283
{
22842284
vim_free(oldarray);
22852285
#ifdef FEAT_FOLDING

src/testdir/test_retab.vim

Lines changed: 3 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -75,6 +75,9 @@ endfunc
7575
func Test_retab_error()
7676
call assert_fails('retab -1', 'E487:')
7777
call assert_fails('retab! -1', 'E487:')
78+
call assert_fails('ret -1000', 'E487:')
79+
call assert_fails('ret 10000', 'E475:')
80+
call assert_fails('ret 80000000000000000000', 'E475:')
7881
endfunc
7982

8083
" vim: shiftwidth=2 sts=2 expandtab

src/version.c

Lines changed: 2 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -755,6 +755,8 @@ static char *(features[]) =
755755

756756
static int included_patches[] =
757757
{ /* Add new patch number below this line */
758+
/**/
759+
3402,
758760
/**/
759761
3401,
760762
/**/

0 commit comments

Comments
 (0)