xref: /csrg-svn/usr.sbin/inetd/inetd.c (revision 32091)
121134Sdist /*
221134Sdist  * Copyright (c) 1983 Regents of the University of California.
321134Sdist  * All rights reserved.  The Berkeley software License Agreement
421134Sdist  * specifies the terms and conditions for redistribution.
521134Sdist  */
621134Sdist 
716374Skarels #ifndef lint
821134Sdist char copyright[] =
921134Sdist "@(#) Copyright (c) 1983 Regents of the University of California.\n\
1021134Sdist  All rights reserved.\n";
1121134Sdist #endif not lint
1216374Skarels 
1321134Sdist #ifndef lint
14*32091Skarels static char sccsid[] = "@(#)inetd.c	5.8 (Berkeley) 09/01/87";
1521134Sdist #endif not lint
1621134Sdist 
1716374Skarels /*
1816374Skarels  * Inetd - Internet super-server
1916374Skarels  *
2016374Skarels  * This program invokes all internet services as needed.
2116374Skarels  * connection-oriented services are invoked each time a
2216374Skarels  * connection is made, by creating a process.  This process
2316374Skarels  * is passed the connection as file descriptor 0 and is
2416374Skarels  * expected to do a getpeername to find out the source host
2516374Skarels  * and port.
2616374Skarels  *
2716374Skarels  * Datagram oriented services are invoked when a datagram
2816374Skarels  * arrives; a process is created and passed a pending message
2916374Skarels  * on file descriptor 0.  Datagram servers may either connect
3016374Skarels  * to their peer, freeing up the original socket for inetd
3116374Skarels  * to receive further messages on, or ``take over the socket'',
3216374Skarels  * processing all arriving datagrams and, eventually, timing
3316374Skarels  * out.	 The first type of server is said to be ``multi-threaded'';
3416374Skarels  * the second type of server ``single-threaded''.
3516374Skarels  *
3616374Skarels  * Inetd uses a configuration file which is read at startup
3716374Skarels  * and, possibly, at some later time in response to a hangup signal.
3816374Skarels  * The configuration file is ``free format'' with fields given in the
3916374Skarels  * order shown below.  Continuation lines for an entry must being with
4016374Skarels  * a space or tab.  All fields must be present in each entry.
4116374Skarels  *
4216374Skarels  *	service name			must be in /etc/services
4316374Skarels  *	socket type			stream/dgram/raw/rdm/seqpacket
4416374Skarels  *	protocol			must be in /etc/protocols
4516374Skarels  *	wait/nowait			single-threaded/multi-threaded
4617346Sbloom  *	user				user to run daemon as
4716374Skarels  *	server program			full path name
4816374Skarels  *	server program arguments	maximum of MAXARGS (5)
4916374Skarels  *
5016374Skarels  * Comment lines are indicated by a `#' in column 1.
5116374Skarels  */
5216374Skarels #include <sys/param.h>
5316374Skarels #include <sys/stat.h>
5416374Skarels #include <sys/ioctl.h>
5516374Skarels #include <sys/socket.h>
5616374Skarels #include <sys/file.h>
5716374Skarels #include <sys/wait.h>
5826921Slepreau #include <sys/time.h>
5926921Slepreau #include <sys/resource.h>
6016374Skarels 
6116374Skarels #include <netinet/in.h>
6216374Skarels #include <arpa/inet.h>
6316374Skarels 
6416374Skarels #include <errno.h>
6516374Skarels #include <stdio.h>
6616374Skarels #include <signal.h>
6716374Skarels #include <netdb.h>
6816510Sralph #include <syslog.h>
6917346Sbloom #include <pwd.h>
7016374Skarels 
7127535Skarels #define	TOOMANY		40		/* don't start more than TOOMANY */
7227535Skarels #define	CNT_INTVL	60		/* servers in CNT_INTVL sec. */
7327535Skarels #define	RETRYTIME	(60*10)		/* retry after bind or server fail */
7427535Skarels 
7527535Skarels #define	SIGBLOCK	(sigmask(SIGCHLD)|sigmask(SIGHUP)|sigmask(SIGALRM))
7627535Skarels 
7716374Skarels extern	int errno;
7816374Skarels 
7927535Skarels int	reapchild(), retry();
8016374Skarels char	*index();
8116374Skarels char	*malloc();
8216374Skarels 
8316374Skarels int	debug = 0;
8425046Skarels int	nsock, maxsock;
8525046Skarels fd_set	allsock;
8616374Skarels int	options;
8727535Skarels int	timingout;
8816374Skarels struct	servent *sp;
8916374Skarels 
9016374Skarels struct	servtab {
9116374Skarels 	char	*se_service;		/* name of service */
9216374Skarels 	int	se_socktype;		/* type of socket to use */
9316374Skarels 	char	*se_proto;		/* protocol used */
9416374Skarels 	short	se_wait;		/* single threaded server */
9516374Skarels 	short	se_checked;		/* looked at during merge */
9617346Sbloom 	char	*se_user;		/* user name to run as */
9726877Skarels 	struct	biltin *se_bi;		/* if built-in, description */
9816374Skarels 	char	*se_server;		/* server program */
9916374Skarels #define MAXARGV 5
10016374Skarels 	char	*se_argv[MAXARGV+1];	/* program arguments */
10116374Skarels 	int	se_fd;			/* open descriptor */
10216374Skarels 	struct	sockaddr_in se_ctrladdr;/* bound address */
10327535Skarels 	int	se_count;		/* number started since se_time */
10427535Skarels 	struct	timeval se_time;	/* start of se_count */
10516374Skarels 	struct	servtab *se_next;
10616374Skarels } *servtab;
10716374Skarels 
10826877Skarels int echo_stream(), discard_stream(), machtime_stream();
10926877Skarels int daytime_stream(), chargen_stream();
11026877Skarels int echo_dg(), discard_dg(), machtime_dg(), daytime_dg(), chargen_dg();
11126877Skarels 
11226877Skarels struct biltin {
11326877Skarels 	char	*bi_service;		/* internally provided service name */
11426877Skarels 	int	bi_socktype;		/* type of socket supported */
11526877Skarels 	short	bi_fork;		/* 1 if should fork before call */
11626877Skarels 	short	bi_wait;		/* 1 if should wait for child */
11726877Skarels 	int	(*bi_fn)();		/* function which performs it */
11826877Skarels } biltins[] = {
11926877Skarels 	/* Echo received data */
12026877Skarels 	"echo",		SOCK_STREAM,	1, 0,	echo_stream,
12126877Skarels 	"echo",		SOCK_DGRAM,	0, 0,	echo_dg,
12226877Skarels 
12326877Skarels 	/* Internet /dev/null */
12426877Skarels 	"discard",	SOCK_STREAM,	1, 0,	discard_stream,
12526877Skarels 	"discard",	SOCK_DGRAM,	0, 0,	discard_dg,
12626877Skarels 
12726877Skarels 	/* Return 32 bit time since 1970 */
12826877Skarels 	"time",		SOCK_STREAM,	0, 0,	machtime_stream,
12926877Skarels 	"time",		SOCK_DGRAM,	0, 0,	machtime_dg,
13026877Skarels 
13126877Skarels 	/* Return human-readable time */
13226877Skarels 	"daytime",	SOCK_STREAM,	0, 0,	daytime_stream,
13326877Skarels 	"daytime",	SOCK_DGRAM,	0, 0,	daytime_dg,
13426877Skarels 
13526877Skarels 	/* Familiar character generator */
13626877Skarels 	"chargen",	SOCK_STREAM,	1, 0,	chargen_stream,
13726877Skarels 	"chargen",	SOCK_DGRAM,	0, 0,	chargen_dg,
13826877Skarels 	0
13926877Skarels };
14026877Skarels 
14126877Skarels #define NUMINT	(sizeof(intab) / sizeof(struct inent))
14216374Skarels char	*CONFIG = "/etc/inetd.conf";
14326877Skarels char	**Argv;
14426877Skarels char 	*LastArg;
14516374Skarels 
14626877Skarels main(argc, argv, envp)
14716374Skarels 	int argc;
14826877Skarels 	char *argv[], *envp[];
14916374Skarels {
15016374Skarels 	register struct servtab *sep;
15117346Sbloom 	register struct passwd *pwd;
15216374Skarels 	char *cp, buf[50];
15327535Skarels 	int pid, i, dofork;
15427535Skarels 	struct sigvec sv;
15516374Skarels 
15626877Skarels 	Argv = argv;
15726877Skarels 	if (envp == 0 || *envp == 0)
15826877Skarels 		envp = argv;
15926877Skarels 	while (*envp)
16026877Skarels 		envp++;
16126877Skarels 	LastArg = envp[-1] + strlen(envp[-1]);
16216374Skarels 	argc--, argv++;
16316374Skarels 	while (argc > 0 && *argv[0] == '-') {
16416374Skarels 		for (cp = &argv[0][1]; *cp; cp++) switch (*cp) {
16516374Skarels 
16616374Skarels 		case 'd':
16716374Skarels 			debug = 1;
16816374Skarels 			options |= SO_DEBUG;
16916374Skarels 			break;
17016374Skarels 
17116374Skarels 		default:
17216374Skarels 			fprintf(stderr,
17316374Skarels 			    "inetd: Unknown flag -%c ignored.\n", *cp);
17416374Skarels 			break;
17516374Skarels 		}
17616374Skarels nextopt:
17716374Skarels 		argc--, argv++;
17816374Skarels 	}
17916374Skarels 	if (argc > 0)
18016374Skarels 		CONFIG = argv[0];
18116374Skarels #ifndef DEBUG
18216374Skarels 	if (fork())
18316374Skarels 		exit(0);
18416374Skarels 	{ int s;
18516374Skarels 	for (s = 0; s < 10; s++)
18616374Skarels 		(void) close(s);
18716374Skarels 	}
18816374Skarels 	(void) open("/", O_RDONLY);
18916374Skarels 	(void) dup2(0, 1);
19016374Skarels 	(void) dup2(0, 2);
19116374Skarels 	{ int tt = open("/dev/tty", O_RDWR);
19216374Skarels 	  if (tt > 0) {
19326921Slepreau 		ioctl(tt, TIOCNOTTY, (char *)0);
19416374Skarels 		close(tt);
19516374Skarels 	  }
19616374Skarels 	}
19716374Skarels #endif
19827535Skarels 	openlog("inetd", LOG_PID | LOG_NOWAIT, LOG_DAEMON);
19927535Skarels 	bzero((char *)&sv, sizeof(sv));
20027535Skarels 	sv.sv_mask = SIGBLOCK;
20127535Skarels 	sv.sv_handler = retry;
20227535Skarels 	sigvec(SIGALRM, &sv, (struct sigvec *)0);
20316374Skarels 	config();
20427535Skarels 	sv.sv_handler = config;
20527535Skarels 	sigvec(SIGHUP, &sv, (struct sigvec *)0);
20627535Skarels 	sv.sv_handler = reapchild;
20727535Skarels 	sigvec(SIGCHLD, &sv, (struct sigvec *)0);
20827535Skarels 
20916374Skarels 	for (;;) {
21027535Skarels 	    int s, ctrl, n;
21127535Skarels 	    fd_set readable;
21216374Skarels 
213*32091Skarels 	    if (nsock == 0) {
214*32091Skarels 		(void) sigblock(SIGBLOCK);
215*32091Skarels 		while (nsock == 0)
21627535Skarels 		    sigpause(0);
217*32091Skarels 		(void) sigsetmask(0);
218*32091Skarels 	    }
21927535Skarels 	    readable = allsock;
22027535Skarels 	    if ((n = select(maxsock + 1, &readable, (fd_set *)0,
22127535Skarels 		(fd_set *)0, (struct timeval *)0)) <= 0) {
22227535Skarels 		    if (n < 0 && errno != EINTR)
223*32091Skarels 			syslog(LOG_WARNING, "select: %m\n");
22427535Skarels 		    sleep(1);
22527535Skarels 		    continue;
22627535Skarels 	    }
22727535Skarels 	    for (sep = servtab; n && sep; sep = sep->se_next)
22827535Skarels 	    if (FD_ISSET(sep->se_fd, &readable)) {
22927535Skarels 		n--;
23016374Skarels 		if (debug)
23116374Skarels 			fprintf(stderr, "someone wants %s\n", sep->se_service);
23225046Skarels 		if (!sep->se_wait && sep->se_socktype == SOCK_STREAM) {
23327535Skarels 			ctrl = accept(sep->se_fd, (struct sockaddr *)0,
23427535Skarels 			    (int *)0);
23516510Sralph 			if (debug)
23616510Sralph 				fprintf(stderr, "accept, ctrl %d\n", ctrl);
23716374Skarels 			if (ctrl < 0) {
23816374Skarels 				if (errno == EINTR)
23916374Skarels 					continue;
24016510Sralph 				syslog(LOG_WARNING, "accept: %m");
24116374Skarels 				continue;
24216374Skarels 			}
24316374Skarels 		} else
24416374Skarels 			ctrl = sep->se_fd;
24527535Skarels 		(void) sigblock(SIGBLOCK);
24626877Skarels 		pid = 0;
24727535Skarels 		dofork = (sep->se_bi == 0 || sep->se_bi->bi_fork);
24827535Skarels 		if (dofork) {
24927535Skarels 			if (sep->se_count++ == 0)
25027535Skarels 			    (void)gettimeofday(&sep->se_time,
25127535Skarels 			        (struct timezone *)0);
25227535Skarels 			else if (sep->se_count >= TOOMANY) {
25327535Skarels 				struct timeval now;
25427535Skarels 
25527535Skarels 				(void)gettimeofday(&now, (struct timezone *)0);
25627535Skarels 				if (now.tv_sec - sep->se_time.tv_sec >
25727535Skarels 				    CNT_INTVL) {
25827535Skarels 					sep->se_time = now;
25927535Skarels 					sep->se_count = 1;
26027535Skarels 				} else {
26127535Skarels 					syslog(LOG_ERR,
26227535Skarels 			"%s/%s server failing (looping), service terminated\n",
26327535Skarels 					    sep->se_service, sep->se_proto);
26427535Skarels 					FD_CLR(sep->se_fd, &allsock);
26527535Skarels 					(void) close(sep->se_fd);
26627535Skarels 					sep->se_fd = -1;
26727535Skarels 					sep->se_count = 0;
26827535Skarels 					nsock--;
26927535Skarels 					sigsetmask(0);
27027535Skarels 					if (!timingout) {
27127535Skarels 						timingout = 1;
27227535Skarels 						alarm(RETRYTIME);
27327535Skarels 					}
27427535Skarels 					continue;
27527535Skarels 				}
27627535Skarels 			}
27726877Skarels 			pid = fork();
27827535Skarels 		}
27916374Skarels 		if (pid < 0) {
28025046Skarels 			if (!sep->se_wait && sep->se_socktype == SOCK_STREAM)
28116374Skarels 				close(ctrl);
28227535Skarels 			sigsetmask(0);
28316374Skarels 			sleep(1);
28416374Skarels 			continue;
28516374Skarels 		}
28627535Skarels 		if (pid && sep->se_wait) {
28716374Skarels 			sep->se_wait = pid;
28827535Skarels 			FD_CLR(sep->se_fd, &allsock);
28925046Skarels 			nsock--;
29016374Skarels 		}
29116374Skarels 		sigsetmask(0);
29216374Skarels 		if (pid == 0) {
29316374Skarels #ifdef	DEBUG
29427535Skarels 			int tt;
29527535Skarels 
29627535Skarels 			if (dofork && (tt = open("/dev/tty", O_RDWR)) > 0) {
29716374Skarels 				ioctl(tt, TIOCNOTTY, 0);
29816374Skarels 				close(tt);
29916374Skarels 			}
30016374Skarels #endif
30127535Skarels 			if (dofork)
30226877Skarels 				for (i = getdtablesize(); --i > 2; )
30326877Skarels 					if (i != ctrl)
30426877Skarels 						close(i);
30526877Skarels 			if (sep->se_bi)
30626877Skarels 				(*sep->se_bi->bi_fn)(ctrl, sep);
30726877Skarels 			else {
30826877Skarels 				dup2(ctrl, 0);
30926877Skarels 				close(ctrl);
31026877Skarels 				dup2(0, 1);
31126877Skarels 				dup2(0, 2);
31226877Skarels 				if ((pwd = getpwnam(sep->se_user)) == NULL) {
31326877Skarels 					syslog(LOG_ERR,
31426877Skarels 						"getpwnam: %s: No such user",
31526877Skarels 						sep->se_user);
31627535Skarels 					if (sep->se_socktype != SOCK_STREAM)
31727535Skarels 						recv(0, buf, sizeof (buf), 0);
31826877Skarels 					_exit(1);
31926877Skarels 				}
32026877Skarels 				if (pwd->pw_uid) {
32126921Slepreau 					(void) setgid((gid_t)pwd->pw_gid);
32226877Skarels 					initgroups(pwd->pw_name, pwd->pw_gid);
32326921Slepreau 					(void) setuid((uid_t)pwd->pw_uid);
32426877Skarels 				}
32526877Skarels 				if (debug)
32626877Skarels 					fprintf(stderr, "%d execl %s\n",
32726877Skarels 					    getpid(), sep->se_server);
32826877Skarels 				execv(sep->se_server, sep->se_argv);
32926877Skarels 				if (sep->se_socktype != SOCK_STREAM)
33026877Skarels 					recv(0, buf, sizeof (buf), 0);
33126877Skarels 				syslog(LOG_ERR, "execv %s: %m", sep->se_server);
33226877Skarels 				_exit(1);
33317346Sbloom 			}
33416374Skarels 		}
33525046Skarels 		if (!sep->se_wait && sep->se_socktype == SOCK_STREAM)
33616374Skarels 			close(ctrl);
33727535Skarels 	    }
33816374Skarels 	}
33916374Skarels }
34016374Skarels 
34116374Skarels reapchild()
34216374Skarels {
34316374Skarels 	union wait status;
34416374Skarels 	int pid;
34516374Skarels 	register struct servtab *sep;
34616374Skarels 
34716374Skarels 	for (;;) {
34826921Slepreau 		pid = wait3(&status, WNOHANG, (struct rusage *)0);
34916374Skarels 		if (pid <= 0)
35016374Skarels 			break;
35116374Skarels 		if (debug)
35216374Skarels 			fprintf(stderr, "%d reaped\n", pid);
35316374Skarels 		for (sep = servtab; sep; sep = sep->se_next)
35416374Skarels 			if (sep->se_wait == pid) {
35516374Skarels 				if (status.w_status)
35616510Sralph 					syslog(LOG_WARNING,
35716510Sralph 					    "%s: exit status 0x%x",
35816374Skarels 					    sep->se_server, status);
35916374Skarels 				if (debug)
36016374Skarels 					fprintf(stderr, "restored %s, fd %d\n",
36116374Skarels 					    sep->se_service, sep->se_fd);
36225046Skarels 				FD_SET(sep->se_fd, &allsock);
36325046Skarels 				nsock++;
36416374Skarels 				sep->se_wait = 1;
36516374Skarels 			}
36616374Skarels 	}
36716374Skarels }
36816374Skarels 
36916374Skarels config()
37016374Skarels {
37116374Skarels 	register struct servtab *sep, *cp, **sepp;
37216374Skarels 	struct servtab *getconfigent(), *enter();
37327535Skarels 	int omask;
37416374Skarels 
37516374Skarels 	if (!setconfig()) {
37616510Sralph 		syslog(LOG_ERR, "%s: %m", CONFIG);
37716374Skarels 		return;
37816374Skarels 	}
37916374Skarels 	for (sep = servtab; sep; sep = sep->se_next)
38016374Skarels 		sep->se_checked = 0;
38116374Skarels 	while (cp = getconfigent()) {
38216374Skarels 		for (sep = servtab; sep; sep = sep->se_next)
38316374Skarels 			if (strcmp(sep->se_service, cp->se_service) == 0 &&
38416374Skarels 			    strcmp(sep->se_proto, cp->se_proto) == 0)
38516374Skarels 				break;
38616374Skarels 		if (sep != 0) {
38716374Skarels 			int i;
38816374Skarels 
38927535Skarels 			omask = sigblock(SIGBLOCK);
39027535Skarels 			if (cp->se_bi == 0)
39127535Skarels 				sep->se_wait = cp->se_wait;
39216374Skarels #define SWAP(a, b) { char *c = a; a = b; b = c; }
39326921Slepreau 			if (cp->se_user)
39426921Slepreau 				SWAP(sep->se_user, cp->se_user);
39516374Skarels 			if (cp->se_server)
39616374Skarels 				SWAP(sep->se_server, cp->se_server);
39716374Skarels 			for (i = 0; i < MAXARGV; i++)
39816374Skarels 				SWAP(sep->se_argv[i], cp->se_argv[i]);
39916374Skarels 			sigsetmask(omask);
40016374Skarels 			freeconfig(cp);
40129794Skarels 			if (debug)
40229794Skarels 				print_service("REDO", sep);
40329794Skarels 		} else {
40416374Skarels 			sep = enter(cp);
40529794Skarels 			if (debug)
40629794Skarels 				print_service("ADD ", sep);
40729794Skarels 		}
40816374Skarels 		sep->se_checked = 1;
40916374Skarels 		sp = getservbyname(sep->se_service, sep->se_proto);
41016374Skarels 		if (sp == 0) {
41116510Sralph 			syslog(LOG_ERR, "%s/%s: unknown service",
41216374Skarels 			    sep->se_service, sep->se_proto);
41316374Skarels 			continue;
41416374Skarels 		}
41527535Skarels 		if (sp->s_port != sep->se_ctrladdr.sin_port) {
41627535Skarels 			sep->se_ctrladdr.sin_port = sp->s_port;
41727535Skarels 			if (sep->se_fd != -1)
41827535Skarels 				(void) close(sep->se_fd);
41927535Skarels 			sep->se_fd = -1;
42016374Skarels 		}
42127535Skarels 		if (sep->se_fd == -1)
42227535Skarels 			setup(sep);
42316374Skarels 	}
42416374Skarels 	endconfig();
42516374Skarels 	/*
42616374Skarels 	 * Purge anything not looked at above.
42716374Skarels 	 */
42827535Skarels 	omask = sigblock(SIGBLOCK);
42916374Skarels 	sepp = &servtab;
43016374Skarels 	while (sep = *sepp) {
43116374Skarels 		if (sep->se_checked) {
43216374Skarels 			sepp = &sep->se_next;
43316374Skarels 			continue;
43416374Skarels 		}
43516374Skarels 		*sepp = sep->se_next;
43616374Skarels 		if (sep->se_fd != -1) {
43725046Skarels 			FD_CLR(sep->se_fd, &allsock);
43825046Skarels 			nsock--;
43916374Skarels 			(void) close(sep->se_fd);
44016374Skarels 		}
44129794Skarels 		if (debug)
44229794Skarels 			print_service("FREE", sep);
44316374Skarels 		freeconfig(sep);
44416374Skarels 		free((char *)sep);
44516374Skarels 	}
44616374Skarels 	(void) sigsetmask(omask);
44716374Skarels }
44816374Skarels 
44927535Skarels retry()
45027535Skarels {
45127535Skarels 	register struct servtab *sep;
45227535Skarels 
45327535Skarels 	timingout = 0;
45427535Skarels 	for (sep = servtab; sep; sep = sep->se_next)
45527535Skarels 		if (sep->se_fd == -1)
45627535Skarels 			setup(sep);
45727535Skarels }
45827535Skarels 
45927535Skarels setup(sep)
46027535Skarels 	register struct servtab *sep;
46127535Skarels {
46227535Skarels 	int on = 1;
46327535Skarels 
46427535Skarels 	if ((sep->se_fd = socket(AF_INET, sep->se_socktype, 0)) < 0) {
46527535Skarels 		syslog(LOG_ERR, "%s/%s: socket: %m",
46627535Skarels 		    sep->se_service, sep->se_proto);
46727535Skarels 		return;
46827535Skarels 	}
46927535Skarels #define	turnon(fd, opt) \
47027535Skarels setsockopt(fd, SOL_SOCKET, opt, (char *)&on, sizeof (on))
47127535Skarels 	if (strcmp(sep->se_proto, "tcp") == 0 && (options & SO_DEBUG) &&
47227535Skarels 	    turnon(sep->se_fd, SO_DEBUG) < 0)
47327535Skarels 		syslog(LOG_ERR, "setsockopt (SO_DEBUG): %m");
47427535Skarels 	if (turnon(sep->se_fd, SO_REUSEADDR) < 0)
47527535Skarels 		syslog(LOG_ERR, "setsockopt (SO_REUSEADDR): %m");
47627535Skarels #undef turnon
47727535Skarels 	if (bind(sep->se_fd, &sep->se_ctrladdr,
47827535Skarels 	    sizeof (sep->se_ctrladdr)) < 0) {
47927535Skarels 		syslog(LOG_ERR, "%s/%s: bind: %m",
48027535Skarels 		    sep->se_service, sep->se_proto);
48127535Skarels 		(void) close(sep->se_fd);
48227535Skarels 		sep->se_fd = -1;
48327535Skarels 		if (!timingout) {
48427535Skarels 			timingout = 1;
48527535Skarels 			alarm(RETRYTIME);
48627535Skarels 		}
48727535Skarels 		return;
48827535Skarels 	}
48927535Skarels 	if (sep->se_socktype == SOCK_STREAM)
49027535Skarels 		listen(sep->se_fd, 10);
49127535Skarels 	FD_SET(sep->se_fd, &allsock);
49227535Skarels 	nsock++;
49327535Skarels 	if (sep->se_fd > maxsock)
49427535Skarels 		maxsock = sep->se_fd;
49527535Skarels }
49627535Skarels 
49716374Skarels struct servtab *
49816374Skarels enter(cp)
49916374Skarels 	struct servtab *cp;
50016374Skarels {
50116374Skarels 	register struct servtab *sep;
50225046Skarels 	int omask;
50316374Skarels 	char *strdup();
50416374Skarels 
50516374Skarels 	sep = (struct servtab *)malloc(sizeof (*sep));
50616374Skarels 	if (sep == (struct servtab *)0) {
50716510Sralph 		syslog(LOG_ERR, "Out of memory.");
50816374Skarels 		exit(-1);
50916374Skarels 	}
51016374Skarels 	*sep = *cp;
51116374Skarels 	sep->se_fd = -1;
51227535Skarels 	omask = sigblock(SIGBLOCK);
51316374Skarels 	sep->se_next = servtab;
51416374Skarels 	servtab = sep;
51516374Skarels 	sigsetmask(omask);
51616374Skarels 	return (sep);
51716374Skarels }
51816374Skarels 
51916374Skarels FILE	*fconfig = NULL;
52016374Skarels struct	servtab serv;
52116374Skarels char	line[256];
52216374Skarels char	*skip(), *nextline();
52316374Skarels 
52416374Skarels setconfig()
52516374Skarels {
52616374Skarels 
52716374Skarels 	if (fconfig != NULL) {
52826921Slepreau 		fseek(fconfig, 0L, L_SET);
52916374Skarels 		return (1);
53016374Skarels 	}
53116374Skarels 	fconfig = fopen(CONFIG, "r");
53216374Skarels 	return (fconfig != NULL);
53316374Skarels }
53416374Skarels 
53516374Skarels endconfig()
53616374Skarels {
53716374Skarels 
53816374Skarels 	if (fconfig == NULL)
53916374Skarels 		return;
54016374Skarels 	fclose(fconfig);
54116374Skarels 	fconfig = NULL;
54216374Skarels }
54316374Skarels 
54416374Skarels struct servtab *
54516374Skarels getconfigent()
54616374Skarels {
54716374Skarels 	register struct servtab *sep = &serv;
54816374Skarels 	char *cp, *arg;
54916374Skarels 	int argc;
55016374Skarels 
55126877Skarels more:
55216374Skarels 	while ((cp = nextline(fconfig)) && *cp == '#')
55316374Skarels 		;
55416374Skarels 	if (cp == NULL)
55516374Skarels 		return ((struct servtab *)0);
55616374Skarels 	sep->se_service = strdup(skip(&cp));
55716374Skarels 	arg = skip(&cp);
55816374Skarels 	if (strcmp(arg, "stream") == 0)
55916374Skarels 		sep->se_socktype = SOCK_STREAM;
56016374Skarels 	else if (strcmp(arg, "dgram") == 0)
56116374Skarels 		sep->se_socktype = SOCK_DGRAM;
56216374Skarels 	else if (strcmp(arg, "rdm") == 0)
56316374Skarels 		sep->se_socktype = SOCK_RDM;
56416374Skarels 	else if (strcmp(arg, "seqpacket") == 0)
56516374Skarels 		sep->se_socktype = SOCK_SEQPACKET;
56616374Skarels 	else if (strcmp(arg, "raw") == 0)
56716374Skarels 		sep->se_socktype = SOCK_RAW;
56816374Skarels 	else
56916374Skarels 		sep->se_socktype = -1;
57016374Skarels 	sep->se_proto = strdup(skip(&cp));
57116374Skarels 	arg = skip(&cp);
57216374Skarels 	sep->se_wait = strcmp(arg, "wait") == 0;
57317346Sbloom 	sep->se_user = strdup(skip(&cp));
57416374Skarels 	sep->se_server = strdup(skip(&cp));
57526877Skarels 	if (strcmp(sep->se_server, "internal") == 0) {
57626877Skarels 		register struct biltin *bi;
57726877Skarels 
57826877Skarels 		for (bi = biltins; bi->bi_service; bi++)
57926877Skarels 			if (bi->bi_socktype == sep->se_socktype &&
58026877Skarels 			    strcmp(bi->bi_service, sep->se_service) == 0)
58126877Skarels 				break;
58226877Skarels 		if (bi->bi_service == 0) {
58326877Skarels 			syslog(LOG_ERR, "internal service %s unknown\n",
58426877Skarels 				sep->se_service);
58526877Skarels 			goto more;
58626877Skarels 		}
58726877Skarels 		sep->se_bi = bi;
58826877Skarels 		sep->se_wait = bi->bi_wait;
58929794Skarels 	} else
59029794Skarels 		sep->se_bi = NULL;
59116374Skarels 	argc = 0;
59216374Skarels 	for (arg = skip(&cp); cp; arg = skip(&cp))
59316374Skarels 		if (argc < MAXARGV)
59416374Skarels 			sep->se_argv[argc++] = strdup(arg);
59516374Skarels 	while (argc <= MAXARGV)
59616374Skarels 		sep->se_argv[argc++] = NULL;
59716374Skarels 	return (sep);
59816374Skarels }
59916374Skarels 
60016374Skarels freeconfig(cp)
60116374Skarels 	register struct servtab *cp;
60216374Skarels {
60316374Skarels 	int i;
60416374Skarels 
60516374Skarels 	if (cp->se_service)
60616374Skarels 		free(cp->se_service);
60716374Skarels 	if (cp->se_proto)
60816374Skarels 		free(cp->se_proto);
60926921Slepreau 	if (cp->se_user)
61026921Slepreau 		free(cp->se_user);
61116374Skarels 	if (cp->se_server)
61216374Skarels 		free(cp->se_server);
61316374Skarels 	for (i = 0; i < MAXARGV; i++)
61416374Skarels 		if (cp->se_argv[i])
61516374Skarels 			free(cp->se_argv[i]);
61616374Skarels }
61716374Skarels 
61816374Skarels char *
61916374Skarels skip(cpp)
62016374Skarels 	char **cpp;
62116374Skarels {
62216374Skarels 	register char *cp = *cpp;
62316374Skarels 	char *start;
62416374Skarels 
62516374Skarels again:
62616374Skarels 	while (*cp == ' ' || *cp == '\t')
62716374Skarels 		cp++;
62816374Skarels 	if (*cp == '\0') {
62916374Skarels 		char c;
63016374Skarels 
63116374Skarels 		c = getc(fconfig);
63216374Skarels 		ungetc(c, fconfig);
63316374Skarels 		if (c == ' ' || c == '\t')
63416374Skarels 			if (cp = nextline(fconfig))
63516374Skarels 				goto again;
63616374Skarels 		*cpp = (char *)0;
63716374Skarels 		return ((char *)0);
63816374Skarels 	}
63916374Skarels 	start = cp;
64016374Skarels 	while (*cp && *cp != ' ' && *cp != '\t')
64116374Skarels 		cp++;
64216374Skarels 	if (*cp != '\0')
64316374Skarels 		*cp++ = '\0';
64416374Skarels 	*cpp = cp;
64516374Skarels 	return (start);
64616374Skarels }
64716374Skarels 
64816374Skarels char *
64916374Skarels nextline(fd)
65016374Skarels 	FILE *fd;
65116374Skarels {
65216374Skarels 	char *cp;
65316374Skarels 
65426921Slepreau 	if (fgets(line, sizeof (line), fd) == NULL)
65516374Skarels 		return ((char *)0);
65616374Skarels 	cp = index(line, '\n');
65716374Skarels 	if (cp)
65816374Skarels 		*cp = '\0';
65916374Skarels 	return (line);
66016374Skarels }
66116374Skarels 
66216374Skarels char *
66316374Skarels strdup(cp)
66416374Skarels 	char *cp;
66516374Skarels {
66616374Skarels 	char *new;
66716374Skarels 
66816374Skarels 	if (cp == NULL)
66916374Skarels 		cp = "";
67026921Slepreau 	new = malloc((unsigned)(strlen(cp) + 1));
67116374Skarels 	if (new == (char *)0) {
67216510Sralph 		syslog(LOG_ERR, "Out of memory.");
67316374Skarels 		exit(-1);
67416374Skarels 	}
67516374Skarels 	strcpy(new, cp);
67616374Skarels 	return (new);
67716374Skarels }
67826877Skarels 
67926877Skarels setproctitle(a, s)
68026877Skarels 	char *a;
68126877Skarels 	int s;
68226877Skarels {
68326877Skarels 	int size;
68426877Skarels 	register char *cp;
68526877Skarels 	struct sockaddr_in sin;
68626877Skarels 	char buf[80];
68726877Skarels 
68826877Skarels 	cp = Argv[0];
68926877Skarels 	size = sizeof(sin);
69026877Skarels 	if (getpeername(s, &sin, &size) == 0)
69126877Skarels 		sprintf(buf, "-%s [%s]", a, inet_ntoa(sin.sin_addr));
69226877Skarels 	else
69326877Skarels 		sprintf(buf, "-%s", a);
69426877Skarels 	strncpy(cp, buf, LastArg - cp);
69526877Skarels 	cp += strlen(cp);
69626877Skarels 	while (cp < LastArg)
69726877Skarels 		*cp++ = ' ';
69826877Skarels }
69926877Skarels 
70026877Skarels /*
70126877Skarels  * Internet services provided internally by inetd:
70226877Skarels  */
70326877Skarels 
70426877Skarels /* ARGSUSED */
70526877Skarels echo_stream(s, sep)		/* Echo service -- echo data back */
70626877Skarels 	int s;
70726877Skarels 	struct servtab *sep;
70826877Skarels {
70926877Skarels 	char buffer[BUFSIZ];
71026877Skarels 	int i;
71126877Skarels 
712*32091Skarels 	setproctitle(sep->se_service, s);
71326877Skarels 	while ((i = read(s, buffer, sizeof(buffer))) > 0 &&
71426877Skarels 	    write(s, buffer, i) > 0)
71526877Skarels 		;
71626877Skarels 	exit(0);
71726877Skarels }
71826877Skarels 
71926877Skarels /* ARGSUSED */
72026877Skarels echo_dg(s, sep)			/* Echo service -- echo data back */
72126877Skarels 	int s;
72226877Skarels 	struct servtab *sep;
72326877Skarels {
72426877Skarels 	char buffer[BUFSIZ];
72526877Skarels 	int i, size;
72626877Skarels 	struct sockaddr sa;
72726877Skarels 
72826877Skarels 	size = sizeof(sa);
72926877Skarels 	if ((i = recvfrom(s, buffer, sizeof(buffer), 0, &sa, &size)) < 0)
73026877Skarels 		return;
73126877Skarels 	(void) sendto(s, buffer, i, 0, &sa, sizeof(sa));
73226877Skarels }
73326877Skarels 
73426877Skarels /* ARGSUSED */
73526877Skarels discard_stream(s, sep)		/* Discard service -- ignore data */
73626877Skarels 	int s;
73726877Skarels 	struct servtab *sep;
73826877Skarels {
73926877Skarels 	char buffer[BUFSIZ];
74026877Skarels 
741*32091Skarels 	setproctitle(sep->se_service, s);
74226877Skarels 	while (1) {
74326877Skarels 		while (read(s, buffer, sizeof(buffer)) > 0)
74426877Skarels 			;
74526877Skarels 		if (errno != EINTR)
74626877Skarels 			break;
74726877Skarels 	}
74826877Skarels 	exit(0);
74926877Skarels }
75026877Skarels 
75126877Skarels /* ARGSUSED */
75226877Skarels discard_dg(s, sep)		/* Discard service -- ignore data */
75326877Skarels 	int s;
75426877Skarels 	struct servtab *sep;
75526877Skarels {
75626877Skarels 	char buffer[BUFSIZ];
75726877Skarels 
75826877Skarels 	(void) read(s, buffer, sizeof(buffer));
75926877Skarels }
76026877Skarels 
76126877Skarels #include <ctype.h>
76226877Skarels #define LINESIZ 72
76326877Skarels char ring[128];
76426877Skarels char *endring;
76526877Skarels 
76626877Skarels initring()
76726877Skarels {
76826877Skarels 	register int i;
76926877Skarels 
77026877Skarels 	endring = ring;
77126877Skarels 
77226877Skarels 	for (i = 0; i <= 128; ++i)
77326877Skarels 		if (isprint(i))
77426877Skarels 			*endring++ = i;
77526877Skarels }
77626877Skarels 
77726877Skarels /* ARGSUSED */
77826877Skarels chargen_stream(s, sep)		/* Character generator */
77926877Skarels 	int s;
78026877Skarels 	struct servtab *sep;
78126877Skarels {
78226877Skarels 	char text[LINESIZ+2];
78326877Skarels 	register int i;
78426877Skarels 	register char *rp, *rs, *dp;
78526877Skarels 
786*32091Skarels 	setproctitle(sep->se_service, s);
78726877Skarels 	if (endring == 0)
78826877Skarels 		initring();
78926877Skarels 
79026877Skarels 	for (rs = ring; ; ++rs) {
79126877Skarels 		if (rs >= endring)
79226877Skarels 			rs = ring;
79326877Skarels 		rp = rs;
79426877Skarels 		dp = text;
79526877Skarels 		i = MIN(LINESIZ, endring - rp);
79626877Skarels 		bcopy(rp, dp, i);
79726877Skarels 		dp += i;
79826877Skarels 		if ((rp += i) >= endring)
79926877Skarels 			rp = ring;
80026877Skarels 		if (i < LINESIZ) {
80126877Skarels 			i = LINESIZ - i;
80226877Skarels 			bcopy(rp, dp, i);
80326877Skarels 			dp += i;
80426877Skarels 			if ((rp += i) >= endring)
80526877Skarels 				rp = ring;
80626877Skarels 		}
80726877Skarels 		*dp++ = '\r';
80826877Skarels 		*dp++ = '\n';
80926877Skarels 
81026877Skarels 		if (write(s, text, dp - text) != dp - text)
81126877Skarels 			break;
81226877Skarels 	}
81326877Skarels 	exit(0);
81426877Skarels }
81526877Skarels 
81626877Skarels /* ARGSUSED */
81726877Skarels chargen_dg(s, sep)		/* Character generator */
81826877Skarels 	int s;
81926877Skarels 	struct servtab *sep;
82026877Skarels {
82126877Skarels 	char text[LINESIZ+2];
82226877Skarels 	register int i;
82326877Skarels 	register char *rp;
82426877Skarels 	static char *rs = ring;
82526877Skarels 	struct sockaddr sa;
82626877Skarels 	int size;
82726877Skarels 
82826877Skarels 	if (endring == 0)
82926877Skarels 		initring();
83026877Skarels 
83126877Skarels 	size = sizeof(sa);
83226877Skarels 	if (recvfrom(s, text, sizeof(text), 0, &sa, &size) < 0)
83326877Skarels 		return;
83426877Skarels 	rp = rs;
83526877Skarels 	if (rs++ >= endring)
83626877Skarels 		rs = ring;
83726877Skarels 	i = MIN(LINESIZ - 2, endring - rp);
83826877Skarels 	bcopy(rp, text, i);
83926877Skarels 	if ((rp += i) >= endring)
84026877Skarels 		rp = ring;
84126877Skarels 	if (i < LINESIZ - 2) {
84226877Skarels 		bcopy(rp, text, i);
84326877Skarels 		if ((rp += i) >= endring)
84426877Skarels 			rp = ring;
84526877Skarels 	}
84626877Skarels 	text[LINESIZ - 2] = '\r';
84726877Skarels 	text[LINESIZ - 1] = '\n';
84826877Skarels 
84926877Skarels 	(void) sendto(s, text, sizeof(text), 0, &sa, sizeof(sa));
85026877Skarels }
85126877Skarels 
85226877Skarels /*
85326877Skarels  * Return a machine readable date and time, in the form of the
85426877Skarels  * number of seconds since midnight, Jan 1, 1900.  Since gettimeofday
85526877Skarels  * returns the number of seconds since midnight, Jan 1, 1970,
85626877Skarels  * we must add 2208988800 seconds to this figure to make up for
85726877Skarels  * some seventy years Bell Labs was asleep.
85826877Skarels  */
85926877Skarels 
86026877Skarels long
86126877Skarels machtime()
86226877Skarels {
86326877Skarels 	struct timeval tv;
86426877Skarels 
86526921Slepreau 	if (gettimeofday(&tv, (struct timezone *)0) < 0) {
86626877Skarels 		fprintf(stderr, "Unable to get time of day\n");
86726921Slepreau 		return (0L);
86826877Skarels 	}
86926877Skarels 	return (htonl((long)tv.tv_sec + 2208988800));
87026877Skarels }
87126877Skarels 
87226877Skarels /* ARGSUSED */
87326877Skarels machtime_stream(s, sep)
87426877Skarels 	int s;
87526877Skarels 	struct servtab *sep;
87626877Skarels {
87726877Skarels 	long result;
87826877Skarels 
87926877Skarels 	result = machtime();
88026877Skarels 	(void) write(s, (char *) &result, sizeof(result));
88126877Skarels }
88226877Skarels 
88326877Skarels /* ARGSUSED */
88426877Skarels machtime_dg(s, sep)
88526877Skarels 	int s;
88626877Skarels 	struct servtab *sep;
88726877Skarels {
88826877Skarels 	long result;
88926877Skarels 	struct sockaddr sa;
89026877Skarels 	int size;
89126877Skarels 
89226877Skarels 	size = sizeof(sa);
89326921Slepreau 	if (recvfrom(s, (char *)&result, sizeof(result), 0, &sa, &size) < 0)
89426877Skarels 		return;
89526877Skarels 	result = machtime();
89626877Skarels 	(void) sendto(s, (char *) &result, sizeof(result), 0, &sa, sizeof(sa));
89726877Skarels }
89826877Skarels 
89926877Skarels /* ARGSUSED */
90026877Skarels daytime_stream(s, sep)		/* Return human-readable time of day */
90126877Skarels 	int s;
90226877Skarels 	struct servtab *sep;
90326877Skarels {
90426877Skarels 	char buffer[256];
90526877Skarels 	time_t time(), clock;
90626877Skarels 	char *ctime();
90726877Skarels 
90826877Skarels 	clock = time((time_t *) 0);
90926877Skarels 
91026877Skarels 	sprintf(buffer, "%s\r", ctime(&clock));
91126921Slepreau 	(void) write(s, buffer, strlen(buffer));
91226877Skarels }
91326877Skarels 
91426877Skarels /* ARGSUSED */
91526877Skarels daytime_dg(s, sep)		/* Return human-readable time of day */
91626877Skarels 	int s;
91726877Skarels 	struct servtab *sep;
91826877Skarels {
91926877Skarels 	char buffer[256];
92026877Skarels 	time_t time(), clock;
92126877Skarels 	struct sockaddr sa;
92226877Skarels 	int size;
92326877Skarels 	char *ctime();
92426877Skarels 
92526877Skarels 	clock = time((time_t *) 0);
92626877Skarels 
92726877Skarels 	size = sizeof(sa);
92826877Skarels 	if (recvfrom(s, buffer, sizeof(buffer), 0, &sa, &size) < 0)
92926877Skarels 		return;
93026877Skarels 	sprintf(buffer, "%s\r", ctime(&clock));
93126877Skarels 	(void) sendto(s, buffer, strlen(buffer), 0, &sa, sizeof(sa));
93226877Skarels }
93329794Skarels 
93429794Skarels /*
93529794Skarels  * print_service:
93629794Skarels  *	Dump relevant information to stderr
93729794Skarels  */
93829794Skarels print_service(action, sep)
93929794Skarels 	char *action;
94029794Skarels 	struct servtab *sep;
94129794Skarels {
94229794Skarels 	fprintf(stderr,
94329794Skarels 	    "%s: %s proto=%s, wait=%d, user=%s builtin=%x server=%s\n",
94429794Skarels 	    action, sep->se_service, sep->se_proto,
94529794Skarels 	    sep->se_wait, sep->se_user, sep->se_bi, sep->se_server);
94629794Skarels }
947