mirror of
https://github.com/neovim/neovim.git
synced 2025-02-25 18:55:25 -06:00
vim-patch:7.4.247
Problem: When passing input to system() there is no way to keep NUL and NL characters separate. Solution: Optionally use a list for the system() input. (ZyX) https://code.google.com/p/vim/source/detail?r=v7-4-247 Implement write_list_to_string() as well since write_list() only works on files.
This commit is contained in:
parent
07bfc11448
commit
d3cd3d2b8f
130
src/nvim/eval.c
130
src/nvim/eval.c
@ -14427,9 +14427,17 @@ static void f_system(typval_T *argvars, typval_T *rettv)
|
|||||||
|
|
||||||
// get input to the shell command (if any), and its length
|
// get input to the shell command (if any), and its length
|
||||||
char_u buf[NUMBUFLEN];
|
char_u buf[NUMBUFLEN];
|
||||||
const char *input = (argvars[1].v_type != VAR_UNKNOWN)
|
const char *input = NULL;
|
||||||
? (char *) get_tv_string_buf_chk(&argvars[1], buf): NULL;
|
size_t input_len = 0;
|
||||||
size_t input_len = input ? strlen(input) : 0;
|
if (argvars[1].v_type == VAR_LIST) {
|
||||||
|
input = write_list_to_string(argvars[1].vval.v_list, &input_len);
|
||||||
|
} else if (argvars[1].v_type != VAR_UNKNOWN) {
|
||||||
|
if ((input = (char *) get_tv_string_buf_chk(&argvars[1], buf))) {
|
||||||
|
input_len = strlen(input);
|
||||||
|
} else {
|
||||||
|
return; // Type error handled in get_tv_string_buf_chk().
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// get shell command to execute
|
// get shell command to execute
|
||||||
const char *cmd = (char *) get_tv_string(&argvars[0]);
|
const char *cmd = (char *) get_tv_string(&argvars[0]);
|
||||||
@ -14439,6 +14447,10 @@ static void f_system(typval_T *argvars, typval_T *rettv)
|
|||||||
char *res = NULL;
|
char *res = NULL;
|
||||||
int status = os_system(cmd, input, input_len, &res, &nread);
|
int status = os_system(cmd, input, input_len, &res, &nread);
|
||||||
|
|
||||||
|
if (argvars[1].v_type == VAR_LIST) {
|
||||||
|
free(input);
|
||||||
|
}
|
||||||
|
|
||||||
set_vim_var_nr(VV_SHELL_ERROR, (long) status);
|
set_vim_var_nr(VV_SHELL_ERROR, (long) status);
|
||||||
|
|
||||||
#ifdef USE_CRNL
|
#ifdef USE_CRNL
|
||||||
@ -15058,6 +15070,59 @@ static void f_winsaveview(typval_T *argvars, typval_T *rettv)
|
|||||||
dict_add_nr_str(dict, "skipcol", (long)curwin->w_skipcol, NULL);
|
dict_add_nr_str(dict, "skipcol", (long)curwin->w_skipcol, NULL);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Writes list of strings to file
|
||||||
|
static bool write_list(FILE *fd, list_T *list, bool binary)
|
||||||
|
{
|
||||||
|
int ret = true;
|
||||||
|
|
||||||
|
for (listitem_T *li = list->lv_first; li != NULL; li = li->li_next) {
|
||||||
|
for (char_u *s = get_tv_string(&li->li_tv); *s != NUL; ++s) {
|
||||||
|
if (putc(*s == '\n' ? NUL : *s, fd) == EOF) {
|
||||||
|
ret = false;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (!binary || li->li_next != NULL) {
|
||||||
|
if (putc('\n', fd) == EOF) {
|
||||||
|
ret = false;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (ret == false) {
|
||||||
|
EMSG(_(e_write));
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Like write_list, but to a string with an out-parameter for the length and
|
||||||
|
/// always assumes binary.
|
||||||
|
static char *write_list_to_string(list_T *list, size_t *len)
|
||||||
|
FUNC_ATTR_MALLOC FUNC_ATTR_NONNULL_ALL FUNC_ATTR_NONNULL_RET
|
||||||
|
{
|
||||||
|
// Calculate the resulting length.
|
||||||
|
*len = 0;
|
||||||
|
for (listitem_T *li = list->lv_first; li != NULL; li = li->li_next) {
|
||||||
|
*len += STRLEN(get_tv_string(&li->li_tv)) + 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
char *ret = xmallocz(*len);
|
||||||
|
char *end = ret;
|
||||||
|
for (listitem_T *li = list->lv_first; li != NULL; li = li->li_next) {
|
||||||
|
for (char *s = (char *) get_tv_string(&li->li_tv); *s != NUL; s++) {
|
||||||
|
*end++ = (*s == '\n') ? NUL : *s;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (li->li_next != NULL) {
|
||||||
|
*end++ = '\n';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
*end = NUL;
|
||||||
|
*len = end - ret;
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* "winwidth(nr)" function
|
* "winwidth(nr)" function
|
||||||
*/
|
*/
|
||||||
@ -15072,68 +15137,43 @@ static void f_winwidth(typval_T *argvars, typval_T *rettv)
|
|||||||
rettv->vval.v_number = wp->w_width;
|
rettv->vval.v_number = wp->w_width;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/// "writefile()" function
|
||||||
* "writefile()" function
|
|
||||||
*/
|
|
||||||
static void f_writefile(typval_T *argvars, typval_T *rettv)
|
static void f_writefile(typval_T *argvars, typval_T *rettv)
|
||||||
{
|
{
|
||||||
int binary = FALSE;
|
rettv->vval.v_number = 0; // Assuming success.
|
||||||
char_u *fname;
|
|
||||||
FILE *fd;
|
|
||||||
listitem_T *li;
|
|
||||||
char_u *s;
|
|
||||||
int ret = 0;
|
|
||||||
int c;
|
|
||||||
|
|
||||||
if (check_restricted() || check_secure())
|
if (check_restricted() || check_secure()) {
|
||||||
return;
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
if (argvars[0].v_type != VAR_LIST) {
|
if (argvars[0].v_type != VAR_LIST) {
|
||||||
EMSG2(_(e_listarg), "writefile()");
|
EMSG2(_(e_listarg), "writefile()");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (argvars[0].vval.v_list == NULL)
|
if (argvars[0].vval.v_list == NULL) {
|
||||||
return;
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool binary = false;
|
||||||
if (argvars[2].v_type != VAR_UNKNOWN
|
if (argvars[2].v_type != VAR_UNKNOWN
|
||||||
&& STRCMP(get_tv_string(&argvars[2]), "b") == 0)
|
&& STRCMP(get_tv_string(&argvars[2]), "b") == 0) {
|
||||||
binary = TRUE;
|
binary = true;
|
||||||
|
}
|
||||||
|
|
||||||
/* Always open the file in binary mode, library functions have a mind of
|
// Always open the file in binary mode, library functions have a mind of
|
||||||
* their own about CR-LF conversion. */
|
// their own about CR-LF conversion.
|
||||||
fname = get_tv_string(&argvars[1]);
|
char_u *fname = get_tv_string(&argvars[1]);
|
||||||
|
FILE *fd;
|
||||||
if (*fname == NUL || (fd = mch_fopen((char *)fname, WRITEBIN)) == NULL) {
|
if (*fname == NUL || (fd = mch_fopen((char *)fname, WRITEBIN)) == NULL) {
|
||||||
EMSG2(_(e_notcreate), *fname == NUL ? (char_u *)_("<empty>") : fname);
|
EMSG2(_(e_notcreate), *fname == NUL ? (char_u *)_("<empty>") : fname);
|
||||||
ret = -1;
|
rettv->vval.v_number = -1;
|
||||||
} else {
|
} else {
|
||||||
for (li = argvars[0].vval.v_list->lv_first; li != NULL;
|
if (write_list(fd, argvars[0].vval.v_list, binary) == false) {
|
||||||
li = li->li_next) {
|
rettv->vval.v_number = -1;
|
||||||
for (s = get_tv_string(&li->li_tv); *s != NUL; ++s) {
|
|
||||||
if (*s == '\n')
|
|
||||||
c = putc(NUL, fd);
|
|
||||||
else
|
|
||||||
c = putc(*s, fd);
|
|
||||||
if (c == EOF) {
|
|
||||||
ret = -1;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (!binary || li->li_next != NULL)
|
|
||||||
if (putc('\n', fd) == EOF) {
|
|
||||||
ret = -1;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
if (ret < 0) {
|
|
||||||
EMSG(_(e_write));
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
fclose(fd);
|
fclose(fd);
|
||||||
}
|
}
|
||||||
|
|
||||||
rettv->vval.v_number = ret;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* "xor(expr, expr)" function
|
* "xor(expr, expr)" function
|
||||||
*/
|
*/
|
||||||
|
@ -394,7 +394,7 @@ static int included_patches[] = {
|
|||||||
//250 NA
|
//250 NA
|
||||||
//249,
|
//249,
|
||||||
//248,
|
//248,
|
||||||
//247,
|
247,
|
||||||
//246,
|
//246,
|
||||||
245,
|
245,
|
||||||
//244,
|
//244,
|
||||||
|
Loading…
Reference in New Issue
Block a user