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.
This commit is contained in:
Dmitrij Tejblum 1999-05-02 12:47:09 +00:00
parent d4af94988c
commit 823da6884b
1 changed files with 64 additions and 37 deletions

View File

@ -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()
{
char *p, *q, line[MAXLINE + 1];
int l, i;
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);
}
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;
char *p, *q;
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;
}
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;
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;