DataMuseum.dkPresents historical artifacts from the history of: Rational R1000/400 Tapes |
This is an automatic "excavation" of a thematic subset of
See our Wiki for more about Rational R1000/400 Tapes Excavated with: AutoArchaeologist - Free & Open Source Software. |
top - downloadIndex: ┃ T m ┃
Length: 72612 (0x11ba4) Types: TextFile Names: »main.c«
└─⟦5f3412b64⟧ Bits:30000745 8mm tape, Rational 1000, ENVIRONMENT 12_6_5 TOOLS └─ ⟦91c658230⟧ »DATA« └─⟦5d656759a⟧ └─⟦50f09e4e4⟧ └─ ⟦this⟧ »./main.c« └─⟦5f3412b64⟧ Bits:30000745 8mm tape, Rational 1000, ENVIRONMENT 12_6_5 TOOLS └─ ⟦91c658230⟧ »DATA« └─⟦5d656759a⟧ └─⟦610eb0a19⟧ └─ ⟦this⟧ »./main.c« └─⟦5f3412b64⟧ Bits:30000745 8mm tape, Rational 1000, ENVIRONMENT 12_6_5 TOOLS └─ ⟦91c658230⟧ »DATA« └─⟦5d656759a⟧ └─⟦d97085656⟧ └─ ⟦this⟧ »./main.c«
#ifndef lint static char rcs_id[] = "$XConsortium: main.c,v 1.99 89/01/04 14:33:47 jim Exp $"; #endif /* lint */ /* * The tty setup code was completely rewritten by Dave Serisky at * Hewlett-Packard and should now work properly under both System V and * 4BSD. However, the toolkit stuff is still pretty crufty. In general, * beware, for there be dragons here. */ #include <X11/copyright.h> /*********************************************************** Copyright 1987, 1988 by Digital Equipment Corporation, Maynard, Massachusetts, and the Massachusetts Institute of Technology, Cambridge, Massachusetts. All Rights Reserved Permission to use, copy, modify, and distribute this software and its documentation for any purpose and without fee is hereby granted, provided that the above copyright notice appear in all copies and that both that copyright notice and this permission notice appear in supporting documentation, and that the names of Digital or MIT not be used in advertising or publicity pertaining to distribution of the software without specific, written prior permission. DIGITAL DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE, INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO EVENT SHALL DIGITAL BE LIABLE FOR ANY SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. ******************************************************************/ /* main.c */ #include <X11/Xos.h> #ifdef hpux #undef SIGCHLD #endif #include <X11/Xlib.h> #include <X11/cursorfont.h> #include <X11/Intrinsic.h> #include <pwd.h> #include <ctype.h> #ifdef att #define USE_USG_PTYS #endif #ifndef att #define USE_HANDSHAKE #endif #include <sys/ioctl.h> #ifdef SYSV #include <sys/termio.h> #ifdef USE_USG_PTYS /* AT&T SYSV has no ptyio.h */ #include <sys/stream.h> /* get typedef used in ptem.h */ #include <sys/ptem.h> /* get struct winsize */ #include <sys/stropts.h> /* for I_PUSH */ #include <poll.h> /* for POLLIN */ #endif #include <sys/stat.h> #define USE_SYSV_TERMIO #define USE_SYSV_SIGNALS #define USE_SYSV_PGRP #define USE_SYSV_ENVVARS /* COLUMNS/LINES vs. TERMCAP */ /* * now get system-specific includes */ #ifdef CRAY #define HAS_UTMP_UT_HOST #define HAS_BSD_GROUPS #endif #ifdef macII #define HAS_UTMP_UT_HOST #define HAS_BSD_GROUPS #include <sys/ttychars.h> #undef USE_SYSV_ENVVARS #undef FIOCLEX #undef FIONCLEX #define setpgrp2 setpgrp #include <sgtty.h> #include <sys/resource.h> #endif #ifdef hpux #define HAS_BSD_GROUPS #include <sys/ptyio.h> #endif #endif /* SYSV */ #ifndef SYSV /* BSD systems */ #include <sgtty.h> #include <sys/resource.h> #define HAS_UTMP_UT_HOST #define HAS_BSD_GROUPS #endif /* !SYSV */ #include <stdio.h> #include <errno.h> #include <setjmp.h> #include <signal.h> #ifdef hpux #include <sys/utsname.h> #endif /* hpux */ #ifdef ultrix #include <fcntl.h> #else #ifndef apollo #include <sys/fcntl.h> #endif #endif #ifdef apollo #define ttyslot() 1 #endif /* apollo */ #include <utmp.h> #ifdef LASTLOG #include <lastlog.h> #endif #include <sys/param.h> /* for NOFILE */ #ifdef PUCC_PTYD #include <local/openpty.h> int Ptyfd; #endif /* PUCC_PTYD */ #ifndef UTMP_FILENAME #define UTMP_FILENAME "/etc/utmp" #endif #ifndef LASTLOG_FILENAME #define LASTLOG_FILENAME "/usr/adm/lastlog" /* only on BSD systems */ #endif #ifndef WTMP_FILENAME #if defined(SYSV) #define WTMP_FILENAME "/etc/wtmp" #else #define WTMP_FILENAME "/usr/adm/wtmp" #endif #endif #include "ptyx.h" #include "data.h" #include "error.h" #include "main.h" #include <X11/StringDefs.h> #include <X11/Shell.h> #include <X11/Scroll.h> #ifdef SIGTSTP #include <sys/wait.h> #ifdef hpux #include <sys/bsdtty.h> #endif #endif #define _RCG_VALUES_ #include "rcg.h" #include "ratmenu.h" #ifdef SIGNALRETURNSINT #define SIGNAL_T int #define SIGNAL_RETURN return 0 #else #define SIGNAL_T void #define SIGNAL_RETURN return #endif SIGNAL_T Exit(); #ifndef ultrix extern char *malloc(); extern char *calloc(); extern char *realloc(); #endif extern char *ttyname(); extern char *getenv(); extern char *strindex (); extern void exit(); extern void sleep(); extern void bcopy(); extern long lseek(); extern XtResource RatMenuResourceList[]; extern int Size_RatMenuResourceList; #ifndef XtNbaseWidth #define XtNbaseWidth "baseWidth" #endif #ifndef XtNbaseHeight #define XtNbaseHeight "baseHeight" #endif int switchfb[] = {0, 2, 1, 3}; static SIGNAL_T reapchild (); static Bool added_utmp_entry = False; static char **command_to_exec; #ifdef USE_SYSV_TERMIO /* The following structures are initialized in main() in order ** to eliminate any assumptions about the internal order of their ** contents. */ static struct termio d_tio; #ifdef TIOCSLTC static struct ltchars d_ltc; #endif /* TIOCSLTC */ #ifdef TIOCLSET static unsigned int d_lmode; #endif /* TIOCLSET */ #else /* not USE_SYSV_TERMIO */ static struct sgttyb d_sg = { 0, 0, 0177, CKILL, EVENP|ODDP|ECHO|XTABS|CRMOD }; static struct tchars d_tc = { CINTR, CQUIT, CSTART, CSTOP, CEOF, CBRK, }; static struct ltchars d_ltc = { CSUSP, CDSUSP, CRPRNT, CFLUSH, CWERASE, CLNEXT }; static int d_disipline = NTTYDISC; static long int d_lmode = LCRTBS|LCRTERA|LCRTKIL|LCTLECH; #endif /* USE_SYSV_TERMIO */ static int parse_tty_modes (); /* * SYSV has the termio.c_cc[V] and ltchars; BSD has tchars and ltchars */ static int override_tty_modes = 0; struct _xttymodes { char *name; int len; int set; char value; } ttymodelist[] = { { "intr", 4, 0, '\0' }, /* tchars.t_intrc ; VINTR */ #define XTTYMODE_intr 0 { "quit", 4, 0, '\0' }, /* tchars.t_quitc ; VQUIT */ #define XTTYMODE_quit 1 { "erase", 5, 0, '\0' }, /* sgttyb.sg_erase ; VERASE */ #define XTTYMODE_erase 2 { "kill", 4, 0, '\0' }, /* sgttyb.sg_kill ; VKILL */ #define XTTYMODE_kill 3 { "eof", 3, 0, '\0' }, /* tchars.t_eofc ; VEOF */ #define XTTYMODE_eof 4 { "eol", 3, 0, '\0' }, /* VEOL */ #define XTTYMODE_eol 5 { "swtch", 5, 0, '\0' }, /* VSWTCH */ #define XTTYMODE_swtch 6 { "start", 5, 0, '\0' }, /* tchars.t_startc */ #define XTTYMODE_start 7 { "stop", 4, 0, '\0' }, /* tchars.t_stopc */ #define XTTYMODE_stop 8 { "brk", 3, 0, '\0' }, /* tchars.t_brkc */ #define XTTYMODE_brk 9 { "susp", 4, 0, '\0' }, /* ltchars.t_suspc */ #define XTTYMODE_susp 10 { "dsusp", 5, 0, '\0' }, /* ltchars.t_dsuspc */ #define XTTYMODE_dsusp 11 { "rprnt", 5, 0, '\0' }, /* ltchars.t_rprntc */ #define XTTYMODE_rprnt 12 { "flush", 5, 0, '\0' }, /* ltchars.t_flushc */ #define XTTYMODE_flush 13 { "weras", 5, 0, '\0' }, /* ltchars.t_werasc */ #define XTTYMODE_weras 14 { "lnext", 5, 0, '\0' }, /* ltchars.t_lnextc */ #define XTTYMODE_lnext 15 #define NXTTYMODES 16 }; #ifdef USE_SYSV_UTMP extern struct utmp *getutent(); extern struct utmp *getutid(); extern struct utmp *getutline(); extern void pututline(); extern void setutent(); extern void endutent(); extern void utmpname(); extern struct passwd *getpwent(); extern struct passwd *getpwuid(); extern struct passwd *getpwnam(); extern void setpwent(); extern void endpwent(); extern struct passwd *fgetpwent(); #else /* not USE_SYSV_UTMP */ static char etc_utmp[] = UTMP_FILENAME; #ifdef LASTLOG static char etc_lastlog[] = LASTLOG_FILENAME; #endif #ifdef WTMP static char etc_wtmp[] = WTMP_FILENAME; #endif #endif /* USE_SYSV_UTMP */ /* * Some people with 4.3bsd /bin/login seem to like to use login -p -f user * to implement xterm -ls. They can turn on USE_LOGIN_DASH_P and turn off * WTMP and LASTLOG. */ #ifdef USE_LOGIN_DASH_P #ifndef LOGIN_FILENAME #define LOGIN_FILENAME "/bin/login" #endif static char bin_login[] = LOGIN_FILENAME; #endif static int inhibit; static char passedPty[2]; /* name if pty if slave */ #ifdef TIOCCONS static int Console; #endif /* TIOCCONS */ #ifndef USE_SYSV_UTMP static int tslot; #endif /* USE_SYSV_UTMP */ static jmp_buf env; char *ProgramName; Boolean sunFunctionKeys; static struct _resource { char *xterm_name; char *icon_geometry; char *title; char *icon_name; char *term_name; char *tty_modes; Boolean utmpInhibit; Boolean AutoRaise; int AutoRaiseDelay; Boolean sunFunctionKeys; /* %%% should be widget resource? */ int VertDragTrigger; int VertDragAmount; int HorizDragTrigger; int HorizDragAmount; char *RcgValue; } resource; Vt100_Resource vt100_resource; Boolean AutoRaise; int AutoRaiseDelay; int VerticalDragTrigger; int VerticalDragAmount; static int defVDT = 4; static int defVDA = 1; int HorizontalDragTrigger; int HorizontalDragAmount; static int defHDT = 20; static int defHDA = 10; int RecognitionValue; #define XtNboldFont "boldFont" #define XtNmenuFont "menuFont" #define XtNscrollBar "scrollBar" #define XtCScrollBar "ScrollBar" #define XtNinternalBorder "internalBorder" /* used by VT (charproc.c) */ extern int defaultIntBorder; /* charproc.c */ #define offset(field) XtOffset(Vt100_Resource *, field) static XtResource vt100_resources[] = { {"x11r2", "X11R2", XtRBoolean, sizeof(Boolean), offset(X11R2), XtRString, "false"}, {XtNgeometry, XtCGeometry, XtRString, sizeof(char *), offset(geometry), XtRString, defGeometry}, {XtNfont, XtCFont, XtRString, sizeof(char *), offset(f_n_name), XtRString, DEFFONT}, {XtNboldFont, "BoldFont", XtRString, sizeof(char *), offset(f_b_name), XtRString, DEFBOLDFONT}, {XtNmenuFont, "MenuFont", XtRString, sizeof(char *), offset(f_m_name), XtRString, DEFMENUFONT}, {XtNfont, XtCFont, XtRFontStruct, sizeof(XFontStruct *), offset(f_n), XtRString, DEFFONT}, {XtNboldFont, "BoldFont", XtRFontStruct, sizeof(XFontStruct *), offset(f_b), XtRString, DEFBOLDFONT}, {XtNmenuFont, "MenuFont", XtRFontStruct, sizeof(XFontStruct *), offset(f_m), XtRString, DEFMENUFONT}, {XtNscrollBar, XtCScrollBar, XtRBoolean, sizeof(Boolean), offset(scrollbar), XtRString, (caddr_t)"false"}, {XtNinternalBorder, XtCBorderWidth, XtRInt, sizeof(int), offset(border), XtRInt, (caddr_t) &defaultIntBorder}, }; #undef offset #define offset(field) XtOffset(struct _resource *, field) static XtResource application_resources[] = { {"autoRaise", "AutoRaise", XtRBoolean, sizeof(Boolean), offset(AutoRaise), XtRString, (caddr_t)"false"}, {"autoRaiseDelay", "AutoRaiseDelay", XtRInt, sizeof(int), offset(AutoRaiseDelay), XtRString, (caddr_t)"2000"}, {"horizDragTrigger", "DragTrigger", XtRInt, sizeof(int), offset(HorizDragTrigger), XtRInt, (caddr_t)&defHDT}, {"vertDragTrigger", "DragTrigger", XtRInt, sizeof(int), offset(VertDragTrigger), XtRInt, (caddr_t)&defVDT}, {"horizDragAmount", "DragAmount", XtRInt, sizeof(int), offset(HorizDragAmount), XtRInt, (caddr_t)&defHDA}, {"vertDragAmount", "DragAmount", XtRInt, sizeof(int), offset(VertDragAmount), XtRInt, (caddr_t)&defVDA}, {"recognition", "Recognition", XtRString, sizeof(char *), offset(RcgValue), XtRString, (caddr_t)defRV}, {"name", "Name", XtRString, sizeof(char *), offset(xterm_name), XtRString, "rxi"}, {"iconGeometry", "IconGeometry", XtRString, sizeof(char *), offset(icon_geometry), XtRString, (caddr_t) NULL}, {XtNtitle, XtCTitle, XtRString, sizeof(char *), offset(title), XtRString, (caddr_t) NULL}, {XtNiconName, XtCIconName, XtRString, sizeof(char *), offset(icon_name), XtRString, (caddr_t) NULL}, {"termName", "TermName", XtRString, sizeof(char *), offset(term_name), XtRString, (caddr_t) NULL}, {"ttyModes", "TtyModes", XtRString, sizeof(char *), offset(tty_modes), XtRString, (caddr_t) NULL}, {"utmpInhibit", "UtmpInhibit", XtRBoolean, sizeof (Boolean), offset(utmpInhibit), XtRString, "false"}, {"sunFunctionKeys", "SunFunctionKeys", XtRBoolean, sizeof (Boolean), offset(sunFunctionKeys), XtRString, "false"}, }; #undef offset /* Command line options table. Only resources are entered here...there is a pass over the remaining options after XtParseCommand is let loose. */ static XrmOptionDescRec optionDescList[] = { {"-geometry", "*vt100.geometry",XrmoptionSepArg, (caddr_t) NULL}, {"-132", "*c132", XrmoptionNoArg, (caddr_t) "on"}, {"+132", "*c132", XrmoptionNoArg, (caddr_t) "off"}, {"-ah", "*alwaysHighlight", XrmoptionNoArg, (caddr_t) "on"}, {"+ah", "*alwaysHighlight", XrmoptionNoArg, (caddr_t) "off"}, {"-ar", "*autoRaise", XrmoptionNoArg, (caddr_t) "on"}, {"+ar", "*autoRaise", XrmoptionNoArg, (caddr_t) "off"}, {"-b", "*internalBorder",XrmoptionSepArg, (caddr_t) NULL}, {"-cb", "*cutToBeginningOfLine", XrmoptionNoArg, (caddr_t) "off"}, {"+cb", "*cutToBeginningOfLine", XrmoptionNoArg, (caddr_t) "on"}, {"-cc", "*charClass", XrmoptionSepArg, (caddr_t) NULL}, {"-cn", "*cutNewline", XrmoptionNoArg, (caddr_t) "off"}, {"+cn", "*cutNewline", XrmoptionNoArg, (caddr_t) "on"}, {"-cr", "*cursorColor", XrmoptionSepArg, (caddr_t) NULL}, {"-cu", "*curses", XrmoptionNoArg, (caddr_t) "on"}, {"+cu", "*curses", XrmoptionNoArg, (caddr_t) "off"}, {"-e", NULL, XrmoptionSkipLine, (caddr_t) NULL}, {"-fb", "*boldFont", XrmoptionSepArg, (caddr_t) NULL}, {"-fn", "*font", XrmoptionSepArg, (caddr_t) NULL}, {"-icongeometry",".iconGeometry",XrmoptionSepArg, (caddr_t) NULL}, {"-icontitle", "*iconName", XrmoptionSepArg, (caddr_t) NULL}, {"-j", "*jumpScroll", XrmoptionNoArg, (caddr_t) "on"}, {"+j", "*jumpScroll", XrmoptionNoArg, (caddr_t) "off"}, {"-l", "*logging", XrmoptionNoArg, (caddr_t) "on"}, {"+l", "*logging", XrmoptionNoArg, (caddr_t) "off"}, {"-lf", "*logFile", XrmoptionSepArg, (caddr_t) NULL}, {"-ls", "*loginShell", XrmoptionNoArg, (caddr_t) "on"}, {"+ls", "*loginShell", XrmoptionNoArg, (caddr_t) "off"}, {"-mb", "*marginBell", XrmoptionNoArg, (caddr_t) "on"}, {"+mb", "*marginBell", XrmoptionNoArg, (caddr_t) "off"}, {"-mc", "*multiClickTime", XrmoptionSepArg, (caddr_t) NULL}, {"-ms", "*pointerColor",XrmoptionSepArg, (caddr_t) NULL}, {"-nb", "*nMarginBell", XrmoptionSepArg, (caddr_t) NULL}, {"-rcg", "*recognition", XrmoptionSepArg, (caddr_t) NULL}, {"-rw", "*reverseWrap", XrmoptionNoArg, (caddr_t) "on"}, {"+rw", "*reverseWrap", XrmoptionNoArg, (caddr_t) "off"}, {"-s", "*multiScroll", XrmoptionNoArg, (caddr_t) "on"}, {"+s", "*multiScroll", XrmoptionNoArg, (caddr_t) "off"}, {"-sb", "*scrollBar", XrmoptionNoArg, (caddr_t) "on"}, {"+sb", "*scrollBar", XrmoptionNoArg, (caddr_t) "off"}, {"-sf", "*sunFunctionKeys", XrmoptionNoArg, (caddr_t) "on"}, {"+sf", "*sunFunctionKeys", XrmoptionNoArg, (caddr_t) "off"}, {"-si", "*scrollTtyOutput",XrmoptionNoArg, (caddr_t) "off"}, {"+si", "*scrollTtyOutput",XrmoptionNoArg, (caddr_t) "on"}, {"-sk", "*scrollKey", XrmoptionNoArg, (caddr_t) "on"}, {"+sk", "*scrollKey", XrmoptionNoArg, (caddr_t) "off"}, {"-sl", "*saveLines", XrmoptionSepArg, (caddr_t) NULL}, {"-tm", "*ttyModes", XrmoptionSepArg, (caddr_t) NULL}, {"-tn", "*termName", XrmoptionSepArg, (caddr_t) NULL}, {"-ut", "*utmpInhibit", XrmoptionNoArg, (caddr_t) "on"}, {"+ut", "*utmpInhibit", XrmoptionNoArg, (caddr_t) "off"}, {"-vb", "*visualBell", XrmoptionNoArg, (caddr_t) "on"}, {"+vb", "*visualBell", XrmoptionNoArg, (caddr_t) "off"}, {"-x11r2", "*X11R2", XrmoptionNoArg, (caddr_t) "on"}, {"+x11r2", "*X11R2", XrmoptionNoArg, (caddr_t) "off"}, {"-x11r3", "*X11R2", XrmoptionNoArg, (caddr_t) "off"}, {"+x11r3", "*X11R2", XrmoptionNoArg, (caddr_t) "on"}, /* bogus old compatibility stuff for which there are standard XtInitialize options now */ {"-rv", "*reverseVideo",XrmoptionNoArg, (caddr_t) "on"}, {"+rv", "*reverseVideo",XrmoptionNoArg, (caddr_t) "off"}, }; static struct _options { char *opt; char *desc; } options[] = { { "-help", "print out this message" }, { "-display displayname", "X server to contact" }, { "-geometry geom", "size (in characters) and position" }, { "-/+rv", "turn on/off reverse video" }, { "-/+ar", "turn on/off auto-raising of the rxi window"}, { "-bg color", "background color" }, { "-fg color", "foreground color" }, { "-bd color", "border color" }, { "-bw number", "border width in pixels" }, { "-fn fontname", "normal text font" }, { "-icongeometry", "geometry for the icon" }, { "-iconic", "start iconic" }, { "-icontitle", "title for the icon" }, { "-name string", "client instance, icon, and title strings" }, { "-title string", "title string" }, { "-xrm resourcestring", "additional resource specifications" }, { "-/+132", "turn on/off column switch inhibiting" }, { "-/+ah", "turn on/off always highlight" }, { "-b number", "internal border in pixels" }, { "-/+cb", "turn on/off cut-to-beginning-of-line inhibit" }, { "-cc classrange", "specify additional character classes" }, { "-/+cn", "turn on/off cut newline inhibit" }, { "-cr color", "text cursor color" }, { "-/+cu", "turn on/off curses emulation" }, { "-e command args", "command to execute" }, { "-fb fontname", "bold text font" }, { "-/+j", "turn on/off jump scroll" }, { "-/+l", "turn on/off logging" }, { "-lf filename", "logging filename" }, { "-/+ls", "turn on/off login shell" }, { "-/+mb", "turn on/off margin bell" }, { "-mc milliseconds", "multiclick time in milliseconds" }, { "-ms color", "pointer color" }, { "-nb number", "margin bell in characters from right end" }, { "-/+rw", "turn on/off reverse wraparound" }, { "-/+s", "turn on/off multiscroll" }, { "-/+sb", "turn on/off scrollbar" }, { "-/+sf", "turn on/off Sun Function Key escape codes" }, { "-/+si", "turn on/off scroll-on-tty-output inhibit" }, { "-/+sk", "turn on/off scroll-on-keypress" }, { "-sl number", "number of scrolled lines to save" }, { "-tm string", "terminal mode keywords and characters" }, { "-tn name", "TERM environment variable name" }, { "-/+ut", "turn on/off utmp inhibit" }, { "-/+vb", "turn on/off visual bell" }, { "-/+x11r2", "server is/not an X11R2 server"}, { "-/+x11r3", "server is/not an X11R3 server"}, { "-C", "intercept console messages, if supported" }, { "-Sxxd", "slave mode on \"ttyxx\", file descriptor \"d\"" }, { NULL, NULL }}; static char *message[] = { "Fonts must be fixed width and, if both normal and bold are specified, must", "have the same size. If only a normal font is specified, it will be used for", "both normal and bold text (by doing overstriking). The -e option, if given,", "must be appear at the end of the command line, otherwise the user's default", "shell will be started. Options that start with a plus sign (+) restore the", "default.", NULL}; static void Syntax (badOption) char *badOption; { struct _options *opt; int col; fprintf (stderr, "%s: bad command line option \"%s\"\r\n\n", ProgramName, badOption); fprintf (stderr, "usage: %s", ProgramName); col = 8 + strlen(ProgramName); for (opt = options; opt->opt; opt++) { int len = 3 + strlen(opt->opt); /* space [ string ] */ if (col + len > 79) { fprintf (stderr, "\r\n "); /* 3 spaces */ col = 3; } fprintf (stderr, " [%s]", opt->opt); col += len; } fprintf (stderr, "\r\n\nType %s -help for a full description.\r\n\n", ProgramName); exit (1); } static void Help () { struct _options *opt; char **cpp; fprintf (stderr, "usage:\n %s [-options ...] [-e command args]\n\n", ProgramName); fprintf (stderr, "where options include:\n"); for (opt = options; opt->opt; opt++) { fprintf (stderr, " %-28s %s\n", opt->opt, opt->desc); } putc ('\n', stderr); for (cpp = message; *cpp; cpp++) { fputs (*cpp, stderr); putc ('\n', stderr); } putc ('\n', stderr); exit (0); } extern WidgetClass xtermWidgetClass; Widget toplevel; char *OrNull( str ) char *str; { if (str == NULL) return "(null)"; return str; } main (argc, argv) int argc; char **argv; { register TScreen *screen; register int i, pty; int Xsocket, mode; char *basename(); int xerror(), xioerror(); ProgramName = argv[0]; ttydev = (char *) malloc ((unsigned)strlen (TTYDEV) + 1); ptydev = (char *) malloc ((unsigned)strlen (PTYDEV) + 1); if (!ttydev || !ptydev) { fprintf (stderr, "%s: unable to allocate memory for ttydev or ptydev\n", ProgramName); exit (1); } strcpy (ttydev, TTYDEV); strcpy (ptydev, PTYDEV); #ifdef USE_SYSV_TERMIO /* Initialization is done here rather than above in order ** to prevent any assumptions about the order of the contents ** of the various terminal structures (which may change from ** implementation to implementation). */ #if defined(macII) || defined(att) d_tio.c_iflag = ICRNL|IXON; d_tio.c_oflag = OPOST|ONLCR|TAB3; d_tio.c_cflag = B9600|CS8|CREAD|PARENB|HUPCL; d_tio.c_lflag = ISIG|ICANON|ECHO|ECHOE|ECHOK; d_tio.c_line = 0; d_tio.c_cc[VINTR] = CINTR; d_tio.c_cc[VQUIT] = CQUIT; d_tio.c_cc[VERASE] = CERASE; d_tio.c_cc[VKILL] = CKILL; d_tio.c_cc[VEOF] = CEOF; d_tio.c_cc[VEOL] = CNUL; d_tio.c_cc[VEOL2] = CNUL; d_tio.c_cc[VSWTCH] = CNUL; #ifdef TIOCSLTC d_ltc.t_suspc = CSUSP; /* t_suspc */ d_ltc.t_dsuspc = CDSUSP; /* t_dsuspc */ d_ltc.t_rprntc = 0; /* reserved...*/ d_ltc.t_flushc = 0; d_ltc.t_werasc = 0; d_ltc.t_lnextc = 0; #endif /* TIOCSLTC */ #else /* else !macII */ d_tio.c_iflag = ICRNL|IXON; d_tio.c_oflag = OPOST|ONLCR|TAB3; #ifdef BAUD_0 d_tio.c_cflag = CS8|CREAD|PARENB|HUPCL; #else /* !BAUD_0 */ d_tio.c_cflag = B9600|CS8|CREAD|PARENB|HUPCL; #endif /* !BAUD_0 */ d_tio.c_lflag = ISIG|ICANON|ECHO|ECHOE|ECHOK; d_tio.c_line = 0; d_tio.c_cc[VINTR] = 0x7f; /* DEL */ d_tio.c_cc[VQUIT] = '\\' & 0x3f; /* '^\' */ d_tio.c_cc[VERASE] = '#'; /* '#' */ d_tio.c_cc[VKILL] = '@'; /* '@' */ d_tio.c_cc[VEOF] = 'D' & 0x3f; /* '^D' */ d_tio.c_cc[VEOL] = '@' & 0x3f; /* '^@' */ #ifdef VSWTCH d_tio.c_cc[VSWTCH] = '@' & 0x3f; /* '^@' */ #endif /* VSWTCH */ /* now, try to inherit tty settings */ { int i; for (i = 0; i <= 2; i++) { struct termio deftio; if (ioctl (i, TCGETA, &deftio) == 0) { d_tio.c_cc[VINTR] = deftio.c_cc[VINTR]; d_tio.c_cc[VQUIT] = deftio.c_cc[VQUIT]; d_tio.c_cc[VERASE] = deftio.c_cc[VERASE]; d_tio.c_cc[VKILL] = deftio.c_cc[VKILL]; d_tio.c_cc[VEOF] = deftio.c_cc[VEOF]; d_tio.c_cc[VEOL] = deftio.c_cc[VEOL]; #ifdef VSWTCH d_tio.c_cc[VSWTCH] = deftio.c_cc[VSWTCH]; #endif /* VSWTCH */ break; } } } #ifdef TIOCSLTC d_ltc.t_suspc = '\000'; /* t_suspc */ d_ltc.t_dsuspc = '\000'; /* t_dsuspc */ d_ltc.t_rprntc = '\377'; /* reserved...*/ d_ltc.t_flushc = '\377'; d_ltc.t_werasc = '\377'; d_ltc.t_lnextc = '\377'; #endif /* TIOCSLTC */ #ifdef TIOCLSET d_lmode = 0; #endif /* TIOCLSET */ #endif /* macII */ #endif /* USE_SYSV_TERMIO */ /* Init the Toolkit. */ toplevel = XtInitialize("rxi", "RXI", optionDescList, XtNumber(optionDescList), &argc, argv); XtGetApplicationResources( toplevel, &resource, application_resources, XtNumber(application_resources), NULL, 0 ); /*--Get those "vt100" resources that we need in order to be able to * set up the "real" geometry and such for our shell. */ vt100_resource.f_n = (XFontStruct*)NULL; vt100_resource.f_b = (XFontStruct*)NULL; vt100_resource.f_m = (XFontStruct*)NULL; XtGetSubresources( (Widget)toplevel, (caddr_t)&vt100_resource, "vt100", "Vt100", vt100_resources, XtNumber(vt100_resources), NULL, 0 ); XtGetSubresources( (Widget)toplevel, (caddr_t)&RatMenuDefaults, "menu", "Menu", RatMenuResourceList, Size_RatMenuResourceList, NULL, 0 ); if (vt100_resource.f_n == (XFontStruct*)NULL) { fprintf ( stderr, "Could not get normal font (-fn %s); using 'fixed'.\n", vt100_resource.f_n_name ); vt100_resource.f_b = (XFontStruct*)NULL; /* Force this Off */ vt100_resource.f_n = XLoadQueryFont( XtDisplay(toplevel), "fixed" ); if (vt100_resource.f_n == (XFontStruct*)NULL) { fprintf( stderr, "Cannot load 'fixed' font either?\n" ); Exit(1); } vt100_resource.f_n_name = "fixed"; VTgcFontMask = 0; } if (vt100_resource.f_b == (XFontStruct*)NULL || vt100_resource.f_b_name == NULL || vt100_resource.f_b_name[0] == '\000' || VTgcFontMask == 0 || strcmp( vt100_resource.f_n_name, vt100_resource.f_b_name ) == 0) { vt100_resource.f_b = vt100_resource.f_n; vt100_resource.enbolden = TRUE; /* no bold font */ } else { vt100_resource.enbolden = FALSE; /* has bold font */ } if (vt100_resource.f_m == (XFontStruct*)NULL || vt100_resource.f_m_name == NULL || vt100_resource.f_m_name[0] == '\000' || VTgcFontMask == 0 || strcmp( vt100_resource.f_n_name, vt100_resource.f_m_name ) == 0) { vt100_resource.f_m = vt100_resource.f_n; } /*--Record the user-specified options for drag and such. */ AutoRaise = resource.AutoRaise; AutoRaiseDelay = resource.AutoRaiseDelay; VerticalDragTrigger = resource.VertDragTrigger; VerticalDragAmount = resource.VertDragAmount; HorizontalDragTrigger = resource.HorizDragTrigger; HorizontalDragAmount = resource.HorizDragAmount; /*---Make sure that we have reasonable control values. */ if (HorizontalDragTrigger <= 0) HorizontalDragTrigger = 1; if (HorizontalDragAmount <= 0) HorizontalDragAmount = 1; if (VerticalDragTrigger <= 0) VerticalDragTrigger = 1; if (VerticalDragAmount <= 0) VerticalDragAmount = 1; /*--Decode the RcgValue. It is a string that contains either a decimal number * or else it contains a terminal name, eg. "xsun3" */ { char *p; int rv; p = resource.RcgValue; while (*p == ' ' || *p == '\t') { ++p; } if (*p >= '0' && *p <= '9') { sscanf( p, "%d", &RecognitionValue ); } else { RecognitionValue = -1; Try_RV_Again: for (rv = 0; Rcg_Values[rv].Name != NULL; ++rv) { if (strcmp( p, Rcg_Values[rv].Name ) == 0) { RecognitionValue = Rcg_Values[rv].Code; break; } } if (RecognitionValue == -1) { fprintf( stderr, "Invalid -rcg/-xrm '*recognition:' value: [%s]\n", p ); if (strcmp( p, defRV ) == 0) { RecognitionValue = 0; /* Bug? */ } else { p = defRV; goto Try_RV_Again; } } } } /* * fill in terminal modes */ if (resource.tty_modes) { int n = parse_tty_modes (resource.tty_modes, ttymodelist, NXTTYMODES); if (n < 0) { fprintf (stderr, "%s: bad tty modes \"%s\"\n", ProgramName, resource.tty_modes); } else if (n > 0) { override_tty_modes = 1; } } xterm_name = resource.xterm_name; sunFunctionKeys = resource.sunFunctionKeys; if (strcmp(xterm_name, "-") == 0) xterm_name = "rxi"; /* Parse the rest of the command line */ for (argc--, argv++ ; argc > 0 ; argc--, argv++) { if(**argv != '-') Syntax (*argv); switch(argv[0][1]) { case 'h': Help (); /* NOTREACHED */ case 'v' : fprintf( stdout,"\n---- Version ------------------------\n\n"); fprintf( stdout, " RXI Version: %s\n", RatRXIVersion[0] ); for (i = 1; RatRXIVersion[i] != NULL; ++i) { fprintf( stdout, " %s\n", RatRXIVersion[i] ); } fprintf( stdout,"\n---- Server -------------------------\n\n"); fprintf( stdout, " X Protocol: %d.%d\n", XProtocolVersion(XtDisplay(toplevel)), XProtocolRevision(XtDisplay(toplevel)) ); fprintf( stdout, " Vendor: %s\n", XServerVendor(XtDisplay(toplevel)) ); fprintf( stdout, " Release: %d\n", XVendorRelease(XtDisplay(toplevel)) ); fprintf( stdout,"\n---- Resources ----------------------\n\n"); fprintf( stdout, " name: %s\n", OrNull(resource.xterm_name) ); fprintf( stdout, " title: %s\n", OrNull(resource.title) ); fprintf( stdout, " recognition: %s\n", OrNull(resource.RcgValue) ); fprintf( stdout, " geometry: %s\n", OrNull(vt100_resource.geometry) ); fprintf( stdout, " envMenuFile: %s\n", OrNull(RatMenuDefaults.env_menu_file) ); fprintf( stdout, "\n" ); exit( 0 ); case 'C': #ifdef TIOCCONS Console = TRUE; #endif /* TIOCCONS */ continue; case 'S': sscanf(*argv + 2, "%c%c%d", passedPty, passedPty+1, &am_slave); if (am_slave <= 0) Syntax(*argv); continue; #ifdef DEBUG case 'D': debug = TRUE; continue; #endif /* DEBUG */ case 'e': if (argc <= 1) Syntax (*argv); command_to_exec = ++argv; break; default: Syntax (*argv); } break; } /*--Create the real toplevel widget. */ { Arg Args[20]; int Argsi = 0; int f_width; int f_height; int min_width; int min_height; Widget new_toplevel; static char Geom[32]; #define SETARG(Nm,Vl) XtSetArg( Args[Argsi], Nm, Vl ); ++Argsi ; /*--Get the width and height "box" for the fonts we are to use. */ f_width = vt100_resource.f_n->max_bounds.width; if (vt100_resource.X11R2) { f_height = vt100_resource.f_n->max_bounds.ascent + vt100_resource.f_n->max_bounds.descent; } else { f_height = vt100_resource.f_n->ascent + vt100_resource.f_n->descent; } /*--Figure out how many pixels to allow for borders and scrollbars. */ if (vt100_resource.scrollbar) { static Arg scrollArg[1] = { {XtNborderWidth, (XtArgVal) 1} }; Widget scroll; scroll = XtCreateWidget( "scrollbar", scrollbarWidgetClass, toplevel, scrollArg, (Cardinal)1 ); min_width = 2 * vt100_resource.border + scroll->core.width; min_height = 2 * vt100_resource.border; XtDestroyWidget( scroll ); } else { min_width = 2 * vt100_resource.border; min_height = 2 * vt100_resource.border; } /*--Figure out what the geometry is with this in mind. */ { int x, y; unsigned int w, h; int flags; flags = XParseGeometry( vt100_resource.geometry, &x, &y, &w, &h ); if (!(flags & WidthValue)) { w = defXColumns; } if (!(flags & HeightValue)) { h = defYLines; } #ifndef XtSpecificationRelease w = w*f_width + min_width; h = h*f_height + min_height; #else #if XtSpecificationRelease <= 3 w = w*f_width + min_width; h = h*f_height + min_height; #endif #endif if (flags & XNegative) { x = -x; } if (flags & YNegative) { y = -y; } if (flags & (XValue | YValue)) { sprintf( Geom, "%dx%d%c%d%c%d", w, h, ((flags & XNegative) ? '-' : '+'), x, ((flags & YNegative) ? '-' : '+'), y ); } else { sprintf( Geom, "%dx%d", w, h ); } } /*--See if there is an Icon geometry specification. */ if (resource.icon_geometry != NULL) { int scr, flags, x, y; unsigned int junk; for(scr = 0; /* yyuucchh */ XtScreen(toplevel) != ScreenOfDisplay(XtDisplay(toplevel),scr); scr++); x = 0; y = 0; flags = XParseGeometry( resource.icon_geometry, &x, &y, &junk, &junk ); if (flags & XNegative) { x += WidthOfScreen(ScreenOfDisplay(XtDisplay(toplevel), scr)); } if (flags & YNegative) { y += HeightOfScreen(ScreenOfDisplay(XtDisplay(toplevel), scr)); } SETARG( XtNiconX, x ); SETARG( XtNiconY, y ); } /*--Set title and icon name if not specified. */ if (command_to_exec) { char window_title[1024]; Arg args[2]; if (!resource.title) { if (command_to_exec) { resource.title = basename (command_to_exec[0]); } /* else not reached */ } SETARG( XtNtitle, resource.title ); } if (resource.icon_name == NULL) { if (resource.title != NULL) { resource.icon_name = resource.title; } else { resource.icon_name = resource.xterm_name; } SETARG( XtNiconName, resource.icon_name ); } /*--Now set up the rest of the toplevel arguments. */ SETARG( XtNallowShellResize, TRUE ); SETARG( XtNinput, TRUE ); SETARG( XtNminWidth, min_width ); SETARG( XtNminHeight, min_height ); SETARG( XtNbaseWidth, min_width ); SETARG( XtNbaseHeight, min_height ); SETARG( XtNwidthInc, f_width ); SETARG( XtNheightInc, f_height ); SETARG( XtNgeometry, Geom ); /*--Destroy the old toplevel widget; it has served the purpose of allowing us * to get the various resources that we require. Then recreate it with the * various control values that we had to calculate and could not know before * we had resources. */ XtDestroyWidget( toplevel ); toplevel = XtCreateApplicationShell( resource.xterm_name, applicationShellWidgetClass, Args, (Cardinal)Argsi ); /*--Create the term widget and get its resources. */ term = (XtermWidget) XtCreateManagedWidget( "vt100", xtermWidgetClass, toplevel, (Arg*)NULL, (Cardinal)0 ); /* this causes the initialize method to be called */ } screen = &term->screen; term->flags |= WRAPAROUND; if (!screen->jumpscroll) { term->flags |= SMOOTHSCROLL; } if (term->misc.reverseWrap) { term->flags |= REVERSEWRAP; } inhibit = 0; if (term->misc.logInhibit) inhibit |= I_LOG; if (term->misc.signalInhibit) inhibit |= I_SIGNAL; term->initflags = term->flags; /* set up stderr properly */ i = -1; #ifdef DEBUG if(debug) { #ifndef ultrix i = open ("rxi.debug.log", O_WRONLY | O_CREAT | O_TRUNC, 0666); #else (void)freopen ("rxi.debug.log", "w", stderr ); i = fileno(stderr); #endif /* ultrix */ } #endif /* DEBUG */ if(i >= 0) { #ifdef USE_SYSV_TERMIO /* SYSV has another pointer which should be part of the ** FILE structure but is actually a seperate array. */ unsigned char *old_bufend; old_bufend = (unsigned char *) _bufend(stderr); fileno(stderr) = i; _bufend(stderr) = old_bufend; #else /* USE_SYSV_TERMIO */ #ifndef ultrix fileno(stderr) = i; #endif #endif /* USE_SYSV_TERMIO */ /* mark this file as close on exec */ (void) fcntl(i, F_SETFD, 1); } /* open a terminal for client */ get_terminal (); spawn (); /* Child process is out there, let's catch it's termination */ signal (SIGCHLD, reapchild); /* Realize procs have now been executed */ Xsocket = screen->display->fd; pty = screen->respond; if (am_slave) { /* Write window id so master end can read and use */ char buf[80]; buf[0] = '\0'; sprintf (buf, "%lx\n", XtWindow (XtParent (term))); write (pty, buf, strlen (buf)); } if (term->misc.log_on) { StartLog(screen); } screen->inhibit = inhibit; #ifdef USE_SYSV_TERMIO if (0 > (mode = fcntl(pty, F_GETFL, 0))) Error(); mode |= O_NDELAY; if (fcntl(pty, F_SETFL, mode)) Error(); #else /* USE_SYSV_TERMIO */ mode = 1; if (ioctl (pty, FIONBIO, (char *)&mode) == -1) SysError (ERROR_FIONBIO); #endif /* USE_SYSV_TERMIO */ pty_mask = 1 << pty; X_mask = 1 << Xsocket; Select_mask = pty_mask | X_mask; max_plus1 = (pty < Xsocket) ? (1 + Xsocket) : (1 + pty); #ifdef DEBUG if (debug) printf ("debugging on\n"); #endif /* DEBUG */ XSetErrorHandler(xerror); XSetIOErrorHandler(xioerror); for( ; ; ) VTRun(); } char *basename(name) char *name; { register char *cp; char *rindex(); return((cp = rindex(name, '/')) ? cp + 1 : name); } /* This function opens up a pty master and stuffs it's value into pty. * If it finds one, it returns a value of 0. If it does not find one, * it returns a value of !0. This routine is designed to be re-entrant, * so that if a pty master is found and later, we find that the slave * has problems, we can re-enter this function and get another one. */ get_pty (pty) int *pty; { static int devindex, letter = 0; #ifdef att if ((*pty = open ("/dev/ptmx", O_RDWR)) < 0) { return 1; } return 0; #else /* !att, need lots of code */ #if defined(umips) && defined (SYSTYPE_SYSV) struct stat fstat_buf; *pty = open ("/dev/ptc", O_RDWR); if (*pty < 0 || (fstat (*pty, &fstat_buf)) < 0) { return(1); } sprintf (ttydev, "/dev/ttyq%d", minor(fstat_buf.st_rdev)); sprintf (ptydev, "/dev/ptyq%d", minor(fstat_buf.st_rdev)); if ((*tty = open (ttydev, O_RDWR)) < 0) { close (*pty); return(1); } /* got one! */ return(0); #else /* not (umips && SYSTYPE_SYSV) */ #ifdef CRAY for (; devindex < 256; devindex++) { sprintf (ttydev, "/dev/ttyp%03d", devindex); sprintf (ptydev, "/dev/pty/%03d", devindex); if ((*pty = open (ptydev, O_RDWR)) >= 0) { /* We need to set things up for our next entry * into this function! */ (void) devindex++; return(0); } } #else /* !CRAY */ while (PTYCHAR1[letter]) { ttydev [strlen(ttydev) - 2] = ptydev [strlen(ptydev) - 2] = PTYCHAR1 [letter]; while (PTYCHAR2[devindex]) { ttydev [strlen(ttydev) - 1] = ptydev [strlen(ptydev) - 1] = PTYCHAR2 [devindex]; if ((*pty = open (ptydev, O_RDWR)) >= 0) { /* We need to set things up for our next entry * into this function! */ (void) devindex++; return(0); } devindex++; } devindex = 0; (void) letter++; } #endif /* CRAY else not CRAY */ /* We were unable to allocate a pty master! Return an error * condition and let our caller terminate cleanly. */ return(1); #endif /* umips && SYSTYPE_SYSV */ #endif /* att */ } get_terminal () /* * sets up X and initializes the terminal structure except for term.buf.fildes. */ { register TScreen *screen = &term->screen; screen->arrow = make_colored_cursor (XC_left_ptr, screen->mousecolor, screen->mousecolorback); } static char *vtterm[] = { "rxi", #ifdef USE_X11TERM "x11term", /* for people who want special term name */ #endif "xterm", /* the prefered name, should be fastest */ "vt102", "vt100", "ansi", "dumb", "rational", "facit", "cit", "cit501", 0 }; /* ARGSUSED */ SIGNAL_T hungtty(i) int i; { longjmp(env, 1); SIGNAL_RETURN; } #ifdef USE_HANDSHAKE typedef enum { /* c == child, p == parent */ PTY_BAD, /* c->p: can't open pty slave for some reason */ PTY_FATALERROR, /* c->p: we had a fatal error with the pty */ PTY_GOOD, /* c->p: we have a good pty, let's go on */ PTY_NEW, /* p->c: here is a new pty slave, try this */ PTY_NOMORE, /* p->c; no more pty's, terminate */ UTMP_ADDED, /* c->p: utmp entry has been added */ UTMP_TTYSLOT, /* c->p: here is my ttyslot */ PTY_EXEC /* p->c: window has been mapped the first time */ } status_t; typedef struct { status_t status; int error; int fatal_error; int tty_slot; int rows; int cols; char buffer[1024]; } handshake_t; /* HsSysError() * * This routine does the equivalent of a SysError but it handshakes * over the errno and error exit to the master process so that it can * display our error message and exit with our exit code so that the * user can see it. */ void HsSysError(pf, error) int pf; int error; { handshake_t handshake; handshake.status = PTY_FATALERROR; handshake.error = errno; handshake.fatal_error = error; strcpy(handshake.buffer, ttydev); write(pf, (char*)&handshake, sizeof(handshake)); exit(error); } static int pc_pipe[2]; /* this pipe is used for parent to child transfer */ static int cp_pipe[2]; /* this pipe is used for child to parent transfer */ first_map_occurred () { handshake_t handshake; register TScreen *screen = &term->screen; if (screen->max_row > 0 && screen->max_col > 0) { handshake.status = PTY_EXEC; handshake.rows = screen->max_row; handshake.cols = screen->max_col; write (pc_pipe[1], (char *) &handshake, sizeof(handshake)); close (cp_pipe[0]); close (pc_pipe[1]); } } #else /* * temporary hack to get xterm working on att ptys */ first_map_occurred () { return; } #define HsSysError(a,b) #endif /* USE_HANDSHAKE else !USE_HANDSHAKE */ spawn () /* * Inits pty and tty and forks a login process. * Does not close fd Xsocket. * If slave, the pty named in passedPty is already open for use */ { extern char *SysErrorMsg(); register TScreen *screen = &term->screen; int Xsocket = screen->display->fd; #ifdef USE_HANDSHAKE handshake_t handshake; #else int fds[2]; #endif int tty; int discipline; int done; #ifdef USE_SYSV_TERMIO struct termio tio; struct termio dummy_tio; #ifdef TIOCLSET unsigned lmode; #endif /* TIOCLSET */ #ifdef TIOCSLTC struct ltchars ltc; #endif /* TIOCSLTC */ int one = 1; int zero = 0; int status; #else /* else not USE_SYSV_TERMIO */ unsigned lmode; struct tchars tc; struct ltchars ltc; struct sgttyb sg; #endif /* USE_SYSV_TERMIO */ char termcap [1024]; char newtc [1024]; char *ptr, *shname, *shname_minus; int i; #ifdef USE_SYSV_TERMIO char *dev_tty_name = (char *) 0; int fd; /* for /etc/wtmp */ #endif /* USE_SYSV_TERMIO */ char **envnew; /* new environment */ char buf[32]; char *TermName; int ldisc = 0; #ifdef sun #ifdef TIOCSSIZE struct ttysize ts; #endif /* TIOCSSIZE */ #else /* not sun */ #ifdef TIOCSWINSZ struct winsize ws; #endif /* TIOCSWINSZ */ #endif /* sun */ struct passwd *pw; #ifdef UTMP struct utmp utmp; #ifdef LASTLOG struct lastlog lastlog; #endif /* LASTLOG */ #endif /* UTMP */ screen->uid = getuid(); screen->gid = getgid(); #ifdef SIGTTOU /* so that TIOCSWINSZ || TIOCSIZE doesn't block */ signal(SIGTTOU,SIG_IGN); #endif if (am_slave) { screen->respond = am_slave; ptydev[strlen(ptydev) - 2] = ttydev[strlen(ttydev) - 2] = passedPty[0]; ptydev[strlen(ptydev) - 1] = ttydev[strlen(ttydev) - 1] = passedPty[1]; setgid (screen->gid); setuid (screen->uid); } else { Bool tty_got_hung = False; /* * Sometimes /dev/tty hangs on open (as in the case of a pty * that has gone away). Simply make up some reasonable * defaults. */ signal(SIGALRM, hungtty); alarm(2); /* alarm(1) might return too soon */ if (! setjmp(env)) { tty = open ("/dev/tty", O_RDWR, 0); alarm(0); } else { tty_got_hung = True; tty = -1; errno = ENXIO; } signal(SIGALRM, SIG_DFL); /* * Check results and ignore current control terminal if * necessary. ENXIO is what is normally returned if there is * no controlling terminal, but some systems (e.g. SunOS 4.0) * seem to return EIO. */ if (tty < 0) { if (tty_got_hung || errno == ENXIO || errno == EIO) { #ifdef USE_SYSV_TERMIO tio = d_tio; #ifdef TIOCSLTC ltc = d_ltc; #endif /* TIOCSLTC */ #ifdef TIOCLSET lmode = d_lmode; #endif /* TIOCLSET */ #else /* not USE_SYSV_TERMIO */ sg = d_sg; tc = d_tc; discipline = d_disipline; ltc = d_ltc; lmode = d_lmode; #endif /* USE_SYSV_TERMIO */ } else { SysError(ERROR_OPDEVTTY); } } else { /* get a copy of the current terminal's state */ #ifdef USE_SYSV_TERMIO if(ioctl(tty, TCGETA, &tio) == -1) SysError(ERROR_TIOCGETP); #ifdef RIOS #undef TIOCSLTC #endif /* RIOS */ #ifdef TIOCSLTC if(ioctl(tty, TIOCGLTC, <c) == -1) SysError(ERROR_TIOCGLTC); #endif /* TIOCSLTC */ #ifdef RIOS #undef TIOCLSET #endif /* RIOS */ #ifdef TIOCLSET if(ioctl(tty, TIOCLGET, &lmode) == -1) SysError(ERROR_TIOCLGET); #endif /* TIOCLSET */ #else /* not USE_SYSV_TERMIO */ if(ioctl(tty, TIOCGETP, (char *)&sg) == -1) SysError (ERROR_TIOCGETP); if(ioctl(tty, TIOCGETC, (char *)&tc) == -1) SysError (ERROR_TIOCGETC); if(ioctl(tty, TIOCGETD, (char *)&discipline) == -1) SysError (ERROR_TIOCGETD); if(ioctl(tty, TIOCGLTC, (char *)<c) == -1) SysError (ERROR_TIOCGLTC); if(ioctl(tty, TIOCLGET, (char *)&lmode) == -1) SysError (ERROR_TIOCLGET); #endif /* USE_SYSV_TERMIO */ close (tty); /* tty is no longer an open fd! */ #ifndef apollo /* apollo compains; stupid */ tty = -1; #endif } #ifdef PUCC_PTYD if(-1 == (screen->respond = openrpty(ttydev, ptydev, (resource.utmpInhibit ? OPTY_NOP : OPTY_LOGIN), getuid(), XDisplayString(screen->display)))) { #else /* not PUCC_PTYD */ if (get_pty (&screen->respond)) { #endif /* PUCC_PTYD */ /* no ptys! */ (void) fprintf(stderr, "%s: no available ptys\n", xterm_name); exit (ERROR_PTYS); #ifdef PUCC_PTYD } #else } /* keep braces balanced for emacs */ #endif #ifdef PUCC_PTYD else { /* * set the fd of the master in a global var so * we can undo all this on exit * */ Ptyfd = screen->respond; } #endif /* PUCC_PTYD */ } /* avoid double MapWindow requests */ XtSetMappedWhenManaged( XtParent(term), False ); /* Realize the VT widget, this calls VTRealize (the widget's Realize * proc) */ XtRealizeWidget (XtParent(term)); envnew = vtterm; ptr = termcap; TermName = NULL; if (resource.term_name) { if (tgetent (ptr, resource.term_name) == 1) { TermName = resource.term_name; resize (screen, TermName, termcap, newtc); } else { fprintf (stderr, "%s: invalid termcap entry \"%s\".\n", ProgramName, resource.term_name); } } if (!TermName) { while (*envnew != NULL) { if(tgetent(ptr, *envnew) == 1) { TermName = *envnew; resize(screen, TermName, termcap, newtc); break; } envnew++; } if (TermName == NULL) { fprintf (stderr, "%s: unable to find usable termcap entry.\n", ProgramName); Exit (1); } } #ifdef sun #ifdef TIOCSSIZE /* tell tty how big window is */ ts.ts_lines = screen->max_row + 1; ts.ts_cols = screen->max_col + 1; #endif /* TIOCSSIZE */ #else /* not sun */ #ifdef TIOCSWINSZ /* tell tty how big window is */ ws.ws_row = screen->max_row + 1; ws.ws_col = screen->max_col + 1; ws.ws_xpixel = FullWidth(screen); ws.ws_ypixel = FullHeight(screen); #endif /* TIOCSWINSZ */ #endif /* sun */ if (!am_slave) { #ifdef USE_HANDSHAKE if (pipe(pc_pipe) || pipe(cp_pipe)) SysError (ERROR_FORK); #endif if ((screen->pid = fork ()) == -1) SysError (ERROR_FORK); if (screen->pid == 0) { /* * now in child process */ extern char **environ; int pgrp = getpid(); #ifdef USE_SYSV_TERMIO char numbuf[12]; #endif /* USE_SYSV_TERMIO */ #ifndef USE_HANDSHAKE int ptyfd; setpgrp(); grantpt (screen->respond); unlockpt (screen->respond); if ((ptyfd = open (ptsname(screen->respond), O_RDWR)) < 0) { SysError (1); } if (ioctl (ptyfd, I_PUSH, "ptem") < 0) { SysError (2); } if (!getenv("CONSEM") && ioctl (ptyfd, I_PUSH, "consem") < 0) { SysError (3); } if (ioctl (ptyfd, I_PUSH, "ldterm") < 0) { SysError (4); } tty = ptyfd; close (screen->respond); #ifdef TIOCSWINSZ /* tell tty how big window is */ ws.ws_row = screen->max_row + 1; ws.ws_col = screen->max_col + 1; ws.ws_xpixel = FullWidth(screen); ws.ws_ypixel = FullHeight(screen); #endif #else /* USE_HANDSHAKE: warning, goes for a long ways */ /* close parent's sides of the pipes */ close (cp_pipe[0]); close (pc_pipe[1]); /* Make sure that our sides of the pipes are not in the * 0, 1, 2 range so that we don't fight with stdin, out * or err. */ if (cp_pipe[1] <= 2) { if ((i = fcntl(cp_pipe[1], F_DUPFD, 3)) >= 0) { (void) close(cp_pipe[1]); cp_pipe[1] = i; } } if (pc_pipe[0] <= 2) { if ((i = fcntl(pc_pipe[0], F_DUPFD, 3)) >= 0) { (void) close(pc_pipe[0]); pc_pipe[0] = i; } } /* we don't need the socket, or the pty master anymore */ close (Xsocket); close (screen->respond); /* Now is the time to set up our process group and * open up the pty slave. */ #ifdef USE_SYSV_PGRP (void) setpgrp(); #endif /* USE_SYSV_PGRP */ while (1) { #ifdef TIOCNOTTY if ((tty = open ("/dev/tty", 2)) >= 0) { ioctl (tty, TIOCNOTTY, (char *) NULL); close (tty); } #endif /* TIOCNOTTY */ if ((tty = open(ttydev, O_RDWR, 0)) >= 0) { #ifdef USE_SYSV_PGRP /* We need to make sure that we are acutally * the process group leader for the pty. If * we are, then we should now be able to open * /dev/tty. */ if ((i = open("/dev/tty", O_RDWR, 0)) >= 0) { /* success! */ close(i); break; } #else /* USE_SYSV_PGRP */ break; #endif /* USE_SYSV_PGRP */ } #ifdef TIOCSCTTY ioctl(tty, TIOCSCTTY, 0); #endif /* let our master know that the open failed */ handshake.status = PTY_BAD; handshake.error = errno; strcpy(handshake.buffer, ttydev); write(cp_pipe[1], (char *) &handshake, sizeof(handshake)); /* get reply from parent */ i = read(pc_pipe[0], (char *) &handshake, sizeof(handshake)); if (i <= 0) { /* parent terminated */ exit(1); } if (handshake.status == PTY_NOMORE) { /* No more ptys, let's shutdown. */ exit(1); } /* We have a new pty to try */ free(ttydev); ttydev = malloc((unsigned) (strlen(handshake.buffer) + 1)); strcpy(ttydev, handshake.buffer); } /* use the same tty name that everyone else will use ** (from ttyname) */ if (ptr = ttyname(tty)) { /* it may be bigger */ ttydev = realloc (ttydev, (unsigned) (strlen(ptr) + 1)); (void) strcpy(ttydev, ptr); } #endif /* !USE_HANDSHAKE else USE_HANDSHAKE - from near fork */ #ifdef USE_TTY_GROUP { #include <grp.h> struct group *ttygrp; if (ttygrp = getgrnam("tty")) { /* change ownership of tty to real uid, "tty" gid */ chown (ttydev, screen->uid, ttygrp->gr_gid); chmod (ttydev, 0620); } else { /* change ownership of tty to real group and user id */ chown (ttydev, screen->uid, screen->gid); chmod (ttydev, 0622); } endgrent(); } #else /* else !USE_TTY_GROUP */ /* change ownership of tty to real group and user id */ chown (ttydev, screen->uid, screen->gid); /* change protection of tty */ chmod (ttydev, 0622); #endif /* USE_TTY_GROUP */ /* * set up the tty modes */ { #ifdef USE_SYSV_TERMIO #ifdef umips /* If the control tty had its modes screwed around with, eg. by lineedit in the shell, or emacs, etc. then tio will have bad values. Let's just get termio from the new tty and tailor it. */ if (ioctl (tty, TCGETA, &tio) == -1) SysError (ERROR_TIOCGETP); tio.c_lflag |= ECHOE; #endif /* umips */ /* Now is also the time to change the modes of the * child pty. */ /* input: nl->nl, don't ignore cr, cr->nl */ tio.c_iflag &= ~(INLCR|IGNCR); tio.c_iflag |= ICRNL; /* ouput: cr->cr, nl is not return, no delays, ln->cr/nl */ tio.c_oflag &= ~(OCRNL|ONLRET|NLDLY|CRDLY|TABDLY|BSDLY|VTDLY|FFDLY); tio.c_oflag |= ONLCR; #ifdef BAUD_0 /* baud rate is 0 (don't care) */ tio.c_cflag &= ~(CBAUD); #else /* !BAUD_0 */ /* baud rate is 9600 (nice default) */ tio.c_cflag &= ~(CBAUD); tio.c_cflag |= B9600; #endif /* !BAUD_0 */ /* enable signals, canonical processing (erase, kill, etc), ** echo */ tio.c_lflag |= ISIG|ICANON|ECHO; /* reset EOL to defalult value */ tio.c_cc[VEOL] = '@' & 0x3f; /* '^@' */ /* certain shells (ksh & csh) change EOF as well */ tio.c_cc[VEOF] = 'D' & 0x3f; /* '^D' */ #define TMODE(ind,var) if (ttymodelist[ind].set) var = ttymodelist[ind].value; if (override_tty_modes) { /* sysv-specific */ TMODE (XTTYMODE_intr, tio.c_cc[VINTR]); TMODE (XTTYMODE_quit, tio.c_cc[VQUIT]); TMODE (XTTYMODE_erase, tio.c_cc[VERASE]); TMODE (XTTYMODE_kill, tio.c_cc[VKILL]); TMODE (XTTYMODE_eof, tio.c_cc[VEOF]); TMODE (XTTYMODE_eol, tio.c_cc[VEOL]); #ifdef VSWTCH TMODE (XTTYMODE_swtch, d_tio.c_cc[VSWTCH]); #endif #ifdef TIOCSLTC /* both SYSV and BSD have ltchars */ TMODE (XTTYMODE_susp, ltc.t_suspc); TMODE (XTTYMODE_dsusp, ltc.t_dsuspc); TMODE (XTTYMODE_rprnt, ltc.t_rprntc); TMODE (XTTYMODE_flush, ltc.t_flushc); TMODE (XTTYMODE_weras, ltc.t_werasc); TMODE (XTTYMODE_lnext, ltc.t_lnextc); #endif } #undef TMODE if (ioctl (tty, TCSETA, &tio) == -1) HsSysError(cp_pipe[1], ERROR_TIOCSETP); #ifdef TIOCSLTC if (ioctl (tty, TIOCSLTC, <c) == -1) HsSysError(cp_pipe[1], ERROR_TIOCSETC); #endif /* TIOCSLTC */ #ifdef TIOCLSET if (ioctl (tty, TIOCLSET, (char *)&lmode) == -1) HsSysError(cp_pipe[1], ERROR_TIOCLSET); #endif /* TIOCLSET */ #ifdef TIOCCONS if (Console) { int on = 1; if (ioctl (tty, TIOCCONS, (char *)&on) == -1) HsSysError(cp_pipe[1], ERROR_TIOCCONS); } #endif /* TIOCCONS */ #else /* USE_SYSV_TERMIO */ sg.sg_flags &= ~(ALLDELAY | XTABS | CBREAK | RAW); sg.sg_flags |= ECHO | CRMOD; /* make sure speed is set on pty so that editors work right*/ sg.sg_ispeed = B9600; sg.sg_ospeed = B9600; /* reset t_brkc to default value */ tc.t_brkc = -1; #define TMODE(ind,var) if (ttymodelist[ind].set) var = ttymodelist[ind].value; if (override_tty_modes) { TMODE (XTTYMODE_intr, tc.t_intrc); TMODE (XTTYMODE_quit, tc.t_quitc); TMODE (XTTYMODE_erase, sg.sg_erase); TMODE (XTTYMODE_kill, sg.sg_kill); TMODE (XTTYMODE_eof, tc.t_eofc); TMODE (XTTYMODE_start, tc.t_startc); TMODE (XTTYMODE_stop, tc.t_stopc); TMODE (XTTYMODE_brk, tc.t_brkc); /* both SYSV and BSD have ltchars */ TMODE (XTTYMODE_susp, ltc.t_suspc); TMODE (XTTYMODE_dsusp, ltc.t_dsuspc); TMODE (XTTYMODE_rprnt, ltc.t_rprntc); TMODE (XTTYMODE_flush, ltc.t_flushc); TMODE (XTTYMODE_weras, ltc.t_werasc); TMODE (XTTYMODE_lnext, ltc.t_lnextc); } #undef TMODE if (ioctl (tty, TIOCSETP, (char *)&sg) == -1) HsSysError (cp_pipe[1], ERROR_TIOCSETP); if (ioctl (tty, TIOCSETC, (char *)&tc) == -1) HsSysError (cp_pipe[1], ERROR_TIOCSETC); if (ioctl (tty, TIOCSETD, (char *)&discipline) == -1) HsSysError (cp_pipe[1], ERROR_TIOCSETD); if (ioctl (tty, TIOCSLTC, (char *)<c) == -1) HsSysError (cp_pipe[1], ERROR_TIOCSLTC); if (ioctl (tty, TIOCLSET, (char *)&lmode) == -1) HsSysError (cp_pipe[1], ERROR_TIOCLSET); #ifdef TIOCCONS if (Console) { int on = 1; if (ioctl (tty, TIOCCONS, (char *)&on) == -1) HsSysError(cp_pipe[1], ERROR_TIOCCONS); } #endif /* TIOCCONS */ #endif /* !USE_SYSV_TERMIO */ } signal (SIGCHLD, SIG_DFL); #ifdef att /* watch out for extra shells (I don't understand either) */ signal (SIGHUP, SIG_DFL); #else signal (SIGHUP, SIG_IGN); #endif /* restore various signals to their defaults */ signal (SIGINT, SIG_DFL); signal (SIGQUIT, SIG_DFL); signal (SIGTERM, SIG_DFL); /* copy the environment before Setenving */ for (i = 0 ; environ [i] != NULL ; i++) ; /* * The `4' (`5' for SYSV) is the number of Setenv() * calls which may add a new entry to the environment. * The `1' is for the NULL terminating entry. */ #ifdef USE_SYSV_ENVVARS envnew = (char **) calloc ((unsigned) i + (5 + 1), sizeof(char *)); #else envnew = (char **) calloc ((unsigned) i + (4 + 1), sizeof(char *)); #endif /* USE_SYSV_ENVVARS */ bcopy((char *)environ, (char *)envnew, i * sizeof(char *)); environ = envnew; Setenv ("TERM=", TermName); if(!TermName) *newtc = 0; sprintf (buf, "%lu", ((unsigned long) XtWindow (XtParent(term)))); Setenv ("WINDOWID=", buf); /* put the display into the environment of the shell*/ Setenv ("DISPLAY=", XDisplayString (screen->display)); signal(SIGTERM, SIG_DFL); /* this is the time to go and set up stdin, out, and err */ { /* dup the tty */ for (i = 0; i <= 2; i++) if (i != tty) { (void) close(i); (void) dup(tty); } #ifndef att /* and close the tty */ if (tty > 2) (void) close(tty); #endif } #ifndef USE_SYSV_PGRP #ifdef TIOCSCTTY setsid(); ioctl(0, TIOCSCTTY, 0); #endif ioctl(0, TIOCSPGRP, (char *)&pgrp); setpgrp(0,0); close(open(ttydev, O_WRONLY, 0)); setpgrp (0, pgrp); #endif /* USE_SYSV_PGRP */ #ifdef UTMP #ifdef USE_SYSV_UTMP /* Set up our utmp entry now. We need to do it here ** for the following reasons: ** - It needs to have our correct process id (for ** login). ** - If our parent was to set it after the fork(), ** it might make it out before we need it. ** - We need to do it before we go and change our ** user and group id's. */ (void) setutent (); /* set up entry to search for */ (void) strncpy(utmp.ut_id,ttydev + strlen(ttydev) - 2, sizeof (utmp.ut_id)); utmp.ut_type = DEAD_PROCESS; /* position to entry in utmp file */ (void) getutid(&utmp); /* set up the new entry */ pw = getpwuid(screen->uid); utmp.ut_type = USER_PROCESS; utmp.ut_exit.e_exit = 2; (void) strncpy(utmp.ut_user, (pw && pw->pw_name) ? pw->pw_name : "????", sizeof(utmp.ut_user)); (void) strncpy(utmp.ut_id, ttydev + strlen(ttydev) - 2, sizeof(utmp.ut_id)); (void) strncpy (utmp.ut_line, ttydev + strlen("/dev/"), sizeof (utmp.ut_line)); #ifdef HAS_UTMP_UT_HOST (void) strncpy(utmp.ut_host, DisplayString(screen->display), sizeof(utmp.ut_host)); #endif (void) strncpy(utmp.ut_name, pw->pw_name, sizeof(utmp.ut_name)); utmp.ut_pid = getpid(); utmp.ut_time = time ((long *) 0); /* write out the entry */ if (!resource.utmpInhibit) (void) pututline(&utmp); /* close the file */ (void) endutent(); #else /* USE_SYSV_UTMP */ /* We can now get our ttyslot! We can also set the initial * UTMP entry. */ tslot = ttyslot(); added_utmp_entry = False; { if ((pw = getpwuid(screen->uid)) && !resource.utmpInhibit && (i = open(etc_utmp, O_WRONLY)) >= 0) { bzero((char *)&utmp, sizeof(struct utmp)); (void) strncpy(utmp.ut_line, ttydev + strlen("/dev/"), sizeof(utmp.ut_line)); (void) strncpy(utmp.ut_name, pw->pw_name, sizeof(utmp.ut_name)); #ifdef HAS_UTMP_UT_HOST (void) strncpy(utmp.ut_host, XDisplayString (screen->display), sizeof(utmp.ut_host)); #endif time(&utmp.ut_time); lseek(i, (long)(tslot * sizeof(struct utmp)), 0); write(i, (char *)&utmp, sizeof(struct utmp)); close(i); added_utmp_entry = True; #ifdef WTMP if (term->misc.login_shell && (i = open(etc_wtmp, O_WRONLY|O_APPEND)) >= 0) { write(i, (char *)&utmp, sizeof(struct utmp)); close(i); } #endif /* WTMP */ #ifdef LASTLOG if (term->misc.login_shell && (i = open(etc_lastlog, O_WRONLY)) >= 0) { bzero((char *)&lastlog, sizeof (struct lastlog)); (void) strncpy(lastlog.ll_line, ttydev + sizeof("/dev"), sizeof (lastlog.ll_line)); (void) strncpy(lastlog.ll_host, XDisplayString (screen->display), sizeof (lastlog.ll_host)); time(&lastlog.ll_time); lseek(i, (long)(screen->uid * sizeof (struct lastlog)), 0); write(i, (char *)&lastlog, sizeof (struct lastlog)); close(i); } #endif /* LASTLOG */ } else tslot = -tslot; } /* Let's pass our ttyslot to our parent so that it can * clean up after us. */ #ifdef USE_HANDSHAKE handshake.tty_slot = tslot; #endif /* USE_HANDSHAKE */ #endif /* USE_SYSV_UTMP */ #ifdef USE_HANDSHAKE /* Let our parent know that we set up our utmp entry * so that it can clean up after us. */ handshake.status = UTMP_ADDED; handshake.error = 0; strcpy(handshake.buffer, ttydev); (void) write(cp_pipe[1], &handshake, sizeof(handshake)); #endif /* USE_HANDSHAKE */ #endif/* UTMP */ (void) setgid (screen->gid); #ifdef HAS_BSD_GROUPS if (geteuid() == 0 && pw) initgroups (pw->pw_name, pw->pw_gid); #endif (void) setuid (screen->uid); #ifdef USE_HANDSHAKE /* mark the pipes as close on exec */ fcntl(cp_pipe[1], F_SETFD, 1); fcntl(pc_pipe[0], F_SETFD, 1); /* We are at the point where we are going to * exec our shell (or whatever). Let our parent * know we arrived safely. */ handshake.status = PTY_GOOD; handshake.error = 0; (void) strcpy(handshake.buffer, ttydev); (void) write(cp_pipe[1], &handshake, sizeof(handshake)); #endif /* USE_HANDSHAKE */ #ifdef USE_SYSV_ENVVARS sprintf (numbuf, "%d", screen->max_col + 1); Setenv("COLUMNS=", numbuf); sprintf (numbuf, "%d", screen->max_row + 1); Setenv("LINES=", numbuf); #else strcpy (termcap, newtc); resize (screen, TermName, termcap, newtc); if (term->misc.titeInhibit) { remove_termcap_entry (newtc, ":ti="); remove_termcap_entry (newtc, ":te="); } Setenv ("TERMCAP=", newtc); #endif /* USE_SYSV_ENVVAR */ /* need to reset after all the ioctl bashing we did above */ #ifdef sun #ifdef TIOCSSIZE ioctl (0, TIOCSSIZE, &ts); #endif /* TIOCSSIZE */ #else /* not sun */ #ifdef TIOCSWINSZ ioctl (0, TIOCSWINSZ, (char *)&ws); #endif /* TIOCSWINSZ */ #endif /* sun */ signal(SIGHUP, SIG_DFL); if (command_to_exec) { execvp(*command_to_exec, command_to_exec); /* print error message on screen */ fprintf(stderr, "%s: Can't execvp %s\n", xterm_name, *command_to_exec); } #ifdef att /* fix pts sh hanging around */ signal (SIGHUP, SIG_DFL); #endif #ifdef UTMP if(((ptr = getenv("SHELL")) == NULL || *ptr == 0) && ((pw == NULL && (pw = getpwuid(screen->uid)) == NULL) || *(ptr = pw->pw_shell) == 0)) #else /* UTMP */ if(((ptr = getenv("SHELL")) == NULL || *ptr == 0) && ((pw = getpwuid(screen->uid)) == NULL || *(ptr = pw->pw_shell) == 0)) #endif /* UTMP */ ptr = "/bin/sh"; if(shname = rindex(ptr, '/')) shname++; else shname = ptr; shname_minus = malloc(strlen(shname) + 2); (void) strcpy(shname_minus, "-"); (void) strcat(shname_minus, shname); #ifndef USE_SYSV_TERMIO ldisc = XStrCmp("csh", shname + strlen(shname) - 3) == 0 ? NTTYDISC : 0; ioctl(0, TIOCSETD, (char *)&ldisc); #endif /* !USE_SYSV_TERMIO */ #ifdef USE_LOGIN_DASH_P if (term->misc.login_shell && pw && added_utmp_entry) execl (bin_login, "login", "-p", "-f", pw->pw_name, 0); #endif execlp (ptr, (term->misc.login_shell ? shname_minus : shname), 0); /* Exec failed. */ fprintf (stderr, "%s: Could not exec %s!\n", xterm_name, ptr); sleep(5); exit(ERROR_EXEC); } /* end if in child after fork */ #ifdef USE_HANDSHAKE /* Parent process. Let's handle handshaked requests to our * child process. */ /* close childs's sides of the pipes */ close (cp_pipe[1]); close (pc_pipe[0]); for (done = 0; !done; ) { if (read(cp_pipe[0], &handshake, sizeof(handshake)) <= 0) { /* Our child is done talking to us. If it terminated * due to an error, we will catch the death of child * and clean up. */ break; } switch(handshake.status) { case PTY_GOOD: /* Success! Let's free up resources and * continue. */ done = 1; break; case PTY_BAD: /* The open of the pty failed! Let's get * another one. */ (void) close(screen->respond); if (get_pty(&screen->respond)) { /* no more ptys! */ (void) fprintf(stderr, "%s: no available ptys\n", xterm_name); handshake.status = PTY_NOMORE; write(pc_pipe[1], &handshake, sizeof(handshake)); exit (ERROR_PTYS); } handshake.status = PTY_NEW; (void) strcpy(handshake.buffer, ttydev); write(pc_pipe[1], &handshake, sizeof(handshake)); break; case PTY_FATALERROR: errno = handshake.error; close(cp_pipe[0]); close(pc_pipe[1]); SysError(handshake.fatal_error); case UTMP_ADDED: /* The utmp entry was set by our slave. Remember * this so that we can reset it later. */ added_utmp_entry = True; #ifndef USE_SYSV_UTMP tslot = handshake.tty_slot; #endif /* USE_SYSV_UTMP */ free(ttydev); ttydev = malloc((unsigned) strlen(handshake.buffer) + 1); strcpy(ttydev, handshake.buffer); break; } } /* close our sides of the pipes */ close (cp_pipe[0]); close (pc_pipe[1]); #endif /* USE_HANDSHAKE */ } /* end if no slave */ /* * still in parent (xterm process) */ #ifdef att /* hung sh problem? */ signal (SIGHUP, SIG_DFL); #else signal (SIGHUP,SIG_IGN); #endif /* * Unfortunately, System V seems to have trouble divorcing the child process * from the process group of xterm. This is a problem because hitting the * INTR or QUIT characters on the keyboard will cause xterm to go away if we * don't ignore the signals. This is annoying. */ #if defined(USE_SYSV_SIGNALS) && !defined(SIGTSTP) signal (SIGINT, SIG_IGN); #ifndef att /* hung shell problem */ signal (SIGQUIT, SIG_IGN); #endif signal (SIGTERM, SIG_IGN); #else /* else is bsd or has job control */ #ifdef SYSV /* if we were spawned by a jobcontrol smart shell (like ksh or csh), * then our pgrp and pid will be the same. If we were spawned by * a jobcontrol dump shell (like /bin/sh), then we will be in out * parents pgrp, and we must ignore keyboard signals, or will will * tank on everything. */ if (getpid() == getpgrp()) { (void) signal(SIGINT, Exit); (void) signal(SIGQUIT, Exit); (void) signal(SIGTERM, Exit); } else { (void) signal(SIGINT, SIG_IGN); (void) signal(SIGQUIT, SIG_IGN); (void) signal(SIGTERM, SIG_IGN); } #else /* SYSV */ signal (SIGINT, Exit); signal (SIGQUIT, Exit); signal (SIGTERM, Exit); #endif /* SYSV */ #endif /* USE_SYSV_SIGNALS and not SIGTSTP */ return; } /* end spawn */ SIGNAL_T Exit(n) int n; { register TScreen *screen = &term->screen; int pty = term->screen.respond; /* file descriptor of pty */ #ifdef UTMP #ifdef USE_SYSV_UTMP struct utmp utmp; struct utmp *utptr; #ifdef WTMP int fd; /* for /etc/wtmp */ #endif /* cleanup the utmp entry we forged earlier */ if (!resource.utmpInhibit && added_utmp_entry) { #ifdef CRAY #define PTYCHARLEN 4 #else #define PTYCHARLEN 2 #endif utmp.ut_type = USER_PROCESS; (void) strncpy(utmp.ut_id, ttydev + strlen(ttydev) - PTYCHARLEN, sizeof(utmp.ut_id)); (void) setutent(); utptr = getutid(&utmp); /* write it out only if it exists, and the pid's match */ if (utptr && (utptr->ut_pid = screen->pid)) { utptr->ut_type = DEAD_PROCESS; utptr->ut_time = time((long *) 0); (void) pututline(utptr); #ifdef WTMP /* set wtmp entry if wtmp file exists */ if ((fd = open(etc_wtmp, O_WRONLY | O_APPEND)) >= 0) { (void) write(fd, utptr, sizeof(utmp)); (void) close(fd); } #endif } (void) endutent(); } #else /* not USE_SYSV_UTMP */ register int i; struct utmp utmp; if (!resource.utmpInhibit && added_utmp_entry && (!am_slave && tslot > 0 && (i = open(etc_utmp, O_WRONLY)) >= 0)) { bzero((char *)&utmp, sizeof(struct utmp)); lseek(i, (long)(tslot * sizeof(struct utmp)), 0); write(i, (char *)&utmp, sizeof(struct utmp)); close(i); #ifdef WTMP if (term->misc.login_shell && (i = open(etc_wtmp, O_WRONLY | O_APPEND)) >= 0) { (void) strncpy(utmp.ut_line, ttydev + sizeof("/dev"), sizeof (utmp.ut_line)); time(&utmp.ut_time); write(i, (char *)&utmp, sizeof(struct utmp)); close(i); } #endif /* WTMP */ } #endif /* USE_SYSV_UTMP */ #endif /* UTMP */ close(pty); /* close explicitly to avoid race with slave side */ if(screen->logging) CloseLog(screen); if (!am_slave) { /* restore ownership of tty and pty */ chown (ttydev, 0, 0); chown (ptydev, 0, 0); /* restore modes of tty and pty */ chmod (ttydev, 0666); chmod (ptydev, 0666); } exit(n); SIGNAL_RETURN; } /* ARGSUSED */ resize(screen, TermName, oldtc, newtc) TScreen *screen; char *TermName; register char *oldtc, *newtc; { #ifndef USE_SYSV_ENVVARS register char *ptr1, *ptr2; register int i; register int li_first = 0; register char *temp; char *index(); if ((ptr1 = strindex (oldtc, "co#")) == NULL){ strcat (oldtc, "co#80:"); ptr1 = strindex (oldtc, "co#"); } if ((ptr2 = strindex (oldtc, "li#")) == NULL){ strcat (oldtc, "li#24:"); ptr2 = strindex (oldtc, "li#"); } if(ptr1 > ptr2) { li_first++; temp = ptr1; ptr1 = ptr2; ptr2 = temp; } ptr1 += 3; ptr2 += 3; strncpy (newtc, oldtc, i = ptr1 - oldtc); newtc += i; sprintf (newtc, "%d", li_first ? screen->max_row + 1 : screen->max_col + 1); newtc += strlen(newtc); ptr1 = index (ptr1, ':'); strncpy (newtc, ptr1, i = ptr2 - ptr1); newtc += i; sprintf (newtc, "%d", li_first ? screen->max_col + 1 : screen->max_row + 1); ptr2 = index (ptr2, ':'); strcat (newtc, ptr2); #endif /* USE_SYSV_ENVVARS */ } static SIGNAL_T reapchild () { #if defined(USE_SYSV_SIGNALS) && !defined(SIGTSTP) int status, pid; pid = wait(&status); if (pid == -1) { (void) signal(SIGCHLD, reapchild); SIGNAL_RETURN; } #else /* defined(USE_SYSV_SIGNALS) && !defined(SIGTSTP) */ union wait status; register int pid; #ifdef DEBUG if (debug) fputs ("Exiting\n", stderr); #endif /* DEBUG */ pid = wait3 (&status, WNOHANG, (struct rusage *)NULL); if (!pid) { #ifdef USE_SYSV_SIGNALS (void) signal(SIGCHLD, reapchild); #endif /* USE_SYSV_SIGNALS */ SIGNAL_RETURN; } #endif /* defined(USE_SYSV_SIGNALS) && !defined(SIGTSTP) */ #ifdef PUCC_PTYD closepty(ttydev, ptydev, (resource.utmpInhibit ? OPTY_NOP : OPTY_LOGIN), Ptyfd); #endif /* PUCC_PTYD */ if (pid != term->screen.pid) { #ifdef USE_SYSV_SIGNALS (void) signal(SIGCHLD, reapchild); #endif /* USE_SYSV_SIGNALS */ SIGNAL_RETURN; } /* * Use pid instead of process group (which would have to get before * the wait call above) so that we don't accidentally hose other * applications. Otherwise, somebody could write a program which put * itself in somebody else's process group. Also, we call Exit instead * of Cleanup so that we don't do a killpg on -1 by accident. Some * operating systems seem to do very nasty things with that. */ if (pid > 1) { killpg (pid, SIGHUP); } Exit (0); SIGNAL_RETURN; } /* VARARGS1 */ consolepr(fmt,x0,x1,x2,x3,x4,x5,x6,x7,x8,x9) char *fmt; { extern int errno; extern char *SysErrorMsg(); int oerrno; int f; char buf[ BUFSIZ ]; oerrno = errno; strcpy(buf, "rxi: "); sprintf(buf+strlen(buf), fmt, x0,x1,x2,x3,x4,x5,x6,x7,x8,x9); strcat(buf, ": "); strcat(buf, SysErrorMsg (oerrno)); strcat(buf, "\n"); f = open("/dev/console",O_WRONLY); write(f, buf, strlen(buf)); close(f); #ifdef TIOCNOTTY if ((f = open("/dev/tty", 2)) >= 0) { ioctl(f, TIOCNOTTY, (char *)NULL); close(f); } #endif /* TIOCNOTTY */ } remove_termcap_entry (buf, str) char *buf; char *str; { register char *strinbuf; strinbuf = strindex (buf, str); if (strinbuf) { register char *colonPtr = index (strinbuf+1, ':'); if (colonPtr) { while (*colonPtr) { *strinbuf++ = *colonPtr++; /* copy down */ } *strinbuf = '\0'; } else { strinbuf[1] = '\0'; } } return; } /* * parse_tty_modes accepts lines of the following form: * * [SETTING] ... * * where setting consists of the words in the modelist followed by a character * or ^char. */ static int parse_tty_modes (s, modelist, nmodes) char *s; struct _xttymodes *modelist; int nmodes; { struct _xttymodes *mp; int c; int count = 0; while (1) { while (*s && isascii(*s) && isspace(*s)) s++; if (!*s) return count; for (mp = modelist; mp->name; mp++) { if (strncmp (s, mp->name, mp->len) == 0) break; } if (!mp->name) return -1; s += mp->len; while (*s && isascii(*s) && isspace(*s)) s++; if (!*s) return -1; if (*s == '^') { s++; c = ((*s == '?') ? 0177 : *s & 31); /* keep control bits */ } else { c = *s; } mp->value = c; mp->set = 1; count++; s++; } } int GetBytesAvailable (fd) int fd; { #ifdef FIONREAD static long arg; ioctl (fd, FIONREAD, (char *) &arg); return (int) arg; #else struct pollfd pollfds[1]; pollfds[0].fd = fd; pollfds[0].events = POLLIN; return poll (pollfds, 1, 0); #endif }