]> granicus.if.org Git - vim/commitdiff
patch 9.0.0476: varargs does not work for replacement function of substitute() v9.0.0476
authorzeertzjq <zeertzjq@outlook.com>
Fri, 16 Sep 2022 11:10:03 +0000 (12:10 +0100)
committerBram Moolenaar <Bram@vim.org>
Fri, 16 Sep 2022 11:10:03 +0000 (12:10 +0100)
Problem:    Varargs does not work for replacement function of substitute().
Solution:   Check the varargs flag of the function. (closes #11142)

src/regexp.c
src/structs.h
src/testdir/test_substitute.vim
src/userfunc.c
src/version.c

index 8e6e9c44058a2a6f56c6f681a48dcd70fd46266c..1bd126169066f6290c29f565e71100708b06ba2c 100644 (file)
@@ -1817,14 +1817,14 @@ static regsubmatch_T rsm;  // can only be used when can_f_submatch is TRUE
  * call_func() by vim_regsub_both().
  */
     static int
-fill_submatch_list(int argc UNUSED, typval_T *argv, int argskip, int argcount)
+fill_submatch_list(int argc UNUSED, typval_T *argv, int argskip, ufunc_T *fp)
 {
     listitem_T *li;
     int                i;
     char_u     *s;
     typval_T   *listarg = argv + argskip;
 
-    if (argcount == argskip)
+    if (!fp->uf_varargs && fp->uf_args.ga_len <= argskip)
        // called function doesn't take a submatches argument
        return argskip;
 
index 581b22eac62fba0ab78d0250ba2aedf8cb541d16..be08f0a53148bd15ee9b9897accc3bcc0514a4f8 100644 (file)
@@ -2052,13 +2052,13 @@ typedef struct
 
 // Struct passed between functions dealing with function call execution.
 //
-// "argv_func", when not NULL, can be used to fill in arguments only when the
+// "fe_argv_func", when not NULL, can be used to fill in arguments only when the
 // invoked function uses them.  It is called like this:
-//   new_argcount = argv_func(current_argcount, argv, partial_argcount,
-//                                                     called_func_argcount)
+//   new_argcount = fe_argv_func(current_argcount, argv, partial_argcount,
+//                                                     called_func)
 //
 typedef struct {
-    int                (* fe_argv_func)(int, typval_T *, int, int);
+    int                (* fe_argv_func)(int, typval_T *, int, ufunc_T *);
     linenr_T   fe_firstline;   // first line of range
     linenr_T   fe_lastline;    // last line of range
     int                *fe_doesrange;  // if not NULL: return: function handled range
index 92e86a9a1035a23019aae46dd8e0cdb2dc5cab95..af450d595b4fb4fe02dc235657b938546c8091f9 100644 (file)
@@ -439,20 +439,24 @@ endfunc
 func SubReplacer(text, submatches)
   return a:text .. a:submatches[0] .. a:text
 endfunc
+func SubReplacerVar(text, ...)
+  return a:text .. a:1[0] .. a:text
+endfunc
 func SubReplacer20(t1, t2, t3, t4, t5, t6, t7, t8, t9, t10, t11, t12, t13, t14, t15, t16, t17, t18, t19, submatches)
   return a:t3 .. a:submatches[0] .. a:t11
 endfunc
 
 func Test_substitute_partial()
-   call assert_equal('1foo2foo3', substitute('123', '2', function('SubReplacer', ['foo']), 'g'))
+  call assert_equal('1foo2foo3', substitute('123', '2', function('SubReplacer', ['foo']), 'g'))
+  call assert_equal('1foo2foo3', substitute('123', '2', function('SubReplacerVar', ['foo']), 'g'))
 
-   " 19 arguments plus one is just OK
-   let Replacer = function('SubReplacer20', repeat(['foo'], 19))
-   call assert_equal('1foo2foo3', substitute('123', '2', Replacer, 'g'))
+  " 19 arguments plus one is just OK
+  let Replacer = function('SubReplacer20', repeat(['foo'], 19))
+  call assert_equal('1foo2foo3', substitute('123', '2', Replacer, 'g'))
 
-   " 20 arguments plus one is too many
-   let Replacer = function('SubReplacer20', repeat(['foo'], 20))
-   call assert_fails("call substitute('123', '2', Replacer, 'g')", 'E118:')
+  " 20 arguments plus one is too many
+  let Replacer = function('SubReplacer20', repeat(['foo'], 20))
+  call assert_fails("call substitute('123', '2', Replacer, 'g')", 'E118:')
 endfunc
 
 func Test_substitute_float()
index 1412caa8e08ff1fc32ee887d5117f20760acc77b..35ce3f308650869fc4ae4b1c12ec6607439f3384 100644 (file)
@@ -3644,7 +3644,7 @@ call_func(
                if (funcexe->fe_argv_func != NULL)
                    // postponed filling in the arguments, do it now
                    argcount = funcexe->fe_argv_func(argcount, argvars,
-                                              argv_clear, fp->uf_args.ga_len);
+                                              argv_clear, fp);
 
                if (funcexe->fe_basetv != NULL)
                {
index ae740816a9dc689c35078f529f2224d211357570..77db15595ace157639902fab4a4bdc84578a068a 100644 (file)
@@ -703,6 +703,8 @@ static char *(features[]) =
 
 static int included_patches[] =
 {   /* Add new patch number below this line */
+/**/
+    476,
 /**/
     475,
 /**/