summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorBram Moolenaar <Bram@vim.org>2020-04-29 22:30:13 +0200
committerBram Moolenaar <Bram@vim.org>2020-04-29 22:30:13 +0200
commitdfc33a665d3b12689aa971575b8e7de4e5202d83 (patch)
tree8c046a483c6b78d558f9ad178ed7e9369c4efe5c
parent339d60c89bef0315343a6128a1b4826510b0efe4 (diff)
patch 8.2.0662: cannot use input() in a channel callbackv8.2.0662
Problem: Cannot use input() in a channel callback. Solution: Reset vgetc_busy. (closes #6010)
-rw-r--r--src/evalfunc.c6
-rw-r--r--src/ex_getln.c8
-rw-r--r--src/globals.h3
-rw-r--r--src/testdir/test_channel.vim17
-rw-r--r--src/version.c2
5 files changed, 33 insertions, 3 deletions
diff --git a/src/evalfunc.c b/src/evalfunc.c
index 02943c717a..1493291954 100644
--- a/src/evalfunc.c
+++ b/src/evalfunc.c
@@ -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;
diff --git a/src/ex_getln.c b/src/ex_getln.c
index 7e92ec8992..6376a5fba4 100644
--- a/src/ex_getln.c
+++ b/src/ex_getln.c
@@ -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
diff --git a/src/globals.h b/src/globals.h
index 725493b320..2dceab5e23 100644
--- a/src/globals.h
+++ b/src/globals.h
@@ -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);
diff --git a/src/testdir/test_channel.vim b/src/testdir/test_channel.vim
index 4a438ea729..019b959251 100644
--- a/src/testdir/test_channel.vim
+++ b/src/testdir/test_channel.vim
@@ -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
diff --git a/src/version.c b/src/version.c
index 0946468501..df2fd1f552 100644
--- a/src/version.c
+++ b/src/version.c
@@ -747,6 +747,8 @@ static char *(features[]) =
static int included_patches[] =
{ /* Add new patch number below this line */
/**/
+ 662,
+/**/
661,
/**/
660,