]> granicus.if.org Git - vim/commitdiff
patch 8.2.0662: cannot use input() in a channel callback v8.2.0662
authorBram Moolenaar <Bram@vim.org>
Wed, 29 Apr 2020 20:30:13 +0000 (22:30 +0200)
committerBram Moolenaar <Bram@vim.org>
Wed, 29 Apr 2020 20:30:13 +0000 (22:30 +0200)
Problem:    Cannot use input() in a channel callback.
Solution:   Reset vgetc_busy. (closes #6010)

src/evalfunc.c
src/ex_getln.c
src/globals.h
src/testdir/test_channel.vim
src/version.c

index 02943c717a87b3a44d5b295e1ce239d77e56af37..1493291954238f882be4c84cc87ac50841a09a27 100644 (file)
@@ -2149,7 +2149,7 @@ f_eval(typval_T *argvars, typval_T *rettv)
     static void
 f_eventhandler(typval_T *argvars UNUSED, typval_T *rettv)
 {
-    rettv->vval.v_number = vgetc_busy;
+    rettv->vval.v_number = vgetc_busy || input_busy;
 }
 
 static garray_T        redir_execute_ga;
@@ -2566,7 +2566,7 @@ f_feedkeys(typval_T *argvars, typval_T *rettv UNUSED)
 #ifdef FEAT_TIMERS
                        || timer_busy
 #endif
-                       )
+                       || input_busy)
                    typebuf_was_filled = TRUE;
            }
            vim_free(keys_esc);
@@ -2887,7 +2887,7 @@ f_funcref(typval_T *argvars, typval_T *rettv)
 }
 
     static type_T *
-ret_f_function(int argcount, type_T **argtypes UNUSED)
+ret_f_function(int argcount, type_T **argtypes)
 {
     if (argcount == 1 && argtypes[0]->tt_type == VAR_STRING)
        return &t_func_any;
index 7e92ec899296049f2b1c4860f45c6e7d06da5eff..6376a5fba41c5fcff4583a1ad2c5b979c42e4b20 100644 (file)
@@ -4468,6 +4468,8 @@ get_user_input(
 
     rettv->v_type = VAR_STRING;
     rettv->vval.v_string = NULL;
+    if (input_busy)
+       return;  // this doesn't work recursively.
 
 #ifdef NO_CONSOLE_INPUT
     // While starting up, there is no place to enter text. When running tests
@@ -4528,12 +4530,18 @@ get_user_input(
        if (defstr != NULL)
        {
            int save_ex_normal_busy = ex_normal_busy;
+           int save_vgetc_busy = vgetc_busy;
+           int save_input_busy = input_busy;
 
+           input_busy |= vgetc_busy;
            ex_normal_busy = 0;
+           vgetc_busy = 0;
            rettv->vval.v_string =
                getcmdline_prompt(secret ? NUL : '@', p, get_echo_attr(),
                                                              xp_type, xp_arg);
            ex_normal_busy = save_ex_normal_busy;
+           vgetc_busy = save_vgetc_busy;
+           input_busy = save_input_busy;
        }
        if (inputdialog && rettv->vval.v_string == NULL
                && argvars[1].v_type != VAR_UNKNOWN
index 725493b3202d153af8320013e7a56fe773f264a4..2dceab5e2363a467c05e67345d6a9674d9abdc4c 100644 (file)
@@ -1814,6 +1814,9 @@ EXTERN int  in_free_unref_items INIT(= FALSE);
 EXTERN int  did_add_timer INIT(= FALSE);
 EXTERN int  timer_busy INIT(= 0);   // when timer is inside vgetc() then > 0
 #endif
+#ifdef FEAT_EVAL
+EXTERN int  input_busy INIT(= 0);   // when inside get_user_input() then > 0
+#endif
 
 #ifdef FEAT_BEVAL_TERM
 EXTERN int  bevalexpr_due_set INIT(= FALSE);
index 4a438ea7292892f4396a5afa0bbc95729e3ff071..019b959251d748bb8c791087a0a22b37ac736408 100644 (file)
@@ -6,6 +6,7 @@ CheckFeature channel
 
 source shared.vim
 source screendump.vim
+source view_util.vim
 
 let s:python = PythonProg()
 if s:python == ''
@@ -2297,4 +2298,20 @@ func Test_job_with_list_args()
   %bw!
 endfunc
 
+func ExitCb_cb_with_input(job, status)
+  call feedkeys(":\<C-u>echo input('', 'default')\<CR>\<CR>", 'nx')
+  call assert_equal('default', Screenline(&lines))
+  let g:wait_exit_cb = 0
+endfunc
+
+func Test_cb_with_input()
+  let g:wait_exit_cb = 1
+
+  call job_start('echo "Vim''s test"',
+        \ {'out_cb': 'ExitCb_cb_with_input'})
+  call WaitForAssert({-> assert_equal(0, g:wait_exit_cb)})
+
+  unlet g:wait_exit_cb
+endfunc
+
 " vim: shiftwidth=2 sts=2 expandtab
index 0946468501f105d0d23af96a6ba27ffed6dcd5e8..df2fd1f55201a6556c7f054c5d84740b4418b571 100644 (file)
@@ -746,6 +746,8 @@ static char *(features[]) =
 
 static int included_patches[] =
 {   /* Add new patch number below this line */
+/**/
+    662,
 /**/
     661,
 /**/