Patch 8.2.0657
Problem:    Vim9: no check if called variable is a FuncRef.
Solution:   Add a type check.
Files:      src/vim9compile.c, src/testdir/test_vim9_script.vim,
            src/testdir/test_vim9_expr.vim


*** ../vim-8.2.0656/src/vim9compile.c   2020-04-27 22:47:45.186176148 +0200
--- src/vim9compile.c   2020-04-28 21:25:07.282335257 +0200
***************
*** 1327,1341 ****
  
  /*
   * Generate an ISN_PCALL instruction.
   */
      static int
! generate_PCALL(cctx_T *cctx, int argcount, int at_top)
  {
      isn_T     *isn;
      garray_T  *stack = &cctx->ctx_type_stack;
  
      RETURN_OK_IF_SKIP(cctx);
  
      if ((isn = generate_instr(cctx, ISN_PCALL)) == NULL)
        return FAIL;
      isn->isn_arg.pfunc.cpf_top = at_top;
--- 1327,1358 ----
  
  /*
   * Generate an ISN_PCALL instruction.
+  * "type" is the type of the FuncRef.
   */
      static int
! 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,1350 ****
      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;
  
      // If partial is above the arguments it must be cleared and replaced with
      // the return value.
--- 1361,1367 ----
      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] = ret_type;
  
      // If partial is above the arguments it must be cleared and replaced with
      // the return value.
***************
*** 2465,2476 ****
      if (STRNCMP(namebuf, "g:", 2) != 0
            && compile_load(&p, namebuf + varlen, cctx, FALSE) == OK)
      {
!       res = generate_PCALL(cctx, argcount, FALSE);
        goto theend;
      }
  
      // A global function may be defined only later.  Need to figure out at
!     // runtime.
      if (STRNCMP(namebuf, "g:", 2) == 0)
        res = generate_UCALL(cctx, name, argcount);
      else
--- 2482,2497 ----
      if (STRNCMP(namebuf, "g:", 2) != 0
            && compile_load(&p, namebuf + varlen, cctx, FALSE) == OK)
      {
!       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.  Also handles a FuncRef at runtime.
      if (STRNCMP(namebuf, "g:", 2) == 0)
        res = generate_UCALL(cctx, name, argcount);
      else
***************
*** 3120,3132 ****
      {
        if (**arg == '(')
        {
!           int     argcount = 0;
  
            // funcref(arg)
            *arg = skipwhite(*arg + 1);
            if (compile_arguments(arg, cctx, &argcount) == FAIL)
                return FAIL;
!           if (generate_PCALL(cctx, argcount, TRUE) == FAIL)
                return FAIL;
        }
        else if (**arg == '-' && (*arg)[1] == '>')
--- 3141,3157 ----
      {
        if (**arg == '(')
        {
!           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, end_leader, type, TRUE) == FAIL)
                return FAIL;
        }
        else if (**arg == '-' && (*arg)[1] == '>')
*** ../vim-8.2.0656/src/testdir/test_vim9_script.vim    2020-04-27 
23:39:26.416849722 +0200
--- src/testdir/test_vim9_script.vim    2020-04-28 21:18:29.515316414 +0200
***************
*** 345,350 ****
--- 345,353 ----
    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()
*** ../vim-8.2.0656/src/testdir/test_vim9_expr.vim      2020-04-27 
22:47:45.186176148 +0200
--- src/testdir/test_vim9_expr.vim      2020-04-28 21:22:18.694750557 +0200
***************
*** 213,218 ****
--- 213,224 ----
    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
*** ../vim-8.2.0656/src/version.c       2020-04-28 20:44:38.872258441 +0200
--- src/version.c       2020-04-28 21:29:08.225742426 +0200
***************
*** 748,749 ****
--- 748,751 ----
  {   /* Add new patch number below this line */
+ /**/
+     657,
  /**/

-- 
How To Keep A Healthy Level Of Insanity:
9. As often as possible, skip rather than walk.

 /// Bram Moolenaar -- [email protected] -- http://www.Moolenaar.net   \\\
///        sponsor Vim, vote for features -- http://www.Vim.org/sponsor/ \\\
\\\  an exciting new programming language -- http://www.Zimbu.org        ///
 \\\            help me help AIDS victims -- http://ICCF-Holland.org    ///

-- 
-- 
You received this message from the "vim_dev" maillist.
Do not top-post! Type your reply below the text you are replying to.
For more information, visit http://www.vim.org/maillist.php

--- 
You received this message because you are subscribed to the Google Groups 
"vim_dev" group.
To unsubscribe from this group and stop receiving emails from it, send an email 
to [email protected].
To view this discussion on the web visit 
https://groups.google.com/d/msgid/vim_dev/202004281930.03SJUBKF031441%40masaka.moolenaar.net.

Raspunde prin e-mail lui