patch 9.0.0419: the :defer command does not check the function arguments
Problem:    The :defer command does not check the function argument count and
            types.
Solution:   Check the function arguments when adding a deferred function.
			
			
This commit is contained in:
		| @ -46,11 +46,14 @@ int generate_JUMP(cctx_T *cctx, jumpwhen_T when, int where); | |||||||
| int generate_JUMP_IF_ARG_SET(cctx_T *cctx, int arg_off); | int generate_JUMP_IF_ARG_SET(cctx_T *cctx, int arg_off); | ||||||
| int generate_FOR(cctx_T *cctx, int loop_idx); | int generate_FOR(cctx_T *cctx, int loop_idx); | ||||||
| int generate_TRYCONT(cctx_T *cctx, int levels, int where); | int generate_TRYCONT(cctx_T *cctx, int levels, int where); | ||||||
|  | int check_internal_func_args(cctx_T *cctx, int func_idx, int argcount, int method_call, type2_T **argtypes, type2_T *shuffled_argtypes); | ||||||
| int generate_BCALL(cctx_T *cctx, int func_idx, int argcount, int method_call); | int generate_BCALL(cctx_T *cctx, int func_idx, int argcount, int method_call); | ||||||
| int generate_LISTAPPEND(cctx_T *cctx); | int generate_LISTAPPEND(cctx_T *cctx); | ||||||
| int generate_BLOBAPPEND(cctx_T *cctx); | int generate_BLOBAPPEND(cctx_T *cctx); | ||||||
|  | int check_args_on_stack(cctx_T *cctx, ufunc_T *ufunc, int argcount); | ||||||
| int generate_CALL(cctx_T *cctx, ufunc_T *ufunc, int pushed_argcount); | int generate_CALL(cctx_T *cctx, ufunc_T *ufunc, int pushed_argcount); | ||||||
| int generate_UCALL(cctx_T *cctx, char_u *name, int argcount); | int generate_UCALL(cctx_T *cctx, char_u *name, int argcount); | ||||||
|  | int check_func_args_from_type(cctx_T *cctx, type_T *type, int argcount, int at_top, char_u *name); | ||||||
| int generate_PCALL(cctx_T *cctx, int argcount, char_u *name, type_T *type, int at_top); | int generate_PCALL(cctx_T *cctx, int argcount, char_u *name, type_T *type, int at_top); | ||||||
| int generate_DEFER(cctx_T *cctx, int var_idx, int argcount); | int generate_DEFER(cctx_T *cctx, int var_idx, int argcount); | ||||||
| int generate_STRINGMEMBER(cctx_T *cctx, char_u *name, size_t len); | int generate_STRINGMEMBER(cctx_T *cctx, char_u *name, size_t len); | ||||||
|  | |||||||
| @ -5,6 +5,7 @@ | |||||||
|  |  | ||||||
| source check.vim | source check.vim | ||||||
| source shared.vim | source shared.vim | ||||||
|  | import './vim9.vim' as v9 | ||||||
|  |  | ||||||
| func Table(title, ...) | func Table(title, ...) | ||||||
|   let ret = a:title |   let ret = a:title | ||||||
| @ -619,7 +620,7 @@ func Test_defer_quitall() | |||||||
|       DeferLevelOne() |       DeferLevelOne() | ||||||
|   END |   END | ||||||
|   call writefile(lines, 'XdeferQuitall', 'D') |   call writefile(lines, 'XdeferQuitall', 'D') | ||||||
|   let res = system(GetVimCommandClean() .. ' -X -S XdeferQuitall') |   let res = system(GetVimCommand() .. ' -X -S XdeferQuitall') | ||||||
|   call assert_equal(0, v:shell_error) |   call assert_equal(0, v:shell_error) | ||||||
|   call assert_false(filereadable('XQuitallOne')) |   call assert_false(filereadable('XQuitallOne')) | ||||||
|   call assert_false(filereadable('XQuitallTwo')) |   call assert_false(filereadable('XQuitallTwo')) | ||||||
| @ -641,7 +642,7 @@ func Test_defer_quitall_in_expr_func() | |||||||
|       call Test_defer_in_funcref() |       call Test_defer_in_funcref() | ||||||
|   END |   END | ||||||
|   call writefile(lines, 'XdeferQuitallExpr', 'D') |   call writefile(lines, 'XdeferQuitallExpr', 'D') | ||||||
|   let res = system(GetVimCommandClean() .. ' -X -S XdeferQuitallExpr') |   let res = system(GetVimCommand() .. ' -X -S XdeferQuitallExpr') | ||||||
|   call assert_equal(0, v:shell_error) |   call assert_equal(0, v:shell_error) | ||||||
|   call assert_false(filereadable('Xentry0')) |   call assert_false(filereadable('Xentry0')) | ||||||
|   call assert_false(filereadable('Xentry1')) |   call assert_false(filereadable('Xentry1')) | ||||||
| @ -695,5 +696,60 @@ def Test_defer_in_funcref() | |||||||
|   assert_false(filereadable('Xentry2')) |   assert_false(filereadable('Xentry2')) | ||||||
| enddef | enddef | ||||||
|  |  | ||||||
|  | func Test_defer_wrong_arguments() | ||||||
|  |   call assert_fails('defer delete()', 'E119:') | ||||||
|  |   call assert_fails('defer FuncIndex(1)', 'E119:') | ||||||
|  |   call assert_fails('defer delete(1, 2, 3)', 'E118:') | ||||||
|  |   call assert_fails('defer FuncIndex(1, 2, 3)', 'E118:') | ||||||
|  |  | ||||||
|  |   let lines =<< trim END | ||||||
|  |       def DeferFunc0() | ||||||
|  |         defer delete() | ||||||
|  |       enddef | ||||||
|  |       defcompile | ||||||
|  |   END | ||||||
|  |   call v9.CheckScriptFailure(lines, 'E119:') | ||||||
|  |   let lines =<< trim END | ||||||
|  |       def DeferFunc3() | ||||||
|  |         defer delete(1, 2, 3) | ||||||
|  |       enddef | ||||||
|  |       defcompile | ||||||
|  |   END | ||||||
|  |   call v9.CheckScriptFailure(lines, 'E118:') | ||||||
|  |   let lines =<< trim END | ||||||
|  |       def DeferFunc2() | ||||||
|  |         defer delete(1, 2) | ||||||
|  |       enddef | ||||||
|  |       defcompile | ||||||
|  |   END | ||||||
|  |   call v9.CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch, expected string but got number') | ||||||
|  |  | ||||||
|  |   def g:FuncOneArg(arg: string) | ||||||
|  |     echo arg | ||||||
|  |   enddef | ||||||
|  |  | ||||||
|  |   let lines =<< trim END | ||||||
|  |       def DeferUserFunc0() | ||||||
|  |         defer g:FuncOneArg() | ||||||
|  |       enddef | ||||||
|  |       defcompile | ||||||
|  |   END | ||||||
|  |   call v9.CheckScriptFailure(lines, 'E119:') | ||||||
|  |   let lines =<< trim END | ||||||
|  |       def DeferUserFunc2() | ||||||
|  |         defer g:FuncOneArg(1, 2) | ||||||
|  |       enddef | ||||||
|  |       defcompile | ||||||
|  |   END | ||||||
|  |   call v9.CheckScriptFailure(lines, 'E118:') | ||||||
|  |   let lines =<< trim END | ||||||
|  |       def DeferUserFunc1() | ||||||
|  |         defer g:FuncOneArg(1) | ||||||
|  |       enddef | ||||||
|  |       defcompile | ||||||
|  |   END | ||||||
|  |   call v9.CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch, expected string but got number') | ||||||
|  | endfunc | ||||||
|  |  | ||||||
|  |  | ||||||
| " vim: shiftwidth=2 sts=2 expandtab | " vim: shiftwidth=2 sts=2 expandtab | ||||||
|  | |||||||
| @ -5608,6 +5608,7 @@ ex_call_inner( | |||||||
| ex_defer_inner( | ex_defer_inner( | ||||||
| 	char_u	    *name, | 	char_u	    *name, | ||||||
| 	char_u	    **arg, | 	char_u	    **arg, | ||||||
|  | 	type_T	    *type, | ||||||
| 	partial_T   *partial, | 	partial_T   *partial, | ||||||
| 	evalarg_T   *evalarg) | 	evalarg_T   *evalarg) | ||||||
| { | { | ||||||
| @ -5640,6 +5641,44 @@ ex_defer_inner( | |||||||
|     r = get_func_arguments(arg, evalarg, FALSE, |     r = get_func_arguments(arg, evalarg, FALSE, | ||||||
| 					    argvars + partial_argc, &argcount); | 					    argvars + partial_argc, &argcount); | ||||||
|     argcount += partial_argc; |     argcount += partial_argc; | ||||||
|  |  | ||||||
|  |     if (r == OK) | ||||||
|  |     { | ||||||
|  | 	if (type != NULL) | ||||||
|  | 	{ | ||||||
|  | 	    // Check that the arguments are OK for the types of the funcref. | ||||||
|  | 	    r = check_argument_types(type, argvars, argcount, NULL, name); | ||||||
|  | 	} | ||||||
|  | 	else if (builtin_function(name, -1)) | ||||||
|  | 	{ | ||||||
|  | 	    int idx = find_internal_func(name); | ||||||
|  |  | ||||||
|  | 	    if (idx < 0) | ||||||
|  | 	    { | ||||||
|  | 		emsg_funcname(e_unknown_function_str, name); | ||||||
|  | 		r = FAIL; | ||||||
|  | 	    } | ||||||
|  | 	    else if (check_internal_func(idx, argcount) == -1) | ||||||
|  | 		r = FAIL; | ||||||
|  | 	} | ||||||
|  | 	else | ||||||
|  | 	{ | ||||||
|  | 	    ufunc_T *ufunc = find_func(name, FALSE); | ||||||
|  |  | ||||||
|  | 	    // we tolerate an unknown function here, it might be defined later | ||||||
|  | 	    if (ufunc != NULL) | ||||||
|  | 	    { | ||||||
|  | 		int error = check_user_func_argcount(ufunc, argcount); | ||||||
|  |  | ||||||
|  | 		if (error != FCERR_UNKNOWN) | ||||||
|  | 		{ | ||||||
|  | 		    user_func_error(error, name, NULL); | ||||||
|  | 		    r = FAIL; | ||||||
|  | 		} | ||||||
|  | 	    } | ||||||
|  | 	} | ||||||
|  |     } | ||||||
|  |  | ||||||
|     if (r == FAIL) |     if (r == FAIL) | ||||||
|     { |     { | ||||||
| 	while (--argcount >= 0) | 	while (--argcount >= 0) | ||||||
| @ -5839,7 +5878,7 @@ ex_call(exarg_T *eap) | |||||||
|     if (eap->cmdidx == CMD_defer) |     if (eap->cmdidx == CMD_defer) | ||||||
|     { |     { | ||||||
| 	arg = startarg; | 	arg = startarg; | ||||||
| 	failed = ex_defer_inner(name, &arg, partial, &evalarg) == FAIL; | 	failed = ex_defer_inner(name, &arg, type, partial, &evalarg) == FAIL; | ||||||
|     } |     } | ||||||
|     else |     else | ||||||
|     { |     { | ||||||
|  | |||||||
| @ -703,6 +703,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 */ | ||||||
|  | /**/ | ||||||
|  |     419, | ||||||
| /**/ | /**/ | ||||||
|     418, |     418, | ||||||
| /**/ | /**/ | ||||||
|  | |||||||
| @ -1684,34 +1684,13 @@ compile_eval(char_u *arg, cctx_T *cctx) | |||||||
|     return skipwhite(p); |     return skipwhite(p); | ||||||
| } | } | ||||||
|  |  | ||||||
| /* |  | ||||||
|  * Get the local variable index for deferred function calls. |  | ||||||
|  * Reserve it when not done already. |  | ||||||
|  * Returns zero for failure. |  | ||||||
|  */ |  | ||||||
|     int |  | ||||||
| get_defer_var_idx(cctx_T *cctx) |  | ||||||
| { |  | ||||||
|     dfunc_T	*dfunc = ((dfunc_T *)def_functions.ga_data) |  | ||||||
| 					       + cctx->ctx_ufunc->uf_dfunc_idx; |  | ||||||
|     if (dfunc->df_defer_var_idx == 0) |  | ||||||
|     { |  | ||||||
| 	lvar_T *lvar = reserve_local(cctx, (char_u *)"@defer@", 7, |  | ||||||
| 							    TRUE, &t_list_any); |  | ||||||
| 	if (lvar == NULL) |  | ||||||
| 	    return 0; |  | ||||||
| 	dfunc->df_defer_var_idx = lvar->lv_idx + 1; |  | ||||||
|     } |  | ||||||
|     return dfunc->df_defer_var_idx; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| /* | /* | ||||||
|  * Compile "defer func(arg)". |  * Compile "defer func(arg)". | ||||||
|  */ |  */ | ||||||
|     char_u * |     char_u * | ||||||
| compile_defer(char_u *arg_start, cctx_T *cctx) | compile_defer(char_u *arg_start, cctx_T *cctx) | ||||||
| { | { | ||||||
|     char_u	*p; |     char_u	*paren; | ||||||
|     char_u	*arg = arg_start; |     char_u	*arg = arg_start; | ||||||
|     int		argcount = 0; |     int		argcount = 0; | ||||||
|     int		defer_var_idx; |     int		defer_var_idx; | ||||||
| @ -1720,13 +1699,13 @@ compile_defer(char_u *arg_start, cctx_T *cctx) | |||||||
|  |  | ||||||
|     // Get a funcref for the function name. |     // Get a funcref for the function name. | ||||||
|     // TODO: better way to find the "(". |     // TODO: better way to find the "(". | ||||||
|     p = vim_strchr(arg, '('); |     paren = vim_strchr(arg, '('); | ||||||
|     if (p == NULL) |     if (paren == NULL) | ||||||
|     { |     { | ||||||
| 	semsg(_(e_missing_parenthesis_str), arg); | 	semsg(_(e_missing_parenthesis_str), arg); | ||||||
| 	return NULL; | 	return NULL; | ||||||
|     } |     } | ||||||
|     *p = NUL; |     *paren = NUL; | ||||||
|     func_idx = find_internal_func(arg); |     func_idx = find_internal_func(arg); | ||||||
|     if (func_idx >= 0) |     if (func_idx >= 0) | ||||||
| 	// TODO: better type | 	// TODO: better type | ||||||
| @ -1734,7 +1713,7 @@ compile_defer(char_u *arg_start, cctx_T *cctx) | |||||||
| 							   &t_func_any, FALSE); | 							   &t_func_any, FALSE); | ||||||
|     else if (compile_expr0(&arg, cctx) == FAIL) |     else if (compile_expr0(&arg, cctx) == FAIL) | ||||||
| 	return NULL; | 	return NULL; | ||||||
|     *p = '('; |     *paren = '('; | ||||||
|  |  | ||||||
|     // check for function type |     // check for function type | ||||||
|     type = get_type_on_stack(cctx, 0); |     type = get_type_on_stack(cctx, 0); | ||||||
| @ -1745,11 +1724,22 @@ compile_defer(char_u *arg_start, cctx_T *cctx) | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     // compile the arguments |     // compile the arguments | ||||||
|     arg = skipwhite(p + 1); |     arg = skipwhite(paren + 1); | ||||||
|     if (compile_arguments(&arg, cctx, &argcount, CA_NOT_SPECIAL) == FAIL) |     if (compile_arguments(&arg, cctx, &argcount, CA_NOT_SPECIAL) == FAIL) | ||||||
| 	return NULL; | 	return NULL; | ||||||
|  |  | ||||||
|     // TODO: check argument count with "type" |     if (func_idx >= 0) | ||||||
|  |     { | ||||||
|  | 	type2_T	*argtypes = NULL; | ||||||
|  | 	type2_T	shuffled_argtypes[MAX_FUNC_ARGS]; | ||||||
|  |  | ||||||
|  | 	if (check_internal_func_args(cctx, func_idx, argcount, FALSE, | ||||||
|  | 					 &argtypes, shuffled_argtypes) == FAIL) | ||||||
|  | 	    return NULL; | ||||||
|  |     } | ||||||
|  |     else if (check_func_args_from_type(cctx, type, argcount, TRUE, | ||||||
|  | 							    arg_start) == FAIL) | ||||||
|  | 	return NULL; | ||||||
|  |  | ||||||
|     defer_var_idx = get_defer_var_idx(cctx); |     defer_var_idx = get_defer_var_idx(cctx); | ||||||
|     if (defer_var_idx == 0) |     if (defer_var_idx == 0) | ||||||
|  | |||||||
							
								
								
									
										201
									
								
								src/vim9instr.c
									
									
									
									
									
								
							
							
						
						
									
										201
									
								
								src/vim9instr.c
									
									
									
									
									
								
							| @ -1329,6 +1329,65 @@ generate_TRYCONT(cctx_T *cctx, int levels, int where) | |||||||
|     return OK; |     return OK; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | /* | ||||||
|  |  * Check "argount" arguments and their types on the type stack. | ||||||
|  |  * Give an error and return FAIL if something is wrong. | ||||||
|  |  * When "method_call" is NULL no code is generated. | ||||||
|  |  */ | ||||||
|  |     int | ||||||
|  | check_internal_func_args( | ||||||
|  | 	cctx_T	*cctx, | ||||||
|  | 	int	func_idx, | ||||||
|  | 	int	argcount, | ||||||
|  | 	int	method_call, | ||||||
|  | 	type2_T **argtypes, | ||||||
|  | 	type2_T *shuffled_argtypes) | ||||||
|  | { | ||||||
|  |     garray_T	*stack = &cctx->ctx_type_stack; | ||||||
|  |     int		argoff = check_internal_func(func_idx, argcount); | ||||||
|  |  | ||||||
|  |     if (argoff < 0) | ||||||
|  | 	return FAIL; | ||||||
|  |  | ||||||
|  |     if (method_call && argoff > 1) | ||||||
|  |     { | ||||||
|  | 	isn_T	*isn = generate_instr(cctx, ISN_SHUFFLE); | ||||||
|  |  | ||||||
|  | 	if (isn  == NULL) | ||||||
|  | 	    return FAIL; | ||||||
|  | 	isn->isn_arg.shuffle.shfl_item = argcount; | ||||||
|  | 	isn->isn_arg.shuffle.shfl_up = argoff - 1; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     if (argcount > 0) | ||||||
|  |     { | ||||||
|  | 	type2_T	*typep = ((type2_T *)stack->ga_data) + stack->ga_len - argcount; | ||||||
|  |  | ||||||
|  | 	// Check the types of the arguments. | ||||||
|  | 	if (method_call && argoff > 1) | ||||||
|  | 	{ | ||||||
|  | 	    int i; | ||||||
|  |  | ||||||
|  | 	    for (i = 0; i < argcount; ++i) | ||||||
|  | 		shuffled_argtypes[i] = (i < argoff - 1) | ||||||
|  | 				    ? typep[i + 1] | ||||||
|  | 				    : (i == argoff - 1) ? typep[0] : typep[i]; | ||||||
|  | 	    *argtypes = shuffled_argtypes; | ||||||
|  | 	} | ||||||
|  | 	else | ||||||
|  | 	{ | ||||||
|  | 	    int i; | ||||||
|  |  | ||||||
|  | 	    for (i = 0; i < argcount; ++i) | ||||||
|  | 		shuffled_argtypes[i] = typep[i]; | ||||||
|  | 	    *argtypes = shuffled_argtypes; | ||||||
|  | 	} | ||||||
|  | 	if (internal_func_check_arg_types(*argtypes, func_idx, argcount, | ||||||
|  | 								 cctx) == FAIL) | ||||||
|  | 	    return FAIL; | ||||||
|  |     } | ||||||
|  |     return OK; | ||||||
|  | } | ||||||
|  |  | ||||||
| /* | /* | ||||||
|  * Generate an ISN_BCALL instruction. |  * Generate an ISN_BCALL instruction. | ||||||
| @ -1340,8 +1399,6 @@ generate_BCALL(cctx_T *cctx, int func_idx, int argcount, int method_call) | |||||||
| { | { | ||||||
|     isn_T	*isn; |     isn_T	*isn; | ||||||
|     garray_T	*stack = &cctx->ctx_type_stack; |     garray_T	*stack = &cctx->ctx_type_stack; | ||||||
|     int		argoff; |  | ||||||
|     type2_T	*typep; |  | ||||||
|     type2_T	*argtypes = NULL; |     type2_T	*argtypes = NULL; | ||||||
|     type2_T	shuffled_argtypes[MAX_FUNC_ARGS]; |     type2_T	shuffled_argtypes[MAX_FUNC_ARGS]; | ||||||
|     type2_T	*maptype = NULL; |     type2_T	*maptype = NULL; | ||||||
| @ -1349,46 +1406,13 @@ generate_BCALL(cctx_T *cctx, int func_idx, int argcount, int method_call) | |||||||
|     type_T	*decl_type; |     type_T	*decl_type; | ||||||
|  |  | ||||||
|     RETURN_OK_IF_SKIP(cctx); |     RETURN_OK_IF_SKIP(cctx); | ||||||
|     argoff = check_internal_func(func_idx, argcount); |  | ||||||
|     if (argoff < 0) |     if (check_internal_func_args(cctx, func_idx, argcount, method_call, | ||||||
|  | 					 &argtypes, shuffled_argtypes) == FAIL) | ||||||
| 	return FAIL; | 	return FAIL; | ||||||
|  |  | ||||||
|     if (method_call && argoff > 1) |     if (internal_func_is_map(func_idx)) | ||||||
|     { | 	maptype = argtypes; | ||||||
| 	if ((isn = generate_instr(cctx, ISN_SHUFFLE)) == NULL) |  | ||||||
| 	    return FAIL; |  | ||||||
| 	isn->isn_arg.shuffle.shfl_item = argcount; |  | ||||||
| 	isn->isn_arg.shuffle.shfl_up = argoff - 1; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     if (argcount > 0) |  | ||||||
|     { |  | ||||||
| 	// Check the types of the arguments. |  | ||||||
| 	typep = ((type2_T *)stack->ga_data) + stack->ga_len - argcount; |  | ||||||
| 	if (method_call && argoff > 1) |  | ||||||
| 	{ |  | ||||||
| 	    int i; |  | ||||||
|  |  | ||||||
| 	    for (i = 0; i < argcount; ++i) |  | ||||||
| 		shuffled_argtypes[i] = (i < argoff - 1) |  | ||||||
| 			    ? typep[i + 1] |  | ||||||
| 				  : (i == argoff - 1) ? typep[0] : typep[i]; |  | ||||||
| 	    argtypes = shuffled_argtypes; |  | ||||||
| 	} |  | ||||||
| 	else |  | ||||||
| 	{ |  | ||||||
| 	    int i; |  | ||||||
|  |  | ||||||
| 	    for (i = 0; i < argcount; ++i) |  | ||||||
| 		shuffled_argtypes[i] = typep[i]; |  | ||||||
| 	    argtypes = shuffled_argtypes; |  | ||||||
| 	} |  | ||||||
| 	if (internal_func_check_arg_types(argtypes, func_idx, argcount, |  | ||||||
| 								 cctx) == FAIL) |  | ||||||
| 	    return FAIL; |  | ||||||
| 	if (internal_func_is_map(func_idx)) |  | ||||||
| 	    maptype = argtypes; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     if ((isn = generate_instr(cctx, ISN_BCALL)) == NULL) |     if ((isn = generate_instr(cctx, ISN_BCALL)) == NULL) | ||||||
| 	return FAIL; | 	return FAIL; | ||||||
| @ -1577,6 +1601,61 @@ generate_UCALL(cctx_T *cctx, char_u *name, int argcount) | |||||||
|     return push_type_stack(cctx, &t_any); |     return push_type_stack(cctx, &t_any); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | /* | ||||||
|  |  * Check the arguments of function "type" against the types on the stack. | ||||||
|  |  * Returns OK or FAIL; | ||||||
|  |  */ | ||||||
|  |     int | ||||||
|  | check_func_args_from_type( | ||||||
|  | 	cctx_T	*cctx, | ||||||
|  | 	type_T	*type, | ||||||
|  | 	int	argcount, | ||||||
|  | 	int	at_top, | ||||||
|  | 	char_u	*name) | ||||||
|  | { | ||||||
|  |     if (type->tt_argcount != -1) | ||||||
|  |     { | ||||||
|  | 	int	    varargs = (type->tt_flags & TTFLAG_VARARGS) ? 1 : 0; | ||||||
|  |  | ||||||
|  | 	if (argcount < type->tt_min_argcount - varargs) | ||||||
|  | 	{ | ||||||
|  | 	    emsg_funcname(e_not_enough_arguments_for_function_str, name); | ||||||
|  | 	    return FAIL; | ||||||
|  | 	} | ||||||
|  | 	if (!varargs && argcount > type->tt_argcount) | ||||||
|  | 	{ | ||||||
|  | 	    emsg_funcname(e_too_many_arguments_for_function_str, name); | ||||||
|  | 	    return FAIL; | ||||||
|  | 	} | ||||||
|  | 	if (type->tt_args != NULL) | ||||||
|  | 	{ | ||||||
|  | 	    int i; | ||||||
|  |  | ||||||
|  | 	    for (i = 0; i < argcount; ++i) | ||||||
|  | 	    { | ||||||
|  | 		int	offset = -argcount + i - (at_top ? 0 : 1); | ||||||
|  | 		type_T	*actual = get_type_on_stack(cctx, -1 - offset); | ||||||
|  | 		type_T	*expected; | ||||||
|  |  | ||||||
|  | 		if (varargs && i >= type->tt_argcount - 1) | ||||||
|  | 		    expected = type->tt_args[type->tt_argcount - 1]->tt_member; | ||||||
|  | 		else if (i >= type->tt_min_argcount | ||||||
|  | 					     && actual->tt_type == VAR_SPECIAL) | ||||||
|  | 		    expected = &t_any; | ||||||
|  | 		else | ||||||
|  | 		    expected = type->tt_args[i]; | ||||||
|  | 		if (need_type(actual, expected, offset, i + 1, | ||||||
|  | 						    cctx, TRUE, FALSE) == FAIL) | ||||||
|  | 		{ | ||||||
|  | 		    arg_type_mismatch(expected, actual, i + 1); | ||||||
|  | 		    return FAIL; | ||||||
|  | 		} | ||||||
|  | 	    } | ||||||
|  | 	} | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return OK; | ||||||
|  | } | ||||||
| /* | /* | ||||||
|  * Generate an ISN_PCALL instruction. |  * Generate an ISN_PCALL instruction. | ||||||
|  * "type" is the type of the FuncRef. |  * "type" is the type of the FuncRef. | ||||||
| @ -1598,47 +1677,9 @@ generate_PCALL( | |||||||
| 	ret_type = &t_any; | 	ret_type = &t_any; | ||||||
|     else if (type->tt_type == VAR_FUNC || type->tt_type == VAR_PARTIAL) |     else if (type->tt_type == VAR_FUNC || type->tt_type == VAR_PARTIAL) | ||||||
|     { |     { | ||||||
| 	if (type->tt_argcount != -1) | 	if (check_func_args_from_type(cctx, type, argcount, at_top, name) == FAIL) | ||||||
| 	{ | 	    return FAIL; | ||||||
| 	    int	    varargs = (type->tt_flags & TTFLAG_VARARGS) ? 1 : 0; |  | ||||||
|  |  | ||||||
| 	    if (argcount < type->tt_min_argcount - varargs) |  | ||||||
| 	    { |  | ||||||
| 		emsg_funcname(e_not_enough_arguments_for_function_str, name); |  | ||||||
| 		return FAIL; |  | ||||||
| 	    } |  | ||||||
| 	    if (!varargs && argcount > type->tt_argcount) |  | ||||||
| 	    { |  | ||||||
| 		emsg_funcname(e_too_many_arguments_for_function_str, name); |  | ||||||
| 		return FAIL; |  | ||||||
| 	    } |  | ||||||
| 	    if (type->tt_args != NULL) |  | ||||||
| 	    { |  | ||||||
| 		int i; |  | ||||||
|  |  | ||||||
| 		for (i = 0; i < argcount; ++i) |  | ||||||
| 		{ |  | ||||||
| 		    int	    offset = -argcount + i - (at_top ? 0 : 1); |  | ||||||
| 		    type_T *actual = get_type_on_stack(cctx, -1 - offset); |  | ||||||
| 		    type_T *expected; |  | ||||||
|  |  | ||||||
| 		    if (varargs && i >= type->tt_argcount - 1) |  | ||||||
| 			expected = type->tt_args[ |  | ||||||
| 					     type->tt_argcount - 1]->tt_member; |  | ||||||
| 		    else if (i >= type->tt_min_argcount |  | ||||||
| 					     && actual->tt_type == VAR_SPECIAL) |  | ||||||
| 			expected = &t_any; |  | ||||||
| 		    else |  | ||||||
| 			expected = type->tt_args[i]; |  | ||||||
| 		    if (need_type(actual, expected, offset, i + 1, |  | ||||||
| 						    cctx, TRUE, FALSE) == FAIL) |  | ||||||
| 		    { |  | ||||||
| 			arg_type_mismatch(expected, actual, i + 1); |  | ||||||
| 			return FAIL; |  | ||||||
| 		    } |  | ||||||
| 		} |  | ||||||
| 	    } |  | ||||||
| 	} |  | ||||||
| 	ret_type = type->tt_member; | 	ret_type = type->tt_member; | ||||||
| 	if (ret_type == &t_unknown) | 	if (ret_type == &t_unknown) | ||||||
| 	    // return type not known yet, use a runtime check | 	    // return type not known yet, use a runtime check | ||||||
|  | |||||||
		Reference in New Issue
	
	Block a user