Skip to content

Commit

Permalink
fix: snprintf buffer overflow detected by -D_FORTIFY_SOURCE=3
Browse files Browse the repository at this point in the history
Thread no. 1 (24 frames)
 neovim#8 snprintf at /usr/include/bits/stdio2.h:54
 neovim#9 set_cmdarg at /usr/src/debug/neovim-0.8.2-2.fc38.x86_64/src/nvim/eval.c:7044
 neovim#10 apply_autocmds_group at /usr/src/debug/neovim-0.8.2-2.fc38.x86_64/src/nvim/autocmd.c:1843
 neovim#11 apply_autocmds_exarg at /usr/src/debug/neovim-0.8.2-2.fc38.x86_64/src/nvim/autocmd.c:1549
 neovim#12 readfile at /usr/src/debug/neovim-0.8.2-2.fc38.x86_64/src/nvim/fileio.c:617
 neovim#13 buf_reload at /usr/src/debug/neovim-0.8.2-2.fc38.x86_64/src/nvim/fileio.c:5038
 neovim#14 buf_check_timestamp at /usr/src/debug/neovim-0.8.2-2.fc38.x86_64/src/nvim/fileio.c:4952
 neovim#15 check_timestamps at /usr/src/debug/neovim-0.8.2-2.fc38.x86_64/src/nvim/fileio.c:4678
 neovim#16 ex_checktime at /usr/src/debug/neovim-0.8.2-2.fc38.x86_64/src/nvim/ex_cmds2.c:765
 neovim#17 execute_cmd0 at /usr/src/debug/neovim-0.8.2-2.fc38.x86_64/src/nvim/ex_docmd.c:1620
 neovim#18 do_one_cmd at /usr/src/debug/neovim-0.8.2-2.fc38.x86_64/src/nvim/ex_docmd.c:2275
 neovim#19 do_cmdline at /usr/src/debug/neovim-0.8.2-2.fc38.x86_64/src/nvim/ex_docmd.c:584
 neovim#20 ex_execute at /usr/src/debug/neovim-0.8.2-2.fc38.x86_64/src/nvim/eval.c:7727
 neovim#21 execute_cmd0 at /usr/src/debug/neovim-0.8.2-2.fc38.x86_64/src/nvim/ex_docmd.c:1620
 neovim#22 do_one_cmd at /usr/src/debug/neovim-0.8.2-2.fc38.x86_64/src/nvim/ex_docmd.c:2275
 neovim#23 do_cmdline at /usr/src/debug/neovim-0.8.2-2.fc38.x86_64/src/nvim/ex_docmd.c:584
 neovim#24 do_ucmd at /usr/src/debug/neovim-0.8.2-2.fc38.x86_64/src/nvim/usercmd.c:1661
 neovim#25 execute_cmd0 at /usr/src/debug/neovim-0.8.2-2.fc38.x86_64/src/nvim/ex_docmd.c:1612
 neovim#26 do_one_cmd at /usr/src/debug/neovim-0.8.2-2.fc38.x86_64/src/nvim/ex_docmd.c:2275
 neovim#27 do_cmdline at /usr/src/debug/neovim-0.8.2-2.fc38.x86_64/src/nvim/ex_docmd.c:584
 neovim#28 nv_colon at /usr/src/debug/neovim-0.8.2-2.fc38.x86_64/src/nvim/normal.c:4058
 neovim#29 normal_execute at /usr/src/debug/neovim-0.8.2-2.fc38.x86_64/src/nvim/normal.c:1172
 neovim#30 state_enter at /usr/src/debug/neovim-0.8.2-2.fc38.x86_64/src/nvim/state.c:88
 neovim#31 normal_enter at /usr/src/debug/neovim-0.8.2-2.fc38.x86_64/src/nvim/normal.c:471

Signed-off-by: Andreas Schneider <asn@cryptomilk.org>
  • Loading branch information
cryptomilk committed Mar 25, 2023
1 parent 42876dd commit f5766ae
Showing 1 changed file with 43 additions and 22 deletions.
65 changes: 43 additions & 22 deletions src/nvim/eval.c
Original file line number Diff line number Diff line change
Expand Up @@ -4076,18 +4076,18 @@ bool garbage_collect(bool testing)
}

// The execution stack can grow big, limit the size.
if (exestack.ga_maxlen - exestack.ga_len > 500) {
if (exestack.ga_marc - exestack.ga_len > 500) {
// Keep 150% of the current size, with a minimum of the growth size.
int n = exestack.ga_len / 2;
if (n < exestack.ga_growsize) {
n = exestack.ga_growsize;
}

// Don't make it bigger though.
if (exestack.ga_len + n < exestack.ga_maxlen) {
if (exestack.ga_len + n < exestack.ga_marc) {
size_t new_len = (size_t)exestack.ga_itemsize * (size_t)(exestack.ga_len + n);
char *pp = xrealloc(exestack.ga_data, new_len);
exestack.ga_maxlen = exestack.ga_len + n;
exestack.ga_marc = exestack.ga_len + n;
exestack.ga_data = pp;
}
}
Expand Down Expand Up @@ -6771,20 +6771,18 @@ char *set_cmdarg(exarg_T *eap, char *oldarg)
{
char *oldval = vimvars[VV_CMDARG].vv_str;
if (eap == NULL) {
xfree(oldval);
vimvars[VV_CMDARG].vv_str = oldarg;
return NULL;
goto error;
}

size_t len = 0;
if (eap->force_bin == FORCE_BIN) {
len = 6;
len += 6; // " ++bin"
} else if (eap->force_bin == FORCE_NOBIN) {
len = 8;
len += 8; // " ++nobin"
}

if (eap->read_edit) {
len += 7;
len += 7; // " ++edit"
}

if (eap->force_ff != 0) {
Expand All @@ -6797,48 +6795,71 @@ char *set_cmdarg(exarg_T *eap, char *oldarg)
len += 7 + 4; // " ++bad=" + "keep" or "drop"
}
if (eap->mkdir_p != 0) {
len += 4;
len += 4; // " ++p"
}

const size_t newval_len = len + 1;
char *newval = xmalloc(newval_len);
int xlen = 0;

if (eap->force_bin == FORCE_BIN) {
snprintf(newval, newval_len, " ++bin");
xlen = snprintf(newval, newval_len, " ++bin");
} else if (eap->force_bin == FORCE_NOBIN) {
snprintf(newval, newval_len, " ++nobin");
xlen = snprintf(newval, newval_len, " ++nobin");
} else {
*newval = NUL;
*newval = '\0';
}
if (xlen < 0) {
goto error;
}

if (eap->read_edit) {
STRCAT(newval, " ++edit");
xlen = snprintf(newval + xlen, newval_len - xlen, " ++edit")
if (xlen < 0) {
goto error;
}
}

if (eap->force_ff != 0) {
snprintf(newval + strlen(newval), newval_len, " ++ff=%s",
xlen = snprintf(newval + xlen, newval_len - xlen, " ++ff=%s",
eap->force_ff == 'u' ? "unix" :
eap->force_ff == 'd' ? "dos" : "mac");
if (xlen < 0) {
goto error;
}
}
if (eap->force_enc != 0) {
snprintf(newval + strlen(newval), newval_len, " ++enc=%s",
eap->cmd + eap->force_enc);
xlen = snprintf(newval + (xlen), newval_len - xlen, " ++enc=%s", eap->cmd + eap->force_enc);
if (xlen < 0) {
goto error;
}
}

if (eap->bad_char == BAD_KEEP) {
STRCPY(newval + strlen(newval), " ++bad=keep");
xlen = snprintf(newval + xlen, newval_len - xlen, " ++bad=keep");
} else if (eap->bad_char == BAD_DROP) {
STRCPY(newval + strlen(newval), " ++bad=drop");
xlen = snprintf(newval + xlen, newval_len - xlen, " ++bad=drop");
} else if (eap->bad_char != 0) {
snprintf(newval + strlen(newval), newval_len, " ++bad=%c",
eap->bad_char);
xlen = snprintf(newval + xlen, newval_len - xlen, " ++bad=%c", eap->bad_char);
}
if (xlen < 0) {
goto error;
}

if (eap->mkdir_p) {
snprintf(newval, newval_len, " ++p");
xlen = snprintf(newval + xlen, newval_len - xlen, " ++p");
if (xlen < 0) {
goto error;
}
}

vimvars[VV_CMDARG].vv_str = newval;
return oldval;

error:
xfree(oldval);
vimvars[VV_CMDARG].vv_str = oldarg;
return NULL;
}

/// Check if variable "name[len]" is a local variable or an argument.
Expand Down

0 comments on commit f5766ae

Please sign in to comment.