0
0
mirror of https://github.com/vim/vim.git synced 2025-09-25 03:54:15 -04:00

patch 7.4.1564

Problem:    An empty list in function() causes an error.
Solution:   Handle an empty list like there is no list of arguments.
This commit is contained in:
Bram Moolenaar
2016-03-15 12:36:08 +01:00
parent 790500a8e6
commit 346418c624
3 changed files with 39 additions and 17 deletions

View File

@@ -11786,6 +11786,10 @@ f_function(typval_T *argvars, typval_T *rettv)
EMSG2(_("E700: Unknown function: %s"), s); EMSG2(_("E700: Unknown function: %s"), s);
else else
{ {
int dict_idx = 0;
int arg_idx = 0;
list_T *list = NULL;
if (STRNCMP(s, "s:", 2) == 0 || STRNCMP(s, "<SID>", 5) == 0) if (STRNCMP(s, "s:", 2) == 0 || STRNCMP(s, "<SID>", 5) == 0)
{ {
char sid_buf[25]; char sid_buf[25];
@@ -11808,10 +11812,6 @@ f_function(typval_T *argvars, typval_T *rettv)
if (argvars[1].v_type != VAR_UNKNOWN) if (argvars[1].v_type != VAR_UNKNOWN)
{ {
partial_T *pt;
int dict_idx = 0;
int arg_idx = 0;
if (argvars[2].v_type != VAR_UNKNOWN) if (argvars[2].v_type != VAR_UNKNOWN)
{ {
/* function(name, [args], dict) */ /* function(name, [args], dict) */
@@ -11824,27 +11824,38 @@ f_function(typval_T *argvars, typval_T *rettv)
else else
/* function(name, [args]) */ /* function(name, [args]) */
arg_idx = 1; arg_idx = 1;
if (dict_idx > 0 && (argvars[dict_idx].v_type != VAR_DICT if (dict_idx > 0)
|| argvars[dict_idx].vval.v_dict == NULL)) {
if (argvars[dict_idx].v_type != VAR_DICT)
{ {
EMSG(_("E922: expected a dict")); EMSG(_("E922: expected a dict"));
vim_free(name); vim_free(name);
return; return;
} }
if (arg_idx > 0 && (argvars[arg_idx].v_type != VAR_LIST if (argvars[dict_idx].vval.v_dict == NULL)
|| argvars[arg_idx].vval.v_list == NULL)) dict_idx = 0;
}
if (arg_idx > 0)
{
if (argvars[arg_idx].v_type != VAR_LIST)
{ {
EMSG(_("E923: Second argument of function() must be a list or a dict")); EMSG(_("E923: Second argument of function() must be a list or a dict"));
vim_free(name); vim_free(name);
return; return;
} }
list = argvars[arg_idx].vval.v_list;
if (list == NULL || list->lv_len == 0)
arg_idx = 0;
}
}
if (dict_idx > 0 || arg_idx > 0)
{
partial_T *pt = (partial_T *)alloc_clear(sizeof(partial_T));
pt = (partial_T *)alloc_clear(sizeof(partial_T));
if (pt != NULL) if (pt != NULL)
{ {
if (arg_idx > 0) if (arg_idx > 0)
{ {
list_T *list = argvars[arg_idx].vval.v_list;
listitem_T *li; listitem_T *li;
int i = 0; int i = 0;

View File

@@ -19,6 +19,9 @@ func Test_partial_args()
call assert_equal("foo/bar/xxx", Cb("xxx")) call assert_equal("foo/bar/xxx", Cb("xxx"))
call assert_equal("foo/bar/yyy", call(Cb, ["yyy"])) call assert_equal("foo/bar/yyy", call(Cb, ["yyy"]))
let Cb = function('MyFunc', [])
call assert_equal("a/b/c", Cb("a", "b", "c"))
let Sort = function('MySort', [1]) let Sort = function('MySort', [1])
call assert_equal([1, 2, 3], sort([3, 1, 2], Sort)) call assert_equal([1, 2, 3], sort([3, 1, 2], Sort))
let Sort = function('MySort', [0]) let Sort = function('MySort', [0])
@@ -34,10 +37,16 @@ func Test_partial_dict()
let Cb = function('MyDictFunc', ["foo", "bar"], dict) let Cb = function('MyDictFunc', ["foo", "bar"], dict)
call assert_equal("hello/foo/bar", Cb()) call assert_equal("hello/foo/bar", Cb())
call assert_fails('Cb("xxx")', 'E492:') call assert_fails('Cb("xxx")', 'E492:')
let Cb = function('MyDictFunc', ["foo"], dict) let Cb = function('MyDictFunc', ["foo"], dict)
call assert_equal("hello/foo/xxx", Cb("xxx")) call assert_equal("hello/foo/xxx", Cb("xxx"))
call assert_fails('Cb()', 'E492:') call assert_fails('Cb()', 'E492:')
let Cb = function('MyDictFunc', [], dict)
call assert_equal("hello/ttt/xxx", Cb("ttt", "xxx"))
call assert_fails('Cb("yyy")', 'E492:')
let Cb = function('MyDictFunc', dict) let Cb = function('MyDictFunc', dict)
call assert_equal("hello/xxx/yyy", Cb("xxx", "yyy")) call assert_equal("hello/xxx/yyy", Cb("xxx", "yyy"))
call assert_fails('Cb()', 'E492:') call assert_fails('Cb("fff")', 'E492:')
endfunc endfunc

View File

@@ -743,6 +743,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 */
/**/
1564,
/**/ /**/
1563, 1563,
/**/ /**/