xref: /freebsd/contrib/libedit/tty.c (revision 732a02b4e77866604a120a275c082bb6221bd2ff)
1 /*	$NetBSD: tty.c,v 1.68 2018/12/02 16:58:13 christos Exp $	*/
2 
3 /*-
4  * Copyright (c) 1992, 1993
5  *	The Regents of the University of California.  All rights reserved.
6  *
7  * This code is derived from software contributed to Berkeley by
8  * Christos Zoulas of Cornell University.
9  *
10  * Redistribution and use in source and binary forms, with or without
11  * modification, are permitted provided that the following conditions
12  * are met:
13  * 1. Redistributions of source code must retain the above copyright
14  *    notice, this list of conditions and the following disclaimer.
15  * 2. Redistributions in binary form must reproduce the above copyright
16  *    notice, this list of conditions and the following disclaimer in the
17  *    documentation and/or other materials provided with the distribution.
18  * 3. Neither the name of the University nor the names of its contributors
19  *    may be used to endorse or promote products derived from this software
20  *    without specific prior written permission.
21  *
22  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
23  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
24  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
25  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
26  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
27  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
28  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
29  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
30  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
31  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
32  * SUCH DAMAGE.
33  */
34 
35 #include "config.h"
36 #if !defined(lint) && !defined(SCCSID)
37 #if 0
38 static char sccsid[] = "@(#)tty.c	8.1 (Berkeley) 6/4/93";
39 #else
40 __RCSID("$NetBSD: tty.c,v 1.68 2018/12/02 16:58:13 christos Exp $");
41 #endif
42 #endif /* not lint && not SCCSID */
43 
44 /*
45  * tty.c: tty interface stuff
46  */
47 #include <assert.h>
48 #include <errno.h>
49 #include <stdlib.h>	/* for abort */
50 #include <string.h>
51 #include <strings.h>	/* for ffs */
52 #include <unistd.h>	/* for isatty */
53 
54 #include "el.h"
55 #include "fcns.h"
56 #include "parse.h"
57 
58 typedef struct ttymodes_t {
59 	const char *m_name;
60 	unsigned int m_value;
61 	int m_type;
62 }          ttymodes_t;
63 
64 typedef struct ttymap_t {
65 	wint_t nch, och;	/* Internal and termio rep of chars */
66 	el_action_t bind[3];	/* emacs, vi, and vi-cmd */
67 } ttymap_t;
68 
69 
70 static const ttyperm_t ttyperm = {
71 	{
72 		{"iflag:", ICRNL, (INLCR | IGNCR)},
73 		{"oflag:", (OPOST | ONLCR), ONLRET},
74 		{"cflag:", 0, 0},
75 		{"lflag:", (ISIG | ICANON | ECHO | ECHOE | ECHOCTL | IEXTEN),
76 		(NOFLSH | ECHONL | EXTPROC | FLUSHO)},
77 		{"chars:", 0, 0},
78 	},
79 	{
80 		{"iflag:", (INLCR | ICRNL), IGNCR},
81 		{"oflag:", (OPOST | ONLCR), ONLRET},
82 		{"cflag:", 0, 0},
83 		{"lflag:", ISIG,
84 		(NOFLSH | ICANON | ECHO | ECHOK | ECHONL | EXTPROC | IEXTEN | FLUSHO)},
85 		{"chars:", (C_SH(C_MIN) | C_SH(C_TIME) | C_SH(C_SWTCH) | C_SH(C_DSWTCH) |
86 			    C_SH(C_SUSP) | C_SH(C_DSUSP) | C_SH(C_EOL) | C_SH(C_DISCARD) |
87 		    C_SH(C_PGOFF) | C_SH(C_PAGE) | C_SH(C_STATUS)), 0}
88 	},
89 	{
90 		{"iflag:", 0, IXON | IXOFF | INLCR | ICRNL},
91 		{"oflag:", 0, 0},
92 		{"cflag:", 0, 0},
93 		{"lflag:", 0, ISIG | IEXTEN},
94 		{"chars:", 0, 0},
95 	}
96 };
97 
98 static const ttychar_t ttychar = {
99 	{
100 		CINTR, CQUIT, CERASE, CKILL,
101 		CEOF, CEOL, CEOL2, CSWTCH,
102 		CDSWTCH, CERASE2, CSTART, CSTOP,
103 		CWERASE, CSUSP, CDSUSP, CREPRINT,
104 		CDISCARD, CLNEXT, CSTATUS, CPAGE,
105 		CPGOFF, CKILL2, CBRK, CMIN,
106 		CTIME
107 	},
108 	{
109 		CINTR, CQUIT, CERASE, CKILL,
110 		_POSIX_VDISABLE, _POSIX_VDISABLE, _POSIX_VDISABLE, _POSIX_VDISABLE,
111 		_POSIX_VDISABLE, CERASE2, CSTART, CSTOP,
112 		_POSIX_VDISABLE, CSUSP, _POSIX_VDISABLE, _POSIX_VDISABLE,
113 		CDISCARD, _POSIX_VDISABLE, _POSIX_VDISABLE, _POSIX_VDISABLE,
114 		_POSIX_VDISABLE, _POSIX_VDISABLE, _POSIX_VDISABLE, 1,
115 		0
116 	},
117 	{
118 		0, 0, 0, 0,
119 		0, 0, 0, 0,
120 		0, 0, 0, 0,
121 		0, 0, 0, 0,
122 		0, 0, 0, 0,
123 		0, 0, 0, 0,
124 		0
125 	}
126 };
127 
128 static const ttymap_t tty_map[] = {
129 #ifdef VERASE
130 	{C_ERASE, VERASE,
131 	{EM_DELETE_PREV_CHAR, VI_DELETE_PREV_CHAR, ED_PREV_CHAR}},
132 #endif /* VERASE */
133 #ifdef VERASE2
134 	{C_ERASE2, VERASE2,
135 	{EM_DELETE_PREV_CHAR, VI_DELETE_PREV_CHAR, ED_PREV_CHAR}},
136 #endif /* VERASE2 */
137 #ifdef VKILL
138 	{C_KILL, VKILL,
139 	{EM_KILL_LINE, VI_KILL_LINE_PREV, ED_UNASSIGNED}},
140 #endif /* VKILL */
141 #ifdef VKILL2
142 	{C_KILL2, VKILL2,
143 	{EM_KILL_LINE, VI_KILL_LINE_PREV, ED_UNASSIGNED}},
144 #endif /* VKILL2 */
145 #ifdef VEOF
146 	{C_EOF, VEOF,
147 	{EM_DELETE_OR_LIST, VI_LIST_OR_EOF, ED_UNASSIGNED}},
148 #endif /* VEOF */
149 #ifdef VWERASE
150 	{C_WERASE, VWERASE,
151 	{ED_DELETE_PREV_WORD, ED_DELETE_PREV_WORD, ED_PREV_WORD}},
152 #endif /* VWERASE */
153 #ifdef VREPRINT
154 	{C_REPRINT, VREPRINT,
155 	{ED_REDISPLAY, ED_INSERT, ED_REDISPLAY}},
156 #endif /* VREPRINT */
157 #ifdef VLNEXT
158 	{C_LNEXT, VLNEXT,
159 	{ED_QUOTED_INSERT, ED_QUOTED_INSERT, ED_UNASSIGNED}},
160 #endif /* VLNEXT */
161 	{(wint_t)-1, (wint_t)-1,
162 	{ED_UNASSIGNED, ED_UNASSIGNED, ED_UNASSIGNED}}
163 };
164 
165 static const ttymodes_t ttymodes[] = {
166 #ifdef	IGNBRK
167 	{"ignbrk", IGNBRK, MD_INP},
168 #endif /* IGNBRK */
169 #ifdef	BRKINT
170 	{"brkint", BRKINT, MD_INP},
171 #endif /* BRKINT */
172 #ifdef	IGNPAR
173 	{"ignpar", IGNPAR, MD_INP},
174 #endif /* IGNPAR */
175 #ifdef	PARMRK
176 	{"parmrk", PARMRK, MD_INP},
177 #endif /* PARMRK */
178 #ifdef	INPCK
179 	{"inpck", INPCK, MD_INP},
180 #endif /* INPCK */
181 #ifdef	ISTRIP
182 	{"istrip", ISTRIP, MD_INP},
183 #endif /* ISTRIP */
184 #ifdef	INLCR
185 	{"inlcr", INLCR, MD_INP},
186 #endif /* INLCR */
187 #ifdef	IGNCR
188 	{"igncr", IGNCR, MD_INP},
189 #endif /* IGNCR */
190 #ifdef	ICRNL
191 	{"icrnl", ICRNL, MD_INP},
192 #endif /* ICRNL */
193 #ifdef	IUCLC
194 	{"iuclc", IUCLC, MD_INP},
195 #endif /* IUCLC */
196 #ifdef	IXON
197 	{"ixon", IXON, MD_INP},
198 #endif /* IXON */
199 #ifdef	IXANY
200 	{"ixany", IXANY, MD_INP},
201 #endif /* IXANY */
202 #ifdef	IXOFF
203 	{"ixoff", IXOFF, MD_INP},
204 #endif /* IXOFF */
205 #ifdef  IMAXBEL
206 	{"imaxbel", IMAXBEL, MD_INP},
207 #endif /* IMAXBEL */
208 
209 #ifdef	OPOST
210 	{"opost", OPOST, MD_OUT},
211 #endif /* OPOST */
212 #ifdef	OLCUC
213 	{"olcuc", OLCUC, MD_OUT},
214 #endif /* OLCUC */
215 #ifdef	ONLCR
216 	{"onlcr", ONLCR, MD_OUT},
217 #endif /* ONLCR */
218 #ifdef	OCRNL
219 	{"ocrnl", OCRNL, MD_OUT},
220 #endif /* OCRNL */
221 #ifdef	ONOCR
222 	{"onocr", ONOCR, MD_OUT},
223 #endif /* ONOCR */
224 #ifdef ONOEOT
225 	{"onoeot", ONOEOT, MD_OUT},
226 #endif /* ONOEOT */
227 #ifdef	ONLRET
228 	{"onlret", ONLRET, MD_OUT},
229 #endif /* ONLRET */
230 #ifdef	OFILL
231 	{"ofill", OFILL, MD_OUT},
232 #endif /* OFILL */
233 #ifdef	OFDEL
234 	{"ofdel", OFDEL, MD_OUT},
235 #endif /* OFDEL */
236 #ifdef	NLDLY
237 	{"nldly", NLDLY, MD_OUT},
238 #endif /* NLDLY */
239 #ifdef	CRDLY
240 	{"crdly", CRDLY, MD_OUT},
241 #endif /* CRDLY */
242 #ifdef	TABDLY
243 	{"tabdly", TABDLY, MD_OUT},
244 #endif /* TABDLY */
245 #ifdef	XTABS
246 	{"xtabs", XTABS, MD_OUT},
247 #endif /* XTABS */
248 #ifdef	BSDLY
249 	{"bsdly", BSDLY, MD_OUT},
250 #endif /* BSDLY */
251 #ifdef	VTDLY
252 	{"vtdly", VTDLY, MD_OUT},
253 #endif /* VTDLY */
254 #ifdef	FFDLY
255 	{"ffdly", FFDLY, MD_OUT},
256 #endif /* FFDLY */
257 #ifdef	PAGEOUT
258 	{"pageout", PAGEOUT, MD_OUT},
259 #endif /* PAGEOUT */
260 #ifdef	WRAP
261 	{"wrap", WRAP, MD_OUT},
262 #endif /* WRAP */
263 
264 #ifdef	CIGNORE
265 	{"cignore", CIGNORE, MD_CTL},
266 #endif /* CBAUD */
267 #ifdef	CBAUD
268 	{"cbaud", CBAUD, MD_CTL},
269 #endif /* CBAUD */
270 #ifdef	CSTOPB
271 	{"cstopb", CSTOPB, MD_CTL},
272 #endif /* CSTOPB */
273 #ifdef	CREAD
274 	{"cread", CREAD, MD_CTL},
275 #endif /* CREAD */
276 #ifdef	PARENB
277 	{"parenb", PARENB, MD_CTL},
278 #endif /* PARENB */
279 #ifdef	PARODD
280 	{"parodd", PARODD, MD_CTL},
281 #endif /* PARODD */
282 #ifdef	HUPCL
283 	{"hupcl", HUPCL, MD_CTL},
284 #endif /* HUPCL */
285 #ifdef	CLOCAL
286 	{"clocal", CLOCAL, MD_CTL},
287 #endif /* CLOCAL */
288 #ifdef	LOBLK
289 	{"loblk", LOBLK, MD_CTL},
290 #endif /* LOBLK */
291 #ifdef	CIBAUD
292 	{"cibaud", CIBAUD, MD_CTL},
293 #endif /* CIBAUD */
294 #ifdef CRTSCTS
295 #ifdef CCTS_OFLOW
296 	{"ccts_oflow", CCTS_OFLOW, MD_CTL},
297 #else
298 	{"crtscts", CRTSCTS, MD_CTL},
299 #endif /* CCTS_OFLOW */
300 #endif /* CRTSCTS */
301 #ifdef CRTS_IFLOW
302 	{"crts_iflow", CRTS_IFLOW, MD_CTL},
303 #endif /* CRTS_IFLOW */
304 #ifdef CDTRCTS
305 	{"cdtrcts", CDTRCTS, MD_CTL},
306 #endif /* CDTRCTS */
307 #ifdef MDMBUF
308 	{"mdmbuf", MDMBUF, MD_CTL},
309 #endif /* MDMBUF */
310 #ifdef RCV1EN
311 	{"rcv1en", RCV1EN, MD_CTL},
312 #endif /* RCV1EN */
313 #ifdef XMT1EN
314 	{"xmt1en", XMT1EN, MD_CTL},
315 #endif /* XMT1EN */
316 
317 #ifdef	ISIG
318 	{"isig", ISIG, MD_LIN},
319 #endif /* ISIG */
320 #ifdef	ICANON
321 	{"icanon", ICANON, MD_LIN},
322 #endif /* ICANON */
323 #ifdef	XCASE
324 	{"xcase", XCASE, MD_LIN},
325 #endif /* XCASE */
326 #ifdef	ECHO
327 	{"echo", ECHO, MD_LIN},
328 #endif /* ECHO */
329 #ifdef	ECHOE
330 	{"echoe", ECHOE, MD_LIN},
331 #endif /* ECHOE */
332 #ifdef	ECHOK
333 	{"echok", ECHOK, MD_LIN},
334 #endif /* ECHOK */
335 #ifdef	ECHONL
336 	{"echonl", ECHONL, MD_LIN},
337 #endif /* ECHONL */
338 #ifdef	NOFLSH
339 	{"noflsh", NOFLSH, MD_LIN},
340 #endif /* NOFLSH */
341 #ifdef	TOSTOP
342 	{"tostop", TOSTOP, MD_LIN},
343 #endif /* TOSTOP */
344 #ifdef	ECHOCTL
345 	{"echoctl", ECHOCTL, MD_LIN},
346 #endif /* ECHOCTL */
347 #ifdef	ECHOPRT
348 	{"echoprt", ECHOPRT, MD_LIN},
349 #endif /* ECHOPRT */
350 #ifdef	ECHOKE
351 	{"echoke", ECHOKE, MD_LIN},
352 #endif /* ECHOKE */
353 #ifdef	DEFECHO
354 	{"defecho", DEFECHO, MD_LIN},
355 #endif /* DEFECHO */
356 #ifdef	FLUSHO
357 	{"flusho", FLUSHO, MD_LIN},
358 #endif /* FLUSHO */
359 #ifdef	PENDIN
360 	{"pendin", PENDIN, MD_LIN},
361 #endif /* PENDIN */
362 #ifdef	IEXTEN
363 	{"iexten", IEXTEN, MD_LIN},
364 #endif /* IEXTEN */
365 #ifdef	NOKERNINFO
366 	{"nokerninfo", NOKERNINFO, MD_LIN},
367 #endif /* NOKERNINFO */
368 #ifdef	ALTWERASE
369 	{"altwerase", ALTWERASE, MD_LIN},
370 #endif /* ALTWERASE */
371 #ifdef	EXTPROC
372 	{"extproc", EXTPROC, MD_LIN},
373 #endif /* EXTPROC */
374 
375 #if defined(VINTR)
376 	{"intr", C_SH(C_INTR), MD_CHAR},
377 #endif /* VINTR */
378 #if defined(VQUIT)
379 	{"quit", C_SH(C_QUIT), MD_CHAR},
380 #endif /* VQUIT */
381 #if defined(VERASE)
382 	{"erase", C_SH(C_ERASE), MD_CHAR},
383 #endif /* VERASE */
384 #if defined(VKILL)
385 	{"kill", C_SH(C_KILL), MD_CHAR},
386 #endif /* VKILL */
387 #if defined(VEOF)
388 	{"eof", C_SH(C_EOF), MD_CHAR},
389 #endif /* VEOF */
390 #if defined(VEOL)
391 	{"eol", C_SH(C_EOL), MD_CHAR},
392 #endif /* VEOL */
393 #if defined(VEOL2)
394 	{"eol2", C_SH(C_EOL2), MD_CHAR},
395 #endif /* VEOL2 */
396 #if defined(VSWTCH)
397 	{"swtch", C_SH(C_SWTCH), MD_CHAR},
398 #endif /* VSWTCH */
399 #if defined(VDSWTCH)
400 	{"dswtch", C_SH(C_DSWTCH), MD_CHAR},
401 #endif /* VDSWTCH */
402 #if defined(VERASE2)
403 	{"erase2", C_SH(C_ERASE2), MD_CHAR},
404 #endif /* VERASE2 */
405 #if defined(VSTART)
406 	{"start", C_SH(C_START), MD_CHAR},
407 #endif /* VSTART */
408 #if defined(VSTOP)
409 	{"stop", C_SH(C_STOP), MD_CHAR},
410 #endif /* VSTOP */
411 #if defined(VWERASE)
412 	{"werase", C_SH(C_WERASE), MD_CHAR},
413 #endif /* VWERASE */
414 #if defined(VSUSP)
415 	{"susp", C_SH(C_SUSP), MD_CHAR},
416 #endif /* VSUSP */
417 #if defined(VDSUSP)
418 	{"dsusp", C_SH(C_DSUSP), MD_CHAR},
419 #endif /* VDSUSP */
420 #if defined(VREPRINT)
421 	{"reprint", C_SH(C_REPRINT), MD_CHAR},
422 #endif /* VREPRINT */
423 #if defined(VDISCARD)
424 	{"discard", C_SH(C_DISCARD), MD_CHAR},
425 #endif /* VDISCARD */
426 #if defined(VLNEXT)
427 	{"lnext", C_SH(C_LNEXT), MD_CHAR},
428 #endif /* VLNEXT */
429 #if defined(VSTATUS)
430 	{"status", C_SH(C_STATUS), MD_CHAR},
431 #endif /* VSTATUS */
432 #if defined(VPAGE)
433 	{"page", C_SH(C_PAGE), MD_CHAR},
434 #endif /* VPAGE */
435 #if defined(VPGOFF)
436 	{"pgoff", C_SH(C_PGOFF), MD_CHAR},
437 #endif /* VPGOFF */
438 #if defined(VKILL2)
439 	{"kill2", C_SH(C_KILL2), MD_CHAR},
440 #endif /* VKILL2 */
441 #if defined(VBRK)
442 	{"brk", C_SH(C_BRK), MD_CHAR},
443 #endif /* VBRK */
444 #if defined(VMIN)
445 	{"min", C_SH(C_MIN), MD_CHAR},
446 #endif /* VMIN */
447 #if defined(VTIME)
448 	{"time", C_SH(C_TIME), MD_CHAR},
449 #endif /* VTIME */
450 	{NULL, 0, -1},
451 };
452 
453 
454 
455 #define	tty__gettabs(td)	((((td)->c_oflag & TAB3) == TAB3) ? 0 : 1)
456 #define	tty__geteightbit(td)	(((td)->c_cflag & CSIZE) == CS8)
457 #define	tty__cooked_mode(td)	((td)->c_lflag & ICANON)
458 
459 static int	tty_getty(EditLine *, struct termios *);
460 static int	tty_setty(EditLine *, int, const struct termios *);
461 static int	tty__getcharindex(int);
462 static void	tty__getchar(struct termios *, unsigned char *);
463 static void	tty__setchar(struct termios *, unsigned char *);
464 static speed_t	tty__getspeed(struct termios *);
465 static int	tty_setup(EditLine *);
466 static void	tty_setup_flags(EditLine *, struct termios *, int);
467 
468 #define	t_qu	t_ts
469 
470 /* tty_getty():
471  *	Wrapper for tcgetattr to handle EINTR
472  */
473 static int
474 tty_getty(EditLine *el, struct termios *t)
475 {
476 	int rv;
477 	while ((rv = tcgetattr(el->el_infd, t)) == -1 && errno == EINTR)
478 		continue;
479 	return rv;
480 }
481 
482 /* tty_setty():
483  *	Wrapper for tcsetattr to handle EINTR
484  */
485 static int
486 tty_setty(EditLine *el, int action, const struct termios *t)
487 {
488 	int rv;
489 	while ((rv = tcsetattr(el->el_infd, action, t)) == -1 && errno == EINTR)
490 		continue;
491 	return rv;
492 }
493 
494 /* tty_setup():
495  *	Get the tty parameters and initialize the editing state
496  */
497 static int
498 tty_setup(EditLine *el)
499 {
500 	int rst = (el->el_flags & NO_RESET) == 0;
501 
502 	if (el->el_flags & EDIT_DISABLED)
503 		return 0;
504 
505 	if (el->el_tty.t_initialized)
506 		return -1;
507 
508 	if (!isatty(el->el_outfd)) {
509 #ifdef DEBUG_TTY
510 		(void) fprintf(el->el_errfile, "%s: isatty: %s\n", __func__,
511 		    strerror(errno));
512 #endif /* DEBUG_TTY */
513 		return -1;
514 	}
515 	if (tty_getty(el, &el->el_tty.t_or) == -1) {
516 #ifdef DEBUG_TTY
517 		(void) fprintf(el->el_errfile, "%s: tty_getty: %s\n", __func__,
518 		    strerror(errno));
519 #endif /* DEBUG_TTY */
520 		return -1;
521 	}
522 	el->el_tty.t_ts = el->el_tty.t_ex = el->el_tty.t_ed = el->el_tty.t_or;
523 
524 	el->el_tty.t_speed = tty__getspeed(&el->el_tty.t_ex);
525 	el->el_tty.t_tabs = tty__gettabs(&el->el_tty.t_ex);
526 	el->el_tty.t_eight = tty__geteightbit(&el->el_tty.t_ex);
527 
528 	tty_setup_flags(el, &el->el_tty.t_ex, EX_IO);
529 
530 	/*
531          * Reset the tty chars to reasonable defaults
532          * If they are disabled, then enable them.
533          */
534 	if (rst) {
535 		if (tty__cooked_mode(&el->el_tty.t_ts)) {
536 			tty__getchar(&el->el_tty.t_ts, el->el_tty.t_c[TS_IO]);
537 			/*
538 	                 * Don't affect CMIN and CTIME for the editor mode
539 	                 */
540 			for (rst = 0; rst < C_NCC - 2; rst++)
541 				if (el->el_tty.t_c[TS_IO][rst] !=
542 				      el->el_tty.t_vdisable
543 				    && el->el_tty.t_c[ED_IO][rst] !=
544 				      el->el_tty.t_vdisable)
545 					el->el_tty.t_c[ED_IO][rst] =
546 					    el->el_tty.t_c[TS_IO][rst];
547 			for (rst = 0; rst < C_NCC; rst++)
548 				if (el->el_tty.t_c[TS_IO][rst] !=
549 				    el->el_tty.t_vdisable)
550 					el->el_tty.t_c[EX_IO][rst] =
551 					    el->el_tty.t_c[TS_IO][rst];
552 		}
553 		tty__setchar(&el->el_tty.t_ex, el->el_tty.t_c[EX_IO]);
554 		if (tty_setty(el, TCSADRAIN, &el->el_tty.t_ex) == -1) {
555 #ifdef DEBUG_TTY
556 			(void) fprintf(el->el_errfile, "%s: tty_setty: %s\n",
557 			    __func__, strerror(errno));
558 #endif /* DEBUG_TTY */
559 			return -1;
560 		}
561 	}
562 
563 	tty_setup_flags(el, &el->el_tty.t_ed, ED_IO);
564 
565 	tty__setchar(&el->el_tty.t_ed, el->el_tty.t_c[ED_IO]);
566 	tty_bind_char(el, 1);
567 	el->el_tty.t_initialized = 1;
568 	return 0;
569 }
570 
571 libedit_private int
572 tty_init(EditLine *el)
573 {
574 
575 	el->el_tty.t_mode = EX_IO;
576 	el->el_tty.t_vdisable = _POSIX_VDISABLE;
577 	el->el_tty.t_initialized = 0;
578 	(void) memcpy(el->el_tty.t_t, ttyperm, sizeof(ttyperm_t));
579 	(void) memcpy(el->el_tty.t_c, ttychar, sizeof(ttychar_t));
580 	return tty_setup(el);
581 }
582 
583 
584 /* tty_end():
585  *	Restore the tty to its original settings
586  */
587 libedit_private void
588 /*ARGSUSED*/
589 tty_end(EditLine *el, int how)
590 {
591 	if (el->el_flags & EDIT_DISABLED)
592 		return;
593 
594 	if (!el->el_tty.t_initialized)
595 		return;
596 
597 	if (tty_setty(el, how, &el->el_tty.t_or) == -1)
598 	{
599 #ifdef DEBUG_TTY
600 		(void) fprintf(el->el_errfile,
601 		    "%s: tty_setty: %s\n", __func__, strerror(errno));
602 #endif /* DEBUG_TTY */
603 	}
604 }
605 
606 
607 /* tty__getspeed():
608  *	Get the tty speed
609  */
610 static speed_t
611 tty__getspeed(struct termios *td)
612 {
613 	speed_t spd;
614 
615 	if ((spd = cfgetispeed(td)) == 0)
616 		spd = cfgetospeed(td);
617 	return spd;
618 }
619 
620 /* tty__getspeed():
621  *	Return the index of the asked char in the c_cc array
622  */
623 static int
624 tty__getcharindex(int i)
625 {
626 	switch (i) {
627 #ifdef VINTR
628 	case C_INTR:
629 		return VINTR;
630 #endif /* VINTR */
631 #ifdef VQUIT
632 	case C_QUIT:
633 		return VQUIT;
634 #endif /* VQUIT */
635 #ifdef VERASE
636 	case C_ERASE:
637 		return VERASE;
638 #endif /* VERASE */
639 #ifdef VKILL
640 	case C_KILL:
641 		return VKILL;
642 #endif /* VKILL */
643 #ifdef VEOF
644 	case C_EOF:
645 		return VEOF;
646 #endif /* VEOF */
647 #ifdef VEOL
648 	case C_EOL:
649 		return VEOL;
650 #endif /* VEOL */
651 #ifdef VEOL2
652 	case C_EOL2:
653 		return VEOL2;
654 #endif /* VEOL2 */
655 #ifdef VSWTCH
656 	case C_SWTCH:
657 		return VSWTCH;
658 #endif /* VSWTCH */
659 #ifdef VDSWTCH
660 	case C_DSWTCH:
661 		return VDSWTCH;
662 #endif /* VDSWTCH */
663 #ifdef VERASE2
664 	case C_ERASE2:
665 		return VERASE2;
666 #endif /* VERASE2 */
667 #ifdef VSTART
668 	case C_START:
669 		return VSTART;
670 #endif /* VSTART */
671 #ifdef VSTOP
672 	case C_STOP:
673 		return VSTOP;
674 #endif /* VSTOP */
675 #ifdef VWERASE
676 	case C_WERASE:
677 		return VWERASE;
678 #endif /* VWERASE */
679 #ifdef VSUSP
680 	case C_SUSP:
681 		return VSUSP;
682 #endif /* VSUSP */
683 #ifdef VDSUSP
684 	case C_DSUSP:
685 		return VDSUSP;
686 #endif /* VDSUSP */
687 #ifdef VREPRINT
688 	case C_REPRINT:
689 		return VREPRINT;
690 #endif /* VREPRINT */
691 #ifdef VDISCARD
692 	case C_DISCARD:
693 		return VDISCARD;
694 #endif /* VDISCARD */
695 #ifdef VLNEXT
696 	case C_LNEXT:
697 		return VLNEXT;
698 #endif /* VLNEXT */
699 #ifdef VSTATUS
700 	case C_STATUS:
701 		return VSTATUS;
702 #endif /* VSTATUS */
703 #ifdef VPAGE
704 	case C_PAGE:
705 		return VPAGE;
706 #endif /* VPAGE */
707 #ifdef VPGOFF
708 	case C_PGOFF:
709 		return VPGOFF;
710 #endif /* VPGOFF */
711 #ifdef VKILL2
712 	case C_KILL2:
713 		return VKILL2;
714 #endif /* KILL2 */
715 #ifdef VMIN
716 	case C_MIN:
717 		return VMIN;
718 #endif /* VMIN */
719 #ifdef VTIME
720 	case C_TIME:
721 		return VTIME;
722 #endif /* VTIME */
723 	default:
724 		return -1;
725 	}
726 }
727 
728 /* tty__getchar():
729  *	Get the tty characters
730  */
731 static void
732 tty__getchar(struct termios *td, unsigned char *s)
733 {
734 
735 #ifdef VINTR
736 	s[C_INTR] = td->c_cc[VINTR];
737 #endif /* VINTR */
738 #ifdef VQUIT
739 	s[C_QUIT] = td->c_cc[VQUIT];
740 #endif /* VQUIT */
741 #ifdef VERASE
742 	s[C_ERASE] = td->c_cc[VERASE];
743 #endif /* VERASE */
744 #ifdef VKILL
745 	s[C_KILL] = td->c_cc[VKILL];
746 #endif /* VKILL */
747 #ifdef VEOF
748 	s[C_EOF] = td->c_cc[VEOF];
749 #endif /* VEOF */
750 #ifdef VEOL
751 	s[C_EOL] = td->c_cc[VEOL];
752 #endif /* VEOL */
753 #ifdef VEOL2
754 	s[C_EOL2] = td->c_cc[VEOL2];
755 #endif /* VEOL2 */
756 #ifdef VSWTCH
757 	s[C_SWTCH] = td->c_cc[VSWTCH];
758 #endif /* VSWTCH */
759 #ifdef VDSWTCH
760 	s[C_DSWTCH] = td->c_cc[VDSWTCH];
761 #endif /* VDSWTCH */
762 #ifdef VERASE2
763 	s[C_ERASE2] = td->c_cc[VERASE2];
764 #endif /* VERASE2 */
765 #ifdef VSTART
766 	s[C_START] = td->c_cc[VSTART];
767 #endif /* VSTART */
768 #ifdef VSTOP
769 	s[C_STOP] = td->c_cc[VSTOP];
770 #endif /* VSTOP */
771 #ifdef VWERASE
772 	s[C_WERASE] = td->c_cc[VWERASE];
773 #endif /* VWERASE */
774 #ifdef VSUSP
775 	s[C_SUSP] = td->c_cc[VSUSP];
776 #endif /* VSUSP */
777 #ifdef VDSUSP
778 	s[C_DSUSP] = td->c_cc[VDSUSP];
779 #endif /* VDSUSP */
780 #ifdef VREPRINT
781 	s[C_REPRINT] = td->c_cc[VREPRINT];
782 #endif /* VREPRINT */
783 #ifdef VDISCARD
784 	s[C_DISCARD] = td->c_cc[VDISCARD];
785 #endif /* VDISCARD */
786 #ifdef VLNEXT
787 	s[C_LNEXT] = td->c_cc[VLNEXT];
788 #endif /* VLNEXT */
789 #ifdef VSTATUS
790 	s[C_STATUS] = td->c_cc[VSTATUS];
791 #endif /* VSTATUS */
792 #ifdef VPAGE
793 	s[C_PAGE] = td->c_cc[VPAGE];
794 #endif /* VPAGE */
795 #ifdef VPGOFF
796 	s[C_PGOFF] = td->c_cc[VPGOFF];
797 #endif /* VPGOFF */
798 #ifdef VKILL2
799 	s[C_KILL2] = td->c_cc[VKILL2];
800 #endif /* KILL2 */
801 #ifdef VMIN
802 	s[C_MIN] = td->c_cc[VMIN];
803 #endif /* VMIN */
804 #ifdef VTIME
805 	s[C_TIME] = td->c_cc[VTIME];
806 #endif /* VTIME */
807 }				/* tty__getchar */
808 
809 
810 /* tty__setchar():
811  *	Set the tty characters
812  */
813 static void
814 tty__setchar(struct termios *td, unsigned char *s)
815 {
816 
817 #ifdef VINTR
818 	td->c_cc[VINTR] = s[C_INTR];
819 #endif /* VINTR */
820 #ifdef VQUIT
821 	td->c_cc[VQUIT] = s[C_QUIT];
822 #endif /* VQUIT */
823 #ifdef VERASE
824 	td->c_cc[VERASE] = s[C_ERASE];
825 #endif /* VERASE */
826 #ifdef VKILL
827 	td->c_cc[VKILL] = s[C_KILL];
828 #endif /* VKILL */
829 #ifdef VEOF
830 	td->c_cc[VEOF] = s[C_EOF];
831 #endif /* VEOF */
832 #ifdef VEOL
833 	td->c_cc[VEOL] = s[C_EOL];
834 #endif /* VEOL */
835 #ifdef VEOL2
836 	td->c_cc[VEOL2] = s[C_EOL2];
837 #endif /* VEOL2 */
838 #ifdef VSWTCH
839 	td->c_cc[VSWTCH] = s[C_SWTCH];
840 #endif /* VSWTCH */
841 #ifdef VDSWTCH
842 	td->c_cc[VDSWTCH] = s[C_DSWTCH];
843 #endif /* VDSWTCH */
844 #ifdef VERASE2
845 	td->c_cc[VERASE2] = s[C_ERASE2];
846 #endif /* VERASE2 */
847 #ifdef VSTART
848 	td->c_cc[VSTART] = s[C_START];
849 #endif /* VSTART */
850 #ifdef VSTOP
851 	td->c_cc[VSTOP] = s[C_STOP];
852 #endif /* VSTOP */
853 #ifdef VWERASE
854 	td->c_cc[VWERASE] = s[C_WERASE];
855 #endif /* VWERASE */
856 #ifdef VSUSP
857 	td->c_cc[VSUSP] = s[C_SUSP];
858 #endif /* VSUSP */
859 #ifdef VDSUSP
860 	td->c_cc[VDSUSP] = s[C_DSUSP];
861 #endif /* VDSUSP */
862 #ifdef VREPRINT
863 	td->c_cc[VREPRINT] = s[C_REPRINT];
864 #endif /* VREPRINT */
865 #ifdef VDISCARD
866 	td->c_cc[VDISCARD] = s[C_DISCARD];
867 #endif /* VDISCARD */
868 #ifdef VLNEXT
869 	td->c_cc[VLNEXT] = s[C_LNEXT];
870 #endif /* VLNEXT */
871 #ifdef VSTATUS
872 	td->c_cc[VSTATUS] = s[C_STATUS];
873 #endif /* VSTATUS */
874 #ifdef VPAGE
875 	td->c_cc[VPAGE] = s[C_PAGE];
876 #endif /* VPAGE */
877 #ifdef VPGOFF
878 	td->c_cc[VPGOFF] = s[C_PGOFF];
879 #endif /* VPGOFF */
880 #ifdef VKILL2
881 	td->c_cc[VKILL2] = s[C_KILL2];
882 #endif /* VKILL2 */
883 #ifdef VMIN
884 	td->c_cc[VMIN] = s[C_MIN];
885 #endif /* VMIN */
886 #ifdef VTIME
887 	td->c_cc[VTIME] = s[C_TIME];
888 #endif /* VTIME */
889 }				/* tty__setchar */
890 
891 
892 /* tty_bind_char():
893  *	Rebind the editline functions
894  */
895 libedit_private void
896 tty_bind_char(EditLine *el, int force)
897 {
898 
899 	unsigned char *t_n = el->el_tty.t_c[ED_IO];
900 	unsigned char *t_o = el->el_tty.t_ed.c_cc;
901 	wchar_t new[2], old[2];
902 	const ttymap_t *tp;
903 	el_action_t *map, *alt;
904 	const el_action_t *dmap, *dalt;
905 	new[1] = old[1] = '\0';
906 
907 	map = el->el_map.key;
908 	alt = el->el_map.alt;
909 	if (el->el_map.type == MAP_VI) {
910 		dmap = el->el_map.vii;
911 		dalt = el->el_map.vic;
912 	} else {
913 		dmap = el->el_map.emacs;
914 		dalt = NULL;
915 	}
916 
917 	for (tp = tty_map; tp->nch != (wint_t)-1; tp++) {
918 		new[0] = (wchar_t)t_n[tp->nch];
919 		old[0] = (wchar_t)t_o[tp->och];
920 		if (new[0] == old[0] && !force)
921 			continue;
922 		/* Put the old default binding back, and set the new binding */
923 		keymacro_clear(el, map, old);
924 		map[(unsigned char)old[0]] = dmap[(unsigned char)old[0]];
925 		keymacro_clear(el, map, new);
926 		/* MAP_VI == 1, MAP_EMACS == 0... */
927 		map[(unsigned char)new[0]] = tp->bind[el->el_map.type];
928 		if (dalt) {
929 			keymacro_clear(el, alt, old);
930 			alt[(unsigned char)old[0]] =
931 			    dalt[(unsigned char)old[0]];
932 			keymacro_clear(el, alt, new);
933 			alt[(unsigned char)new[0]] =
934 			    tp->bind[el->el_map.type + 1];
935 		}
936 	}
937 }
938 
939 
940 static tcflag_t *
941 tty__get_flag(struct termios *t, int kind) {
942 	switch (kind) {
943 	case MD_INP:
944 		return &t->c_iflag;
945 	case MD_OUT:
946 		return &t->c_oflag;
947 	case MD_CTL:
948 		return &t->c_cflag;
949 	case MD_LIN:
950 		return &t->c_lflag;
951 	default:
952 		abort();
953 		/*NOTREACHED*/
954 	}
955 }
956 
957 
958 static tcflag_t
959 tty_update_flag(EditLine *el, tcflag_t f, int mode, int kind)
960 {
961 	f &= ~el->el_tty.t_t[mode][kind].t_clrmask;
962 	f |= el->el_tty.t_t[mode][kind].t_setmask;
963 	return f;
964 }
965 
966 
967 static void
968 tty_update_flags(EditLine *el, int kind)
969 {
970 	tcflag_t *tt, *ed, *ex;
971 	tt = tty__get_flag(&el->el_tty.t_ts, kind);
972 	ed = tty__get_flag(&el->el_tty.t_ed, kind);
973 	ex = tty__get_flag(&el->el_tty.t_ex, kind);
974 
975 	if (*tt != *ex && (kind != MD_CTL || *tt != *ed)) {
976 		*ed = tty_update_flag(el, *tt, ED_IO, kind);
977 		*ex = tty_update_flag(el, *tt, EX_IO, kind);
978 	}
979 }
980 
981 
982 static void
983 tty_update_char(EditLine *el, int mode, int c) {
984 	if (!((el->el_tty.t_t[mode][MD_CHAR].t_setmask & C_SH(c)))
985 	    && (el->el_tty.t_c[TS_IO][c] != el->el_tty.t_c[EX_IO][c]))
986 		el->el_tty.t_c[mode][c] = el->el_tty.t_c[TS_IO][c];
987 	if (el->el_tty.t_t[mode][MD_CHAR].t_clrmask & C_SH(c))
988 		el->el_tty.t_c[mode][c] = el->el_tty.t_vdisable;
989 }
990 
991 
992 /* tty_rawmode():
993  *	Set terminal into 1 character at a time mode.
994  */
995 libedit_private int
996 tty_rawmode(EditLine *el)
997 {
998 
999 	if (el->el_tty.t_mode == ED_IO || el->el_tty.t_mode == QU_IO)
1000 		return 0;
1001 
1002 	if (el->el_flags & EDIT_DISABLED)
1003 		return 0;
1004 
1005 	if (tty_getty(el, &el->el_tty.t_ts) == -1) {
1006 #ifdef DEBUG_TTY
1007 		(void) fprintf(el->el_errfile, "%s: tty_getty: %s\n", __func__,
1008 		    strerror(errno));
1009 #endif /* DEBUG_TTY */
1010 		return -1;
1011 	}
1012 	/*
1013          * We always keep up with the eight bit setting and the speed of the
1014          * tty. But we only believe changes that are made to cooked mode!
1015          */
1016 	el->el_tty.t_eight = tty__geteightbit(&el->el_tty.t_ts);
1017 	el->el_tty.t_speed = tty__getspeed(&el->el_tty.t_ts);
1018 
1019 	if (tty__getspeed(&el->el_tty.t_ex) != el->el_tty.t_speed ||
1020 	    tty__getspeed(&el->el_tty.t_ed) != el->el_tty.t_speed) {
1021 		(void) cfsetispeed(&el->el_tty.t_ex, el->el_tty.t_speed);
1022 		(void) cfsetospeed(&el->el_tty.t_ex, el->el_tty.t_speed);
1023 		(void) cfsetispeed(&el->el_tty.t_ed, el->el_tty.t_speed);
1024 		(void) cfsetospeed(&el->el_tty.t_ed, el->el_tty.t_speed);
1025 	}
1026 	if (tty__cooked_mode(&el->el_tty.t_ts)) {
1027 		int i;
1028 
1029 		for (i = MD_INP; i <= MD_LIN; i++)
1030 			tty_update_flags(el, i);
1031 
1032 		if (tty__gettabs(&el->el_tty.t_ex) == 0)
1033 			el->el_tty.t_tabs = 0;
1034 		else
1035 			el->el_tty.t_tabs = EL_CAN_TAB ? 1 : 0;
1036 
1037 		tty__getchar(&el->el_tty.t_ts, el->el_tty.t_c[TS_IO]);
1038 		/*
1039 		 * Check if the user made any changes.
1040 		 * If he did, then propagate the changes to the
1041 		 * edit and execute data structures.
1042 		 */
1043 		for (i = 0; i < C_NCC; i++)
1044 			if (el->el_tty.t_c[TS_IO][i] !=
1045 			    el->el_tty.t_c[EX_IO][i])
1046 				break;
1047 
1048 		if (i != C_NCC) {
1049 			/*
1050 			 * Propagate changes only to the unlibedit_private
1051 			 * chars that have been modified just now.
1052 			 */
1053 			for (i = 0; i < C_NCC; i++)
1054 				tty_update_char(el, ED_IO, i);
1055 
1056 			tty_bind_char(el, 0);
1057 			tty__setchar(&el->el_tty.t_ed, el->el_tty.t_c[ED_IO]);
1058 
1059 			for (i = 0; i < C_NCC; i++)
1060 				tty_update_char(el, EX_IO, i);
1061 
1062 			tty__setchar(&el->el_tty.t_ex, el->el_tty.t_c[EX_IO]);
1063 		}
1064 	}
1065 	if (tty_setty(el, TCSADRAIN, &el->el_tty.t_ed) == -1) {
1066 #ifdef DEBUG_TTY
1067 		(void) fprintf(el->el_errfile, "%s: tty_setty: %s\n", __func__,
1068 		    strerror(errno));
1069 #endif /* DEBUG_TTY */
1070 		return -1;
1071 	}
1072 	el->el_tty.t_mode = ED_IO;
1073 	return 0;
1074 }
1075 
1076 
1077 /* tty_cookedmode():
1078  *	Set the tty back to normal mode
1079  */
1080 libedit_private int
1081 tty_cookedmode(EditLine *el)
1082 {				/* set tty in normal setup */
1083 
1084 	if (el->el_tty.t_mode == EX_IO)
1085 		return 0;
1086 
1087 	if (el->el_flags & EDIT_DISABLED)
1088 		return 0;
1089 
1090 	if (tty_setty(el, TCSADRAIN, &el->el_tty.t_ex) == -1) {
1091 #ifdef DEBUG_TTY
1092 		(void) fprintf(el->el_errfile, "%s: tty_setty: %s\n", __func__,
1093 		    strerror(errno));
1094 #endif /* DEBUG_TTY */
1095 		return -1;
1096 	}
1097 	el->el_tty.t_mode = EX_IO;
1098 	return 0;
1099 }
1100 
1101 
1102 /* tty_quotemode():
1103  *	Turn on quote mode
1104  */
1105 libedit_private int
1106 tty_quotemode(EditLine *el)
1107 {
1108 	if (el->el_tty.t_mode == QU_IO)
1109 		return 0;
1110 
1111 	el->el_tty.t_qu = el->el_tty.t_ed;
1112 
1113 	tty_setup_flags(el, &el->el_tty.t_qu, QU_IO);
1114 
1115 	if (tty_setty(el, TCSADRAIN, &el->el_tty.t_qu) == -1) {
1116 #ifdef DEBUG_TTY
1117 		(void) fprintf(el->el_errfile, "%s: tty_setty: %s\n", __func__,
1118 		    strerror(errno));
1119 #endif /* DEBUG_TTY */
1120 		return -1;
1121 	}
1122 	el->el_tty.t_mode = QU_IO;
1123 	return 0;
1124 }
1125 
1126 
1127 /* tty_noquotemode():
1128  *	Turn off quote mode
1129  */
1130 libedit_private int
1131 tty_noquotemode(EditLine *el)
1132 {
1133 
1134 	if (el->el_tty.t_mode != QU_IO)
1135 		return 0;
1136 	if (tty_setty(el, TCSADRAIN, &el->el_tty.t_ed) == -1) {
1137 #ifdef DEBUG_TTY
1138 		(void) fprintf(el->el_errfile, "%s: tty_setty: %s\n", __func__,
1139 		    strerror(errno));
1140 #endif /* DEBUG_TTY */
1141 		return -1;
1142 	}
1143 	el->el_tty.t_mode = ED_IO;
1144 	return 0;
1145 }
1146 
1147 
1148 /* tty_stty():
1149  *	Stty builtin
1150  */
1151 libedit_private int
1152 /*ARGSUSED*/
1153 tty_stty(EditLine *el, int argc __attribute__((__unused__)),
1154     const wchar_t **argv)
1155 {
1156 	const ttymodes_t *m;
1157 	char x;
1158 	int aflag = 0;
1159 	const wchar_t *s, *d;
1160         char name[EL_BUFSIZ];
1161 	struct termios *tios = &el->el_tty.t_ex;
1162 	int z = EX_IO;
1163 
1164 	if (argv == NULL)
1165 		return -1;
1166 	strncpy(name, ct_encode_string(*argv++, &el->el_scratch), sizeof(name));
1167         name[sizeof(name) - 1] = '\0';
1168 
1169 	while (argv && *argv && argv[0][0] == '-' && argv[0][2] == '\0')
1170 		switch (argv[0][1]) {
1171 		case 'a':
1172 			aflag++;
1173 			argv++;
1174 			break;
1175 		case 'd':
1176 			argv++;
1177 			tios = &el->el_tty.t_ed;
1178 			z = ED_IO;
1179 			break;
1180 		case 'x':
1181 			argv++;
1182 			tios = &el->el_tty.t_ex;
1183 			z = EX_IO;
1184 			break;
1185 		case 'q':
1186 			argv++;
1187 			tios = &el->el_tty.t_ts;
1188 			z = QU_IO;
1189 			break;
1190 		default:
1191 			(void) fprintf(el->el_errfile,
1192 			    "%s: Unknown switch `%lc'.\n",
1193 			    name, (wint_t)argv[0][1]);
1194 			return -1;
1195 		}
1196 
1197 	if (!argv || !*argv) {
1198 		int i = -1;
1199 		size_t len = 0, st = 0, cu;
1200 		for (m = ttymodes; m->m_name; m++) {
1201 			if (m->m_type != i) {
1202 				(void) fprintf(el->el_outfile, "%s%s",
1203 				    i != -1 ? "\n" : "",
1204 				    el->el_tty.t_t[z][m->m_type].t_name);
1205 				i = m->m_type;
1206 				st = len =
1207 				    strlen(el->el_tty.t_t[z][m->m_type].t_name);
1208 			}
1209 			if (i != -1) {
1210 			    x = (el->el_tty.t_t[z][i].t_setmask & m->m_value)
1211 				?  '+' : '\0';
1212 
1213 			    if (el->el_tty.t_t[z][i].t_clrmask & m->m_value)
1214 				x = '-';
1215 			} else {
1216 			    x = '\0';
1217 			}
1218 
1219 			if (x != '\0' || aflag) {
1220 
1221 				cu = strlen(m->m_name) + (x != '\0') + 1;
1222 
1223 				if (len + cu >=
1224 				    (size_t)el->el_terminal.t_size.h) {
1225 					(void) fprintf(el->el_outfile, "\n%*s",
1226 					    (int)st, "");
1227 					len = st + cu;
1228 				} else
1229 					len += cu;
1230 
1231 				if (x != '\0')
1232 					(void) fprintf(el->el_outfile, "%c%s ",
1233 					    x, m->m_name);
1234 				else
1235 					(void) fprintf(el->el_outfile, "%s ",
1236 					    m->m_name);
1237 			}
1238 		}
1239 		(void) fprintf(el->el_outfile, "\n");
1240 		return 0;
1241 	}
1242 	while (argv && (s = *argv++)) {
1243 		const wchar_t *p;
1244 		switch (*s) {
1245 		case '+':
1246 		case '-':
1247 			x = (char)*s++;
1248 			break;
1249 		default:
1250 			x = '\0';
1251 			break;
1252 		}
1253 		d = s;
1254 		p = wcschr(s, L'=');
1255 		for (m = ttymodes; m->m_name; m++)
1256 			if ((p ? strncmp(m->m_name, ct_encode_string(d,
1257 			    &el->el_scratch), (size_t)(p - d)) :
1258 			    strcmp(m->m_name, ct_encode_string(d,
1259 			    &el->el_scratch))) == 0 &&
1260 			    (p == NULL || m->m_type == MD_CHAR))
1261 				break;
1262 
1263 		if (!m->m_name) {
1264 			(void) fprintf(el->el_errfile,
1265 			    "%s: Invalid argument `%ls'.\n", name, d);
1266 			return -1;
1267 		}
1268 		if (p) {
1269 			int c = ffs((int)m->m_value);
1270 			int v = *++p ? parse__escape(&p) :
1271 			    el->el_tty.t_vdisable;
1272 			assert(c != 0);
1273 			c--;
1274 			c = tty__getcharindex(c);
1275 			assert(c != -1);
1276 			tios->c_cc[c] = (cc_t)v;
1277 			continue;
1278 		}
1279 		switch (x) {
1280 		case '+':
1281 			el->el_tty.t_t[z][m->m_type].t_setmask |= m->m_value;
1282 			el->el_tty.t_t[z][m->m_type].t_clrmask &= ~m->m_value;
1283 			break;
1284 		case '-':
1285 			el->el_tty.t_t[z][m->m_type].t_setmask &= ~m->m_value;
1286 			el->el_tty.t_t[z][m->m_type].t_clrmask |= m->m_value;
1287 			break;
1288 		default:
1289 			el->el_tty.t_t[z][m->m_type].t_setmask &= ~m->m_value;
1290 			el->el_tty.t_t[z][m->m_type].t_clrmask &= ~m->m_value;
1291 			break;
1292 		}
1293 	}
1294 
1295 	tty_setup_flags(el, tios, z);
1296 	if (el->el_tty.t_mode == z) {
1297 		if (tty_setty(el, TCSADRAIN, tios) == -1) {
1298 #ifdef DEBUG_TTY
1299 			(void) fprintf(el->el_errfile, "%s: tty_setty: %s\n",
1300 			    __func__, strerror(errno));
1301 #endif /* DEBUG_TTY */
1302 			return -1;
1303 		}
1304 	}
1305 
1306 	return 0;
1307 }
1308 
1309 
1310 #ifdef notyet
1311 /* tty_printchar():
1312  *	DEbugging routine to print the tty characters
1313  */
1314 static void
1315 tty_printchar(EditLine *el, unsigned char *s)
1316 {
1317 	ttyperm_t *m;
1318 	int i;
1319 
1320 	for (i = 0; i < C_NCC; i++) {
1321 		for (m = el->el_tty.t_t; m->m_name; m++)
1322 			if (m->m_type == MD_CHAR && C_SH(i) == m->m_value)
1323 				break;
1324 		if (m->m_name)
1325 			(void) fprintf(el->el_errfile, "%s ^%c ",
1326 			    m->m_name, s[i] + 'A' - 1);
1327 		if (i % 5 == 0)
1328 			(void) fprintf(el->el_errfile, "\n");
1329 	}
1330 	(void) fprintf(el->el_errfile, "\n");
1331 }
1332 #endif /* notyet */
1333 
1334 
1335 static void
1336 tty_setup_flags(EditLine *el, struct termios *tios, int mode)
1337 {
1338 	int kind;
1339 	for (kind = MD_INP; kind <= MD_LIN; kind++) {
1340 		tcflag_t *f = tty__get_flag(tios, kind);
1341 		*f = tty_update_flag(el, *f, mode, kind);
1342 	}
1343 }
1344 
1345 libedit_private int
1346 tty_get_signal_character(EditLine *el, int sig)
1347 {
1348 #ifdef ECHOCTL
1349 	tcflag_t *ed = tty__get_flag(&el->el_tty.t_ed, MD_INP);
1350 	if ((*ed & ECHOCTL) == 0)
1351 		return -1;
1352 #endif
1353 	switch (sig) {
1354 #ifdef SIGINT
1355 	case SIGINT:
1356 		return el->el_tty.t_c[ED_IO][VINTR];
1357 #endif
1358 #ifdef SIGQUIT
1359 	case SIGQUIT:
1360 		return el->el_tty.t_c[ED_IO][VQUIT];
1361 #endif
1362 #ifdef SIGINFO
1363 	case SIGINFO:
1364 		return el->el_tty.t_c[ED_IO][VSTATUS];
1365 #endif
1366 #ifdef SIGTSTP
1367 	case SIGTSTP:
1368 		return el->el_tty.t_c[ED_IO][VSUSP];
1369 #endif
1370 	default:
1371 		return -1;
1372 	}
1373 }
1374