mirror of
https://github.com/vim/vim.git
synced 2025-09-26 04:04:07 -04:00
patch 8.2.3117: Vim9: type not properly checked in for loop
Problem: Vim9: type not properly checked in for loop. Solution: Have items() return a list of lists. Add runtime type checks. (closes #8515)
This commit is contained in:
@@ -522,6 +522,11 @@ ret_list_dict_any(int argcount UNUSED, type_T **argtypes UNUSED)
|
|||||||
return &t_list_dict_any;
|
return &t_list_dict_any;
|
||||||
}
|
}
|
||||||
static type_T *
|
static type_T *
|
||||||
|
ret_list_items(int argcount, type_T **argtypes UNUSED)
|
||||||
|
{
|
||||||
|
return &t_list_list_any;
|
||||||
|
}
|
||||||
|
static type_T *
|
||||||
ret_dict_any(int argcount UNUSED, type_T **argtypes UNUSED)
|
ret_dict_any(int argcount UNUSED, type_T **argtypes UNUSED)
|
||||||
{
|
{
|
||||||
return &t_dict_any;
|
return &t_dict_any;
|
||||||
@@ -1166,7 +1171,7 @@ static funcentry_T global_functions[] =
|
|||||||
{"isnan", 1, 1, FEARG_1, arg1_float_or_nr,
|
{"isnan", 1, 1, FEARG_1, arg1_float_or_nr,
|
||||||
ret_number_bool, MATH_FUNC(f_isnan)},
|
ret_number_bool, MATH_FUNC(f_isnan)},
|
||||||
{"items", 1, 1, FEARG_1, arg1_dict,
|
{"items", 1, 1, FEARG_1, arg1_dict,
|
||||||
ret_list_any, f_items},
|
ret_list_items, f_items},
|
||||||
{"job_getchannel", 1, 1, FEARG_1, NULL,
|
{"job_getchannel", 1, 1, FEARG_1, NULL,
|
||||||
ret_channel, JOB_FUNC(f_job_getchannel)},
|
ret_channel, JOB_FUNC(f_job_getchannel)},
|
||||||
{"job_info", 0, 1, FEARG_1, NULL,
|
{"job_info", 0, 1, FEARG_1, NULL,
|
||||||
@@ -3687,6 +3692,7 @@ ret_f_function(int argcount, type_T **argtypes)
|
|||||||
{
|
{
|
||||||
if (argcount == 1 && argtypes[0]->tt_type == VAR_STRING)
|
if (argcount == 1 && argtypes[0]->tt_type == VAR_STRING)
|
||||||
return &t_func_any;
|
return &t_func_any;
|
||||||
|
// Need to check the type at runtime, the function may be defined later.
|
||||||
return &t_func_unknown;
|
return &t_func_unknown;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -441,6 +441,7 @@ EXTERN type_T t_list_number INIT6(VAR_LIST, 0, 0, TTFLAG_STATIC, &t_number, NULL
|
|||||||
EXTERN type_T t_list_string INIT6(VAR_LIST, 0, 0, TTFLAG_STATIC, &t_string, NULL);
|
EXTERN type_T t_list_string INIT6(VAR_LIST, 0, 0, TTFLAG_STATIC, &t_string, NULL);
|
||||||
EXTERN type_T t_list_job INIT6(VAR_LIST, 0, 0, TTFLAG_STATIC, &t_job, NULL);
|
EXTERN type_T t_list_job INIT6(VAR_LIST, 0, 0, TTFLAG_STATIC, &t_job, NULL);
|
||||||
EXTERN type_T t_list_dict_any INIT6(VAR_LIST, 0, 0, TTFLAG_STATIC, &t_dict_any, NULL);
|
EXTERN type_T t_list_dict_any INIT6(VAR_LIST, 0, 0, TTFLAG_STATIC, &t_dict_any, NULL);
|
||||||
|
EXTERN type_T t_list_list_any INIT6(VAR_LIST, 0, 0, TTFLAG_STATIC, &t_list_any, NULL);
|
||||||
|
|
||||||
EXTERN type_T t_dict_bool INIT6(VAR_DICT, 0, 0, TTFLAG_STATIC, &t_bool, NULL);
|
EXTERN type_T t_dict_bool INIT6(VAR_DICT, 0, 0, TTFLAG_STATIC, &t_bool, NULL);
|
||||||
EXTERN type_T t_dict_number INIT6(VAR_DICT, 0, 0, TTFLAG_STATIC, &t_number, NULL);
|
EXTERN type_T t_dict_number INIT6(VAR_DICT, 0, 0, TTFLAG_STATIC, &t_number, NULL);
|
||||||
|
@@ -2573,6 +2573,14 @@ def Test_for_loop_fails()
|
|||||||
endfor
|
endfor
|
||||||
END
|
END
|
||||||
CheckDefAndScriptFailure(lines, 'E1059:', 1)
|
CheckDefAndScriptFailure(lines, 'E1059:', 1)
|
||||||
|
|
||||||
|
lines =<< trim END
|
||||||
|
var d: dict<number> = {a: 1, b: 2}
|
||||||
|
for [k: job, v: job] in d->items()
|
||||||
|
echo k v
|
||||||
|
endfor
|
||||||
|
END
|
||||||
|
CheckDefExecAndScriptFailure(lines, 'E1012: Type mismatch; expected job but got string', 2)
|
||||||
enddef
|
enddef
|
||||||
|
|
||||||
def Test_for_loop_script_var()
|
def Test_for_loop_script_var()
|
||||||
|
@@ -755,6 +755,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 */
|
||||||
|
/**/
|
||||||
|
3117,
|
||||||
/**/
|
/**/
|
||||||
3116,
|
3116,
|
||||||
/**/
|
/**/
|
||||||
|
@@ -7932,8 +7932,11 @@ compile_for(char_u *arg_start, cctx_T *cctx)
|
|||||||
if (lhs_type == &t_any)
|
if (lhs_type == &t_any)
|
||||||
lhs_type = item_type;
|
lhs_type = item_type;
|
||||||
else if (item_type != &t_unknown
|
else if (item_type != &t_unknown
|
||||||
&& !(var_list && item_type == &t_any)
|
&& ((var_list && item_type == &t_any)
|
||||||
&& check_type(lhs_type, item_type, TRUE, where) == FAIL)
|
? need_type(item_type, lhs_type,
|
||||||
|
-1, 0, cctx, FALSE, FALSE)
|
||||||
|
: check_type(lhs_type, item_type, TRUE, where))
|
||||||
|
== FAIL)
|
||||||
goto failed;
|
goto failed;
|
||||||
var_lvar = reserve_local(cctx, arg, varlen, TRUE, lhs_type);
|
var_lvar = reserve_local(cctx, arg, varlen, TRUE, lhs_type);
|
||||||
if (var_lvar == NULL)
|
if (var_lvar == NULL)
|
||||||
|
Reference in New Issue
Block a user