/*
* CDDL HEADER START
*
* The contents of this file are subject to the terms of the
* Common Development and Distribution License (the "License").
* You may not use this file except in compliance with the License.
*
* You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
* or http://www.opensolaris.org/os/licensing.
* See the License for the specific language governing permissions
* and limitations under the License.
*
* When distributing Covered Code, include this CDDL HEADER in each
* file and include the License file at usr/src/OPENSOLARIS.LICENSE.
* If applicable, add the following below this CDDL HEADER, with the
* fields enclosed by brackets "[]" replaced with your own identifying
* information: Portions Copyright [yyyy] [name of copyright owner]
*
* CDDL HEADER END
*/
/*
* Copyright 2007 Sun Microsystems, Inc. All rights reserved.
* Use is subject to license terms.
*/
#pragma ident "%Z%%M% %I% %E% SMI"
#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);
/*
* dhcpmsg(): logs a message to the console or to syslog
*
* input: int: the level to log the message at
* const char *: a printf-like format string
* ...: arguments to the format string
* output: void
*/
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);
/*
* either log to stderr, or log to syslog. print out unix
* error message if errlevel is MSG_ERR and errno is set
*/
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);
}
/*
* dhcpmsg_init(): opens and initializes the DHCP messaging facility
*
* input: const char *: the name of the executable
* boolean_t: whether the executable is a daemon
* boolean_t: whether the executable is running "verbosely"
* int: the debugging level the executable is being run at
* output: void
*/
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"));
}
}
}
/*
* dhcpmsg_fini(): closes the DHCP messaging facility.
*
* input: void
* output: void
*/
void
dhcpmsg_fini(void)
{
if (is_daemon) {
if (is_verbose) {
syslog(err_to_syslog(MSG_VERBOSE), "%s",
dgettext(TEXT_DOMAIN, "Daemon terminated"));
}
closelog();
}
}
/*
* err_to_syslog(): converts a dhcpmsg log level into a syslog log level
*
* input: int: the dhcpmsg log level
* output: int: the syslog log level
*/
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);
}
/*
* err_to_string(): converts a log level into a string
*
* input: int: the log level
* output: const char *: the stringified log level
*/
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>");
}