1997-06-02 17:21:41 +00:00
|
|
|
/*
|
|
|
|
* Copyright (c) 1983, 1988 Regents of the University of California.
|
|
|
|
* All rights reserved.
|
|
|
|
*
|
|
|
|
* Redistribution and use in source and binary forms are permitted provided
|
|
|
|
* that: (1) source distributions retain this entire copyright notice and
|
|
|
|
* comment, and (2) distributions including binaries display the following
|
|
|
|
* acknowledgement: ``This product includes software developed by the
|
|
|
|
* University of California, Berkeley and its contributors'' in the
|
|
|
|
* documentation or other materials provided with the distribution and in
|
|
|
|
* all advertising materials mentioning features or use of this software.
|
|
|
|
* Neither the name of the University nor the names of its contributors may
|
|
|
|
* be used to endorse or promote products derived from this software without
|
|
|
|
* specific prior written permission.
|
|
|
|
* THIS SOFTWARE IS PROVIDED ``AS IS'' AND WITHOUT ANY EXPRESS OR IMPLIED
|
|
|
|
* WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED WARRANTIES OF
|
|
|
|
* MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#if defined(LIBC_SCCS) && !defined(lint)
|
|
|
|
static char sccsid[] = "@(#)syslog.c 5.28 (Berkeley) 6/27/90";
|
|
|
|
#endif /* LIBC_SCCS and not lint */
|
|
|
|
|
|
|
|
/*
|
|
|
|
* SYSLOG -- print message on log file
|
|
|
|
*
|
|
|
|
* This routine looks a lot like printf, except that it outputs to the
|
|
|
|
* log file instead of the standard output. Also:
|
|
|
|
* adds a timestamp,
|
|
|
|
* prints the module name in front of the message,
|
|
|
|
* has some other formatting types (or will sometime),
|
|
|
|
* adds a newline on the end of the message.
|
|
|
|
*
|
|
|
|
* The output of this routine is intended to be read by syslogd(8).
|
|
|
|
*
|
|
|
|
* Author: Eric Allman
|
|
|
|
* Modified to use UNIX domain IPC by Ralph Campbell
|
|
|
|
*
|
|
|
|
* Sat Dec 11 11:58:31 CST 1993: Dr. Wettstein
|
|
|
|
* Changes to allow compilation with no complains under -Wall.
|
|
|
|
*
|
|
|
|
* Thu Jan 18 11:16:11 CST 1996: Dr. Wettstein
|
|
|
|
* Added patch to close potential security hole. This is the same
|
|
|
|
* patch which was announced in the linux-security mailing lists
|
|
|
|
* and incorporated into the libc version of syslog.c.
|
|
|
|
*
|
. Removed kernel.patch since it is no longer required
. Changed SOCK_STREAM to SOCK_DGRAM in syslog.c
. klogd will only change the console log level if `-c' is supplied
. syslogd.c by Bill Nottingham <notting@redhat.com>
Um, if the directory is invalid, the bind() call in
create_unix_socket fails. Without the return -1, we return the
invalid fd that we just closed. When syslogd then starts
listening, select goes into a hard loop getting EBADF, IIRC.
. klogd.c by Troels Walsted Hansen <troels@thule.no>
I found a bug in the sysklogd package version 1.4. When it
encounters a zero byte in the kernel logging output, the text
parser enters a busy loop. I came upon it when the 3c59x driver
from kernel 2.4.0 started outputting two zero bytes for the product
code of my laptop's 3Com card. It could be argued that the kernel
should never output zero bytes in the logging info, but obviously
that will happen from time to time.
I fear this bug might be considered a security issue as well, if
the kernel can be coerced to output a zero byte somehow, all kernel
logging will stop.
Wolfgang Oertl <Wolfgang.Oertl@uibk.ac.at> had a similar bugfix
idea
. klogd.c by Thomas Roessler <roessler@does-not-exist.org>
Additionally, the patch prevents LogLine from being invoked with a
negative counter as an argument.
2001-03-11 19:35:52 +00:00
|
|
|
* Sun Mar 11 20:23:44 CET 2001: Martin Schulze <joey@infodrom.ffis.de>
|
|
|
|
* Use SOCK_DGRAM for loggin, renables it to work.
|
|
|
|
*
|
2003-08-27 15:56:01 +00:00
|
|
|
* Wed Aug 27 17:48:16 CEST 2003: Martin Schulze <joey@Infodrom.org>
|
|
|
|
* Improved patch by Michael Pomraning <mjp@securepipe.com> to
|
|
|
|
* reconnect klogd to the logger after it went away.
|
1997-06-02 17:21:41 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/socket.h>
|
|
|
|
#include <sys/file.h>
|
|
|
|
#include <sys/signal.h>
|
|
|
|
#include <sys/syslog.h>
|
|
|
|
#if 0
|
|
|
|
#include "syslog.h"
|
|
|
|
#include "pathnames.h"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include <sys/uio.h>
|
|
|
|
#include <sys/wait.h>
|
|
|
|
#include <netdb.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <time.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <errno.h>
|
|
|
|
#include <stdarg.h>
|
|
|
|
#include <paths.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
|
|
|
|
#define _PATH_LOGNAME "/dev/log"
|
|
|
|
|
|
|
|
static int LogFile = -1; /* fd for log */
|
|
|
|
static int connected; /* have done connect */
|
|
|
|
static int LogStat = 0; /* status bits, set by openlog() */
|
|
|
|
static const char *LogTag = "syslog"; /* string to tag the entry with */
|
|
|
|
static int LogFacility = LOG_USER; /* default facility code */
|
|
|
|
|
|
|
|
void
|
|
|
|
syslog(int pri, const char *fmt, ...)
|
|
|
|
{
|
|
|
|
va_list ap;
|
|
|
|
|
|
|
|
va_start(ap, fmt);
|
|
|
|
vsyslog(pri, fmt, ap);
|
|
|
|
va_end(ap);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
vsyslog(pri, fmt, ap)
|
|
|
|
int pri;
|
|
|
|
const char *fmt;
|
|
|
|
va_list ap;
|
|
|
|
{
|
|
|
|
register int cnt;
|
|
|
|
register char *p;
|
|
|
|
time_t now;
|
2003-08-27 15:56:01 +00:00
|
|
|
int fd, saved_errno;
|
|
|
|
int result;
|
1997-06-02 17:21:41 +00:00
|
|
|
char tbuf[2048], fmt_cpy[1024], *stdp = (char *) 0;
|
|
|
|
|
|
|
|
saved_errno = errno;
|
|
|
|
|
|
|
|
/* see if we should just throw out this message */
|
|
|
|
if (!LOG_MASK(LOG_PRI(pri)) || (pri &~ (LOG_PRIMASK|LOG_FACMASK)))
|
|
|
|
return;
|
|
|
|
if (LogFile < 0 || !connected)
|
1999-08-21 10:28:44 +00:00
|
|
|
openlog(LogTag, LogStat | LOG_NDELAY, LogFacility);
|
1997-06-02 17:21:41 +00:00
|
|
|
|
|
|
|
/* set default facility if none specified */
|
|
|
|
if ((pri & LOG_FACMASK) == 0)
|
|
|
|
pri |= LogFacility;
|
|
|
|
|
|
|
|
/* build the message */
|
|
|
|
(void)time(&now);
|
|
|
|
(void)sprintf(tbuf, "<%d>%.15s ", pri, ctime(&now) + 4);
|
|
|
|
for (p = tbuf; *p; ++p);
|
|
|
|
if (LogStat & LOG_PERROR)
|
|
|
|
stdp = p;
|
|
|
|
if (LogTag) {
|
|
|
|
(void)strcpy(p, LogTag);
|
|
|
|
for (; *p; ++p);
|
|
|
|
}
|
|
|
|
if (LogStat & LOG_PID) {
|
|
|
|
(void)sprintf(p, "[%d]", getpid());
|
|
|
|
for (; *p; ++p);
|
|
|
|
}
|
|
|
|
if (LogTag) {
|
|
|
|
*p++ = ':';
|
|
|
|
*p++ = ' ';
|
|
|
|
}
|
|
|
|
|
|
|
|
/* substitute error message for %m */
|
|
|
|
{
|
|
|
|
register char ch, *t1, *t2;
|
|
|
|
char *strerror();
|
|
|
|
|
|
|
|
for (t1 = fmt_cpy;
|
|
|
|
(ch = *fmt) != '\0' && t1<fmt_cpy+sizeof(fmt_cpy);
|
|
|
|
++fmt)
|
|
|
|
if (ch == '%' && fmt[1] == 'm') {
|
|
|
|
++fmt;
|
|
|
|
for (t2 = strerror(saved_errno);
|
|
|
|
(*t1 = *t2++); ++t1);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
*t1++ = ch;
|
|
|
|
*t1 = '\0';
|
|
|
|
}
|
|
|
|
|
|
|
|
(void)vsprintf(p, fmt_cpy, ap);
|
|
|
|
|
|
|
|
cnt = strlen(tbuf);
|
|
|
|
|
|
|
|
/* output to stderr if requested */
|
|
|
|
if (LogStat & LOG_PERROR) {
|
|
|
|
struct iovec iov[2];
|
|
|
|
register struct iovec *v = iov;
|
|
|
|
|
|
|
|
v->iov_base = stdp;
|
|
|
|
v->iov_len = cnt - (stdp - tbuf);
|
|
|
|
++v;
|
|
|
|
v->iov_base = "\n";
|
|
|
|
v->iov_len = 1;
|
|
|
|
(void)writev(2, iov, 2);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* output the message to the local logger */
|
2003-08-27 15:56:01 +00:00
|
|
|
result = write(LogFile, tbuf, cnt + 1);
|
2003-08-27 15:55:04 +00:00
|
|
|
|
2003-08-27 15:56:01 +00:00
|
|
|
if (result == -1
|
|
|
|
&& (errno == ECONNRESET || errno == ENOTCONN || errno == ECONNREFUSED)) {
|
2003-08-27 15:55:04 +00:00
|
|
|
closelog();
|
2003-08-27 15:56:01 +00:00
|
|
|
openlog(LogTag, LogStat | LOG_NDELAY, LogFacility);
|
|
|
|
result = write(LogFile, tbuf, cnt + 1);
|
2003-08-27 15:55:04 +00:00
|
|
|
}
|
|
|
|
|
2003-08-27 15:56:01 +00:00
|
|
|
if (result >= 0 || !(LogStat&LOG_CONS))
|
1997-06-02 17:21:41 +00:00
|
|
|
return;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* output the message to the console; don't worry about
|
|
|
|
* blocking, if console blocks everything will.
|
|
|
|
*/
|
|
|
|
if ((fd = open(_PATH_CONSOLE, O_WRONLY|O_NOCTTY, 0)) < 0)
|
|
|
|
return;
|
|
|
|
(void)strcat(tbuf, "\r\n");
|
|
|
|
cnt += 2;
|
|
|
|
p = index(tbuf, '>') + 1;
|
|
|
|
(void)write(fd, p, cnt - (p - tbuf));
|
|
|
|
(void)close(fd);
|
|
|
|
}
|
|
|
|
|
1998-10-12 16:58:12 +00:00
|
|
|
#ifndef TESTING
|
1997-06-02 17:21:41 +00:00
|
|
|
static struct sockaddr SyslogAddr; /* AF_UNIX address of local logger */
|
1998-10-12 16:58:12 +00:00
|
|
|
#endif
|
1997-06-02 17:21:41 +00:00
|
|
|
/*
|
|
|
|
* OPENLOG -- open system log
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
openlog(ident, logstat, logfac)
|
|
|
|
const char *ident;
|
|
|
|
int logstat, logfac;
|
|
|
|
{
|
|
|
|
if (ident != NULL)
|
|
|
|
LogTag = ident;
|
|
|
|
LogStat = logstat;
|
|
|
|
|
|
|
|
#ifdef ALLOW_KERNEL_LOGGING
|
|
|
|
if ((logfac &~ LOG_FACMASK) == 0)
|
|
|
|
#else
|
|
|
|
if (logfac != 0 && (logfac &~ LOG_FACMASK) == 0)
|
|
|
|
#endif
|
|
|
|
LogFacility = logfac;
|
|
|
|
|
1998-10-12 16:58:12 +00:00
|
|
|
#ifndef TESTING
|
1997-06-02 17:21:41 +00:00
|
|
|
if (LogFile == -1) {
|
|
|
|
SyslogAddr.sa_family = AF_UNIX;
|
|
|
|
strncpy(SyslogAddr.sa_data, _PATH_LOGNAME,
|
|
|
|
sizeof(SyslogAddr.sa_data));
|
|
|
|
if (LogStat & LOG_NDELAY) {
|
. Removed kernel.patch since it is no longer required
. Changed SOCK_STREAM to SOCK_DGRAM in syslog.c
. klogd will only change the console log level if `-c' is supplied
. syslogd.c by Bill Nottingham <notting@redhat.com>
Um, if the directory is invalid, the bind() call in
create_unix_socket fails. Without the return -1, we return the
invalid fd that we just closed. When syslogd then starts
listening, select goes into a hard loop getting EBADF, IIRC.
. klogd.c by Troels Walsted Hansen <troels@thule.no>
I found a bug in the sysklogd package version 1.4. When it
encounters a zero byte in the kernel logging output, the text
parser enters a busy loop. I came upon it when the 3c59x driver
from kernel 2.4.0 started outputting two zero bytes for the product
code of my laptop's 3Com card. It could be argued that the kernel
should never output zero bytes in the logging info, but obviously
that will happen from time to time.
I fear this bug might be considered a security issue as well, if
the kernel can be coerced to output a zero byte somehow, all kernel
logging will stop.
Wolfgang Oertl <Wolfgang.Oertl@uibk.ac.at> had a similar bugfix
idea
. klogd.c by Thomas Roessler <roessler@does-not-exist.org>
Additionally, the patch prevents LogLine from being invoked with a
negative counter as an argument.
2001-03-11 19:35:52 +00:00
|
|
|
LogFile = socket(AF_UNIX, SOCK_DGRAM, 0);
|
1997-06-02 17:21:41 +00:00
|
|
|
/* fcntl(LogFile, F_SETFD, 1); */
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (LogFile != -1 && !connected &&
|
|
|
|
connect(LogFile, &SyslogAddr, sizeof(SyslogAddr.sa_family)+
|
|
|
|
strlen(SyslogAddr.sa_data)) != -1)
|
1998-10-12 16:58:12 +00:00
|
|
|
#else
|
|
|
|
LogFile = fileno(stdout);
|
|
|
|
#endif
|
1997-06-02 17:21:41 +00:00
|
|
|
connected = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* CLOSELOG -- close the system log
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
closelog()
|
|
|
|
{
|
1998-10-12 16:58:12 +00:00
|
|
|
#ifndef TESTING
|
1997-06-02 17:21:41 +00:00
|
|
|
(void) close(LogFile);
|
1998-10-12 16:58:12 +00:00
|
|
|
#endif
|
1997-06-02 17:21:41 +00:00
|
|
|
LogFile = -1;
|
|
|
|
connected = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int LogMask = 0xff; /* mask of priorities to be logged */
|
|
|
|
/*
|
|
|
|
* SETLOGMASK -- set the log mask level
|
|
|
|
*/
|
|
|
|
int
|
|
|
|
setlogmask(pmask)
|
|
|
|
int pmask;
|
|
|
|
{
|
|
|
|
int omask;
|
|
|
|
|
|
|
|
omask = LogMask;
|
|
|
|
if (pmask != 0)
|
|
|
|
LogMask = pmask;
|
|
|
|
return (omask);
|
|
|
|
}
|