taisei/src/log.c
Andrei "Akari" Alexeyev 3a957cb3f2 More control over logging output via env variables and macros
In release builds, everything is logged to the log file by default, and
only warnings and errors go to the console (stderr). In debug builds,
all output goes both to the console (stdout/stderr) and the log file.

You can override this behaviour with environment variables. Some
examples:

    Don't log anything to the console at all:
        TAISEI_LOGLVLS_CONSOLE=-a

    Suppress info and debug output in the log file:
        TAISEI_LOGLVLS_FILE=-di

    More verbose console output in release builds:
        TAISEI_LOGLVLS_CONSOLE=+i

    Override defaults and enable all enable the wanted loglevels explicitly:
        TAISEI_LOGLVLS_CONSOLE=-a+diwe

    Redirect debug output to stderr:
        TAISEI_LOGLVLS_STDOUT=-d
        TAISEI_LOGLVLS_STDERR=+d
2017-03-13 18:57:44 +02:00

170 lines
4 KiB
C

#include <assert.h>
#include <SDL_bits.h>
#include <SDL_mutex.h>
#include "log.h"
#include "util.h"
#include "list.h"
#ifdef __WINDOWS__
#define LOG_EOL "\r\n"
#else
#define LOG_EOL "\n"
#endif
typedef struct Logger {
struct Logger *next;
struct Logger *prev;
SDL_RWops *out;
unsigned int levels;
} Logger;
static Logger *loggers = NULL;
static unsigned int enabled_log_levels;
static SDL_mutex *log_mutex;
// order must much the LogLevel enum after LOG_NONE
static const char *level_prefix_map[] = { "D", "I", "W", "E" };
static const char* level_prefix(LogLevel lvl) {
int idx = SDL_MostSignificantBitIndex32(lvl);
assert(idx >= 0 && idx < sizeof(level_prefix_map));
return level_prefix_map[idx];
}
static char* format_log_string(LogLevel lvl, const char *funcname, const char *fmt, va_list args) {
const char *pref = level_prefix(lvl);
char *msg = vstrfmt(fmt, args);
char *final = strfmt("%-9d %s: %s(): %s%s", SDL_GetTicks(), pref, funcname, msg, LOG_EOL);
free(msg);
// TODO: maybe convert all \n in the message to LOG_EOL
return final;
}
noreturn static void log_abort(const char *msg) {
#ifdef LOG_FATAL_MSGBOX
if(msg) {
SDL_ShowSimpleMessageBox(SDL_MESSAGEBOX_ERROR, "Taisei error", msg, NULL);
}
#endif
// abort() doesn't clean up, but it lets us get a backtrace, which is more useful
log_shutdown();
abort();
}
static void log_internal(LogLevel lvl, const char *funcname, const char *fmt, va_list args) {
assert(fmt[strlen(fmt)-1] != '\n');
char *str = NULL;
size_t slen = 0;
lvl &= enabled_log_levels;
if(lvl == LOG_NONE) {
return;
}
for(Logger *l = loggers; l; l = l->next) {
if(l->levels & lvl) {
if(!str) {
str = format_log_string(lvl, funcname, fmt, args);
slen = strlen(str);
}
SDL_LockMutex(log_mutex);
SDL_RWwrite(l->out, str, 1, slen);
SDL_UnlockMutex(log_mutex);
}
}
free(str);
if(lvl & LOG_FATAL) {
log_abort(str);
}
}
void _taisei_log(LogLevel lvl, const char *funcname, const char *fmt, ...) {
va_list args;
va_start(args, fmt);
log_internal(lvl, funcname, fmt, args);
va_end(args);
}
noreturn void _taisei_log_fatal(LogLevel lvl, const char *funcname, const char *fmt, ...) {
va_list args;
va_start(args, fmt);
log_internal(lvl, funcname, fmt, args);
va_end(args);
// should usually not get here, log_internal will abort earlier if lvl is LOG_FATAL
// that is unless LOG_FATAL is disabled for some reason
log_abort(NULL);
}
static void delete_logger(void **loggers, void *logger) {
Logger *l = logger;
SDL_RWclose(l->out);
delete_element(loggers, logger);
}
void log_init(LogLevel lvls) {
enabled_log_levels = lvls;
log_mutex = SDL_CreateMutex();
}
void log_shutdown(void) {
delete_all_elements((void**)&loggers, delete_logger);
SDL_DestroyMutex(log_mutex);
log_mutex = NULL;
}
void log_add_output(LogLevel levels, SDL_RWops *output) {
if(!output || !(levels & enabled_log_levels)) {
SDL_RWclose(output);
return;
}
Logger *l = create_element((void**)&loggers, sizeof(Logger));
l->levels = levels;
l->out = output;
}
static LogLevel chr2lvl(char c) {
c = toupper(c);
for(int i = 0; i < sizeof(level_prefix_map) / sizeof(char*); ++i) {
if(c == level_prefix_map[i][0]) {
return (1 << i);
} else if(c == 'A') {
return LOG_ALL;
}
}
return 0;
}
LogLevel log_parse_levels(LogLevel lvls, const char *lvlmod) {
if(!lvlmod) {
return lvls;
}
bool enable = true;
for(const char *c = lvlmod; *c; ++c) {
if(*c == '+') {
enable = true;
} else if(*c == '-') {
enable = false;
} else if(enable) {
lvls |= chr2lvl(*c);
} else {
lvls &= ~chr2lvl(*c);
}
}
return lvls;
}