#include <stdarg.h>
#include <errno.h>
#include <stdio.h>
#include <syslog.h>
#include <libintl.h>
#include <string.h>
#include <limits.h>
#include "dhcpmsg.h"
static boolean_t is_daemon = B_FALSE;
static boolean_t is_verbose = B_FALSE;
static char program[PATH_MAX] = "<unknown>";
static int debug_level;
static const char *err_to_string(int);
static int err_to_syslog(int);
void
dhcpmsg(int errlevel, const char *fmt, ...)
{
va_list ap;
char buf[512];
char *errmsg;
if ((errlevel == MSG_DEBUG2 && (debug_level < 2)) ||
(errlevel == MSG_DEBUG && (debug_level < 1)) ||
(errlevel == MSG_VERBOSE && !is_verbose))
return;
va_start(ap, fmt);
if (is_daemon) {
(void) snprintf(buf, sizeof (buf), (errlevel == MSG_ERR &&
errno != 0) ? "%s: %%m\n" : "%s\n", gettext(fmt));
(void) vsyslog(err_to_syslog(errlevel), buf, ap);
} else {
errmsg = strerror(errno);
if (errmsg == NULL)
errmsg = dgettext(TEXT_DOMAIN, "<unknown error>");
(void) snprintf(buf, sizeof (buf), (errlevel == MSG_ERR &&
errno != 0) ? "%s: %s: %s: %s\n" : "%s: %s: %s\n", program,
dgettext(TEXT_DOMAIN, err_to_string(errlevel)),
gettext(fmt), errmsg);
(void) vfprintf(stderr, buf, ap);
}
va_end(ap);
}
void
dhcpmsg_init(const char *program_name, boolean_t daemon, boolean_t verbose,
int level)
{
(void) strlcpy(program, program_name, sizeof (program));
debug_level = level;
is_verbose = verbose;
if (daemon) {
is_daemon = B_TRUE;
(void) openlog(program, LOG_PID, LOG_DAEMON);
if (is_verbose) {
syslog(err_to_syslog(MSG_VERBOSE), "%s",
dgettext(TEXT_DOMAIN, "Daemon started"));
}
}
}
void
dhcpmsg_fini(void)
{
if (is_daemon) {
if (is_verbose) {
syslog(err_to_syslog(MSG_VERBOSE), "%s",
dgettext(TEXT_DOMAIN, "Daemon terminated"));
}
closelog();
}
}
static int
err_to_syslog(int errlevel)
{
switch (errlevel) {
case MSG_DEBUG:
case MSG_DEBUG2:
return (LOG_DEBUG);
case MSG_ERROR:
case MSG_ERR:
return (LOG_ERR);
case MSG_WARNING:
return (LOG_WARNING);
case MSG_NOTICE:
return (LOG_NOTICE);
case MSG_CRIT:
return (LOG_CRIT);
case MSG_VERBOSE:
case MSG_INFO:
return (LOG_INFO);
}
return (LOG_INFO);
}
static const char *
err_to_string(int errlevel)
{
switch (errlevel) {
case MSG_DEBUG:
case MSG_DEBUG2:
return ("debug");
case MSG_ERR:
case MSG_ERROR:
return ("error");
case MSG_WARNING:
return ("warning");
case MSG_NOTICE:
return ("notice");
case MSG_CRIT:
return ("CRITICAL");
case MSG_VERBOSE:
case MSG_INFO:
return ("info");
}
return ("<unknown>");
}