diff options
Diffstat (limited to 'tools/log.c')
-rw-r--r-- | tools/log.c | 149 |
1 files changed, 149 insertions, 0 deletions
diff --git a/tools/log.c b/tools/log.c new file mode 100644 index 0000000..3317a35 --- /dev/null +++ b/tools/log.c @@ -0,0 +1,149 @@ +/* + * kmod - log infrastructure + * + * Copyright (C) 2012-2013 ProFUSION embedded systems + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 2 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see <http://www.gnu.org/licenses/>. + */ + +#include <errno.h> +#include <stdio.h> +#include <stdlib.h> +#include <syslog.h> + +#include <libkmod/libkmod.h> + +#include "kmod.h" + +#define PRIO_MAX_SIZE 32 + +static bool log_use_syslog; +static int log_priority = LOG_WARNING; + +static const char *prio_to_str(char buf[static PRIO_MAX_SIZE], int prio) +{ + const char *prioname; + + switch (prio) { + case LOG_CRIT: + prioname = "FATAL"; + break; + case LOG_ERR: + prioname = "ERROR"; + break; + case LOG_WARNING: + prioname = "WARNING"; + break; + case LOG_NOTICE: + prioname = "NOTICE"; + break; + case LOG_INFO: + prioname = "INFO"; + break; + case LOG_DEBUG: + prioname = "DEBUG"; + break; + default: + snprintf(buf, PRIO_MAX_SIZE, "LOG-%03d", prio); + prioname = buf; + } + + return prioname; +} + +_printf_format_(6, 0) +static void log_kmod(void *data, int priority, const char *file, int line, + const char *fn, const char *format, va_list args) +{ + char buf[PRIO_MAX_SIZE]; + const char *prioname; + char *str; + + prioname = prio_to_str(buf, priority); + + if (vasprintf(&str, format, args) < 0) + return; + + if (log_use_syslog) { +#ifdef ENABLE_DEBUG + syslog(priority, "%s: %s:%d %s() %s", prioname, file, line, + fn, str); +#else + syslog(priority, "%s: %s", prioname, str); +#endif + } else { +#ifdef ENABLE_DEBUG + fprintf(stderr, "%s: %s: %s:%d %s() %s", + program_invocation_short_name, prioname, file, line, + fn, str); +#else + fprintf(stderr, "%s: %s: %s", program_invocation_short_name, + prioname, str); +#endif + } + + free(str); + (void)data; +} +void log_open(bool use_syslog) +{ + log_use_syslog = use_syslog; + + if (log_use_syslog) + openlog(program_invocation_short_name, LOG_CONS, LOG_DAEMON); +} + +void log_close(void) +{ + if (log_use_syslog) + closelog(); +} + +void log_printf(int prio, const char *fmt, ...) +{ + char buf[PRIO_MAX_SIZE]; + const char *prioname; + char *msg; + va_list args; + + if (prio > log_priority) + return; + + va_start(args, fmt); + if (vasprintf(&msg, fmt, args) < 0) + msg = NULL; + va_end(args); + if (msg == NULL) + return; + + prioname = prio_to_str(buf, prio); + + if (log_use_syslog) + syslog(prio, "%s: %s", prioname, msg); + else + fprintf(stderr, "%s: %s: %s", program_invocation_short_name, + prioname, msg); + free(msg); + + if (prio <= LOG_CRIT) + exit(EXIT_FAILURE); +} + +void log_setup_kmod_log(struct kmod_ctx *ctx, int priority) +{ + log_priority = priority; + + kmod_set_log_priority(ctx, log_priority); + kmod_set_log_fn(ctx, log_kmod, NULL); +} |