2 * Copyright (c) 1988, 1993
3 * The Regents of the University of California. All rights reserved.
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions
8 * 1. Redistributions of source code must retain the above copyright
9 * notice, this list of conditions and the following disclaimer.
10 * 2. Redistributions in binary form must reproduce the above copyright
11 * notice, this list of conditions and the following disclaimer in the
12 * documentation and/or other materials provided with the distribution.
13 * 3. All advertising materials mentioning features or use of this software
14 * must display the following acknowledgement:
15 * This product includes software developed by the University of
16 * California, Berkeley and its contributors.
17 * 4. Neither the name of the University nor the names of its contributors
18 * may be used to endorse or promote products derived from this software
19 * without specific prior written permission.
21 * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
22 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
23 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
24 * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
25 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
26 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
27 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
28 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
29 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
30 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
33 * @(#)hack.tty.c 8.1 (Berkeley) 5/31/93
34 * $FreeBSD: src/games/hack/hack.tty.c,v 1.6.2.1 2000/07/20 10:35:07 kris Exp $
35 * $DragonFly: src/games/hack/hack.tty.c,v 1.4 2006/08/21 19:45:32 pavalos Exp $
38 /* Copyright (c) Stichting Mathematisch Centrum, Amsterdam, 1985. */
39 /* hack.tty.c - version 1.0.3 */
40 /* With thanks to the people who sent code for SYSV - hpscdi!jon,
41 arnold@ucsf-cgl, wcs@bo95b, cbcephus!pds and others. */
46 * The distinctions here are not BSD - rest but rather USG - rest, as
47 * BSD still has the old sgttyb structure, but SYSV has termio. Thus:
56 * Some systems may have getchar() return EOF for various reasons, and
57 * we should not quit before seeing at least NR_OF_EOFS consecutive EOFs.
67 #define termstruct termio
68 #define kill_sym c_cc[VKILL]
69 #define erase_sym c_cc[VERASE]
71 #define tabflgs c_oflag
72 #define echoflgs c_lflag
73 #define cbrkflgs c_lflag
74 #define CBRKMASK ICANON
75 #define CBRKON ! /* reverse condition */
76 #define OSPEED(x) ((x).c_cflag & CBAUD)
77 #define GTTY(x) (ioctl(0, TCGETA, x))
78 #define STTY(x) (ioctl(0, TCSETA, x)) /* TCSETAF? TCSETAW? */
83 #define termstruct sgttyb
84 #define kill_sym sg_kill
85 #define erase_sym sg_erase
87 #define tabflgs sg_flags
88 #define echoflgs sg_flags
89 #define cbrkflgs sg_flags
90 #define CBRKMASK CBREAK
91 #define CBRKON /* empty */
92 #define OSPEED(x) (x).sg_ospeed
93 #define GTTY(x) (ioctl(0, TIOCGETP, x))
94 #define STTY(x) (ioctl(0, TIOCSETP, x))
98 static char erase_char
, kill_char
;
99 static boolean settty_needed
= FALSE
;
100 struct termstruct inittyb
, curttyb
;
102 static void setctty(void);
105 * Get initial state of terminal, set ospeed (for termcap routines)
106 * and switch off tab expansion if necessary.
107 * Called by startup() in termcap.c and after returning from ! or ^Z
112 if(GTTY(&inittyb
) < 0)
113 perror("Hack (gettty)");
115 erase_char
= inittyb
.erase_sym
;
116 kill_char
= inittyb
.kill_sym
;
119 /* do not expand tabs - they might be needed inside a cm sequence */
120 if(curttyb
.tabflgs
& EXTABS
) {
121 curttyb
.tabflgs
&= ~EXTABS
;
124 settty_needed
= TRUE
;
127 /* reset terminal to original state */
129 settty(const char *s
)
133 if(s
) printf("%s", s
);
135 if(STTY(&inittyb
) < 0)
136 perror("Hack (settty)");
137 flags
.echo
= (inittyb
.echoflgs
& ECHO
) ? ON
: OFF
;
138 flags
.cbreak
= (CBRKON(inittyb
.cbrkflgs
& CBRKMASK
)) ? ON
: OFF
;
145 if(STTY(&curttyb
) < 0)
146 perror("Hack (setctty)");
152 int ef
= 0; /* desired value of flags & ECHO */
153 int cf
= CBRKON(CBRKMASK
); /* desired value of flags & CBREAK */
157 /* Should use (ECHO|CRMOD) here instead of ECHO */
158 if((curttyb
.echoflgs
& ECHO
) != ef
){
159 curttyb
.echoflgs
&= ~ECHO
;
162 if((curttyb
.cbrkflgs
& CBRKMASK
) != cf
){
163 curttyb
.cbrkflgs
&= ~CBRKMASK
;
164 curttyb
.cbrkflgs
|= cf
;
166 /* be satisfied with one character; no timeout */
167 curttyb
.c_cc
[VMIN
] = 1; /* was VEOF */
168 curttyb
.c_cc
[VTIME
] = 0; /* was VEOL */
182 error(const char *s
, ...)
195 * Read a line closed with '\n' into the array char bufp[BUFSZ].
196 * (The '\n' is not stored. The string is closed with a '\0'.)
197 * Reading can be interrupted by an escape ('\033') - now the
198 * resulting string is "\033".
206 flags
.toplin
= 2; /* nonempty, no --More-- required */
209 if((c
= getchar()) == EOF
) {
218 if(c
== erase_char
|| c
== '\b') {
221 putstr("\b \b"); /* putsym converts \b */
223 } else if(c
== '\n') {
226 } else if(' ' <= c
&& c
< '\177') {
227 /* avoid isprint() - some people don't have it
228 ' ' is not always a printing char */
232 if(bufp
-obufp
< BUFSZ
-1 && bufp
-obufp
< COLNO
)
234 } else if(c
== kill_char
|| c
== '\177') { /* Robert Viduya */
235 /* this test last - @ might be the kill_char */
236 while(bufp
!= obufp
) {
252 cgetret(const char *s
)
258 putstr(flags
.cbreak
? "space" : "return");
259 putstr(" to continue: ");
265 char morc
; /* tell the outside world what char he used */
268 xwaitforspace(const char *s
) /* chars allowed besides space or return */
274 while((c
= readchar()) != '\n') {
277 if(s
&& index(s
,c
)) {
289 static char inputline
[COLNO
];
293 if(!Invisible
) curs_on_u(); else home();
294 while((foo
= readchar()) >= '0' && foo
<= '9')
295 multi
= 10*multi
+foo
-'0';
302 if(foo
== 'f' || foo
== 'F'){
303 inputline
[1] = getchar();
305 if(inputline
[1] == foo
) inputline
[2] = getchar(); else
309 if(foo
== 'm' || foo
== 'M'){
310 inputline
[1] = getchar();
323 if((sym
= getchar()) == EOF
)
326 * Some SYSV systems seem to return EOFs for various reasons
327 * (?like when one hits break or for interrupted systemcalls?),
328 * and we must see several before we quit.
330 int cnt
= NR_OF_EOFS
;
332 clearerr(stdin
); /* omit if clearerr is undefined */
333 if((sym
= getchar()) != EOF
) goto noteof
;
340 #endif /* NR_OF_EOFS */
341 if(flags
.toplin
== 1)
349 settty("End of input?\n");