patch 9.0.1723: Fix regression in {func} argument of reduce()
Problem: Fix regression in {func} argument of reduce()
Solution: pass function name as string again
Before patch 9.0.0548, passing a string as {func} argument of reduce()
is treated as a function name, but after patch 9.0.0548 it is treated as
an expression instead, which is useless as reduce() doesn't set any v:
variables. This PR restores the behavior of {func} before that patch.
Also correct an emsg() call, as e_string_list_or_blob_required doesn't
contain format specifiers.
closes: #12824
Signed-off-by: Christian Brabandt <cb@256bit.org>
Co-authored-by: zeertzjq <zeertzjq@outlook.com>
			
			
This commit is contained in:
		
				
					committed by
					
						 Christian Brabandt
						Christian Brabandt
					
				
			
			
				
	
			
			
			
						parent
						
							4f389e7c0f
						
					
				
				
					commit
					ad0c442f1f
				
			| @ -769,7 +769,7 @@ blob_reduce( | |||||||
| 	argv[1].v_type = VAR_NUMBER; | 	argv[1].v_type = VAR_NUMBER; | ||||||
| 	argv[1].vval.v_number = blob_get(b, i); | 	argv[1].vval.v_number = blob_get(b, i); | ||||||
|  |  | ||||||
| 	r = eval_expr_typval(expr, argv, 2, NULL, rettv); | 	r = eval_expr_typval(expr, TRUE, argv, 2, NULL, rettv); | ||||||
|  |  | ||||||
| 	clear_tv(&argv[0]); | 	clear_tv(&argv[0]); | ||||||
| 	if (r == FAIL || called_emsg != called_emsg_start) | 	if (r == FAIL || called_emsg != called_emsg_start) | ||||||
|  | |||||||
							
								
								
									
										28
									
								
								src/eval.c
									
									
									
									
									
								
							
							
						
						
									
										28
									
								
								src/eval.c
									
									
									
									
									
								
							| @ -252,12 +252,14 @@ eval_expr_get_funccal(typval_T *expr, typval_T *rettv) | |||||||
| /* | /* | ||||||
|  * Evaluate an expression, which can be a function, partial or string. |  * Evaluate an expression, which can be a function, partial or string. | ||||||
|  * Pass arguments "argv[argc]". |  * Pass arguments "argv[argc]". | ||||||
|  |  * If "want_func" is TRUE treat a string as a function name, not an expression. | ||||||
|  * "fc_arg" is from eval_expr_get_funccal() or NULL; |  * "fc_arg" is from eval_expr_get_funccal() or NULL; | ||||||
|  * Return the result in "rettv" and OK or FAIL. |  * Return the result in "rettv" and OK or FAIL. | ||||||
|  */ |  */ | ||||||
|     int |     int | ||||||
| eval_expr_typval( | eval_expr_typval( | ||||||
| 	typval_T    *expr, | 	typval_T    *expr, | ||||||
|  | 	int	    want_func, | ||||||
| 	typval_T    *argv, | 	typval_T    *argv, | ||||||
| 	int	    argc, | 	int	    argc, | ||||||
| 	funccall_T  *fc_arg, | 	funccall_T  *fc_arg, | ||||||
| @ -267,17 +269,7 @@ eval_expr_typval( | |||||||
|     char_u	buf[NUMBUFLEN]; |     char_u	buf[NUMBUFLEN]; | ||||||
|     funcexe_T	funcexe; |     funcexe_T	funcexe; | ||||||
|  |  | ||||||
|     if (expr->v_type == VAR_FUNC) |     if (expr->v_type == VAR_PARTIAL) | ||||||
|     { |  | ||||||
| 	s = expr->vval.v_string; |  | ||||||
| 	if (s == NULL || *s == NUL) |  | ||||||
| 	    return FAIL; |  | ||||||
| 	CLEAR_FIELD(funcexe); |  | ||||||
| 	funcexe.fe_evaluate = TRUE; |  | ||||||
| 	if (call_func(s, -1, rettv, argc, argv, &funcexe) == FAIL) |  | ||||||
| 	    return FAIL; |  | ||||||
|     } |  | ||||||
|     else if (expr->v_type == VAR_PARTIAL) |  | ||||||
|     { |     { | ||||||
| 	partial_T   *partial = expr->vval.v_partial; | 	partial_T   *partial = expr->vval.v_partial; | ||||||
|  |  | ||||||
| @ -318,6 +310,18 @@ eval_expr_typval( | |||||||
|     { |     { | ||||||
| 	return exe_typval_instr(expr, rettv); | 	return exe_typval_instr(expr, rettv); | ||||||
|     } |     } | ||||||
|  |     else if (expr->v_type == VAR_FUNC || want_func) | ||||||
|  |     { | ||||||
|  | 	s = expr->v_type == VAR_FUNC | ||||||
|  | 		? expr->vval.v_string | ||||||
|  | 		: tv_get_string_buf_chk_strict(expr, buf, in_vim9script()); | ||||||
|  | 	if (s == NULL || *s == NUL) | ||||||
|  | 	    return FAIL; | ||||||
|  | 	CLEAR_FIELD(funcexe); | ||||||
|  | 	funcexe.fe_evaluate = TRUE; | ||||||
|  | 	if (call_func(s, -1, rettv, argc, argv, &funcexe) == FAIL) | ||||||
|  | 	    return FAIL; | ||||||
|  |     } | ||||||
|     else |     else | ||||||
|     { |     { | ||||||
| 	s = tv_get_string_buf_chk_strict(expr, buf, in_vim9script()); | 	s = tv_get_string_buf_chk_strict(expr, buf, in_vim9script()); | ||||||
| @ -346,7 +350,7 @@ eval_expr_to_bool(typval_T *expr, int *error) | |||||||
|     typval_T	rettv; |     typval_T	rettv; | ||||||
|     int		res; |     int		res; | ||||||
|  |  | ||||||
|     if (eval_expr_typval(expr, NULL, 0, NULL, &rettv) == FAIL) |     if (eval_expr_typval(expr, FALSE, NULL, 0, NULL, &rettv) == FAIL) | ||||||
|     { |     { | ||||||
| 	*error = TRUE; | 	*error = TRUE; | ||||||
| 	return FALSE; | 	return FALSE; | ||||||
|  | |||||||
| @ -6938,7 +6938,7 @@ indexof_eval_expr(typval_T *expr) | |||||||
|     argv[1] = *get_vim_var_tv(VV_VAL); |     argv[1] = *get_vim_var_tv(VV_VAL); | ||||||
|     newtv.v_type = VAR_UNKNOWN; |     newtv.v_type = VAR_UNKNOWN; | ||||||
|  |  | ||||||
|     if (eval_expr_typval(expr, argv, 2, NULL, &newtv) == FAIL) |     if (eval_expr_typval(expr, FALSE, argv, 2, NULL, &newtv) == FAIL) | ||||||
| 	return FALSE; | 	return FALSE; | ||||||
|  |  | ||||||
|     found = tv_get_bool_chk(&newtv, &error); |     found = tv_get_bool_chk(&newtv, &error); | ||||||
|  | |||||||
| @ -1616,7 +1616,7 @@ checkitem_common(void *context, char_u *name, dict_T *dict) | |||||||
| 	argv[0].vval.v_dict = dict; | 	argv[0].vval.v_dict = dict; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     if (eval_expr_typval(expr, argv, 1, NULL, &rettv) == FAIL) |     if (eval_expr_typval(expr, FALSE, argv, 1, NULL, &rettv) == FAIL) | ||||||
| 	goto theend; | 	goto theend; | ||||||
|  |  | ||||||
|     // We want to use -1, but also true/false should be allowed. |     // We want to use -1, but also true/false should be allowed. | ||||||
|  | |||||||
| @ -2333,7 +2333,7 @@ filter_map_one( | |||||||
|     copy_tv(tv, get_vim_var_tv(VV_VAL)); |     copy_tv(tv, get_vim_var_tv(VV_VAL)); | ||||||
|     argv[0] = *get_vim_var_tv(VV_KEY); |     argv[0] = *get_vim_var_tv(VV_KEY); | ||||||
|     argv[1] = *get_vim_var_tv(VV_VAL); |     argv[1] = *get_vim_var_tv(VV_VAL); | ||||||
|     if (eval_expr_typval(expr, argv, 2, fc, newtv) == FAIL) |     if (eval_expr_typval(expr, FALSE, argv, 2, fc, newtv) == FAIL) | ||||||
| 	goto theend; | 	goto theend; | ||||||
|     if (filtermap == FILTERMAP_FILTER) |     if (filtermap == FILTERMAP_FILTER) | ||||||
|     { |     { | ||||||
| @ -3084,7 +3084,7 @@ list_reduce( | |||||||
| 	else | 	else | ||||||
| 	    argv[1] = li->li_tv; | 	    argv[1] = li->li_tv; | ||||||
|  |  | ||||||
| 	r = eval_expr_typval(expr, argv, 2, fc, rettv); | 	r = eval_expr_typval(expr, TRUE, argv, 2, fc, rettv); | ||||||
|  |  | ||||||
| 	if (argv[0].v_type != VAR_NUMBER && argv[0].v_type != VAR_UNKNOWN) | 	if (argv[0].v_type != VAR_NUMBER && argv[0].v_type != VAR_UNKNOWN) | ||||||
| 	    clear_tv(&argv[0]); | 	    clear_tv(&argv[0]); | ||||||
| @ -3125,7 +3125,7 @@ f_reduce(typval_T *argvars, typval_T *rettv) | |||||||
| 	    && argvars[0].v_type != VAR_LIST | 	    && argvars[0].v_type != VAR_LIST | ||||||
| 	    && argvars[0].v_type != VAR_BLOB) | 	    && argvars[0].v_type != VAR_BLOB) | ||||||
|     { |     { | ||||||
| 	semsg(_(e_string_list_or_blob_required), "reduce()"); | 	emsg(_(e_string_list_or_blob_required)); | ||||||
| 	return; | 	return; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  | |||||||
| @ -7,7 +7,7 @@ void fill_evalarg_from_eap(evalarg_T *evalarg, exarg_T *eap, int skip); | |||||||
| int eval_to_bool(char_u *arg, int *error, exarg_T *eap, int skip, int use_simple_function); | int eval_to_bool(char_u *arg, int *error, exarg_T *eap, int skip, int use_simple_function); | ||||||
| int eval_expr_valid_arg(typval_T *tv); | int eval_expr_valid_arg(typval_T *tv); | ||||||
| funccall_T *eval_expr_get_funccal(typval_T *expr, typval_T *rettv); | funccall_T *eval_expr_get_funccal(typval_T *expr, typval_T *rettv); | ||||||
| int eval_expr_typval(typval_T *expr, typval_T *argv, int argc, funccall_T *fc_arg, typval_T *rettv); | int eval_expr_typval(typval_T *expr, int prefer_func, typval_T *argv, int argc, funccall_T *fc_arg, typval_T *rettv); | ||||||
| int eval_expr_to_bool(typval_T *expr, int *error); | int eval_expr_to_bool(typval_T *expr, int *error); | ||||||
| char_u *eval_to_string_skip(char_u *arg, exarg_T *eap, int skip); | char_u *eval_to_string_skip(char_u *arg, exarg_T *eap, int skip); | ||||||
| void init_evalarg(evalarg_T *evalarg); | void init_evalarg(evalarg_T *evalarg); | ||||||
|  | |||||||
| @ -1032,7 +1032,7 @@ string_reduce( | |||||||
| 	    break; | 	    break; | ||||||
| 	len = (int)STRLEN(argv[1].vval.v_string); | 	len = (int)STRLEN(argv[1].vval.v_string); | ||||||
|  |  | ||||||
| 	r = eval_expr_typval(expr, argv, 2, fc, rettv); | 	r = eval_expr_typval(expr, TRUE, argv, 2, fc, rettv); | ||||||
|  |  | ||||||
| 	clear_tv(&argv[0]); | 	clear_tv(&argv[0]); | ||||||
| 	clear_tv(&argv[1]); | 	clear_tv(&argv[1]); | ||||||
|  | |||||||
| @ -1037,6 +1037,10 @@ func Test_reduce() | |||||||
|       call assert_equal('Å,s,t,r,ö,m', reduce('Åström', LSTART acc, val LMIDDLE acc .. ',' .. val LEND)) |       call assert_equal('Å,s,t,r,ö,m', reduce('Åström', LSTART acc, val LMIDDLE acc .. ',' .. val LEND)) | ||||||
|       call assert_equal('Å,s,t,r,ö,m', reduce('Åström', LSTART acc, val LMIDDLE acc .. ',' .. val LEND)) |       call assert_equal('Å,s,t,r,ö,m', reduce('Åström', LSTART acc, val LMIDDLE acc .. ',' .. val LEND)) | ||||||
|       call assert_equal(',a,b,c', reduce('abc', LSTART acc, val LMIDDLE acc .. ',' .. val LEND, test_null_string())) |       call assert_equal(',a,b,c', reduce('abc', LSTART acc, val LMIDDLE acc .. ',' .. val LEND, test_null_string())) | ||||||
|  |  | ||||||
|  |       call assert_equal(0x7d, reduce([0x30, 0x25, 0x08, 0x61], 'or')) | ||||||
|  |       call assert_equal(0x7d, reduce(0z30250861, 'or')) | ||||||
|  |       call assert_equal('β', reduce('ββββ', 'matchstr')) | ||||||
|   END |   END | ||||||
|   call v9.CheckLegacyAndVim9Success(lines) |   call v9.CheckLegacyAndVim9Success(lines) | ||||||
|  |  | ||||||
| @ -1052,7 +1056,7 @@ func Test_reduce() | |||||||
|  |  | ||||||
|   call assert_fails("call reduce({}, { acc, val -> acc + val }, 1)", 'E1098:') |   call assert_fails("call reduce({}, { acc, val -> acc + val }, 1)", 'E1098:') | ||||||
|   call assert_fails("call reduce(0, { acc, val -> acc + val }, 1)", 'E1098:') |   call assert_fails("call reduce(0, { acc, val -> acc + val }, 1)", 'E1098:') | ||||||
|   call assert_fails("call reduce([1, 2], 'Xdoes_not_exist')", 'E121:') |   call assert_fails("call reduce([1, 2], 'Xdoes_not_exist')", 'E117:') | ||||||
|   call assert_fails("echo reduce(0z01, { acc, val -> 2 * acc + val }, '')", 'E1210:') |   call assert_fails("echo reduce(0z01, { acc, val -> 2 * acc + val }, '')", 'E1210:') | ||||||
|  |  | ||||||
|   call assert_fails("vim9 reduce(0, (acc, val) => (acc .. val), '')", 'E1252:') |   call assert_fails("vim9 reduce(0, (acc, val) => (acc .. val), '')", 'E1252:') | ||||||
|  | |||||||
| @ -695,6 +695,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 */ | ||||||
|  | /**/ | ||||||
|  |     1723, | ||||||
| /**/ | /**/ | ||||||
|     1722, |     1722, | ||||||
| /**/ | /**/ | ||||||
|  | |||||||
		Reference in New Issue
	
	Block a user