# HG changeset patch # User Bram Moolenaar # Date 1591534804 -7200 # Node ID ab27db64f1fb01dc88ba74f9cde92384dc6bd7fb # Parent 6f44f49988ee58800ab9884c356432eba29db392 patch 8.2.0918: duplicate code for evaluating expression argument Commit: https://github.com/vim/vim/commit/a9c010494767e43a51c443cac35ebc80d0831d0b Author: Bram Moolenaar Date: Sun Jun 7 14:50:50 2020 +0200 patch 8.2.0918: duplicate code for evaluating expression argument Problem: Duplicate code for evaluating expression argument. Solution: Merge the code and make the use more flexible. diff --git a/src/eval.c b/src/eval.c --- a/src/eval.c +++ b/src/eval.c @@ -212,6 +212,18 @@ eval1_emsg(char_u **arg, typval_T *rettv } /* + * Return whether a typval is a valid expression to pass to eval_expr_typval() + * or eval_expr_to_bool(). An empty string returns FALSE; + */ + int +eval_expr_valid_arg(typval_T *tv) +{ + return tv->v_type != VAR_UNKNOWN + && (tv->v_type != VAR_STRING + || (tv->vval.v_string != NULL && *tv->vval.v_string != NUL)); +} + +/* * Evaluate an expression, which can be a function, partial or string. * Pass arguments "argv[argc]". * Return the result in "rettv" and OK or FAIL. diff --git a/src/evalfunc.c b/src/evalfunc.c --- a/src/evalfunc.c +++ b/src/evalfunc.c @@ -6399,11 +6399,9 @@ search_cmn(typval_T *argvars, pos_T *mat int options = SEARCH_KEEP; int subpatnum; searchit_arg_T sia; - evalarg_T skip; + int use_skip = FALSE; pos_T firstpos; - CLEAR_FIELD(skip); - pat = tv_get_string(&argvars[0]); dir = get_search_arg(&argvars[1], flagsp); // may set p_ws if (dir == 0) @@ -6429,9 +6427,7 @@ search_cmn(typval_T *argvars, pos_T *mat if (time_limit < 0) goto theend; #endif - if (argvars[4].v_type != VAR_UNKNOWN - && evalarg_get(&argvars[4], &skip) == FAIL) - goto theend; + use_skip = eval_expr_valid_arg(&argvars[4]); } } @@ -6471,19 +6467,20 @@ search_cmn(typval_T *argvars, pos_T *mat if (firstpos.lnum != 0 && EQUAL_POS(pos, firstpos)) subpatnum = FAIL; - if (subpatnum == FAIL || !evalarg_valid(&skip)) + if (subpatnum == FAIL || !use_skip) // didn't find it or no skip argument break; firstpos = pos; - // If the skip pattern matches, ignore this match. + // If the skip expression matches, ignore this match. { int do_skip; int err; pos_T save_pos = curwin->w_cursor; curwin->w_cursor = pos; - do_skip = evalarg_call_bool(&skip, &err); + err = FALSE; + do_skip = eval_expr_to_bool(&argvars[4], &err); curwin->w_cursor = save_pos; if (err) { @@ -6523,7 +6520,6 @@ search_cmn(typval_T *argvars, pos_T *mat curwin->w_set_curswant = TRUE; theend: p_ws = save_p_ws; - evalarg_clean(&skip); return retval; } @@ -6791,14 +6787,9 @@ searchpair_cmn(typval_T *argvars, pos_T skip = NULL; else { + // Type is checked later. skip = &argvars[4]; - if (skip->v_type != VAR_FUNC && skip->v_type != VAR_PARTIAL - && skip->v_type != VAR_STRING) - { - // Type error - semsg(_(e_invarg2), tv_get_string(&argvars[4])); - goto theend; - } + if (argvars[5].v_type != VAR_UNKNOWN) { lnum_stop = (long)tv_get_number_chk(&argvars[5], NULL); @@ -6922,12 +6913,7 @@ do_searchpair( options |= SEARCH_START; if (skip != NULL) - { - // Empty string means to not use the skip expression. - if (skip->v_type == VAR_STRING || skip->v_type == VAR_FUNC) - use_skip = skip->vval.v_string != NULL - && *skip->vval.v_string != NUL; - } + use_skip = eval_expr_valid_arg(skip); save_cursor = curwin->w_cursor; pos = curwin->w_cursor; diff --git a/src/evalvars.c b/src/evalvars.c --- a/src/evalvars.c +++ b/src/evalvars.c @@ -3811,79 +3811,4 @@ free_callback(callback_T *callback) callback->cb_name = NULL; } -/* - * Process a function argument that can be a string expression or a function - * reference. - * "tv" must remain valid until calling evalarg_clean()! - * Returns FAIL when the argument is invalid. - */ - int -evalarg_get(typval_T *tv, evalarg_T *eva) -{ - if (tv->v_type == VAR_STRING - || tv->v_type == VAR_NUMBER - || tv->v_type == VAR_BOOL - || tv->v_type == VAR_SPECIAL) - { - eva->eva_string = tv_get_string_buf(tv, eva->eva_buf); - return OK; - } - - eva->eva_callback = get_callback(tv); - return eva->eva_callback.cb_name == NULL ? FAIL : OK; -} - -/* - * Return whether "eva" has a valid expression or callback. - */ - int -evalarg_valid(evalarg_T *eva) -{ - return eva->eva_string != NULL || eva->eva_callback.cb_name != NULL; -} - -/* - * Invoke the expression or callback "eva" and return the result in "tv". - * Returns FAIL if something failed - */ - int -evalarg_call(evalarg_T *eva, typval_T *tv) -{ - typval_T argv[1]; - - if (eva->eva_string != NULL) - return eval0(eva->eva_string, tv, NULL, EVAL_EVALUATE); - - argv[0].v_type = VAR_UNKNOWN; - return call_callback(&eva->eva_callback, -1, tv, 0, argv); -} - -/* - * Like evalarg_call(), but just return TRUE of FALSE. - * Sets "error" to TRUE if evaluation failed. - */ - int -evalarg_call_bool(evalarg_T *eva, int *error) -{ - typval_T tv; - int r; - - if (evalarg_call(eva, &tv) == FAIL) - { - *error = TRUE; - return FALSE; - } - r = tv_get_number(&tv); - clear_tv(&tv); - *error = FALSE; - return r; -} - - void -evalarg_clean(evalarg_T *eva) -{ - if (eva->eva_string == NULL) - free_callback(&eva->eva_callback); -} - #endif // FEAT_EVAL diff --git a/src/proto/eval.pro b/src/proto/eval.pro --- a/src/proto/eval.pro +++ b/src/proto/eval.pro @@ -4,6 +4,7 @@ varnumber_T num_modulus(varnumber_T n1, void eval_init(void); void eval_clear(void); int eval_to_bool(char_u *arg, int *error, char_u **nextcmd, int skip); +int eval_expr_valid_arg(typval_T *tv); int eval_expr_typval(typval_T *expr, typval_T *argv, int argc, typval_T *rettv); int eval_expr_to_bool(typval_T *expr, int *error); char_u *eval_to_string_skip(char_u *arg, char_u **nextcmd, int skip); @@ -19,14 +20,14 @@ void *call_func_retlist(char_u *func, in int eval_foldexpr(char_u *arg, int *cp); char_u *get_lval(char_u *name, typval_T *rettv, lval_T *lp, int unlet, int skip, int flags, int fne_flags); void clear_lval(lval_T *lp); -void set_var_lval(lval_T *lp, char_u *endp, typval_T *rettv, int copy, int is_const, char_u *op); +void set_var_lval(lval_T *lp, char_u *endp, typval_T *rettv, int copy, int flags, char_u *op); void *eval_for_line(char_u *arg, int *errp, char_u **nextcmdp, int skip); int next_for_item(void *fi_void, char_u *arg); void free_for_info(void *fi_void); void set_context_for_expression(expand_T *xp, char_u *arg, cmdidx_T cmdidx); int pattern_match(char_u *pat, char_u *text, int ic); -int eval0(char_u *arg, typval_T *rettv, char_u **nextcmd, int evaluate); -int eval1(char_u **arg, typval_T *rettv, int evaluate); +int eval0(char_u *arg, typval_T *rettv, char_u **nextcmd, int flags); +int eval1(char_u **arg, typval_T *rettv, int flags); void eval_addblob(typval_T *tv1, typval_T *tv2); int eval_addlist(typval_T *tv1, typval_T *tv2); char_u *partial_name(partial_T *pt); @@ -50,7 +51,7 @@ int get_name_len(char_u **arg, char_u ** char_u *find_name_end(char_u *arg, char_u **expr_start, char_u **expr_end, int flags); int eval_isnamec(int c); int eval_isnamec1(int c); -int handle_subscript(char_u **arg, typval_T *rettv, int evaluate, int verbose, char_u *start_leader, char_u **end_leaderp); +int handle_subscript(char_u **arg, typval_T *rettv, int flags, int verbose, char_u *start_leader, char_u **end_leaderp); int item_copy(typval_T *from, typval_T *to, int deep, int copyID); void echo_one(typval_T *rettv, int with_space, int *atstart, int *needclr); void ex_echo(exarg_T *eap); diff --git a/src/proto/evalvars.pro b/src/proto/evalvars.pro --- a/src/proto/evalvars.pro +++ b/src/proto/evalvars.pro @@ -88,9 +88,4 @@ callback_T get_callback(typval_T *arg); void put_callback(callback_T *cb, typval_T *tv); void set_callback(callback_T *dest, callback_T *src); void free_callback(callback_T *callback); -int evalarg_get(typval_T *tv, evalarg_T *eva); -int evalarg_valid(evalarg_T *eva); -int evalarg_call(evalarg_T *eva, typval_T *tv); -int evalarg_call_bool(evalarg_T *eva, int *error); -void evalarg_clean(evalarg_T *eva); /* vim: set ft=c : */ diff --git a/src/structs.h b/src/structs.h --- a/src/structs.h +++ b/src/structs.h @@ -4130,20 +4130,6 @@ typedef struct int sa_wrapped; // search wrapped around } searchit_arg_T; -/* - * Function argument that can be a string, funcref or partial. - * - declare: evalarg_T name; - * - init: CLEAR_FIELD(name); - * - set: evalarg_get(&argvars[3], &name); - * - use: if (evalarg_valid(&name)) res = evalarg_call(&name); - * - cleanup: evalarg_clean(&name); - */ -typedef struct -{ - char_u eva_buf[NUMBUFLEN]; // buffer for get_tv_string_buf() - char_u *eva_string; - callback_T eva_callback; -} evalarg_T; #define WRITEBUFSIZE 8192 // size of normal write buffer diff --git a/src/version.c b/src/version.c --- a/src/version.c +++ b/src/version.c @@ -755,6 +755,8 @@ static char *(features[]) = static int included_patches[] = { /* Add new patch number below this line */ /**/ + 918, +/**/ 917, /**/ 916,