mirror of
https://github.com/vim/vim.git
synced 2025-09-25 03:54:15 -04:00
patch 8.2.2740: Vim9: lambda with varargs doesn't work
Problem: Vim9: lambda with varargs doesn't work. Solution: Make "...name" work. Require type to be a list.
This commit is contained in:
@@ -395,3 +395,5 @@ EXTERN char e_cannot_lock_unlock_local_variable[]
|
|||||||
INIT(= N_("E1178: Cannot lock or unlock a local variable"));
|
INIT(= N_("E1178: Cannot lock or unlock a local variable"));
|
||||||
EXTERN char e_failed_to_extract_pwd_from_str_check_your_shell_config[]
|
EXTERN char e_failed_to_extract_pwd_from_str_check_your_shell_config[]
|
||||||
INIT(= N_("E1179: Failed to extract PWD from %s, check your shell's config related to OSC 7"));
|
INIT(= N_("E1179: Failed to extract PWD from %s, check your shell's config related to OSC 7"));
|
||||||
|
EXTERN char e_variable_arguments_type_must_be_list_str[]
|
||||||
|
INIT(= N_("E1180: Variable arguments type must be a list: %s"));
|
||||||
|
@@ -791,10 +791,18 @@ def Test_call_funcref_wrong_args()
|
|||||||
enddef
|
enddef
|
||||||
|
|
||||||
def Test_call_lambda_args()
|
def Test_call_lambda_args()
|
||||||
|
var lines =<< trim END
|
||||||
|
var Callback = (..._) => 'anything'
|
||||||
|
assert_equal('anything', Callback())
|
||||||
|
assert_equal('anything', Callback(1))
|
||||||
|
assert_equal('anything', Callback('a', 2))
|
||||||
|
END
|
||||||
|
CheckDefAndScriptSuccess(lines)
|
||||||
|
|
||||||
CheckDefFailure(['echo ((i) => 0)()'],
|
CheckDefFailure(['echo ((i) => 0)()'],
|
||||||
'E119: Not enough arguments for function: ((i) => 0)()')
|
'E119: Not enough arguments for function: ((i) => 0)()')
|
||||||
|
|
||||||
var lines =<< trim END
|
lines =<< trim END
|
||||||
var Ref = (x: number, y: number) => x + y
|
var Ref = (x: number, y: number) => x + y
|
||||||
echo Ref(1, 'x')
|
echo Ref(1, 'x')
|
||||||
END
|
END
|
||||||
@@ -923,13 +931,22 @@ def Test_call_def_varargs()
|
|||||||
|
|
||||||
lines =<< trim END
|
lines =<< trim END
|
||||||
vim9script
|
vim9script
|
||||||
def Func(...l: any)
|
def Func(...l: list<any>)
|
||||||
echo l
|
echo l
|
||||||
enddef
|
enddef
|
||||||
Func(0)
|
Func(0)
|
||||||
END
|
END
|
||||||
CheckScriptSuccess(lines)
|
CheckScriptSuccess(lines)
|
||||||
|
|
||||||
|
lines =<< trim END
|
||||||
|
vim9script
|
||||||
|
def Func(...l: any)
|
||||||
|
echo l
|
||||||
|
enddef
|
||||||
|
Func(0)
|
||||||
|
END
|
||||||
|
CheckScriptFailure(lines, 'E1180:', 2)
|
||||||
|
|
||||||
lines =<< trim END
|
lines =<< trim END
|
||||||
vim9script
|
vim9script
|
||||||
def Func(..._l: list<string>)
|
def Func(..._l: list<string>)
|
||||||
|
@@ -3644,7 +3644,7 @@ enddef
|
|||||||
def Test_catch_exception_in_callback()
|
def Test_catch_exception_in_callback()
|
||||||
var lines =<< trim END
|
var lines =<< trim END
|
||||||
vim9script
|
vim9script
|
||||||
def Callback(...l: any)
|
def Callback(...l: list<any>)
|
||||||
try
|
try
|
||||||
var x: string
|
var x: string
|
||||||
var y: string
|
var y: string
|
||||||
@@ -3669,10 +3669,10 @@ def Test_no_unknown_error_after_error()
|
|||||||
var lines =<< trim END
|
var lines =<< trim END
|
||||||
vim9script
|
vim9script
|
||||||
var source: list<number>
|
var source: list<number>
|
||||||
def Out_cb(...l: any)
|
def Out_cb(...l: list<any>)
|
||||||
eval [][0]
|
eval [][0]
|
||||||
enddef
|
enddef
|
||||||
def Exit_cb(...l: any)
|
def Exit_cb(...l: list<any>)
|
||||||
sleep 1m
|
sleep 1m
|
||||||
source += l
|
source += l
|
||||||
enddef
|
enddef
|
||||||
|
@@ -68,6 +68,7 @@ one_function_arg(
|
|||||||
garray_T *argtypes,
|
garray_T *argtypes,
|
||||||
int types_optional,
|
int types_optional,
|
||||||
evalarg_T *evalarg,
|
evalarg_T *evalarg,
|
||||||
|
int is_vararg,
|
||||||
int skip)
|
int skip)
|
||||||
{
|
{
|
||||||
char_u *p = arg;
|
char_u *p = arg;
|
||||||
@@ -155,7 +156,8 @@ one_function_arg(
|
|||||||
{
|
{
|
||||||
if (type == NULL && types_optional)
|
if (type == NULL && types_optional)
|
||||||
// lambda arguments default to "any" type
|
// lambda arguments default to "any" type
|
||||||
type = vim_strsave((char_u *)"any");
|
type = vim_strsave((char_u *)
|
||||||
|
(is_vararg ? "list<any>" : "any"));
|
||||||
((char_u **)argtypes->ga_data)[argtypes->ga_len++] = type;
|
((char_u **)argtypes->ga_data)[argtypes->ga_len++] = type;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -250,7 +252,7 @@ get_function_args(
|
|||||||
|
|
||||||
arg = p;
|
arg = p;
|
||||||
p = one_function_arg(p, newargs, argtypes, types_optional,
|
p = one_function_arg(p, newargs, argtypes, types_optional,
|
||||||
evalarg, skip);
|
evalarg, TRUE, skip);
|
||||||
if (p == arg)
|
if (p == arg)
|
||||||
break;
|
break;
|
||||||
if (*skipwhite(p) == '=')
|
if (*skipwhite(p) == '=')
|
||||||
@@ -264,7 +266,7 @@ get_function_args(
|
|||||||
{
|
{
|
||||||
arg = p;
|
arg = p;
|
||||||
p = one_function_arg(p, newargs, argtypes, types_optional,
|
p = one_function_arg(p, newargs, argtypes, types_optional,
|
||||||
evalarg, skip);
|
evalarg, FALSE, skip);
|
||||||
if (p == arg)
|
if (p == arg)
|
||||||
break;
|
break;
|
||||||
|
|
||||||
@@ -360,12 +362,14 @@ err_ret:
|
|||||||
static int
|
static int
|
||||||
parse_argument_types(ufunc_T *fp, garray_T *argtypes, int varargs)
|
parse_argument_types(ufunc_T *fp, garray_T *argtypes, int varargs)
|
||||||
{
|
{
|
||||||
|
int len = 0;
|
||||||
|
|
||||||
ga_init2(&fp->uf_type_list, sizeof(type_T *), 10);
|
ga_init2(&fp->uf_type_list, sizeof(type_T *), 10);
|
||||||
if (argtypes->ga_len > 0)
|
if (argtypes->ga_len > 0)
|
||||||
{
|
{
|
||||||
// When "varargs" is set the last name/type goes into uf_va_name
|
// When "varargs" is set the last name/type goes into uf_va_name
|
||||||
// and uf_va_type.
|
// and uf_va_type.
|
||||||
int len = argtypes->ga_len - (varargs ? 1 : 0);
|
len = argtypes->ga_len - (varargs ? 1 : 0);
|
||||||
|
|
||||||
if (len > 0)
|
if (len > 0)
|
||||||
fp->uf_arg_types = ALLOC_CLEAR_MULT(type_T *, len);
|
fp->uf_arg_types = ALLOC_CLEAR_MULT(type_T *, len);
|
||||||
@@ -388,6 +392,8 @@ parse_argument_types(ufunc_T *fp, garray_T *argtypes, int varargs)
|
|||||||
fp->uf_arg_types[i] = type;
|
fp->uf_arg_types[i] = type;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (varargs)
|
if (varargs)
|
||||||
{
|
{
|
||||||
char_u *p;
|
char_u *p;
|
||||||
@@ -399,14 +405,22 @@ parse_argument_types(ufunc_T *fp, garray_T *argtypes, int varargs)
|
|||||||
--fp->uf_args.ga_len;
|
--fp->uf_args.ga_len;
|
||||||
p = ((char_u **)argtypes->ga_data)[len];
|
p = ((char_u **)argtypes->ga_data)[len];
|
||||||
if (p == NULL)
|
if (p == NULL)
|
||||||
// todo: get type from default value
|
// TODO: get type from default value
|
||||||
fp->uf_va_type = &t_any;
|
fp->uf_va_type = &t_list_any;
|
||||||
else
|
else
|
||||||
|
{
|
||||||
fp->uf_va_type = parse_type(&p, &fp->uf_type_list, TRUE);
|
fp->uf_va_type = parse_type(&p, &fp->uf_type_list, TRUE);
|
||||||
if (fp->uf_va_type == NULL)
|
if (fp->uf_va_type != NULL && fp->uf_va_type->tt_type != VAR_LIST)
|
||||||
|
{
|
||||||
|
semsg(_(e_variable_arguments_type_must_be_list_str),
|
||||||
|
((char_u **)argtypes->ga_data)[len]);
|
||||||
return FAIL;
|
return FAIL;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
if (fp->uf_va_type == NULL)
|
||||||
|
return FAIL;
|
||||||
|
}
|
||||||
|
|
||||||
return OK;
|
return OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1236,7 +1250,8 @@ get_lambda_tv(
|
|||||||
ga_init(&fp->uf_def_args);
|
ga_init(&fp->uf_def_args);
|
||||||
if (types_optional)
|
if (types_optional)
|
||||||
{
|
{
|
||||||
if (parse_argument_types(fp, &argtypes, FALSE) == FAIL)
|
if (parse_argument_types(fp, &argtypes,
|
||||||
|
in_vim9script() && varargs) == FAIL)
|
||||||
goto errret;
|
goto errret;
|
||||||
if (ret_type != NULL)
|
if (ret_type != NULL)
|
||||||
{
|
{
|
||||||
@@ -1264,8 +1279,8 @@ get_lambda_tv(
|
|||||||
if (sandbox)
|
if (sandbox)
|
||||||
flags |= FC_SANDBOX;
|
flags |= FC_SANDBOX;
|
||||||
// In legacy script a lambda can be called with more args than
|
// In legacy script a lambda can be called with more args than
|
||||||
// uf_args.ga_len.
|
// uf_args.ga_len. In Vim9 script "...name" has to be used.
|
||||||
fp->uf_varargs = !in_vim9script();
|
fp->uf_varargs = !in_vim9script() || varargs;
|
||||||
fp->uf_flags = flags;
|
fp->uf_flags = flags;
|
||||||
fp->uf_calls = 0;
|
fp->uf_calls = 0;
|
||||||
fp->uf_script_ctx = current_sctx;
|
fp->uf_script_ctx = current_sctx;
|
||||||
@@ -3190,7 +3205,7 @@ list_func_head(ufunc_T *fp, int indent)
|
|||||||
msg_puts(", ");
|
msg_puts(", ");
|
||||||
msg_puts("...");
|
msg_puts("...");
|
||||||
msg_puts((char *)fp->uf_va_name);
|
msg_puts((char *)fp->uf_va_name);
|
||||||
if (fp->uf_va_type)
|
if (fp->uf_va_type != NULL)
|
||||||
{
|
{
|
||||||
char *tofree;
|
char *tofree;
|
||||||
|
|
||||||
|
@@ -750,6 +750,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 */
|
||||||
|
/**/
|
||||||
|
2740,
|
||||||
/**/
|
/**/
|
||||||
2739,
|
2739,
|
||||||
/**/
|
/**/
|
||||||
|
@@ -1856,7 +1856,8 @@ generate_CALL(cctx_T *cctx, ufunc_T *ufunc, int pushed_argcount)
|
|||||||
continue;
|
continue;
|
||||||
expected = ufunc->uf_arg_types[i];
|
expected = ufunc->uf_arg_types[i];
|
||||||
}
|
}
|
||||||
else if (ufunc->uf_va_type == NULL || ufunc->uf_va_type == &t_any)
|
else if (ufunc->uf_va_type == NULL
|
||||||
|
|| ufunc->uf_va_type == &t_list_any)
|
||||||
// possibly a lambda or "...: any"
|
// possibly a lambda or "...: any"
|
||||||
expected = &t_any;
|
expected = &t_any;
|
||||||
else
|
else
|
||||||
@@ -9069,7 +9070,7 @@ set_function_type(ufunc_T *ufunc)
|
|||||||
if (varargs)
|
if (varargs)
|
||||||
{
|
{
|
||||||
ufunc->uf_func_type->tt_args[argcount] =
|
ufunc->uf_func_type->tt_args[argcount] =
|
||||||
ufunc->uf_va_type == NULL ? &t_any : ufunc->uf_va_type;
|
ufunc->uf_va_type == NULL ? &t_list_any : ufunc->uf_va_type;
|
||||||
ufunc->uf_func_type->tt_flags = TTFLAG_VARARGS;
|
ufunc->uf_func_type->tt_flags = TTFLAG_VARARGS;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -1374,7 +1374,7 @@ call_def_function(
|
|||||||
// Check the type of the list items.
|
// Check the type of the list items.
|
||||||
tv = STACK_TV_BOT(-1);
|
tv = STACK_TV_BOT(-1);
|
||||||
if (ufunc->uf_va_type != NULL
|
if (ufunc->uf_va_type != NULL
|
||||||
&& ufunc->uf_va_type != &t_any
|
&& ufunc->uf_va_type != &t_list_any
|
||||||
&& ufunc->uf_va_type->tt_member != &t_any
|
&& ufunc->uf_va_type->tt_member != &t_any
|
||||||
&& tv->vval.v_list != NULL)
|
&& tv->vval.v_list != NULL)
|
||||||
{
|
{
|
||||||
|
Reference in New Issue
Block a user