-/* tip.c 4.13 83/05/17 */
+/*
+ * Copyright (c) 1983 The Regents of the University of California.
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms are permitted
+ * provided that: (1) source distributions retain this entire copyright
+ * notice and comment, and (2) distributions including binaries display
+ * the following acknowledgement: ``This product includes software
+ * developed by the University of California, Berkeley and its contributors''
+ * in the documentation or other materials provided with the distribution
+ * and in all advertising materials mentioning features or use of this
+ * software. Neither the name of the University nor the names of its
+ * contributors may be used to endorse or promote products derived
+ * from this software without specific prior written permission.
+ * THIS SOFTWARE IS PROVIDED ``AS IS'' AND WITHOUT ANY EXPRESS OR
+ * IMPLIED WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED
+ * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE.
+ */
+
+#ifndef lint
+char copyright[] =
+"@(#) Copyright (c) 1983 The Regents of the University of California.\n\
+ All rights reserved.\n";
+#endif /* not lint */
+
+#ifndef lint
+static char sccsid[] = "@(#)tip.c 5.13 (Berkeley) 6/1/90";
+#endif /* not lint */
/*
* tip - UNIX link to other systems
* cu phone-number [-s speed] [-l line] [-a acu]
*/
#include "tip.h"
+#include "pathnames.h"
/*
* Baud rate mapping table
1200, 1800, 2400, 4800, 9600, 19200, -1
};
-#ifdef VMUNIX
int disc = OTTYDISC; /* tip normally runs this way */
-#endif
-
-int intprompt();
-int timeout();
-int cleanup();
+void intprompt();
+void timeout();
+void cleanup();
char *sname();
-extern char *sprintf();
+char PNbuf[256]; /* This limits the size of a number */
main(argc, argv)
char *argv[];
register char *p;
char sbuf[12];
+ gid = getgid();
+ egid = getegid();
+ uid = getuid();
+ euid = geteuid();
if (equal(sname(argv[0]), "cu")) {
- cumain(argc, argv);
cumode = 1;
+ cumain(argc, argv);
goto cucommon;
}
if (system == NOSTR)
goto notnumber;
+ if (isalpha(*system))
+ goto notnumber;
+ /*
+ * System name is really a phone number...
+ * Copy the number then stomp on the original (in case the number
+ * is private, we don't want 'ps' or 'w' to find it).
+ */
+ if (strlen(system) > sizeof PNbuf - 1) {
+ fprintf(stderr, "tip: phone number too long (max = %d bytes)\n",
+ sizeof PNbuf - 1);
+ exit(1);
+ }
+ strncpy( PNbuf, system, sizeof PNbuf - 1 );
for (p = system; *p; p++)
- if (isalpha(*p))
- goto notnumber;
- PN = system; /* system name is really a phone number */
- system = sprintf(sbuf, "tip%d", BR);
+ *p = '\0';
+ PN = PNbuf;
+ (void)sprintf(sbuf, "tip%d", BR);
+ system = sbuf;
notnumber:
- signal(SIGINT, cleanup);
- signal(SIGQUIT, cleanup);
- signal(SIGHUP, cleanup);
- signal(SIGTERM, cleanup);
+ (void)signal(SIGINT, cleanup);
+ (void)signal(SIGQUIT, cleanup);
+ (void)signal(SIGHUP, cleanup);
+ (void)signal(SIGTERM, cleanup);
if ((i = hunt(system)) == 0) {
printf("all ports busy\n");
}
if (i == -1) {
printf("link down\n");
- delock(uucplock);
+ (void)uu_unlock(uucplock);
exit(3);
}
setbuf(stdout, NULL);
loginit();
- /*
- * Now that we have the logfile and the ACU open
- * return to the real uid and gid. These things will
- * be closed on exit. Note that we can't run as root,
- * because locking mechanism on the tty and the accounting
- * will be bypassed.
- */
- setuid(getuid());
- setgid(getgid());
/*
* Kludge, their's no easy way to get the initialization
* in the right order, so force it here
*/
if ((PH = getenv("PHONES")) == NOSTR)
- PH = "/etc/phones";
+ PH = _PATH_PHONES;
vinit(); /* init variables */
+ setparity("even"); /* set the parity table */
if ((i = speed(number(value(BAUDRATE)))) == NULL) {
printf("tip: bad baud rate %d\n", number(value(BAUDRATE)));
- delock(uucplock);
+ (void)uu_unlock(uucplock);
exit(3);
}
+ /*
+ * Now that we have the logfile and the ACU open
+ * return to the real uid and gid. These things will
+ * be closed on exit. Swap real and effective uid's
+ * so we can get the original permissions back
+ * for removing the uucp lock.
+ */
+ user_uid();
+
/*
* Hardwired connections require the
* line speed set before they make any transmissions
ttysetup(i);
if (p = connect()) {
printf("\07%s\n[EOT]\n", p);
- delock(uucplock);
+ daemon_uid();
+ (void)uu_unlock(uucplock);
exit(1);
}
if (!HW)
* From here down the code is shared with
* the "cu" version of tip.
*/
+
ioctl(0, TIOCGETP, (char *)&defarg);
ioctl(0, TIOCGETC, (char *)&defchars);
ioctl(0, TIOCGLTC, (char *)&deflchars);
-#ifdef VMUNIX
ioctl(0, TIOCGETD, (char *)&odisc);
-#endif
arg = defarg;
arg.sg_flags = ANYP | CBREAK;
tchars = defchars;
raw();
pipe(fildes); pipe(repdes);
- signal(SIGALRM, timeout);
+ (void)signal(SIGALRM, timeout);
/*
* Everything's set up now:
- * connection established (hardwired or diaulup)
+ * connection established (hardwired or dialup)
* line conditioned (baud rate, mode, etc.)
* internal data structures (variables)
* so, fork one process for local side and one for remote.
/*NOTREACHED*/
}
+void
cleanup()
{
- delock(uucplock);
-#ifdef VMUNIX
+
+ daemon_uid();
+ (void)uu_unlock(uucplock);
if (odisc)
ioctl(0, TIOCSETD, (char *)&odisc);
-#endif
exit(0);
}
+/*
+ * Muck with user ID's. We are setuid to the owner of the lock
+ * directory when we start. user_uid() reverses real and effective
+ * ID's after startup, to run with the user's permissions.
+ * daemon_uid() switches back to the privileged uid for unlocking.
+ * Finally, to avoid running a shell with the wrong real uid,
+ * shell_uid() sets real and effective uid's to the user's real ID.
+ */
+static int uidswapped;
+
+user_uid()
+{
+ if (uidswapped == 0) {
+ setregid(egid, gid);
+ setreuid(euid, uid);
+ uidswapped = 1;
+ }
+}
+
+daemon_uid()
+{
+
+ if (uidswapped) {
+ setreuid(uid, euid);
+ setregid(gid, egid);
+ uidswapped = 0;
+ }
+}
+
+shell_uid()
+{
+
+ setreuid(uid, uid);
+ setregid(gid, gid);
+}
+
/*
* put the controlling keyboard into raw mode
*/
raw()
{
+
ioctl(0, TIOCSETP, &arg);
ioctl(0, TIOCSETC, &tchars);
ioctl(0, TIOCSLTC, <chars);
-#ifdef VMUNIX
ioctl(0, TIOCSETD, (char *)&disc);
-#endif
}
*/
unraw()
{
-#ifdef VMUNIX
+
ioctl(0, TIOCSETD, (char *)&odisc);
-#endif
ioctl(0, TIOCSETP, (char *)&defarg);
ioctl(0, TIOCSETC, (char *)&defchars);
ioctl(0, TIOCSLTC, (char *)&deflchars);
}
+static jmp_buf promptbuf;
+
/*
* Print string ``s'', then read a string
* in from the terminal. Handles signals & allows use of
register char *p;
{
register char *b = p;
+ sig_t oint, oquit;
stoprompt = 0;
- signal(SIGINT, intprompt);
- signal(SIGQUIT, SIG_IGN);
+ oint = signal(SIGINT, intprompt);
+ oquit = signal(SIGQUIT, SIG_IGN);
unraw();
printf("%s", s);
- while ((*p = getchar()) != EOF && *p != '\n') {
- if (stoprompt)
- goto pbreak;
- p++;
- }
+ if (setjmp(promptbuf) == 0)
+ while ((*p = getchar()) != EOF && *p != '\n')
+ p++;
*p = '\0';
-pbreak:
+
raw();
- signal(SIGINT, SIG_DFL);
- signal(SIGQUIT,SIG_DFL);
- return(stoprompt || p == b);
+ (void)signal(SIGINT, oint);
+ (void)signal(SIGQUIT, oquit);
+ return (stoprompt || p == b);
}
/*
* Interrupt service routine during prompting
*/
+void
intprompt()
{
- signal(SIGINT, SIG_IGN);
+
+ (void)signal(SIGINT, SIG_IGN);
stoprompt = 1;
printf("\r\n");
+ longjmp(promptbuf, 1);
}
/*
continue;
} else if (gch == '\r') {
bol = 1;
- write(FD, &gch, 1);
+ pwrite(FD, &gch, 1);
+ if (boolean(value(HALFDUPLEX)))
+ printf("\r\n");
continue;
} else if (!cumode && gch == character(value(FORCE)))
gch = getchar()&0177;
bol = any(gch, value(EOL));
if (boolean(value(RAISE)) && islower(gch))
- toupper(gch);
- write(FD, &gch, 1);
+ gch = toupper(gch);
+ pwrite(FD, &gch, 1);
+ if (boolean(value(HALFDUPLEX)))
+ printf("%c", gch);
}
}
gch = (getchar()&0177);
for (p = etable; p->e_char; p++)
if (p->e_char == gch) {
- if ((p->e_flags&PRIV) && getuid())
+ if ((p->e_flags&PRIV) && uid)
continue;
printf("%s", ctrl(c));
(*p->e_func)(gch);
- return(0);
+ return (0);
}
/* ESCAPE ESCAPE forces ESCAPE */
if (c != gch)
- write(FD, &c, 1);
- return(gch);
+ pwrite(FD, &c, 1);
+ return (gch);
}
speed(n)
+ int n;
{
register int *p;
for (p = bauds; *p != -1; p++)
if (*p == n)
- return(p-bauds);
- return(NULL);
+ return (p - bauds);
+ return (NULL);
}
any(c, p)
register char c, *p;
{
- if (p)
- while (*p)
+ while (p && *p)
if (*p++ == c)
- return(1);
- return(0);
+ return (1);
+ return (0);
}
size(s)
register char *s;
{
- register int i = 0;
+ register int i = 0;
- while (*s++) i++;
- return(i);
+ while (s && *s++)
+ i++;
+ return (i);
}
char *
;
}
*p = '\0';
- return(buf);
+ return (buf);
}
char *
s[0] = c;
s[1] = '\0';
}
- return(s);
+ return (s);
}
/*
printf("%c\r\n", c);
for (p = etable; p->e_char; p++) {
- if ((p->e_flags&PRIV) && getuid())
+ if ((p->e_flags&PRIV) && uid)
continue;
printf("%2s", ctrl(character(value(ESCAPE))));
printf("%-2s %c %s\r\n", ctrl(p->e_char),
* Set up the "remote" tty's state
*/
ttysetup(speed)
+ int speed;
{
-#ifdef VMUNIX
unsigned bits = LDECCTQ;
-#endif
arg.sg_ispeed = arg.sg_ospeed = speed;
- arg.sg_flags = TANDEM|RAW;
+ arg.sg_flags = RAW;
+ if (boolean(value(TAND)))
+ arg.sg_flags |= TANDEM;
ioctl(FD, TIOCSETP, (char *)&arg);
-#ifdef VMUNIX
ioctl(FD, TIOCLBIS, (char *)&bits);
-#endif
}
/*
p = s;
return (p);
}
+
+static char partab[0200];
+static int bits8;
+
+/*
+ * Do a write to the remote machine with the correct parity.
+ * We are doing 8 bit wide output, so we just generate a character
+ * with the right parity and output it.
+ */
+pwrite(fd, buf, n)
+ int fd;
+ char *buf;
+ register int n;
+{
+ register int i;
+ register char *bp;
+ extern int errno;
+
+ bp = buf;
+ if (bits8 == 0)
+ for (i = 0; i < n; i++) {
+ *bp = partab[(*bp) & 0177];
+ bp++;
+ }
+ if (write(fd, buf, n) < 0) {
+ if (errno == EIO)
+ abort("Lost carrier.");
+ /* this is questionable */
+ perror("write");
+ }
+}
+
+/*
+ * Build a parity table with appropriate high-order bit.
+ */
+setparity(defparity)
+ char *defparity;
+{
+ register int i;
+ char *parity;
+ extern char evenpartab[];
+
+ if (value(PARITY) == NOSTR)
+ value(PARITY) = defparity;
+ parity = value(PARITY);
+ if (equal(parity, "none")) {
+ bits8 = 1;
+ return;
+ } else
+ bits8 = 0;
+ for (i = 0; i < 0200; i++)
+ partab[i] = evenpartab[i];
+ if (equal(parity, "even"))
+ return;
+ if (equal(parity, "odd")) {
+ for (i = 0; i < 0200; i++)
+ partab[i] ^= 0200; /* reverse bit 7 */
+ return;
+ }
+ if (equal(parity, "zero")) {
+ for (i = 0; i < 0200; i++)
+ partab[i] &= ~0200; /* turn off bit 7 */
+ return;
+ }
+ if (equal(parity, "one")) {
+ for (i = 0; i < 0200; i++)
+ partab[i] |= 0200; /* turn on bit 7 */
+ return;
+ }
+ fprintf(stderr, "%s: unknown parity value\n", PA);
+ fflush(stderr);
+}