bemanitools/src/main/util/log.c
icex2 96d7c4ed91 feat(util/log): Expose proper log level definition with enum
Internally, this was already used but it lacked a clean
public interface to set different log levels from the outside.

Useful for various command line tooling to implement verbosity
levels, e.g. -v, -vv, -vvv
2023-06-25 13:24:07 +02:00

135 lines
3.7 KiB
C

#include "util/log.h"
#include "util/str.h"
#include <windows.h>
#include <stdarg.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
static log_writer_t log_writer;
static void *log_writer_ctx;
static enum log_level log_level;
static void log_builtin_fatal(const char *module, const char *fmt, ...);
static void log_builtin_info(const char *module, const char *fmt, ...);
static void log_builtin_misc(const char *module, const char *fmt, ...);
static void log_builtin_warning(const char *module, const char *fmt, ...);
static void log_builtin_format(
enum log_level msg_level, const char *module, const char *fmt, va_list ap);
#define IMPLEMENT_SINK(name, msg_level) \
static void name(const char *module, const char *fmt, ...) \
{ \
va_list ap; \
\
va_start(ap, fmt); \
log_builtin_format(msg_level, module, fmt, ap); \
va_end(ap); \
}
IMPLEMENT_SINK(log_builtin_info, LOG_LEVEL_INFO)
IMPLEMENT_SINK(log_builtin_misc, LOG_LEVEL_MISC)
IMPLEMENT_SINK(log_builtin_warning, LOG_LEVEL_WARNING)
static void log_builtin_fatal(const char *module, const char *fmt, ...)
{
va_list ap;
va_start(ap, fmt);
log_builtin_format(LOG_LEVEL_FATAL, module, fmt, ap);
va_end(ap);
DebugBreak();
ExitProcess(EXIT_FAILURE);
}
static void log_builtin_format(
enum log_level msg_level, const char *module, const char *fmt, va_list ap)
{
static const char chars[] = "FWIM";
/* 64k so we can log data dumps of rs232 without crashing */
char line[65536];
char msg[65536];
int result;
if (msg_level <= log_level) {
str_vformat(msg, sizeof(msg), fmt, ap);
result = str_format(
line, sizeof(line), "%c:%s: %s\n", chars[msg_level], module, msg);
log_writer(log_writer_ctx, line, result);
}
}
void log_assert_body(const char *file, int line, const char *function)
{
log_impl_fatal("assert", "%s:%d: function `%s'", file, line, function);
}
void log_to_external(
log_formatter_t misc,
log_formatter_t info,
log_formatter_t warning,
log_formatter_t fatal)
{
log_impl_misc = misc;
log_impl_info = info;
log_impl_warning = warning;
log_impl_fatal = fatal;
}
void log_to_writer(log_writer_t writer, void *ctx)
{
log_impl_misc = log_builtin_misc;
log_impl_info = log_builtin_info;
log_impl_warning = log_builtin_warning;
log_impl_fatal = log_builtin_fatal;
if (writer != NULL) {
log_writer = writer;
log_writer_ctx = ctx;
} else {
log_writer = log_writer_null;
}
}
void log_set_level(enum log_level new_level)
{
log_level = new_level;
}
void log_writer_debug(void *ctx, const char *chars, size_t nchars)
{
OutputDebugStringA(chars);
}
void log_writer_stdout(void *ctx, const char *chars, size_t nchars)
{
printf("%s", chars);
}
void log_writer_stderr(void *ctx, const char *chars, size_t nchars)
{
fprintf(stderr, "%s", chars);
}
void log_writer_file(void *ctx, const char *chars, size_t nchars)
{
fwrite(chars, 1, nchars, (FILE *) ctx);
fflush((FILE *) ctx);
}
void log_writer_null(void *ctx, const char *chars, size_t nchars)
{
}
log_formatter_t log_impl_misc = log_builtin_misc;
log_formatter_t log_impl_info = log_builtin_info;
log_formatter_t log_impl_warning = log_builtin_warning;
log_formatter_t log_impl_fatal = log_builtin_fatal;
static log_writer_t log_writer = log_writer_null;
static enum log_level log_level = LOG_LEVEL_MISC;