Home | History | Annotate | Line # | Download | only in libedit
tty.c revision 1.41
      1 /*	$NetBSD: tty.c,v 1.41 2011/10/04 15:27:04 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.41 2011/10/04 15:27:04 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 <unistd.h>	/* for isatty */
     50 #include <strings.h>	/* for ffs */
     51 #include "el.h"
     52 #include "tty.h"
     53 
     54 typedef struct ttymodes_t {
     55 	const char *m_name;
     56 	unsigned int m_value;
     57 	int m_type;
     58 }          ttymodes_t;
     59 
     60 typedef struct ttymap_t {
     61 	Int nch, och;		/* Internal and termio rep of chars */
     62 	el_action_t bind[3];	/* emacs, vi, and vi-cmd */
     63 } ttymap_t;
     64 
     65 
     66 private const ttyperm_t ttyperm = {
     67 	{
     68 		{"iflag:", ICRNL, (INLCR | IGNCR)},
     69 		{"oflag:", (OPOST | ONLCR), ONLRET},
     70 		{"cflag:", 0, 0},
     71 		{"lflag:", (ISIG | ICANON | ECHO | ECHOE | ECHOCTL | IEXTEN),
     72 		(NOFLSH | ECHONL | EXTPROC | FLUSHO)},
     73 		{"chars:", 0, 0},
     74 	},
     75 	{
     76 		{"iflag:", (INLCR | ICRNL), IGNCR},
     77 		{"oflag:", (OPOST | ONLCR), ONLRET},
     78 		{"cflag:", 0, 0},
     79 		{"lflag:", ISIG,
     80 		(NOFLSH | ICANON | ECHO | ECHOK | ECHONL | EXTPROC | IEXTEN | FLUSHO)},
     81 		{"chars:", (C_SH(C_MIN) | C_SH(C_TIME) | C_SH(C_SWTCH) | C_SH(C_DSWTCH) |
     82 			    C_SH(C_SUSP) | C_SH(C_DSUSP) | C_SH(C_EOL) | C_SH(C_DISCARD) |
     83 		    C_SH(C_PGOFF) | C_SH(C_PAGE) | C_SH(C_STATUS)), 0}
     84 	},
     85 	{
     86 		{"iflag:", 0, IXON | IXOFF | INLCR | ICRNL},
     87 		{"oflag:", 0, 0},
     88 		{"cflag:", 0, 0},
     89 		{"lflag:", 0, ISIG | IEXTEN},
     90 		{"chars:", 0, 0},
     91 	}
     92 };
     93 
     94 private const ttychar_t ttychar = {
     95 	{
     96 		CINTR, CQUIT, CERASE, CKILL,
     97 		CEOF, CEOL, CEOL2, CSWTCH,
     98 		CDSWTCH, CERASE2, CSTART, CSTOP,
     99 		CWERASE, CSUSP, CDSUSP, CREPRINT,
    100 		CDISCARD, CLNEXT, CSTATUS, CPAGE,
    101 		CPGOFF, CKILL2, CBRK, CMIN,
    102 		CTIME
    103 	},
    104 	{
    105 		CINTR, CQUIT, CERASE, CKILL,
    106 		_POSIX_VDISABLE, _POSIX_VDISABLE, _POSIX_VDISABLE, _POSIX_VDISABLE,
    107 		_POSIX_VDISABLE, CERASE2, CSTART, CSTOP,
    108 		_POSIX_VDISABLE, CSUSP, _POSIX_VDISABLE, _POSIX_VDISABLE,
    109 		CDISCARD, _POSIX_VDISABLE, _POSIX_VDISABLE, _POSIX_VDISABLE,
    110 		_POSIX_VDISABLE, _POSIX_VDISABLE, _POSIX_VDISABLE, 1,
    111 		0
    112 	},
    113 	{
    114 		0, 0, 0, 0,
    115 		0, 0, 0, 0,
    116 		0, 0, 0, 0,
    117 		0, 0, 0, 0,
    118 		0, 0, 0, 0,
    119 		0, 0, 0, 0,
    120 		0
    121 	}
    122 };
    123 
    124 private const ttymap_t tty_map[] = {
    125 #ifdef VERASE
    126 	{C_ERASE, VERASE,
    127 	{EM_DELETE_PREV_CHAR, VI_DELETE_PREV_CHAR, ED_PREV_CHAR}},
    128 #endif /* VERASE */
    129 #ifdef VERASE2
    130 	{C_ERASE2, VERASE2,
    131 	{EM_DELETE_PREV_CHAR, VI_DELETE_PREV_CHAR, ED_PREV_CHAR}},
    132 #endif /* VERASE2 */
    133 #ifdef VKILL
    134 	{C_KILL, VKILL,
    135 	{EM_KILL_LINE, VI_KILL_LINE_PREV, ED_UNASSIGNED}},
    136 #endif /* VKILL */
    137 #ifdef VKILL2
    138 	{C_KILL2, VKILL2,
    139 	{EM_KILL_LINE, VI_KILL_LINE_PREV, ED_UNASSIGNED}},
    140 #endif /* VKILL2 */
    141 #ifdef VEOF
    142 	{C_EOF, VEOF,
    143 	{EM_DELETE_OR_LIST, VI_LIST_OR_EOF, ED_UNASSIGNED}},
    144 #endif /* VEOF */
    145 #ifdef VWERASE
    146 	{C_WERASE, VWERASE,
    147 	{ED_DELETE_PREV_WORD, ED_DELETE_PREV_WORD, ED_PREV_WORD}},
    148 #endif /* VWERASE */
    149 #ifdef VREPRINT
    150 	{C_REPRINT, VREPRINT,
    151 	{ED_REDISPLAY, ED_INSERT, ED_REDISPLAY}},
    152 #endif /* VREPRINT */
    153 #ifdef VLNEXT
    154 	{C_LNEXT, VLNEXT,
    155 	{ED_QUOTED_INSERT, ED_QUOTED_INSERT, ED_UNASSIGNED}},
    156 #endif /* VLNEXT */
    157 	{(Int)-1, (Int)-1,
    158 	{ED_UNASSIGNED, ED_UNASSIGNED, ED_UNASSIGNED}}
    159 };
    160 
    161 private const ttymodes_t ttymodes[] = {
    162 #ifdef	IGNBRK
    163 	{"ignbrk", IGNBRK, MD_INP},
    164 #endif /* IGNBRK */
    165 #ifdef	BRKINT
    166 	{"brkint", BRKINT, MD_INP},
    167 #endif /* BRKINT */
    168 #ifdef	IGNPAR
    169 	{"ignpar", IGNPAR, MD_INP},
    170 #endif /* IGNPAR */
    171 #ifdef	PARMRK
    172 	{"parmrk", PARMRK, MD_INP},
    173 #endif /* PARMRK */
    174 #ifdef	INPCK
    175 	{"inpck", INPCK, MD_INP},
    176 #endif /* INPCK */
    177 #ifdef	ISTRIP
    178 	{"istrip", ISTRIP, MD_INP},
    179 #endif /* ISTRIP */
    180 #ifdef	INLCR
    181 	{"inlcr", INLCR, MD_INP},
    182 #endif /* INLCR */
    183 #ifdef	IGNCR
    184 	{"igncr", IGNCR, MD_INP},
    185 #endif /* IGNCR */
    186 #ifdef	ICRNL
    187 	{"icrnl", ICRNL, MD_INP},
    188 #endif /* ICRNL */
    189 #ifdef	IUCLC
    190 	{"iuclc", IUCLC, MD_INP},
    191 #endif /* IUCLC */
    192 #ifdef	IXON
    193 	{"ixon", IXON, MD_INP},
    194 #endif /* IXON */
    195 #ifdef	IXANY
    196 	{"ixany", IXANY, MD_INP},
    197 #endif /* IXANY */
    198 #ifdef	IXOFF
    199 	{"ixoff", IXOFF, MD_INP},
    200 #endif /* IXOFF */
    201 #ifdef  IMAXBEL
    202 	{"imaxbel", IMAXBEL, MD_INP},
    203 #endif /* IMAXBEL */
    204 
    205 #ifdef	OPOST
    206 	{"opost", OPOST, MD_OUT},
    207 #endif /* OPOST */
    208 #ifdef	OLCUC
    209 	{"olcuc", OLCUC, MD_OUT},
    210 #endif /* OLCUC */
    211 #ifdef	ONLCR
    212 	{"onlcr", ONLCR, MD_OUT},
    213 #endif /* ONLCR */
    214 #ifdef	OCRNL
    215 	{"ocrnl", OCRNL, MD_OUT},
    216 #endif /* OCRNL */
    217 #ifdef	ONOCR
    218 	{"onocr", ONOCR, MD_OUT},
    219 #endif /* ONOCR */
    220 #ifdef ONOEOT
    221 	{"onoeot", ONOEOT, MD_OUT},
    222 #endif /* ONOEOT */
    223 #ifdef	ONLRET
    224 	{"onlret", ONLRET, MD_OUT},
    225 #endif /* ONLRET */
    226 #ifdef	OFILL
    227 	{"ofill", OFILL, MD_OUT},
    228 #endif /* OFILL */
    229 #ifdef	OFDEL
    230 	{"ofdel", OFDEL, MD_OUT},
    231 #endif /* OFDEL */
    232 #ifdef	NLDLY
    233 	{"nldly", NLDLY, MD_OUT},
    234 #endif /* NLDLY */
    235 #ifdef	CRDLY
    236 	{"crdly", CRDLY, MD_OUT},
    237 #endif /* CRDLY */
    238 #ifdef	TABDLY
    239 	{"tabdly", TABDLY, MD_OUT},
    240 #endif /* TABDLY */
    241 #ifdef	XTABS
    242 	{"xtabs", XTABS, MD_OUT},
    243 #endif /* XTABS */
    244 #ifdef	BSDLY
    245 	{"bsdly", BSDLY, MD_OUT},
    246 #endif /* BSDLY */
    247 #ifdef	VTDLY
    248 	{"vtdly", VTDLY, MD_OUT},
    249 #endif /* VTDLY */
    250 #ifdef	FFDLY
    251 	{"ffdly", FFDLY, MD_OUT},
    252 #endif /* FFDLY */
    253 #ifdef	PAGEOUT
    254 	{"pageout", PAGEOUT, MD_OUT},
    255 #endif /* PAGEOUT */
    256 #ifdef	WRAP
    257 	{"wrap", WRAP, MD_OUT},
    258 #endif /* WRAP */
    259 
    260 #ifdef	CIGNORE
    261 	{"cignore", CIGNORE, MD_CTL},
    262 #endif /* CBAUD */
    263 #ifdef	CBAUD
    264 	{"cbaud", CBAUD, MD_CTL},
    265 #endif /* CBAUD */
    266 #ifdef	CSTOPB
    267 	{"cstopb", CSTOPB, MD_CTL},
    268 #endif /* CSTOPB */
    269 #ifdef	CREAD
    270 	{"cread", CREAD, MD_CTL},
    271 #endif /* CREAD */
    272 #ifdef	PARENB
    273 	{"parenb", PARENB, MD_CTL},
    274 #endif /* PARENB */
    275 #ifdef	PARODD
    276 	{"parodd", PARODD, MD_CTL},
    277 #endif /* PARODD */
    278 #ifdef	HUPCL
    279 	{"hupcl", HUPCL, MD_CTL},
    280 #endif /* HUPCL */
    281 #ifdef	CLOCAL
    282 	{"clocal", CLOCAL, MD_CTL},
    283 #endif /* CLOCAL */
    284 #ifdef	LOBLK
    285 	{"loblk", LOBLK, MD_CTL},
    286 #endif /* LOBLK */
    287 #ifdef	CIBAUD
    288 	{"cibaud", CIBAUD, MD_CTL},
    289 #endif /* CIBAUD */
    290 #ifdef CRTSCTS
    291 #ifdef CCTS_OFLOW
    292 	{"ccts_oflow", CCTS_OFLOW, MD_CTL},
    293 #else
    294 	{"crtscts", CRTSCTS, MD_CTL},
    295 #endif /* CCTS_OFLOW */
    296 #endif /* CRTSCTS */
    297 #ifdef CRTS_IFLOW
    298 	{"crts_iflow", CRTS_IFLOW, MD_CTL},
    299 #endif /* CRTS_IFLOW */
    300 #ifdef CDTRCTS
    301 	{"cdtrcts", CDTRCTS, MD_CTL},
    302 #endif /* CDTRCTS */
    303 #ifdef MDMBUF
    304 	{"mdmbuf", MDMBUF, MD_CTL},
    305 #endif /* MDMBUF */
    306 #ifdef RCV1EN
    307 	{"rcv1en", RCV1EN, MD_CTL},
    308 #endif /* RCV1EN */
    309 #ifdef XMT1EN
    310 	{"xmt1en", XMT1EN, MD_CTL},
    311 #endif /* XMT1EN */
    312 
    313 #ifdef	ISIG
    314 	{"isig", ISIG, MD_LIN},
    315 #endif /* ISIG */
    316 #ifdef	ICANON
    317 	{"icanon", ICANON, MD_LIN},
    318 #endif /* ICANON */
    319 #ifdef	XCASE
    320 	{"xcase", XCASE, MD_LIN},
    321 #endif /* XCASE */
    322 #ifdef	ECHO
    323 	{"echo", ECHO, MD_LIN},
    324 #endif /* ECHO */
    325 #ifdef	ECHOE
    326 	{"echoe", ECHOE, MD_LIN},
    327 #endif /* ECHOE */
    328 #ifdef	ECHOK
    329 	{"echok", ECHOK, MD_LIN},
    330 #endif /* ECHOK */
    331 #ifdef	ECHONL
    332 	{"echonl", ECHONL, MD_LIN},
    333 #endif /* ECHONL */
    334 #ifdef	NOFLSH
    335 	{"noflsh", NOFLSH, MD_LIN},
    336 #endif /* NOFLSH */
    337 #ifdef	TOSTOP
    338 	{"tostop", TOSTOP, MD_LIN},
    339 #endif /* TOSTOP */
    340 #ifdef	ECHOCTL
    341 	{"echoctl", ECHOCTL, MD_LIN},
    342 #endif /* ECHOCTL */
    343 #ifdef	ECHOPRT
    344 	{"echoprt", ECHOPRT, MD_LIN},
    345 #endif /* ECHOPRT */
    346 #ifdef	ECHOKE
    347 	{"echoke", ECHOKE, MD_LIN},
    348 #endif /* ECHOKE */
    349 #ifdef	DEFECHO
    350 	{"defecho", DEFECHO, MD_LIN},
    351 #endif /* DEFECHO */
    352 #ifdef	FLUSHO
    353 	{"flusho", FLUSHO, MD_LIN},
    354 #endif /* FLUSHO */
    355 #ifdef	PENDIN
    356 	{"pendin", PENDIN, MD_LIN},
    357 #endif /* PENDIN */
    358 #ifdef	IEXTEN
    359 	{"iexten", IEXTEN, MD_LIN},
    360 #endif /* IEXTEN */
    361 #ifdef	NOKERNINFO
    362 	{"nokerninfo", NOKERNINFO, MD_LIN},
    363 #endif /* NOKERNINFO */
    364 #ifdef	ALTWERASE
    365 	{"altwerase", ALTWERASE, MD_LIN},
    366 #endif /* ALTWERASE */
    367 #ifdef	EXTPROC
    368 	{"extproc", EXTPROC, MD_LIN},
    369 #endif /* EXTPROC */
    370 
    371 #if defined(VINTR)
    372 	{"intr", C_SH(C_INTR), MD_CHAR},
    373 #endif /* VINTR */
    374 #if defined(VQUIT)
    375 	{"quit", C_SH(C_QUIT), MD_CHAR},
    376 #endif /* VQUIT */
    377 #if defined(VERASE)
    378 	{"erase", C_SH(C_ERASE), MD_CHAR},
    379 #endif /* VERASE */
    380 #if defined(VKILL)
    381 	{"kill", C_SH(C_KILL), MD_CHAR},
    382 #endif /* VKILL */
    383 #if defined(VEOF)
    384 	{"eof", C_SH(C_EOF), MD_CHAR},
    385 #endif /* VEOF */
    386 #if defined(VEOL)
    387 	{"eol", C_SH(C_EOL), MD_CHAR},
    388 #endif /* VEOL */
    389 #if defined(VEOL2)
    390 	{"eol2", C_SH(C_EOL2), MD_CHAR},
    391 #endif /* VEOL2 */
    392 #if defined(VSWTCH)
    393 	{"swtch", C_SH(C_SWTCH), MD_CHAR},
    394 #endif /* VSWTCH */
    395 #if defined(VDSWTCH)
    396 	{"dswtch", C_SH(C_DSWTCH), MD_CHAR},
    397 #endif /* VDSWTCH */
    398 #if defined(VERASE2)
    399 	{"erase2", C_SH(C_ERASE2), MD_CHAR},
    400 #endif /* VERASE2 */
    401 #if defined(VSTART)
    402 	{"start", C_SH(C_START), MD_CHAR},
    403 #endif /* VSTART */
    404 #if defined(VSTOP)
    405 	{"stop", C_SH(C_STOP), MD_CHAR},
    406 #endif /* VSTOP */
    407 #if defined(VWERASE)
    408 	{"werase", C_SH(C_WERASE), MD_CHAR},
    409 #endif /* VWERASE */
    410 #if defined(VSUSP)
    411 	{"susp", C_SH(C_SUSP), MD_CHAR},
    412 #endif /* VSUSP */
    413 #if defined(VDSUSP)
    414 	{"dsusp", C_SH(C_DSUSP), MD_CHAR},
    415 #endif /* VDSUSP */
    416 #if defined(VREPRINT)
    417 	{"reprint", C_SH(C_REPRINT), MD_CHAR},
    418 #endif /* VREPRINT */
    419 #if defined(VDISCARD)
    420 	{"discard", C_SH(C_DISCARD), MD_CHAR},
    421 #endif /* VDISCARD */
    422 #if defined(VLNEXT)
    423 	{"lnext", C_SH(C_LNEXT), MD_CHAR},
    424 #endif /* VLNEXT */
    425 #if defined(VSTATUS)
    426 	{"status", C_SH(C_STATUS), MD_CHAR},
    427 #endif /* VSTATUS */
    428 #if defined(VPAGE)
    429 	{"page", C_SH(C_PAGE), MD_CHAR},
    430 #endif /* VPAGE */
    431 #if defined(VPGOFF)
    432 	{"pgoff", C_SH(C_PGOFF), MD_CHAR},
    433 #endif /* VPGOFF */
    434 #if defined(VKILL2)
    435 	{"kill2", C_SH(C_KILL2), MD_CHAR},
    436 #endif /* VKILL2 */
    437 #if defined(VBRK)
    438 	{"brk", C_SH(C_BRK), MD_CHAR},
    439 #endif /* VBRK */
    440 #if defined(VMIN)
    441 	{"min", C_SH(C_MIN), MD_CHAR},
    442 #endif /* VMIN */
    443 #if defined(VTIME)
    444 	{"time", C_SH(C_TIME), MD_CHAR},
    445 #endif /* VTIME */
    446 	{NULL, 0, -1},
    447 };
    448 
    449 
    450 
    451 #define	tty__gettabs(td)	((((td)->c_oflag & TAB3) == TAB3) ? 0 : 1)
    452 #define	tty__geteightbit(td)	(((td)->c_cflag & CSIZE) == CS8)
    453 #define	tty__cooked_mode(td)	((td)->c_lflag & ICANON)
    454 
    455 private int	tty_getty(EditLine *, struct termios *);
    456 private int	tty_setty(EditLine *, int, const struct termios *);
    457 private int	tty__getcharindex(int);
    458 private void	tty__getchar(struct termios *, unsigned char *);
    459 private void	tty__setchar(struct termios *, unsigned char *);
    460 private speed_t	tty__getspeed(struct termios *);
    461 private int	tty_setup(EditLine *);
    462 
    463 #define	t_qu	t_ts
    464 
    465 /* tty_getty():
    466  *	Wrapper for tcgetattr to handle EINTR
    467  */
    468 private int
    469 tty_getty(EditLine *el, struct termios *t)
    470 {
    471 	int rv;
    472 	while ((rv = tcgetattr(el->el_infd, t)) == -1 && errno == EINTR)
    473 		continue;
    474 	return rv;
    475 }
    476 
    477 /* tty_setty():
    478  *	Wrapper for tcsetattr to handle EINTR
    479  */
    480 private int
    481 tty_setty(EditLine *el, int action, const struct termios *t)
    482 {
    483 	int rv;
    484 	while ((rv = tcsetattr(el->el_infd, action, t)) == -1 && errno == EINTR)
    485 		continue;
    486 	return rv;
    487 }
    488 
    489 /* tty_setup():
    490  *	Get the tty parameters and initialize the editing state
    491  */
    492 private int
    493 tty_setup(EditLine *el)
    494 {
    495 	int rst = 1;
    496 
    497 	if (el->el_flags & EDIT_DISABLED)
    498 		return 0;
    499 
    500 	if (!isatty(el->el_outfd)) {
    501 #ifdef DEBUG_TTY
    502 		(void) fprintf(el->el_errfile,
    503 		    "tty_setup: isatty: %s\n", strerror(errno));
    504 #endif /* DEBUG_TTY */
    505 		return -1;
    506 	}
    507 	if (tty_getty(el, &el->el_tty.t_ed) == -1) {
    508 #ifdef DEBUG_TTY
    509 		(void) fprintf(el->el_errfile,
    510 		    "tty_setup: tty_getty: %s\n", strerror(errno));
    511 #endif /* DEBUG_TTY */
    512 		return -1;
    513 	}
    514 	el->el_tty.t_ts = el->el_tty.t_ex = el->el_tty.t_ed;
    515 
    516 	el->el_tty.t_speed = tty__getspeed(&el->el_tty.t_ex);
    517 	el->el_tty.t_tabs = tty__gettabs(&el->el_tty.t_ex);
    518 	el->el_tty.t_eight = tty__geteightbit(&el->el_tty.t_ex);
    519 
    520 	el->el_tty.t_ex.c_iflag &= ~el->el_tty.t_t[EX_IO][MD_INP].t_clrmask;
    521 	el->el_tty.t_ex.c_iflag |= el->el_tty.t_t[EX_IO][MD_INP].t_setmask;
    522 
    523 	el->el_tty.t_ex.c_oflag &= ~el->el_tty.t_t[EX_IO][MD_OUT].t_clrmask;
    524 	el->el_tty.t_ex.c_oflag |= el->el_tty.t_t[EX_IO][MD_OUT].t_setmask;
    525 
    526 	el->el_tty.t_ex.c_cflag &= ~el->el_tty.t_t[EX_IO][MD_CTL].t_clrmask;
    527 	el->el_tty.t_ex.c_cflag |= el->el_tty.t_t[EX_IO][MD_CTL].t_setmask;
    528 
    529 	el->el_tty.t_ex.c_lflag &= ~el->el_tty.t_t[EX_IO][MD_LIN].t_clrmask;
    530 	el->el_tty.t_ex.c_lflag |= el->el_tty.t_t[EX_IO][MD_LIN].t_setmask;
    531 
    532 	/*
    533          * Reset the tty chars to reasonable defaults
    534          * If they are disabled, then enable them.
    535          */
    536 	if (rst) {
    537 		if (tty__cooked_mode(&el->el_tty.t_ts)) {
    538 			tty__getchar(&el->el_tty.t_ts, el->el_tty.t_c[TS_IO]);
    539 			/*
    540 	                 * Don't affect CMIN and CTIME for the editor mode
    541 	                 */
    542 			for (rst = 0; rst < C_NCC - 2; rst++)
    543 				if (el->el_tty.t_c[TS_IO][rst] !=
    544 				      el->el_tty.t_vdisable
    545 				    && el->el_tty.t_c[ED_IO][rst] !=
    546 				      el->el_tty.t_vdisable)
    547 					el->el_tty.t_c[ED_IO][rst] =
    548 					    el->el_tty.t_c[TS_IO][rst];
    549 			for (rst = 0; rst < C_NCC; rst++)
    550 				if (el->el_tty.t_c[TS_IO][rst] !=
    551 				    el->el_tty.t_vdisable)
    552 					el->el_tty.t_c[EX_IO][rst] =
    553 					    el->el_tty.t_c[TS_IO][rst];
    554 		}
    555 		tty__setchar(&el->el_tty.t_ex, el->el_tty.t_c[EX_IO]);
    556 		if (tty_setty(el, TCSADRAIN, &el->el_tty.t_ex) == -1) {
    557 #ifdef DEBUG_TTY
    558 			(void) fprintf(el->el_errfile,
    559 			    "tty_setup: tty_setty: %s\n",
    560 			    strerror(errno));
    561 #endif /* DEBUG_TTY */
    562 			return -1;
    563 		}
    564 	}
    565 
    566 	el->el_tty.t_ed.c_iflag &= ~el->el_tty.t_t[ED_IO][MD_INP].t_clrmask;
    567 	el->el_tty.t_ed.c_iflag |= el->el_tty.t_t[ED_IO][MD_INP].t_setmask;
    568 
    569 	el->el_tty.t_ed.c_oflag &= ~el->el_tty.t_t[ED_IO][MD_OUT].t_clrmask;
    570 	el->el_tty.t_ed.c_oflag |= el->el_tty.t_t[ED_IO][MD_OUT].t_setmask;
    571 
    572 	el->el_tty.t_ed.c_cflag &= ~el->el_tty.t_t[ED_IO][MD_CTL].t_clrmask;
    573 	el->el_tty.t_ed.c_cflag |= el->el_tty.t_t[ED_IO][MD_CTL].t_setmask;
    574 
    575 	el->el_tty.t_ed.c_lflag &= ~el->el_tty.t_t[ED_IO][MD_LIN].t_clrmask;
    576 	el->el_tty.t_ed.c_lflag |= el->el_tty.t_t[ED_IO][MD_LIN].t_setmask;
    577 
    578 	tty__setchar(&el->el_tty.t_ed, el->el_tty.t_c[ED_IO]);
    579 	tty_bind_char(el, 1);
    580 	return 0;
    581 }
    582 
    583 protected int
    584 tty_init(EditLine *el)
    585 {
    586 
    587 	el->el_tty.t_mode = EX_IO;
    588 	el->el_tty.t_vdisable = _POSIX_VDISABLE;
    589 	(void) memcpy(el->el_tty.t_t, ttyperm, sizeof(ttyperm_t));
    590 	(void) memcpy(el->el_tty.t_c, ttychar, sizeof(ttychar_t));
    591 	return tty_setup(el);
    592 }
    593 
    594 
    595 /* tty_end():
    596  *	Restore the tty to its original settings
    597  */
    598 protected void
    599 /*ARGSUSED*/
    600 tty_end(EditLine *el __attribute__((__unused__)))
    601 {
    602 
    603 	/* XXX: Maybe reset to an initial state? */
    604 }
    605 
    606 
    607 /* tty__getspeed():
    608  *	Get the tty speed
    609  */
    610 private 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 private 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 private 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 private 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 protected 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 	Char 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 != (Int)-1; tp++) {
    918 		new[0] = t_n[tp->nch];
    919 		old[0] = 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[UC(old[0])] = dmap[UC(old[0])];
    925 		keymacro_clear(el, map, new);
    926 		/* MAP_VI == 1, MAP_EMACS == 0... */
    927 		map[UC(new[0])] = tp->bind[el->el_map.type];
    928 		if (dalt) {
    929 			keymacro_clear(el, alt, old);
    930 			alt[UC(old[0])] = dalt[UC(old[0])];
    931 			keymacro_clear(el, alt, new);
    932 			alt[UC(new[0])] = tp->bind[el->el_map.type + 1];
    933 		}
    934 	}
    935 }
    936 
    937 
    938 /* tty_rawmode():
    939  * 	Set terminal into 1 character at a time mode.
    940  */
    941 protected int
    942 tty_rawmode(EditLine *el)
    943 {
    944 
    945 	if (el->el_tty.t_mode == ED_IO || el->el_tty.t_mode == QU_IO)
    946 		return 0;
    947 
    948 	if (el->el_flags & EDIT_DISABLED)
    949 		return 0;
    950 
    951 	if (tty_getty(el, &el->el_tty.t_ts) == -1) {
    952 #ifdef DEBUG_TTY
    953 		(void) fprintf(el->el_errfile, "tty_rawmode: tty_getty: %s\n",
    954 		    strerror(errno));
    955 #endif /* DEBUG_TTY */
    956 		return -1;
    957 	}
    958 	/*
    959          * We always keep up with the eight bit setting and the speed of the
    960          * tty. But we only believe changes that are made to cooked mode!
    961          */
    962 	el->el_tty.t_eight = tty__geteightbit(&el->el_tty.t_ts);
    963 	el->el_tty.t_speed = tty__getspeed(&el->el_tty.t_ts);
    964 
    965 	if (tty__getspeed(&el->el_tty.t_ex) != el->el_tty.t_speed ||
    966 	    tty__getspeed(&el->el_tty.t_ed) != el->el_tty.t_speed) {
    967 		(void) cfsetispeed(&el->el_tty.t_ex, el->el_tty.t_speed);
    968 		(void) cfsetospeed(&el->el_tty.t_ex, el->el_tty.t_speed);
    969 		(void) cfsetispeed(&el->el_tty.t_ed, el->el_tty.t_speed);
    970 		(void) cfsetospeed(&el->el_tty.t_ed, el->el_tty.t_speed);
    971 	}
    972 	if (tty__cooked_mode(&el->el_tty.t_ts)) {
    973 		if (el->el_tty.t_ts.c_cflag != el->el_tty.t_ex.c_cflag) {
    974 			el->el_tty.t_ex.c_cflag =
    975 			    el->el_tty.t_ts.c_cflag;
    976 			el->el_tty.t_ex.c_cflag &=
    977 			    ~el->el_tty.t_t[EX_IO][MD_CTL].t_clrmask;
    978 			el->el_tty.t_ex.c_cflag |=
    979 			    el->el_tty.t_t[EX_IO][MD_CTL].t_setmask;
    980 
    981 			el->el_tty.t_ed.c_cflag =
    982 			    el->el_tty.t_ts.c_cflag;
    983 			el->el_tty.t_ed.c_cflag &=
    984 			    ~el->el_tty.t_t[ED_IO][MD_CTL].t_clrmask;
    985 			el->el_tty.t_ed.c_cflag |=
    986 			    el->el_tty.t_t[ED_IO][MD_CTL].t_setmask;
    987 		}
    988 		if ((el->el_tty.t_ts.c_lflag != el->el_tty.t_ex.c_lflag) &&
    989 		    (el->el_tty.t_ts.c_lflag != el->el_tty.t_ed.c_lflag)) {
    990 			el->el_tty.t_ex.c_lflag =
    991 			    el->el_tty.t_ts.c_lflag;
    992 			el->el_tty.t_ex.c_lflag &=
    993 			    ~el->el_tty.t_t[EX_IO][MD_LIN].t_clrmask;
    994 			el->el_tty.t_ex.c_lflag |=
    995 			    el->el_tty.t_t[EX_IO][MD_LIN].t_setmask;
    996 
    997 			el->el_tty.t_ed.c_lflag =
    998 			    el->el_tty.t_ts.c_lflag;
    999 			el->el_tty.t_ed.c_lflag &=
   1000 			    ~el->el_tty.t_t[ED_IO][MD_LIN].t_clrmask;
   1001 			el->el_tty.t_ed.c_lflag |=
   1002 			    el->el_tty.t_t[ED_IO][MD_LIN].t_setmask;
   1003 		}
   1004 		if ((el->el_tty.t_ts.c_iflag != el->el_tty.t_ex.c_iflag) &&
   1005 		    (el->el_tty.t_ts.c_iflag != el->el_tty.t_ed.c_iflag)) {
   1006 			el->el_tty.t_ex.c_iflag =
   1007 			    el->el_tty.t_ts.c_iflag;
   1008 			el->el_tty.t_ex.c_iflag &=
   1009 			    ~el->el_tty.t_t[EX_IO][MD_INP].t_clrmask;
   1010 			el->el_tty.t_ex.c_iflag |=
   1011 			    el->el_tty.t_t[EX_IO][MD_INP].t_setmask;
   1012 
   1013 			el->el_tty.t_ed.c_iflag =
   1014 			    el->el_tty.t_ts.c_iflag;
   1015 			el->el_tty.t_ed.c_iflag &=
   1016 			    ~el->el_tty.t_t[ED_IO][MD_INP].t_clrmask;
   1017 			el->el_tty.t_ed.c_iflag |=
   1018 			    el->el_tty.t_t[ED_IO][MD_INP].t_setmask;
   1019 		}
   1020 		if ((el->el_tty.t_ts.c_oflag != el->el_tty.t_ex.c_oflag) &&
   1021 		    (el->el_tty.t_ts.c_oflag != el->el_tty.t_ed.c_oflag)) {
   1022 			el->el_tty.t_ex.c_oflag =
   1023 			    el->el_tty.t_ts.c_oflag;
   1024 			el->el_tty.t_ex.c_oflag &=
   1025 			    ~el->el_tty.t_t[EX_IO][MD_OUT].t_clrmask;
   1026 			el->el_tty.t_ex.c_oflag |=
   1027 			    el->el_tty.t_t[EX_IO][MD_OUT].t_setmask;
   1028 
   1029 			el->el_tty.t_ed.c_oflag =
   1030 			    el->el_tty.t_ts.c_oflag;
   1031 			el->el_tty.t_ed.c_oflag &=
   1032 			    ~el->el_tty.t_t[ED_IO][MD_OUT].t_clrmask;
   1033 			el->el_tty.t_ed.c_oflag |=
   1034 			    el->el_tty.t_t[ED_IO][MD_OUT].t_setmask;
   1035 		}
   1036 		if (tty__gettabs(&el->el_tty.t_ex) == 0)
   1037 			el->el_tty.t_tabs = 0;
   1038 		else
   1039 			el->el_tty.t_tabs = EL_CAN_TAB ? 1 : 0;
   1040 
   1041 		{
   1042 			int i;
   1043 
   1044 			tty__getchar(&el->el_tty.t_ts, el->el_tty.t_c[TS_IO]);
   1045 			/*
   1046 		         * Check if the user made any changes.
   1047 		         * If he did, then propagate the changes to the
   1048 		         * edit and execute data structures.
   1049 		         */
   1050 			for (i = 0; i < C_NCC; i++)
   1051 				if (el->el_tty.t_c[TS_IO][i] !=
   1052 				    el->el_tty.t_c[EX_IO][i])
   1053 					break;
   1054 
   1055 			if (i != C_NCC) {
   1056 				/*
   1057 				 * Propagate changes only to the unprotected
   1058 				 * chars that have been modified just now.
   1059 				 */
   1060 				for (i = 0; i < C_NCC; i++) {
   1061 					if (!((el->el_tty.t_t[ED_IO][MD_CHAR].t_setmask & C_SH(i)))
   1062 					    && (el->el_tty.t_c[TS_IO][i] != el->el_tty.t_c[EX_IO][i]))
   1063 						el->el_tty.t_c[ED_IO][i] = el->el_tty.t_c[TS_IO][i];
   1064 					if (el->el_tty.t_t[ED_IO][MD_CHAR].t_clrmask & C_SH(i))
   1065 						el->el_tty.t_c[ED_IO][i] = el->el_tty.t_vdisable;
   1066 				}
   1067 				tty_bind_char(el, 0);
   1068 				tty__setchar(&el->el_tty.t_ed, el->el_tty.t_c[ED_IO]);
   1069 
   1070 				for (i = 0; i < C_NCC; i++) {
   1071 					if (!((el->el_tty.t_t[EX_IO][MD_CHAR].t_setmask & C_SH(i)))
   1072 					    && (el->el_tty.t_c[TS_IO][i] != el->el_tty.t_c[EX_IO][i]))
   1073 						el->el_tty.t_c[EX_IO][i] = el->el_tty.t_c[TS_IO][i];
   1074 					if (el->el_tty.t_t[EX_IO][MD_CHAR].t_clrmask & C_SH(i))
   1075 						el->el_tty.t_c[EX_IO][i] = el->el_tty.t_vdisable;
   1076 				}
   1077 				tty__setchar(&el->el_tty.t_ex, el->el_tty.t_c[EX_IO]);
   1078 			}
   1079 		}
   1080 	}
   1081 	if (tty_setty(el, TCSADRAIN, &el->el_tty.t_ed) == -1) {
   1082 #ifdef DEBUG_TTY
   1083 		(void) fprintf(el->el_errfile, "tty_rawmode: tty_setty: %s\n",
   1084 		    strerror(errno));
   1085 #endif /* DEBUG_TTY */
   1086 		return -1;
   1087 	}
   1088 	el->el_tty.t_mode = ED_IO;
   1089 	return 0;
   1090 }
   1091 
   1092 
   1093 /* tty_cookedmode():
   1094  *	Set the tty back to normal mode
   1095  */
   1096 protected int
   1097 tty_cookedmode(EditLine *el)
   1098 {				/* set tty in normal setup */
   1099 
   1100 	if (el->el_tty.t_mode == EX_IO)
   1101 		return 0;
   1102 
   1103 	if (el->el_flags & EDIT_DISABLED)
   1104 		return 0;
   1105 
   1106 	if (tty_setty(el, TCSADRAIN, &el->el_tty.t_ex) == -1) {
   1107 #ifdef DEBUG_TTY
   1108 		(void) fprintf(el->el_errfile,
   1109 		    "tty_cookedmode: tty_setty: %s\n",
   1110 		    strerror(errno));
   1111 #endif /* DEBUG_TTY */
   1112 		return -1;
   1113 	}
   1114 	el->el_tty.t_mode = EX_IO;
   1115 	return 0;
   1116 }
   1117 
   1118 
   1119 /* tty_quotemode():
   1120  *	Turn on quote mode
   1121  */
   1122 protected int
   1123 tty_quotemode(EditLine *el)
   1124 {
   1125 	if (el->el_tty.t_mode == QU_IO)
   1126 		return 0;
   1127 
   1128 	el->el_tty.t_qu = el->el_tty.t_ed;
   1129 
   1130 	el->el_tty.t_qu.c_iflag &= ~el->el_tty.t_t[QU_IO][MD_INP].t_clrmask;
   1131 	el->el_tty.t_qu.c_iflag |= el->el_tty.t_t[QU_IO][MD_INP].t_setmask;
   1132 
   1133 	el->el_tty.t_qu.c_oflag &= ~el->el_tty.t_t[QU_IO][MD_OUT].t_clrmask;
   1134 	el->el_tty.t_qu.c_oflag |= el->el_tty.t_t[QU_IO][MD_OUT].t_setmask;
   1135 
   1136 	el->el_tty.t_qu.c_cflag &= ~el->el_tty.t_t[QU_IO][MD_CTL].t_clrmask;
   1137 	el->el_tty.t_qu.c_cflag |= el->el_tty.t_t[QU_IO][MD_CTL].t_setmask;
   1138 
   1139 	el->el_tty.t_qu.c_lflag &= ~el->el_tty.t_t[QU_IO][MD_LIN].t_clrmask;
   1140 	el->el_tty.t_qu.c_lflag |= el->el_tty.t_t[QU_IO][MD_LIN].t_setmask;
   1141 
   1142 	if (tty_setty(el, TCSADRAIN, &el->el_tty.t_qu) == -1) {
   1143 #ifdef DEBUG_TTY
   1144 		(void) fprintf(el->el_errfile, "QuoteModeOn: tty_setty: %s\n",
   1145 		    strerror(errno));
   1146 #endif /* DEBUG_TTY */
   1147 		return -1;
   1148 	}
   1149 	el->el_tty.t_mode = QU_IO;
   1150 	return 0;
   1151 }
   1152 
   1153 
   1154 /* tty_noquotemode():
   1155  *	Turn off quote mode
   1156  */
   1157 protected int
   1158 tty_noquotemode(EditLine *el)
   1159 {
   1160 
   1161 	if (el->el_tty.t_mode != QU_IO)
   1162 		return 0;
   1163 	if (tty_setty(el, TCSADRAIN, &el->el_tty.t_ed) == -1) {
   1164 #ifdef DEBUG_TTY
   1165 		(void) fprintf(el->el_errfile, "QuoteModeOff: tty_setty: %s\n",
   1166 		    strerror(errno));
   1167 #endif /* DEBUG_TTY */
   1168 		return -1;
   1169 	}
   1170 	el->el_tty.t_mode = ED_IO;
   1171 	return 0;
   1172 }
   1173 
   1174 
   1175 /* tty_stty():
   1176  *	Stty builtin
   1177  */
   1178 protected int
   1179 /*ARGSUSED*/
   1180 tty_stty(EditLine *el, int argc __attribute__((__unused__)), const Char **argv)
   1181 {
   1182 	const ttymodes_t *m;
   1183 	char x;
   1184 	int aflag = 0;
   1185 	const Char *s, *d;
   1186         char name[EL_BUFSIZ];
   1187 	struct termios *tios = &el->el_tty.t_ex;
   1188 	int z = EX_IO;
   1189 
   1190 	if (argv == NULL)
   1191 		return -1;
   1192 	strncpy(name, ct_encode_string(*argv++, &el->el_scratch), sizeof(name));
   1193         name[sizeof(name) - 1] = '\0';
   1194 
   1195 	while (argv && *argv && argv[0][0] == '-' && argv[0][2] == '\0')
   1196 		switch (argv[0][1]) {
   1197 		case 'a':
   1198 			aflag++;
   1199 			argv++;
   1200 			break;
   1201 		case 'd':
   1202 			argv++;
   1203 			tios = &el->el_tty.t_ed;
   1204 			z = ED_IO;
   1205 			break;
   1206 		case 'x':
   1207 			argv++;
   1208 			tios = &el->el_tty.t_ex;
   1209 			z = EX_IO;
   1210 			break;
   1211 		case 'q':
   1212 			argv++;
   1213 			tios = &el->el_tty.t_ts;
   1214 			z = QU_IO;
   1215 			break;
   1216 		default:
   1217 			(void) fprintf(el->el_errfile,
   1218 			    "%s: Unknown switch `%c'.\n",
   1219 			    name, argv[0][1]);
   1220 			return -1;
   1221 		}
   1222 
   1223 	if (!argv || !*argv) {
   1224 		int i = -1;
   1225 		size_t len = 0, st = 0, cu;
   1226 		for (m = ttymodes; m->m_name; m++) {
   1227 			if (m->m_type != i) {
   1228 				(void) fprintf(el->el_outfile, "%s%s",
   1229 				    i != -1 ? "\n" : "",
   1230 				    el->el_tty.t_t[z][m->m_type].t_name);
   1231 				i = m->m_type;
   1232 				st = len =
   1233 				    strlen(el->el_tty.t_t[z][m->m_type].t_name);
   1234 			}
   1235 			if (i != -1) {
   1236 			    x = (el->el_tty.t_t[z][i].t_setmask & m->m_value)
   1237 				?  '+' : '\0';
   1238 
   1239 			    if (el->el_tty.t_t[z][i].t_clrmask & m->m_value)
   1240 				x = '-';
   1241 			} else {
   1242 			    x = '\0';
   1243 			}
   1244 
   1245 			if (x != '\0' || aflag) {
   1246 
   1247 				cu = strlen(m->m_name) + (x != '\0') + 1;
   1248 
   1249 				if (len + cu >= (size_t)el->el_terminal.t_size.h) {
   1250 					(void) fprintf(el->el_outfile, "\n%*s",
   1251 					    (int)st, "");
   1252 					len = st + cu;
   1253 				} else
   1254 					len += cu;
   1255 
   1256 				if (x != '\0')
   1257 					(void) fprintf(el->el_outfile, "%c%s ",
   1258 					    x, m->m_name);
   1259 				else
   1260 					(void) fprintf(el->el_outfile, "%s ",
   1261 					    m->m_name);
   1262 			}
   1263 		}
   1264 		(void) fprintf(el->el_outfile, "\n");
   1265 		return 0;
   1266 	}
   1267 	while (argv && (s = *argv++)) {
   1268 		const Char *p;
   1269 		switch (*s) {
   1270 		case '+':
   1271 		case '-':
   1272 			x = (char)*s++;
   1273 			break;
   1274 		default:
   1275 			x = '\0';
   1276 			break;
   1277 		}
   1278 		d = s;
   1279 		p = Strchr(s, '=');
   1280 		for (m = ttymodes; m->m_name; m++)
   1281 			if ((p ? strncmp(m->m_name, ct_encode_string(d, &el->el_scratch), (size_t)(p - d)) :
   1282 			    strcmp(m->m_name, ct_encode_string(d, &el->el_scratch))) == 0 &&
   1283 			    (p == NULL || m->m_type == MD_CHAR))
   1284 				break;
   1285 
   1286 		if (!m->m_name) {
   1287 			(void) fprintf(el->el_errfile,
   1288 			    "%s: Invalid argument `" FSTR "'.\n", name, d);
   1289 			return -1;
   1290 		}
   1291 		if (p) {
   1292 			int c = ffs((int)m->m_value);
   1293 			int v = *++p ? parse__escape(&p) :
   1294 			    el->el_tty.t_vdisable;
   1295 			assert(c != 0);
   1296 			c--;
   1297 			c = tty__getcharindex(c);
   1298 			assert(c != -1);
   1299 			tios->c_cc[c] = (cc_t)v;
   1300 			continue;
   1301 		}
   1302 		switch (x) {
   1303 		case '+':
   1304 			el->el_tty.t_t[z][m->m_type].t_setmask |= m->m_value;
   1305 			el->el_tty.t_t[z][m->m_type].t_clrmask &= ~m->m_value;
   1306 			break;
   1307 		case '-':
   1308 			el->el_tty.t_t[z][m->m_type].t_setmask &= ~m->m_value;
   1309 			el->el_tty.t_t[z][m->m_type].t_clrmask |= m->m_value;
   1310 			break;
   1311 		default:
   1312 			el->el_tty.t_t[z][m->m_type].t_setmask &= ~m->m_value;
   1313 			el->el_tty.t_t[z][m->m_type].t_clrmask &= ~m->m_value;
   1314 			break;
   1315 		}
   1316 	}
   1317 
   1318 	if (el->el_tty.t_mode == z) {
   1319 		if (tty_setty(el, TCSADRAIN, tios) == -1) {
   1320 #ifdef DEBUG_TTY
   1321 			(void) fprintf(el->el_errfile,
   1322 			    "tty_stty: tty_setty: %s\n", strerror(errno));
   1323 #endif /* DEBUG_TTY */
   1324 			return -1;
   1325 		}
   1326 	}
   1327 
   1328 	return 0;
   1329 }
   1330 
   1331 
   1332 #ifdef notyet
   1333 /* tty_printchar():
   1334  *	DEbugging routine to print the tty characters
   1335  */
   1336 private void
   1337 tty_printchar(EditLine *el, unsigned char *s)
   1338 {
   1339 	ttyperm_t *m;
   1340 	int i;
   1341 
   1342 	for (i = 0; i < C_NCC; i++) {
   1343 		for (m = el->el_tty.t_t; m->m_name; m++)
   1344 			if (m->m_type == MD_CHAR && C_SH(i) == m->m_value)
   1345 				break;
   1346 		if (m->m_name)
   1347 			(void) fprintf(el->el_errfile, "%s ^%c ",
   1348 			    m->m_name, s[i] + 'A' - 1);
   1349 		if (i % 5 == 0)
   1350 			(void) fprintf(el->el_errfile, "\n");
   1351 	}
   1352 	(void) fprintf(el->el_errfile, "\n");
   1353 }
   1354 #endif /* notyet */
   1355