1
0
forked from aniani/vim

patch 8.1.1820: using expr->FuncRef() does not work

Problem:    Using expr->FuncRef() does not work.
Solution:   Make FuncRef work as a method.
This commit is contained in:
Bram Moolenaar
2019-08-05 23:10:16 +02:00
parent 1b6d9c4215
commit 761fdf01c6
4 changed files with 113 additions and 88 deletions

View File

@@ -3654,6 +3654,74 @@ pattern_match(char_u *pat, char_u *text, int ic)
return matches; return matches;
} }
/*
* Handle a name followed by "(". Both for just "name(arg)" and for
* "expr->name(arg)".
* Returns OK or FAIL.
*/
static int
eval_func(
char_u **arg, // points to "(", will be advanced
char_u *name,
int name_len,
typval_T *rettv,
int evaluate,
typval_T *basetv) // "expr" for "expr->name(arg)"
{
char_u *s = name;
int len = name_len;
partial_T *partial;
int ret = OK;
if (!evaluate)
check_vars(s, len);
/* If "s" is the name of a variable of type VAR_FUNC
* use its contents. */
s = deref_func_name(s, &len, &partial, !evaluate);
/* Need to make a copy, in case evaluating the arguments makes
* the name invalid. */
s = vim_strsave(s);
if (s == NULL)
ret = FAIL;
else
{
funcexe_T funcexe;
// Invoke the function.
vim_memset(&funcexe, 0, sizeof(funcexe));
funcexe.firstline = curwin->w_cursor.lnum;
funcexe.lastline = curwin->w_cursor.lnum;
funcexe.doesrange = &len;
funcexe.evaluate = evaluate;
funcexe.partial = partial;
funcexe.basetv = basetv;
ret = get_func_tv(s, len, rettv, arg, &funcexe);
}
vim_free(s);
/* If evaluate is FALSE rettv->v_type was not set in
* get_func_tv, but it's needed in handle_subscript() to parse
* what follows. So set it here. */
if (rettv->v_type == VAR_UNKNOWN && !evaluate && **arg == '(')
{
rettv->vval.v_string = NULL;
rettv->v_type = VAR_FUNC;
}
/* Stop the expression evaluation when immediately
* aborting on error, or when an interrupt occurred or
* an exception was thrown but not caught. */
if (evaluate && aborting())
{
if (ret == OK)
clear_tv(rettv);
ret = FAIL;
}
return ret;
}
/* /*
* The "evaluate" argument: When FALSE, the argument is only parsed but not * The "evaluate" argument: When FALSE, the argument is only parsed but not
* executed. The function may return OK, but the rettv will be of type * executed. The function may return OK, but the rettv will be of type
@@ -4671,55 +4739,7 @@ eval7(
else else
{ {
if (**arg == '(') /* recursive! */ if (**arg == '(') /* recursive! */
{ ret = eval_func(arg, s, len, rettv, evaluate, NULL);
partial_T *partial;
if (!evaluate)
check_vars(s, len);
/* If "s" is the name of a variable of type VAR_FUNC
* use its contents. */
s = deref_func_name(s, &len, &partial, !evaluate);
/* Need to make a copy, in case evaluating the arguments makes
* the name invalid. */
s = vim_strsave(s);
if (s == NULL)
ret = FAIL;
else
{
funcexe_T funcexe;
// Invoke the function.
vim_memset(&funcexe, 0, sizeof(funcexe));
funcexe.firstline = curwin->w_cursor.lnum;
funcexe.lastline = curwin->w_cursor.lnum;
funcexe.doesrange = &len;
funcexe.evaluate = evaluate;
funcexe.partial = partial;
ret = get_func_tv(s, len, rettv, arg, &funcexe);
}
vim_free(s);
/* If evaluate is FALSE rettv->v_type was not set in
* get_func_tv, but it's needed in handle_subscript() to parse
* what follows. So set it here. */
if (rettv->v_type == VAR_UNKNOWN && !evaluate && **arg == '(')
{
rettv->vval.v_string = NULL;
rettv->v_type = VAR_FUNC;
}
/* Stop the expression evaluation when immediately
* aborting on error, or when an interrupt occurred or
* an exception was thrown but not caught. */
if (evaluate && aborting())
{
if (ret == OK)
clear_tv(rettv);
ret = FAIL;
}
}
else if (evaluate) else if (evaluate)
ret = get_var_tv(s, len, rettv, NULL, TRUE, FALSE); ret = get_var_tv(s, len, rettv, NULL, TRUE, FALSE);
else else
@@ -4815,55 +4835,42 @@ eval_method(
{ {
char_u *name; char_u *name;
long len; long len;
funcexe_T funcexe; char_u *alias;
int ret = OK;
typval_T base = *rettv; typval_T base = *rettv;
int ret;
// Skip over the ->. // Skip over the ->.
*arg += 2; *arg += 2;
rettv->v_type = VAR_UNKNOWN;
// Locate the method name.
name = *arg; name = *arg;
for (len = 0; eval_isnamec(name[len]); ++len) len = get_name_len(arg, &alias, evaluate, TRUE);
; if (alias != NULL)
if (len == 0) name = alias;
if (len <= 0)
{ {
if (verbose) if (verbose)
emsg(_("E260: Missing name after ->")); emsg(_("E260: Missing name after ->"));
return FAIL; ret = FAIL;
} }
else
// Check for the "(". Skip over white space after it.
if (name[len] != '(')
{ {
if (verbose) if (**arg != '(')
semsg(_(e_missingparen), name); {
return FAIL; if (verbose)
semsg(_(e_missingparen), name);
ret = FAIL;
}
else
ret = eval_func(arg, name, len, rettv, evaluate, &base);
} }
*arg += len;
// TODO: if "name" is a function reference, resolve it.
vim_memset(&funcexe, 0, sizeof(funcexe));
funcexe.evaluate = evaluate;
funcexe.basetv = &base;
rettv->v_type = VAR_UNKNOWN;
ret = get_func_tv(name, len, rettv, arg, &funcexe);
/* Clear the funcref afterwards, so that deleting it while /* Clear the funcref afterwards, so that deleting it while
* evaluating the arguments is possible (see test55). */ * evaluating the arguments is possible (see test55). */
if (evaluate) if (evaluate)
clear_tv(&base); clear_tv(&base);
/* Stop the expression evaluation when immediately aborting on
* error, or when an interrupt occurred or an exception was thrown
* but not caught. */
if (aborting())
{
if (ret == OK)
clear_tv(rettv);
ret = FAIL;
}
return ret; return ret;
} }

View File

@@ -1,6 +1,6 @@
" Tests for ->method() " Tests for ->method()
func Test_list() func Test_list_method()
let l = [1, 2, 3] let l = [1, 2, 3]
call assert_equal([1, 2, 3, 4], [1, 2, 3]->add(4)) call assert_equal([1, 2, 3, 4], [1, 2, 3]->add(4))
eval l->assert_equal(l) eval l->assert_equal(l)
@@ -34,7 +34,7 @@ func Test_list()
call assert_fails('eval l->values()', 'E715:') call assert_fails('eval l->values()', 'E715:')
endfunc endfunc
func Test_dict() func Test_dict_method()
let d = #{one: 1, two: 2, three: 3} let d = #{one: 1, two: 2, three: 3}
call assert_equal(d, d->copy()) call assert_equal(d, d->copy())
@@ -66,7 +66,7 @@ func Test_dict()
call assert_equal([1, 2, 3], d->values()) call assert_equal([1, 2, 3], d->values())
endfunc endfunc
func Test_string() func Test_string_method()
call assert_equal(['1', '2', '3'], '1 2 3'->split()) call assert_equal(['1', '2', '3'], '1 2 3'->split())
call assert_equal([1, 2, 3], '1 2 3'->split()->map({i, v -> str2nr(v)})) call assert_equal([1, 2, 3], '1 2 3'->split()->map({i, v -> str2nr(v)}))
call assert_equal([65, 66, 67], 'ABC'->str2list()) call assert_equal([65, 66, 67], 'ABC'->str2list())
@@ -76,7 +76,7 @@ func Test_string()
call assert_equal('axc', 'abc'->substitute('b', 'x', '')) call assert_equal('axc', 'abc'->substitute('b', 'x', ''))
endfunc endfunc
func Test_append() func Test_method_append()
new new
eval ['one', 'two', 'three']->append(1) eval ['one', 'two', 'three']->append(1)
call assert_equal(['', 'one', 'two', 'three'], getline(1, '$')) call assert_equal(['', 'one', 'two', 'three'], getline(1, '$'))
@@ -89,3 +89,16 @@ func Test_append()
exe 'bwipe! ' .. bnr exe 'bwipe! ' .. bnr
endfunc endfunc
func Test_method_funcref()
func Concat(one, two, three)
return a:one .. a:two .. a:three
endfunc
let FuncRef = function('Concat')
eval 'foo'->FuncRef('bar', 'tail')->assert_equal('foobartail')
let Partial = function('Concat', ['two'])
eval 'one'->Partial('three')->assert_equal('onetwothree')
delfunc Concat
endfunc

View File

@@ -1498,6 +1498,7 @@ call_func(
typval_T argv[MAX_FUNC_ARGS + 1]; // used when "partial" or typval_T argv[MAX_FUNC_ARGS + 1]; // used when "partial" or
// "funcexe->basetv" is not NULL // "funcexe->basetv" is not NULL
int argv_clear = 0; int argv_clear = 0;
int argv_base = 0;
partial_T *partial = funcexe->partial; partial_T *partial = funcexe->partial;
// Make a copy of the name, if it comes from a funcref variable it could // Make a copy of the name, if it comes from a funcref variable it could
@@ -1590,9 +1591,9 @@ call_func(
mch_memmove(&argv[1], argvars, sizeof(typval_T) * argcount); mch_memmove(&argv[1], argvars, sizeof(typval_T) * argcount);
argv[0] = *funcexe->basetv; argv[0] = *funcexe->basetv;
argcount++; argcount++;
argvars = argv;
argv_base = 1;
} }
else
memcpy(argv, argvars, sizeof(typval_T) * argcount);
if (fp->uf_flags & FC_RANGE && funcexe->doesrange != NULL) if (fp->uf_flags & FC_RANGE && funcexe->doesrange != NULL)
*funcexe->doesrange = TRUE; *funcexe->doesrange = TRUE;
@@ -1621,7 +1622,7 @@ call_func(
did_save_redo = TRUE; did_save_redo = TRUE;
} }
++fp->uf_calls; ++fp->uf_calls;
call_user_func(fp, argcount, argv, rettv, call_user_func(fp, argcount, argvars, rettv,
funcexe->firstline, funcexe->lastline, funcexe->firstline, funcexe->lastline,
(fp->uf_flags & FC_DICT) ? selfdict : NULL); (fp->uf_flags & FC_DICT) ? selfdict : NULL);
if (--fp->uf_calls <= 0 && fp->uf_refcount <= 0) if (--fp->uf_calls <= 0 && fp->uf_refcount <= 0)
@@ -1698,8 +1699,10 @@ call_func(
} }
} }
// clear the copies made from the partial
while (argv_clear > 0) while (argv_clear > 0)
clear_tv(&argv[--argv_clear]); clear_tv(&argv[--argv_clear + argv_base]);
vim_free(tofree); vim_free(tofree);
vim_free(name); vim_free(name);

View File

@@ -773,6 +773,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 */
/**/
1820,
/**/ /**/
1819, 1819,
/**/ /**/