diff --git a/ctdb/common/ctdb_logging.c b/ctdb/common/ctdb_logging.c
index bf67f096b54..9b5103a20dd 100644
--- a/ctdb/common/ctdb_logging.c
+++ b/ctdb/common/ctdb_logging.c
@@ -23,8 +23,6 @@
#include "../include/ctdb_private.h"
#include "../include/ctdb_client.h"
-int DEBUGLEVEL = DEBUG_NOTICE;
-int this_log_level = 0;
const char *debug_extra = "";
struct debug_levels debug_levels[] = {
diff --git a/ctdb/include/ctdb_logging.h b/ctdb/include/ctdb_logging.h
index f6f904aeade..c0c02660051 100644
--- a/ctdb/include/ctdb_logging.h
+++ b/ctdb/include/ctdb_logging.h
@@ -20,8 +20,6 @@
#ifndef _CTDB_LOGGING_H_
#define _CTDB_LOGGING_H_
-extern int DEBUGLEVEL;
-extern int this_log_level;
extern const char *debug_extra;
enum debug_level {
@@ -35,8 +33,4 @@ enum debug_level {
DEBUG_DEBUG = 4,
};
-#define DEBUGLVL(lvl) ((lvl) <= DEBUGLEVEL)
-#define DEBUG(lvl, x) do { this_log_level = (lvl); if ((lvl) <= DEBUGLEVEL) { do_debug x; }} while (0)
-#define DEBUGADD(lvl, x) do { if ((lvl) <= DEBUGLEVEL) { this_log_level = (lvl); do_debug_add x; }} while (0)
-
#endif /* _CTDB_LOGGING_H_ */
diff --git a/ctdb/lib/tdb_wrap/tdb_wrap.c b/ctdb/lib/tdb_wrap/tdb_wrap.c
index f39ac7a0911..57f8efc4e9f 100644
--- a/ctdb/lib/tdb_wrap/tdb_wrap.c
+++ b/ctdb/lib/tdb_wrap/tdb_wrap.c
@@ -50,12 +50,19 @@ static void log_fn(struct tdb_context *tdb, enum tdb_debug_level level, const ch
{
if (level <= TDB_DEBUG_ERROR) {
va_list ap;
- char newfmt[strlen(tdb_name(tdb)) + 1 + strlen(fmt) + 1];
- this_log_level = level;
- sprintf(newfmt, "%s:%s", tdb_name(tdb), fmt);
+ char *ptr = NULL;
+ int ret;
+
va_start(ap, fmt);
- do_debug_v(newfmt, ap);
+ ret = vasprintf(&ptr, fmt, ap);
va_end(ap);
+
+ if (ret != -1) {
+ const char *name = tdb_name(tdb);
+ DEBUG(level,
+ ("%s:%s", name ? name : "unnamed tdb", ptr));
+ free(ptr);
+ }
}
}
diff --git a/ctdb/lib/util/debug.c b/ctdb/lib/util/debug.c
index 40dda1a70a5..e15944403da 100644
--- a/ctdb/lib/util/debug.c
+++ b/ctdb/lib/util/debug.c
@@ -17,78 +17,13 @@
along with this program; if not, see .
*/
-#include "includes.h"
-#include "system/time.h"
-#include
+#include "replace.h"
+#include "system/filesys.h"
#include
+#include
+#include "debug.h"
-static void _do_debug_v(const char *format, va_list ap)
-{
- struct timeval t;
- char *s = NULL;
- struct tm *tm;
- char tbuf[100];
- int ret;
-
- ret = vasprintf(&s, format, ap);
- if (ret == -1) {
- fprintf(stderr, "vasprintf failed in _do_debug_v, cannot print debug message.\n");
- fflush(stderr);
- return;
- }
-
- t = timeval_current();
- tm = localtime(&t.tv_sec);
-
- strftime(tbuf,sizeof(tbuf)-1,"%Y/%m/%d %H:%M:%S", tm);
-
- fprintf(stderr, "%s.%06u [%s%5u]: %s", tbuf, (unsigned)t.tv_usec,
- debug_extra, (unsigned)getpid(), s);
- fflush(stderr);
- free(s);
-}
-
-/* default logging function */
-void (*do_debug_v)(const char *, va_list ap) = _do_debug_v;
-
-void do_debug(const char *format, ...)
-{
- va_list ap;
-
- va_start(ap, format);
- do_debug_v(format, ap);
- va_end(ap);
-}
-
-
-static void _do_debug_add_v(const char *format, va_list ap)
-{
- char *s = NULL;
- int ret;
-
- ret = vasprintf(&s, format, ap);
- if (ret == -1) {
- fprintf(stderr, "vasprintf failed in _do_debug_add_v, cannot print debug message.\n");
- fflush(stderr);
- return;
- }
-
- fprintf(stderr, "%s", s);
- fflush(stderr);
- free(s);
-}
-
-/* default logging function */
-void (*do_debug_add_v)(const char *, va_list ap) = _do_debug_add_v;
-
-void do_debug_add(const char *format, ...)
-{
- va_list ap;
-
- va_start(ap, format);
- do_debug_add_v(format, ap);
- va_end(ap);
-}
+int DEBUGLEVEL;
static void print_asc(int level, const uint8_t *buf, size_t len)
{
@@ -133,3 +68,48 @@ void dump_data(int level, const uint8_t *buf, size_t len)
DEBUG(level, (__location__ " dump data of size %i finished\n", (int)len));
}
+/* state variables for the debug system */
+static struct {
+ debug_callback_fn callback;
+ void *callback_private;
+} state;
+
+static int current_msg_level = 0;
+
+void debug_set_callback(void *private_ptr, debug_callback_fn fn)
+{
+ assert(fn != NULL);
+
+ state.callback_private = private_ptr;
+ state.callback = fn;
+}
+
+bool dbghdr(int level, const char *location, const char *func)
+{
+ current_msg_level = level;
+ return true;
+}
+
+bool dbgtext( const char *format_str, ... )
+{
+ va_list ap;
+ char *msgbuf = NULL;
+ int res;
+
+ va_start(ap, format_str);
+ res = vasprintf(&msgbuf, format_str, ap);
+ va_end(ap);
+ if (res == -1) {
+ return false;
+ }
+
+ if (state.callback != NULL) {
+ state.callback(state.callback_private,
+ current_msg_level, msgbuf);
+ } else {
+ write(2, msgbuf, strlen(msgbuf));
+ }
+
+ free(msgbuf);
+ return true;
+}
diff --git a/ctdb/lib/util/debug.h b/ctdb/lib/util/debug.h
index a590bc0f843..ac5e2c77cfa 100644
--- a/ctdb/lib/util/debug.h
+++ b/ctdb/lib/util/debug.h
@@ -20,10 +20,20 @@
#ifndef UTIL_DEBUG_H
#define UTIL_DEBUG_H
-extern void (*do_debug_v)(const char *, va_list ap);
-extern void (*do_debug_add_v)(const char *, va_list ap);
-void do_debug(const char *format, ...) PRINTF_ATTRIBUTE(1, 2);
-void do_debug_add(const char *format, ...) PRINTF_ATTRIBUTE(1, 2);
+bool dbgtext( const char *, ... ) PRINTF_ATTRIBUTE(1,2);
+bool dbghdr( int level, const char *location, const char *func);
void dump_data(int level, const uint8_t *buf1, size_t len);
+extern int DEBUGLEVEL;
+
+#define DEBUGLVL(lvl) ((lvl) <= DEBUGLEVEL)
+#define DEBUG( lvl, body ) \
+ (void)( ((lvl) <= DEBUGLEVEL) \
+ && (dbghdr( lvl, __location__, __FUNCTION__ )) \
+ && (dbgtext body) )
+#define DEBUGADD(lvl, body) DEBUG(lvl, body)
+
+typedef void (*debug_callback_fn)(void *private_ptr, int level, const char *msg);
+void debug_set_callback(void *private_ptr, debug_callback_fn fn);
+
#endif /* UTIL_DEBUG_H */
diff --git a/ctdb/server/ctdb_logging.c b/ctdb/server/ctdb_logging.c
index 97cc3beda6f..3aba37b81d5 100644
--- a/ctdb/server/ctdb_logging.c
+++ b/ctdb/server/ctdb_logging.c
@@ -23,6 +23,7 @@
#include "system/syslog.h"
#include "system/time.h"
#include "system/filesys.h"
+#include "lib/util/debug.h"
struct syslog_message {
uint32_t level;
@@ -208,21 +209,15 @@ static struct ctdb_log_state *log_state;
/*
syslog logging function
*/
-static void ctdb_syslog_log(const char *format, va_list ap)
+static void ctdb_syslog_log(void *private_ptr, int dbglevel, const char *s)
{
struct syslog_message *msg;
int level = LOG_DEBUG;
- char *s = NULL;
- int len, ret;
+ int len;
int syslog_fd;
struct sockaddr_in syslog_sin;
- ret = vasprintf(&s, format, ap);
- if (ret == -1) {
- return;
- }
-
- switch (this_log_level) {
+ switch (dbglevel) {
case DEBUG_EMERG:
level = LOG_EMERG;
break;
@@ -252,7 +247,6 @@ static void ctdb_syslog_log(const char *format, va_list ap)
len = offsetof(struct syslog_message, message) + strlen(debug_extra) + strlen(s) + 1;
msg = malloc(len);
if (msg == NULL) {
- free(s);
return;
}
msg->level = level;
@@ -266,7 +260,6 @@ static void ctdb_syslog_log(const char *format, va_list ap)
syslog_fd = socket(AF_INET, SOCK_DGRAM, IPPROTO_UDP);
if (syslog_fd == -1) {
printf("Failed to create syslog socket\n");
- free(s);
free(msg);
return;
}
@@ -275,15 +268,14 @@ static void ctdb_syslog_log(const char *format, va_list ap)
syslog_sin.sin_port = htons(CTDB_PORT);
syslog_sin.sin_addr.s_addr = htonl(INADDR_LOOPBACK);
- ret = sendto(syslog_fd, msg, len, 0,
- (struct sockaddr *)&syslog_sin,
- sizeof(syslog_sin));
+ (void) sendto(syslog_fd, msg, len, 0,
+ (struct sockaddr *)&syslog_sin,
+ sizeof(syslog_sin));
/* no point in checking here since we cant log an error */
close(syslog_fd);
}
- free(s);
free(msg);
}
@@ -291,23 +283,14 @@ static void ctdb_syslog_log(const char *format, va_list ap)
/*
log file logging function
*/
-static void ctdb_logfile_log(const char *format, va_list ap)
+static void ctdb_logfile_log(void *private_ptr, int dbglevel, const char *s)
{
struct timeval t;
- char *s = NULL;
struct tm *tm;
char tbuf[100];
char *s2 = NULL;
int ret;
- ret = vasprintf(&s, format, ap);
- if (ret == -1) {
- const char *errstr = "vasprintf failed\n";
-
- sys_write(log_state->fd, errstr, strlen(errstr));
- return;
- }
-
t = timeval_current();
tm = localtime(&t.tv_sec);
@@ -316,7 +299,6 @@ static void ctdb_logfile_log(const char *format, va_list ap)
ret = asprintf(&s2, "%s.%06u [%s%5u]: %s",
tbuf, (unsigned)t.tv_usec,
debug_extra, (unsigned)getpid(), s);
- free(s);
if (ret == -1) {
const char *errstr = "asprintf failed\n";
sys_write(log_state->fd, errstr, strlen(errstr));
@@ -328,32 +310,12 @@ static void ctdb_logfile_log(const char *format, va_list ap)
}
}
-static void ctdb_logfile_log_add(const char *format, va_list ap)
-{
- char *s = NULL;
- int ret;
-
- ret = vasprintf(&s, format, ap);
- if (ret == -1) {
- const char *errstr = "vasprintf failed\n";
-
- sys_write(log_state->fd, errstr, strlen(errstr));
- return;
- }
-
- if (s) {
- sys_write(log_state->fd, s, strlen(s));
- free(s);
- }
-}
-
-
-
/*
choose the logfile location
*/
int ctdb_set_logfile(struct ctdb_context *ctdb, const char *logfile, bool use_syslog)
{
+ debug_callback_fn callback;
int ret;
ctdb->log = talloc_zero(ctdb, struct ctdb_log_state);
@@ -366,12 +328,10 @@ int ctdb_set_logfile(struct ctdb_context *ctdb, const char *logfile, bool use_sy
log_state = ctdb->log;
if (use_syslog) {
- do_debug_v = ctdb_syslog_log;
- do_debug_add_v = ctdb_syslog_log;
+ callback = ctdb_syslog_log;
ctdb->log->use_syslog = true;
} else if (logfile == NULL || strcmp(logfile, "-") == 0) {
- do_debug_v = ctdb_logfile_log;
- do_debug_add_v = ctdb_logfile_log_add;
+ callback = ctdb_logfile_log;
ctdb->log->fd = 1;
/* also catch stderr of subcommands to stdout */
ret = dup2(1, 2);
@@ -380,8 +340,7 @@ int ctdb_set_logfile(struct ctdb_context *ctdb, const char *logfile, bool use_sy
abort();
}
} else {
- do_debug_v = ctdb_logfile_log;
- do_debug_add_v = ctdb_logfile_log_add;
+ callback = ctdb_logfile_log;
ctdb->log->fd = open(logfile, O_WRONLY|O_APPEND|O_CREAT, 0666);
if (ctdb->log->fd == -1) {
@@ -390,6 +349,8 @@ int ctdb_set_logfile(struct ctdb_context *ctdb, const char *logfile, bool use_sy
}
}
+ debug_set_callback(NULL, callback);
+
return 0;
}
@@ -399,9 +360,9 @@ static void write_to_log(struct ctdb_log_state *log,
{
if (script_log_level <= DEBUGLEVEL) {
if (log != NULL && log->prefix != NULL) {
- do_debug("%s: %*.*s\n", log->prefix, len, len, buf);
+ dbgtext("%s: %*.*s\n", log->prefix, len, len, buf);
} else {
- do_debug("%*.*s\n", len, len, buf);
+ dbgtext("%*.*s\n", len, len, buf);
}
/* log it in the eventsystem as well */
if (log && log->logfn) {
@@ -435,8 +396,6 @@ static void ctdb_log_handler(struct event_context *ev, struct fd_event *fde,
return;
}
- this_log_level = script_log_level;
-
while (log->buf_used > 0 &&
(p = memchr(log->buf, '\n', log->buf_used)) != NULL) {
int n1 = (p - log->buf)+1;
@@ -462,7 +421,6 @@ static int log_context_destructor(struct ctdb_log_state *log)
{
/* Flush buffer in case it wasn't \n-terminated. */
if (log->buf_used > 0) {
- this_log_level = script_log_level;
write_to_log(log, log->buf, log->buf_used);
}
return 0;
@@ -629,8 +587,7 @@ static void ctdb_tevent_logging(void *private_data,
}
if (lvl <= DEBUGLEVEL) {
- this_log_level = lvl;
- do_debug_v(fmt, ap);
+ dbgtext(fmt, ap);
}
}