xref: /dflybsd-src/usr.bin/top/screen.c (revision dc4f0af10c302f890123c000480a85f47d731199)
1*dc4f0af1Szrj /*
2*dc4f0af1Szrj  * Copyright (c) 1984 through 2008, William LeFebvre
3*dc4f0af1Szrj  * All rights reserved.
4*dc4f0af1Szrj  *
5*dc4f0af1Szrj  * Redistribution and use in source and binary forms, with or without
6*dc4f0af1Szrj  * modification, are permitted provided that the following conditions are met:
7*dc4f0af1Szrj  *
8*dc4f0af1Szrj  *     * Redistributions of source code must retain the above copyright
9*dc4f0af1Szrj  * notice, this list of conditions and the following disclaimer.
10*dc4f0af1Szrj  *
11*dc4f0af1Szrj  *     * Redistributions in binary form must reproduce the above
12*dc4f0af1Szrj  * copyright notice, this list of conditions and the following disclaimer
13*dc4f0af1Szrj  * in the documentation and/or other materials provided with the
14*dc4f0af1Szrj  * distribution.
15*dc4f0af1Szrj  *
16*dc4f0af1Szrj  *     * Neither the name of William LeFebvre nor the names of other
17*dc4f0af1Szrj  * contributors may be used to endorse or promote products derived from
18*dc4f0af1Szrj  * this software without specific prior written permission.
19*dc4f0af1Szrj  *
20*dc4f0af1Szrj  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
21*dc4f0af1Szrj  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
22*dc4f0af1Szrj  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
23*dc4f0af1Szrj  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
24*dc4f0af1Szrj  * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
25*dc4f0af1Szrj  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
26*dc4f0af1Szrj  * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
27*dc4f0af1Szrj  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
28*dc4f0af1Szrj  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
29*dc4f0af1Szrj  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
30*dc4f0af1Szrj  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
31*dc4f0af1Szrj  */
32*dc4f0af1Szrj 
33*dc4f0af1Szrj /*
34*dc4f0af1Szrj  *  Top users/processes display for Unix
35*dc4f0af1Szrj  *  Version 3
36*dc4f0af1Szrj  */
37*dc4f0af1Szrj 
38*dc4f0af1Szrj /*  This file contains the routines that interface to termcap and stty/gtty.
39*dc4f0af1Szrj  *
40*dc4f0af1Szrj  *  Paul Vixie, February 1987: converted to use ioctl() instead of stty/gtty.
41*dc4f0af1Szrj  *
42*dc4f0af1Szrj  *  I put in code to turn on the TOSTOP bit while top was running, but I
43*dc4f0af1Szrj  *  didn't really like the results.  If you desire it, turn on the
44*dc4f0af1Szrj  *  preprocessor variable "TOStop".   --wnl
45*dc4f0af1Szrj  */
46*dc4f0af1Szrj 
47*dc4f0af1Szrj #include "os.h"
48*dc4f0af1Szrj #include "top.h"
49*dc4f0af1Szrj 
50*dc4f0af1Szrj #if HAVE_CURSES_H && HAVE_TERM_H
51*dc4f0af1Szrj #include <curses.h>
52*dc4f0af1Szrj #include <term.h>
53*dc4f0af1Szrj #else
54*dc4f0af1Szrj #if HAVE_TERMCAP_H
55*dc4f0af1Szrj #include <termcap.h>
56*dc4f0af1Szrj #else
57*dc4f0af1Szrj #if HAVE_CURSES_H
58*dc4f0af1Szrj #include <curses.h>
59*dc4f0af1Szrj #endif
60*dc4f0af1Szrj #endif
61*dc4f0af1Szrj #endif
62*dc4f0af1Szrj 
63*dc4f0af1Szrj #if !HAVE_DECL_TPUTS
64*dc4f0af1Szrj int tputs(const char *, int, int (*)(int));
65*dc4f0af1Szrj #endif
66*dc4f0af1Szrj #if !HAVE_DECL_TGOTO
67*dc4f0af1Szrj char *tgoto(const char *, int, int);
68*dc4f0af1Szrj #endif
69*dc4f0af1Szrj #if !HAVE_DECL_TGETENT
70*dc4f0af1Szrj int tgetent(const char *, char *);
71*dc4f0af1Szrj #endif
72*dc4f0af1Szrj #if !HAVE_DECL_TGETFLAG
73*dc4f0af1Szrj int tgetflag(const char *);
74*dc4f0af1Szrj #endif
75*dc4f0af1Szrj #if !HAVE_DECL_TGETNUM
76*dc4f0af1Szrj int tgetnum(const char *);
77*dc4f0af1Szrj #endif
78*dc4f0af1Szrj #if !HAVE_DECL_TGETSTR
79*dc4f0af1Szrj char *tgetstr(const char *, char **);
80*dc4f0af1Szrj #endif
81*dc4f0af1Szrj 
82*dc4f0af1Szrj #include <sys/ioctl.h>
83*dc4f0af1Szrj #ifdef CBREAK
84*dc4f0af1Szrj # include <sgtty.h>
85*dc4f0af1Szrj # define USE_SGTTY
86*dc4f0af1Szrj #else
87*dc4f0af1Szrj # ifdef TCGETA
88*dc4f0af1Szrj #  define USE_TERMIO
89*dc4f0af1Szrj #  include <termio.h>
90*dc4f0af1Szrj # else
91*dc4f0af1Szrj #  define USE_TERMIOS
92*dc4f0af1Szrj #  include <termios.h>
93*dc4f0af1Szrj # endif
94*dc4f0af1Szrj #endif
95*dc4f0af1Szrj #if defined(USE_TERMIO) || defined(USE_TERMIOS)
96*dc4f0af1Szrj # ifndef TAB3
97*dc4f0af1Szrj #  ifdef OXTABS
98*dc4f0af1Szrj #   define TAB3 OXTABS
99*dc4f0af1Szrj #  else
100*dc4f0af1Szrj #   define TAB3 0
101*dc4f0af1Szrj #  endif
102*dc4f0af1Szrj # endif
103*dc4f0af1Szrj #endif
104*dc4f0af1Szrj 
105*dc4f0af1Szrj #include "screen.h"
106*dc4f0af1Szrj #include "boolean.h"
107*dc4f0af1Szrj 
108*dc4f0af1Szrj #define putcap(str)     (void)((str) != NULL ? tputs(str, 1, putstdout) : 0)
109*dc4f0af1Szrj 
110*dc4f0af1Szrj extern char *myname;
111*dc4f0af1Szrj 
112*dc4f0af1Szrj char ch_erase;
113*dc4f0af1Szrj char ch_kill;
114*dc4f0af1Szrj char ch_werase;
115*dc4f0af1Szrj char smart_terminal;
116*dc4f0af1Szrj int  screen_length;
117*dc4f0af1Szrj int  screen_width;
118*dc4f0af1Szrj 
119*dc4f0af1Szrj char PC;
120*dc4f0af1Szrj 
121*dc4f0af1Szrj static int  tc_overstrike;
122*dc4f0af1Szrj static char termcap_buf[1024];
123*dc4f0af1Szrj static char string_buffer[1024];
124*dc4f0af1Szrj static char home[15];
125*dc4f0af1Szrj static char lower_left[15];
126*dc4f0af1Szrj static char *tc_clear_line;
127*dc4f0af1Szrj static char *tc_clear_screen;
128*dc4f0af1Szrj static char *tc_clear_to_end;
129*dc4f0af1Szrj static char *tc_cursor_motion;
130*dc4f0af1Szrj static char *tc_start_standout;
131*dc4f0af1Szrj static char *tc_end_standout;
132*dc4f0af1Szrj static char *terminal_init;
133*dc4f0af1Szrj static char *terminal_end;
134*dc4f0af1Szrj 
135*dc4f0af1Szrj #ifdef USE_SGTTY
136*dc4f0af1Szrj static struct sgttyb old_settings;
137*dc4f0af1Szrj static struct sgttyb new_settings;
138*dc4f0af1Szrj #endif
139*dc4f0af1Szrj #ifdef USE_TERMIO
140*dc4f0af1Szrj static struct termio old_settings;
141*dc4f0af1Szrj static struct termio new_settings;
142*dc4f0af1Szrj #endif
143*dc4f0af1Szrj #ifdef USE_TERMIOS
144*dc4f0af1Szrj static struct termios old_settings;
145*dc4f0af1Szrj static struct termios new_settings;
146*dc4f0af1Szrj #endif
147*dc4f0af1Szrj static char is_a_terminal = No;
148*dc4f0af1Szrj #ifdef TOStop
149*dc4f0af1Szrj static int old_lword;
150*dc4f0af1Szrj static int new_lword;
151*dc4f0af1Szrj #endif
152*dc4f0af1Szrj 
153*dc4f0af1Szrj #define	STDIN	0
154*dc4f0af1Szrj #define	STDOUT	1
155*dc4f0af1Szrj #define	STDERR	2
156*dc4f0af1Szrj 
157*dc4f0af1Szrj /* This has to be defined as a subroutine for tputs (instead of a macro) */
158*dc4f0af1Szrj 
159*dc4f0af1Szrj static int
putstdout(TPUTS_PUTC_ARGTYPE ch)160*dc4f0af1Szrj putstdout(TPUTS_PUTC_ARGTYPE ch)
161*dc4f0af1Szrj 
162*dc4f0af1Szrj {
163*dc4f0af1Szrj     return putchar((int)ch);
164*dc4f0af1Szrj }
165*dc4f0af1Szrj 
166*dc4f0af1Szrj void
screen_getsize()167*dc4f0af1Szrj screen_getsize()
168*dc4f0af1Szrj 
169*dc4f0af1Szrj {
170*dc4f0af1Szrj 
171*dc4f0af1Szrj #ifdef TIOCGWINSZ
172*dc4f0af1Szrj 
173*dc4f0af1Szrj     struct winsize ws;
174*dc4f0af1Szrj 
175*dc4f0af1Szrj     if (ioctl (1, TIOCGWINSZ, &ws) != -1)
176*dc4f0af1Szrj     {
177*dc4f0af1Szrj 	if (ws.ws_row != 0)
178*dc4f0af1Szrj 	{
179*dc4f0af1Szrj 	    screen_length = ws.ws_row;
180*dc4f0af1Szrj 	}
181*dc4f0af1Szrj 	if (ws.ws_col != 0)
182*dc4f0af1Szrj 	{
183*dc4f0af1Szrj 	    screen_width = ws.ws_col - 1;
184*dc4f0af1Szrj 	}
185*dc4f0af1Szrj     }
186*dc4f0af1Szrj 
187*dc4f0af1Szrj #else
188*dc4f0af1Szrj #ifdef TIOCGSIZE
189*dc4f0af1Szrj 
190*dc4f0af1Szrj     struct ttysize ts;
191*dc4f0af1Szrj 
192*dc4f0af1Szrj     if (ioctl (1, TIOCGSIZE, &ts) != -1)
193*dc4f0af1Szrj     {
194*dc4f0af1Szrj 	if (ts.ts_lines != 0)
195*dc4f0af1Szrj 	{
196*dc4f0af1Szrj 	    screen_length = ts.ts_lines;
197*dc4f0af1Szrj 	}
198*dc4f0af1Szrj 	if (ts.ts_cols != 0)
199*dc4f0af1Szrj 	{
200*dc4f0af1Szrj 	    screen_width = ts.ts_cols - 1;
201*dc4f0af1Szrj 	}
202*dc4f0af1Szrj     }
203*dc4f0af1Szrj 
204*dc4f0af1Szrj #endif /* TIOCGSIZE */
205*dc4f0af1Szrj #endif /* TIOCGWINSZ */
206*dc4f0af1Szrj 
207*dc4f0af1Szrj     (void) strcpy(lower_left, tgoto(tc_cursor_motion, 0, screen_length - 1));
208*dc4f0af1Szrj }
209*dc4f0af1Szrj 
210*dc4f0af1Szrj int
screen_readtermcap(int interactive)211*dc4f0af1Szrj screen_readtermcap(int interactive)
212*dc4f0af1Szrj 
213*dc4f0af1Szrj {
214*dc4f0af1Szrj     char *bufptr;
215*dc4f0af1Szrj     char *PCptr;
216*dc4f0af1Szrj     char *term_name;
217*dc4f0af1Szrj     char *getenv();
218*dc4f0af1Szrj     int status;
219*dc4f0af1Szrj 
220*dc4f0af1Szrj     /* set defaults in case we aren't smart */
221*dc4f0af1Szrj     screen_width = MAX_COLS;
222*dc4f0af1Szrj     screen_length = 0;
223*dc4f0af1Szrj 
224*dc4f0af1Szrj     if (interactive == No)
225*dc4f0af1Szrj     {
226*dc4f0af1Szrj 	/* pretend we have a dumb terminal */
227*dc4f0af1Szrj 	smart_terminal = No;
228*dc4f0af1Szrj 	return No;
229*dc4f0af1Szrj     }
230*dc4f0af1Szrj 
231*dc4f0af1Szrj     /* assume we have a smart terminal until proven otherwise */
232*dc4f0af1Szrj     smart_terminal = Yes;
233*dc4f0af1Szrj 
234*dc4f0af1Szrj     /* get the terminal name */
235*dc4f0af1Szrj     term_name = getenv("TERM");
236*dc4f0af1Szrj 
237*dc4f0af1Szrj     /* if there is no TERM, assume it's a dumb terminal */
238*dc4f0af1Szrj     /* patch courtesy of Sam Horrocks at telegraph.ics.uci.edu */
239*dc4f0af1Szrj     if (term_name == NULL)
240*dc4f0af1Szrj     {
241*dc4f0af1Szrj 	smart_terminal = No;
242*dc4f0af1Szrj 	return No;
243*dc4f0af1Szrj     }
244*dc4f0af1Szrj 
245*dc4f0af1Szrj     /* now get the termcap entry */
246*dc4f0af1Szrj     if ((status = tgetent(termcap_buf, term_name)) != 1)
247*dc4f0af1Szrj     {
248*dc4f0af1Szrj 	if (status == -1)
249*dc4f0af1Szrj 	{
250*dc4f0af1Szrj 	    fprintf(stderr, "%s: can't open termcap file\n", myname);
251*dc4f0af1Szrj 	}
252*dc4f0af1Szrj 	else
253*dc4f0af1Szrj 	{
254*dc4f0af1Szrj 	    fprintf(stderr, "%s: no termcap entry for a `%s' terminal\n",
255*dc4f0af1Szrj 		    myname, term_name);
256*dc4f0af1Szrj 	}
257*dc4f0af1Szrj 
258*dc4f0af1Szrj 	/* pretend it's dumb and proceed */
259*dc4f0af1Szrj 	smart_terminal = No;
260*dc4f0af1Szrj 	return No;
261*dc4f0af1Szrj     }
262*dc4f0af1Szrj 
263*dc4f0af1Szrj     /* "hardcopy" immediately indicates a very stupid terminal */
264*dc4f0af1Szrj     if (tgetflag("hc"))
265*dc4f0af1Szrj     {
266*dc4f0af1Szrj 	smart_terminal = No;
267*dc4f0af1Szrj 	return No;
268*dc4f0af1Szrj     }
269*dc4f0af1Szrj 
270*dc4f0af1Szrj     /* set up common terminal capabilities */
271*dc4f0af1Szrj     if ((screen_length = tgetnum("li")) <= 0)
272*dc4f0af1Szrj     {
273*dc4f0af1Szrj 	screen_length = smart_terminal = 0;
274*dc4f0af1Szrj 	return No;
275*dc4f0af1Szrj     }
276*dc4f0af1Szrj 
277*dc4f0af1Szrj     /* screen_width is a little different */
278*dc4f0af1Szrj     if ((screen_width = tgetnum("co")) == -1)
279*dc4f0af1Szrj     {
280*dc4f0af1Szrj 	screen_width = 79;
281*dc4f0af1Szrj     }
282*dc4f0af1Szrj     else
283*dc4f0af1Szrj     {
284*dc4f0af1Szrj 	screen_width -= 1;
285*dc4f0af1Szrj     }
286*dc4f0af1Szrj 
287*dc4f0af1Szrj     /* terminals that overstrike need special attention */
288*dc4f0af1Szrj     tc_overstrike = tgetflag("os");
289*dc4f0af1Szrj 
290*dc4f0af1Szrj     /* initialize the pointer into the termcap string buffer */
291*dc4f0af1Szrj     bufptr = string_buffer;
292*dc4f0af1Szrj 
293*dc4f0af1Szrj     /* get "ce", clear to end */
294*dc4f0af1Szrj     if (!tc_overstrike)
295*dc4f0af1Szrj     {
296*dc4f0af1Szrj 	tc_clear_line = tgetstr("ce", &bufptr);
297*dc4f0af1Szrj     }
298*dc4f0af1Szrj 
299*dc4f0af1Szrj     /* get necessary capabilities */
300*dc4f0af1Szrj     if ((tc_clear_screen  = tgetstr("cl", &bufptr)) == NULL ||
301*dc4f0af1Szrj 	(tc_cursor_motion = tgetstr("cm", &bufptr)) == NULL)
302*dc4f0af1Szrj     {
303*dc4f0af1Szrj 	smart_terminal = No;
304*dc4f0af1Szrj 	return No;
305*dc4f0af1Szrj     }
306*dc4f0af1Szrj 
307*dc4f0af1Szrj     /* get some more sophisticated stuff -- these are optional */
308*dc4f0af1Szrj     tc_clear_to_end   = tgetstr("cd", &bufptr);
309*dc4f0af1Szrj     terminal_init  = tgetstr("ti", &bufptr);
310*dc4f0af1Szrj     terminal_end   = tgetstr("te", &bufptr);
311*dc4f0af1Szrj     tc_start_standout = tgetstr("so", &bufptr);
312*dc4f0af1Szrj     tc_end_standout   = tgetstr("se", &bufptr);
313*dc4f0af1Szrj 
314*dc4f0af1Szrj     /* pad character */
315*dc4f0af1Szrj     PC = (PCptr = tgetstr("pc", &bufptr)) ? *PCptr : 0;
316*dc4f0af1Szrj 
317*dc4f0af1Szrj     /* set convenience strings */
318*dc4f0af1Szrj     (void) strcpy(home, tgoto(tc_cursor_motion, 0, 0));
319*dc4f0af1Szrj     /* (lower_left is set in screen_getsize) */
320*dc4f0af1Szrj 
321*dc4f0af1Szrj     /* get the actual screen size with an ioctl, if needed */
322*dc4f0af1Szrj     /* This may change screen_width and screen_length, and it always
323*dc4f0af1Szrj        sets lower_left. */
324*dc4f0af1Szrj     screen_getsize();
325*dc4f0af1Szrj 
326*dc4f0af1Szrj     /* if stdout is not a terminal, pretend we are a dumb terminal */
327*dc4f0af1Szrj #ifdef USE_SGTTY
328*dc4f0af1Szrj     if (ioctl(STDOUT, TIOCGETP, &old_settings) == -1)
329*dc4f0af1Szrj     {
330*dc4f0af1Szrj 	smart_terminal = No;
331*dc4f0af1Szrj     }
332*dc4f0af1Szrj #endif
333*dc4f0af1Szrj #ifdef USE_TERMIO
334*dc4f0af1Szrj     if (ioctl(STDOUT, TCGETA, &old_settings) == -1)
335*dc4f0af1Szrj     {
336*dc4f0af1Szrj 	smart_terminal = No;
337*dc4f0af1Szrj     }
338*dc4f0af1Szrj #endif
339*dc4f0af1Szrj #ifdef USE_TERMIOS
340*dc4f0af1Szrj     if (tcgetattr(STDOUT, &old_settings) == -1)
341*dc4f0af1Szrj     {
342*dc4f0af1Szrj 	smart_terminal = No;
343*dc4f0af1Szrj     }
344*dc4f0af1Szrj #endif
345*dc4f0af1Szrj 
346*dc4f0af1Szrj     return smart_terminal;
347*dc4f0af1Szrj }
348*dc4f0af1Szrj 
349*dc4f0af1Szrj void
screen_init()350*dc4f0af1Szrj screen_init()
351*dc4f0af1Szrj 
352*dc4f0af1Szrj {
353*dc4f0af1Szrj     /* get the old settings for safe keeping */
354*dc4f0af1Szrj #ifdef USE_SGTTY
355*dc4f0af1Szrj     if (ioctl(STDOUT, TIOCGETP, &old_settings) != -1)
356*dc4f0af1Szrj     {
357*dc4f0af1Szrj 	/* copy the settings so we can modify them */
358*dc4f0af1Szrj 	new_settings = old_settings;
359*dc4f0af1Szrj 
360*dc4f0af1Szrj 	/* turn on CBREAK and turn off character echo and tab expansion */
361*dc4f0af1Szrj 	new_settings.sg_flags |= CBREAK;
362*dc4f0af1Szrj 	new_settings.sg_flags &= ~(ECHO|XTABS);
363*dc4f0af1Szrj 	(void) ioctl(STDOUT, TIOCSETP, &new_settings);
364*dc4f0af1Szrj 
365*dc4f0af1Szrj 	/* remember the erase and kill characters */
366*dc4f0af1Szrj 	ch_erase = old_settings.sg_erase;
367*dc4f0af1Szrj 	ch_kill  = old_settings.sg_kill;
368*dc4f0af1Szrj 	ch_werase  = old_settings.sg_werase;
369*dc4f0af1Szrj 
370*dc4f0af1Szrj #ifdef TOStop
371*dc4f0af1Szrj 	/* get the local mode word */
372*dc4f0af1Szrj 	(void) ioctl(STDOUT, TIOCLGET, &old_lword);
373*dc4f0af1Szrj 
374*dc4f0af1Szrj 	/* modify it */
375*dc4f0af1Szrj 	new_lword = old_lword | LTOSTOP;
376*dc4f0af1Szrj 	(void) ioctl(STDOUT, TIOCLSET, &new_lword);
377*dc4f0af1Szrj #endif
378*dc4f0af1Szrj 	/* remember that it really is a terminal */
379*dc4f0af1Szrj 	is_a_terminal = Yes;
380*dc4f0af1Szrj 
381*dc4f0af1Szrj 	/* send the termcap initialization string */
382*dc4f0af1Szrj 	putcap(terminal_init);
383*dc4f0af1Szrj     }
384*dc4f0af1Szrj #endif
385*dc4f0af1Szrj #ifdef USE_TERMIO
386*dc4f0af1Szrj     if (ioctl(STDOUT, TCGETA, &old_settings) != -1)
387*dc4f0af1Szrj     {
388*dc4f0af1Szrj 	/* copy the settings so we can modify them */
389*dc4f0af1Szrj 	new_settings = old_settings;
390*dc4f0af1Szrj 
391*dc4f0af1Szrj 	/* turn off ICANON, character echo and tab expansion */
392*dc4f0af1Szrj 	new_settings.c_lflag &= ~(ICANON|ECHO);
393*dc4f0af1Szrj 	new_settings.c_oflag &= ~(TAB3);
394*dc4f0af1Szrj 	new_settings.c_cc[VMIN] = 1;
395*dc4f0af1Szrj 	new_settings.c_cc[VTIME] = 0;
396*dc4f0af1Szrj 	(void) ioctl(STDOUT, TCSETA, &new_settings);
397*dc4f0af1Szrj 
398*dc4f0af1Szrj 	/* remember the erase and kill characters */
399*dc4f0af1Szrj 	ch_erase  = old_settings.c_cc[VERASE];
400*dc4f0af1Szrj 	ch_kill   = old_settings.c_cc[VKILL];
401*dc4f0af1Szrj 	ch_werase = old_settings.c_cc[VWERASE];
402*dc4f0af1Szrj 
403*dc4f0af1Szrj 	/* remember that it really is a terminal */
404*dc4f0af1Szrj 	is_a_terminal = Yes;
405*dc4f0af1Szrj 
406*dc4f0af1Szrj 	/* send the termcap initialization string */
407*dc4f0af1Szrj 	putcap(terminal_init);
408*dc4f0af1Szrj     }
409*dc4f0af1Szrj #endif
410*dc4f0af1Szrj #ifdef USE_TERMIOS
411*dc4f0af1Szrj     if (tcgetattr(STDOUT, &old_settings) != -1)
412*dc4f0af1Szrj     {
413*dc4f0af1Szrj 	/* copy the settings so we can modify them */
414*dc4f0af1Szrj 	new_settings = old_settings;
415*dc4f0af1Szrj 
416*dc4f0af1Szrj 	/* turn off ICANON, character echo and tab expansion */
417*dc4f0af1Szrj 	new_settings.c_lflag &= ~(ICANON|ECHO);
418*dc4f0af1Szrj 	new_settings.c_oflag &= ~(TAB3);
419*dc4f0af1Szrj 	new_settings.c_cc[VMIN] = 1;
420*dc4f0af1Szrj 	new_settings.c_cc[VTIME] = 0;
421*dc4f0af1Szrj 	(void) tcsetattr(STDOUT, TCSADRAIN, &new_settings);
422*dc4f0af1Szrj 
423*dc4f0af1Szrj 	/* remember the erase and kill characters */
424*dc4f0af1Szrj 	ch_erase  = old_settings.c_cc[VERASE];
425*dc4f0af1Szrj 	ch_kill   = old_settings.c_cc[VKILL];
426*dc4f0af1Szrj 	ch_werase = old_settings.c_cc[VWERASE];
427*dc4f0af1Szrj 
428*dc4f0af1Szrj 	/* remember that it really is a terminal */
429*dc4f0af1Szrj 	is_a_terminal = Yes;
430*dc4f0af1Szrj 
431*dc4f0af1Szrj 	/* send the termcap initialization string */
432*dc4f0af1Szrj 	putcap(terminal_init);
433*dc4f0af1Szrj     }
434*dc4f0af1Szrj #endif
435*dc4f0af1Szrj 
436*dc4f0af1Szrj     if (!is_a_terminal)
437*dc4f0af1Szrj     {
438*dc4f0af1Szrj 	/* not a terminal at all---consider it dumb */
439*dc4f0af1Szrj 	smart_terminal = No;
440*dc4f0af1Szrj     }
441*dc4f0af1Szrj }
442*dc4f0af1Szrj 
443*dc4f0af1Szrj void
screen_end()444*dc4f0af1Szrj screen_end()
445*dc4f0af1Szrj 
446*dc4f0af1Szrj {
447*dc4f0af1Szrj     /* move to the lower left, clear the line and send "te" */
448*dc4f0af1Szrj     if (smart_terminal)
449*dc4f0af1Szrj     {
450*dc4f0af1Szrj 	putcap(lower_left);
451*dc4f0af1Szrj 	putcap(tc_clear_line);
452*dc4f0af1Szrj 	fflush(stdout);
453*dc4f0af1Szrj 	putcap(terminal_end);
454*dc4f0af1Szrj     }
455*dc4f0af1Szrj 
456*dc4f0af1Szrj     /* if we have settings to reset, then do so */
457*dc4f0af1Szrj     if (is_a_terminal)
458*dc4f0af1Szrj     {
459*dc4f0af1Szrj #ifdef USE_SGTTY
460*dc4f0af1Szrj 	(void) ioctl(STDOUT, TIOCSETP, &old_settings);
461*dc4f0af1Szrj #ifdef TOStop
462*dc4f0af1Szrj 	(void) ioctl(STDOUT, TIOCLSET, &old_lword);
463*dc4f0af1Szrj #endif
464*dc4f0af1Szrj #endif
465*dc4f0af1Szrj #ifdef USE_TERMIO
466*dc4f0af1Szrj 	(void) ioctl(STDOUT, TCSETA, &old_settings);
467*dc4f0af1Szrj #endif
468*dc4f0af1Szrj #ifdef USE_TERMIOS
469*dc4f0af1Szrj 	(void) tcsetattr(STDOUT, TCSADRAIN, &old_settings);
470*dc4f0af1Szrj #endif
471*dc4f0af1Szrj     }
472*dc4f0af1Szrj }
473*dc4f0af1Szrj 
474*dc4f0af1Szrj void
screen_reinit()475*dc4f0af1Szrj screen_reinit()
476*dc4f0af1Szrj 
477*dc4f0af1Szrj {
478*dc4f0af1Szrj     /* install our settings if it is a terminal */
479*dc4f0af1Szrj     if (is_a_terminal)
480*dc4f0af1Szrj     {
481*dc4f0af1Szrj #ifdef USE_SGTTY
482*dc4f0af1Szrj 	(void) ioctl(STDOUT, TIOCSETP, &new_settings);
483*dc4f0af1Szrj #ifdef TOStop
484*dc4f0af1Szrj 	(void) ioctl(STDOUT, TIOCLSET, &new_lword);
485*dc4f0af1Szrj #endif
486*dc4f0af1Szrj #endif
487*dc4f0af1Szrj #ifdef USE_TERMIO
488*dc4f0af1Szrj 	(void) ioctl(STDOUT, TCSETA, &new_settings);
489*dc4f0af1Szrj #endif
490*dc4f0af1Szrj #ifdef USE_TERMIOS
491*dc4f0af1Szrj 	(void) tcsetattr(STDOUT, TCSADRAIN, &new_settings);
492*dc4f0af1Szrj #endif
493*dc4f0af1Szrj     }
494*dc4f0af1Szrj 
495*dc4f0af1Szrj     /* send init string */
496*dc4f0af1Szrj     if (smart_terminal)
497*dc4f0af1Szrj     {
498*dc4f0af1Szrj 	putcap(terminal_init);
499*dc4f0af1Szrj     }
500*dc4f0af1Szrj }
501*dc4f0af1Szrj 
502*dc4f0af1Szrj void
screen_move(int x,int y)503*dc4f0af1Szrj screen_move(int x, int y)
504*dc4f0af1Szrj 
505*dc4f0af1Szrj {
506*dc4f0af1Szrj     tputs(tgoto(tc_cursor_motion, x, y), 1, putstdout);
507*dc4f0af1Szrj }
508*dc4f0af1Szrj 
509*dc4f0af1Szrj void
screen_standout(char * msg)510*dc4f0af1Szrj screen_standout(char *msg)
511*dc4f0af1Szrj 
512*dc4f0af1Szrj {
513*dc4f0af1Szrj     if (smart_terminal)
514*dc4f0af1Szrj     {
515*dc4f0af1Szrj 	putcap(tc_start_standout);
516*dc4f0af1Szrj 	fputs(msg, stdout);
517*dc4f0af1Szrj 	putcap(tc_end_standout);
518*dc4f0af1Szrj     }
519*dc4f0af1Szrj     else
520*dc4f0af1Szrj     {
521*dc4f0af1Szrj 	fputs(msg, stdout);
522*dc4f0af1Szrj     }
523*dc4f0af1Szrj }
524*dc4f0af1Szrj 
525*dc4f0af1Szrj void
screen_clear()526*dc4f0af1Szrj screen_clear()
527*dc4f0af1Szrj 
528*dc4f0af1Szrj {
529*dc4f0af1Szrj     if (smart_terminal)
530*dc4f0af1Szrj     {
531*dc4f0af1Szrj 	putcap(tc_clear_screen);
532*dc4f0af1Szrj     }
533*dc4f0af1Szrj }
534*dc4f0af1Szrj 
535*dc4f0af1Szrj int
screen_cte()536*dc4f0af1Szrj screen_cte()
537*dc4f0af1Szrj 
538*dc4f0af1Szrj {
539*dc4f0af1Szrj     if (smart_terminal)
540*dc4f0af1Szrj     {
541*dc4f0af1Szrj 	if (tc_clear_to_end)
542*dc4f0af1Szrj 	{
543*dc4f0af1Szrj 	    putcap(tc_clear_to_end);
544*dc4f0af1Szrj 	    return(Yes);
545*dc4f0af1Szrj 	}
546*dc4f0af1Szrj     }
547*dc4f0af1Szrj     return(No);
548*dc4f0af1Szrj }
549*dc4f0af1Szrj 
550*dc4f0af1Szrj void
screen_cleareol(int len)551*dc4f0af1Szrj screen_cleareol(int len)
552*dc4f0af1Szrj 
553*dc4f0af1Szrj {
554*dc4f0af1Szrj     int i;
555*dc4f0af1Szrj 
556*dc4f0af1Szrj     if (smart_terminal && !tc_overstrike && len > 0)
557*dc4f0af1Szrj     {
558*dc4f0af1Szrj 	if (tc_clear_line)
559*dc4f0af1Szrj 	{
560*dc4f0af1Szrj 	    putcap(tc_clear_line);
561*dc4f0af1Szrj 	    return;
562*dc4f0af1Szrj 	}
563*dc4f0af1Szrj 	else
564*dc4f0af1Szrj 	{
565*dc4f0af1Szrj 	    i = 0;
566*dc4f0af1Szrj 	    while (i++ < 0)
567*dc4f0af1Szrj 	    {
568*dc4f0af1Szrj 		putchar(' ');
569*dc4f0af1Szrj 	    }
570*dc4f0af1Szrj 	    i = 0;
571*dc4f0af1Szrj 	    while (i++ < 0)
572*dc4f0af1Szrj 	    {
573*dc4f0af1Szrj 		putchar('\b');
574*dc4f0af1Szrj 	    }
575*dc4f0af1Szrj 	    return;
576*dc4f0af1Szrj 	}
577*dc4f0af1Szrj     }
578*dc4f0af1Szrj     return;
579*dc4f0af1Szrj }
580*dc4f0af1Szrj 
581*dc4f0af1Szrj void
screen_home()582*dc4f0af1Szrj screen_home()
583*dc4f0af1Szrj 
584*dc4f0af1Szrj {
585*dc4f0af1Szrj     if (smart_terminal)
586*dc4f0af1Szrj     {
587*dc4f0af1Szrj 	putcap(home);
588*dc4f0af1Szrj     }
589*dc4f0af1Szrj }
590*dc4f0af1Szrj 
591*dc4f0af1Szrj 
592