From a60c9cfa3a8359247f7ca33b3ebc8574c5c7194a Mon Sep 17 00:00:00 2001
From: dt
Date: Sun, 2 May 1999 12:47:09 +0000
Subject: Fight with false newlines in kernel message logs. Output a line into
log only after we read a newline, or we have nothing to read from /dev/klog.
Read /dev/klog in non-blocking mode.
---
usr.sbin/syslogd/syslogd.c | 101 ++++++++++++++++++++++++++++-----------------
1 file changed, 64 insertions(+), 37 deletions(-)
(limited to 'usr.sbin/syslogd')
diff --git a/usr.sbin/syslogd/syslogd.c b/usr.sbin/syslogd/syslogd.c
index 516a912..fd74b54 100644
--- a/usr.sbin/syslogd/syslogd.c
+++ b/usr.sbin/syslogd/syslogd.c
@@ -42,7 +42,7 @@ static const char copyright[] =
static char sccsid[] = "@(#)syslogd.c 8.3 (Berkeley) 4/4/94";
#endif
static const char rcsid[] =
- "$Id: syslogd.c,v 1.46 1998/12/29 23:14:50 cwt Exp $";
+ "$Id: syslogd.c,v 1.47 1999/04/30 12:51:20 des Exp $";
#endif /* not lint */
/*
@@ -252,6 +252,7 @@ int Debug; /* debug flag */
char LocalHostName[MAXHOSTNAMELEN+1]; /* our hostname */
char *LocalDomain; /* our local domain name */
int finet = -1; /* Internet datagram socket */
+int fklog = -1; /* /dev/klog */
int LogPort; /* port number for INET connections */
int Initialized = 0; /* set when we have initialized ourselves */
int MarkInterval = 20 * 60; /* interval between marks in seconds */
@@ -282,6 +283,7 @@ void logmsg __P((int, char *, char *, int));
void printline __P((char *, char *));
void printsys __P((char *));
int p_open __P((char *, pid_t *));
+void readklog __P((void));
void reapchild __P((int));
char *ttymsg __P((struct iovec *, int, char *, int));
static void usage __P((void));
@@ -295,7 +297,7 @@ main(argc, argv)
int argc;
char *argv[];
{
- int ch, i, l, fklog, len;
+ int ch, i, l, len;
struct sockaddr_un sunx, fromunix;
struct sockaddr_in sin, frominet;
FILE *fp;
@@ -416,7 +418,10 @@ main(argc, argv)
}
}
- if ((fklog = open(_PATH_KLOG, O_RDONLY, 0)) < 0)
+ if ((fklog = open(_PATH_KLOG, O_RDONLY, 0)) >= 0)
+ if (fcntl(fklog, F_SETFL, O_NONBLOCK) < 0)
+ fklog = -1;
+ if (fklog < 0)
dprintf("can't open %s (%d)\n", _PATH_KLOG, errno);
/* tuck my process id away */
@@ -474,16 +479,8 @@ main(argc, argv)
continue;
}
/*dprintf("got a message (%d, %#x)\n", nfds, readfds);*/
- if (fklog != -1 && FD_ISSET(fklog, &readfds)) {
- i = read(fklog, line, MAXLINE - 1);
- if (i > 0) {
- line[i] = '\0';
- printsys(line);
- } else if (i < 0 && errno != EINTR) {
- logerror("klog");
- fklog = -1;
- }
- }
+ if (fklog != -1 && FD_ISSET(fklog, &readfds))
+ readklog();
if (finet != -1 && FD_ISSET(finet, &readfds)) {
len = sizeof(frominet);
l = recvfrom(finet, line, MAXLINE, 0,
@@ -581,36 +578,66 @@ printline(hname, msg)
}
/*
- * Take a raw input line from /dev/klog, split and format similar to syslog().
+ * Read /dev/klog while data are available, split into lines.
*/
void
-printsys(msg)
- char *msg;
+readklog()
{
- int pri, flags;
- char *p, *q;
+ char *p, *q, line[MAXLINE + 1];
+ int l, i;
- for (p = msg; *p != '\0'; ) {
- flags = ISKERNEL | SYNC_FILE | ADDDATE; /* fsync after write */
- pri = DEFSPRI;
- if (*p == '<') {
- pri = 0;
- while (isdigit(*++p))
- pri = 10 * pri + (*p - '0');
- if (*p == '>')
- ++p;
- } else {
- /* kernel printf's come out on console */
- flags |= IGN_CONS;
+ l = 0;
+ for (;;) {
+ i = read(fklog, line + l, MAXLINE - 1 - l);
+ if (i > 0)
+ line[i + l] = '\0';
+ else if (i < 0 && errno != EINTR && errno != EAGAIN) {
+ logerror("klog");
+ fklog = -1;
+ break;
+ } else
+ break;
+
+ for (p = line; (q = strchr(p, '\n')) != NULL; p = q + 1) {
+ *q = '\0';
+ printsys(p);
}
- if (pri &~ (LOG_FACMASK|LOG_PRIMASK))
- pri = DEFSPRI;
- for (q = p; *q != '\0' && *q != '\n'; q++);
- if (*q != '\0')
- *q++ = '\0';
- logmsg(pri, p, LocalHostName, flags);
- p = q;
+ l = strlen(p);
+ if (l >= MAXLINE - 1) {
+ printsys(p);
+ l = 0;
+ }
+ if (l > 0)
+ memmove(line, p, l);
}
+ if (l > 0)
+ printsys(line);
+}
+
+/*
+ * Take a raw input line from /dev/klog, format similar to syslog().
+ */
+void
+printsys(p)
+ char *p;
+{
+ int pri, flags;
+
+ flags = ISKERNEL | SYNC_FILE | ADDDATE; /* fsync after write */
+ pri = DEFSPRI;
+ if (*p == '<') {
+ pri = 0;
+ while (isdigit(*++p))
+ pri = 10 * pri + (*p - '0');
+ if (*p == '>')
+ ++p;
+ } else {
+ /* kernel printf's come out on console */
+ flags |= IGN_CONS;
+ }
+ if (pri &~ (LOG_FACMASK|LOG_PRIMASK))
+ pri = DEFSPRI;
+ logmsg(pri, p, LocalHostName, flags);
}
time_t now;
--
cgit v1.1