Annotation of embedaddon/libpdel/sys/alog.c, revision 1.1
1.1 ! misho 1:
! 2: /*
! 3: * Copyright (c) 2001-2002 Packet Design, LLC.
! 4: * All rights reserved.
! 5: *
! 6: * Subject to the following obligations and disclaimer of warranty,
! 7: * use and redistribution of this software, in source or object code
! 8: * forms, with or without modifications are expressly permitted by
! 9: * Packet Design; provided, however, that:
! 10: *
! 11: * (i) Any and all reproductions of the source or object code
! 12: * must include the copyright notice above and the following
! 13: * disclaimer of warranties; and
! 14: * (ii) No rights are granted, in any manner or form, to use
! 15: * Packet Design trademarks, including the mark "PACKET DESIGN"
! 16: * on advertising, endorsements, or otherwise except as such
! 17: * appears in the above copyright notice or in the software.
! 18: *
! 19: * THIS SOFTWARE IS BEING PROVIDED BY PACKET DESIGN "AS IS", AND
! 20: * TO THE MAXIMUM EXTENT PERMITTED BY LAW, PACKET DESIGN MAKES NO
! 21: * REPRESENTATIONS OR WARRANTIES, EXPRESS OR IMPLIED, REGARDING
! 22: * THIS SOFTWARE, INCLUDING WITHOUT LIMITATION, ANY AND ALL IMPLIED
! 23: * WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE,
! 24: * OR NON-INFRINGEMENT. PACKET DESIGN DOES NOT WARRANT, GUARANTEE,
! 25: * OR MAKE ANY REPRESENTATIONS REGARDING THE USE OF, OR THE RESULTS
! 26: * OF THE USE OF THIS SOFTWARE IN TERMS OF ITS CORRECTNESS, ACCURACY,
! 27: * RELIABILITY OR OTHERWISE. IN NO EVENT SHALL PACKET DESIGN BE
! 28: * LIABLE FOR ANY DAMAGES RESULTING FROM OR ARISING OUT OF ANY USE
! 29: * OF THIS SOFTWARE, INCLUDING WITHOUT LIMITATION, ANY DIRECT,
! 30: * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, PUNITIVE, OR CONSEQUENTIAL
! 31: * DAMAGES, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES, LOSS OF
! 32: * USE, DATA OR PROFITS, HOWEVER CAUSED AND UNDER ANY THEORY OF
! 33: * LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
! 34: * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF
! 35: * THE USE OF THIS SOFTWARE, EVEN IF PACKET DESIGN IS ADVISED OF
! 36: * THE POSSIBILITY OF SUCH DAMAGE.
! 37: *
! 38: * Author: Archie Cobbs <archie@freebsd.org>
! 39: */
! 40:
! 41: #include <sys/types.h>
! 42: #include <sys/param.h>
! 43: #include <sys/socket.h>
! 44: #include <sys/uio.h>
! 45: #include <sys/queue.h>
! 46:
! 47: #include <netinet/in.h>
! 48:
! 49: #include <stdlib.h>
! 50: #include <stddef.h>
! 51: #include <stdio.h>
! 52: #include <syslog.h>
! 53: #include <assert.h>
! 54: #include <string.h>
! 55: #include <stdarg.h>
! 56: #include <unistd.h>
! 57: #include <fcntl.h>
! 58: #include <errno.h>
! 59: #include <regex.h>
! 60: #include <pthread.h>
! 61:
! 62: #include "structs/structs.h"
! 63: #include "structs/type/array.h"
! 64: #include "structs/type/int.h"
! 65: #include "structs/type/ip4.h"
! 66: #include "structs/type/pointer.h"
! 67: #include "structs/type/string.h"
! 68: #include "structs/type/struct.h"
! 69: #include "structs/type/time.h"
! 70: #include "sys/alog.h"
! 71: #include "sys/logfile.h"
! 72: #include "util/typed_mem.h"
! 73:
! 74: /* Memory allocation types */
! 75: #define MEM_TYPE "alog"
! 76: #define MEM_TYPE_ENTRY "alog_entry"
! 77: #define MEM_TYPE_HISTORY "alog_history"
! 78: #define MEM_TYPE_CURCHAN "alog_curchan"
! 79:
! 80: /* Last message timeout functions */
! 81: #define LASTMSG_TIMEOUT_INITIAL 5
! 82: #define LASTMSG_TIMEOUT_1(t, d) ((t) + (d) + 1 / 2)
! 83: #define LASTMSG_TIMEOUT_2(t, d) ((t) + (d) + 1 / 2)
! 84: #define LASTMSG_TIMEOUT_3(t) (t)
! 85:
! 86: #define LASTMSG_REPEATED_FMT "last message repeated %u times"
! 87:
! 88: /* Misc defs */
! 89: #define SYSLOG_PORT 514 /* syslogd udp port */
! 90: #define MAX_NAME 64 /* max identifier length */
! 91: #define EST_ENTRY_SIZE 100 /* estimated avg. size per entry */
! 92:
! 93: enum alog_type {
! 94: ALOG_NONE = 0, /* not initialized */
! 95: ALOG_NULL, /* log to nowhere */
! 96: ALOG_STDERR, /* log to stderr */
! 97: ALOG_SYSLOG_LOCAL, /* log to local machine syslog */
! 98: ALOG_SYSLOG_REMOTE /* log to remote machine syslog */
! 99: };
! 100:
! 101: /* Current alog channel (thread-specific variable) */
! 102: struct alog_curchan {
! 103: int channel;
! 104: };
! 105:
! 106: /* Stolen from syslog.h */
! 107: struct nameval {
! 108: const char *name;
! 109: int val;
! 110: };
! 111:
! 112: /* 'Last message' state for a channel */
! 113: struct alog_lastmsg {
! 114: char *msg; /* previously logged message */
! 115: int sev; /* severity for 'msg' */
! 116: time_t when; /* timestamp for 'msg' */
! 117: u_int repeat; /* number of unlogged repeats */
! 118: u_int timeout; /* timeout for logging repeat */
! 119: time_t timer_expiry; /* timer running & expiry */
! 120: };
! 121:
! 122: /* Structure describing one channel */
! 123: struct alog_channel {
! 124: char name[MAX_NAME]; /* local syslog(3) identifier */
! 125: struct logfile *logfile; /* history logfile, or NULL */
! 126: enum alog_type type; /* channel configuration type */
! 127: int sock; /* remote syslog(3) socket */
! 128: int facility; /* channel facility */
! 129: int min_severity; /* min channel severity */
! 130: u_char debug; /* debug enabled flag */
! 131: pthread_mutex_t mutex; /* mutex, if not ALOG_NONE */
! 132: struct alog_lastmsg last; /* 'last message' info */
! 133: };
! 134:
! 135: /*
! 136: * Internal variables
! 137: */
! 138: static struct alog_channel alog_channels[ALOG_MAX_CHANNELS];
! 139: static pthread_key_t alog_current_channel;
! 140:
! 141: /* Stolen from syslog.h */
! 142: static struct nameval prioritynames[] = {
! 143: { "alert", LOG_ALERT, },
! 144: { "crit", LOG_CRIT, },
! 145: { "debug", LOG_DEBUG, },
! 146: { "emerg", LOG_EMERG, },
! 147: { "err", LOG_ERR, },
! 148: { "error", LOG_ERR, }, /* DEPRECATED */
! 149: { "info", LOG_INFO, },
! 150: { "notice", LOG_NOTICE, },
! 151: { "panic", LOG_EMERG, }, /* DEPRECATED */
! 152: { "warn", LOG_WARNING, }, /* DEPRECATED */
! 153: { "warning", LOG_WARNING, },
! 154: { NULL, -1, }
! 155: };
! 156:
! 157: /* Stolen from syslog.h */
! 158: static struct nameval facilitynames[] = {
! 159: { "auth", LOG_AUTH, },
! 160: { "authpriv", LOG_AUTHPRIV, },
! 161: { "cron", LOG_CRON, },
! 162: { "daemon", LOG_DAEMON, },
! 163: { "ftp", LOG_FTP, },
! 164: { "kern", LOG_KERN, },
! 165: { "lpr", LOG_LPR, },
! 166: { "mail", LOG_MAIL, },
! 167: { "news", LOG_NEWS, },
! 168: #ifdef LOG_NTP
! 169: { "ntp", LOG_NTP, },
! 170: #endif
! 171: #ifdef LOG_SECURITY
! 172: { "security", LOG_SECURITY, },
! 173: #endif
! 174: { "syslog", LOG_SYSLOG, },
! 175: { "user", LOG_USER, },
! 176: { "uucp", LOG_UUCP, },
! 177: { "local0", LOG_LOCAL0, },
! 178: { "local1", LOG_LOCAL1, },
! 179: { "local2", LOG_LOCAL2, },
! 180: { "local3", LOG_LOCAL3, },
! 181: { "local4", LOG_LOCAL4, },
! 182: { "local5", LOG_LOCAL5, },
! 183: { "local6", LOG_LOCAL6, },
! 184: { "local7", LOG_LOCAL7, },
! 185: { NULL, -1, }
! 186: };
! 187:
! 188: /*
! 189: * Internal functions
! 190: */
! 191: static void alog_write(struct alog_channel *ch,
! 192: int sev, time_t when, const char *msg);
! 193: static void alog_flush_lastmsg(struct alog_channel *ch, u_int timeout);
! 194: static void alog_last_check(struct alog_channel *ch);
! 195: static int alog_get_channel(void);
! 196: static struct alog_curchan *alog_get_current_channel(int create);
! 197: static void alog_init_current_channel(void);
! 198: static void alog_curchan_destroy(void *arg);
! 199:
! 200: /*
! 201: * Initialize or reconfigure a logging channel.
! 202: */
! 203: int
! 204: alog_configure(int channel, const struct alog_config *conf)
! 205: {
! 206: struct alog_channel *const ch = &alog_channels[channel];
! 207: struct sockaddr_in sin;
! 208: int init_mutex = 0;
! 209: int esave;
! 210: int debug;
! 211:
! 212: /* Sanity check */
! 213: if (channel < 0 || channel >= ALOG_MAX_CHANNELS) {
! 214: errno = EINVAL;
! 215: return (-1);
! 216: }
! 217:
! 218: /* If already initialized, shut it down */
! 219: if (ch->type != ALOG_NONE)
! 220: alog_shutdown(channel);
! 221:
! 222: /* Initialize channel */
! 223: debug = ch->debug;
! 224: memset(ch, 0, sizeof(*ch));
! 225: ch->sock = -1;
! 226: ch->debug = debug;
! 227: if (conf->name != NULL)
! 228: strlcpy(ch->name, conf->name, sizeof(ch->name));
! 229: ch->facility = (ch->debug || conf->facility == NULL) ?
! 230: -1 : alog_facility(conf->facility);
! 231: ch->min_severity = conf->min_severity;
! 232:
! 233: /* Open logfile */
! 234: if (conf->histlen > 0) {
! 235: if ((ch->logfile = logfile_open(conf->path, 0,
! 236: conf->histlen, conf->histlen * EST_ENTRY_SIZE)) == NULL)
! 237: goto fail;
! 238: }
! 239:
! 240: /* Initialize mutex */
! 241: if ((errno = pthread_mutex_init(&ch->mutex, NULL)) != 0)
! 242: goto fail;
! 243: init_mutex = 1;
! 244:
! 245: /* Handle stderr case */
! 246: if (ch->facility == -1) {
! 247: ch->type = ALOG_STDERR;
! 248: return (0);
! 249: }
! 250:
! 251: /* Handle NULL case */
! 252: if (conf->name == NULL) {
! 253: ch->type = ALOG_NULL;
! 254: return (0);
! 255: }
! 256:
! 257: /* Handle local syslog case */
! 258: if (conf->remote_server.s_addr == 0) {
! 259: ch->type = ALOG_SYSLOG_LOCAL;
! 260: return (0);
! 261: }
! 262:
! 263: /* Handle remote syslog case */
! 264: if ((ch->sock = socket(PF_INET, SOCK_DGRAM, IPPROTO_UDP)) == -1)
! 265: goto fail;
! 266: (void)fcntl(ch->sock, F_SETFD, 1);
! 267: if (shutdown(ch->sock, SHUT_RD) == -1)
! 268: goto fail;
! 269: memset(&sin, 0, sizeof(sin));
! 270: #ifndef __linux__
! 271: sin.sin_len = sizeof(sin);
! 272: #endif
! 273: sin.sin_family = AF_INET;
! 274: sin.sin_port = htons(SYSLOG_PORT);
! 275: sin.sin_addr = conf->remote_server;
! 276: if (connect(ch->sock, (struct sockaddr *)&sin, sizeof(sin)) == -1)
! 277: goto fail;
! 278: ch->type = ALOG_SYSLOG_REMOTE;
! 279: return (0);
! 280:
! 281: fail:
! 282: /* Clean up after failure */
! 283: esave = errno;
! 284: logfile_close(&ch->logfile); /* ok if null */
! 285: if (ch->sock != -1) {
! 286: (void)close(ch->sock);
! 287: ch->sock = -1;
! 288: }
! 289: if (init_mutex)
! 290: pthread_mutex_destroy(&ch->mutex);
! 291: errno = esave;
! 292: return (-1);
! 293: }
! 294:
! 295: /*
! 296: * Shutdown one logging channel.
! 297: */
! 298: int
! 299: alog_shutdown(int channel)
! 300: {
! 301: struct alog_channel *const ch = &alog_channels[channel];
! 302:
! 303: /* Sanity check */
! 304: if (channel < 0 || channel >= ALOG_MAX_CHANNELS) {
! 305: errno = EINVAL;
! 306: return (-1);
! 307: }
! 308:
! 309: /* Cancel timer */
! 310: ch->last.timer_expiry = 0;
! 311:
! 312: /* Flush any repeated message */
! 313: alog_flush_lastmsg(ch, 0);
! 314:
! 315: /* Shut down channel */
! 316: switch (ch->type) {
! 317: case ALOG_SYSLOG_REMOTE:
! 318: (void)close(ch->sock);
! 319: ch->sock = -1;
! 320: pthread_mutex_destroy(&ch->mutex);
! 321: break;
! 322: case ALOG_SYSLOG_LOCAL:
! 323: closelog();
! 324: pthread_mutex_destroy(&ch->mutex);
! 325: break;
! 326: case ALOG_NULL:
! 327: case ALOG_STDERR:
! 328: pthread_mutex_destroy(&ch->mutex);
! 329: break;
! 330: case ALOG_NONE:
! 331: break;
! 332: }
! 333: ch->type = ALOG_NONE;
! 334:
! 335: /* Free saved last message */
! 336: FREE(MEM_TYPE, ch->last.msg);
! 337: ch->last.msg = NULL;
! 338:
! 339: /* Close logfile */
! 340: logfile_close(&ch->logfile); /* ok if null */
! 341:
! 342: /* Done */
! 343: return (0);
! 344: }
! 345:
! 346: /*
! 347: * Get current channel.
! 348: */
! 349: static int
! 350: alog_get_channel(void)
! 351: {
! 352: struct alog_curchan *cc;
! 353:
! 354: if ((cc = alog_get_current_channel(0)) == NULL)
! 355: return (0);
! 356: return (cc->channel);
! 357: }
! 358:
! 359: /*
! 360: * Set active channel.
! 361: */
! 362: int
! 363: alog_set_channel(int channel)
! 364: {
! 365: struct alog_curchan *cc;
! 366:
! 367: /* Sanity check */
! 368: if (channel < 0 || channel >= ALOG_MAX_CHANNELS) {
! 369: errno = EINVAL;
! 370: return (-1);
! 371: }
! 372:
! 373: /* Compare with current channel */
! 374: if ((cc = alog_get_current_channel(channel != 0)) == NULL) {
! 375: if (channel != 0)
! 376: return (-1);
! 377: return (0);
! 378: }
! 379: if (channel == cc->channel)
! 380: return (0);
! 381:
! 382: /* Set to new channel, even if it's uninitialized */
! 383: cc->channel = channel;
! 384:
! 385: /* Done */
! 386: return (0);
! 387: }
! 388:
! 389: /*
! 390: * Get current channel per-thread variable.
! 391: */
! 392: static struct alog_curchan *
! 393: alog_get_current_channel(int create)
! 394: {
! 395: static pthread_once_t init_channel_once = PTHREAD_ONCE_INIT;
! 396: struct alog_curchan *cc;
! 397:
! 398: /* Initialize per-thread variable (once) */
! 399: if ((errno = pthread_once(&init_channel_once,
! 400: alog_init_current_channel)) != 0) {
! 401: fprintf(stderr, "%s: %s: %s\n",
! 402: "alog", "pthread_once", strerror(errno));
! 403: return (NULL);
! 404: }
! 405:
! 406: /* Get instance for this thread; create if necessary */
! 407: if ((cc = pthread_getspecific(alog_current_channel)) == NULL) {
! 408: if (!create)
! 409: return (NULL);
! 410: if ((cc = MALLOC(MEM_TYPE_CURCHAN, sizeof(*cc))) == NULL) {
! 411: fprintf(stderr, "%s: %s: %s\n",
! 412: "alog", "malloc", strerror(errno));
! 413: return (NULL);
! 414: }
! 415: cc->channel = 0;
! 416: if ((errno = pthread_setspecific(alog_current_channel,
! 417: cc)) != 0) {
! 418: fprintf(stderr, "%s: %s: %s\n",
! 419: "alog", "pthread_setspecific", strerror(errno));
! 420: FREE(MEM_TYPE_CURCHAN, cc);
! 421: return (NULL);
! 422: }
! 423: }
! 424:
! 425: /* Return current channel */
! 426: return (cc);
! 427: }
! 428:
! 429: /*
! 430: * Initialize "current channel" per-thread variable.
! 431: */
! 432: static void
! 433: alog_init_current_channel(void)
! 434: {
! 435: int err;
! 436:
! 437: if ((err = pthread_key_create(&alog_current_channel,
! 438: alog_curchan_destroy)) != 0) {
! 439: fprintf(stderr, "%s: %s: %s\n",
! 440: "alog", "pthread_key_create", strerror(err));
! 441: assert(0);
! 442: }
! 443: }
! 444:
! 445: /*
! 446: * Destroy an instance of the per-thread current channel variable.
! 447: */
! 448: static void
! 449: alog_curchan_destroy(void *arg)
! 450: {
! 451: struct alog_curchan *const cc = arg;
! 452:
! 453: FREE(MEM_TYPE_CURCHAN, cc);
! 454: }
! 455:
! 456: /*
! 457: * Enable debugging on a channel.
! 458: */
! 459: void
! 460: alog_set_debug(int channel, int enabled)
! 461: {
! 462: struct alog_channel *const ch = &alog_channels[channel];
! 463:
! 464: ch->debug = !!enabled;
! 465: }
! 466:
! 467: /*
! 468: * Log something to the currently selected channel. Preserves errno.
! 469: */
! 470: void
! 471: alog(int sev, const char *fmt, ...)
! 472: {
! 473: va_list args;
! 474:
! 475: va_start(args, fmt);
! 476: valog(sev, fmt, args);
! 477: va_end(args);
! 478: }
! 479:
! 480: /*
! 481: * Log something to the currently selected channel. Preserves errno.
! 482: */
! 483: void
! 484: valog(int sev, const char *fmt, va_list args)
! 485: {
! 486: const int errno_save = errno;
! 487: struct alog_channel *const ch = &alog_channels[alog_get_channel()];
! 488: char fmtbuf[1024];
! 489: time_t now;
! 490: char *msg;
! 491: int r;
! 492:
! 493: /* Lock channel */
! 494: if (ch->type != ALOG_NONE) {
! 495: r = pthread_mutex_lock(&ch->mutex);
! 496: assert(r == 0);
! 497: }
! 498:
! 499: /* Check last message timer */
! 500: alog_last_check(ch);
! 501:
! 502: /* Filter out unwanted verbosity */
! 503: if (ch->type != ALOG_NONE && sev > ch->min_severity)
! 504: goto done;
! 505:
! 506: /* Allow for things like ``alog(LOG_DEBUG + 1, ...)'' */
! 507: if (sev >= LOG_DEBUG)
! 508: sev = LOG_DEBUG;
! 509:
! 510: /* Build the message */
! 511: alog_expand(fmt, errno_save, fmtbuf, sizeof(fmtbuf));
! 512: VASPRINTF(MEM_TYPE, &msg, fmtbuf, args);
! 513: if (msg == NULL) {
! 514: fprintf(stderr, "%s: %s: %s\n",
! 515: __FUNCTION__, "vasprintf", strerror(errno));
! 516: goto done;
! 517: }
! 518:
! 519: /* Get current time */
! 520: now = time(NULL);
! 521:
! 522: /* If not configured, don't do the repeated messages thing */
! 523: if (ch->type == ALOG_NONE) {
! 524: alog_write(ch, sev, now, msg);
! 525: FREE(MEM_TYPE, msg);
! 526: goto done;
! 527: }
! 528:
! 529: /* Handle repeated messages */
! 530: if (ch->last.msg == NULL || strcmp(msg, ch->last.msg) != 0) {
! 531: alog_flush_lastmsg(ch, LASTMSG_TIMEOUT_INITIAL);
! 532: FREE(MEM_TYPE, ch->last.msg);
! 533: ch->last.msg = msg;
! 534: ch->last.sev = sev;
! 535: ch->last.when = now;
! 536: alog_write(ch, sev, now, msg);
! 537: } else {
! 538: time_t delay;
! 539:
! 540: delay = MAX(now - ch->last.when, 0);
! 541: if (++ch->last.repeat == 1) {
! 542: ch->last.timeout = LASTMSG_TIMEOUT_1(
! 543: ch->last.timeout, delay);
! 544: } else {
! 545: ch->last.timeout = LASTMSG_TIMEOUT_2(
! 546: ch->last.timeout, delay);
! 547: }
! 548: ch->last.when = now;
! 549: ch->last.timer_expiry = now + ch->last.timeout;
! 550: FREE(MEM_TYPE, msg);
! 551: }
! 552:
! 553: done:
! 554: /* Unlock channel and restore errno */
! 555: if (ch->type != ALOG_NONE) {
! 556: r = pthread_mutex_unlock(&ch->mutex);
! 557: assert(r == 0);
! 558: }
! 559: errno = errno_save;
! 560: }
! 561:
! 562: /*
! 563: * Check for expiration of 'last mesasge' timer.
! 564: *
! 565: * This assumes the channel is locked.
! 566: */
! 567: static void
! 568: alog_last_check(struct alog_channel *ch)
! 569: {
! 570: /* Is the timer "running"? */
! 571: if (ch->last.timer_expiry == 0)
! 572: return;
! 573:
! 574: /* Has the timer "expired"? */
! 575: if (ch->last.timer_expiry > time(NULL))
! 576: return;
! 577:
! 578: /* "Stop" timer */
! 579: ch->last.timer_expiry = 0;
! 580:
! 581: /* Flush last message */
! 582: alog_flush_lastmsg(ch, LASTMSG_TIMEOUT_3(ch->last.timeout));
! 583: }
! 584:
! 585: /*
! 586: * Log repeated message(s).
! 587: *
! 588: * This assumes the channel is locked.
! 589: */
! 590: static void
! 591: alog_flush_lastmsg(struct alog_channel *ch, u_int timeout)
! 592: {
! 593: switch (ch->last.repeat) {
! 594: case 0:
! 595: break;
! 596: case 1:
! 597: if (ch->last.msg != NULL) {
! 598: alog_write(ch, ch->last.sev,
! 599: ch->last.when, ch->last.msg);
! 600: }
! 601: break;
! 602: default:
! 603: {
! 604: char buf[sizeof(LASTMSG_REPEATED_FMT) + 32];
! 605:
! 606: snprintf(buf, sizeof(buf),
! 607: LASTMSG_REPEATED_FMT, ch->last.repeat);
! 608: alog_write(ch, ch->last.sev, ch->last.when, buf);
! 609: break;
! 610: }
! 611: }
! 612: ch->last.repeat = 0;
! 613: ch->last.timeout = timeout;
! 614: ch->last.timer_expiry = 0;
! 615: }
! 616:
! 617: /*
! 618: * Write a message to the log.
! 619: *
! 620: * This assumes the channel is locked.
! 621: */
! 622: static void
! 623: alog_write(struct alog_channel *ch, int sev, time_t when, const char *msg)
! 624: {
! 625: /* Log it to wherever */
! 626: switch (ch->type) {
! 627: case ALOG_NONE:
! 628: case ALOG_STDERR:
! 629: fprintf(stderr, "%s: %s\n",
! 630: alog_severity_name(LOG_PRI(sev)), msg);
! 631: break;
! 632: case ALOG_NULL:
! 633: break;
! 634: case ALOG_SYSLOG_LOCAL:
! 635: openlog(ch->name, 0, ch->facility); /* XXX race condition */
! 636: syslog(sev, "%s", msg); /* XXX race condition */
! 637: break;
! 638: case ALOG_SYSLOG_REMOTE:
! 639: {
! 640: char ascbuf[32];
! 641: struct tm tm;
! 642: char *netmsg;
! 643: int len;
! 644:
! 645: if (ch->sock == -1)
! 646: break;
! 647: gmtime_r(&when, &tm);
! 648: asctime_r(&tm, ascbuf);
! 649: len = ASPRINTF(TYPED_MEM_TEMP, &netmsg, "<%d>%.15s %s",
! 650: LOG_MAKEPRI(ch->facility, sev), ascbuf + 4, msg);
! 651: if (netmsg != NULL)
! 652: write(ch->sock, netmsg, len);
! 653: FREE(TYPED_MEM_TEMP, netmsg);
! 654: break;
! 655: }
! 656: }
! 657:
! 658: /* Save message in history buffer */
! 659: if (ch->logfile != NULL) {
! 660: struct alog_entry *ent;
! 661: int mlen;
! 662:
! 663: /* Setup entry */
! 664: mlen = strlen(msg) + 1;
! 665: if ((ent = MALLOC(MEM_TYPE, sizeof(*ent) + mlen)) == NULL) {
! 666: fprintf(stderr, "%s: %s: %s\n",
! 667: __FUNCTION__, "malloc", strerror(errno));
! 668: return;
! 669: }
! 670: ent->when = when;
! 671: ent->sev = LOG_PRI(sev);
! 672: memcpy(ent->msg, msg, mlen);
! 673:
! 674: /* Add to history file */
! 675: if (logfile_put(ch->logfile, ent, sizeof(*ent) + mlen) == -1) {
! 676: fprintf(stderr, "%s: %s: %s\n",
! 677: __FUNCTION__, "logfile_put", strerror(errno));
! 678: }
! 679: FREE(MEM_TYPE, ent);
! 680: }
! 681: }
! 682:
! 683: /*
! 684: * Get channel history
! 685: */
! 686: int
! 687: alog_get_history(int channel, int min_sev, int max_num, time_t max_age,
! 688: const regex_t *preg, struct alog_history *list)
! 689: {
! 690: struct alog_channel *const ch = &alog_channels[channel];
! 691: struct alog_entry *copy;
! 692: regmatch_t pmatch;
! 693: u_int num_ent;
! 694: int rtn = -1;
! 695: time_t now;
! 696: int i, j;
! 697: u_int num;
! 698: int r;
! 699:
! 700: /* Sanity check */
! 701: if (channel < 0 || channel >= ALOG_MAX_CHANNELS) {
! 702: errno = EINVAL;
! 703: return (-1);
! 704: }
! 705:
! 706: /* Lock channel */
! 707: if (ch->type != ALOG_NONE) {
! 708: r = pthread_mutex_lock(&ch->mutex);
! 709: assert(r == 0);
! 710: }
! 711:
! 712: /* Check last message timer */
! 713: alog_last_check(ch);
! 714:
! 715: /* Initialize array */
! 716: memset(list, 0, sizeof(*list));
! 717:
! 718: /* Nothing to return? */
! 719: if (ch->logfile == NULL
! 720: || max_num == 0
! 721: || (num_ent = logfile_num_entries(ch->logfile)) == 0) {
! 722: rtn = 0;
! 723: goto done;
! 724: }
! 725:
! 726: /* Get current time */
! 727: now = time(NULL);
! 728:
! 729: #define MATCH(sev, when, msg, msglen) \
! 730: (sev <= min_sev && (now - when) <= max_age && (preg == NULL \
! 731: || (pmatch.rm_so = 0, pmatch.rm_eo = msglen, \
! 732: regexec(preg, msg, 0, &pmatch, REG_STARTEND) == 0)))
! 733:
! 734: /* Count how many entries we'll be returning */
! 735: num = 0;
! 736: if (ch->last.repeat > 0
! 737: && ch->last.msg != NULL
! 738: && MATCH(ch->last.sev, ch->last.when,
! 739: ch->last.msg, strlen(ch->last.msg)))
! 740: num++;
! 741: for (i = -1; num < max_num && i >= -num_ent; i--) {
! 742: const struct alog_entry *ent;
! 743: int len;
! 744:
! 745: /* Get entry, check severity and age */
! 746: if ((ent = logfile_get(ch->logfile, i, &len)) != NULL
! 747: && len >= sizeof(*ent) + 1
! 748: && MATCH(ent->sev, ent->when,
! 749: ent->msg, len - sizeof(*ent) - 1))
! 750: num++;
! 751: }
! 752: if (num == 0) {
! 753: rtn = 0;
! 754: goto done;
! 755: }
! 756:
! 757: /* Allocate array of pointers */
! 758: if ((list->elems = MALLOC(MEM_TYPE_HISTORY,
! 759: num * sizeof(*list->elems))) == NULL)
! 760: goto done;
! 761:
! 762: /* Fill array, starting with the most recent first */
! 763: j = num;
! 764: if (ch->last.repeat > 0
! 765: && ch->last.msg != NULL
! 766: && MATCH(ch->last.sev, ch->last.when,
! 767: ch->last.msg, strlen(ch->last.msg))) {
! 768: char buf[sizeof(LASTMSG_REPEATED_FMT) + 32];
! 769: const char *s;
! 770:
! 771: if (ch->last.repeat > 1) {
! 772: snprintf(buf, sizeof(buf),
! 773: LASTMSG_REPEATED_FMT, ch->last.repeat);
! 774: s = buf;
! 775: } else
! 776: s = ch->last.msg;
! 777: if ((copy = MALLOC(MEM_TYPE_ENTRY,
! 778: sizeof(*copy) + strlen(s) + 1)) != NULL) {
! 779: copy->when = ch->last.when;
! 780: copy->sev = ch->last.sev;
! 781: strcpy(copy->msg, s);
! 782: list->elems[--j] = copy;
! 783: }
! 784: }
! 785: for (i = -1; j > 0 && i >= -num_ent; i--) {
! 786: const struct alog_entry *ent;
! 787: int len;
! 788:
! 789: /* Copy it if it matches */
! 790: if ((ent = logfile_get(ch->logfile, i, &len)) != NULL
! 791: && len >= sizeof(*ent) + 1
! 792: && MATCH(ent->sev, ent->when,
! 793: ent->msg, len - sizeof(*ent) - 1)) {
! 794: if ((copy = MALLOC(MEM_TYPE_ENTRY, len)) != NULL) {
! 795: memcpy(copy, ent, len - 1);
! 796: ((char *)copy)[len - 1] = '\0'; /* safety */
! 797: list->elems[--j] = copy;
! 798: }
! 799: }
! 800: }
! 801:
! 802: /* Collapse entries dropped because of memory problems or whatever */
! 803: if (j > 0) {
! 804: num -= j;
! 805: memcpy(list->elems, list->elems + j,
! 806: num * sizeof(*list->elems));
! 807: }
! 808: list->length = num;
! 809: rtn = 0;
! 810:
! 811: done:
! 812: /* Unlock channel and return */
! 813: if (ch->type != ALOG_NONE) {
! 814: r = pthread_mutex_unlock(&ch->mutex);
! 815: assert(r == 0);
! 816: }
! 817: return (rtn);
! 818: }
! 819:
! 820: /*
! 821: * Clear log history.
! 822: */
! 823: int
! 824: alog_clear_history(int channel)
! 825: {
! 826: struct alog_channel *const ch = &alog_channels[channel];
! 827: int r;
! 828:
! 829: /* Sanity check */
! 830: if (channel < 0 || channel >= ALOG_MAX_CHANNELS) {
! 831: errno = EINVAL;
! 832: return (-1);
! 833: }
! 834:
! 835: /* Lock channel */
! 836: if (ch->type != ALOG_NONE) {
! 837: r = pthread_mutex_lock(&ch->mutex);
! 838: assert(r == 0);
! 839: }
! 840:
! 841: /* Reset last message state */
! 842: ch->last.timer_expiry = 0;
! 843: FREE(MEM_TYPE, ch->last.msg);
! 844: memset(&ch->last, 0, sizeof(ch->last));
! 845:
! 846: /* Zero out logfile */
! 847: if (ch->logfile != NULL)
! 848: logfile_trim(ch->logfile, 0);
! 849:
! 850: /* Unlock channel */
! 851: if (ch->type != ALOG_NONE) {
! 852: r = pthread_mutex_unlock(&ch->mutex);
! 853: assert(r == 0);
! 854: }
! 855:
! 856: /* Done */
! 857: return (0);
! 858: }
! 859:
! 860: /*
! 861: * Decode syslog facility name
! 862: */
! 863: int
! 864: alog_facility(const char *name)
! 865: {
! 866: int i;
! 867:
! 868: for (i = 0; facilitynames[i].name != NULL; i++) {
! 869: if (strcmp(name, facilitynames[i].name) == 0)
! 870: return (facilitynames[i].val);
! 871: }
! 872: return (-1);
! 873: }
! 874:
! 875: /*
! 876: * Decode syslog facility name
! 877: */
! 878: const char *
! 879: alog_facility_name(int facility)
! 880: {
! 881: int i;
! 882:
! 883: for (i = 0; facilitynames[i].name != NULL; i++) {
! 884: if (facilitynames[i].val == facility)
! 885: return (facilitynames[i].name);
! 886: }
! 887: return (NULL);
! 888: }
! 889:
! 890: /*
! 891: * Decode syslog severity name
! 892: */
! 893: int
! 894: alog_severity(const char *name)
! 895: {
! 896: char *eptr;
! 897: int i;
! 898:
! 899: for (i = 0; prioritynames[i].name != NULL; i++) {
! 900: if (strcmp(name, prioritynames[i].name) == 0)
! 901: return (prioritynames[i].val);
! 902: }
! 903: i = (int)strtol(name, &eptr, 0);
! 904: if (*name != '\0' && *eptr == '\0')
! 905: return (i);
! 906: return (-1);
! 907: }
! 908:
! 909: /*
! 910: * Decode syslog severity name
! 911: */
! 912: const char *
! 913: alog_severity_name(int severity)
! 914: {
! 915: int i;
! 916:
! 917: for (i = 0; prioritynames[i].name != NULL; i++) {
! 918: if (prioritynames[i].val == severity)
! 919: return (prioritynames[i].name);
! 920: }
! 921: return (NULL);
! 922: }
! 923:
! 924: /*
! 925: * Kludge to deal with %m format when not using syslog
! 926: */
! 927: void
! 928: alog_expand(const char *fmt, int errnum, char *buf, size_t bufsize)
! 929: {
! 930: const char *errstr;
! 931: const char *s;
! 932: int errlen;
! 933: int i;
! 934:
! 935: /* Find "%m" which is usually at the end */
! 936: for (s = fmt + strlen(fmt) - 2;
! 937: s >= fmt && !(s[0] == '%' && s[1] == 'm');
! 938: s--);
! 939:
! 940: /* Check if we should bother doing anything */
! 941: if (s < fmt || (i = s - fmt) > bufsize - 2) {
! 942: strlcpy(buf, fmt, bufsize);
! 943: return;
! 944: }
! 945:
! 946: /* Convert "%m" to error string */
! 947: errstr = strerror(errnum);
! 948: errlen = strlen(errstr);
! 949: strlcpy(buf, fmt, i + 1);
! 950: strlcpy(buf + i, errstr, bufsize - i);
! 951: strlcpy(buf + i + errlen, fmt + i + 2, bufsize - i - errlen);
! 952: }
! 953:
! 954: /*********************************************************************
! 955: STRUCTS TYPE DEFINITIONS
! 956: *********************************************************************/
! 957:
! 958: /* Type for "path" and "name" strings in struct alog_config */
! 959: static const struct structs_type alog_path_type
! 960: = STRUCTS_STRING_TYPE("alog_config.path", 1);
! 961: static const struct structs_type alog_name_type
! 962: = STRUCTS_STRING_TYPE("alog_config.name", 1);
! 963:
! 964: /* Fields list for struct alog_config */
! 965: static const struct structs_field alog_config_fields[] = {
! 966: STRUCTS_STRUCT_FIELD(alog_config, path, &alog_path_type),
! 967: STRUCTS_STRUCT_FIELD(alog_config, name, &alog_name_type),
! 968: STRUCTS_STRUCT_FIELD(alog_config, facility, &alog_facility_type),
! 969: STRUCTS_STRUCT_FIELD(alog_config, remote_server, &structs_type_ip4),
! 970: STRUCTS_STRUCT_FIELD(alog_config, min_severity, &alog_severity_type),
! 971: STRUCTS_STRUCT_FIELD(alog_config, histlen, &structs_type_int),
! 972: STRUCTS_STRUCT_FIELD_END
! 973: };
! 974:
! 975: /* Type for struct alog_config */
! 976: const struct structs_type alog_config_type
! 977: = STRUCTS_STRUCT_TYPE(alog_config, &alog_config_fields);
! 978:
! 979: /* Override method for alog_facility_type */
! 980: static structs_init_t alog_facility_type_init;
! 981: static structs_binify_t alog_facility_type_binify;
! 982:
! 983: /* Type for syslog facility, which we store as a string. */
! 984: const struct structs_type alog_facility_type = {
! 985: sizeof(char *),
! 986: "alog",
! 987: STRUCTS_TYPE_PRIMITIVE,
! 988: alog_facility_type_init,
! 989: structs_ascii_copy,
! 990: structs_string_equal,
! 991: structs_string_ascify,
! 992: alog_facility_type_binify,
! 993: structs_string_encode,
! 994: structs_string_decode,
! 995: structs_string_free,
! 996: { { (void *)"alog_config.facility" }, { (void *)1 } }
! 997: };
! 998:
! 999: /*
! 1000: * Initializer for alog_facility_type
! 1001: */
! 1002: static int
! 1003: alog_facility_type_init(const struct structs_type *type, void *data)
! 1004: {
! 1005: return (structs_string_binify(type, "daemon", data, NULL, 0));
! 1006: }
! 1007:
! 1008: /*
! 1009: * Binifier for alog_facility_type
! 1010: */
! 1011: static int
! 1012: alog_facility_type_binify(const struct structs_type *type,
! 1013: const char *ascii, void *data, char *ebuf, size_t emax)
! 1014: {
! 1015: if (*ascii != '\0' && alog_facility(ascii) == -1) {
! 1016: strlcpy(ebuf, "invalid syslog facility", emax);
! 1017: errno = EINVAL;
! 1018: return (-1);
! 1019: }
! 1020: return (structs_string_binify(type, ascii, data, NULL, 0));
! 1021: }
! 1022:
! 1023: /* Override methods for alog_severity_type */
! 1024: static structs_init_t alog_severity_init;
! 1025: static structs_ascify_t alog_severity_ascify;
! 1026: static structs_binify_t alog_severity_binify;
! 1027:
! 1028: /* Type for severity, which defaults to LOG_INFO. The input can be either
! 1029: a symbolic name or a numeric value. */
! 1030: const struct structs_type alog_severity_type = {
! 1031: sizeof(int),
! 1032: "int",
! 1033: STRUCTS_TYPE_PRIMITIVE,
! 1034: alog_severity_init,
! 1035: structs_region_copy,
! 1036: structs_region_equal,
! 1037: alog_severity_ascify,
! 1038: alog_severity_binify,
! 1039: structs_region_encode_netorder,
! 1040: structs_region_decode_netorder,
! 1041: structs_nothing_free,
! 1042: { { (void *)2 }, { (void *)1 } }
! 1043: };
! 1044:
! 1045: /*
! 1046: * Initializer for severity
! 1047: */
! 1048: static int
! 1049: alog_severity_init(const struct structs_type *type, void *data)
! 1050: {
! 1051: *((int *)data) = LOG_INFO;
! 1052: return (0);
! 1053: }
! 1054:
! 1055: /*
! 1056: * Ascifier for severity
! 1057: */
! 1058: static char *
! 1059: alog_severity_ascify(const struct structs_type *type,
! 1060: const char *mtype, const void *data)
! 1061: {
! 1062: const int sev = *((int *)data);
! 1063: const char *s;
! 1064: char buf[32];
! 1065:
! 1066: if ((s = alog_severity_name(sev)) == NULL) {
! 1067: snprintf(buf, sizeof(buf), "%d", sev);
! 1068: s = buf;
! 1069: }
! 1070: return (STRDUP(mtype, s));
! 1071: }
! 1072:
! 1073: /*
! 1074: * Binifier for severity
! 1075: */
! 1076: static int
! 1077: alog_severity_binify(const struct structs_type *type,
! 1078: const char *ascii, void *data, char *ebuf, size_t emax)
! 1079: {
! 1080: int sev;
! 1081:
! 1082: if ((sev = alog_severity(ascii)) == -1
! 1083: && structs_int_binify(type, ascii, data, NULL, 0) == -1) {
! 1084: strlcpy(ebuf, "invalid syslog severity", emax);
! 1085: return (-1);
! 1086: }
! 1087: *((int *)data) = sev;
! 1088: return (0);
! 1089: }
! 1090:
! 1091: /*
! 1092: * Ascifier for log entry message.
! 1093: */
! 1094: static char *
! 1095: alog_entry_msg_ascify(const struct structs_type *type,
! 1096: const char *mtype, const void *data)
! 1097: {
! 1098: return (STRDUP(mtype, data));
! 1099: }
! 1100:
! 1101: /* Type for log entry message. This is a "read only" type. */
! 1102: static const struct structs_type alog_entry_msg_type = {
! 1103: 0,
! 1104: "alog_entry_msg_type",
! 1105: STRUCTS_TYPE_PRIMITIVE,
! 1106: structs_notsupp_init,
! 1107: structs_notsupp_copy,
! 1108: structs_notsupp_equal,
! 1109: alog_entry_msg_ascify,
! 1110: structs_notsupp_binify,
! 1111: structs_notsupp_encode,
! 1112: structs_notsupp_decode,
! 1113: structs_nothing_free,
! 1114: };
! 1115:
! 1116: /* Type for history list: an array of pointers to log entries */
! 1117: static const struct structs_field alog_entry_fields[] = {
! 1118: STRUCTS_STRUCT_FIELD(alog_entry, when, &structs_type_time_abs),
! 1119: STRUCTS_STRUCT_FIELD(alog_entry, sev, &alog_severity_type),
! 1120: STRUCTS_STRUCT_FIELD(alog_entry, msg, &alog_entry_msg_type),
! 1121: STRUCTS_STRUCT_FIELD_END
! 1122: };
! 1123: static const struct structs_type alog_entry_type
! 1124: = STRUCTS_STRUCT_TYPE(alog_entry, &alog_entry_fields);
! 1125: static const struct structs_type alog_history_ptr_type
! 1126: = STRUCTS_POINTER_TYPE(&alog_entry_type, MEM_TYPE_ENTRY);
! 1127: const struct structs_type alog_history_type
! 1128: = STRUCTS_ARRAY_TYPE(&alog_history_ptr_type, MEM_TYPE_HISTORY, "entry");
! 1129:
FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>