patch 8.0.0976: cannot send lines to a terminal job
Problem:    Cannot send lines to a terminal job.
Solution:   Make [range]terminal send selected lines to the job.
            Use ++rows and ++cols for the terminal size.
			
			
This commit is contained in:
		| @ -1,4 +1,4 @@ | |||||||
| *terminal.txt*	For Vim version 8.0.  Last change: 2017 Aug 12 | *terminal.txt*	For Vim version 8.0.  Last change: 2017 Aug 20 | ||||||
|  |  | ||||||
|  |  | ||||||
| 		  VIM REFERENCE MANUAL	  by Bram Moolenaar | 		  VIM REFERENCE MANUAL	  by Bram Moolenaar | ||||||
| @ -102,10 +102,9 @@ Syntax ~ | |||||||
| 			parentheses.  E.g. if "gdb" exists the second terminal | 			parentheses.  E.g. if "gdb" exists the second terminal | ||||||
| 			buffer will use "!gdb (1)". | 			buffer will use "!gdb (1)". | ||||||
|  |  | ||||||
| 			If [range] is given it is used for the terminal size. | 			If [range] is given the specified lines are used as | ||||||
| 			One number specifies the number of rows.  Unless the | 			input for the job.  It will not be possible to type | ||||||
| 			"vertical" modifier is used, then it is the number of | 			keys in the terminal window. | ||||||
| 			columns. |  | ||||||
|  |  | ||||||
| 			Two comma separated numbers are used as "rows,cols". | 			Two comma separated numbers are used as "rows,cols". | ||||||
| 			E.g. `:24,80gdb` opens a terminal with 24 rows and 80 | 			E.g. `:24,80gdb` opens a terminal with 24 rows and 80 | ||||||
| @ -125,6 +124,10 @@ Syntax ~ | |||||||
| 					cannot be |abandon|ed. | 					cannot be |abandon|ed. | ||||||
| 			++hidden	Open the terminal in a hidden buffer, | 			++hidden	Open the terminal in a hidden buffer, | ||||||
| 					no window will be used. | 					no window will be used. | ||||||
|  | 			++rows={height} Use {height} for the terminal window | ||||||
|  | 					height. | ||||||
|  | 			++cols={width}  Use {width} for the terminal window | ||||||
|  | 					width. | ||||||
|  |  | ||||||
| 			If you want to use more options use the |term_start()| | 			If you want to use more options use the |term_start()| | ||||||
| 			function. | 			function. | ||||||
|  | |||||||
| @ -1484,8 +1484,8 @@ EX(CMD_tearoff,		"tearoff",	ex_tearoff, | |||||||
| 			NEEDARG|EXTRA|TRLBAR|NOTRLCOM|CMDWIN, | 			NEEDARG|EXTRA|TRLBAR|NOTRLCOM|CMDWIN, | ||||||
| 			ADDR_LINES), | 			ADDR_LINES), | ||||||
| EX(CMD_terminal,	"terminal",	ex_terminal, | EX(CMD_terminal,	"terminal",	ex_terminal, | ||||||
| 			RANGE|NOTADR|BANG|FILES|TRLBAR|CMDWIN, | 			RANGE|BANG|FILES|TRLBAR|CMDWIN, | ||||||
| 			ADDR_OTHER), | 			ADDR_LINES), | ||||||
| EX(CMD_tfirst,		"tfirst",	ex_tag, | EX(CMD_tfirst,		"tfirst",	ex_tag, | ||||||
| 			RANGE|NOTADR|BANG|TRLBAR|ZEROR, | 			RANGE|NOTADR|BANG|TRLBAR|ZEROR, | ||||||
| 			ADDR_LINES), | 			ADDR_LINES), | ||||||
|  | |||||||
| @ -5238,6 +5238,7 @@ mch_job_start(char **argv, job_T *job, jobopt_T *options) | |||||||
|     int		use_file_for_in = options->jo_io[PART_IN] == JIO_FILE; |     int		use_file_for_in = options->jo_io[PART_IN] == JIO_FILE; | ||||||
|     int		use_file_for_out = options->jo_io[PART_OUT] == JIO_FILE; |     int		use_file_for_out = options->jo_io[PART_OUT] == JIO_FILE; | ||||||
|     int		use_file_for_err = options->jo_io[PART_ERR] == JIO_FILE; |     int		use_file_for_err = options->jo_io[PART_ERR] == JIO_FILE; | ||||||
|  |     int		use_buffer_for_in = options->jo_io[PART_IN] == JIO_BUFFER; | ||||||
|     int		use_out_for_err = options->jo_io[PART_ERR] == JIO_OUT; |     int		use_out_for_err = options->jo_io[PART_ERR] == JIO_OUT; | ||||||
|     SIGSET_DECL(curset) |     SIGSET_DECL(curset) | ||||||
|  |  | ||||||
| @ -5247,7 +5248,10 @@ mch_job_start(char **argv, job_T *job, jobopt_T *options) | |||||||
|     /* default is to fail */ |     /* default is to fail */ | ||||||
|     job->jv_status = JOB_FAILED; |     job->jv_status = JOB_FAILED; | ||||||
|  |  | ||||||
|     if (options->jo_pty) |     if (options->jo_pty | ||||||
|  | 	    && (!(use_file_for_in || use_null_for_in) | ||||||
|  | 		|| !(use_file_for_in || use_null_for_out) | ||||||
|  | 		|| !(use_out_for_err || use_file_for_err || use_null_for_err))) | ||||||
| 	open_pty(&pty_master_fd, &pty_slave_fd, &job->jv_tty_name); | 	open_pty(&pty_master_fd, &pty_slave_fd, &job->jv_tty_name); | ||||||
|  |  | ||||||
|     /* TODO: without the channel feature connect the child to /dev/null? */ |     /* TODO: without the channel feature connect the child to /dev/null? */ | ||||||
| @ -5263,8 +5267,12 @@ mch_job_start(char **argv, job_T *job, jobopt_T *options) | |||||||
| 	    goto failed; | 	    goto failed; | ||||||
| 	} | 	} | ||||||
|     } |     } | ||||||
|     else if (!use_null_for_in && pty_master_fd < 0 && pipe(fd_in) < 0) |     else | ||||||
| 	goto failed; | 	/* When writing buffer lines to the input don't use the pty, so that | ||||||
|  | 	 * the pipe can be closed when all lines were written. */ | ||||||
|  | 	if (!use_null_for_in && (pty_master_fd < 0 || use_buffer_for_in) | ||||||
|  | 							    && pipe(fd_in) < 0) | ||||||
|  | 	    goto failed; | ||||||
|  |  | ||||||
|     if (use_file_for_out) |     if (use_file_for_out) | ||||||
|     { |     { | ||||||
|  | |||||||
| @ -38,7 +38,7 @@ | |||||||
|  * in tl_scrollback are no longer used. |  * in tl_scrollback are no longer used. | ||||||
|  * |  * | ||||||
|  * TODO: |  * TODO: | ||||||
|  * - make [range]terminal pipe [range] lines to the terminal |  * - test writing lines to terminal job when implemented for MS-Windows | ||||||
|  * - implement term_setsize() |  * - implement term_setsize() | ||||||
|  * - add test for giving error for invalid 'termsize' value. |  * - add test for giving error for invalid 'termsize' value. | ||||||
|  * - support minimal size when 'termsize' is "rows*cols". |  * - support minimal size when 'termsize' is "rows*cols". | ||||||
| @ -447,10 +447,14 @@ ex_terminal(exarg_T *eap) | |||||||
|     cmd = eap->arg; |     cmd = eap->arg; | ||||||
|     while (*cmd && *cmd == '+' && *(cmd + 1) == '+') |     while (*cmd && *cmd == '+' && *(cmd + 1) == '+') | ||||||
|     { |     { | ||||||
| 	char_u  *p; | 	char_u  *p, *ep; | ||||||
|  |  | ||||||
| 	cmd += 2; | 	cmd += 2; | ||||||
| 	p = skiptowhite(cmd); | 	p = skiptowhite(cmd); | ||||||
|  | 	ep = vim_strchr(cmd, '='); | ||||||
|  | 	if (ep != NULL && ep < p) | ||||||
|  | 	    p = ep; | ||||||
|  |  | ||||||
| 	if ((int)(p - cmd) == 5 && STRNICMP(cmd, "close", 5) == 0) | 	if ((int)(p - cmd) == 5 && STRNICMP(cmd, "close", 5) == 0) | ||||||
| 	    opt.jo_term_finish = 'c'; | 	    opt.jo_term_finish = 'c'; | ||||||
| 	else if ((int)(p - cmd) == 4 && STRNICMP(cmd, "open", 4) == 0) | 	else if ((int)(p - cmd) == 4 && STRNICMP(cmd, "open", 4) == 0) | ||||||
| @ -459,6 +463,20 @@ ex_terminal(exarg_T *eap) | |||||||
| 	    opt.jo_curwin = 1; | 	    opt.jo_curwin = 1; | ||||||
| 	else if ((int)(p - cmd) == 6 && STRNICMP(cmd, "hidden", 6) == 0) | 	else if ((int)(p - cmd) == 6 && STRNICMP(cmd, "hidden", 6) == 0) | ||||||
| 	    opt.jo_hidden = 1; | 	    opt.jo_hidden = 1; | ||||||
|  | 	else if ((int)(p - cmd) == 4 && STRNICMP(cmd, "rows", 4) == 0 | ||||||
|  | 		&& ep != NULL && isdigit(ep[1])) | ||||||
|  | 	{ | ||||||
|  | 	    opt.jo_set2 |= JO2_TERM_ROWS; | ||||||
|  | 	    opt.jo_term_rows = atoi((char *)ep + 1); | ||||||
|  | 	    p = skiptowhite(cmd); | ||||||
|  | 	} | ||||||
|  | 	else if ((int)(p - cmd) == 4 && STRNICMP(cmd, "cols", 4) == 0 | ||||||
|  | 		&& ep != NULL && isdigit(ep[1])) | ||||||
|  | 	{ | ||||||
|  | 	    opt.jo_set2 |= JO2_TERM_COLS; | ||||||
|  | 	    opt.jo_term_cols = atoi((char *)ep + 1); | ||||||
|  | 	    p = skiptowhite(cmd); | ||||||
|  | 	} | ||||||
| 	else | 	else | ||||||
| 	{ | 	{ | ||||||
| 	    if (*p) | 	    if (*p) | ||||||
| @ -472,17 +490,14 @@ ex_terminal(exarg_T *eap) | |||||||
| 	/* Make a copy, an autocommand may set 'shell'. */ | 	/* Make a copy, an autocommand may set 'shell'. */ | ||||||
| 	tofree = cmd = vim_strsave(p_sh); | 	tofree = cmd = vim_strsave(p_sh); | ||||||
|  |  | ||||||
|     if (eap->addr_count == 2) |     if (eap->addr_count > 0) | ||||||
|     { |     { | ||||||
| 	opt.jo_term_rows = eap->line1; | 	/* Write lines from current buffer to the job. */ | ||||||
| 	opt.jo_term_cols = eap->line2; | 	opt.jo_set |= JO_IN_IO | JO_IN_BUF | JO_IN_TOP | JO_IN_BOT; | ||||||
|     } | 	opt.jo_io[PART_IN] = JIO_BUFFER; | ||||||
|     else if (eap->addr_count == 1) | 	opt.jo_io_buf[PART_IN] = curbuf->b_fnum; | ||||||
|     { | 	opt.jo_in_top = eap->line1; | ||||||
| 	if (cmdmod.split & WSP_VERT) | 	opt.jo_in_bot = eap->line2; | ||||||
| 	    opt.jo_term_cols = eap->line2; |  | ||||||
| 	else |  | ||||||
| 	    opt.jo_term_rows = eap->line2; |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     argvar.v_type = VAR_STRING; |     argvar.v_type = VAR_STRING; | ||||||
| @ -1077,6 +1092,29 @@ term_vgetc() | |||||||
|     return c; |     return c; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | /* | ||||||
|  |  * Get the part that is connected to the tty. Normally this is PART_IN, but | ||||||
|  |  * when writing buffer lines to the job it can be another.  This makes it | ||||||
|  |  * possible to do "1,5term vim -". | ||||||
|  |  */ | ||||||
|  |     static ch_part_T | ||||||
|  | get_tty_part(term_T *term) | ||||||
|  | { | ||||||
|  | #ifdef UNIX | ||||||
|  |     ch_part_T	parts[3] = {PART_IN, PART_OUT, PART_ERR}; | ||||||
|  |     int		i; | ||||||
|  |  | ||||||
|  |     for (i = 0; i < 3; ++i) | ||||||
|  |     { | ||||||
|  | 	int fd = term->tl_job->jv_channel->ch_part[parts[i]].ch_fd; | ||||||
|  |  | ||||||
|  | 	if (isatty(fd)) | ||||||
|  | 	    return parts[i]; | ||||||
|  |     } | ||||||
|  | #endif | ||||||
|  |     return PART_IN; | ||||||
|  | } | ||||||
|  |  | ||||||
| /* | /* | ||||||
|  * Send keys to terminal. |  * Send keys to terminal. | ||||||
|  * Return FAIL when the key needs to be handled in Normal mode. |  * Return FAIL when the key needs to be handled in Normal mode. | ||||||
| @ -1148,8 +1186,8 @@ send_keys_to_term(term_T *term, int c, int typed) | |||||||
|     len = term_convert_key(term, c, msg); |     len = term_convert_key(term, c, msg); | ||||||
|     if (len > 0) |     if (len > 0) | ||||||
| 	/* TODO: if FAIL is returned, stop? */ | 	/* TODO: if FAIL is returned, stop? */ | ||||||
| 	channel_send(term->tl_job->jv_channel, PART_IN, | 	channel_send(term->tl_job->jv_channel, get_tty_part(term), | ||||||
| 						 (char_u *)msg, (int)len, NULL); | 						(char_u *)msg, (int)len, NULL); | ||||||
|  |  | ||||||
|     return OK; |     return OK; | ||||||
| } | } | ||||||
| @ -1332,7 +1370,8 @@ terminal_loop(void) | |||||||
|  |  | ||||||
| #ifdef UNIX | #ifdef UNIX | ||||||
|     { |     { | ||||||
| 	int fd = curbuf->b_term->tl_job->jv_channel->ch_part[PART_IN].ch_fd; | 	int part = get_tty_part(curbuf->b_term); | ||||||
|  | 	int fd = curbuf->b_term->tl_job->jv_channel->ch_part[part].ch_fd; | ||||||
|  |  | ||||||
| 	if (isatty(fd)) | 	if (isatty(fd)) | ||||||
| 	{ | 	{ | ||||||
| @ -2823,7 +2862,12 @@ dyn_winpty_init(int verbose) | |||||||
|  * Return OK or FAIL. |  * Return OK or FAIL. | ||||||
|  */ |  */ | ||||||
|     static int |     static int | ||||||
| term_and_job_init(term_T *term, int rows, int cols, typval_T *argvar, jobopt_T *opt) | term_and_job_init( | ||||||
|  | 	term_T	    *term, | ||||||
|  | 	int	    rows, | ||||||
|  | 	int	    cols, | ||||||
|  | 	typval_T    *argvar, | ||||||
|  | 	jobopt_T    *opt) | ||||||
| { | { | ||||||
|     WCHAR	    *p = NULL; |     WCHAR	    *p = NULL; | ||||||
|     channel_T	    *channel = NULL; |     channel_T	    *channel = NULL; | ||||||
| @ -3020,7 +3064,12 @@ terminal_enabled(void) | |||||||
|  * Return OK or FAIL. |  * Return OK or FAIL. | ||||||
|  */ |  */ | ||||||
|     static int |     static int | ||||||
| term_and_job_init(term_T *term, int rows, int cols, typval_T *argvar, jobopt_T *opt) | term_and_job_init( | ||||||
|  | 	term_T	    *term, | ||||||
|  | 	int	    rows, | ||||||
|  | 	int	    cols, | ||||||
|  | 	typval_T    *argvar, | ||||||
|  | 	jobopt_T    *opt) | ||||||
| { | { | ||||||
|     create_vterm(term, rows, cols); |     create_vterm(term, rows, cols); | ||||||
|  |  | ||||||
|  | |||||||
| @ -251,7 +251,7 @@ endfunc | |||||||
| func Test_terminal_size() | func Test_terminal_size() | ||||||
|   let cmd = Get_cat_123_cmd() |   let cmd = Get_cat_123_cmd() | ||||||
|  |  | ||||||
|   exe '5terminal ' . cmd |   exe 'terminal ++rows=5 ' . cmd | ||||||
|   let size = term_getsize('') |   let size = term_getsize('') | ||||||
|   bwipe! |   bwipe! | ||||||
|   call assert_equal(5, size[0]) |   call assert_equal(5, size[0]) | ||||||
| @ -262,7 +262,7 @@ func Test_terminal_size() | |||||||
|   call assert_equal(6, size[0]) |   call assert_equal(6, size[0]) | ||||||
|  |  | ||||||
|   vsplit |   vsplit | ||||||
|   exe '5,33terminal ' . cmd |   exe 'terminal ++rows=5 ++cols=33 ' . cmd | ||||||
|   let size = term_getsize('') |   let size = term_getsize('') | ||||||
|   bwipe! |   bwipe! | ||||||
|   call assert_equal([5, 33], size) |   call assert_equal([5, 33], size) | ||||||
| @ -272,7 +272,7 @@ func Test_terminal_size() | |||||||
|   bwipe! |   bwipe! | ||||||
|   call assert_equal([6, 36], size) |   call assert_equal([6, 36], size) | ||||||
|  |  | ||||||
|   exe 'vertical 20terminal ' . cmd |   exe 'vertical terminal ++cols=20 ' . cmd | ||||||
|   let size = term_getsize('') |   let size = term_getsize('') | ||||||
|   bwipe! |   bwipe! | ||||||
|   call assert_equal(20, size[1]) |   call assert_equal(20, size[1]) | ||||||
| @ -283,7 +283,7 @@ func Test_terminal_size() | |||||||
|   call assert_equal(26, size[1]) |   call assert_equal(26, size[1]) | ||||||
|  |  | ||||||
|   split |   split | ||||||
|   exe 'vertical 6,20terminal ' . cmd |   exe 'vertical terminal ++rows=6 ++cols=20 ' . cmd | ||||||
|   let size = term_getsize('') |   let size = term_getsize('') | ||||||
|   bwipe! |   bwipe! | ||||||
|   call assert_equal([6, 20], size) |   call assert_equal([6, 20], size) | ||||||
|  | |||||||
| @ -769,6 +769,8 @@ static char *(features[]) = | |||||||
|  |  | ||||||
| static int included_patches[] = | static int included_patches[] = | ||||||
| {   /* Add new patch number below this line */ | {   /* Add new patch number below this line */ | ||||||
|  | /**/ | ||||||
|  |     976, | ||||||
| /**/ | /**/ | ||||||
|     975, |     975, | ||||||
| /**/ | /**/ | ||||||
|  | |||||||
		Reference in New Issue
	
	Block a user