get definition of f_uname right (?)
[unix-history] / usr / src / usr.sbin / syslogd / syslogd.c
index 8023974..e8b93d0 100644 (file)
@@ -1,62 +1,95 @@
+/*
+ * Copyright (c) 1983 Regents of the University of California.
+ * All rights reserved.  The Berkeley software License Agreement
+ * specifies the terms and conditions for redistribution.
+ */
+
+#ifndef lint
+char copyright[] =
+"@(#) Copyright (c) 1983 Regents of the University of California.\n\
+ All rights reserved.\n";
+#endif not lint
+
 #ifndef lint
 #ifndef lint
-static char SccsId[] = "@(#)syslogd.c  4.2 (Berkeley) %G%";
-#endif
+static char sccsid[] = "@(#)syslogd.c  5.3 (Berkeley) %G%";
+#endif not lint
+
+#define COMPAT         /* include 4.3 Alpha compatibility */
 
 /*
  *  syslogd -- log system messages
  *
  * This program implements a system log. It takes a series of lines.
  * Each line may have a priority, signified as "<n>" as
 
 /*
  *  syslogd -- log system messages
  *
  * This program implements a system log. It takes a series of lines.
  * Each line may have a priority, signified as "<n>" as
- * the first three characters of the line.  If this is
- * not present, a default priority (DefPri) is used, which
- * starts out as LOG_NOTICE.  The default priority can get
- * changed using "<*>n".
+ * the first characters of the line.  If this is
+ * not present, a default priority is used.
  *
  * To kill syslogd, send a signal 15 (terminate).  A signal 1 (hup) will
  * cause it to reread its configuration file.
  *
  * Defined Constants:
  *
  *
  * To kill syslogd, send a signal 15 (terminate).  A signal 1 (hup) will
  * cause it to reread its configuration file.
  *
  * Defined Constants:
  *
- * DAEMON -- Userid number to setuid to after setup.
  * MAXLINE -- the maximimum line length that can be handled.
  * MAXLINE -- the maximimum line length that can be handled.
- * NLOGS -- the maximum number of simultaneous log files.
- * NUSERS -- the maximum number of people that can
- *     be designated as "superusers" on your system.
+ * NLOGS   -- the maximum number of simultaneous log files.
+ * DEFUPRI -- the default priority for user messages
+ * DEFSPRI -- the default priority for kernel messages
+ *
+ * Author: Eric Allman
+ * extensive changes by Ralph Campbell
  */
 
  */
 
-#define DAEMON         1       /* Daemon user-id */
-#define        NLOGS           10      /* max number of log files */
-#define        NSUSERS         10      /* max number of special users */
-#define        MAXLINE         1024    /* maximum line length */
+#define        NLOGS           20              /* max number of log files */
+#define        MAXLINE         1024            /* maximum line length */
+#define DEFUPRI                (LOG_USER|LOG_NOTICE)
+#define DEFSPRI                (LOG_KERN|LOG_CRIT)
 
 #include <syslog.h>
 #include <errno.h>
 #include <stdio.h>
 #include <utmp.h>
 #include <ctype.h>
 
 #include <syslog.h>
 #include <errno.h>
 #include <stdio.h>
 #include <utmp.h>
 #include <ctype.h>
+#include <signal.h>
+#include <sysexits.h>
+#include <strings.h>
+
 #include <sys/types.h>
 #include <sys/ioctl.h>
 #include <sys/stat.h>
 #include <sys/types.h>
 #include <sys/ioctl.h>
 #include <sys/stat.h>
-#include <signal.h>
-#include <sysexits.h>
+#include <sys/wait.h>
 #include <sys/socket.h>
 #include <sys/file.h>
 #include <sys/socket.h>
 #include <sys/file.h>
+#include <sys/msgbuf.h>
+#include <sys/uio.h>
 #include <sys/un.h>
 #include <sys/un.h>
+#include <sys/time.h>
+#include <sys/resource.h>
+
 #include <netinet/in.h>
 #include <netdb.h>
 
 #include <netinet/in.h>
 #include <netdb.h>
 
-char   logname[] = "/dev/log";
-char   defconf[] = "/etc/syslog.conf";
-char   defpid[] = "/etc/syslog.pid";
+char   *LogName = "/dev/log";
+char   *ConfFile = "/etc/syslog.conf";
+char   *PidFile = "/etc/syslog.pid";
 char   ctty[] = "/dev/console";
 
 char   ctty[] = "/dev/console";
 
-typedef char   bool;
-#define        TRUE    1
-#define        FALSE   0
+#define FDMASK(fd)     (1 << (fd))
 
 #define        dprintf         if (Debug) printf
 
 
 #define        dprintf         if (Debug) printf
 
-#define UNAMESZ        8       /* length of a login name */
+#define UNAMESZ                8       /* length of a login name */
+#define MAXUNAMES      20      /* maximum number of user names */
+#define MAXFNAME       200     /* max file pathname length */
+
+#define NOPRI          0x10    /* the "no priority" priority */
+
+/*
+ * Flags to logmsg().
+ */
+
+#define IGN_CONS       0x001   /* don't print on console */
+#define SYNC_FILE      0x002   /* do fsync on file after printing */
+#define NOCOPY         0x004   /* don't suppress duplicate messages */
+#define ADDDATE                0x008   /* add a date to the message */
 
 /*
  * This structure represents the files that will have log
 
 /*
  * This structure represents the files that will have log
@@ -64,38 +97,93 @@ typedef char        bool;
  */
 
 struct filed {
  */
 
 struct filed {
-       int     f_file;                 /* file descriptor */
-       short   f_pmask;                /* priority mask */
-       short   f_flags;                /* see #defines below */
-       struct  sockaddr_in f_addr;     /* forwarding address */
-       char    f_name[248];            /* filename */
+       short   f_type;                 /* entry type, see below */
+       short   f_file;                 /* file descriptor */
+       u_char  f_pmask[LOG_NFACILITIES];       /* priority mask */
+       union {
+               char    f_uname[MAXUNAMES][UNAMESZ+1];
+               struct
+               {
+                       char    f_hname[32];
+                       struct sockaddr_in      f_addr;
+               }       f_forw;         /* forwarding address */
+               char    f_fname[MAXFNAME];
+       }       f_un;
 };
 
 };
 
-#define F_TTY  01              /* file is a tty */
-#define F_MARK 02              /* write to the file periodically */
-#define F_FORW 04              /* forward message to another host */
-
-struct filed   Files[NLOGS];
-
-/* list of superusers */
-struct susers {
-       short   s_pmask;                /* priority mask */
-       char    s_name[UNAMESZ+1];
+/* values for f_type */
+#define F_UNUSED       0               /* unused entry */
+#define F_FILE         1               /* regular file */
+#define F_TTY          2               /* terminal */
+#define F_CONSOLE      3               /* console terminal */
+#define F_FORW         4               /* remote machine */
+#define F_USERS                5               /* list of users */
+#define F_WALL         6               /* everyone logged on */
+
+char   *TypeNames[7] = {
+       "UNUSED",       "FILE",         "TTY",          "CONSOLE",
+       "FORW",         "USERS",        "WALL"
 };
 
 };
 
-struct susers Susers[NSUSERS];
+struct filed   Files[NLOGS];
 
 int    Debug;                  /* debug flag */
 
 int    Debug;                  /* debug flag */
-int    LogFile;                /* log file descriptor */
-int    DefPri = LOG_NOTICE;    /* the default priority for untagged msgs */
-int    Sumask = LOG_SALERT;    /* lowest priority written to super-users */
-int    MarkIntvl = 15;         /* mark interval in minutes */
-char   *ConfFile = defconf;    /* configuration file */
-char   rhost[32];              /* hostname of sender (forwarded messages) */
+char   LocalHostName[32];      /* our hostname */
+int    InetInuse = 0;          /* non-zero if INET sockets are being used */
+int    LogPort;                /* port number for INET connections */
+char   PrevLine[MAXLINE + 1];  /* copy of last line to supress repeats */
+char   PrevHost[32];           /* previous host */
+int    PrevFlags;
+int    PrevPri;
+int    PrevCount = 0;          /* number of times seen */
+int    Initialized = 0;        /* set when we have initialized ourselves */
 
 extern int errno, sys_nerr;
 extern char *sys_errlist[];
 
 extern int errno, sys_nerr;
 extern char *sys_errlist[];
-extern char *ctime();
+extern char *ctime(), *index();
+
+#ifdef COMPAT
+int    CompatMode = 0;         /* run in compatibility mode */
+int    CompatCodes[32] = {
+               LOG_USER|LOG_ALERT,     /* 0 -- undefined */
+                       /* kernel priorities */
+               LOG_KERN|LOG_EMERG,     /* KERN_EMERG */
+               LOG_KERN|LOG_ALERT,     /* KERN_ALERT */
+               LOG_KERN|LOG_CRIT,      /* KERN_ERR */
+               LOG_KERN|LOG_ERR,       /* KERN_FAIL */
+               LOG_KERN|LOG_WARNING,   /* KERN_RECOV */
+               LOG_KERN|LOG_INFO,      /* KERN_INFO */
+                       /* user abnormal conditions priorities */
+               LOG_USER|LOG_EMERG,     /* LOG_EMERG */
+               LOG_USER|LOG_ALERT,     /* LOG_ALERT */
+               LOG_USER|LOG_CRIT,      /* LOG_CRIT */
+               LOG_USER|LOG_ERR,       /* LOG_ERR */
+               LOG_USER|LOG_ERR,       /* LOG_ERR */
+               LOG_USER|LOG_WARNING,   /* LOG_WARNING */
+                       /* user priorities */
+               LOG_USER|LOG_ALERT,     /* LOG_SALERT */
+               LOG_AUTH|LOG_NOTICE,    /* LOG_SECURITY */
+               LOG_USER|LOG_INFO,      /* LOG_FIXED */
+               LOG_MAIL|LOG_ERR,       /* LOG_MAIL */
+               LOG_DAEMON|LOG_ERR,     /* LOG_REJECT */
+               LOG_USER|LOG_NOTICE,    /* LOG_NOTICE */
+                       /* user information priorities */
+               LOG_USER|LOG_INFO,      /* LOG_INFO */
+               LOG_LOCAL1|LOG_INFO,    /* LOG_INFO1 */
+               LOG_LOCAL2|LOG_INFO,    /* LOG_INFO2 */
+               LOG_LOCAL3|LOG_INFO,    /* LOG_INFO3 */
+               LOG_LOCAL4|LOG_INFO,    /* LOG_INFO4 */
+               LOG_LOCAL5|LOG_INFO,    /* LOG_INFO5 */
+                       /* user debug/local priorities */
+               LOG_USER|LOG_DEBUG,     /* LOG_DEBUG */
+               LOG_LOCAL1|LOG_DEBUG,   /* LOG_LOCAL1 */
+               LOG_LOCAL2|LOG_DEBUG,   /* LOG_LOCAL2 */
+               LOG_LOCAL3|LOG_DEBUG,   /* LOG_LOCAL3 */
+               LOG_LOCAL4|LOG_DEBUG,   /* LOG_LOCAL4 */
+               LOG_LOCAL5|LOG_DEBUG,   /* LOG_LOCAL5 */
+               LOG_LOCAL6|LOG_DEBUG,   /* LOG_LOCAL6 */
+};
+#endif COMPAT
 
 main(argc, argv)
        int argc;
 
 main(argc, argv)
        int argc;
@@ -103,42 +191,40 @@ main(argc, argv)
 {
        register int i;
        register char *p;
 {
        register int i;
        register char *p;
-       int funix, finet, defreadfds, len;
+       int funix, finet, inetm, fklog, klogm, len;
        struct sockaddr_un sun, fromunix;
        struct sockaddr_in sin, frominet;
        FILE *fp;
        struct sockaddr_un sun, fromunix;
        struct sockaddr_in sin, frominet;
        FILE *fp;
-       char line[MAXLINE + 1];
-       extern die();
-       extern int domark();
-
-       sun.sun_family = AF_UNIX;
-       strncpy(sun.sun_path, logname, sizeof sun.sun_path);
+       char line[MSG_BSIZE + 1];
+       extern int die(), domark(), reapchild();
 
        while (--argc > 0) {
                p = *++argv;
 
        while (--argc > 0) {
                p = *++argv;
-               if (p[0] == '-') {
-                       switch (p[1]) {
-                       case 'm':               /* set mark interval */
-                               MarkIntvl = atoi(&p[2]);
-                               if (MarkIntvl <= 0)
-                                       MarkIntvl = 1;
-                               break;
+               if (p[0] != '-')
+                       usage();
+               switch (p[1]) {
+               case 'f':               /* configuration file */
+                       if (p[2] != '\0')
+                               ConfFile = &p[2];
+                       break;
 
 
-                       case 'f':               /* configuration file */
-                               if (p[2] != '\0')
-                                       ConfFile = &p[2];
-                               break;
+               case 'd':               /* debug */
+                       Debug++;
+                       break;
 
 
-                       case 'd':               /* debug */
-                               Debug++;
-                               break;
+#ifdef COMPAT
+               case 'C':               /* run in compat mode */
+                       CompatMode++;
+                       break;
+#endif COMPAT
 
 
-                       case 'p':               /* port */
-                               if (p[2] != '\0')
-                                       strncpy(sun.sun_path, &p[2], 
-                                               sizeof sun.sun_path);
-                               break;
-                       }
+               case 'p':               /* path */
+                       if (p[2] != '\0')
+                               LogName = &p[2];
+                       break;
+
+               default:
+                       usage();
                }
        }
 
                }
        }
 
@@ -150,26 +236,27 @@ main(argc, argv)
                (void) open("/", 0);
                (void) dup2(0, 1);
                (void) dup2(0, 2);
                (void) open("/", 0);
                (void) dup2(0, 1);
                (void) dup2(0, 2);
-               i = open("/dev/tty", 2);
-               if (i >= 0) {
-                       ioctl(i, TIOCNOTTY, (char *)0);
-                       (void) close(i);
-               }
-       }
-       signal(SIGTERM, die);
+               untty();
+       } else
+               setlinebuf(stdout);
+
+       (void) signal(SIGTERM, die);
+       (void) signal(SIGINT, die);
+       (void) signal(SIGCHLD, reapchild);
+       (void) unlink(LogName);
+
+       sun.sun_family = AF_UNIX;
+       (void) strncpy(sun.sun_path, LogName, sizeof sun.sun_path);
+       (void) gethostname(LocalHostName, sizeof LocalHostName);
        funix = socket(AF_UNIX, SOCK_DGRAM, 0);
        funix = socket(AF_UNIX, SOCK_DGRAM, 0);
-       if (funix >= 0 && bind(funix, &sun,
-           sizeof(sun.sun_family)+strlen(sun.sun_path)) < 0) {
-               close(funix);
-               funix = -1;
-       }
-       if (funix < 0) {
-               fp = fopen(ctty, "w");
-               fprintf(fp, "\r\nsyslog: cannot create %s (%d)\r\n", logname, errno);
-               dprintf("cannot create %s (%d)\n", logname, errno);
-               exit(1);
+       if (funix < 0 || bind(funix, (struct sockaddr *) &sun,
+           sizeof(sun.sun_family)+strlen(sun.sun_path)) < 0 ||
+           chmod(LogName, 0666) < 0) {
+               (void) sprintf(line, "cannot create %s", LogName);
+               logerror(line);
+               dprintf("cannot create %s (%d)\n", LogName, errno);
+               die(0);
        }
        }
-       defreadfds = 1 << funix;
        finet = socket(AF_INET, SOCK_DGRAM, 0);
        if (finet >= 0) {
                struct servent *sp;
        finet = socket(AF_INET, SOCK_DGRAM, 0);
        if (finet >= 0) {
                struct servent *sp;
@@ -178,63 +265,104 @@ main(argc, argv)
                if (sp == NULL) {
                        errno = 0;
                        logerror("syslog/udp: unknown service");
                if (sp == NULL) {
                        errno = 0;
                        logerror("syslog/udp: unknown service");
-                       die();
+                       die(0);
                }
                sin.sin_family = AF_INET;
                }
                sin.sin_family = AF_INET;
-               sin.sin_port = sp->s_port;
-               if (bind(finet, &sin, sizeof(sin), 0) < 0) {
+               sin.sin_port = LogPort = sp->s_port;
+               if (bind(finet, &sin, sizeof(sin)) < 0) {
                        logerror("bind");
                        logerror("bind");
-                       die();
+                       if (!Debug)
+                               die(0);
+               } else {
+                       inetm = FDMASK(finet);
+                       InetInuse = 1;
                }
                }
-               defreadfds |= 1 << finet;
+       }
+       if ((fklog = open("/dev/klog", O_RDONLY)) >= 0)
+               klogm = FDMASK(fklog);
+       else {
+               dprintf("can't open /dev/klog (%d)\n", errno);
+               klogm = 0;
        }
 
        /* tuck my process id away */
        }
 
        /* tuck my process id away */
-       fp = fopen(defpid, "w");
+       fp = fopen(PidFile, "w");
        if (fp != NULL) {
                fprintf(fp, "%d\n", getpid());
        if (fp != NULL) {
                fprintf(fp, "%d\n", getpid());
-               fclose(fp);
+               (void) fclose(fp);
        }
 
        dprintf("off & running....\n");
 
        }
 
        dprintf("off & running....\n");
 
-       for (i = 0; i < NLOGS; i++)
-               Files[i].f_file = -1;
        init();
        init();
-       signal(SIGALRM, domark);
-       alarm(MarkIntvl * 60);
+       (void) signal(SIGHUP, init);
 
        for (;;) {
 
        for (;;) {
-               int domain, nfds, readfds = defreadfds;
+               int nfds, readfds = FDMASK(funix) | inetm | klogm;
 
 
-               nfds = select(20, &readfds, 0, 0, 0);
+               errno = 0;
+               dprintf("readfds = %#x\n", readfds, funix, finet, fklog);
+               nfds = select(20, (fd_set *) &readfds, (fd_set *) NULL,
+                                 (fd_set *) NULL, (struct timeval *) NULL);
+               dprintf("got a message (%d, %#x)\n", nfds, readfds);
                if (nfds == 0)
                        continue;
                if (nfds < 0) {
                if (nfds == 0)
                        continue;
                if (nfds < 0) {
-                       if (errno == EINTR)
-                               continue;
-                       logerror("select");
+                       if (errno != EINTR)
+                               logerror("select");
                        continue;
                }
                        continue;
                }
-               if (readfds & (1 << funix)) {
-                       domain = AF_UNIX;
+               if (readfds & klogm) {
+                       i = read(fklog, line, sizeof(line) - 1);
+                       if (i > 0) {
+                               line[i] = '\0';
+                               printsys(line);
+                       } else if (i < 0 && errno != EINTR) {
+                               logerror("klog");
+                               fklog = -1;
+                               klogm = 0;
+                       }
+               }
+               if (readfds & FDMASK(funix)) {
                        len = sizeof fromunix;
                        len = sizeof fromunix;
-                       i = recvfrom(funix, line, MAXLINE, 0, &fromunix, &len);
-               } else if (readfds & (1 << finet)) {
-                       domain = AF_INET;
+                       i = recvfrom(funix, line, MAXLINE, 0,
+                                    (struct sockaddr *) &fromunix, &len);
+                       if (i > 0) {
+                               line[i] = '\0';
+                               printline(LocalHostName, line);
+                       } else if (i < 0 && errno != EINTR)
+                               logerror("recvfrom unix");
+               }
+               if (readfds & inetm) {
                        len = sizeof frominet;
                        i = recvfrom(finet, line, MAXLINE, 0, &frominet, &len);
                        len = sizeof frominet;
                        i = recvfrom(finet, line, MAXLINE, 0, &frominet, &len);
+                       if (i > 0) {
+                               extern char *cvthname();
+
+                               line[i] = '\0';
+                               printline(cvthname(&frominet), line);
+                       } else if (i < 0 && errno != EINTR)
+                               logerror("recvfrom inet");
+               } 
+       }
+}
+
+usage()
+{
+       fprintf(stderr, "usage: syslogd [-d] [-ppath] [-fconffile]\n");
+       exit(1);
+}
+
+untty()
+{
+       int i;
+
+       if (!Debug) {
+               i = open("/dev/tty", O_RDWR);
+               if (i >= 0) {
+                       (void) ioctl(i, (int) TIOCNOTTY, (char *)0);
+                       (void) close(i);
                }
                }
-               if (i < 0) {
-                       if (errno == EINTR)
-                               continue;
-                       logerror("recvfrom");
-                       continue;
-               }
-               if (domain == AF_INET && !chkhost(&frominet))
-                       continue;
-               line[i] = '\0';
-               printline(domain == AF_UNIX, line);
        }
 }
 
        }
 }
 
@@ -243,268 +371,283 @@ main(argc, argv)
  * on the appropriate log files.
  */
 
  * on the appropriate log files.
  */
 
-printline(local, msg)
-       int local;
+printline(hname, msg)
+       char *hname;
        char *msg;
 {
        register char *p, *q;
        register int c;
        char *msg;
 {
        register char *p, *q;
        register int c;
-       char line[MAXLINE + sizeof(rhost) + 4];
+       char line[MAXLINE + 1];
        int pri;
 
        /* test for special codes */
        int pri;
 
        /* test for special codes */
-       pri = DefPri;
+       pri = DEFUPRI;
        p = msg;
        p = msg;
-       if (p[0] == '<' && p[2] == '>') {
-               switch (p[1]) {
-               case '*':       /* reset default message priority */
-                       dprintf("default priority = %c\n", p[3]);
-                       c = p[3] - '0';
-                       if ((unsigned) c <= 9)
-                               DefPri = c;
-                       break;
-
-               case '$':       /* reconfigure */
-                       dprintf("reconfigure\n");
-                       init();
-               }
-               p++;
-               pri = *p++ - '0';
-               p++;
-               if ((unsigned) pri > LOG_DEBUG)
-                       pri = DefPri;
+       if (*p == '<') {
+               pri = 0;
+               while (isdigit(*++p))
+                       pri = 10 * pri + (*p - '0');
+               if (*p == '>')
+                       ++p;
+               if (pri <= 0 || pri >= (LOG_NFACILITIES << 3))
+                       pri = DEFUPRI;
        }
 
        }
 
+       /* don't allow users to log kernel messages */
+       if ((pri & LOG_PRIMASK) == LOG_KERN)
+               pri |= LOG_USER;
+
        q = line;
        q = line;
-       if (!local) {
-               sprintf(q, "%s: ", rhost);
-               q += strlen(q);
+#ifdef COMPAT
+       if (CompatMode) {
+               register char *lp = index(p, ':');
+
+               if (lp && lp[1] == ' ' && lp[17] == '-' && lp[18] == '-') {
+                       /*
+                        * Old format message
+                        */
+                       dprintf("mapping <%d> to <%d>\n", pri, CompatCodes[pri]);
+                       pri = CompatCodes[pri];
+                       (void) strncpy(q, lp + 2, 15);
+                       q += 15;
+                       *q++ = ' ';
+                       (void) strncpy(q, p, lp - p + 1);
+                       q += lp - p + 1;
+                       p = lp + 19;
+               }
        }
        }
+#endif COMPAT
+
        while ((c = *p++ & 0177) != '\0' && c != '\n' &&
        while ((c = *p++ & 0177) != '\0' && c != '\n' &&
-           q < &line[sizeof(line) - 2]) {
+           q < &line[sizeof(line) - 1]) {
                if (iscntrl(c)) {
                        *q++ = '^';
                        *q++ = c ^ 0100;
                } else
                        *q++ = c;
        }
                if (iscntrl(c)) {
                        *q++ = '^';
                        *q++ = c ^ 0100;
                } else
                        *q++ = c;
        }
-       *q++ = '\n';
        *q = '\0';
 
        *q = '\0';
 
-       logmsg(pri, line);
+       logmsg(pri, line, hname, 0);
 }
 
 /*
 }
 
 /*
- * Log a message to the appropriate log files, users, etc. based on
- * the priority.
+ * Take a raw input line from /dev/klog, split and format similar to syslog().
  */
 
  */
 
-logmsg(pri, msg)
-       int     pri;
-       char    *msg;
+printsys(msg)
+       char *msg;
 {
 {
+       register char *p, *q;
+       register int c;
        char line[MAXLINE + 1];
        char line[MAXLINE + 1];
-       register struct filed *f;
-       register int l;
-
-       /* log the message to the particular outputs */
-       for (f = Files; f < &Files[NLOGS]; f++) {
-               if (f->f_file < 0 || f->f_pmask < pri)
-                       continue;
-               if (f->f_flags & F_FORW) {
-                       sprintf(line, "<%d>%s", pri, msg);
-                       l = strlen(line);
-                       if (l > MAXLINE)
-                               l = MAXLINE;
-                       if (sendto(f->f_file, line, l, 0,
-                           &f->f_addr, sizeof f->f_addr) != l)
-                               logerror("sendto");
-                       continue;
-               }
-               l = strlen(msg);
-               if (write(f->f_file, msg, l) != l) {
-                       logerror(f->f_name);
-                       (void) close(f->f_file);
-                       f->f_file = -1;
-               }
-               if ((f->f_flags & F_TTY) && write(f->f_file, "\r", 1) != 1) {
-                       logerror(f->f_name);
-                       (void) close(f->f_file);
-                       f->f_file = -1;
+       int pri, flags;
+       char *lp;
+       time_t now;
+
+       (void) time(&now);
+       (void) sprintf(line, "%.15s vmunix: ", ctime(&now) + 4);
+       lp = line + strlen(line);
+       for (p = msg; *p != '\0'; ) {
+               flags = SYNC_FILE;      /* fsync file after write */
+               pri = DEFSPRI;
+               if (*p == '<') {
+                       pri = 0;
+                       while (isdigit(*++p))
+                               pri = 10 * pri + (*p - '0');
+                       if (*p == '>')
+                               ++p;
+                       if (pri <= 0 || pri >= (LOG_NFACILITIES << 3))
+                               pri = DEFSPRI;
+#ifdef COMPAT
+                       else if (CompatMode) {
+                               dprintf("mapping <%d> to <%d>\n", pri, CompatCodes[pri]);
+                               pri = CompatCodes[pri];
+                       }
+#endif COMPAT
+               } else {
+                       /* kernel printf's come out on console */
+                       flags |= IGN_CONS;
                }
                }
+               q = lp;
+               while (*p != '\0' && (c = *p++) != '\n' &&
+                   q < &line[MAXLINE])
+                       *q++ = c;
+               *q = '\0';
+               logmsg(pri, line, LocalHostName, flags);
        }
        }
-
-       /*
-        * Output high priority messages to terminals.
-        */
-       if (pri <= Sumask)
-               wallmsg(pri, msg);
 }
 
 /*
 }
 
 /*
- *  INIT -- Initialize syslog from configuration table
- *
- *     The configuration table consists of a series of lines
- *     broken into two sections by a blank line.  The first
- *     section gives a list of files to log on.  The first
- *     character is a digit which is the priority mask for
- *     that file.  If the second character is an asterisk, then
- *     syslog arranges for something to be printed every fifteen
- *     minutes (even if only a null line), so that crashes and
- *     other events can be localized.  The rest of the line is
- *     the pathname of the log file.  The second section is
- *     a list of user names; these people are all notified
- *     when subalert messages occur (if they are logged on).
- *
- *     The configuration table will be reread by this routine
- *     if a signal 1 occurs; for that reason, it is tricky
- *     about not re-opening files and closing files it will
- *     not be using.
+ * Log a message to the appropriate log files, users, etc. based on
+ * the priority.
  */
 
  */
 
-init()
+logmsg(pri, msg, from, flags)
+       int pri;
+       char *msg, *from;
+       int flags;
 {
 {
-       register int i;
-       register FILE *cf;
+       char line[MAXLINE + 1];
        register struct filed *f;
        register struct filed *f;
-       register char *p;
-       char cline[BUFSIZ];
-       struct servent *sp;
-       struct hostent *hp;
-       int pmask, flags;
-       long now;
+       register int l;
+       int fac, prilev;
+       struct iovec iov[6];
+       register struct iovec *v = iov;
+       int omask;
 
 
-       dprintf("init\n");
+       dprintf("logmsg: pri %o, flags %x, from %s, msg %s\n", pri, flags, from, msg);
 
 
-       /* ignore interrupts during this routine */
-       signal(SIGHUP, SIG_IGN);
+       omask = sigblock(sigmask(SIGALRM)|sigmask(SIGHUP));
 
        /*
 
        /*
-        *  Close all open log files.
+        * Check to see if msg looks non-standard.
         */
         */
-       for (f = Files; f < &Files[NLOGS]; f++) {
-               if (f->f_file < 0)
-                       (void) close(f->f_file);
-               f->f_file = -1;
-       }
-
-       /* open the configuration file */
-       if ((cf = fopen(ConfFile, "r")) == NULL) {
-               dprintf("cannot open %s\n", ConfFile);
-               f = Files;
-               if ((f->f_file = open(ctty, O_WRONLY)) >= 0) {
-                       strncpy(f->f_name, ctty, sizeof(f->f_name)-1);
-                       f->f_pmask = LOG_CRIT;
-                       f->f_flags = F_TTY|F_MARK;
+       if (!(flags & ADDDATE) && (strlen(msg) < 16 ||
+           msg[3] != ' ' || msg[6] != ' ' ||
+           msg[9] != ':' || msg[12] != ':' || msg[15] != ' '))
+               flags |= ADDDATE;
+
+       if ((flags & NOCOPY) == 0) {
+               if (flags & ADDDATE)
+                       flushmsg();
+               else if (!strcmp(msg + 16, PrevLine + 16)) {
+                       /* we found a match, update the time */
+                       (void) strncpy(PrevLine, msg, 15);
+                       PrevCount++;
+                       (void) sigsetmask(omask);
+                       return;
+               } else {
+                       /* new line, save it */
+                       flushmsg();
+                       (void) strcpy(PrevLine, msg);
+                       (void) strcpy(PrevHost, from);
+                       PrevFlags = flags;
+                       PrevPri = pri;
                }
                }
-               return;
        }
 
        }
 
-       /*
-        *  Foreach line in the conf table, open that file.
-        */
-       f = Files;
-       sp = getservbyname("syslogd", "udp");
-       while (fgets(cline, sizeof cline, cf) != NULL) {
-               /* check for end-of-section */
-               if (cline[0] == '\n')
-                       break;
-
-               /* strip off newline character */
-               for (p = cline; *p != '\0'; p++)
-                       if (*p == '\n') {
-                               *p = '\0';
-                               break;
-                       }
+       if (flags & ADDDATE) {
+               time_t now;
+
+               (void) time(&now);
+               v->iov_base = ctime(&now) + 4;
+       } else
+               v->iov_base = msg;
+       v->iov_len = 15;
+       v++;
+       v->iov_base = " ";
+       v->iov_len = 1;
+       v++;
+       v->iov_base = from;
+       v->iov_len = strlen(v->iov_base);
+       v++;
+       v->iov_base = " ";
+       v->iov_len = 1;
+       v++;
+       if (flags & ADDDATE)
+               v->iov_base = msg;
+       else
+               v->iov_base = msg + 16;
+       v->iov_len = strlen(v->iov_base);
+       v++;
 
 
-               dprintf("F: got line '%s'\n", cline);
+       /* extract facility and priority level */
+       fac = (pri & LOG_FACMASK) >> 3;
+       prilev = pri & LOG_PRIMASK;
 
 
-               /* extract priority mask and mark flag */
-               p = cline;
-               flags = 0;
-               pmask = *p++ - '0';
-               if (*p == '*') {
-                       p++;
-                       flags |= F_MARK;
+       /* log the message to the particular outputs */
+       if (!Initialized) {
+               int cfd = open(ctty, O_WRONLY);
+
+               if (cfd >= 0) {
+                       v->iov_base = "\r\n";
+                       v->iov_len = 2;
+                       (void) writev(cfd, iov, 6);
+                       (void) close(cfd);
                }
                }
-
-               if (f >= &Files[NLOGS])
+               return;
+       }
+       for (f = Files; f < &Files[NLOGS]; f++) {
+               /* skip messages that are incorrect priority */
+               if (f->f_pmask[fac] < prilev || f->f_pmask[fac] == NOPRI)
                        continue;
 
                        continue;
 
-               /* mark entry as used and update flags */
-               if (*p == '@') {
-                       hp = gethostbyname(++p);
-                       if (sp != NULL && hp != NULL) {
-                               bzero(&f->f_addr, sizeof f->f_addr);
-                               f->f_addr.sin_family = AF_INET;
-                               f->f_addr.sin_port = sp->s_port;
-                               bcopy(hp->h_addr, (char *) &f->f_addr.sin_addr, hp->h_length);
-                               f->f_file = socket(AF_INET, SOCK_DGRAM, 0);
-                               if (f->f_file < 0) {
-                                       logerror("socket");
-                                       continue;
-                               }
+               dprintf("Logging to %s", TypeNames[f->f_type]);
+               switch (f->f_type) {
+               case F_UNUSED:
+                       dprintf("\n");
+                       break;
+
+               case F_FORW:
+                       dprintf(" %s\n", f->f_un.f_forw.f_hname);
+                       (void) sprintf(line, "<%d>%.15s %s", pri, v[0].iov_base,
+                               v[4].iov_base);
+                       l = strlen(line);
+                       if (l > MAXLINE)
+                               l = MAXLINE;
+                       if (sendto(f->f_file, line, l, 0,
+                           &f->f_un.f_forw.f_addr,
+                           sizeof f->f_un.f_forw.f_addr) != l) {
+                               int e = errno;
+                               (void) close(f->f_file);
+                               f->f_type = F_UNUSED;
+                               errno = e;
+                               logerror("sendto");
                        }
                        }
-                       flags |= F_FORW;
-                       f->f_pmask = pmask;
-                       f->f_flags = flags;
-                       dprintf("Host %s pmask %d flags %o\n", p, pmask, flags);
-                       f++;
-                       continue;
-               }
-               strncpy(f->f_name, p, sizeof(f->f_name)-1);
-               if ((f->f_file = open(p, O_WRONLY|O_APPEND)) < 0) {
-                       logerror(p);
-                       continue;
-               }
-               if (isatty(f->f_file))
-                       flags |= F_TTY;
-               f->f_pmask = pmask;
-               f->f_flags = flags;
-               dprintf("File %s pmask %d flags %o\n", p, pmask, flags);
-               f++;
-       }
+                       break;
 
 
-       /*
-        *  Read the list of users.
-        *
-        *      Anyone in this list is informed directly if s/he
-        *      is logged in when a high priority message comes through.
-        */
-       for (i = 0; i < NSUSERS && fgets(cline, sizeof cline, cf) != NULL; i++) {
-               /* strip off newline */
-               for (p = cline; *p != '\0'; p++)
-                       if (*p == '\n') {
-                               *p = '\0';
+               case F_CONSOLE:
+                       if (flags & IGN_CONS) {
+                               dprintf(" (ignored)\n");
                                break;
                        }
                                break;
                        }
-               dprintf("U: got line '%s'\n", cline);
-               p = cline;
-               if (isdigit(*p)) {
-                       Susers[i].s_pmask = pmask = *p++ - '0';
-                       if (pmask > Sumask)
-                               Sumask = pmask;
-               } else
-                       Susers[i].s_pmask = LOG_SALERT;
-               strncpy(Susers[i].s_name, p, UNAMESZ);
-               dprintf("Suser %s pmask %d\n", p, pmask);
-       }
 
 
-       /* zero the rest of the old superusers */
-       while (i < NSUSERS)
-               Susers[i++].s_name[0] = '\0';
-
-       /* close the configuration file */
-       (void) fclose(cf);
+               case F_TTY:
+               case F_FILE:
+                       dprintf(" %s\n", f->f_un.f_fname);
+                       if (f->f_type != F_FILE) {
+                               v->iov_base = "\r\n";
+                               v->iov_len = 2;
+                       } else {
+                               v->iov_base = "\n";
+                               v->iov_len = 1;
+                       }
+                       if (writev(f->f_file, iov, 6) < 0) {
+                               int e = errno;
+                               (void) close(f->f_file);
+                               /*
+                                * Check for EBADF on TTY's due to vhangup() XXX
+                                */
+                               if (e == EBADF && f->f_type != F_FILE) {
+                                       f->f_file = open(f->f_un.f_fname, O_WRONLY|O_APPEND);
+                                       if (f->f_file < 0) {
+                                               f->f_type = F_UNUSED;
+                                               logerror(f->f_un.f_fname);
+                                       }
+                               } else {
+                                       f->f_type = F_UNUSED;
+                                       errno = e;
+                                       logerror(f->f_un.f_fname);
+                               }
+                       } else if (flags & SYNC_FILE)
+                               (void) fsync(f->f_file);
+                       break;
 
 
-       time(&now);
-       sprintf(cline, "syslog restarted %s", ctime(&now));
-       logmsg(LOG_DEBUG, cline);
+               case F_USERS:
+               case F_WALL:
+                       dprintf("\n");
+                       v->iov_base = "\r\n";
+                       v->iov_len = 2;
+                       wallmsg(f, iov);
+                       break;
+               }
+       }
 
 
-       /* arrange for signal 1 to reconfigure */
-       signal(SIGHUP, init);
+       (void) sigsetmask(omask);
 }
 
 }
 
+
 /*
  *  WALLMSG -- Write a message to the world at large
  *
 /*
  *  WALLMSG -- Write a message to the world at large
  *
@@ -512,130 +655,127 @@ init()
  *     world, or a list of approved users.
  */
 
  *     world, or a list of approved users.
  */
 
-wallmsg(pri, msg)
-       int pri;
-       char *msg;
+wallmsg(f, iov)
+       register struct filed *f;
+       struct iovec *iov;
 {
        register char *p;
        register int i;
 {
        register char *p;
        register int i;
-       int f, uf, flags, len, e;
+       int ttyf, len;
+       FILE *uf;
+       static int reenter = 0;
        struct utmp ut;
        struct utmp ut;
-       long now;
-       char line[MAXLINE + 100];
-       char hbuf[32];
+       time_t now;
+       char greetings[200];
+
+       if (reenter++)
+               return;
 
        /* open the user login file */
 
        /* open the user login file */
-       uf = open("/etc/utmp", 0);
-       if (uf < 0)
+       if ((uf = fopen("/etc/utmp", "r")) == NULL) {
+               logerror("/etc/utmp");
+               reenter = 0;
                return;
                return;
+       }
 
 
-       time(&now);
-       gethostname(hbuf, sizeof hbuf);
-       sprintf(line,
-           "\r\n\007Message from syslogd@%s at %.24s ...\r\n%s\r",
-               hbuf, ctime(&now), msg);
-       len = strlen(line);
+       (void) time(&now);
+       (void) sprintf(greetings,
+           "\r\n\7Message from syslogd@%s at %.24s ...\r\n",
+               iov[2].iov_base, ctime(&now));
+       len = strlen(greetings);
 
        /* scan the user login file */
 
        /* scan the user login file */
-       while (read(uf, &ut, sizeof ut) == sizeof ut) {
+       while (fread((char *) &ut, sizeof ut, 1, uf) == 1) {
                /* is this slot used? */
                if (ut.ut_name[0] == '\0')
                        continue;
 
                /* should we send the message to this user? */
                /* is this slot used? */
                if (ut.ut_name[0] == '\0')
                        continue;
 
                /* should we send the message to this user? */
-               if (pri != LOG_ALERT) {
-                       for (i = 0; i < NSUSERS; i++) {
-                               if (pri > Susers[i].s_pmask)
-                                       continue;
-                               if (strncmp(Susers[i].s_name, ut.ut_name,
-                                   sizeof ut.ut_name) == 0)
-                                       goto prmsg;
+               if (f->f_type == F_USERS) {
+                       for (i = 0; i < MAXUNAMES; i++) {
+                               if (!f->f_un.f_uname[i][0]) {
+                                       i = MAXUNAMES;
+                                       break;
+                               }
+                               if (strncmp(f->f_un.f_uname[i], ut.ut_name,
+                                   UNAMESZ) == 0)
+                                       break;
                        }
                        }
-                       continue;
+                       if (i >= MAXUNAMES)
+                               continue;
                }
                }
-               prmsg:
 
                /* compute the device name */
                p = "/dev/12345678";
                strcpyn(&p[5], ut.ut_line, UNAMESZ);
 
 
                /* compute the device name */
                p = "/dev/12345678";
                strcpyn(&p[5], ut.ut_line, UNAMESZ);
 
-               /* open the terminal */
-               f = open(p, O_WRONLY|O_NDELAY);
-               if (f < 0)
-                       continue;
-               if ((flags = fcntl(f, F_GETFL, 0)) == -1)
-                       continue;
-               if (fcntl(f, F_SETFL, flags | FNDELAY) == -1)
-                       goto oldway;
-               i = write(f, line, len);
-               e = errno;
-               (void) fcntl(f, F_SETFL, flags);
-               if (i == len || e != EWOULDBLOCK) {
-                       (void) close(f);
-                       continue;
-               }
-       oldway:
+               /*
+                * Might as well fork instead of using nonblocking I/O
+                * and doing notty().
+                */
                if (fork() == 0) {
                if (fork() == 0) {
-                       (void) write(f, line, len);
+                       if (f->f_type == F_WALL) {
+                               iov[0].iov_base = greetings;
+                               iov[0].iov_len = len;
+                       }
+                       (void) signal(SIGALRM, SIG_DFL);
+                       (void) alarm(30);
+                       /* open the terminal */
+                       ttyf = open(p, O_WRONLY);
+                       if (ttyf >= 0)
+                               (void) writev(ttyf, iov, 6);
                        exit(0);
                }
                        exit(0);
                }
-               (void) close(f);
+               /* avoid having them all pile up at once */
+               sleep(1);
        }
        }
-
        /* close the user login file */
        /* close the user login file */
-       (void) close(uf);
+       (void) fclose(uf);
+       reenter = 0;
 }
 
 }
 
-/*
- * Make sure every marked file gets written to periodically.
- * Reset the alarm clock to call itself after MarkIntvl minutes.
- */
-domark()
+reapchild()
 {
 {
-       register struct filed *f;
-       struct stat stb;
-       char buf[40];
-       long now;
-
-       dprintf("domark\n");
+       union wait status;
 
 
-       time(&now);
-       for (f = Files; f < &Files[NLOGS]; f++) {
-               if (!(f->f_flags & F_MARK))
-                       continue;
-               if (f->f_file < 0 || fstat(f->f_file, &stb) < 0)
-                       continue;
-               if (stb.st_mtime >= now - MarkIntvl * 60)
-                       continue;
-               sprintf(buf, " --- MARK --- %s", ctime(&now));
-               logmsg(-1, buf);
-       }
-       alarm(MarkIntvl * 60);
+       while (wait3(&status, WNOHANG, (struct rusage *) NULL) > 0)
+               ;
 }
 
 /*
 }
 
 /*
- * Check to see if we should log this message.
+ * Return a printable representation of a host address.
  */
  */
-chkhost(f)
+char *
+cvthname(f)
        struct sockaddr_in *f;
 {
        struct hostent *hp;
        extern char *inet_ntoa();
 
        struct sockaddr_in *f;
 {
        struct hostent *hp;
        extern char *inet_ntoa();
 
-       dprintf("chkhost\n");
+       dprintf("cvthname(%s)\n", inet_ntoa(f->sin_addr));
 
        if (f->sin_family != AF_INET) {
                dprintf("Malformed from address\n");
 
        if (f->sin_family != AF_INET) {
                dprintf("Malformed from address\n");
-               return (0);
+               return ("???");
        }
        hp = gethostbyaddr(&f->sin_addr, sizeof(struct in_addr), f->sin_family);
        if (hp == 0) {
                dprintf("Host name for your address (%s) unknown\n",
                        inet_ntoa(f->sin_addr));
        }
        hp = gethostbyaddr(&f->sin_addr, sizeof(struct in_addr), f->sin_family);
        if (hp == 0) {
                dprintf("Host name for your address (%s) unknown\n",
                        inet_ntoa(f->sin_addr));
-               return (0);
+               return (inet_ntoa(f->sin_addr));
        }
        }
-       strncpy(rhost, hp->h_name, sizeof rhost);
-       return (1);
+       return (hp->h_name);
+}
+
+flushmsg()
+{
+       if (PrevCount == 0)
+               return;
+       if (PrevCount > 1)
+               (void) sprintf(PrevLine+16, "last message repeated %d times", PrevCount);
+       PrevCount = 0;
+       logmsg(PrevPri, PrevLine, PrevHost, PrevFlags|NOCOPY);
+       PrevLine[0] = '\0';
 }
 
 /*
 }
 
 /*
@@ -647,19 +787,323 @@ logerror(type)
        char buf[100];
 
        if (errno == 0)
        char buf[100];
 
        if (errno == 0)
-               sprintf(buf, "syslogd: %s\n", type);
+               (void) sprintf(buf, "syslogd: %s", type);
        else if ((unsigned) errno > sys_nerr)
        else if ((unsigned) errno > sys_nerr)
-               sprintf(buf, "syslogd: %s: error %d\n", type, errno);
+               (void) sprintf(buf, "syslogd: %s: error %d", type, errno);
        else
        else
-               sprintf(buf, "syslogd: %s: %s\n", type, sys_errlist[errno]);
+               (void) sprintf(buf, "syslogd: %s: %s", type, sys_errlist[errno]);
        errno = 0;
        errno = 0;
-       dprintf(buf);
-       logmsg(LOG_ERR, buf);
+       dprintf("%s\n", buf);
+       logmsg(LOG_DAEMON|LOG_ERR, buf, LocalHostName, ADDDATE);
 }
 
 }
 
-die()
+die(sig)
 {
 {
-       logmsg(LOG_DEBUG, "syslog: down\n");
-       (void) unlink(logname);
+       char buf[100];
+
+       dprintf("syslogd: going down on signal %d\n", sig);
+       flushmsg();
+       (void) sprintf(buf, "going down on signal %d", sig);
+       logerror(buf);
+       (void) unlink(LogName);
        exit(0);
 }
        exit(0);
 }
+
+/*
+ *  INIT -- Initialize syslogd from configuration table
+ */
+
+init()
+{
+       register int i;
+       register FILE *cf;
+       register struct filed *f;
+       register char *p;
+       char cline[BUFSIZ];
+
+       dprintf("init\n");
+
+       /* flush any pending output */
+       flushmsg();
+
+       /*
+        *  Close all open log files.
+        */
+       for (f = Files; f < &Files[NLOGS]; f++) {
+               if (f->f_type == F_FILE || f->f_type == F_TTY)
+                       (void) close(f->f_file);
+               f->f_type = F_UNUSED;
+       }
+
+       /* open the configuration file */
+       if ((cf = fopen(ConfFile, "r")) == NULL) {
+               dprintf("cannot open %s\n", ConfFile);
+               cfline("*.ERR\t/dev/console", &Files[0]);
+               cfline("*.PANIC\t*", &Files[1]);
+               return;
+       }
+
+       /*
+        *  Foreach line in the conf table, open that file.
+        */
+       f = Files;
+       while (fgets(cline, sizeof cline, cf) != NULL && f < &Files[NLOGS]) {
+               /* check for end-of-section */
+               if (cline[0] == '\n' || cline[0] == '#')
+                       continue;
+
+               /* strip off newline character */
+               p = index(cline, '\n');
+               if (p)
+                       *p = '\0';
+
+               cfline(cline, f++);
+       }
+
+       /* close the configuration file */
+       (void) fclose(cf);
+
+       Initialized = 1;
+
+       if (Debug) {
+               for (f = Files; f < &Files[NLOGS]; f++) {
+                       for (i = 0; i < LOG_NFACILITIES; i++)
+                               if (f->f_pmask[i] == NOPRI)
+                                       printf("X ");
+                               else
+                                       printf("%d ", f->f_pmask[i]);
+                       printf("%s: ", TypeNames[f->f_type]);
+                       switch (f->f_type) {
+                       case F_FILE:
+                       case F_TTY:
+                       case F_CONSOLE:
+                               printf("%s", f->f_un.f_fname);
+                               break;
+
+                       case F_FORW:
+                               printf("%s", f->f_un.f_forw.f_hname);
+                               break;
+
+                       case F_USERS:
+                               for (i = 0; i < MAXUNAMES && *f->f_un.f_uname[i]; i++)
+                                       printf("%s, ", f->f_un.f_uname[i]);
+                               break;
+                       }
+                       printf("\n");
+               }
+       }
+
+       logmsg(LOG_DAEMON|LOG_INFO, "syslogd: restart", LocalHostName, ADDDATE);
+       dprintf("syslogd: restarted\n");
+}
+
+/*
+ * Crack a configuration file line
+ */
+
+struct code {
+       char    *c_name;
+       int     c_val;
+};
+
+struct code    PriNames[] = {
+       "panic",        LOG_EMERG,
+       "emerg",        LOG_EMERG,
+       "alert",        LOG_ALERT,
+       "crit",         LOG_CRIT,
+       "err",          LOG_ERR,
+       "error",        LOG_ERR,
+       "warn",         LOG_WARNING,
+       "warning",      LOG_WARNING,
+       "notice",       LOG_NOTICE,
+       "info",         LOG_INFO,
+       "debug",        LOG_DEBUG,
+       NULL,           -1
+};
+
+struct code    FacNames[] = {
+       "kern",         LOG_KERN,
+       "user",         LOG_USER,
+       "mail",         LOG_MAIL,
+       "auth",         LOG_AUTH,
+       "security",     LOG_AUTH,
+       "local0",       LOG_LOCAL0,
+       "local1",       LOG_LOCAL1,
+       "local2",       LOG_LOCAL2,
+       "local3",       LOG_LOCAL3,
+       "local4",       LOG_LOCAL4,
+       "local5",       LOG_LOCAL5,
+       "local6",       LOG_LOCAL6,
+       "local7",       LOG_LOCAL7,
+       NULL,           -1
+};
+
+cfline(line, f)
+       char *line;
+       register struct filed *f;
+{
+       register char *p;
+       register char *q;
+       register int i;
+       char *bp;
+       int pri;
+       struct hostent *hp;
+       char buf[MAXLINE];
+
+       dprintf("cfline(%s)\n", line);
+
+       /* clear out file entry */
+       bzero((char *) f, sizeof *f);
+       for (i = 0; i < LOG_NFACILITIES; i++)
+               f->f_pmask[i] = NOPRI;
+
+       /* scan through the list of selectors */
+       for (p = line; *p && *p != '\t';) {
+
+               /* find the end of this facility name list */
+               for (q = p; *q && *q != '\t' && *q++ != '.'; )
+                       continue;
+
+               /* collect priority name */
+               for (bp = buf; *q && !index("\t,;", *q); )
+                       *bp++ = *q++;
+               *bp = '\0';
+
+               /* skip cruft */
+               while (index(", ;", *q))
+                       q++;
+
+               /* decode priority name */
+               pri = decode(buf, PriNames);
+               if (pri < 0) {
+                       char xbuf[200];
+
+                       (void) sprintf(xbuf, "unknown priority name \"%s\"", buf);
+                       logerror(xbuf);
+                       return;
+               }
+
+               /* scan facilities */
+               while (*p && !index("\t.;", *p)) {
+                       int i;
+
+                       for (bp = buf; *p && !index("\t,;.", *p); )
+                               *bp++ = *p++;
+                       *bp = '\0';
+                       if (*buf == '*')
+                               for (i = 0; i < LOG_NFACILITIES; i++)
+                                       f->f_pmask[i] = pri;
+                       else {
+                               i = decode(buf, FacNames);
+                               if (i < 0) {
+                                       char xbuf[200];
+
+                                       (void) sprintf(xbuf, "unknown facility name \"%s\"", buf);
+                                       logerror(xbuf);
+                                       return;
+                               }
+                               f->f_pmask[i >> 3] = pri;
+                       }
+                       while (*p == ',' || *p == ' ')
+                               p++;
+               }
+
+               p = q;
+       }
+
+       /* skip to action part */
+       while (*p == '\t')
+               p++;
+
+       switch (*p)
+       {
+       case '@':
+               if (!InetInuse)
+                       break;
+               (void) strcpy(f->f_un.f_forw.f_hname, ++p);
+               hp = gethostbyname(p);
+               if (hp == NULL) {
+                       char buf[100];
+
+                       (void) sprintf(buf, "unknown host %s", p);
+                       errno = 0;
+                       logerror(buf);
+                       break;
+               }
+               bzero((char *) &f->f_un.f_forw.f_addr,
+                        sizeof f->f_un.f_forw.f_addr);
+               f->f_un.f_forw.f_addr.sin_family = AF_INET;
+               f->f_un.f_forw.f_addr.sin_port = LogPort;
+               bcopy(hp->h_addr, (char *) &f->f_un.f_forw.f_addr.sin_addr, hp->h_length);
+               f->f_file = socket(AF_INET, SOCK_DGRAM, 0);
+               if (f->f_file < 0) {
+                       logerror("socket");
+                       break;
+               }
+               f->f_type = F_FORW;
+               break;
+
+       case '/':
+               (void) strcpy(f->f_un.f_fname, p);
+               if ((f->f_file = open(p, O_WRONLY|O_APPEND)) < 0) {
+                       logerror(p);
+                       break;
+               }
+               if (isatty(f->f_file)) {
+                       f->f_type = F_TTY;
+                       untty();
+               }
+               else
+                       f->f_type = F_FILE;
+               if (strcmp(p, ctty) == 0)
+                       f->f_type = F_CONSOLE;
+               break;
+
+       case '*':
+               f->f_type = F_WALL;
+               break;
+
+       default:
+               for (i = 0; i < MAXUNAMES && *p; i++) {
+                       for (q = p; *q && *q != ','; )
+                               q++;
+                       (void) strncpy(f->f_un.f_uname[i], p, UNAMESZ);
+                       if ((q - p) > UNAMESZ)
+                               f->f_un.f_uname[i][UNAMESZ] = '\0';
+                       else
+                               f->f_un.f_uname[i][q - p] = '\0';
+                       while (*q == ',' || *q == ' ')
+                               q++;
+                       p = q;
+               }
+               f->f_type = F_USERS;
+               break;
+       }
+}
+
+
+/*
+ *  Decode a symbolic name to a numeric value
+ */
+
+decode(name, codetab)
+       char *name;
+       struct code *codetab;
+{
+       register struct code *c;
+       register char *p;
+       char buf[40];
+
+       if (isdigit(*name))
+               return (atoi(name));
+
+       (void) strcpy(buf, name);
+       for (p = buf; *p; p++)
+               if (isupper(*p))
+                       *p = tolower(*p);
+       for (c = codetab; c->c_name; c++)
+               if (!strcmp(buf, c->c_name))
+                       return (c->c_val);
+
+       return (-1);
+}