[9] | 1 | /*
|
---|
| 2 | ** Copyright (c) 1983, 1988
|
---|
| 3 | ** The Regents of the University of California. All rights reserved.
|
---|
| 4 | **
|
---|
| 5 | ** Redistribution and use in source and binary forms, with or without
|
---|
| 6 | ** modification, are permitted provided that the following conditions
|
---|
| 7 | ** are met:
|
---|
| 8 | ** 1. Redistributions of source code must retain the above copyright
|
---|
| 9 | ** notice, this list of conditions and the following disclaimer.
|
---|
| 10 | ** 2. Redistributions in binary form must reproduce the above copyright
|
---|
| 11 | ** notice, this list of conditions and the following disclaimer in the
|
---|
| 12 | ** documentation and/or other materials provided with the distribution.
|
---|
| 13 | ** 3. All advertising materials mentioning features or use of this software
|
---|
| 14 | ** must display the following acknowledgement:
|
---|
| 15 | ** This product includes software developed by the University of
|
---|
| 16 | ** California, Berkeley and its contributors.
|
---|
| 17 | ** 4. Neither the name of the University nor the names of its contributors
|
---|
| 18 | ** may be used to endorse or promote products derived from this software
|
---|
| 19 | ** without specific prior written permission.
|
---|
| 20 | **
|
---|
| 21 | ** THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
|
---|
| 22 | ** ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
---|
| 23 | ** IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
---|
| 24 | ** ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
|
---|
| 25 | ** FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
---|
| 26 | ** DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
|
---|
| 27 | ** OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
---|
| 28 | ** HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
---|
| 29 | ** LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
|
---|
| 30 | ** OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
---|
| 31 | ** SUCH DAMAGE.
|
---|
| 32 | **
|
---|
| 33 | ** #ifndef lint
|
---|
| 34 | ** char copyright2[] =
|
---|
| 35 | ** "@(#) Copyright (c) 1983, 1988 Regents of the University of California.\n\
|
---|
| 36 | ** All rights reserved.\n";
|
---|
| 37 | ** #endif
|
---|
| 38 | **
|
---|
| 39 | ** #ifndef lint
|
---|
| 40 | ** static char sccsid[] = "@(#)syslogd.c 5.27 (Berkeley) 10/10/88";
|
---|
| 41 | ** #endif
|
---|
| 42 | **
|
---|
| 43 | ** -----------------------------------------------------------------------
|
---|
| 44 | **
|
---|
| 45 | ** SYSLOGD -- log system messages
|
---|
| 46 | ** This program implements a system log.
|
---|
| 47 | ** It takes a series of lines and outputs them according to the setup
|
---|
| 48 | ** defined in the configuration file.
|
---|
| 49 | ** Each line may have a priority, signified as "<n>" as
|
---|
| 50 | ** the first characters of the line. If this is
|
---|
| 51 | ** not present, a default priority is used.
|
---|
| 52 | **
|
---|
| 53 | ** To kill syslogd, send a signal 15 (terminate).
|
---|
| 54 | ** A signal 1 (hup) will cause it to reread its configuration file.
|
---|
| 55 | **
|
---|
| 56 | ** Defined Constants:
|
---|
| 57 | ** MAXLINE -- the maximimum line length that can be handled.
|
---|
| 58 | ** MAXSVLINE -- the length of saved messages (for filtering)
|
---|
| 59 | ** DEFUPRI -- the default priority for user messages
|
---|
| 60 | ** DEFSPRI -- the default priority for kernel messages
|
---|
| 61 | **
|
---|
| 62 | ** Author: Eric Allman
|
---|
| 63 | ** extensive changes by Ralph Campbell
|
---|
| 64 | ** more extensive changes by Eric Allman (again)
|
---|
| 65 | ** changes by Steve Lord
|
---|
| 66 | **
|
---|
| 67 | ** Extensive rewriting by G. Falzoni <gfalzoni@inwind.it> for porting to Minix
|
---|
| 68 | **
|
---|
| 69 | ** $Log: syslogd.c,v $
|
---|
| 70 | ** Revision 1.3 2006/04/04 14:22:40 beng
|
---|
| 71 | ** Fix
|
---|
| 72 | **
|
---|
| 73 | ** Revision 1.2 2006/04/04 14:18:16 beng
|
---|
| 74 | ** Make syslogd work, even if it can only open klog and not udp or vice versa
|
---|
| 75 | ** (but not neither)
|
---|
| 76 | **
|
---|
| 77 | ** Revision 1.1 2006/04/03 13:07:42 beng
|
---|
| 78 | ** Kick out usyslogd in favour of syslogd Giovanni's syslogd port
|
---|
| 79 | **
|
---|
| 80 | ** Revision 1.3 2005/09/16 10:10:12 lsodgf0
|
---|
| 81 | ** Rework for Minix 3. Adds kernel logs from /dev/klogd
|
---|
| 82 | **
|
---|
| 83 | ** $Id: syslogd.c,v 1.3 2006/04/04 14:22:40 beng Exp $
|
---|
| 84 | */
|
---|
| 85 |
|
---|
| 86 | #include <sys/types.h>
|
---|
| 87 | #include <stdio.h>
|
---|
| 88 | #include <ctype.h>
|
---|
| 89 | #include <string.h>
|
---|
| 90 | #include <unistd.h>
|
---|
| 91 | #include <stdlib.h>
|
---|
| 92 | #include <signal.h>
|
---|
| 93 | #include <errno.h>
|
---|
| 94 | #include <fcntl.h>
|
---|
| 95 | #include <limits.h>
|
---|
| 96 | #include <time.h>
|
---|
| 97 | #include <sys/ioctl.h>
|
---|
| 98 | #include <sys/select.h>
|
---|
| 99 | #include <sys/wait.h>
|
---|
| 100 | #include <net/netlib.h>
|
---|
| 101 | #include <net/hton.h>
|
---|
| 102 | #include <net/gen/in.h>
|
---|
| 103 | #include <net/gen/udp.h>
|
---|
| 104 | #include <net/gen/udp_io.h>
|
---|
| 105 | #include <net/gen/netdb.h>
|
---|
| 106 |
|
---|
| 107 | #define SYSLOG_NAMES
|
---|
| 108 | #include <syslog.h>
|
---|
| 109 | #define KLOGD 1
|
---|
| 110 | /** Define following values to your requirements **/
|
---|
| 111 | #define MAXLINE 512 /* maximum line length */
|
---|
| 112 | #define MAXSVLINE 256 /* maximum saved line length */
|
---|
| 113 |
|
---|
| 114 | #define DEFUPRI (LOG_USER|LOG_NOTICE)
|
---|
| 115 | #define DEFSPRI (LOG_KERN|LOG_CRIT)
|
---|
| 116 |
|
---|
| 117 | /* Flags to logmsg() */
|
---|
| 118 | #define IGN_CONS 0x001 /* don't print on console */
|
---|
| 119 | #define SYNC_FILE 0x002 /* do fsync on file after printing */
|
---|
| 120 | #define ADDDATE 0x004 /* add a date to the message */
|
---|
| 121 | #define MARK 0x008 /* this message is a mark */
|
---|
| 122 |
|
---|
| 123 | #define CTTY "/dev/log" /* Minix log device (console) */
|
---|
| 124 |
|
---|
| 125 | #define dprintf if(DbgOpt!=0)printf
|
---|
| 126 | #if debug == 0
|
---|
| 127 | #define DEBUG(statement)
|
---|
| 128 | #else
|
---|
| 129 | #define DEBUG(statement) statement
|
---|
| 130 | #endif
|
---|
| 131 | #if !defined PIDFILE
|
---|
| 132 | #define PIDFILE "/var/run/syslogd.pid"
|
---|
| 133 | #endif
|
---|
| 134 |
|
---|
| 135 | #define UNAMESZ 8 /* length of a login name */
|
---|
| 136 | #define MAXUNAMES 20 /* maximum number of user names */
|
---|
| 137 | #define MAXFNAME 200 /* max file pathname length */
|
---|
| 138 | #define MAXHOSTNAMELEN 64 /* max length of FQDN host name */
|
---|
| 139 |
|
---|
| 140 | /* Intervals at which we flush out "message repeated" messages,
|
---|
| 141 | * in seconds after previous message is logged. After each flush,
|
---|
| 142 | * we move to the next interval until we reach the largest. */
|
---|
| 143 | #define TIMERINTVL 30 /* interval for checking flush, mark */
|
---|
| 144 | #define INTERVAL1 30
|
---|
| 145 | #define INTERVAL2 60
|
---|
| 146 | #define MAXREPEAT ((sizeof(repeatinterval)/sizeof(repeatinterval[0]))-1)
|
---|
| 147 | #define REPEATTIME(f) ((f)->f_time+repeatinterval[(f)->f_repeatcount])
|
---|
| 148 | #define BACKOFF(f) {if(++(f)->f_repeatcount>MAXREPEAT)(f)->f_repeatcount=MAXREPEAT;}
|
---|
| 149 |
|
---|
| 150 | /* Values for f_type */
|
---|
| 151 | #define F_UNUSED 0 /* unused entry */
|
---|
| 152 | #define F_FILE 1 /* regular file */
|
---|
| 153 | #define F_TTY 2 /* terminal */
|
---|
| 154 | #define F_CONSOLE 3 /* console terminal */
|
---|
| 155 | #define F_FORW 4 /* remote machine */
|
---|
| 156 | #define F_USERS 5 /* list of users */
|
---|
| 157 | #define F_WALL 6 /* everyone logged on */
|
---|
| 158 |
|
---|
| 159 | #define max(a,b) ((a)>=(b)?(a):(b))
|
---|
| 160 |
|
---|
| 161 | /* This structure represents the files that will have log copies printed */
|
---|
| 162 | struct filed {
|
---|
| 163 | struct filed *f_next; /* next in linked list */
|
---|
| 164 | short f_type; /* entry type, see below */
|
---|
| 165 | short f_file; /* file descriptor */
|
---|
| 166 | time_t f_time; /* time this was last written */
|
---|
| 167 | char f_pmask[LOG_NFACILITIES + 1]; /* priority mask */
|
---|
| 168 | union {
|
---|
| 169 | char f_uname[MAXUNAMES][UNAMESZ + 1];
|
---|
| 170 | char f_fname[MAXFNAME];
|
---|
| 171 | } f_un;
|
---|
| 172 | char f_prevline[MAXSVLINE]; /* last message logged */
|
---|
| 173 | char f_lasttime[16]; /* time of last occurrence */
|
---|
| 174 | char f_prevhost[MAXHOSTNAMELEN + 1]; /* host from which recd. */
|
---|
| 175 | int f_prevpri; /* pri of f_prevline */
|
---|
| 176 | int f_prevlen; /* length of f_prevline */
|
---|
| 177 | int f_prevcount; /* repetition cnt of prevline */
|
---|
| 178 | int f_repeatcount; /* number of "repeated" msgs */
|
---|
| 179 | int f_flags; /* store some additional flags */
|
---|
| 180 | };
|
---|
| 181 |
|
---|
| 182 | static const char *const TypeNames[] =
|
---|
| 183 | {
|
---|
| 184 | "UNUSED", "FILE", "TTY", "CONSOLE", "FORW", "USERS", "WALL", NULL,
|
---|
| 185 | };
|
---|
| 186 |
|
---|
| 187 | static struct filed *Files = NULL;
|
---|
| 188 | static struct filed consfile;
|
---|
| 189 | static int DbgOpt = 0; /* debug flag */
|
---|
| 190 | static char LocalHostName[MAXHOSTNAMELEN + 1]; /* our hostname */
|
---|
| 191 | static int Initialized = 0; /* set when we have initialized ourselves */
|
---|
| 192 | static int MarkInterval = 20 * 60; /* interval between marks in seconds */
|
---|
| 193 | static int MarkSeq = 0; /* mark sequence number */
|
---|
| 194 | static time_t now;
|
---|
| 195 |
|
---|
| 196 | static const char *ConfFile = "/etc/syslog.conf";
|
---|
| 197 | static const char *PidFile = PIDFILE; /* "/var/run/syslogd.pid" */
|
---|
| 198 | static const char ctty[] = CTTY;
|
---|
| 199 |
|
---|
| 200 | static const char ProgName[] = "syslogd:";
|
---|
| 201 | static const char version[] = "1.3 (Minix)";
|
---|
| 202 | static const char usage[] =
|
---|
| 203 | /* */ "usage:\tsyslogd [-d] [-m markinterval] [-f conf-file]\n"
|
---|
| 204 | "\t\t[-p listeningport] [-v] [-?]\n" ;
|
---|
| 205 | static const int repeatinterval[] =
|
---|
| 206 | /* */ {INTERVAL1, INTERVAL2,}; /* # of secs before flush */
|
---|
| 207 |
|
---|
| 208 | /*
|
---|
| 209 | ** Name: void wallmsg(struct filed *fLog, char *message);
|
---|
| 210 | ** Function: Write the specified message to either the entire
|
---|
| 211 | ** world, or a list of approved users.
|
---|
| 212 | */
|
---|
| 213 | void wallmsg(struct filed * fLog, char *message)
|
---|
| 214 | {
|
---|
| 215 |
|
---|
| 216 | return;
|
---|
| 217 | }
|
---|
| 218 |
|
---|
| 219 | /*
|
---|
| 220 | ** Name: void fprintlog(struct filed *fLog, int flags, char *message);
|
---|
| 221 | ** Function:
|
---|
| 222 | */
|
---|
| 223 | void fprintlog(struct filed * fLog, int flags, char *message)
|
---|
| 224 | {
|
---|
| 225 | int len;
|
---|
| 226 | char line[MAXLINE + 1];
|
---|
| 227 | char repbuf[80];
|
---|
| 228 |
|
---|
| 229 | if (message == NULL) {
|
---|
| 230 | if (fLog->f_prevcount > 1) {
|
---|
| 231 | sprintf(repbuf, "last message repeated %d times", fLog->f_prevcount);
|
---|
| 232 | message = repbuf;
|
---|
| 233 | } else
|
---|
| 234 | message = fLog->f_prevline;
|
---|
| 235 | }
|
---|
| 236 | sprintf(line, "%s %s %s", fLog->f_lasttime, fLog->f_prevhost, message);
|
---|
| 237 | DEBUG(dprintf("Logging to %s", TypeNames[fLog->f_type]);)
|
---|
| 238 | fLog->f_time = now;
|
---|
| 239 | switch (fLog->f_type) {
|
---|
| 240 | case F_UNUSED: /* */
|
---|
| 241 | DEBUG(dprintf("\n");)
|
---|
| 242 | break;
|
---|
| 243 | case F_CONSOLE:
|
---|
| 244 | if (flags & IGN_CONS) {
|
---|
| 245 | case F_FORW: /* */
|
---|
| 246 | DEBUG(dprintf(" (ignored)\n");)
|
---|
| 247 | break;
|
---|
| 248 | } /* else Fall Through */
|
---|
| 249 | case F_TTY:
|
---|
| 250 | case F_FILE:
|
---|
| 251 | DEBUG(dprintf(" %s\n", fLog->f_un.f_fname);)
|
---|
| 252 | strcat(line, fLog->f_type != F_FILE ? "\r\n" : "\n");
|
---|
| 253 | len = strlen(line);
|
---|
| 254 | if (write(fLog->f_file, line, len) != len) {
|
---|
| 255 | /* Handle errors */ ;
|
---|
| 256 | } else if (flags & SYNC_FILE)
|
---|
| 257 | sync();
|
---|
| 258 | break;
|
---|
| 259 | case F_USERS:
|
---|
| 260 | case F_WALL:
|
---|
| 261 | DEBUG(dprintf("\n");)
|
---|
| 262 | strcat(line, "\r\n");
|
---|
| 263 | wallmsg(fLog, line);
|
---|
| 264 | break;
|
---|
| 265 | }
|
---|
| 266 | fLog->f_prevcount = 0;
|
---|
| 267 | return;
|
---|
| 268 | }
|
---|
| 269 |
|
---|
| 270 | /*
|
---|
| 271 | ** Name: void logmsg(int pri, char *msg, char *from, int flags);
|
---|
| 272 | ** Function: Log a message to the appropriate log files, users, etc.
|
---|
| 273 | ** based on the priority.
|
---|
| 274 | */
|
---|
| 275 | void logmsg(int pri, char *msg, char *from, int flags)
|
---|
| 276 | {
|
---|
| 277 | struct filed *f;
|
---|
| 278 | int fac, prilev;
|
---|
| 279 | int omask, msglen;
|
---|
| 280 | char *timestamp;
|
---|
| 281 |
|
---|
| 282 | DEBUG(dprintf("logmsg: pri %o, flags %x, from %s, msg %s\n", pri, flags, from, msg);)
|
---|
| 283 | /*
|
---|
| 284 | omask = sigblock(__sigmask(SIGHUP) | __sigmask(SIGALRM));
|
---|
| 285 | */
|
---|
| 286 | /* Check to see if msg looks non-standard. */
|
---|
| 287 | msglen = strlen(msg);
|
---|
| 288 | if (msglen < 16 || msg[3] != ' ' || msg[6] != ' ' ||
|
---|
| 289 | msg[9] != ':' || msg[12] != ':' || msg[15] != ' ')
|
---|
| 290 | flags |= ADDDATE;
|
---|
| 291 |
|
---|
| 292 | time(&now);
|
---|
| 293 | if (flags & ADDDATE)
|
---|
| 294 | timestamp = ctime(&now) + 4;
|
---|
| 295 | else {
|
---|
| 296 | timestamp = msg;
|
---|
| 297 | msg += 16;
|
---|
| 298 | msglen -= 16;
|
---|
| 299 | }
|
---|
| 300 |
|
---|
| 301 | /* Extract facility and priority level */
|
---|
| 302 | fac = (flags & MARK) ? LOG_NFACILITIES : LOG_FAC(pri);
|
---|
| 303 | prilev = LOG_PRI(pri);
|
---|
| 304 |
|
---|
| 305 | /* Log the message to the particular outputs */
|
---|
| 306 | if (!Initialized) {
|
---|
| 307 | /* Not yet initialized. Every message goes to console */
|
---|
| 308 | f = &consfile;
|
---|
| 309 | f->f_file = open(ctty, O_WRONLY | O_NOCTTY);
|
---|
| 310 | if (f->f_file >= 0) {
|
---|
| 311 | if (!DbgOpt) setsid();
|
---|
| 312 | fprintlog(f, flags, msg);
|
---|
| 313 | close(f->f_file);
|
---|
| 314 | }
|
---|
| 315 | } else {
|
---|
| 316 | for (f = Files; f; f = f->f_next) {
|
---|
| 317 |
|
---|
| 318 | /* Skip messages that are incorrect priority */
|
---|
| 319 | if (f->f_pmask[fac] < prilev || f->f_pmask[fac] == INTERNAL_NOPRI)
|
---|
| 320 | continue;
|
---|
| 321 |
|
---|
| 322 | if (f->f_type == F_CONSOLE && (flags & IGN_CONS)) continue;
|
---|
| 323 |
|
---|
| 324 | /* Don't output marks to recently written files */
|
---|
| 325 | if ((flags & MARK) && (now - f->f_time) < MarkInterval / 2)
|
---|
| 326 | continue;
|
---|
| 327 |
|
---|
| 328 | /* Suppress duplicate lines to this file */
|
---|
| 329 | if ((flags & MARK) == 0 && msglen == f->f_prevlen &&
|
---|
| 330 | !strcmp(msg, f->f_prevline) &&
|
---|
| 331 | !strcmp(from, f->f_prevhost)) {
|
---|
| 332 | strncpy(f->f_lasttime, timestamp, 15);
|
---|
| 333 | f->f_prevcount += 1;
|
---|
| 334 | DEBUG(dprintf("msg repeated %d times, %ld sec of %d\n",
|
---|
| 335 | f->f_prevcount, now - f->f_time,
|
---|
| 336 | repeatinterval[f->f_repeatcount]);)
|
---|
| 337 | /* If domark would have logged this by now,
|
---|
| 338 | * flush it now (so we don't hold isolated
|
---|
| 339 | * messages), but back off so we'll flush
|
---|
| 340 | * less often in the future. */
|
---|
| 341 | if (now > REPEATTIME(f)) {
|
---|
| 342 | fprintlog(f, flags, (char *) NULL);
|
---|
| 343 | BACKOFF(f);
|
---|
| 344 | }
|
---|
| 345 | } else {
|
---|
| 346 | /* New line, save it */
|
---|
| 347 | if (f->f_prevcount) fprintlog(f, 0, (char *) NULL);
|
---|
| 348 | f->f_repeatcount = 0;
|
---|
| 349 | strncpy(f->f_lasttime, timestamp, 15);
|
---|
| 350 | strncpy(f->f_prevhost, from, sizeof(f->f_prevhost));
|
---|
| 351 | if (msglen < MAXSVLINE) {
|
---|
| 352 | f->f_prevlen = msglen;
|
---|
| 353 | f->f_prevpri = pri;
|
---|
| 354 | strcpy(f->f_prevline, msg);
|
---|
| 355 | fprintlog(f, flags, (char *) NULL);
|
---|
| 356 | } else {
|
---|
| 357 | f->f_prevline[0] = 0;
|
---|
| 358 | f->f_prevlen = 0;
|
---|
| 359 | fprintlog(f, flags, msg);
|
---|
| 360 | }
|
---|
| 361 | }
|
---|
| 362 | }
|
---|
| 363 | }
|
---|
| 364 |
|
---|
| 365 | /*
|
---|
| 366 | sigsetmask(omask);
|
---|
| 367 | */
|
---|
| 368 | return;
|
---|
| 369 | }
|
---|
| 370 |
|
---|
| 371 | /*
|
---|
| 372 | ** Name: void logerror(char *type);
|
---|
| 373 | ** Function: Prints syslogd errors in some place.
|
---|
| 374 | */
|
---|
| 375 | void logerror(char *type)
|
---|
| 376 | {
|
---|
| 377 | char buf[100];
|
---|
| 378 |
|
---|
| 379 | if (errno == 0) sprintf(buf, "%s %s", ProgName, type);
|
---|
| 380 |
|
---|
| 381 | else if (errno >= _NERROR)
|
---|
| 382 | sprintf(buf, "%s %s - error %d", ProgName, type, errno);
|
---|
| 383 |
|
---|
| 384 | else
|
---|
| 385 | sprintf(buf, "%s %s - %s", ProgName, type, strerror(errno));
|
---|
| 386 |
|
---|
| 387 | errno = 0;
|
---|
| 388 | dprintf("%s\n", buf);
|
---|
| 389 | logmsg(LOG_SYSLOG | LOG_ERR, buf, LocalHostName, ADDDATE);
|
---|
| 390 | return;
|
---|
| 391 | }
|
---|
| 392 |
|
---|
| 393 | /*
|
---|
| 394 | ** Name: void die(int sig);
|
---|
| 395 | ** Function: Signal handler for kill signals.
|
---|
| 396 | */
|
---|
| 397 | void die(int sig)
|
---|
| 398 | {
|
---|
| 399 | struct filed *f;
|
---|
| 400 | char buf[100];
|
---|
| 401 |
|
---|
| 402 | for (f = Files; f != NULL; f = f->f_next) {
|
---|
| 403 | /* Flush any pending output */
|
---|
| 404 | if (f->f_prevcount) fprintlog(f, 0, NULL);
|
---|
| 405 | }
|
---|
| 406 | if (sig >= 0) {
|
---|
| 407 | DEBUG(dprintf("%s exiting on signal %d\n", ProgName, sig);)
|
---|
| 408 | sprintf(buf, "exiting on signal %d", sig);
|
---|
| 409 | errno = 0;
|
---|
| 410 | logerror(buf);
|
---|
| 411 | }
|
---|
| 412 | unlink(PidFile);
|
---|
| 413 | exit(sig == (-1) ? EXIT_FAILURE : EXIT_SUCCESS);
|
---|
| 414 | }
|
---|
| 415 |
|
---|
| 416 | /*
|
---|
| 417 | ** Name: void domark(int sig);
|
---|
| 418 | ** Function: Signal handler for alarm.
|
---|
| 419 | ** Used for messages filtering and mark facility.
|
---|
| 420 | */
|
---|
| 421 | void domark(int sig)
|
---|
| 422 | {
|
---|
| 423 | struct filed *f;
|
---|
| 424 |
|
---|
| 425 | now = time(NULL);
|
---|
| 426 | MarkSeq += TIMERINTVL;
|
---|
| 427 | if (MarkSeq >= MarkInterval) {
|
---|
| 428 | logmsg(LOG_INFO, "-- MARK --", LocalHostName, ADDDATE | MARK);
|
---|
| 429 | MarkSeq = 0;
|
---|
| 430 | }
|
---|
| 431 | for (f = Files; f; f = f->f_next) {
|
---|
| 432 | if (f->f_prevcount && now >= REPEATTIME(f)) {
|
---|
| 433 | DEBUG(dprintf("flush %s: repeated %d times, %d sec.\n",
|
---|
| 434 | TypeNames[f->f_type], f->f_prevcount,
|
---|
| 435 | repeatinterval[f->f_repeatcount]);)
|
---|
| 436 | fprintlog(f, 0, NULL);
|
---|
| 437 | BACKOFF(f);
|
---|
| 438 | }
|
---|
| 439 | }
|
---|
| 440 | signal(SIGALRM, domark);
|
---|
| 441 | alarm(TIMERINTVL);
|
---|
| 442 | return;
|
---|
| 443 | }
|
---|
| 444 |
|
---|
| 445 | /*
|
---|
| 446 | ** Name: int decode(char *name, struct _code *codetab);
|
---|
| 447 | ** Function: Decode a symbolic name to a numeric value
|
---|
| 448 | */
|
---|
| 449 | int decode(char *name, const struct _code *codetab)
|
---|
| 450 | {
|
---|
| 451 | const struct _code *c;
|
---|
| 452 | char *p;
|
---|
| 453 | char buf[40];
|
---|
| 454 |
|
---|
| 455 | DEBUG(dprintf("symbolic name: %s", name);)
|
---|
| 456 | if (isdigit(*name)) return (atoi(name));
|
---|
| 457 |
|
---|
| 458 | strcpy(buf, name);
|
---|
| 459 | for (p = buf; *p; p += 1) {
|
---|
| 460 | if (isupper(*p)) *p = tolower(*p);
|
---|
| 461 | }
|
---|
| 462 | for (c = codetab; c->c_name; c += 1) {
|
---|
| 463 | if (!strcmp(buf, c->c_name)) {
|
---|
| 464 | DEBUG(dprintf(" ==> %d\n", c->c_val);)
|
---|
| 465 | return (c->c_val);
|
---|
| 466 | }
|
---|
| 467 | }
|
---|
| 468 | return (-1);
|
---|
| 469 | }
|
---|
| 470 |
|
---|
| 471 | /*
|
---|
| 472 | ** Name: void cfline(char *line, struct filed *f);
|
---|
| 473 | ** Function: Parse a configuration file line
|
---|
| 474 | */
|
---|
| 475 | void cfline(char *line, struct filed * fLog)
|
---|
| 476 | {
|
---|
| 477 | char *p, *q, *bp;
|
---|
| 478 | int ix, pri;
|
---|
| 479 | char buf[MAXLINE];
|
---|
| 480 | char xbuf[200];
|
---|
| 481 |
|
---|
| 482 | DEBUG(dprintf("cfline(%s)\n", line);)
|
---|
| 483 |
|
---|
| 484 | /* Keep sys_errlist stuff out of logerror messages */
|
---|
| 485 | errno = 0;
|
---|
| 486 |
|
---|
| 487 | /* Clear out file entry */
|
---|
| 488 | memset(fLog, 0, sizeof(*fLog));
|
---|
| 489 | for (ix = 0; ix <= LOG_NFACILITIES; ix += 1) /* */
|
---|
| 490 | fLog->f_pmask[ix] = INTERNAL_NOPRI;
|
---|
| 491 |
|
---|
| 492 | /* Scan through the list of selectors */
|
---|
| 493 | for (p = line; *p && *p != '\t';) {
|
---|
| 494 |
|
---|
| 495 | /* Find the end of this facility name list */
|
---|
| 496 | for (q = p; *q && *q != '\t' && *q++ != '.';) continue;
|
---|
| 497 |
|
---|
| 498 | /* Collect priority name */
|
---|
| 499 | for (bp = buf; *q && !strchr("\t,;", *q);) *bp++ = *q++;
|
---|
| 500 | *bp = '\0';
|
---|
| 501 |
|
---|
| 502 | /* Skip cruft */
|
---|
| 503 | while (strchr(", ;", *q)) q++;
|
---|
| 504 |
|
---|
| 505 | /* Decode priority name */
|
---|
| 506 | pri = decode(buf, PriNames);
|
---|
| 507 | if (pri < 0) {
|
---|
| 508 | sprintf(xbuf, "unknown priority name \"%s\"", buf);
|
---|
| 509 | logerror(xbuf);
|
---|
| 510 | return;
|
---|
| 511 | }
|
---|
| 512 |
|
---|
| 513 | /* Scan facilities */
|
---|
| 514 | while (*p && !strchr("\t.;", *p)) {
|
---|
| 515 | for (bp = buf; *p && !strchr("\t,;.", *p);) *bp++ = *p++;
|
---|
| 516 | *bp = '\0';
|
---|
| 517 | if (*buf == '*') {
|
---|
| 518 | for (ix = 0; ix <= LOG_NFACILITIES; ix += 1)
|
---|
| 519 | if ((fLog->f_pmask[ix] < pri) ||
|
---|
| 520 | (fLog->f_pmask[ix] == INTERNAL_NOPRI)) {
|
---|
| 521 | fLog->f_pmask[ix] = pri;
|
---|
| 522 | }
|
---|
| 523 | } else {
|
---|
| 524 | ix = decode(buf, FacNames);
|
---|
| 525 | if (ix < 0) {
|
---|
| 526 | sprintf(xbuf, "unknown facility name \"%s\"", buf);
|
---|
| 527 | logerror(xbuf);
|
---|
| 528 | return;
|
---|
| 529 | }
|
---|
| 530 | if ((fLog->f_pmask[ix >> 3] < pri) ||
|
---|
| 531 | (fLog->f_pmask[ix >> 3] == INTERNAL_NOPRI)) {
|
---|
| 532 | fLog->f_pmask[ix >> 3] = pri;
|
---|
| 533 | }
|
---|
| 534 | }
|
---|
| 535 | while (*p == ',' || *p == ' ') p++;
|
---|
| 536 | }
|
---|
| 537 | p = q;
|
---|
| 538 | }
|
---|
| 539 |
|
---|
| 540 | /* Skip to action part */
|
---|
| 541 | while (*p == '\t' || *p == ' ') p++;
|
---|
| 542 |
|
---|
| 543 | DEBUG(dprintf("leading char in action: %c\n", *p);)
|
---|
| 544 | switch (*p) {
|
---|
| 545 | case '@': /* Logging to a remote host */
|
---|
| 546 | break; /* NOT IMPLEMENTED */
|
---|
| 547 |
|
---|
| 548 | case '/': /* Logging to a local file/device */
|
---|
| 549 | strcpy(fLog->f_un.f_fname, p);
|
---|
| 550 | DEBUG(dprintf("filename: %s\n", p); /* ASP */)
|
---|
| 551 | if ((fLog->f_file = open(p, O_WRONLY | O_APPEND | O_CREAT | O_NOCTTY, 0644)) < 0) {
|
---|
| 552 | fLog->f_file = F_UNUSED;
|
---|
| 553 | sprintf(xbuf, "unknown file/device (%s)", p);
|
---|
| 554 | logerror(xbuf);
|
---|
| 555 | break;
|
---|
| 556 | }
|
---|
| 557 | if (isatty(fLog->f_file)) {
|
---|
| 558 | if (!DbgOpt) setsid();
|
---|
| 559 | fLog->f_type = F_TTY;
|
---|
| 560 | } else
|
---|
| 561 | fLog->f_type = F_FILE;
|
---|
| 562 | if (strcmp(p, ctty) == 0) fLog->f_type = F_CONSOLE;
|
---|
| 563 | break;
|
---|
| 564 |
|
---|
| 565 | case '*': /* Logging to all users */
|
---|
| 566 | DEBUG(dprintf("write-all\n");)
|
---|
| 567 | fLog->f_type = F_WALL;
|
---|
| 568 | break;
|
---|
| 569 |
|
---|
| 570 | default: /* Logging to selected users */
|
---|
| 571 | DEBUG(dprintf("users: %s\n", p); /* ASP */)
|
---|
| 572 | for (ix = 0; ix < MAXUNAMES && *p; ix += 1) {
|
---|
| 573 | for (q = p; *q && *q != ',';) q += 1;
|
---|
| 574 | strncpy(fLog->f_un.f_uname[ix], p, UNAMESZ);
|
---|
| 575 | if ((q - p) > UNAMESZ)
|
---|
| 576 | fLog->f_un.f_uname[ix][UNAMESZ] = '\0';
|
---|
| 577 | else
|
---|
| 578 | fLog->f_un.f_uname[ix][q - p] = '\0';
|
---|
| 579 | while (*q == ',' || *q == ' ') q++;
|
---|
| 580 | p = q;
|
---|
| 581 | }
|
---|
| 582 | fLog->f_type = F_USERS;
|
---|
| 583 | break;
|
---|
| 584 | }
|
---|
| 585 | }
|
---|
| 586 |
|
---|
| 587 | /*
|
---|
| 588 | ** Name: void printline(char *hname, char *msg);
|
---|
| 589 | ** Function: Takes a raw input line, decodes the message and
|
---|
| 590 | ** prints the message on the appropriate log files.
|
---|
| 591 | */
|
---|
| 592 | void printline(char *hname, char *msg)
|
---|
| 593 | {
|
---|
| 594 | char line[MAXLINE + 1];
|
---|
| 595 | char *p = msg, *q = line;
|
---|
| 596 | int ch, pri = DEFUPRI;
|
---|
| 597 |
|
---|
| 598 | /* Test for special codes */
|
---|
| 599 | if (*p == '<') {
|
---|
| 600 | pri = 0;
|
---|
| 601 | while (isdigit(*++p)) {
|
---|
| 602 | if ((*p - '0') < 8) {
|
---|
| 603 | /* Only 3 bits allocated for pri -- ASP */
|
---|
| 604 | pri = 10 * pri + (*p - '0');
|
---|
| 605 | } else
|
---|
| 606 | pri = 10 * pri + 7;
|
---|
| 607 | }
|
---|
| 608 | if (*p == '>') ++p;
|
---|
| 609 | }
|
---|
| 610 | if (pri & ~(LOG_FACMASK | LOG_PRIMASK)) pri = DEFUPRI;
|
---|
| 611 |
|
---|
| 612 | /* Does not allow users to log kernel messages */
|
---|
| 613 | if (LOG_FAC(pri) == LOG_KERN) pri = LOG_MAKEPRI(LOG_USER, LOG_PRI(pri));
|
---|
| 614 |
|
---|
| 615 | /* Copies message to local buffer, translating control characters */
|
---|
| 616 | while ((ch = *p++ & 0177) != '\0' && q < &line[sizeof(line) - 1]) {
|
---|
| 617 | if (ch == '\n') /* Removes newlines */
|
---|
| 618 | *q++ = ' ';
|
---|
| 619 | else if (iscntrl(ch)) { /* Translates control characters */
|
---|
| 620 | *q++ = '^';
|
---|
| 621 | *q++ = ch ^ 0100;
|
---|
| 622 | } else
|
---|
| 623 | *q++ = ch;
|
---|
| 624 | }
|
---|
| 625 | *q = '\0';
|
---|
| 626 |
|
---|
| 627 | logmsg(pri, line, hname, 0);
|
---|
| 628 | return;
|
---|
| 629 | }
|
---|
| 630 |
|
---|
| 631 | /*
|
---|
| 632 | ** Name: void printkline(char *hname, char *msg);
|
---|
| 633 | ** Function: Takes a raw input line from kernel and
|
---|
| 634 | ** prints the message on the appropriate log files.
|
---|
| 635 | */
|
---|
| 636 | void printkline(char *hname, char *msg)
|
---|
| 637 | {
|
---|
| 638 | char line[MAXLINE + 1];
|
---|
| 639 | char *p = msg, *q = line;
|
---|
| 640 | int ch, pri = DEFUPRI;
|
---|
| 641 |
|
---|
| 642 | /* Copies message to local buffer, adding source program tag */
|
---|
| 643 | sprintf(line, "kernel: %s", msg);
|
---|
| 644 |
|
---|
| 645 | logmsg(LOG_KERN | LOG_INFO, line, hname, ADDDATE);
|
---|
| 646 | return;
|
---|
| 647 | }
|
---|
| 648 |
|
---|
| 649 | /*
|
---|
| 650 | ** Name: void init(int sig);
|
---|
| 651 | ** Function: Initialize syslogd from configuration file.
|
---|
| 652 | ** Used at startup or after a SIGHUP signal.
|
---|
| 653 | */
|
---|
| 654 | void init(int sig)
|
---|
| 655 | {
|
---|
| 656 | int i;
|
---|
| 657 | FILE *cf;
|
---|
| 658 | struct filed *fLog, *next, **nextp;
|
---|
| 659 | char *p;
|
---|
| 660 | char cline[BUFSIZ];
|
---|
| 661 |
|
---|
| 662 | DEBUG(dprintf("init\n");)
|
---|
| 663 |
|
---|
| 664 | /* Close all open log files. */
|
---|
| 665 | Initialized = 0;
|
---|
| 666 | for (fLog = Files; fLog != NULL; fLog = next) {
|
---|
| 667 |
|
---|
| 668 | /* Flush any pending output */
|
---|
| 669 | if (fLog->f_prevcount) fprintlog(fLog, 0, NULL);
|
---|
| 670 |
|
---|
| 671 | switch (fLog->f_type) {
|
---|
| 672 | case F_FILE:
|
---|
| 673 | case F_TTY:
|
---|
| 674 | case F_CONSOLE: close(fLog->f_file); break;
|
---|
| 675 | }
|
---|
| 676 | next = fLog->f_next;
|
---|
| 677 | free((char *) fLog);
|
---|
| 678 | }
|
---|
| 679 | Files = NULL;
|
---|
| 680 | nextp = &Files;
|
---|
| 681 |
|
---|
| 682 | /* Open the configuration file */
|
---|
| 683 | if ((cf = fopen(ConfFile, "r")) != NULL) {
|
---|
| 684 | /* Foreach line in the configuration table, open that file. */
|
---|
| 685 | fLog = NULL;
|
---|
| 686 | while (fgets(cline, sizeof(cline), cf) != NULL) {
|
---|
| 687 | /* Check for end-of-section, comments, strip off
|
---|
| 688 | * trailing spaces and newline character. */
|
---|
| 689 | for (p = cline; isspace(*p); p += 1);
|
---|
| 690 | if (*p == '\0' || *p == '#') continue;
|
---|
| 691 | for (p = strchr(cline, '\0'); isspace(*--p););
|
---|
| 692 | *++p = '\0';
|
---|
| 693 | fLog = (struct filed *) calloc(1, sizeof(*fLog));
|
---|
| 694 | *nextp = fLog;
|
---|
| 695 | nextp = &fLog->f_next;
|
---|
| 696 | cfline(cline, fLog);
|
---|
| 697 | }
|
---|
| 698 |
|
---|
| 699 | /* Close the configuration file */
|
---|
| 700 | fclose(cf);
|
---|
| 701 | Initialized = 1;
|
---|
| 702 | DEBUG (
|
---|
| 703 | if (DbgOpt) {
|
---|
| 704 | for (fLog = Files; fLog; fLog = fLog->f_next) {
|
---|
| 705 | for (i = 0; i <= LOG_NFACILITIES; i += 1)
|
---|
| 706 | if (fLog->f_pmask[i] == INTERNAL_NOPRI)
|
---|
| 707 | printf("X ");
|
---|
| 708 | else
|
---|
| 709 | printf("%d ", fLog->f_pmask[i]);
|
---|
| 710 | printf("%s: ", TypeNames[fLog->f_type]);
|
---|
| 711 | switch (fLog->f_type) {
|
---|
| 712 | case F_FILE:
|
---|
| 713 | case F_TTY:
|
---|
| 714 | case F_CONSOLE:
|
---|
| 715 | printf("%s", fLog->f_un.f_fname);
|
---|
| 716 | break;
|
---|
| 717 | case F_FORW:
|
---|
| 718 | break;
|
---|
| 719 | case F_USERS:
|
---|
| 720 | for (i = 0; i < MAXUNAMES && *fLog->f_un.f_uname[i]; i += 1)
|
---|
| 721 | printf("%s, ", fLog->f_un.f_uname[i]);
|
---|
| 722 | break;
|
---|
| 723 | }
|
---|
| 724 | printf("\n");
|
---|
| 725 | }
|
---|
| 726 | }
|
---|
| 727 | )
|
---|
| 728 | logmsg(LOG_SYSLOG | LOG_INFO, "syslogd: restart", LocalHostName, ADDDATE);
|
---|
| 729 | signal(SIGHUP, init);
|
---|
| 730 | DEBUG(dprintf("%s restarted\n", ProgName);)
|
---|
| 731 | } else {
|
---|
| 732 | DEBUG(dprintf("cannot open %s\n", ConfFile);)
|
---|
| 733 | *nextp = (struct filed *) calloc(1, sizeof(*fLog));
|
---|
| 734 | cfline("*.ERR\t" CTTY, *nextp);
|
---|
| 735 | (*nextp)->f_next = (struct filed *) calloc(1, sizeof(*fLog));
|
---|
| 736 | cfline("*.PANIC\t*", (*nextp)->f_next);
|
---|
| 737 | Initialized = 1;
|
---|
| 738 | }
|
---|
| 739 | return;
|
---|
| 740 | }
|
---|
| 741 |
|
---|
| 742 | /*
|
---|
| 743 | ** Name: void daemonize(char *line);
|
---|
| 744 | ** Function: Clone itself and becomes a daemon releasing unnecessay resources.
|
---|
| 745 | */
|
---|
| 746 | void daemonize(char *line)
|
---|
| 747 | {
|
---|
| 748 | int lfd, len, pid;
|
---|
| 749 |
|
---|
| 750 | if ((lfd = open(PidFile, O_CREAT | O_RDWR, 0600)) > 0) {
|
---|
| 751 | len = read(lfd, line, 10);
|
---|
| 752 | line[len] = '\0';
|
---|
| 753 | close(lfd);
|
---|
| 754 | if ((kill(len = atoi(line), 0) < 0 && errno == ESRCH) || len == 0) {
|
---|
| 755 | if (!DbgOpt) {
|
---|
| 756 | /* Parent ends and child becomes a daemon */
|
---|
| 757 | if ((pid = fork()) > 0) {
|
---|
| 758 | /* Write process id. in pid file */
|
---|
| 759 | lfd = open(PidFile, O_TRUNC | O_WRONLY);
|
---|
| 760 | len = sprintf(line, "%5d", pid);
|
---|
| 761 | write(lfd, line, len);
|
---|
| 762 | close(lfd);
|
---|
| 763 |
|
---|
| 764 | /* Wait for initialization to complete */
|
---|
| 765 | exit(EXIT_SUCCESS);
|
---|
| 766 | }
|
---|
| 767 | sleep(1);
|
---|
| 768 | setsid(); /* Set as session leader */
|
---|
| 769 | chdir("/"); /* Change to the root directory */
|
---|
| 770 | /* Get rid of all open files */
|
---|
| 771 | for (lfd = STDERR_FILENO + 1; lfd < OPEN_MAX; lfd += 1)
|
---|
| 772 | close(lfd);
|
---|
| 773 | }
|
---|
| 774 | } else {
|
---|
| 775 | fprintf(stderr, "\n%s already running\n", ProgName);
|
---|
| 776 | exit(EXIT_FAILURE);
|
---|
| 777 | }
|
---|
| 778 | } else {
|
---|
| 779 | fprintf(stderr, "\n%s can't open %s (%s)\n", ProgName, PidFile, strerror(errno));
|
---|
| 780 | exit(EXIT_FAILURE);
|
---|
| 781 | }
|
---|
| 782 | return;
|
---|
| 783 | }
|
---|
| 784 |
|
---|
| 785 | /*
|
---|
| 786 | ** Name: int main(int argc, char **argv);
|
---|
| 787 | ** Function: Syslog daemon entry point
|
---|
| 788 | */
|
---|
| 789 | int main(int argc, char **argv)
|
---|
| 790 | {
|
---|
| 791 | char *p, *udpdev, *eol;
|
---|
| 792 | int nfd, kfd, len, fdmax;
|
---|
| 793 | int ch, port = 0;
|
---|
| 794 | fd_set fdset;
|
---|
| 795 | struct nwio_udpopt udpopt;
|
---|
| 796 | struct servent *sp;
|
---|
| 797 | char line[MAXLINE + 1];
|
---|
| 798 |
|
---|
| 799 | while ((ch = getopt(argc, argv, "df:m:p:v?")) != EOF) {
|
---|
| 800 | switch ((char) ch) {
|
---|
| 801 | case 'd': /* Debug */
|
---|
| 802 | DbgOpt += 1;
|
---|
| 803 | break;
|
---|
| 804 | case 'f': /* Set configuration file */
|
---|
| 805 | ConfFile = optarg;
|
---|
| 806 | break;
|
---|
| 807 | case 'm': /* Set mark interval */
|
---|
| 808 | MarkInterval = atoi(optarg) * 60;
|
---|
| 809 | break;
|
---|
| 810 | case 'p': /* Set listening port */
|
---|
| 811 | port = atoi(optarg);
|
---|
| 812 | break;
|
---|
| 813 | case 'v': /* Print version */
|
---|
| 814 | fprintf(stderr, "%s version %s\n", ProgName, version);
|
---|
| 815 | return EXIT_FAILURE;
|
---|
| 816 | case '?': /* Help */
|
---|
| 817 | default:
|
---|
| 818 | fprintf(stderr, usage);
|
---|
| 819 | return EXIT_FAILURE;
|
---|
| 820 | }
|
---|
| 821 | }
|
---|
| 822 | if (argc -= optind) {
|
---|
| 823 | fprintf(stderr, usage);
|
---|
| 824 | return EXIT_FAILURE;
|
---|
| 825 | }
|
---|
| 826 |
|
---|
| 827 | daemonize(line);
|
---|
| 828 |
|
---|
| 829 | /* Get the official name of local host. */
|
---|
| 830 | gethostname(LocalHostName, sizeof(LocalHostName) - 1);
|
---|
| 831 | if ((p = strchr(LocalHostName, '.'))) *p = '\0';
|
---|
| 832 |
|
---|
| 833 | udpdev = (p = getenv("UDP_DEVICE")) ? p : UDP_DEVICE;
|
---|
| 834 | sp = getservbyname("syslog", "udp");
|
---|
| 835 |
|
---|
| 836 | signal(SIGTERM, die);
|
---|
| 837 | signal(SIGINT, DbgOpt ? die : SIG_IGN);
|
---|
| 838 | signal(SIGQUIT, DbgOpt ? die : SIG_IGN);
|
---|
| 839 | signal(SIGALRM, domark);
|
---|
| 840 |
|
---|
| 841 | alarm(TIMERINTVL);
|
---|
| 842 |
|
---|
| 843 | /* Open UDP device */
|
---|
| 844 | nfd = open(udpdev, O_NONBLOCK | O_RDONLY);
|
---|
| 845 |
|
---|
| 846 | /* Configures the UDP device */
|
---|
| 847 | udpopt.nwuo_flags = NWUO_SHARED | NWUO_LP_SET | NWUO_EN_LOC |
|
---|
| 848 | NWUO_DI_BROAD | NWUO_RP_SET | NWUO_RA_SET |
|
---|
| 849 | NWUO_RWDATONLY | NWUO_DI_IPOPT;
|
---|
| 850 | udpopt.nwuo_locport = udpopt.nwuo_remport =
|
---|
| 851 | port == 0 ? sp->s_port : htons(port);
|
---|
| 852 | udpopt.nwuo_remaddr = udpopt.nwuo_locaddr = htonl(0x7F000001L);
|
---|
| 853 |
|
---|
| 854 | if(nfd >= 0) {
|
---|
| 855 | while (ioctl(nfd, NWIOSUDPOPT, &udpopt) < 0 ||
|
---|
| 856 | ioctl(nfd, NWIOGUDPOPT, &udpopt) < 0) {
|
---|
| 857 | if (errno == EAGAIN) {
|
---|
| 858 | sleep(1);
|
---|
| 859 | continue;
|
---|
| 860 | }
|
---|
| 861 | logerror("Set/Get UDP options failed");
|
---|
| 862 | return EXIT_FAILURE;
|
---|
| 863 | }
|
---|
| 864 | }
|
---|
| 865 |
|
---|
| 866 | /* Open kernel log device */
|
---|
| 867 | kfd = open("/dev/klog", O_NONBLOCK | O_RDONLY);
|
---|
| 868 |
|
---|
| 869 | if(kfd < 0 && nfd < 0) {
|
---|
| 870 | logerror("open /dev/klog and udp device failed - can't log anything");
|
---|
| 871 | return EXIT_FAILURE;
|
---|
| 872 | }
|
---|
| 873 |
|
---|
| 874 | fdmax = max(nfd, kfd) + 1;
|
---|
| 875 |
|
---|
| 876 | DEBUG(dprintf("off & running....\n");)
|
---|
| 877 |
|
---|
| 878 | init(-1); /* Initilizes log data structures */
|
---|
| 879 |
|
---|
| 880 | for (;;) { /* Main loop */
|
---|
| 881 |
|
---|
| 882 | FD_ZERO(&fdset); /* Setup descriptors for select */
|
---|
| 883 | if(nfd >= 0) FD_SET(nfd, &fdset);
|
---|
| 884 | if(kfd >= 0) FD_SET(kfd, &fdset);
|
---|
| 885 |
|
---|
| 886 | if (select(fdmax, &fdset, NULL, NULL, NULL) <= 0) {
|
---|
| 887 | sleep(1);
|
---|
| 888 | continue;
|
---|
| 889 |
|
---|
| 890 | }
|
---|
| 891 | if (nfd >= 0 && FD_ISSET(nfd, &fdset)) {
|
---|
| 892 |
|
---|
| 893 | /* Read a message from application programs */
|
---|
| 894 | len = read(nfd, line, MAXLINE);
|
---|
| 895 | if (len > 0) { /* Got a message */
|
---|
| 896 | line[len] = '\0';
|
---|
| 897 | dprintf("got a message (%d, %#x)\n", nfd, len);
|
---|
| 898 | printline(LocalHostName, line);
|
---|
| 899 |
|
---|
| 900 | } else if (len < 0) { /* Got an error or signal while reading */
|
---|
| 901 | if (errno != EINTR) /* */
|
---|
| 902 | logerror("Receive error from UDP channel");
|
---|
| 903 |
|
---|
| 904 | } else { /* (len == 0) Channel has been closed */
|
---|
| 905 | logerror("UDP channel has closed");
|
---|
| 906 | close(nfd);
|
---|
| 907 | die(-1);
|
---|
| 908 | }
|
---|
| 909 | }
|
---|
| 910 | if (kfd >= 0 && FD_ISSET(kfd, &fdset)) {
|
---|
| 911 | static char linebuf[5*1024];
|
---|
| 912 |
|
---|
| 913 | /* Read a message from kernel (klog) */
|
---|
| 914 | len = read(kfd, linebuf, sizeof(linebuf)-2);
|
---|
| 915 | dprintf("got a message (%d, %#x)\n", kfd, len);
|
---|
| 916 | for (ch = 0; ch < len; ch += 1)
|
---|
| 917 | if (linebuf[ch] == '\0') linebuf[ch] = ' ';
|
---|
| 918 | if (linebuf[len - 1] == '\n') len -= 1;
|
---|
| 919 | linebuf[len] = '\n';
|
---|
| 920 | linebuf[len + 1] = '\0';
|
---|
| 921 | p = linebuf;
|
---|
| 922 | while(eol = strchr(p, '\n')) {
|
---|
| 923 | *eol = '\0';
|
---|
| 924 | printkline(LocalHostName, p);
|
---|
| 925 | p = eol+1;
|
---|
| 926 | }
|
---|
| 927 | }
|
---|
| 928 | }
|
---|
| 929 | /* Control never gets here */
|
---|
| 930 | }
|
---|
| 931 |
|
---|
| 932 | /** syslogd.c **/
|
---|