blob: 33b97ba94e23e288ccfd0a8624c15eb72ea54677 [file] [log] [blame]
Glenn L McGrath06e95652003-02-09 06:51:14 +00001/*
2 * Copyright (c) 1983,1991 The Regents of the University of California.
3 * All rights reserved.
4 *
5 * This code is derived from software contributed to Berkeley by
6 * David A. Holland.
7 *
8 * Busybox port by Vladimir Oleynik (C) 2001-2003 <dzo@simtreas.ru>
9 *
10 * This program is free software; you can redistribute it and/or modify
11 * it under the terms of the GNU General Public License as published by
12 * the Free Software Foundation; either version 2 of the License, or
13 * (at your option) any later version.
14 *
15 * This program is distributed in the hope that it will be useful,
16 * but WITHOUT ANY WARRANTY; without even the implied warranty of
17 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
18 * General Public License for more details.
19 *
20 * You should have received a copy of the GNU General Public License
21 * along with this program; if not, write to the Free Software
22 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
23 *
24 */
25
26/*
27 * Inetd - Internet super-server
28 *
29 * This program invokes all internet services as needed.
30 * connection-oriented services are invoked each time a
31 * connection is made, by creating a process. This process
32 * is passed the connection as file descriptor 0 and is
33 * expected to do a getpeername to find out the source host
34 * and port.
35 *
36 * Datagram oriented services are invoked when a datagram
37 * arrives; a process is created and passed a pending message
38 * on file descriptor 0. Datagram servers may either connect
39 * to their peer, freeing up the original socket for inetd
40 * to receive further messages on, or ``take over the socket'',
41 * processing all arriving datagrams and, eventually, timing
42 * out. The first type of server is said to be ``multi-threaded'';
43 * the second type of server ``single-threaded''.
44 *
45 * Inetd uses a configuration file which is read at startup
46 * and, possibly, at some later time in response to a hangup signal.
47 * The configuration file is ``free format'' with fields given in the
48 * order shown below. Continuation lines for an entry must being with
49 * a space or tab. All fields must be present in each entry.
50 *
51 * service name must be in /etc/services
52 * socket type stream/dgram/raw/rdm/seqpacket
53 * protocol must be in /etc/protocols
54 * wait/nowait[.max] single-threaded/multi-threaded, max #
55 * user[.group] user/group to run daemon as
56 * server program full path name
57 * server program arguments maximum of MAXARGS (20)
58 *
59 * RPC services unsuported
60 *
61 * Comment lines are indicated by a `#' in column 1.
62 */
63
64/*
65 * Here's the scoop concerning the user.group feature:
66 *
67 * 1) No group listed.
68 *
69 * a) for root: NO setuid() or setgid() is done
70 *
71 * b) nonroot: setuid()
72 * setgid(primary group as found in passwd)
73 * initgroups(name, primary group)
74 *
75 * 2) set-group-option on.
76 *
77 * a) for root: NO setuid()
78 * setgid(specified group)
79 * setgroups(1, specified group)
80 *
81 * b) nonroot: setuid()
82 * setgid(specified group)
83 * initgroups(name, specified group)
84 *
85 * All supplementary groups are discarded at startup in case inetd was
86 * run manually.
87 */
88
89#define __USE_BSD_SIGNAL
90
91#include "busybox.h"
92
93#include <sys/param.h>
94#include <sys/stat.h>
95#include <sys/ioctl.h>
96#include <sys/socket.h>
97#include <sys/un.h>
98#include <sys/file.h>
99#include <sys/wait.h>
100#include <sys/time.h>
101#include <sys/resource.h>
102
103#ifndef __linux__
104#ifndef RLIMIT_NOFILE
105#define RLIMIT_NOFILE RLIMIT_OFILE
106#endif
107#endif
108
109#include <sys/param.h>
110#include <sys/stat.h>
111#include <sys/ioctl.h>
112#include <sys/socket.h>
113#include <sys/file.h>
114#include <sys/wait.h>
115#include <sys/time.h>
116#include <sys/resource.h>
117
118#include <netinet/in.h>
119#include <netinet/ip.h>
120#include <arpa/inet.h>
121
122#include <errno.h>
123#include <signal.h>
124#include <netdb.h>
125#include <syslog.h>
Glenn L McGrath06e95652003-02-09 06:51:14 +0000126#include <stdio.h>
127#include <stdlib.h>
128#include <string.h>
129#include <getopt.h>
130#include <unistd.h>
131#include <stdarg.h>
Manuel Novoa III c2843562003-02-11 07:06:06 +0000132#include <time.h>
Glenn L McGrath06e95652003-02-09 06:51:14 +0000133
134#define _PATH_INETDCONF "/etc/inetd.conf"
135#define _PATH_INETDPID "/var/run/inetd.pid"
136
137#ifndef MIN
138#define MIN(a, b) ((a) < (b) ? (a) : (b))
139#endif
140
141#define TOOMANY 40 /* don't start more than TOOMANY */
142#define CNT_INTVL 60 /* servers in CNT_INTVL sec. */
143#define RETRYTIME (60*10) /* retry after bind or server fail */
144
145#ifndef OPEN_MAX
146#define OPEN_MAX 64
147#endif
148
149
150/* Reserve some descriptors, 3 stdio + at least: 1 log, 1 conf. file */
151#define FD_MARGIN (8)
152static int rlim_ofile_cur = OPEN_MAX;
153
154#ifdef RLIMIT_NOFILE
155static struct rlimit rlim_ofile;
156#endif
157
Glenn L McGrathb1207b32003-02-10 22:31:09 +0000158#define INETD_UNSUPPORT_BILTIN 1
159
160/* Check unsupporting builtin */
161#ifdef CONFIG_FEATURE_INETD_SUPPORT_BILTIN_ECHO
162#undef INETD_UNSUPPORT_BILTIN
163#endif
164#ifdef CONFIG_FEATURE_INETD_SUPPORT_BILTIN_DISCARD
165#undef INETD_UNSUPPORT_BILTIN
166#endif
167#ifdef CONFIG_FEATURE_INETD_SUPPORT_BILTIN_TIME
168#undef INETD_UNSUPPORT_BILTIN
169#endif
170#ifdef CONFIG_FEATURE_INETD_SUPPORT_BILTIN_DAYTIME
171#undef INETD_UNSUPPORT_BILTIN
172#endif
173#ifdef CONFIG_FEATURE_INETD_SUPPORT_BILTIN_CHARGEN
174#undef INETD_UNSUPPORT_BILTIN
175#endif
Glenn L McGrath06e95652003-02-09 06:51:14 +0000176
177static struct servtab {
178 char *se_service; /* name of service */
179 int se_socktype; /* type of socket to use */
180 int se_family; /* address family */
181 char *se_proto; /* protocol used */
182 short se_wait; /* single threaded server */
183 short se_checked; /* looked at during merge */
184 char *se_user; /* user name to run as */
185 char *se_group; /* group name to run as */
186#ifndef INETD_UNSUPPORT_BILTIN
187 const struct biltin *se_bi; /* if built-in, description */
188#endif
189 char *se_server; /* server program */
190#define MAXARGV 20
191 char *se_argv[MAXARGV+1]; /* program arguments */
192 int se_fd; /* open descriptor */
193 union {
194 struct sockaddr se_un_ctrladdr;
195 struct sockaddr_in se_un_ctrladdr_in;
196 struct sockaddr_un se_un_ctrladdr_un;
197 } se_un; /* bound address */
198#define se_ctrladdr se_un.se_un_ctrladdr
199#define se_ctrladdr_in se_un.se_un_ctrladdr_in
200#define se_ctrladdr_un se_un.se_un_ctrladdr_un
201 int se_ctrladdr_size;
202 int se_max; /* max # of instances of this service */
203 int se_count; /* number started since se_time */
204 struct timeval se_time; /* start of se_count */
205 struct servtab *se_next;
206} *servtab;
207
208/* Length of socket listen queue. Should be per-service probably. */
209static int global_queuelen = 128;
210
211static int nsock, maxsock;
212static fd_set allsock;
213static int timingout;
214static sigset_t blockmask, emptymask;
215
216
Glenn L McGrath06e95652003-02-09 06:51:14 +0000217 /* Echo received data */
218#ifdef CONFIG_FEATURE_INETD_SUPPORT_BILTIN_ECHO
Glenn L McGrath06e95652003-02-09 06:51:14 +0000219static void echo_stream(int, struct servtab *);
220static void echo_dg(int, struct servtab *);
221#endif
222 /* Internet /dev/null */
223#ifdef CONFIG_FEATURE_INETD_SUPPORT_BILTIN_DISCARD
Glenn L McGrath06e95652003-02-09 06:51:14 +0000224static void discard_stream(int, struct servtab *);
225static void discard_dg(int, struct servtab *);
226#endif
227 /* Return 32 bit time since 1900 */
228#ifdef CONFIG_FEATURE_INETD_SUPPORT_BILTIN_TIME
Glenn L McGrath06e95652003-02-09 06:51:14 +0000229static void machtime_stream(int, struct servtab *);
230static void machtime_dg(int, struct servtab *);
231#endif
232 /* Return human-readable time */
233#ifdef CONFIG_FEATURE_INETD_SUPPORT_BILTIN_DAYTIME
Glenn L McGrath06e95652003-02-09 06:51:14 +0000234static void daytime_stream(int, struct servtab *);
235static void daytime_dg(int, struct servtab *);
236#endif
237 /* Familiar character generator */
238#ifdef CONFIG_FEATURE_INETD_SUPPORT_BILTIN_CHARGEN
Glenn L McGrath06e95652003-02-09 06:51:14 +0000239static void chargen_stream(int, struct servtab *);
240static void chargen_dg(int, struct servtab *);
241#endif
242
243#ifndef INETD_UNSUPPORT_BILTIN
244struct biltin {
245 const char *bi_service; /* internally provided service name */
246 int bi_socktype; /* type of socket supported */
247 short bi_fork; /* 1 if should fork before call */
248 short bi_wait; /* 1 if should wait for child */
249 void (*bi_fn)(int, struct servtab *); /* fn which performs it */
250};
251
252static const struct biltin biltins[] = {
253#ifdef CONFIG_FEATURE_INETD_SUPPORT_BILTIN_ECHO
254 /* Echo received data */
255 { "echo", SOCK_STREAM, 1, 0, echo_stream, },
256 { "echo", SOCK_DGRAM, 0, 0, echo_dg, },
257#endif
258#ifdef CONFIG_FEATURE_INETD_SUPPORT_BILTIN_DISCARD
259 /* Internet /dev/null */
260 { "discard", SOCK_STREAM, 1, 0, discard_stream, },
261 { "discard", SOCK_DGRAM, 0, 0, discard_dg, },
262#endif
263#ifdef CONFIG_FEATURE_INETD_SUPPORT_BILTIN_TIME
264 /* Return 32 bit time since 1900 */
265 { "time", SOCK_STREAM, 0, 0, machtime_stream, },
266 { "time", SOCK_DGRAM, 0, 0, machtime_dg, },
267#endif
268#ifdef CONFIG_FEATURE_INETD_SUPPORT_BILTIN_DAYTIME
269 /* Return human-readable time */
270 { "daytime", SOCK_STREAM, 0, 0, daytime_stream, },
271 { "daytime", SOCK_DGRAM, 0, 0, daytime_dg, },
272#endif
273#ifdef CONFIG_FEATURE_INETD_SUPPORT_BILTIN_CHARGEN
274 /* Familiar character generator */
275 { "chargen", SOCK_STREAM, 1, 0, chargen_stream, },
276 { "chargen", SOCK_DGRAM, 0, 0, chargen_dg, },
277#endif
278 { NULL, 0, 0, 0, NULL }
279};
280#endif /* INETD_UNSUPPORT_BILTIN */
281
282#define NUMINT (sizeof(intab) / sizeof(struct inent))
283static const char *CONFIG = _PATH_INETDCONF;
284
285#define BCOPY(s, d, z) memcpy(d, s, z)
286
287static void
288syslog_err_and_discard_dg(int se_socktype, const char *msg, ...)
289 __attribute__ ((noreturn, format (printf, 2, 3)));
290
291static void
292syslog_err_and_discard_dg(int se_socktype, const char *msg, ...)
293{
294 char buf[50];
295 va_list p;
296
297 va_start(p, msg);
298 vsyslog(LOG_ERR, msg, p);
299 if (se_socktype != SOCK_STREAM)
300 recv(0, buf, sizeof (buf), 0);
301 _exit(1);
302}
303
304static FILE *fconfig;
305static char line[256];
306
307static FILE *
308setconfig(void)
309{
310 FILE *f = fconfig;
311
312 if (f != NULL) {
313 fseek(f, 0L, L_SET);
314 } else {
315 f = fconfig = fopen(CONFIG, "r");
316 if(f == NULL)
317 syslog(LOG_ERR, "%s: %m", CONFIG);
318 }
319 return f;
320}
321
322static char *
323nextline(void)
324{
325 char *cp;
326 FILE *fd = fconfig;
327
328 if (fgets(line, sizeof (line), fd) == NULL)
329 return ((char *)0);
330 cp = strchr(line, '\n');
331 if (cp)
332 *cp = '\0';
333 return (line);
334}
335
336static char *
337skip(char **cpp)
338{
339 char *cp = *cpp;
340 char *start;
341
342 if (*cpp == NULL)
343 return ((char *)0);
344
345again:
346 while (*cp == ' ' || *cp == '\t')
347 cp++;
348 if (*cp == '\0') {
349 int c;
350
351 c = getc(fconfig);
352 (void) ungetc(c, fconfig);
353 if (c == ' ' || c == '\t')
354 if ((cp = nextline()) != NULL)
355 goto again;
356 *cpp = NULL;
357 return NULL;
358 }
359 start = cp;
360 while (*cp && *cp != ' ' && *cp != '\t')
361 cp++;
362 if (*cp != '\0')
363 *cp++ = '\0';
364 *cpp = cp;
365 return (start);
366}
367
368static char *
369newstr(char *cp)
370{
371 cp = strdup(cp ? cp : "");
372 if (cp)
373 return(cp);
374
375 syslog_err_and_discard_dg(SOCK_STREAM, "strdup: %m");
376}
377
378
379static struct servtab *
380getconfigent(void)
381{
382 static struct servtab serv;
383 struct servtab *sep = &serv;
384 int argc;
385 char *cp, *arg;
386
387more:
388 while ((cp = nextline()) && *cp == '#')
389 ;
390 if (cp == NULL)
391 return ((struct servtab *)0);
392 memset((char *)sep, 0, sizeof *sep);
393 sep->se_service = newstr(skip(&cp));
394 arg = skip(&cp);
395 if (arg == NULL)
396 goto more;
397
398 if (strcmp(arg, "stream") == 0)
399 sep->se_socktype = SOCK_STREAM;
400 else if (strcmp(arg, "dgram") == 0)
401 sep->se_socktype = SOCK_DGRAM;
402 else if (strcmp(arg, "rdm") == 0)
403 sep->se_socktype = SOCK_RDM;
404 else if (strcmp(arg, "seqpacket") == 0)
405 sep->se_socktype = SOCK_SEQPACKET;
406 else if (strcmp(arg, "raw") == 0)
407 sep->se_socktype = SOCK_RAW;
408 else
409 sep->se_socktype = -1;
410
411 sep->se_proto = newstr(skip(&cp));
412 if (strcmp(sep->se_proto, "unix") == 0) {
413 sep->se_family = AF_UNIX;
414 } else {
415 sep->se_family = AF_INET;
416 if (strncmp(sep->se_proto, "rpc/", 4) == 0) {
417 syslog(LOG_ERR, "%s: rpc services not suported",
418 sep->se_service);
419 goto more;
420 }
421 }
422 arg = skip(&cp);
423 if (arg == NULL)
424 goto more;
425 {
426 char *s = strchr(arg, '.');
427 if (s) {
428 *s++ = '\0';
429 sep->se_max = atoi(s);
430 } else
431 sep->se_max = TOOMANY;
432 }
433 sep->se_wait = strcmp(arg, "wait") == 0;
434 sep->se_user = newstr(skip(&cp));
435 sep->se_group = strchr(sep->se_user, '.');
436 if (sep->se_group) {
437 *sep->se_group++ = '\0';
438 }
439 sep->se_server = newstr(skip(&cp));
440 if (strcmp(sep->se_server, "internal") == 0) {
441#ifndef INETD_UNSUPPORT_BILTIN
442 const struct biltin *bi;
443
444 for (bi = biltins; bi->bi_service; bi++)
445 if (bi->bi_socktype == sep->se_socktype &&
446 strcmp(bi->bi_service, sep->se_service) == 0)
447 break;
448 if (bi->bi_service == 0) {
449 syslog(LOG_ERR, "internal service %s unknown",
450 sep->se_service);
451 goto more;
452 }
453 sep->se_bi = bi;
454 sep->se_wait = bi->bi_wait;
455#else
456 syslog(LOG_ERR, "internal service %s unknown",
457 sep->se_service);
458 goto more;
459#endif
460 } else
461#ifndef INETD_UNSUPPORT_BILTIN
462 sep->se_bi = NULL
463#endif
464 ;
465 argc = 0;
466 for (arg = skip(&cp); cp; arg = skip(&cp)) {
467 if (argc < MAXARGV)
468 sep->se_argv[argc++] = newstr(arg);
469 }
470 while (argc <= MAXARGV)
471 sep->se_argv[argc++] = NULL;
472 return (sep);
473}
474
475static void
476freeconfig(struct servtab *cp)
477{
478 int i;
479
480 free(cp->se_service);
481 free(cp->se_proto);
482 free(cp->se_user);
483 /* Note: se_group is part of the newstr'ed se_user */
484 free(cp->se_server);
485 for (i = 0; i < MAXARGV; i++)
486 free(cp->se_argv[i]);
487}
488
489#ifndef INETD_UNSUPPORT_BILTIN
490static char **Argv;
491static char *LastArg;
492
493static void
494setproctitle(char *a, int s)
495{
496 size_t size;
497 char *cp;
498 struct sockaddr_in sn;
499 char buf[80];
500
501 cp = Argv[0];
502 size = sizeof(sn);
503 if (getpeername(s, (struct sockaddr *)&sn, &size) == 0)
504 (void) sprintf(buf, "-%s [%s]", a, inet_ntoa(sn.sin_addr));
505 else
506 (void) sprintf(buf, "-%s", a);
507 strncpy(cp, buf, LastArg - cp);
508 cp += strlen(cp);
509 while (cp < LastArg)
510 *cp++ = ' ';
511}
512#endif /* INETD_UNSUPPORT_BILTIN */
513
514static struct servtab *
515enter(struct servtab *cp)
516{
517 struct servtab *sep;
518 sigset_t oldmask;
519
520 sep = (struct servtab *)malloc(sizeof (*sep));
521 if (sep == NULL) {
Manuel Novoa III cad53642003-03-19 09:13:01 +0000522 syslog_err_and_discard_dg(SOCK_STREAM, bb_msg_memory_exhausted);
Glenn L McGrath06e95652003-02-09 06:51:14 +0000523 }
524 *sep = *cp;
525 sep->se_fd = -1;
526 sigprocmask(SIG_BLOCK, &blockmask, &oldmask);
527 sep->se_next = servtab;
528 servtab = sep;
529 sigprocmask(SIG_SETMASK, &oldmask, NULL);
530 return (sep);
531}
532
533static int
534bump_nofile(void)
535{
536#ifdef RLIMIT_NOFILE
537
538#define FD_CHUNK 32
539
540 struct rlimit rl;
541
542 if (getrlimit(RLIMIT_NOFILE, &rl) < 0) {
543 syslog(LOG_ERR, "getrlimit: %m");
544 return -1;
545 }
546 rl.rlim_cur = MIN(rl.rlim_max, rl.rlim_cur + FD_CHUNK);
547 if (rl.rlim_cur <= rlim_ofile_cur) {
548 syslog(LOG_ERR,
Glenn L McGrathb1207b32003-02-10 22:31:09 +0000549#if _FILE_OFFSET_BITS == 64
550 "bump_nofile: cannot extend file limit, max = %lld",
551#else
552 "bump_nofile: cannot extend file limit, max = %ld",
553#endif
Glenn L McGrath06e95652003-02-09 06:51:14 +0000554 rl.rlim_cur);
555 return -1;
556 }
557
558 if (setrlimit(RLIMIT_NOFILE, &rl) < 0) {
559 syslog(LOG_ERR, "setrlimit: %m");
560 return -1;
561 }
562
563 rlim_ofile_cur = rl.rlim_cur;
564 return 0;
565
566#else
567 syslog(LOG_ERR, "bump_nofile: cannot extend file limit");
568 return -1;
569#endif
570}
571
572
573static void
574setup(struct servtab *sep)
575{
576 int on = 1;
577
578 if ((sep->se_fd = socket(sep->se_family, sep->se_socktype, 0)) < 0) {
579 syslog(LOG_ERR, "%s/%s: socket: %m",
580 sep->se_service, sep->se_proto);
581 return;
582 }
583 if (setsockopt(sep->se_fd, SOL_SOCKET, SO_REUSEADDR, (void *)&on,
584 sizeof(on)) < 0)
585 syslog(LOG_ERR, "setsockopt (SO_REUSEADDR): %m");
586 if (bind(sep->se_fd, &sep->se_ctrladdr, sep->se_ctrladdr_size) < 0) {
587 syslog(LOG_ERR, "%s/%s: bind: %m",
588 sep->se_service, sep->se_proto);
589 (void) close(sep->se_fd);
590 sep->se_fd = -1;
591 if (!timingout) {
592 timingout = 1;
593 alarm(RETRYTIME);
594 }
595 return;
596 }
597 if (sep->se_socktype == SOCK_STREAM)
598 listen(sep->se_fd, global_queuelen);
599
600 FD_SET(sep->se_fd, &allsock);
601 nsock++;
602 if (sep->se_fd > maxsock) {
603 maxsock = sep->se_fd;
604 if (maxsock > rlim_ofile_cur - FD_MARGIN)
605 bump_nofile();
606 }
607}
608
609static void
610config(int signum)
611{
612 struct servtab *sep, *cp, **sepp;
613 sigset_t oldmask;
614 unsigned n;
615
616 (void)signum;
617 if (setconfig() == NULL)
618 return;
619
620 for (sep = servtab; sep; sep = sep->se_next)
621 sep->se_checked = 0;
622 while ((cp = getconfigent()) != NULL) {
623 for (sep = servtab; sep; sep = sep->se_next)
624 if (strcmp(sep->se_service, cp->se_service) == 0 &&
625 strcmp(sep->se_proto, cp->se_proto) == 0)
626 break;
627 if (sep != 0) {
628 int i;
629
630#define SWAP(type, a, b) {type c=(type)a; (type)a=(type)b; (type)b=(type)c;}
631
632 sigprocmask(SIG_BLOCK, &emptymask, &oldmask);
633 /*
634 * sep->se_wait may be holding the pid of a daemon
635 * that we're waiting for. If so, don't overwrite
636 * it unless the config file explicitly says don't
637 * wait.
638 */
639 if (
640#ifndef INETD_UNSUPPORT_BILTIN
641 cp->se_bi == 0 &&
642#endif
643 (sep->se_wait == 1 || cp->se_wait == 0))
644 sep->se_wait = cp->se_wait;
645 if (cp->se_max != sep->se_max)
646 SWAP(int, cp->se_max, sep->se_max);
647 if (cp->se_user)
648 SWAP(char *, sep->se_user, cp->se_user);
649 if (cp->se_group)
650 SWAP(char *, sep->se_group, cp->se_group);
651 if (cp->se_server)
652 SWAP(char *, sep->se_server, cp->se_server);
653 for (i = 0; i < MAXARGV; i++)
654 SWAP(char *, sep->se_argv[i], cp->se_argv[i]);
655#undef SWAP
656 sigprocmask(SIG_SETMASK, &oldmask, NULL);
657 freeconfig(cp);
658 } else {
659 sep = enter(cp);
660 }
661 sep->se_checked = 1;
662
663 switch (sep->se_family) {
664 case AF_UNIX:
665 if (sep->se_fd != -1)
666 break;
667 (void)unlink(sep->se_service);
668 n = strlen(sep->se_service);
669 if (n > sizeof(sep->se_ctrladdr_un.sun_path) - 1)
670 n = sizeof(sep->se_ctrladdr_un.sun_path) - 1;
671 strncpy(sep->se_ctrladdr_un.sun_path, sep->se_service, n);
672 sep->se_ctrladdr_un.sun_family = AF_UNIX;
673 sep->se_ctrladdr_size = n +
674 sizeof sep->se_ctrladdr_un.sun_family;
675 setup(sep);
676 break;
677 case AF_INET:
678 sep->se_ctrladdr_in.sin_family = AF_INET;
679 sep->se_ctrladdr_size = sizeof sep->se_ctrladdr_in;
680 {
681 u_short port = htons(atoi(sep->se_service));
682
683 if (!port) {
684 struct servent *sp;
685 sp = getservbyname(sep->se_service,
686 sep->se_proto);
687 if (sp == 0) {
688 syslog(LOG_ERR,
689 "%s/%s: unknown service",
690 sep->se_service, sep->se_proto);
691 continue;
692 }
693 port = sp->s_port;
694 }
695 if (port != sep->se_ctrladdr_in.sin_port) {
696 sep->se_ctrladdr_in.sin_port = port;
697 if (sep->se_fd != -1) {
698 FD_CLR(sep->se_fd, &allsock);
699 nsock--;
700 (void) close(sep->se_fd);
701 }
702 sep->se_fd = -1;
703 }
704 if (sep->se_fd == -1)
705 setup(sep);
706 }
707 }
708 }
709 if (fconfig) {
710 (void) fclose(fconfig);
711 fconfig = NULL;
712 }
713 /*
714 * Purge anything not looked at above.
715 */
716 sigprocmask(SIG_SETMASK, &blockmask, &oldmask);
717 sepp = &servtab;
718 while ((sep = *sepp) != NULL) {
719 if (sep->se_checked) {
720 sepp = &sep->se_next;
721 continue;
722 }
723 *sepp = sep->se_next;
724 if (sep->se_fd != -1) {
725 FD_CLR(sep->se_fd, &allsock);
726 nsock--;
727 (void) close(sep->se_fd);
728 }
729 if (sep->se_family == AF_UNIX)
730 (void)unlink(sep->se_service);
731 freeconfig(sep);
732 free((char *)sep);
733 }
734 sigprocmask(SIG_SETMASK, &oldmask, NULL);
735}
736
737
738
739static void
740reapchild(int signum)
741{
742 int status;
743 int pid;
744 struct servtab *sep;
745
746 (void)signum;
747 for (;;) {
748 pid = wait3(&status, WNOHANG, (struct rusage *)0);
749 if (pid <= 0)
750 break;
751 for (sep = servtab; sep; sep = sep->se_next)
752 if (sep->se_wait == pid) {
753 if (WIFEXITED(status) && WEXITSTATUS(status))
754 syslog(LOG_WARNING,
755 "%s: exit status 0x%x",
756 sep->se_server, WEXITSTATUS(status));
757 else if (WIFSIGNALED(status))
758 syslog(LOG_WARNING,
759 "%s: exit signal 0x%x",
760 sep->se_server, WTERMSIG(status));
761 sep->se_wait = 1;
762 FD_SET(sep->se_fd, &allsock);
763 nsock++;
764 }
765 }
766}
767
768static void
769retry(int signum)
770{
771 struct servtab *sep;
772
773 (void)signum;
774 timingout = 0;
775 for (sep = servtab; sep; sep = sep->se_next) {
776 if (sep->se_fd == -1) {
777 switch (sep->se_family) {
778 case AF_UNIX:
779 case AF_INET:
780 setup(sep);
781 break;
782 }
783 }
784 }
785}
786
787static void
788goaway(int signum)
789{
790 struct servtab *sep;
791
792 (void)signum;
793 for (sep = servtab; sep; sep = sep->se_next)
794 if (sep->se_fd != -1 && sep->se_family == AF_UNIX)
795 (void)unlink(sep->se_service);
796 (void)unlink(_PATH_INETDPID);
797 exit(0);
798}
799
800
801
802extern int
803inetd_main(int argc, char *argv[])
804{
805 struct servtab *sep;
806 struct passwd *pwd;
807 struct group *grp = NULL;
808 struct sigaction sa;
809 int ch, pid;
810 gid_t gid;
811
812#ifdef INETD_UNSUPPORT_BILTIN
813# define dofork 1
814#else
815 int dofork;
816 extern char **environ;
817#endif
818
819 gid = getgid();
820 setgroups(1, &gid);
821
822#ifndef INETD_UNSUPPORT_BILTIN
823 Argv = argv;
824 if (environ == 0 || *environ == 0)
825 environ = argv;
826 while (*environ)
827 environ++;
828 LastArg = environ[-1] + strlen(environ[-1]);
829#endif
830
831 while ((ch = getopt(argc, argv, "q:")) != EOF)
832 switch(ch) {
833 case 'q':
834 global_queuelen = atoi(optarg);
835 if (global_queuelen < 8) global_queuelen=8;
836 break;
837 default:
Manuel Novoa III cad53642003-03-19 09:13:01 +0000838 bb_show_usage(); // "[-q len] [conf]"
Glenn L McGrath06e95652003-02-09 06:51:14 +0000839 }
840 argc -= optind;
841 argv += optind;
842
843 if (argc > 0)
844 CONFIG = argv[0];
845
846 daemon(0, 0);
Manuel Novoa III cad53642003-03-19 09:13:01 +0000847 openlog(bb_applet_name, LOG_PID | LOG_NOWAIT, LOG_DAEMON);
Glenn L McGrath06e95652003-02-09 06:51:14 +0000848 {
849 FILE *fp;
850
851 if ((fp = fopen(_PATH_INETDPID, "w")) != NULL) {
852 fprintf(fp, "%u\n", getpid());
853 (void)fclose(fp);
854 }
855 }
856
857#ifdef RLIMIT_NOFILE
858 if (getrlimit(RLIMIT_NOFILE, &rlim_ofile) < 0) {
859 syslog(LOG_ERR, "getrlimit: %m");
860 } else {
861 rlim_ofile_cur = rlim_ofile.rlim_cur;
862 if (rlim_ofile_cur == RLIM_INFINITY) /* ! */
863 rlim_ofile_cur = OPEN_MAX;
864 }
865#endif
866
867 config(0);
868
869 sigemptyset(&emptymask);
870 sigemptyset(&blockmask);
871 sigaddset(&blockmask, SIGCHLD);
872 sigaddset(&blockmask, SIGHUP);
873 sigaddset(&blockmask, SIGALRM);
874
875 memset(&sa, 0, sizeof(sa));
876 sa.sa_mask = blockmask;
877 sa.sa_handler = retry;
878 sigaction(SIGALRM, &sa, NULL);
879 sa.sa_handler = config;
880 sigaction(SIGHUP, &sa, NULL);
881 sa.sa_handler = reapchild;
882 sigaction(SIGCHLD, &sa, NULL);
883 sa.sa_handler = goaway;
884 sigaction(SIGTERM, &sa, NULL);
885 sa.sa_handler = goaway;
886 sigaction(SIGINT, &sa, NULL);
887 sa.sa_handler = SIG_IGN;
888 sigaction(SIGPIPE, &sa, NULL);
889
890 {
891 /* space for daemons to overwrite environment for ps */
892#define DUMMYSIZE 100
893 char dummy[DUMMYSIZE];
894
895 (void)memset(dummy, 'x', DUMMYSIZE - 1);
896 dummy[DUMMYSIZE - 1] = '\0';
897
898 (void)setenv("inetd_dummy", dummy, 1);
899 }
900
901 for (;;) {
902 int n, ctrl;
903 fd_set readable;
904
905 if (nsock == 0) {
906 sigprocmask(SIG_BLOCK, &blockmask, NULL);
907 while (nsock == 0)
908 sigsuspend(&emptymask);
909 sigprocmask(SIG_SETMASK, &emptymask, NULL);
910 }
911 readable = allsock;
912 if ((n = select(maxsock + 1, &readable, (fd_set *)0,
913 (fd_set *)0, (struct timeval *)0)) <= 0) {
914 if (n < 0 && errno != EINTR)
915 syslog(LOG_WARNING, "select: %m");
916 sleep(1);
917 continue;
918 }
919 for (sep = servtab; n && sep; sep = sep->se_next)
920 if (sep->se_fd != -1 && FD_ISSET(sep->se_fd, &readable)) {
921 n--;
922 if (!sep->se_wait && sep->se_socktype == SOCK_STREAM) {
923 /* Fixed AGC */
924 fcntl(sep->se_fd, F_SETFL, O_NDELAY);
925 /* --------- */
926 ctrl = accept(sep->se_fd, NULL, NULL);
927 fcntl(sep->se_fd, F_SETFL, 0);
928 if (ctrl < 0) {
929 if (errno == EINTR || errno == EWOULDBLOCK)
930 continue;
931 syslog(LOG_WARNING, "accept (for %s): %m",
932 sep->se_service);
933 continue;
934 }
935 } else
936 ctrl = sep->se_fd;
937 sigprocmask(SIG_BLOCK, &blockmask, NULL);
938 pid = 0;
939#ifndef INETD_UNSUPPORT_BILTIN
940 dofork = (sep->se_bi == 0 || sep->se_bi->bi_fork);
941#endif
942 if (dofork) {
943 if (sep->se_count++ == 0)
944 (void)gettimeofday(&sep->se_time,
945 (struct timezone *)0);
946 else if (sep->se_count >= sep->se_max) {
947 struct timeval now;
948
949 (void)gettimeofday(&now, (struct timezone *)0);
950 if (now.tv_sec - sep->se_time.tv_sec >
951 CNT_INTVL) {
952 sep->se_time = now;
953 sep->se_count = 1;
954 } else {
955 syslog(LOG_ERR,
956 "%s/%s server failing (looping), service terminated",
957 sep->se_service, sep->se_proto);
958 FD_CLR(sep->se_fd, &allsock);
959 (void) close(sep->se_fd);
960 sep->se_fd = -1;
961 sep->se_count = 0;
962 nsock--;
963 sigprocmask(SIG_SETMASK, &emptymask,
964 NULL);
965 if (!timingout) {
966 timingout = 1;
967 alarm(RETRYTIME);
968 }
969 continue;
970 }
971 }
972 pid = fork();
973 }
974 if (pid < 0) {
975 syslog(LOG_ERR, "fork: %m");
976 if (sep->se_socktype == SOCK_STREAM)
977 close(ctrl);
978 sigprocmask(SIG_SETMASK, &emptymask, NULL);
979 sleep(1);
980 continue;
981 }
982 if (pid && sep->se_wait) {
983 sep->se_wait = pid;
984 FD_CLR(sep->se_fd, &allsock);
985 nsock--;
986 }
987 sigprocmask(SIG_SETMASK, &emptymask, NULL);
988 if (pid == 0) {
989#ifndef INETD_UNSUPPORT_BILTIN
990 if (sep->se_bi)
991 (*sep->se_bi->bi_fn)(ctrl, sep);
992 else
993#endif
994 {
995 if ((pwd = getpwnam(sep->se_user)) == NULL) {
996 syslog_err_and_discard_dg(
997 sep->se_socktype,
998 "getpwnam: %s: No such user",
999 sep->se_user);
1000 }
1001 if (sep->se_group &&
1002 (grp = getgrnam(sep->se_group)) == NULL) {
1003 syslog_err_and_discard_dg(
1004 sep->se_socktype,
1005 "getgrnam: %s: No such group",
1006 sep->se_group);
1007 }
1008 /*
1009 * Ok. There are four cases here:
1010 * 1. nonroot user, no group specified
1011 * 2. nonroot user, some group specified
1012 * 3. root user, no group specified
1013 * 4. root user, some group specified
1014 * In cases 2 and 4 we setgid to the specified
1015 * group. In cases 1 and 2 we run initgroups
1016 * to run with the groups of the given user.
1017 * In case 4 we do setgroups to run with the
1018 * given group. In case 3 we do nothing.
1019 */
1020 if (pwd->pw_uid) {
1021 if (sep->se_group)
1022 pwd->pw_gid = grp->gr_gid;
1023 setgid((gid_t)pwd->pw_gid);
1024 initgroups(pwd->pw_name, pwd->pw_gid);
1025 setuid((uid_t)pwd->pw_uid);
1026 } else if (sep->se_group) {
1027 setgid((gid_t)grp->gr_gid);
1028 setgroups(1, &grp->gr_gid);
1029 }
1030 dup2(ctrl, 0);
1031 close(ctrl);
1032 dup2(0, 1);
1033 dup2(0, 2);
1034#ifdef RLIMIT_NOFILE
1035 if (rlim_ofile.rlim_cur != rlim_ofile_cur) {
1036 if (setrlimit(RLIMIT_NOFILE,
1037 &rlim_ofile) < 0)
1038 syslog(LOG_ERR,"setrlimit: %m");
1039 }
1040#endif
1041 for (ctrl = rlim_ofile_cur-1; --ctrl > 2; )
1042 (void)close(ctrl);
1043
1044 memset(&sa, 0, sizeof(sa));
1045 sa.sa_handler = SIG_DFL;
1046 sigaction(SIGPIPE, &sa, NULL);
1047
1048 execv(sep->se_server, sep->se_argv);
1049 syslog_err_and_discard_dg(sep->se_socktype,
1050 "execv %s: %m", sep->se_server);
1051 }
1052 }
1053 if (!sep->se_wait && sep->se_socktype == SOCK_STREAM)
1054 close(ctrl);
1055 }
1056 }
1057}
1058
1059
1060/*
1061 * Internet services provided internally by inetd:
1062 */
1063#define BUFSIZE 4096
1064
1065#ifdef CONFIG_FEATURE_INETD_SUPPORT_BILTIN_ECHO
1066/* Echo service -- echo data back */
1067static void
1068echo_stream(int s, struct servtab *sep)
1069{
1070 char buffer[BUFSIZE];
1071 int i;
1072
1073 setproctitle(sep->se_service, s);
1074 while ((i = read(s, buffer, sizeof(buffer))) > 0 &&
1075 write(s, buffer, i) > 0)
1076 ;
1077 exit(0);
1078}
1079
1080/* Echo service -- echo data back */
1081static void
1082echo_dg(int s, struct servtab *sep)
1083{
1084 char buffer[BUFSIZE];
1085 int i;
1086 size_t size;
1087 struct sockaddr sa;
1088
1089 (void)sep;
1090
1091 size = sizeof(sa);
1092 if ((i = recvfrom(s, buffer, sizeof(buffer), 0, &sa, &size)) < 0)
1093 return;
1094 (void) sendto(s, buffer, i, 0, &sa, sizeof(sa));
1095}
1096#endif /* CONFIG_FEATURE_INETD_SUPPORT_BILTIN_ECHO */
1097
1098
1099#ifdef CONFIG_FEATURE_INETD_SUPPORT_BILTIN_DISCARD
1100/* Discard service -- ignore data */
1101static void
1102discard_stream(int s, struct servtab *sep)
1103{
1104 char buffer[BUFSIZE];
1105
1106 setproctitle(sep->se_service, s);
1107 while ((errno = 0, read(s, buffer, sizeof(buffer)) > 0) ||
1108 errno == EINTR)
1109 ;
1110 exit(0);
1111}
1112
1113/* Discard service -- ignore data */
1114static void
1115discard_dg(int s, struct servtab *sep)
1116{
1117 char buffer[BUFSIZE];
1118 (void)sep;
1119 read(s, buffer, sizeof(buffer));
1120}
1121#endif /* CONFIG_FEATURE_INETD_SUPPORT_BILTIN_DISCARD */
1122
1123
1124#ifdef CONFIG_FEATURE_INETD_SUPPORT_BILTIN_CHARGEN
1125#include <ctype.h>
1126#define LINESIZ 72
1127static char ring[128];
1128static char *endring;
1129
1130static void
1131initring(void)
1132{
1133 int i;
1134
1135 endring = ring;
1136
1137 for (i = 0; i <= 128; ++i)
1138 if (isprint(i))
1139 *endring++ = i;
1140}
1141
1142/* Character generator */
1143static void
1144chargen_stream(int s, struct servtab *sep)
1145{
1146 char *rs;
1147 int len;
1148 char text[LINESIZ+2];
1149
1150 setproctitle(sep->se_service, s);
1151
1152 if (!endring) {
1153 initring();
1154 rs = ring;
1155 }
1156
1157 text[LINESIZ] = '\r';
1158 text[LINESIZ + 1] = '\n';
1159 for (rs = ring;;) {
1160 if ((len = endring - rs) >= LINESIZ)
1161 BCOPY(rs, text, LINESIZ);
1162 else {
1163 BCOPY(rs, text, len);
1164 BCOPY(ring, text + len, LINESIZ - len);
1165 }
1166 if (++rs == endring)
1167 rs = ring;
1168 if (write(s, text, sizeof(text)) != sizeof(text))
1169 break;
1170 }
1171 exit(0);
1172}
1173
1174/* Character generator */
1175static void
1176chargen_dg(int s, struct servtab *sep)
1177{
1178 struct sockaddr sa;
1179 static char *rs;
1180 size_t len, size;
1181 char text[LINESIZ+2];
1182
1183 (void)sep;
1184
1185 if (endring == 0) {
1186 initring();
1187 rs = ring;
1188 }
1189
1190 size = sizeof(sa);
1191 if (recvfrom(s, text, sizeof(text), 0, &sa, &size) < 0)
1192 return;
1193
1194 if ((len = endring - rs) >= LINESIZ)
1195 BCOPY(rs, text, LINESIZ);
1196 else {
1197 BCOPY(rs, text, len);
1198 BCOPY(ring, text + len, LINESIZ - len);
1199 }
1200 if (++rs == endring)
1201 rs = ring;
1202 text[LINESIZ] = '\r';
1203 text[LINESIZ + 1] = '\n';
1204 (void) sendto(s, text, sizeof(text), 0, &sa, sizeof(sa));
1205}
1206#endif /* CONFIG_FEATURE_INETD_SUPPORT_BILTIN_CHARGEN */
1207
1208
1209#ifdef CONFIG_FEATURE_INETD_SUPPORT_BILTIN_TIME
1210/*
1211 * Return a machine readable date and time, in the form of the
1212 * number of seconds since midnight, Jan 1, 1900. Since gettimeofday
1213 * returns the number of seconds since midnight, Jan 1, 1970,
1214 * we must add 2208988800 seconds to this figure to make up for
1215 * some seventy years Bell Labs was asleep.
1216 */
1217
1218static long
1219machtime(void)
1220{
1221 struct timeval tv;
1222
1223 if (gettimeofday(&tv, (struct timezone *)0) < 0) {
1224 fprintf(stderr, "Unable to get time of day\n");
1225 return (0L);
1226 }
1227 return (htonl((long)tv.tv_sec + 2208988800UL));
1228}
1229
1230static void
1231machtime_stream(int s, struct servtab *sep)
1232{
1233 long result;
1234 (void)sep;
1235
1236 result = machtime();
1237 write(s, (char *) &result, sizeof(result));
1238}
1239
1240static void
1241machtime_dg(int s, struct servtab *sep)
1242{
1243 long result;
1244 struct sockaddr sa;
1245 size_t size;
1246 (void)sep;
1247
1248 size = sizeof(sa);
1249 if (recvfrom(s, (char *)&result, sizeof(result), 0, &sa, &size) < 0)
1250 return;
1251 result = machtime();
1252 (void) sendto(s, (char *) &result, sizeof(result), 0, &sa, sizeof(sa));
1253}
1254#endif /* CONFIG_FEATURE_INETD_SUPPORT_BILTIN_TIME */
1255
1256
1257#ifdef CONFIG_FEATURE_INETD_SUPPORT_BILTIN_DAYTIME
1258/* Return human-readable time of day */
1259static int
1260human_readable_time_sprintf(char *buffer)
1261{
1262 time_t clocc = time(NULL);
1263
1264 return sprintf(buffer, "%.24s\r\n", ctime(&clocc));
1265}
1266
1267static void
1268daytime_stream(int s, struct servtab *sep)
1269{
1270 char buffer[256];
1271 size_t st = human_readable_time_sprintf(buffer);
1272
1273 (void)sep;
1274
1275 write(s, buffer, st);
1276}
1277
1278/* Return human-readable time of day */
1279static void
1280daytime_dg(int s, struct servtab *sep)
1281{
1282 char buffer[256];
1283 struct sockaddr sa;
1284 size_t size;
1285
1286 (void)sep;
1287
1288 size = sizeof(sa);
1289 if (recvfrom(s, buffer, sizeof(buffer), 0, &sa, &size) < 0)
1290 return;
1291 size = human_readable_time_sprintf(buffer);
1292 sendto(s, buffer, size, 0, &sa, sizeof(sa));
1293}
1294#endif /* CONFIG_FEATURE_INETD_SUPPORT_BILTIN_DAYTIME */