/* This file is part of gacopyz.
Copyright (C) 2006-2020 Sergey Poznyakoff
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 3, 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 . */
#include
void (*__gacopyz_log_printer)(int, char *, va_list) =
gacopyz_syslog_log_printer;
/* Logging */
void
gacopyz_syslog_log_printer(int level, char *fmt, va_list ap)
{
switch (level) {
case SMI_LOG_PROTO:
case SMI_LOG_DEBUG:
level = LOG_DEBUG;
break;
case SMI_LOG_INFO:
level = LOG_INFO;
break;
case SMI_LOG_WARN:
level = LOG_WARNING;
break;
case SMI_LOG_ERR:
level = LOG_ERR;
break;
case SMI_LOG_FATAL:
default:
level = LOG_EMERG;
}
vsyslog(level, fmt, ap); /* FIXME: if absent? */
}
static char *level_name[] = {
"PROTO",
"DEBUG",
"INFO",
"WARN",
"ERR",
"FATAL"
};
int
gacopyz_string_to_log_level(const char *str)
{
int i;
for (i = 0; i < sizeof level_name / sizeof level_name[0]; i++)
if (strcasecmp (str, level_name[i]) == 0)
return i;
return -1;
}
const char *
gacopyz_log_level_to_string(int level)
{
if (level < 0 || level > sizeof level_name / sizeof level_name[0])
return NULL;
return level_name[level];
}
void
gacopyz_stderr_log_printer(int level, char *fmt, va_list ap)
{
fprintf(stderr, "Gacopyz %s: ", gacopyz_log_level_to_string(level));
vfprintf(stderr, fmt, ap);
fprintf(stderr, "\n");
}
void
gacopyz_log(int level, char *fmt, ...)
{
va_list ap;
va_start(ap, fmt);
__gacopyz_log_printer(level, fmt, ap);
va_end(ap);
}
void
gacopyz_io_log(gacopyz_iod_t iod, int level, char *fmt, ...)
{
if (iod->logmask & SMI_LOG_MASK(level)) {
va_list ap;
va_start(ap, fmt);
__gacopyz_log_printer(level, fmt, ap);
va_end(ap);
}
}
size_t
gacopyz_format_vbuf(char vbuf[GACOPYZ_VBUFSIZE], const char *buf, size_t size)
{
char *p = vbuf;
char *q = vbuf + 51;
int i;
size_t j = 0;
for (i = 0; i < 16; i++) {
unsigned c;
if (j < size) {
c = *(const unsigned char*)buf++;
j++;
} else
c = 0;
sprintf(p, "%02X ", c);
p += 3;
*q++ = isprint(c) ? c : '.';
if (i == 7) {
*p++ = ' ';
*q++ = ' ';
}
}
*p++ = ' ';
*p = ' ';
*q = 0;
return j;
}
void
gacopyz_logdump(int level, const char *pfx, const char *buf, size_t size)
{
char vbuf[GACOPYZ_VBUFSIZE];
while (size) {
size_t rd = gacopyz_format_vbuf(vbuf, buf, size);
gacopyz_log(level, "%s: %s", pfx, vbuf);
size -= rd;
buf += rd;
}
}
void
gacopyz_io_logdump(gacopyz_iod_t iod, const char *pfx,
const char *buf, size_t size)
{
if (iod->logmask & SMI_LOG_MASK(SMI_LOG_PROTO))
gacopyz_logdump(SMI_LOG_PROTO, pfx, buf, size);
else if (iod->logmask & SMI_LOG_MASK(SMI_LOG_DEBUG))
gacopyz_log(SMI_LOG_DEBUG, "%s", pfx);
}
void
gacopyz_set_logger(void (*logger)(int, char *, va_list))
{
__gacopyz_log_printer = logger;
}