[comp.bugs.4bsd.ucb-fixes] V1.60

bostic@OKEEFFE.BERKELEY.EDU (Keith Bostic) (06/11/88)

Subject: rlogind/telnetd security problem
Index: etc/{rlogind.c,telnetd.c} 4.3BSD

Description:
	There is a security problem in rlogind and telnetd as
	distributed in 4.3BSD.
Fix:
	Apply the attached patches to rlogind.c and telnetd.c.

*** rlogind.c.orig	Sun May 22 16:41:03 1988
--- rlogind.c.new	Sun May 22 16:48:06 1988
***************
*** 44,59 ****
  # define TIOCPKT_WINDOW 0x80
  # endif TIOCPKT_WINDOW
  
! extern	errno;
  int	reapchild();
  struct	passwd *getpwnam();
  char	*malloc();
  
  main(argc, argv)
  	int argc;
  	char **argv;
  {
! 	int on = 1, options = 0, fromlen;
  	struct sockaddr_in from;
  
  	openlog("rlogind", LOG_PID | LOG_AUTH, LOG_AUTH);
--- 44,60 ----
  # define TIOCPKT_WINDOW 0x80
  # endif TIOCPKT_WINDOW
  
! extern	int errno;
  int	reapchild();
  struct	passwd *getpwnam();
  char	*malloc();
  
+ /*ARGSUSED*/
  main(argc, argv)
  	int argc;
  	char **argv;
  {
! 	int on = 1, fromlen;
  	struct sockaddr_in from;
  
  	openlog("rlogind", LOG_PID | LOG_AUTH, LOG_AUTH);
***************
*** 72,78 ****
  int	child;
  int	cleanup();
  int	netf;
- extern	errno;
  char	*line;
  extern	char	*inet_ntoa();
  
--- 73,78 ----
***************
*** 115,122 ****
  		if (stat(line, &stb) < 0)
  			break;
  		for (i = 0; i < 16; i++) {
! 			line[strlen("/dev/ptyp")] = "0123456789abcdef"[i];
! 			p = open(line, 2);
  			if (p > 0)
  				goto gotpty;
  		}
--- 115,122 ----
  		if (stat(line, &stb) < 0)
  			break;
  		for (i = 0; i < 16; i++) {
! 			line[sizeof("/dev/ptyp") - 1] = "0123456789abcdef"[i];
! 			p = open(line, O_RDWR);
  			if (p > 0)
  				goto gotpty;
  		}
***************
*** 127,155 ****
  	(void) ioctl(p, TIOCSWINSZ, &win);
  	netf = f;
  	line[strlen("/dev/")] = 't';
  #ifdef DEBUG
! 	{ int tt = open("/dev/tty", 2);
! 	  if (tt > 0) {
! 		ioctl(tt, TIOCNOTTY, 0);
! 		close(tt);
! 	  }
  	}
  #endif
- 	t = open(line, 2);
- 	if (t < 0)
- 		fatalperror(f, line, errno);
- 	{ struct sgttyb b;
- 	  gtty(t, &b); b.sg_flags = RAW|ANYP; stty(t, &b);
- 	}
  	pid = fork();
  	if (pid < 0)
! 		fatalperror(f, "", errno);
  	if (pid == 0) {
  		close(f), close(p);
  		dup2(t, 0), dup2(t, 1), dup2(t, 2);
  		close(t);
  		execl("/bin/login", "login", "-r", hp->h_name, 0);
! 		fatalperror(2, "/bin/login", errno);
  		/*NOTREACHED*/
  	}
  	close(t);
--- 127,168 ----
  	(void) ioctl(p, TIOCSWINSZ, &win);
  	netf = f;
  	line[strlen("/dev/")] = 't';
+ 	t = open(line, O_RDWR);
+ 	if (t < 0)
+ 		fatalperror(f, line);
+ 	if (fchmod(t, 0))
+ 		fatalperror(f, line);
+ 	(void)signal(SIGHUP, SIG_IGN);
+ 	vhangup();
+ 	(void)signal(SIGHUP, SIG_DFL);
+ 	t = open(line, O_RDWR);
+ 	if (t < 0)
+ 		fatalperror(f, line);
+ 	{
+ 		struct sgttyb b;
+ 
+ 		(void)ioctl(t, TIOCGETP, &b);
+ 		b.sg_flags = RAW|ANYP;
+ 		(void)ioctl(t, TIOCSETP, &b);
+ 	}
  #ifdef DEBUG
! 	{
! 		int tt = open("/dev/tty", O_RDWR);
! 		if (tt > 0) {
! 			(void)ioctl(tt, TIOCNOTTY, 0);
! 			(void)close(tt);
! 		}
  	}
  #endif
  	pid = fork();
  	if (pid < 0)
! 		fatalperror(f, "");
  	if (pid == 0) {
  		close(f), close(p);
  		dup2(t, 0), dup2(t, 1), dup2(t, 2);
  		close(t);
  		execl("/bin/login", "login", "-r", hp->h_name, 0);
! 		fatalperror(2, "/bin/login");
  		/*NOTREACHED*/
  	}
  	close(t);
***************
*** 226,232 ****
  		if (select(16, &ibits, &obits, &ebits, 0) < 0) {
  			if (errno == EINTR)
  				continue;
! 			fatalperror(f, "select", errno);
  		}
  		if (ibits == 0 && obits == 0 && ebits == 0) {
  			/* shouldn't happen... */
--- 239,245 ----
  		if (select(16, &ibits, &obits, &ebits, 0) < 0) {
  			if (errno == EINTR)
  				continue;
! 			fatalperror(f, "select");
  		}
  		if (ibits == 0 && obits == 0 && ebits == 0) {
  			/* shouldn't happen... */
***************
*** 335,344 ****
  	exit(1);
  }
  
! fatalperror(f, msg, errno)
  	int f;
  	char *msg;
- 	int errno;
  {
  	char buf[BUFSIZ];
  	extern int sys_nerr;
--- 348,356 ----
  	exit(1);
  }
  
! fatalperror(f, msg)
  	int f;
  	char *msg;
  {
  	char buf[BUFSIZ];
  	extern int sys_nerr;

*** telnetd.c.orig	Sun May 22 16:40:37 1988
--- telnetd.c.new	Sun May 22 16:44:52 1988
***************
*** 254,261 ****
  		if (stat(line, &stb) < 0)
  			break;
  		for (i = 0; i < 16; i++) {
! 			line[strlen("/dev/ptyp")] = "0123456789abcdef"[i];
! 			p = open(line, 2);
  			if (p > 0)
  				goto gotpty;
  		}
--- 254,261 ----
  		if (stat(line, &stb) < 0)
  			break;
  		for (i = 0; i < 16; i++) {
! 			line[sizeof("/dev/ptyp") - 1] = "0123456789abcdef"[i];
! 			p = open(line, O_RDWR);
  			if (p > 0)
  				goto gotpty;
  		}
***************
*** 272,278 ****
  	}
  	t = open(line, O_RDWR);
  	if (t < 0)
! 		fatalperror(f, line, errno);
  	ioctl(t, TIOCGETP, &b);
  	b.sg_flags = CRMOD|XTABS|ANYP;
  	ioctl(t, TIOCSETP, &b);
--- 272,286 ----
  	}
  	t = open(line, O_RDWR);
  	if (t < 0)
! 		fatalperror(f, line);
! 	if (fchmod(t, 0))
! 		fatalperror(f, line);
! 	(void)signal(SIGHUP, SIG_IGN);
! 	vhangup();
! 	(void)signal(SIGHUP, SIG_DFL);
! 	t = open(line, O_RDWR);
! 	if (t < 0)
! 		fatalperror(f, line);
  	ioctl(t, TIOCGETP, &b);
  	b.sg_flags = CRMOD|XTABS|ANYP;
  	ioctl(t, TIOCSETP, &b);
***************
*** 295,301 ****
  	getterminaltype();
  
  	if ((i = fork()) < 0)
! 		fatalperror(f, "fork", errno);
  	if (i)
  		telnet(f, p);
  	close(f);
--- 303,309 ----
  	getterminaltype();
  
  	if ((i = fork()) < 0)
! 		fatalperror(f, "fork");
  	if (i)
  		telnet(f, p);
  	close(f);
***************
*** 315,321 ****
  	 */
  	execl("/bin/login", "login", "-h", host,
  					terminaltype ? "-p" : 0, 0);
! 	fatalperror(f, "/bin/login", errno);
  	/*NOTREACHED*/
  }
  
--- 323,329 ----
  	 */
  	execl("/bin/login", "login", "-h", host,
  					terminaltype ? "-p" : 0, 0);
! 	fatalperror(f, "/bin/login");
  	/*NOTREACHED*/
  }
  
***************
*** 330,339 ****
  	exit(1);
  }
  
! fatalperror(f, msg, errno)
  	int f;
  	char *msg;
- 	int errno;
  {
  	char buf[BUFSIZ];
  	extern char *sys_errlist[];
--- 338,346 ----
  	exit(1);
  }
  
! fatalperror(f, msg)
  	int f;
  	char *msg;
  {
  	char buf[BUFSIZ];
  	extern char *sys_errlist[];
***************
*** 362,368 ****
      } while ((value == -1) && (errno == EINTR));
  
      if (value < 0) {
! 	fatalperror(pty, "select", errno);
      }
      if (FD_ISSET(s, &excepts)) {
  	return 1;
--- 369,375 ----
      } while ((value == -1) && (errno == EINTR));
  
      if (value < 0) {
! 	fatalperror(pty, "select");
      }
      if (FD_ISSET(s, &excepts)) {
  	return 1;