]> granicus.if.org Git - vim/commitdiff
patch 8.2.0657: Vim9: no check if called variable is a FuncRef v8.2.0657
authorBram Moolenaar <Bram@vim.org>
Tue, 28 Apr 2020 19:29:34 +0000 (21:29 +0200)
committerBram Moolenaar <Bram@vim.org>
Tue, 28 Apr 2020 19:29:34 +0000 (21:29 +0200)
Problem:    Vim9: no check if called variable is a FuncRef.
Solution:   Add a type check.

src/testdir/test_vim9_expr.vim
src/testdir/test_vim9_script.vim
src/version.c
src/vim9compile.c

index 463227257a76d654d697c7036ba0079785024a69..ace3d2f2560a0a34f0434e428ca3c10ef4c1e827 100644 (file)
@@ -213,6 +213,12 @@ def Test_expr4_equal()
   assert_equal(true, function('g:Test_expr4_equal', [123]) == function('g:Test_expr4_equal', [123]))
   assert_equal(false, function('g:Test_expr4_equal', [123]) == function('g:Test_expr4_is', [123]))
   assert_equal(false, function('g:Test_expr4_equal', [123]) == function('g:Test_expr4_equal', [999]))
+
+  let OneFunc: func
+  let TwoFunc: func
+  OneFunc = function('len')
+  TwoFunc = function('len')
+  assert_equal(true, OneFunc('abc') == TwoFunc('123'))
 enddef
 
 " test != comperator
index 4c821ec513d92eb01f10c29abbf5012df4656469..bc1f87377f1362d17ffd143ef1cd4758dcedeea7 100644 (file)
@@ -345,6 +345,9 @@ func Test_wrong_type()
   call CheckDefFailure(['let var: pam'], 'E1010:')
   call CheckDefFailure(['let var: sam'], 'E1010:')
   call CheckDefFailure(['let var: vim'], 'E1010:')
+
+  call CheckDefFailure(['let Ref: number', 'Ref()'], 'E1085:')
+  call CheckDefFailure(['let Ref: string', 'let res = Ref()'], 'E1085:')
 endfunc
 
 func Test_const()
index ebe266eda9395b377343fdc642144472129f55e2..eac5c9adfa55b086663d58d866c44d6e32eb08ab 100644 (file)
@@ -746,6 +746,8 @@ static char *(features[]) =
 
 static int included_patches[] =
 {   /* Add new patch number below this line */
+/**/
+    657,
 /**/
     656,
 /**/
index 89a86779a633e38e38f214de79d0a216d7a62ecb..aa3cd07d3d1fff295877df313891871e0400517a 100644 (file)
@@ -1327,15 +1327,32 @@ generate_UCALL(cctx_T *cctx, char_u *name, int argcount)
 
 /*
  * Generate an ISN_PCALL instruction.
+ * "type" is the type of the FuncRef.
  */
     static int
-generate_PCALL(cctx_T *cctx, int argcount, int at_top)
+generate_PCALL(
+       cctx_T  *cctx,
+       int     argcount,
+       char_u  *name,
+       type_T  *type,
+       int     at_top)
 {
     isn_T      *isn;
     garray_T   *stack = &cctx->ctx_type_stack;
+    type_T     *ret_type;
 
     RETURN_OK_IF_SKIP(cctx);
 
+    if (type->tt_type == VAR_ANY)
+       ret_type = &t_any;
+    else if (type->tt_type == VAR_FUNC || type->tt_type == VAR_PARTIAL)
+       ret_type = type->tt_member;
+    else
+    {
+       semsg(_("E1085: Not a callable type: %s"), name);
+       return FAIL;
+    }
+
     if ((isn = generate_instr(cctx, ISN_PCALL)) == NULL)
        return FAIL;
     isn->isn_arg.pfunc.cpf_top = at_top;
@@ -1344,7 +1361,7 @@ generate_PCALL(cctx_T *cctx, int argcount, int at_top)
     stack->ga_len -= argcount; // drop the arguments
 
     // drop the funcref/partial, get back the return value
-    ((type_T **)stack->ga_data)[stack->ga_len - 1] = &t_any;
+    ((type_T **)stack->ga_data)[stack->ga_len - 1] = ret_type;
 
     // If partial is above the arguments it must be cleared and replaced with
     // the return value.
@@ -2465,12 +2482,16 @@ compile_call(char_u **arg, size_t varlen, cctx_T *cctx, int argcount_init)
     if (STRNCMP(namebuf, "g:", 2) != 0
            && compile_load(&p, namebuf + varlen, cctx, FALSE) == OK)
     {
-       res = generate_PCALL(cctx, argcount, FALSE);
+       garray_T    *stack = &cctx->ctx_type_stack;
+       type_T      *type;
+
+       type = ((type_T **)stack->ga_data)[stack->ga_len - 1];
+       res = generate_PCALL(cctx, argcount, namebuf, type, FALSE);
        goto theend;
     }
 
     // A global function may be defined only later.  Need to figure out at
-    // runtime.
+    // runtime.  Also handles a FuncRef at runtime.
     if (STRNCMP(namebuf, "g:", 2) == 0)
        res = generate_UCALL(cctx, name, argcount);
     else
@@ -3120,13 +3141,17 @@ compile_subscript(
     {
        if (**arg == '(')
        {
-           int     argcount = 0;
+           garray_T    *stack = &cctx->ctx_type_stack;
+           type_T      *type;
+           int         argcount = 0;
 
            // funcref(arg)
+           type = ((type_T **)stack->ga_data)[stack->ga_len - 1];
+
            *arg = skipwhite(*arg + 1);
            if (compile_arguments(arg, cctx, &argcount) == FAIL)
                return FAIL;
-           if (generate_PCALL(cctx, argcount, TRUE) == FAIL)
+           if (generate_PCALL(cctx, argcount, end_leader, type, TRUE) == FAIL)
                return FAIL;
        }
        else if (**arg == '-' && (*arg)[1] == '>')