blob: 94376fe52f9cb1f3a9af1d713569700164a7b584 [file] [log] [blame]
Mark Whitley6317c4b2001-03-12 22:51:50 +00001/* vi: set sw=4 ts=4: */
2/*
3 * Mini klogd implementation for busybox
4 *
Glenn L McGrath6ed77592002-12-12 10:54:48 +00005 * Copyright (C) 2001 by Gennady Feldman <gfeldman@gena01.com>.
Mark Whitley6317c4b2001-03-12 22:51:50 +00006 * Changes: Made this a standalone busybox module which uses standalone
7 * syslog() client interface.
8 *
Eric Andersenc7bda1c2004-03-15 08:29:22 +00009 * Copyright (C) 1999-2004 by Erik Andersen <andersen@codepoet.org>
Mark Whitley6317c4b2001-03-12 22:51:50 +000010 *
11 * Copyright (C) 2000 by Karl M. Hegbloom <karlheg@debian.org>
12 *
Glenn L McGrath6ed77592002-12-12 10:54:48 +000013 * "circular buffer" Copyright (C) 2000 by Gennady Feldman <gfeldman@gena01.com>
Mark Whitley6317c4b2001-03-12 22:51:50 +000014 *
Glenn L McGrath6ed77592002-12-12 10:54:48 +000015 * Maintainer: Gennady Feldman <gfeldman@gena01.com> as of Mar 12, 2001
Mark Whitley6bff9cc2001-03-12 23:41:34 +000016 *
Mark Whitley6317c4b2001-03-12 22:51:50 +000017 * This program is free software; you can redistribute it and/or modify
18 * it under the terms of the GNU General Public License as published by
19 * the Free Software Foundation; either version 2 of the License, or
20 * (at your option) any later version.
21 *
22 * This program is distributed in the hope that it will be useful,
23 * but WITHOUT ANY WARRANTY; without even the implied warranty of
24 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
25 * General Public License for more details.
26 *
27 * You should have received a copy of the GNU General Public License
28 * along with this program; if not, write to the Free Software
29 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
30 *
31 */
32
33#include <stdio.h>
34#include <stdlib.h>
Glenn L McGrath9fef17d2002-08-22 18:41:20 +000035#include <signal.h> /* for our signal() handlers */
36#include <string.h> /* strncpy() */
37#include <errno.h> /* errno and friends */
Mark Whitley6317c4b2001-03-12 22:51:50 +000038#include <unistd.h>
39#include <ctype.h>
40#include <sys/syslog.h>
Eric Andersen85e5e722003-07-22 08:56:55 +000041#include <sys/klog.h>
Eric Andersene76c3b02001-04-05 03:14:39 +000042
Mark Whitley6317c4b2001-03-12 22:51:50 +000043#include "busybox.h"
44
45static void klogd_signal(int sig)
46{
47 klogctl(7, NULL, 0);
48 klogctl(0, 0, 0);
Glenn L McGrath9fef17d2002-08-22 18:41:20 +000049 /* logMessage(0, "Kernel log daemon exiting."); */
Eric Andersen36adca82004-06-22 10:07:17 +000050 syslog(LOG_NOTICE, "Kernel log daemon exiting.");
51 exit(EXIT_SUCCESS);
Mark Whitley6317c4b2001-03-12 22:51:50 +000052}
53
Eric Andersen0373cf12004-06-25 11:23:03 +000054static void doKlogd(const int console_log_level) __attribute__ ((noreturn));
55static void doKlogd(const int console_log_level)
Mark Whitley6317c4b2001-03-12 22:51:50 +000056{
57 int priority = LOG_INFO;
58 char log_buffer[4096];
59 int i, n, lastc;
60 char *start;
61
Eric Andersen36adca82004-06-22 10:07:17 +000062 openlog("kernel", 0, LOG_KERN);
63
Mark Whitley6317c4b2001-03-12 22:51:50 +000064 /* Set up sig handlers */
65 signal(SIGINT, klogd_signal);
66 signal(SIGKILL, klogd_signal);
67 signal(SIGTERM, klogd_signal);
68 signal(SIGHUP, SIG_IGN);
69
70 /* "Open the log. Currently a NOP." */
71 klogctl(1, NULL, 0);
72
Glenn L McGrathe1ad6722002-12-01 11:31:58 +000073 /* Set level of kernel console messaging.. */
Eric Andersen0373cf12004-06-25 11:23:03 +000074 if (console_log_level != -1)
Glenn L McGrathe1ad6722002-12-01 11:31:58 +000075 klogctl(8, NULL, console_log_level);
76
Eric Andersen36adca82004-06-22 10:07:17 +000077 syslog(LOG_NOTICE, "klogd started: " BB_BANNER);
Mark Whitley6317c4b2001-03-12 22:51:50 +000078
79 while (1) {
80 /* Use kernel syscalls */
81 memset(log_buffer, '\0', sizeof(log_buffer));
82 n = klogctl(2, log_buffer, sizeof(log_buffer));
83 if (n < 0) {
Mark Whitley6317c4b2001-03-12 22:51:50 +000084 if (errno == EINTR)
85 continue;
Eric Andersen0373cf12004-06-25 11:23:03 +000086 syslog(LOG_ERR, "klogd: Error return from sys_sycall: %d - %m.\n", errno);
Eric Andersen36adca82004-06-22 10:07:17 +000087 exit(EXIT_FAILURE);
Mark Whitley6317c4b2001-03-12 22:51:50 +000088 }
89
90 /* klogctl buffer parsing modelled after code in dmesg.c */
Glenn L McGrath9fef17d2002-08-22 18:41:20 +000091 start = &log_buffer[0];
92 lastc = '\0';
93 for (i = 0; i < n; i++) {
Mark Whitley6317c4b2001-03-12 22:51:50 +000094 if (lastc == '\0' && log_buffer[i] == '<') {
95 priority = 0;
96 i++;
97 while (isdigit(log_buffer[i])) {
Glenn L McGrath9fef17d2002-08-22 18:41:20 +000098 priority = priority * 10 + (log_buffer[i] - '0');
Mark Whitley6317c4b2001-03-12 22:51:50 +000099 i++;
100 }
Glenn L McGrath9fef17d2002-08-22 18:41:20 +0000101 if (log_buffer[i] == '>')
102 i++;
Mark Whitley6317c4b2001-03-12 22:51:50 +0000103 start = &log_buffer[i];
104 }
105 if (log_buffer[i] == '\n') {
Glenn L McGrath9fef17d2002-08-22 18:41:20 +0000106 log_buffer[i] = '\0'; /* zero terminate this message */
Eric Andersen36adca82004-06-22 10:07:17 +0000107 syslog(priority, start);
Glenn L McGrath9fef17d2002-08-22 18:41:20 +0000108 start = &log_buffer[i + 1];
Mark Whitley6317c4b2001-03-12 22:51:50 +0000109 priority = LOG_INFO;
110 }
111 lastc = log_buffer[i];
112 }
113 }
114}
115
Mark Whitley6317c4b2001-03-12 22:51:50 +0000116extern int klogd_main(int argc, char **argv)
117{
118 /* no options, no getopt */
Eric Andersene5c24df2001-03-29 21:58:33 +0000119 int opt;
Mark Whitley6317c4b2001-03-12 22:51:50 +0000120 int doFork = TRUE;
Eric Andersen0373cf12004-06-25 11:23:03 +0000121 unsigned char console_log_level = -1;
Mark Whitley6317c4b2001-03-12 22:51:50 +0000122
123 /* do normal option parsing */
Glenn L McGrathe1ad6722002-12-01 11:31:58 +0000124 while ((opt = getopt(argc, argv, "c:n")) > 0) {
Mark Whitley6317c4b2001-03-12 22:51:50 +0000125 switch (opt) {
Glenn L McGrathe1ad6722002-12-01 11:31:58 +0000126 case 'c':
127 if ((optarg == NULL) || (optarg[1] != '\0')) {
Manuel Novoa III cad53642003-03-19 09:13:01 +0000128 bb_show_usage();
Glenn L McGrathe1ad6722002-12-01 11:31:58 +0000129 }
130 /* Valid levels are between 1 and 8 */
131 console_log_level = *optarg - '1';
132 if (console_log_level > 7) {
Manuel Novoa III cad53642003-03-19 09:13:01 +0000133 bb_show_usage();
Glenn L McGrathe1ad6722002-12-01 11:31:58 +0000134 }
135 console_log_level++;
Eric Andersenc7bda1c2004-03-15 08:29:22 +0000136
Glenn L McGrathe1ad6722002-12-01 11:31:58 +0000137 break;
Glenn L McGrath9fef17d2002-08-22 18:41:20 +0000138 case 'n':
139 doFork = FALSE;
140 break;
141 default:
Manuel Novoa III cad53642003-03-19 09:13:01 +0000142 bb_show_usage();
Mark Whitley6317c4b2001-03-12 22:51:50 +0000143 }
144 }
145
Matt Kraai1f0c4362001-12-20 23:13:26 +0000146 if (doFork) {
Russ Dilla1fece22003-12-15 21:57:44 +0000147#if defined(__uClinux__)
148 vfork_daemon_rexec(0, 1, argc, argv, "-n");
149#else /* __uClinux__ */
Eric Andersene5c24df2001-03-29 21:58:33 +0000150 if (daemon(0, 1) < 0)
Manuel Novoa III cad53642003-03-19 09:13:01 +0000151 bb_perror_msg_and_die("daemon");
Russ Dilla1fece22003-12-15 21:57:44 +0000152#endif /* __uClinux__ */
Mark Whitley6317c4b2001-03-12 22:51:50 +0000153 }
Glenn L McGrathe1ad6722002-12-01 11:31:58 +0000154 doKlogd(console_log_level);
Glenn L McGrath9fef17d2002-08-22 18:41:20 +0000155
Mark Whitley6317c4b2001-03-12 22:51:50 +0000156 return EXIT_SUCCESS;
157}
158
159/*
160Local Variables
161c-file-style: "linux"
162c-basic-offset: 4
163tab-width: 4
164End:
165*/