mirror of
https://git.hardenedbsd.org/hardenedbsd/HardenedBSD.git
synced 2025-01-01 00:18:15 +01:00
1145 lines
30 KiB
C
1145 lines
30 KiB
C
/*-
|
|
* Copyright (c) 1992, 1993
|
|
* The Regents of the University of California. All rights reserved.
|
|
*
|
|
* This code is derived from software contributed to Berkeley by
|
|
* Christos Zoulas of Cornell University.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions
|
|
* are met:
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions and the following disclaimer.
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
* documentation and/or other materials provided with the distribution.
|
|
* 3. All advertising materials mentioning features or use of this software
|
|
* must display the following acknowledgement:
|
|
* This product includes software developed by the University of
|
|
* California, Berkeley and its contributors.
|
|
* 4. Neither the name of the University nor the names of its contributors
|
|
* may be used to endorse or promote products derived from this software
|
|
* without specific prior written permission.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
|
|
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
* ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
|
|
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
|
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
|
|
* OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
|
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
|
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
|
|
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
|
* SUCH DAMAGE.
|
|
*/
|
|
|
|
#if !defined(lint) && !defined(SCCSID)
|
|
static char sccsid[] = "@(#)tty.c 8.1 (Berkeley) 6/4/93";
|
|
#endif /* not lint && not SCCSID */
|
|
|
|
/*
|
|
* tty.c: tty interface stuff
|
|
*/
|
|
#include "sys.h"
|
|
#include "tty.h"
|
|
#include "el.h"
|
|
|
|
typedef struct ttymodes_t {
|
|
char *m_name;
|
|
int m_value;
|
|
int m_type;
|
|
} ttymodes_t;
|
|
|
|
typedef struct ttymap_t {
|
|
int nch, och; /* Internal and termio rep of chars */
|
|
el_action_t bind[3]; /* emacs, vi, and vi-cmd */
|
|
} ttymap_t;
|
|
|
|
|
|
private ttyperm_t ttyperm = {
|
|
{
|
|
{ "iflag:", ICRNL, (INLCR|IGNCR) },
|
|
{ "oflag:", (OPOST|ONLCR), ONLRET },
|
|
{ "cflag:", 0, 0 },
|
|
{ "lflag:", (ISIG|ICANON|ECHO|ECHOE|ECHOCTL|IEXTEN),
|
|
(NOFLSH|ECHONL|EXTPROC|FLUSHO) },
|
|
{ "chars:", 0, 0 },
|
|
},
|
|
{
|
|
{ "iflag:", (INLCR|ICRNL), IGNCR },
|
|
{ "oflag:", (OPOST|ONLCR), ONLRET },
|
|
{ "cflag:", 0, 0 },
|
|
{ "lflag:", ISIG,
|
|
(NOFLSH|ICANON|ECHO|ECHOK|ECHONL|EXTPROC|IEXTEN|FLUSHO) },
|
|
{ "chars:", (C_SH(C_MIN)|C_SH(C_TIME)|C_SH(C_SWTCH)|C_SH(C_DSWTCH)|
|
|
C_SH(C_SUSP)|C_SH(C_DSUSP)|C_SH(C_EOL)|C_SH(C_DISCARD)|
|
|
C_SH(C_PGOFF)|C_SH(C_PAGE)|C_SH(C_STATUS)), 0 }
|
|
},
|
|
{
|
|
{ "iflag:", 0, IXON | IXOFF },
|
|
{ "oflag:", 0, 0 },
|
|
{ "cflag:", 0, 0 },
|
|
{ "lflag:", 0, ISIG | IEXTEN },
|
|
{ "chars:", 0, 0 },
|
|
}
|
|
};
|
|
|
|
private ttychar_t ttychar = {
|
|
{
|
|
CINTR, CQUIT, CERASE, CKILL,
|
|
CEOF, CEOL, CEOL2, CSWTCH,
|
|
CDSWTCH, CERASE2, CSTART, CSTOP,
|
|
CWERASE, CSUSP, CDSUSP, CREPRINT,
|
|
CDISCARD, CLNEXT, CSTATUS, CPAGE,
|
|
CPGOFF, CKILL2, CBRK, CMIN,
|
|
CTIME
|
|
},
|
|
{
|
|
CINTR, CQUIT, CERASE, CKILL,
|
|
_POSIX_VDISABLE, _POSIX_VDISABLE, _POSIX_VDISABLE, _POSIX_VDISABLE,
|
|
_POSIX_VDISABLE, CERASE2, CSTART, CSTOP,
|
|
_POSIX_VDISABLE, CSUSP, _POSIX_VDISABLE, _POSIX_VDISABLE,
|
|
CDISCARD, _POSIX_VDISABLE, _POSIX_VDISABLE, _POSIX_VDISABLE,
|
|
_POSIX_VDISABLE, _POSIX_VDISABLE, _POSIX_VDISABLE, 1,
|
|
0
|
|
},
|
|
{
|
|
0, 0, 0, 0,
|
|
0, 0, 0, 0,
|
|
0, 0, 0, 0,
|
|
0, 0, 0, 0,
|
|
0, 0, 0, 0,
|
|
0, 0, 0, 0,
|
|
0
|
|
}
|
|
};
|
|
|
|
private ttymap_t tty_map[] = {
|
|
#ifdef VERASE
|
|
{ C_ERASE, VERASE,
|
|
{ ED_DELETE_PREV_CHAR, VI_DELETE_PREV_CHAR, ED_PREV_CHAR } },
|
|
#endif /* VERASE */
|
|
#ifdef VERASE2
|
|
{ C_ERASE2, VERASE2,
|
|
{ ED_DELETE_PREV_CHAR, VI_DELETE_PREV_CHAR, ED_PREV_CHAR } },
|
|
#endif /* VERASE2 */
|
|
#ifdef VKILL
|
|
{ C_KILL, VKILL,
|
|
{ EM_KILL_LINE, VI_KILL_LINE_PREV, ED_UNASSIGNED } },
|
|
#endif /* VKILL */
|
|
#ifdef VKILL2
|
|
{ C_KILL2, VKILL2,
|
|
{ EM_KILL_LINE, VI_KILL_LINE_PREV, ED_UNASSIGNED } },
|
|
#endif /* VKILL2 */
|
|
#ifdef VEOF
|
|
{ C_EOF, VEOF,
|
|
{ EM_DELETE_OR_LIST, VI_LIST_OR_EOF, ED_UNASSIGNED } },
|
|
#endif /* VEOF */
|
|
#ifdef VWERASE
|
|
{ C_WERASE, VWERASE,
|
|
{ ED_DELETE_PREV_WORD, ED_DELETE_PREV_WORD, ED_PREV_WORD } },
|
|
#endif /* VWERASE */
|
|
#ifdef VREPRINT
|
|
{ C_REPRINT, VREPRINT,
|
|
{ ED_REDISPLAY, ED_INSERT, ED_REDISPLAY } },
|
|
#endif /* VREPRINT */
|
|
#ifdef VLNEXT
|
|
{ C_LNEXT, VLNEXT,
|
|
{ ED_QUOTED_INSERT, ED_QUOTED_INSERT, ED_UNASSIGNED } },
|
|
#endif /* VLNEXT */
|
|
{ -1, -1,
|
|
{ ED_UNASSIGNED, ED_UNASSIGNED, ED_UNASSIGNED } }
|
|
};
|
|
|
|
private ttymodes_t ttymodes[] = {
|
|
# ifdef IGNBRK
|
|
{ "ignbrk", IGNBRK, M_INP },
|
|
# endif /* IGNBRK */
|
|
# ifdef BRKINT
|
|
{ "brkint", BRKINT, M_INP },
|
|
# endif /* BRKINT */
|
|
# ifdef IGNPAR
|
|
{ "ignpar", IGNPAR, M_INP },
|
|
# endif /* IGNPAR */
|
|
# ifdef PARMRK
|
|
{ "parmrk", PARMRK, M_INP },
|
|
# endif /* PARMRK */
|
|
# ifdef INPCK
|
|
{ "inpck", INPCK, M_INP },
|
|
# endif /* INPCK */
|
|
# ifdef ISTRIP
|
|
{ "istrip", ISTRIP, M_INP },
|
|
# endif /* ISTRIP */
|
|
# ifdef INLCR
|
|
{ "inlcr", INLCR, M_INP },
|
|
# endif /* INLCR */
|
|
# ifdef IGNCR
|
|
{ "igncr", IGNCR, M_INP },
|
|
# endif /* IGNCR */
|
|
# ifdef ICRNL
|
|
{ "icrnl", ICRNL, M_INP },
|
|
# endif /* ICRNL */
|
|
# ifdef IUCLC
|
|
{ "iuclc", IUCLC, M_INP },
|
|
# endif /* IUCLC */
|
|
# ifdef IXON
|
|
{ "ixon", IXON, M_INP },
|
|
# endif /* IXON */
|
|
# ifdef IXANY
|
|
{ "ixany", IXANY, M_INP },
|
|
# endif /* IXANY */
|
|
# ifdef IXOFF
|
|
{ "ixoff", IXOFF, M_INP },
|
|
# endif /* IXOFF */
|
|
# ifdef IMAXBEL
|
|
{ "imaxbel",IMAXBEL,M_INP },
|
|
# endif /* IMAXBEL */
|
|
|
|
# ifdef OPOST
|
|
{ "opost", OPOST, M_OUT },
|
|
# endif /* OPOST */
|
|
# ifdef OLCUC
|
|
{ "olcuc", OLCUC, M_OUT },
|
|
# endif /* OLCUC */
|
|
# ifdef ONLCR
|
|
{ "onlcr", ONLCR, M_OUT },
|
|
# endif /* ONLCR */
|
|
# ifdef OCRNL
|
|
{ "ocrnl", OCRNL, M_OUT },
|
|
# endif /* OCRNL */
|
|
# ifdef ONOCR
|
|
{ "onocr", ONOCR, M_OUT },
|
|
# endif /* ONOCR */
|
|
# ifdef ONOEOT
|
|
{ "onoeot", ONOEOT, M_OUT },
|
|
# endif /* ONOEOT */
|
|
# ifdef ONLRET
|
|
{ "onlret", ONLRET, M_OUT },
|
|
# endif /* ONLRET */
|
|
# ifdef OFILL
|
|
{ "ofill", OFILL, M_OUT },
|
|
# endif /* OFILL */
|
|
# ifdef OFDEL
|
|
{ "ofdel", OFDEL, M_OUT },
|
|
# endif /* OFDEL */
|
|
# ifdef NLDLY
|
|
{ "nldly", NLDLY, M_OUT },
|
|
# endif /* NLDLY */
|
|
# ifdef CRDLY
|
|
{ "crdly", CRDLY, M_OUT },
|
|
# endif /* CRDLY */
|
|
# ifdef TABDLY
|
|
{ "tabdly", TABDLY, M_OUT },
|
|
# endif /* TABDLY */
|
|
# ifdef XTABS
|
|
{ "xtabs", XTABS, M_OUT },
|
|
# endif /* XTABS */
|
|
# ifdef BSDLY
|
|
{ "bsdly", BSDLY, M_OUT },
|
|
# endif /* BSDLY */
|
|
# ifdef VTDLY
|
|
{ "vtdly", VTDLY, M_OUT },
|
|
# endif /* VTDLY */
|
|
# ifdef FFDLY
|
|
{ "ffdly", FFDLY, M_OUT },
|
|
# endif /* FFDLY */
|
|
# ifdef PAGEOUT
|
|
{ "pageout",PAGEOUT,M_OUT },
|
|
# endif /* PAGEOUT */
|
|
# ifdef WRAP
|
|
{ "wrap", WRAP, M_OUT },
|
|
# endif /* WRAP */
|
|
|
|
# ifdef CIGNORE
|
|
{ "cignore",CIGNORE,M_CTL },
|
|
# endif /* CBAUD */
|
|
# ifdef CBAUD
|
|
{ "cbaud", CBAUD, M_CTL },
|
|
# endif /* CBAUD */
|
|
# ifdef CSTOPB
|
|
{ "cstopb", CSTOPB, M_CTL },
|
|
# endif /* CSTOPB */
|
|
# ifdef CREAD
|
|
{ "cread", CREAD, M_CTL },
|
|
# endif /* CREAD */
|
|
# ifdef PARENB
|
|
{ "parenb", PARENB, M_CTL },
|
|
# endif /* PARENB */
|
|
# ifdef PARODD
|
|
{ "parodd", PARODD, M_CTL },
|
|
# endif /* PARODD */
|
|
# ifdef HUPCL
|
|
{ "hupcl", HUPCL, M_CTL },
|
|
# endif /* HUPCL */
|
|
# ifdef CLOCAL
|
|
{ "clocal", CLOCAL, M_CTL },
|
|
# endif /* CLOCAL */
|
|
# ifdef LOBLK
|
|
{ "loblk", LOBLK, M_CTL },
|
|
# endif /* LOBLK */
|
|
# ifdef CIBAUD
|
|
{ "cibaud", CIBAUD, M_CTL },
|
|
# endif /* CIBAUD */
|
|
# ifdef CRTSCTS
|
|
# ifdef CCTS_OFLOW
|
|
{ "ccts_oflow",CCTS_OFLOW,M_CTL },
|
|
# else
|
|
{ "crtscts",CRTSCTS,M_CTL },
|
|
# endif /* CCTS_OFLOW */
|
|
# endif /* CRTSCTS */
|
|
# ifdef CRTS_IFLOW
|
|
{ "crts_iflow",CRTS_IFLOW,M_CTL },
|
|
# endif /* CRTS_IFLOW */
|
|
# ifdef MDMBUF
|
|
{ "mdmbuf", MDMBUF, M_CTL },
|
|
# endif /* MDMBUF */
|
|
# ifdef RCV1EN
|
|
{ "rcv1en", RCV1EN, M_CTL },
|
|
# endif /* RCV1EN */
|
|
# ifdef XMT1EN
|
|
{ "xmt1en", XMT1EN, M_CTL },
|
|
# endif /* XMT1EN */
|
|
|
|
# ifdef ISIG
|
|
{ "isig", ISIG, M_LIN },
|
|
# endif /* ISIG */
|
|
# ifdef ICANON
|
|
{ "icanon", ICANON, M_LIN },
|
|
# endif /* ICANON */
|
|
# ifdef XCASE
|
|
{ "xcase", XCASE, M_LIN },
|
|
# endif /* XCASE */
|
|
# ifdef ECHO
|
|
{ "echo", ECHO, M_LIN },
|
|
# endif /* ECHO */
|
|
# ifdef ECHOE
|
|
{ "echoe", ECHOE, M_LIN },
|
|
# endif /* ECHOE */
|
|
# ifdef ECHOK
|
|
{ "echok", ECHOK, M_LIN },
|
|
# endif /* ECHOK */
|
|
# ifdef ECHONL
|
|
{ "echonl", ECHONL, M_LIN },
|
|
# endif /* ECHONL */
|
|
# ifdef NOFLSH
|
|
{ "noflsh", NOFLSH, M_LIN },
|
|
# endif /* NOFLSH */
|
|
# ifdef TOSTOP
|
|
{ "tostop", TOSTOP, M_LIN },
|
|
# endif /* TOSTOP */
|
|
# ifdef ECHOCTL
|
|
{ "echoctl",ECHOCTL,M_LIN },
|
|
# endif /* ECHOCTL */
|
|
# ifdef ECHOPRT
|
|
{ "echoprt",ECHOPRT,M_LIN },
|
|
# endif /* ECHOPRT */
|
|
# ifdef ECHOKE
|
|
{ "echoke", ECHOKE, M_LIN },
|
|
# endif /* ECHOKE */
|
|
# ifdef DEFECHO
|
|
{ "defecho",DEFECHO,M_LIN },
|
|
# endif /* DEFECHO */
|
|
# ifdef FLUSHO
|
|
{ "flusho", FLUSHO, M_LIN },
|
|
# endif /* FLUSHO */
|
|
# ifdef PENDIN
|
|
{ "pendin", PENDIN, M_LIN },
|
|
# endif /* PENDIN */
|
|
# ifdef IEXTEN
|
|
{ "iexten", IEXTEN, M_LIN },
|
|
# endif /* IEXTEN */
|
|
# ifdef NOKERNINFO
|
|
{ "nokerninfo",NOKERNINFO,M_LIN },
|
|
# endif /* NOKERNINFO */
|
|
# ifdef ALTWERASE
|
|
{ "altwerase",ALTWERASE,M_LIN },
|
|
# endif /* ALTWERASE */
|
|
# ifdef EXTPROC
|
|
{ "extproc",EXTPROC, M_LIN },
|
|
# endif /* EXTPROC */
|
|
|
|
# if defined(VINTR)
|
|
{ "intr", C_SH(C_INTR), M_CHAR },
|
|
# endif /* VINTR */
|
|
# if defined(VQUIT)
|
|
{ "quit", C_SH(C_QUIT), M_CHAR },
|
|
# endif /* VQUIT */
|
|
# if defined(VERASE)
|
|
{ "erase", C_SH(C_ERASE), M_CHAR },
|
|
# endif /* VERASE */
|
|
# if defined(VKILL)
|
|
{ "kill", C_SH(C_KILL), M_CHAR },
|
|
# endif /* VKILL */
|
|
# if defined(VEOF)
|
|
{ "eof", C_SH(C_EOF), M_CHAR },
|
|
# endif /* VEOF */
|
|
# if defined(VEOL)
|
|
{ "eol", C_SH(C_EOL), M_CHAR },
|
|
# endif /* VEOL */
|
|
# if defined(VEOL2)
|
|
{ "eol2", C_SH(C_EOL2), M_CHAR },
|
|
# endif /* VEOL2 */
|
|
# if defined(VSWTCH)
|
|
{ "swtch", C_SH(C_SWTCH), M_CHAR },
|
|
# endif /* VSWTCH */
|
|
# if defined(VDSWTCH)
|
|
{ "dswtch", C_SH(C_DSWTCH), M_CHAR },
|
|
# endif /* VDSWTCH */
|
|
# if defined(VERASE2)
|
|
{ "erase2", C_SH(C_ERASE2), M_CHAR },
|
|
# endif /* VERASE2 */
|
|
# if defined(VSTART)
|
|
{ "start", C_SH(C_START), M_CHAR },
|
|
# endif /* VSTART */
|
|
# if defined(VSTOP)
|
|
{ "stop", C_SH(C_STOP), M_CHAR },
|
|
# endif /* VSTOP */
|
|
# if defined(VWERASE)
|
|
{ "werase", C_SH(C_WERASE), M_CHAR },
|
|
# endif /* VWERASE */
|
|
# if defined(VSUSP)
|
|
{ "susp", C_SH(C_SUSP), M_CHAR },
|
|
# endif /* VSUSP */
|
|
# if defined(VDSUSP)
|
|
{ "dsusp", C_SH(C_DSUSP), M_CHAR },
|
|
# endif /* VDSUSP */
|
|
# if defined(VREPRINT)
|
|
{ "reprint", C_SH(C_REPRINT),M_CHAR },
|
|
# endif /* VREPRINT */
|
|
# if defined(VDISCARD)
|
|
{ "discard", C_SH(C_DISCARD),M_CHAR },
|
|
# endif /* VDISCARD */
|
|
# if defined(VLNEXT)
|
|
{ "lnext", C_SH(C_LNEXT), M_CHAR },
|
|
# endif /* VLNEXT */
|
|
# if defined(VSTATUS)
|
|
{ "status", C_SH(C_STATUS), M_CHAR },
|
|
# endif /* VSTATUS */
|
|
# if defined(VPAGE)
|
|
{ "page", C_SH(C_PAGE), M_CHAR },
|
|
# endif /* VPAGE */
|
|
# if defined(VPGOFF)
|
|
{ "pgoff", C_SH(C_PGOFF), M_CHAR },
|
|
# endif /* VPGOFF */
|
|
# if defined(VKILL2)
|
|
{ "kill2", C_SH(C_KILL2), M_CHAR },
|
|
# endif /* VKILL2 */
|
|
# if defined(VBRK)
|
|
{ "brk", C_SH(C_BRK), M_CHAR },
|
|
# endif /* VBRK */
|
|
# if defined(VMIN)
|
|
{ "min", C_SH(C_MIN), M_CHAR },
|
|
# endif /* VMIN */
|
|
# if defined(VTIME)
|
|
{ "time", C_SH(C_TIME), M_CHAR },
|
|
# endif /* VTIME */
|
|
{ NULL, 0, -1 },
|
|
};
|
|
|
|
|
|
|
|
#define tty_getty(el, td) tcgetattr((el)->el_infd, (td))
|
|
#define tty_setty(el, td) tcsetattr((el)->el_infd, TCSADRAIN, (td))
|
|
|
|
#define tty__gettabs(td) ((((td)->c_oflag & TAB3) == TAB3) ? 0 : 1)
|
|
#define tty__geteightbit(td) (((td)->c_cflag & CSIZE) == CS8)
|
|
#define tty__cooked_mode(td) ((td)->c_lflag & ICANON)
|
|
|
|
private void tty__getchar __P((struct termios *, unsigned char *));
|
|
private void tty__setchar __P((struct termios *, unsigned char *));
|
|
private speed_t tty__getspeed __P((struct termios *));
|
|
private int tty_setup __P((EditLine *));
|
|
|
|
#define t_qu t_ts
|
|
|
|
|
|
/* tty_setup():
|
|
* Get the tty parameters and initialize the editing state
|
|
*/
|
|
private int
|
|
tty_setup(el)
|
|
EditLine *el;
|
|
{
|
|
int rst = 1;
|
|
if (tty_getty(el, &el->el_tty.t_ed) == -1) {
|
|
#ifdef DEBUG_TTY
|
|
(void) fprintf(el->el_errfile,
|
|
"tty_setup: tty_getty: %s\n", strerror(errno));
|
|
#endif /* DEBUG_TTY */
|
|
return(-1);
|
|
}
|
|
el->el_tty.t_ts = el->el_tty.t_ex = el->el_tty.t_ed;
|
|
|
|
el->el_tty.t_speed = tty__getspeed(&el->el_tty.t_ex);
|
|
el->el_tty.t_tabs = tty__gettabs(&el->el_tty.t_ex);
|
|
el->el_tty.t_eight = tty__geteightbit(&el->el_tty.t_ex);
|
|
|
|
el->el_tty.t_ex.c_iflag &= ~el->el_tty.t_t[EX_IO][M_INP].t_clrmask;
|
|
el->el_tty.t_ex.c_iflag |= el->el_tty.t_t[EX_IO][M_INP].t_setmask;
|
|
|
|
el->el_tty.t_ex.c_oflag &= ~el->el_tty.t_t[EX_IO][M_OUT].t_clrmask;
|
|
el->el_tty.t_ex.c_oflag |= el->el_tty.t_t[EX_IO][M_OUT].t_setmask;
|
|
|
|
el->el_tty.t_ex.c_cflag &= ~el->el_tty.t_t[EX_IO][M_CTL].t_clrmask;
|
|
el->el_tty.t_ex.c_cflag |= el->el_tty.t_t[EX_IO][M_CTL].t_setmask;
|
|
|
|
el->el_tty.t_ex.c_lflag &= ~el->el_tty.t_t[EX_IO][M_LIN].t_clrmask;
|
|
el->el_tty.t_ex.c_lflag |= el->el_tty.t_t[EX_IO][M_LIN].t_setmask;
|
|
|
|
/*
|
|
* Reset the tty chars to reasonable defaults
|
|
* If they are disabled, then enable them.
|
|
*/
|
|
if (rst) {
|
|
if (tty__cooked_mode(&el->el_tty.t_ts)) {
|
|
tty__getchar(&el->el_tty.t_ts, el->el_tty.t_c[TS_IO]);
|
|
/*
|
|
* Don't affect CMIN and CTIME for the editor mode
|
|
*/
|
|
for (rst = 0; rst < C_NCC - 2; rst++)
|
|
if (el->el_tty.t_c[TS_IO][rst] != el->el_tty.t_vdisable &&
|
|
el->el_tty.t_c[ED_IO][rst] != el->el_tty.t_vdisable)
|
|
el->el_tty.t_c[ED_IO][rst] = el->el_tty.t_c[TS_IO][rst];
|
|
for (rst = 0; rst < C_NCC; rst++)
|
|
if (el->el_tty.t_c[TS_IO][rst] != el->el_tty.t_vdisable &&
|
|
el->el_tty.t_c[EX_IO][rst] != el->el_tty.t_vdisable)
|
|
el->el_tty.t_c[EX_IO][rst] = el->el_tty.t_c[TS_IO][rst];
|
|
}
|
|
tty__setchar(&el->el_tty.t_ex, el->el_tty.t_c[EX_IO]);
|
|
if (tty_setty(el, &el->el_tty.t_ex) == -1) {
|
|
#ifdef DEBUG_TTY
|
|
(void) fprintf(el->el_errfile, "tty_setup: tty_setty: %s\n",
|
|
strerror(errno));
|
|
#endif /* DEBUG_TTY */
|
|
return(-1);
|
|
}
|
|
}
|
|
else
|
|
tty__setchar(&el->el_tty.t_ex, el->el_tty.t_c[EX_IO]);
|
|
|
|
el->el_tty.t_ed.c_iflag &= ~el->el_tty.t_t[ED_IO][M_INP].t_clrmask;
|
|
el->el_tty.t_ed.c_iflag |= el->el_tty.t_t[ED_IO][M_INP].t_setmask;
|
|
|
|
el->el_tty.t_ed.c_oflag &= ~el->el_tty.t_t[ED_IO][M_OUT].t_clrmask;
|
|
el->el_tty.t_ed.c_oflag |= el->el_tty.t_t[ED_IO][M_OUT].t_setmask;
|
|
|
|
el->el_tty.t_ed.c_cflag &= ~el->el_tty.t_t[ED_IO][M_CTL].t_clrmask;
|
|
el->el_tty.t_ed.c_cflag |= el->el_tty.t_t[ED_IO][M_CTL].t_setmask;
|
|
|
|
el->el_tty.t_ed.c_lflag &= ~el->el_tty.t_t[ED_IO][M_LIN].t_clrmask;
|
|
el->el_tty.t_ed.c_lflag |= el->el_tty.t_t[ED_IO][M_LIN].t_setmask;
|
|
|
|
tty__setchar(&el->el_tty.t_ed, el->el_tty.t_c[ED_IO]);
|
|
return 0;
|
|
}
|
|
|
|
protected int
|
|
tty_init(el)
|
|
EditLine *el;
|
|
{
|
|
el->el_tty.t_mode = EX_IO;
|
|
el->el_tty.t_vdisable = _POSIX_VDISABLE;
|
|
(void) memcpy(el->el_tty.t_t, ttyperm, sizeof(ttyperm_t));
|
|
(void) memcpy(el->el_tty.t_c, ttychar, sizeof(ttychar_t));
|
|
return tty_setup(el);
|
|
} /* end tty_init */
|
|
|
|
|
|
/* tty_end():
|
|
* Restore the tty to its original settings
|
|
*/
|
|
protected void
|
|
/*ARGSUSED*/
|
|
tty_end(el)
|
|
EditLine *el;
|
|
{
|
|
/* XXX: Maybe reset to an initial state? */
|
|
}
|
|
|
|
|
|
/* tty__getspeed():
|
|
* Get the tty speed
|
|
*/
|
|
private speed_t
|
|
tty__getspeed(td)
|
|
struct termios *td;
|
|
{
|
|
speed_t spd;
|
|
|
|
if ((spd = cfgetispeed(td)) == 0)
|
|
spd = cfgetospeed(td);
|
|
return spd;
|
|
} /* end tty__getspeed */
|
|
|
|
|
|
/* tty__getchar():
|
|
* Get the tty characters
|
|
*/
|
|
private void
|
|
tty__getchar(td, s)
|
|
struct termios *td;
|
|
unsigned char *s;
|
|
{
|
|
# ifdef VINTR
|
|
s[C_INTR] = td->c_cc[VINTR];
|
|
# endif /* VINTR */
|
|
# ifdef VQUIT
|
|
s[C_QUIT] = td->c_cc[VQUIT];
|
|
# endif /* VQUIT */
|
|
# ifdef VERASE
|
|
s[C_ERASE] = td->c_cc[VERASE];
|
|
# endif /* VERASE */
|
|
# ifdef VKILL
|
|
s[C_KILL] = td->c_cc[VKILL];
|
|
# endif /* VKILL */
|
|
# ifdef VEOF
|
|
s[C_EOF] = td->c_cc[VEOF];
|
|
# endif /* VEOF */
|
|
# ifdef VEOL
|
|
s[C_EOL] = td->c_cc[VEOL];
|
|
# endif /* VEOL */
|
|
# ifdef VEOL2
|
|
s[C_EOL2] = td->c_cc[VEOL2];
|
|
# endif /* VEOL2 */
|
|
# ifdef VSWTCH
|
|
s[C_SWTCH] = td->c_cc[VSWTCH];
|
|
# endif /* VSWTCH */
|
|
# ifdef VDSWTCH
|
|
s[C_DSWTCH] = td->c_cc[VDSWTCH];
|
|
# endif /* VDSWTCH */
|
|
# ifdef VERASE2
|
|
s[C_ERASE2] = td->c_cc[VERASE2];
|
|
# endif /* VERASE2 */
|
|
# ifdef VSTART
|
|
s[C_START] = td->c_cc[VSTART];
|
|
# endif /* VSTART */
|
|
# ifdef VSTOP
|
|
s[C_STOP] = td->c_cc[VSTOP];
|
|
# endif /* VSTOP */
|
|
# ifdef VWERASE
|
|
s[C_WERASE] = td->c_cc[VWERASE];
|
|
# endif /* VWERASE */
|
|
# ifdef VSUSP
|
|
s[C_SUSP] = td->c_cc[VSUSP];
|
|
# endif /* VSUSP */
|
|
# ifdef VDSUSP
|
|
s[C_DSUSP] = td->c_cc[VDSUSP];
|
|
# endif /* VDSUSP */
|
|
# ifdef VREPRINT
|
|
s[C_REPRINT]= td->c_cc[VREPRINT];
|
|
# endif /* VREPRINT */
|
|
# ifdef VDISCARD
|
|
s[C_DISCARD]= td->c_cc[VDISCARD];
|
|
# endif /* VDISCARD */
|
|
# ifdef VLNEXT
|
|
s[C_LNEXT] = td->c_cc[VLNEXT];
|
|
# endif /* VLNEXT */
|
|
# ifdef VSTATUS
|
|
s[C_STATUS] = td->c_cc[VSTATUS];
|
|
# endif /* VSTATUS */
|
|
# ifdef VPAGE
|
|
s[C_PAGE] = td->c_cc[VPAGE];
|
|
# endif /* VPAGE */
|
|
# ifdef VPGOFF
|
|
s[C_PGOFF] = td->c_cc[VPGOFF];
|
|
# endif /* VPGOFF */
|
|
# ifdef VKILL2
|
|
s[C_KILL2] = td->c_cc[VKILL2];
|
|
# endif /* KILL2 */
|
|
# ifdef VMIN
|
|
s[C_MIN] = td->c_cc[VMIN];
|
|
# endif /* VMIN */
|
|
# ifdef VTIME
|
|
s[C_TIME] = td->c_cc[VTIME];
|
|
# endif /* VTIME */
|
|
} /* tty__getchar */
|
|
|
|
|
|
/* tty__setchar():
|
|
* Set the tty characters
|
|
*/
|
|
private void
|
|
tty__setchar(td, s)
|
|
struct termios *td;
|
|
unsigned char *s;
|
|
{
|
|
# ifdef VINTR
|
|
td->c_cc[VINTR] = s[C_INTR];
|
|
# endif /* VINTR */
|
|
# ifdef VQUIT
|
|
td->c_cc[VQUIT] = s[C_QUIT];
|
|
# endif /* VQUIT */
|
|
# ifdef VERASE
|
|
td->c_cc[VERASE] = s[C_ERASE];
|
|
# endif /* VERASE */
|
|
# ifdef VKILL
|
|
td->c_cc[VKILL] = s[C_KILL];
|
|
# endif /* VKILL */
|
|
# ifdef VEOF
|
|
td->c_cc[VEOF] = s[C_EOF];
|
|
# endif /* VEOF */
|
|
# ifdef VEOL
|
|
td->c_cc[VEOL] = s[C_EOL];
|
|
# endif /* VEOL */
|
|
# ifdef VEOL2
|
|
td->c_cc[VEOL2] = s[C_EOL2];
|
|
# endif /* VEOL2 */
|
|
# ifdef VSWTCH
|
|
td->c_cc[VSWTCH] = s[C_SWTCH];
|
|
# endif /* VSWTCH */
|
|
# ifdef VDSWTCH
|
|
td->c_cc[VDSWTCH] = s[C_DSWTCH];
|
|
# endif /* VDSWTCH */
|
|
# ifdef VERASE2
|
|
td->c_cc[VERASE2] = s[C_ERASE2];
|
|
# endif /* VERASE2 */
|
|
# ifdef VSTART
|
|
td->c_cc[VSTART] = s[C_START];
|
|
# endif /* VSTART */
|
|
# ifdef VSTOP
|
|
td->c_cc[VSTOP] = s[C_STOP];
|
|
# endif /* VSTOP */
|
|
# ifdef VWERASE
|
|
td->c_cc[VWERASE] = s[C_WERASE];
|
|
# endif /* VWERASE */
|
|
# ifdef VSUSP
|
|
td->c_cc[VSUSP] = s[C_SUSP];
|
|
# endif /* VSUSP */
|
|
# ifdef VDSUSP
|
|
td->c_cc[VDSUSP] = s[C_DSUSP];
|
|
# endif /* VDSUSP */
|
|
# ifdef VREPRINT
|
|
td->c_cc[VREPRINT] = s[C_REPRINT];
|
|
# endif /* VREPRINT */
|
|
# ifdef VDISCARD
|
|
td->c_cc[VDISCARD] = s[C_DISCARD];
|
|
# endif /* VDISCARD */
|
|
# ifdef VLNEXT
|
|
td->c_cc[VLNEXT] = s[C_LNEXT];
|
|
# endif /* VLNEXT */
|
|
# ifdef VSTATUS
|
|
td->c_cc[VSTATUS] = s[C_STATUS];
|
|
# endif /* VSTATUS */
|
|
# ifdef VPAGE
|
|
td->c_cc[VPAGE] = s[C_PAGE];
|
|
# endif /* VPAGE */
|
|
# ifdef VPGOFF
|
|
td->c_cc[VPGOFF] = s[C_PGOFF];
|
|
# endif /* VPGOFF */
|
|
# ifdef VKILL2
|
|
td->c_cc[VKILL2] = s[C_KILL2];
|
|
# endif /* VKILL2 */
|
|
# ifdef VMIN
|
|
td->c_cc[VMIN] = s[C_MIN];
|
|
# endif /* VMIN */
|
|
# ifdef VTIME
|
|
td->c_cc[VTIME] = s[C_TIME];
|
|
# endif /* VTIME */
|
|
} /* tty__setchar */
|
|
|
|
|
|
/* tty_bind_char():
|
|
* Rebind the editline functions
|
|
*/
|
|
protected void
|
|
tty_bind_char(el, force)
|
|
EditLine *el;
|
|
int force;
|
|
{
|
|
unsigned char *t_n = el->el_tty.t_c[ED_IO];
|
|
unsigned char *t_o = el->el_tty.t_ed.c_cc;
|
|
char new[2], old[2];
|
|
ttymap_t *tp;
|
|
el_action_t *dmap, *dalt, *map, *alt;
|
|
new[1] = old[1] = '\0';
|
|
|
|
|
|
map = el->el_map.key;
|
|
alt = el->el_map.alt;
|
|
if (el->el_map.type == MAP_VI) {
|
|
dmap = el->el_map.vii;
|
|
dalt = el->el_map.vic;
|
|
}
|
|
else {
|
|
dmap = el->el_map.emacs;
|
|
dalt = NULL;
|
|
}
|
|
|
|
for (tp = tty_map; tp->nch != -1; tp++) {
|
|
new[0] = t_n[tp->nch];
|
|
old[0] = t_o[tp->och];
|
|
if (new[0] == old[0] && !force)
|
|
continue;
|
|
/* Put the old default binding back, and set the new binding */
|
|
key_clear(el, map, old);
|
|
map[old[0]] = dmap[old[0]];
|
|
key_clear(el, map, new);
|
|
/* MAP_VI == 1, MAP_EMACS == 0... */
|
|
map[new[0]] = tp->bind[el->el_map.type];
|
|
if (dalt) {
|
|
key_clear(el, alt, old);
|
|
alt[old[0]] = dalt[old[0]];
|
|
key_clear(el, alt, new);
|
|
alt[new[0]] = tp->bind[el->el_map.type+1];
|
|
}
|
|
}
|
|
}
|
|
|
|
/* tty_rawmode():
|
|
* Set terminal into 1 character at a time mode.
|
|
*/
|
|
protected int
|
|
tty_rawmode(el)
|
|
EditLine *el;
|
|
{
|
|
if (el->el_tty.t_mode == ED_IO)
|
|
return (0);
|
|
|
|
if (tty_getty(el, &el->el_tty.t_ts) == -1) {
|
|
#ifdef DEBUG_TTY
|
|
(void) fprintf(el->el_errfile, "tty_rawmode: tty_getty: %s\n", strerror(errno));
|
|
#endif /* DEBUG_TTY */
|
|
return(-1);
|
|
}
|
|
|
|
/*
|
|
* We always keep up with the eight bit setting and the speed of the
|
|
* tty. But only we only believe changes that are made to cooked mode!
|
|
*/
|
|
el->el_tty.t_eight = tty__geteightbit(&el->el_tty.t_ts);
|
|
el->el_tty.t_speed = tty__getspeed(&el->el_tty.t_ts);
|
|
|
|
if (tty__getspeed(&el->el_tty.t_ex) != el->el_tty.t_speed ||
|
|
tty__getspeed(&el->el_tty.t_ed) != el->el_tty.t_speed) {
|
|
(void) cfsetispeed(&el->el_tty.t_ex, el->el_tty.t_speed);
|
|
(void) cfsetospeed(&el->el_tty.t_ex, el->el_tty.t_speed);
|
|
(void) cfsetispeed(&el->el_tty.t_ed, el->el_tty.t_speed);
|
|
(void) cfsetospeed(&el->el_tty.t_ed, el->el_tty.t_speed);
|
|
}
|
|
|
|
if (tty__cooked_mode(&el->el_tty.t_ts)) {
|
|
if (el->el_tty.t_ts.c_cflag != el->el_tty.t_ex.c_cflag) {
|
|
el->el_tty.t_ex.c_cflag = el->el_tty.t_ts.c_cflag;
|
|
el->el_tty.t_ex.c_cflag &= ~el->el_tty.t_t[EX_IO][M_CTL].t_clrmask;
|
|
el->el_tty.t_ex.c_cflag |= el->el_tty.t_t[EX_IO][M_CTL].t_setmask;
|
|
|
|
el->el_tty.t_ed.c_cflag = el->el_tty.t_ts.c_cflag;
|
|
el->el_tty.t_ed.c_cflag &= ~el->el_tty.t_t[ED_IO][M_CTL].t_clrmask;
|
|
el->el_tty.t_ed.c_cflag |= el->el_tty.t_t[ED_IO][M_CTL].t_setmask;
|
|
}
|
|
|
|
if ((el->el_tty.t_ts.c_lflag != el->el_tty.t_ex.c_lflag) &&
|
|
(el->el_tty.t_ts.c_lflag != el->el_tty.t_ed.c_lflag)) {
|
|
el->el_tty.t_ex.c_lflag = el->el_tty.t_ts.c_lflag;
|
|
el->el_tty.t_ex.c_lflag &= ~el->el_tty.t_t[EX_IO][M_LIN].t_clrmask;
|
|
el->el_tty.t_ex.c_lflag |= el->el_tty.t_t[EX_IO][M_LIN].t_setmask;
|
|
|
|
el->el_tty.t_ed.c_lflag = el->el_tty.t_ts.c_lflag;
|
|
el->el_tty.t_ed.c_lflag &= ~el->el_tty.t_t[ED_IO][M_LIN].t_clrmask;
|
|
el->el_tty.t_ed.c_lflag |= el->el_tty.t_t[ED_IO][M_LIN].t_setmask;
|
|
}
|
|
|
|
if ((el->el_tty.t_ts.c_iflag != el->el_tty.t_ex.c_iflag) &&
|
|
(el->el_tty.t_ts.c_iflag != el->el_tty.t_ed.c_iflag)) {
|
|
el->el_tty.t_ex.c_iflag = el->el_tty.t_ts.c_iflag;
|
|
el->el_tty.t_ex.c_iflag &= ~el->el_tty.t_t[EX_IO][M_INP].t_clrmask;
|
|
el->el_tty.t_ex.c_iflag |= el->el_tty.t_t[EX_IO][M_INP].t_setmask;
|
|
|
|
el->el_tty.t_ed.c_iflag = el->el_tty.t_ts.c_iflag;
|
|
el->el_tty.t_ed.c_iflag &= ~el->el_tty.t_t[ED_IO][M_INP].t_clrmask;
|
|
el->el_tty.t_ed.c_iflag |= el->el_tty.t_t[ED_IO][M_INP].t_setmask;
|
|
}
|
|
|
|
if ((el->el_tty.t_ts.c_oflag != el->el_tty.t_ex.c_oflag) &&
|
|
(el->el_tty.t_ts.c_oflag != el->el_tty.t_ed.c_oflag)) {
|
|
el->el_tty.t_ex.c_oflag = el->el_tty.t_ts.c_oflag;
|
|
el->el_tty.t_ex.c_oflag &= ~el->el_tty.t_t[EX_IO][M_OUT].t_clrmask;
|
|
el->el_tty.t_ex.c_oflag |= el->el_tty.t_t[EX_IO][M_OUT].t_setmask;
|
|
|
|
el->el_tty.t_ed.c_oflag = el->el_tty.t_ts.c_oflag;
|
|
el->el_tty.t_ed.c_oflag &= ~el->el_tty.t_t[ED_IO][M_OUT].t_clrmask;
|
|
el->el_tty.t_ed.c_oflag |= el->el_tty.t_t[ED_IO][M_OUT].t_setmask;
|
|
}
|
|
|
|
if (tty__gettabs(&el->el_tty.t_ex) == 0)
|
|
el->el_tty.t_tabs = 0;
|
|
else
|
|
el->el_tty.t_tabs = EL_CAN_TAB ? 1 : 0;
|
|
|
|
{
|
|
int i;
|
|
|
|
tty__getchar(&el->el_tty.t_ts, el->el_tty.t_c[TS_IO]);
|
|
/*
|
|
* Check if the user made any changes.
|
|
* If he did, then propagate the changes to the
|
|
* edit and execute data structures.
|
|
*/
|
|
for (i = 0; i < C_NCC; i++)
|
|
if (el->el_tty.t_c[TS_IO][i] != el->el_tty.t_c[EX_IO][i])
|
|
break;
|
|
|
|
if (i != C_NCC) {
|
|
/*
|
|
* Propagate changes only to the unprotected chars
|
|
* that have been modified just now.
|
|
*/
|
|
for (i = 0; i < C_NCC; i++) {
|
|
if (!((el->el_tty.t_t[ED_IO][M_CHAR].t_setmask & C_SH(i)))
|
|
&& (el->el_tty.t_c[TS_IO][i] != el->el_tty.t_c[EX_IO][i]))
|
|
el->el_tty.t_c[ED_IO][i] = el->el_tty.t_c[TS_IO][i];
|
|
if (el->el_tty.t_t[ED_IO][M_CHAR].t_clrmask & C_SH(i))
|
|
el->el_tty.t_c[ED_IO][i] = el->el_tty.t_vdisable;
|
|
}
|
|
tty_bind_char(el, 0);
|
|
tty__setchar(&el->el_tty.t_ed, el->el_tty.t_c[ED_IO]);
|
|
|
|
for (i = 0; i < C_NCC; i++) {
|
|
if (!((el->el_tty.t_t[EX_IO][M_CHAR].t_setmask & C_SH(i)))
|
|
&& (el->el_tty.t_c[TS_IO][i] != el->el_tty.t_c[EX_IO][i]))
|
|
el->el_tty.t_c[EX_IO][i] = el->el_tty.t_c[TS_IO][i];
|
|
if (el->el_tty.t_t[EX_IO][M_CHAR].t_clrmask & C_SH(i))
|
|
el->el_tty.t_c[EX_IO][i] = el->el_tty.t_vdisable;
|
|
}
|
|
tty__setchar(&el->el_tty.t_ex, el->el_tty.t_c[EX_IO]);
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
if (tty_setty(el, &el->el_tty.t_ed) == -1) {
|
|
#ifdef DEBUG_TTY
|
|
(void) fprintf(el->el_errfile, "tty_rawmode: tty_setty: %s\n",
|
|
strerror(errno));
|
|
#endif /* DEBUG_TTY */
|
|
return -1;
|
|
}
|
|
el->el_tty.t_mode = ED_IO;
|
|
return (0);
|
|
} /* end tty_rawmode */
|
|
|
|
|
|
/* tty_cookedmode():
|
|
* Set the tty back to normal mode
|
|
*/
|
|
protected int
|
|
tty_cookedmode(el)
|
|
EditLine *el;
|
|
{ /* set tty in normal setup */
|
|
if (el->el_tty.t_mode == EX_IO)
|
|
return (0);
|
|
|
|
if (tty_setty(el, &el->el_tty.t_ex) == -1) {
|
|
#ifdef DEBUG_TTY
|
|
(void) fprintf(el->el_errfile, "tty_cookedmode: tty_setty: %s\n",
|
|
strerror(errno));
|
|
#endif /* DEBUG_TTY */
|
|
return -1;
|
|
}
|
|
el->el_tty.t_mode = EX_IO;
|
|
return (0);
|
|
} /* end tty_cookedmode */
|
|
|
|
|
|
/* tty_quotemode():
|
|
* Turn on quote mode
|
|
*/
|
|
protected int
|
|
tty_quotemode(el)
|
|
EditLine *el;
|
|
{
|
|
if (el->el_tty.t_mode == QU_IO)
|
|
return 0;
|
|
|
|
el->el_tty.t_qu = el->el_tty.t_ed;
|
|
|
|
el->el_tty.t_qu.c_iflag &= ~el->el_tty.t_t[QU_IO][M_INP].t_clrmask;
|
|
el->el_tty.t_qu.c_iflag |= el->el_tty.t_t[QU_IO][M_INP].t_setmask;
|
|
|
|
el->el_tty.t_qu.c_oflag &= ~el->el_tty.t_t[QU_IO][M_OUT].t_clrmask;
|
|
el->el_tty.t_qu.c_oflag |= el->el_tty.t_t[QU_IO][M_OUT].t_setmask;
|
|
|
|
el->el_tty.t_qu.c_cflag &= ~el->el_tty.t_t[QU_IO][M_CTL].t_clrmask;
|
|
el->el_tty.t_qu.c_cflag |= el->el_tty.t_t[QU_IO][M_CTL].t_setmask;
|
|
|
|
el->el_tty.t_qu.c_lflag &= ~el->el_tty.t_t[QU_IO][M_LIN].t_clrmask;
|
|
el->el_tty.t_qu.c_lflag |= el->el_tty.t_t[QU_IO][M_LIN].t_setmask;
|
|
|
|
if (tty_setty(el, &el->el_tty.t_qu) == -1) {
|
|
#ifdef DEBUG_TTY
|
|
(void) fprintf(el->el_errfile, "QuoteModeOn: tty_setty: %s\n",
|
|
strerror(errno));
|
|
#endif /* DEBUG_TTY */
|
|
return -1;
|
|
}
|
|
el->el_tty.t_mode = QU_IO;
|
|
return 0;
|
|
} /* end tty_quotemode */
|
|
|
|
|
|
/* tty_noquotemode():
|
|
* Turn off quote mode
|
|
*/
|
|
protected int
|
|
tty_noquotemode(el)
|
|
EditLine *el;
|
|
{
|
|
if (el->el_tty.t_mode != QU_IO)
|
|
return 0;
|
|
if (tty_setty(el, &el->el_tty.t_ed) == -1) {
|
|
#ifdef DEBUG_TTY
|
|
(void) fprintf(el->el_errfile, "QuoteModeOff: tty_setty: %s\n",
|
|
strerror(errno));
|
|
#endif /* DEBUG_TTY */
|
|
return -1;
|
|
}
|
|
el->el_tty.t_mode = ED_IO;
|
|
return 0;
|
|
}
|
|
|
|
/* tty_stty():
|
|
* Stty builtin
|
|
*/
|
|
protected int
|
|
/*ARGSUSED*/
|
|
tty_stty(el, argc, argv)
|
|
EditLine *el;
|
|
int argc;
|
|
char **argv;
|
|
{
|
|
ttymodes_t *m;
|
|
char x, *d;
|
|
int aflag = 0;
|
|
char *s;
|
|
char *name;
|
|
int z = EX_IO;
|
|
|
|
if (argv == NULL)
|
|
return -1;
|
|
name = *argv++;
|
|
|
|
while (argv && *argv && argv[0][0] == '-' && argv[0][2] == '\0')
|
|
switch (argv[0][1]) {
|
|
case 'a':
|
|
aflag++;
|
|
argv++;
|
|
break;
|
|
case 'd':
|
|
argv++;
|
|
z = ED_IO;
|
|
break;
|
|
case 'x':
|
|
argv++;
|
|
z = EX_IO;
|
|
break;
|
|
case 'q':
|
|
argv++;
|
|
z = QU_IO;
|
|
break;
|
|
default:
|
|
(void) fprintf(el->el_errfile, "%s: Unknown switch `%c'.\n",
|
|
name, argv[0][1]);
|
|
return -1;
|
|
}
|
|
|
|
if (!argv || !*argv) {
|
|
int i = -1;
|
|
int len = 0, st = 0, cu;
|
|
for (m = ttymodes; m->m_name; m++) {
|
|
if (m->m_type != i) {
|
|
(void) fprintf(el->el_outfile, "%s%s", i != -1 ? "\n" : "",
|
|
el->el_tty.t_t[z][m->m_type].t_name);
|
|
i = m->m_type;
|
|
st = len = strlen(el->el_tty.t_t[z][m->m_type].t_name);
|
|
}
|
|
|
|
x = (el->el_tty.t_t[z][i].t_setmask & m->m_value) ? '+' : '\0';
|
|
x = (el->el_tty.t_t[z][i].t_clrmask & m->m_value) ? '-' : x;
|
|
|
|
if (x != '\0' || aflag) {
|
|
|
|
cu = strlen(m->m_name) + (x != '\0') + 1;
|
|
|
|
if (len + cu >= el->el_term.t_size.h) {
|
|
(void) fprintf(el->el_outfile, "\n%*s", st, "");
|
|
len = st + cu;
|
|
}
|
|
else
|
|
len += cu;
|
|
|
|
if (x != '\0')
|
|
(void) fprintf(el->el_outfile, "%c%s ", x, m->m_name);
|
|
else
|
|
(void) fprintf(el->el_outfile, "%s ", m->m_name);
|
|
}
|
|
}
|
|
(void) fprintf(el->el_outfile, "\n");
|
|
return 0;
|
|
}
|
|
|
|
while (argv && (s = *argv++)) {
|
|
switch (*s) {
|
|
case '+':
|
|
case '-':
|
|
x = *s++;
|
|
break;
|
|
default:
|
|
x = '\0';
|
|
break;
|
|
}
|
|
d = s;
|
|
for (m = ttymodes; m->m_name; m++)
|
|
if (strcmp(m->m_name, d) == 0)
|
|
break;
|
|
|
|
if (!m->m_name) {
|
|
(void) fprintf(el->el_errfile, "%s: Invalid argument `%s'.\n",
|
|
name, d);
|
|
return -1;
|
|
}
|
|
|
|
switch (x) {
|
|
case '+':
|
|
el->el_tty.t_t[z][m->m_type].t_setmask |= m->m_value;
|
|
el->el_tty.t_t[z][m->m_type].t_clrmask &= ~m->m_value;
|
|
break;
|
|
case '-':
|
|
el->el_tty.t_t[z][m->m_type].t_setmask &= ~m->m_value;
|
|
el->el_tty.t_t[z][m->m_type].t_clrmask |= m->m_value;
|
|
break;
|
|
default:
|
|
el->el_tty.t_t[z][m->m_type].t_setmask &= ~m->m_value;
|
|
el->el_tty.t_t[z][m->m_type].t_clrmask &= ~m->m_value;
|
|
break;
|
|
}
|
|
}
|
|
return 0;
|
|
} /* end tty_stty */
|
|
|
|
|
|
#ifdef notyet
|
|
/* tty_printchar():
|
|
* DEbugging routine to print the tty characters
|
|
*/
|
|
private void
|
|
tty_printchar(el, s)
|
|
EditLine *el;
|
|
unsigned char *s;
|
|
{
|
|
ttyperm_t *m;
|
|
int i;
|
|
|
|
for (i = 0; i < C_NCC; i++) {
|
|
for (m = el->el_tty.t_t; m->m_name; m++)
|
|
if (m->m_type == M_CHAR && C_SH(i) == m->m_value)
|
|
break;
|
|
if (m->m_name)
|
|
(void) fprintf(el->el_errfile, "%s ^%c ", m->m_name, s[i] + 'A'-1);
|
|
if (i % 5 == 0)
|
|
(void) fprintf(el->el_errfile, "\n");
|
|
}
|
|
(void) fprintf(el->el_errfile, "\n");
|
|
}
|
|
#endif /* notyet */
|