]> granicus.if.org Git - vim/commitdiff
patch 8.1.2280: crash when passing partial to substitute() v8.1.2280
authorBram Moolenaar <Bram@vim.org>
Sat, 9 Nov 2019 21:28:11 +0000 (22:28 +0100)
committerBram Moolenaar <Bram@vim.org>
Sat, 9 Nov 2019 21:28:11 +0000 (22:28 +0100)
Problem:    Crash when passing partial to substitute().
Solution:   Take extra arguments into account. (closes #5186)

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

index 490bda4b88ed8fb93a1cc98be87d9ec48f65299f..01f1fff0d7e8430182e2f94350c672b58f9d50ee 100644 (file)
@@ -1,8 +1,6 @@
 /* regexp.c */
 int re_multiline(regprog_T *prog);
 char_u *skip_regexp(char_u *startp, int dirc, int magic, char_u **newp);
-int vim_regcomp_had_eol(void);
-void free_regexp_stuff(void);
 reg_extmatch_T *ref_extmatch(reg_extmatch_T *em);
 void unref_extmatch(reg_extmatch_T *em);
 char_u *regtilde(char_u *source, int magic);
@@ -10,8 +8,10 @@ int vim_regsub(regmatch_T *rmp, char_u *source, typval_T *expr, char_u *dest, in
 int vim_regsub_multi(regmmatch_T *rmp, linenr_T lnum, char_u *source, char_u *dest, int copy, int magic, int backslash);
 char_u *reg_submatch(int no);
 list_T *reg_submatch_list(int no);
+int vim_regcomp_had_eol(void);
 regprog_T *vim_regcomp(char_u *expr_arg, int re_flags);
 void vim_regfree(regprog_T *prog);
+void free_regexp_stuff(void);
 int regprog_in_use(regprog_T *prog);
 int vim_regexec_prog(regprog_T **prog, int ignore_case, char_u *line, colnr_T col);
 int vim_regexec(regmatch_T *rmp, char_u *line, colnr_T col);
index b952315b21c0b44c3ee8c719e531fc65f0e59eca..42f34c2f9957622856c7b6045c18bb8f1979ef9f 100644 (file)
@@ -1784,25 +1784,26 @@ static regsubmatch_T rsm;  /* can only be used when can_f_submatch is TRUE */
 #ifdef FEAT_EVAL
 
 /*
- * Put the submatches in "argv[0]" which is a list passed into call_func() by
- * vim_regsub_both().
+ * Put the submatches in "argv[argskip]" which is a list passed into
+ * call_func() by vim_regsub_both().
  */
     static int
-fill_submatch_list(int argc UNUSED, typval_T *argv, int argcount)
+fill_submatch_list(int argc UNUSED, typval_T *argv, int argskip, int argcount)
 {
     listitem_T *li;
     int                i;
     char_u     *s;
+    typval_T   *listarg = argv + argskip;
 
-    if (argcount == 0)
-       /* called function doesn't take an argument */
-       return 0;
+    if (argcount == argskip)
+       // called function doesn't take a submatches argument
+       return argskip;
 
-    /* Relies on sl_list to be the first item in staticList10_T. */
-    init_static_list((staticList10_T *)(argv->vval.v_list));
+    // Relies on sl_list to be the first item in staticList10_T.
+    init_static_list((staticList10_T *)(listarg->vval.v_list));
 
-    /* There are always 10 list items in staticList10_T. */
-    li = argv->vval.v_list->lv_first;
+    // There are always 10 list items in staticList10_T.
+    li = listarg->vval.v_list->lv_first;
     for (i = 0; i < 10; ++i)
     {
        s = rsm.sm_match->startp[i];
@@ -1814,7 +1815,7 @@ fill_submatch_list(int argc UNUSED, typval_T *argv, int argcount)
        li->li_tv.vval.v_string = s;
        li = li->li_next;
     }
-    return 1;
+    return argskip + 1;
 }
 
     static void
index fd3a1d072b7aa3fdd98766761e07ac9e31057d86..69dc7825e1d51bf3e8ad2a63b960a7d166514026 100644 (file)
@@ -1627,10 +1627,11 @@ typedef struct
 //
 // "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, called_func_argcount)
+//   new_argcount = argv_func(current_argcount, argv, partial_argcount,
+//                                                     called_func_argcount)
 //
 typedef struct {
-    int                (* argv_func)(int, typval_T *, int);
+    int                (* argv_func)(int, typval_T *, int, int);
     linenr_T   firstline;      // first line of range
     linenr_T   lastline;       // last line of range
     int                *doesrange;     // if not NULL: return: function handled range
index e8b0e49c2e402a8e044eeaea431209668a1c5bb3..b30464ff1ac9d8637e6db154713518fbe36d7be6 100644 (file)
@@ -405,6 +405,14 @@ func Test_sub_replace_10()
    call assert_equal('1aaa', substitute('123', '1\zs\|[23]', 'a', 'g'))
 endfunc
 
+func SubReplacer(text, submatches)
+  return a:text .. a:submatches[0] .. a:text
+endfunc
+
+func Test_substitute_partial()
+   call assert_equal('1foo2foo3', substitute('123', '2', function('SubReplacer', ['foo']), 'g'))
+endfunc
+
 " Tests for *sub-replace-special* and *sub-replace-expression* on :substitute.
 
 " Execute a list of :substitute command tests
index 40eb14401d11dee3a3e3baa0312c2f3cf3bae3a8..cfc52befe4aacd4ae3b407ce7eda2203cf043871 100644 (file)
@@ -1588,7 +1588,8 @@ call_func(
            else if (fp != NULL)
            {
                if (funcexe->argv_func != NULL)
-                   argcount = funcexe->argv_func(argcount, argvars,
+                   // postponed filling in the arguments, do it now
+                   argcount = funcexe->argv_func(argcount, argvars, argv_clear,
                                                           fp->uf_args.ga_len);
 
                if (funcexe->basetv != NULL)
index a38d3357ccb367afb326994a1a4a2eba8563f8e7..2d79cbed6b4154eb12933e1204670d780c86b2fe 100644 (file)
@@ -741,6 +741,8 @@ static char *(features[]) =
 
 static int included_patches[] =
 {   /* Add new patch number below this line */
+/**/
+    2280,
 /**/
     2279,
 /**/