#include "log.h" #include #include #include #include #include #include #include #include #include "debug.h" #include "util.h" #include "xsnprintf.h" static bool colorize = false; static bool do_syslog = true; static enum log_class log_level = LOG_CLASS_INFO; static const char log_level_map[][8] = { [LOG_CLASS_ERROR] = "error", [LOG_CLASS_WARNING] = "warning", [LOG_CLASS_INFO] = "info", #if defined(_DEBUG) [LOG_CLASS_DEBUG] = "debug", #endif }; void log_init(enum log_colorize _colorize, bool _do_syslog, enum log_facility syslog_facility, enum log_class _log_level) { static const int facility_map[] = { [LOG_FACILITY_USER] = LOG_USER, [LOG_FACILITY_DAEMON] = LOG_DAEMON, }; static const int level_map[] = { [LOG_CLASS_ERROR] = LOG_ERR, [LOG_CLASS_WARNING] = LOG_WARNING, [LOG_CLASS_INFO] = LOG_INFO, [LOG_CLASS_DEBUG] = LOG_DEBUG, }; colorize = _colorize == LOG_COLORIZE_NEVER ? false : _colorize == LOG_COLORIZE_ALWAYS ? true : isatty(STDERR_FILENO); do_syslog = _do_syslog; log_level = _log_level; if (do_syslog) { openlog(NULL, /*LOG_PID*/0, facility_map[syslog_facility]); setlogmask(LOG_UPTO(level_map[_log_level])); } } void log_deinit(void) { if (do_syslog) closelog(); } static void _log(enum log_class log_class, const char *module, const char *file, int lineno, const char *fmt, int sys_errno, va_list va) { if (log_class > log_level) return; const char *class = "abcd"; int class_clr = 0; switch (log_class) { case LOG_CLASS_ERROR: class = " err"; class_clr = 31; break; case LOG_CLASS_WARNING: class = "warn"; class_clr = 33; break; case LOG_CLASS_INFO: class = "info"; class_clr = 97; break; case LOG_CLASS_DEBUG: class = " dbg"; class_clr = 36; break; } char clr[16]; snprintf(clr, sizeof(clr), "\033[%dm", class_clr); fprintf(stderr, "%s%s%s: ", colorize ? clr : "", class, colorize ? "\033[0m" : ""); if (colorize) fprintf(stderr, "\033[2m"); fprintf(stderr, "%s:%d: ", file, lineno); if (colorize) fprintf(stderr, "\033[0m"); vfprintf(stderr, fmt, va); if (sys_errno != 0) fprintf(stderr, ": %s", strerror(sys_errno)); fprintf(stderr, "\n"); } static void _sys_log(enum log_class log_class, const char *module, const char UNUSED *file, int UNUSED lineno, const char *fmt, int sys_errno, va_list va) { if (!do_syslog) return; /* Map our log level to syslog's level */ int level = -1; switch (log_class) { case LOG_CLASS_ERROR: level = LOG_ERR; break; case LOG_CLASS_WARNING: level = LOG_WARNING; break; case LOG_CLASS_INFO: level = LOG_INFO; break; case LOG_CLASS_DEBUG: level = LOG_DEBUG; break; } xassert(level != -1); char msg[4096]; int n = vsnprintf(msg, sizeof(msg), fmt, va); xassert(n >= 0); if (sys_errno != 0 && (size_t)n < sizeof(msg)) snprintf(msg + n, sizeof(msg) - n, ": %s", strerror(sys_errno)); syslog(level, "%s: %s", module, msg); } void log_msg(enum log_class log_class, const char *module, const char *file, int lineno, const char *fmt, ...) { va_list ap1, ap2; va_start(ap1, fmt); va_copy(ap2, ap1); _log(log_class, module, file, lineno, fmt, 0, ap1); _sys_log(log_class, module, file, lineno, fmt, 0, ap2); va_end(ap1); va_end(ap2); } void log_errno(enum log_class log_class, const char *module, const char *file, int lineno, const char *fmt, ...) { va_list ap1, ap2; va_start(ap1, fmt); va_copy(ap2, ap1); _log(log_class, module, file, lineno, fmt, errno, ap1); _sys_log(log_class, module, file, lineno, fmt, errno, ap2); va_end(ap1); va_end(ap2); } void log_errno_provided(enum log_class log_class, const char *module, const char *file, int lineno, int _errno, const char *fmt, ...) { va_list ap1, ap2; va_start(ap1, fmt); va_copy(ap2, ap1); _log(log_class, module, file, lineno, fmt, _errno, ap1); _sys_log(log_class, module, file, lineno, fmt, _errno, ap2); va_end(ap1); va_end(ap2); } int log_level_from_string(const char *str) { if (unlikely(str[0] == '\0')) return -1; for (int i = 0, n = ALEN(log_level_map); i < n; i++) if (strcmp(str, log_level_map[i]) == 0) return i; return -1; } const char * log_level_string_hint(void) { static char buf[64]; if (buf[0] != '\0') return buf; for (size_t i = 0, pos = 0, n = ALEN(log_level_map); i < n; i++) { const char *entry = log_level_map[i]; const char *delim = (i + 1 < n) ? ", " : ""; pos += xsnprintf(buf + pos, sizeof(buf) - pos, "'%s'%s", entry, delim); } return buf; }