1
0
mirror of https://github.com/samba-team/samba.git synced 2025-01-10 01:18:15 +03:00
samba-mirror/lib/util/debug.c
Andrew Bartlett 83fe7a0316 lib/util: Add "debug syslog format = always", which logs to stdout in syslog style
Signed-off-by: Andrew Bartlett <abartlet@samba.org>
Reviewed-by: Stefan Metzmacher <metze@samba.org>
2023-04-06 12:51:30 +00:00

1998 lines
48 KiB
C

/*
Unix SMB/CIFS implementation.
Samba utility functions
Copyright (C) Andrew Tridgell 1992-1998
Copyright (C) Elrond 2002
Copyright (C) Simo Sorce 2002
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 of the License, 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 <http://www.gnu.org/licenses/>.
*/
#include "replace.h"
#include <talloc.h>
#include "system/filesys.h"
#include "system/syslog.h"
#include "system/locale.h"
#include "system/network.h"
#include "system/time.h"
#include "time_basic.h"
#include "close_low_fd.h"
#include "memory.h"
#include "util_strlist.h" /* LIST_SEP */
#include "blocking.h"
#include "debug.h"
#include <assert.h>
/* define what facility to use for syslog */
#ifndef SYSLOG_FACILITY
#define SYSLOG_FACILITY LOG_DAEMON
#endif
/* -------------------------------------------------------------------------- **
* Defines...
*/
/*
* format_bufr[FORMAT_BUFR_SIZE - 1] should always be reserved
* for a terminating null byte.
*/
#define FORMAT_BUFR_SIZE 4096
/* -------------------------------------------------------------------------- **
* This module implements Samba's debugging utility.
*
* The syntax of a debugging log file is represented as:
*
* <debugfile> :== { <debugmsg> }
*
* <debugmsg> :== <debughdr> '\n' <debugtext>
*
* <debughdr> :== '[' TIME ',' LEVEL ']' [ [FILENAME ':'] [FUNCTION '()'] ]
*
* <debugtext> :== { <debugline> }
*
* <debugline> :== TEXT '\n'
*
* TEXT is a string of characters excluding the newline character.
* LEVEL is the DEBUG level of the message (an integer in the range 0..10).
* TIME is a timestamp.
* FILENAME is the name of the file from which the debug message was generated.
* FUNCTION is the function from which the debug message was generated.
*
* Basically, what that all means is:
*
* - A debugging log file is made up of debug messages.
*
* - Each debug message is made up of a header and text. The header is
* separated from the text by a newline.
*
* - The header begins with the timestamp and debug level of the message
* enclosed in brackets. The filename and function from which the
* message was generated may follow. The filename is terminated by a
* colon, and the function name is terminated by parenthesis.
*
* - The message text is made up of zero or more lines, each terminated by
* a newline.
*/
/* state variables for the debug system */
static struct {
bool initialized;
enum debug_logtype logtype; /* The type of logging we are doing: eg stdout, file, stderr */
char prog_name[255];
char hostname[HOST_NAME_MAX+1];
bool reopening_logs;
bool schedule_reopen_logs;
struct debug_settings settings;
debug_callback_fn callback;
void *callback_private;
char header_str[300];
char header_str_no_nl[300];
size_t hs_len;
char msg_no_nl[FORMAT_BUFR_SIZE];
} state = {
.settings = {
.timestamp_logs = true
},
};
struct debug_class {
/*
* The debug loglevel of the class.
*/
int loglevel;
/*
* An optional class specific logfile, may be NULL in which case the
* "global" logfile is used and fd is -1.
*/
char *logfile;
int fd;
/* inode number of the logfile to detect logfile rotation */
ino_t ino;
};
/*
* default_classname_table[] is read in from debug-classname-table.c
* so that test_logging.c can use it too.
*/
#include "lib/util/debug-classes/debug-classname-table.c"
/*
* This is to allow reading of dbgc_config before the debug
* system has been initialized.
*/
static struct debug_class debug_class_list_initial[ARRAY_SIZE(default_classname_table)] = {
[DBGC_ALL] = { .fd = 2 },
};
static size_t debug_num_classes = 0;
static struct debug_class *dbgc_config = debug_class_list_initial;
static int current_msg_level = 0;
static int current_msg_class = 0;
/*
* DBG_DEV(): when and how to user it.
*
* As a developer, you sometimes want verbose logging between point A and
* point B, where the relationship between these points is not easily defined
* in terms of the call stack.
*
* For example, you might be interested in what is going on in functions in
* lib/util/util_str.c in an ldap worker process after a particular query. If
* you use gdb, something will time out and you won't get the full
* conversation. If you add fprintf() or DBG_ERR()s to util_str.c, you'll get
* a massive flood, and there's a chance one will accidentally slip into a
* release and the whole world will flood. DBG_DEV is a solution.
*
* On start-up, DBG_DEV() is switched OFF. Nothing is printed.
*
* 1. Add `DBG_DEV("formatted msg %d, etc\n", i);` where needed.
*
* 2. At each point you want to start debugging, add `debug_developer_enable()`.
*
* 3. At each point you want debugging to stop, add `debug_developer_disable()`.
*
* In DEVELOPER builds, the message will be printed at level 0, as with
* DBG_ERR(). In production builds, the macro resolves to nothing.
*
* The messages are printed with a "<function_name>:DEV:<pid>:" prefix.
*/
static bool debug_developer_is_enabled = false;
bool debug_developer_enabled(void)
{
return debug_developer_is_enabled;
}
/*
* debug_developer_disable() will turn DBG_DEV() on in the current
* process and children.
*/
void debug_developer_enable(void)
{
debug_developer_is_enabled = true;
}
/*
* debug_developer_disable() will make DBG_DEV() do nothing in the current
* process (and children).
*/
void debug_developer_disable(void)
{
debug_developer_is_enabled = false;
}
/*
* Within debug.c, DBG_DEV() always writes to stderr, because some functions
* here will attempt infinite recursion with normal DEBUG macros.
*/
#ifdef DEVELOPER
#undef DBG_DEV
#define DBG_DEV(fmt, ...) \
(void)((debug_developer_enabled()) \
&& (fprintf(stderr, "%s:DEV:%d: " fmt "%s", \
__func__, getpid(), ##__VA_ARGS__, "")) )
#endif
#if defined(WITH_SYSLOG) || defined(HAVE_LIBSYSTEMD_JOURNAL) || defined(HAVE_LIBSYSTEMD)
static int debug_level_to_priority(int level)
{
/*
* map debug levels to syslog() priorities
*/
static const int priority_map[] = {
LOG_ERR, /* 0 */
LOG_WARNING, /* 1 */
LOG_NOTICE, /* 2 */
LOG_NOTICE, /* 3 */
LOG_NOTICE, /* 4 */
LOG_NOTICE, /* 5 */
LOG_INFO, /* 6 */
LOG_INFO, /* 7 */
LOG_INFO, /* 8 */
LOG_INFO, /* 9 */
};
int priority;
if (level < 0 || (size_t)level >= ARRAY_SIZE(priority_map))
priority = LOG_DEBUG;
else
priority = priority_map[level];
return priority;
}
#endif
/* -------------------------------------------------------------------------- **
* Produce a version of the given buffer without any trailing newlines.
*/
#if defined(HAVE_LIBSYSTEMD_JOURNAL) || defined(HAVE_LIBSYSTEMD) || \
defined(HAVE_LTTNG_TRACEF) || defined(HAVE_GPFS)
static void copy_no_nl(char *out,
size_t out_size,
const char *in,
size_t in_len)
{
size_t len;
/*
* Some backends already add an extra newline, so also provide
* a buffer without the newline character.
*/
len = MIN(in_len, out_size - 1);
if ((len > 0) && (in[len - 1] == '\n')) {
len--;
}
memcpy(out, in, len);
out[len] = '\0';
}
static void ensure_copy_no_nl(char *out,
size_t out_size,
const char *in,
size_t in_len)
{
/*
* Assume out is a static buffer that is reused as a cache.
* If it isn't empty then this has already been done with the
* same input.
*/
if (out[0] != '\0') {
return;
}
copy_no_nl(out, out_size, in, in_len);
}
#endif
/* -------------------------------------------------------------------------- **
* Debug backends. When logging to DEBUG_FILE, send the log entries to
* all active backends.
*/
static void debug_file_log(int msg_level, const char *msg, size_t msg_len)
{
struct iovec iov[] = {
{
.iov_base = discard_const(state.header_str),
.iov_len = state.hs_len,
},
{
.iov_base = discard_const(msg),
.iov_len = msg_len,
},
};
ssize_t ret;
int fd;
check_log_size();
if (dbgc_config[current_msg_class].fd != -1) {
fd = dbgc_config[current_msg_class].fd;
} else {
fd = dbgc_config[DBGC_ALL].fd;
}
do {
ret = writev(fd, iov, ARRAY_SIZE(iov));
} while (ret == -1 && errno == EINTR);
}
#ifdef WITH_SYSLOG
static void debug_syslog_reload(bool enabled, bool previously_enabled,
const char *prog_name, char *option)
{
if (enabled && !previously_enabled) {
const char *ident = NULL;
if ((prog_name != NULL) && (prog_name[0] != '\0')) {
ident = prog_name;
}
#ifdef LOG_DAEMON
openlog(ident, LOG_PID, SYSLOG_FACILITY);
#else
/* for old systems that have no facility codes. */
openlog(ident, LOG_PID);
#endif
return;
}
if (!enabled && previously_enabled) {
closelog();
}
}
static void debug_syslog_log(int msg_level, const char *msg, size_t msg_len)
{
int priority;
priority = debug_level_to_priority(msg_level);
/*
* Specify the facility to interoperate with other syslog
* callers (vfs_full_audit for example).
*/
priority |= SYSLOG_FACILITY;
if (state.hs_len > 0) {
syslog(priority, "%s", state.header_str);
}
syslog(priority, "%s", msg);
}
#endif /* WITH_SYSLOG */
#if defined(HAVE_LIBSYSTEMD_JOURNAL) || defined(HAVE_LIBSYSTEMD)
#include <systemd/sd-journal.h>
static void debug_systemd_log(int msg_level, const char *msg, size_t msg_len)
{
if (state.hs_len > 0) {
ensure_copy_no_nl(state.header_str_no_nl,
sizeof(state.header_str_no_nl),
state.header_str,
state.hs_len);
sd_journal_send("MESSAGE=%s",
state.header_str_no_nl,
"PRIORITY=%d",
debug_level_to_priority(msg_level),
"LEVEL=%d",
msg_level,
NULL);
}
ensure_copy_no_nl(state.msg_no_nl,
sizeof(state.msg_no_nl),
msg, msg_len);
sd_journal_send("MESSAGE=%s", state.msg_no_nl,
"PRIORITY=%d", debug_level_to_priority(msg_level),
"LEVEL=%d", msg_level,
NULL);
}
#endif
#ifdef HAVE_LTTNG_TRACEF
#include <lttng/tracef.h>
static void debug_lttng_log(int msg_level, const char *msg, size_t msg_len)
{
if (state.hs_len > 0) {
ensure_copy_no_nl(state.header_str_no_nl,
sizeof(state.header_str_no_nl),
state.header_str,
state.hs_len);
tracef(state.header_str_no_nl);
}
ensure_copy_no_nl(state.msg_no_nl,
sizeof(state.msg_no_nl),
msg, msg_len);
tracef(state.msg_no_nl);
}
#endif /* WITH_LTTNG_TRACEF */
#ifdef HAVE_GPFS
#include "gpfswrap.h"
static void debug_gpfs_reload(bool enabled, bool previously_enabled,
const char *prog_name, char *option)
{
gpfswrap_init();
if (enabled && !previously_enabled) {
gpfswrap_init_trace();
return;
}
if (!enabled && previously_enabled) {
gpfswrap_fini_trace();
return;
}
if (enabled) {
/*
* Trigger GPFS library to adjust state if necessary.
*/
gpfswrap_query_trace();
}
}
static void debug_gpfs_log(int msg_level, const char *msg, size_t msg_len)
{
if (state.hs_len > 0) {
ensure_copy_no_nl(state.header_str_no_nl,
sizeof(state.header_str_no_nl),
state.header_str,
state.hs_len);
gpfswrap_add_trace(msg_level, state.header_str_no_nl);
}
ensure_copy_no_nl(state.msg_no_nl,
sizeof(state.msg_no_nl),
msg, msg_len);
gpfswrap_add_trace(msg_level, state.msg_no_nl);
}
#endif /* HAVE_GPFS */
#define DEBUG_RINGBUF_SIZE (1024 * 1024)
#define DEBUG_RINGBUF_SIZE_OPT "size="
static char *debug_ringbuf;
static size_t debug_ringbuf_size;
static size_t debug_ringbuf_ofs;
/* We ensure in debug_ringbuf_log() that this is always \0 terminated */
char *debug_get_ringbuf(void)
{
return debug_ringbuf;
}
/* Return the size of the ringbuf (including a \0 terminator) */
size_t debug_get_ringbuf_size(void)
{
return debug_ringbuf_size;
}
static void debug_ringbuf_reload(bool enabled, bool previously_enabled,
const char *prog_name, char *option)
{
bool cmp;
size_t optlen = strlen(DEBUG_RINGBUF_SIZE_OPT);
debug_ringbuf_size = DEBUG_RINGBUF_SIZE;
debug_ringbuf_ofs = 0;
SAFE_FREE(debug_ringbuf);
if (!enabled) {
return;
}
if (option != NULL) {
cmp = strncmp(option, DEBUG_RINGBUF_SIZE_OPT, optlen);
if (cmp == 0) {
debug_ringbuf_size = (size_t)strtoull(
option + optlen, NULL, 10);
}
}
debug_ringbuf = calloc(debug_ringbuf_size, sizeof(char));
if (debug_ringbuf == NULL) {
return;
}
}
static void _debug_ringbuf_log(int msg_level, const char *msg, size_t msg_len)
{
size_t allowed_size;
if (debug_ringbuf == NULL) {
return;
}
/* Ensure the buffer is always \0 terminated */
allowed_size = debug_ringbuf_size - 1;
if (msg_len > allowed_size) {
return;
}
if ((debug_ringbuf_ofs + msg_len) < debug_ringbuf_ofs) {
return;
}
if ((debug_ringbuf_ofs + msg_len) > allowed_size) {
debug_ringbuf_ofs = 0;
}
memcpy(debug_ringbuf + debug_ringbuf_ofs, msg, msg_len);
debug_ringbuf_ofs += msg_len;
}
static void debug_ringbuf_log(int msg_level, const char *msg, size_t msg_len)
{
if (state.hs_len > 0) {
_debug_ringbuf_log(msg_level, state.header_str, state.hs_len);
}
_debug_ringbuf_log(msg_level, msg, msg_len);
}
static struct debug_backend {
const char *name;
int log_level;
int new_log_level;
void (*reload)(bool enabled, bool prev_enabled,
const char *prog_name, char *option);
void (*log)(int msg_level,
const char *msg,
size_t len);
char *option;
} debug_backends[] = {
{
.name = "file",
.log = debug_file_log,
},
#ifdef WITH_SYSLOG
{
.name = "syslog",
.reload = debug_syslog_reload,
.log = debug_syslog_log,
},
#endif
#if defined(HAVE_LIBSYSTEMD_JOURNAL) || defined(HAVE_LIBSYSTEMD)
{
.name = "systemd",
.log = debug_systemd_log,
},
#endif
#ifdef HAVE_LTTNG_TRACEF
{
.name = "lttng",
.log = debug_lttng_log,
},
#endif
#ifdef HAVE_GPFS
{
.name = "gpfs",
.reload = debug_gpfs_reload,
.log = debug_gpfs_log,
},
#endif
{
.name = "ringbuf",
.log = debug_ringbuf_log,
.reload = debug_ringbuf_reload,
},
};
static struct debug_backend *debug_find_backend(const char *name)
{
unsigned i;
for (i = 0; i < ARRAY_SIZE(debug_backends); i++) {
if (strcmp(name, debug_backends[i].name) == 0) {
return &debug_backends[i];
}
}
return NULL;
}
/*
* parse "backend[:option][@loglevel]
*/
static void debug_backend_parse_token(char *tok)
{
char *backend_name_option, *backend_name,*backend_level, *saveptr;
char *backend_option;
struct debug_backend *b;
/*
* First parse into backend[:option] and loglevel
*/
backend_name_option = strtok_r(tok, "@\0", &saveptr);
if (backend_name_option == NULL) {
return;
}
backend_level = strtok_r(NULL, "\0", &saveptr);
/*
* Now parse backend[:option]
*/
backend_name = strtok_r(backend_name_option, ":\0", &saveptr);
if (backend_name == NULL) {
return;
}
backend_option = strtok_r(NULL, "\0", &saveptr);
/*
* Find and update backend
*/
b = debug_find_backend(backend_name);
if (b == NULL) {
return;
}
if (backend_level == NULL) {
b->new_log_level = MAX_DEBUG_LEVEL;
} else {
b->new_log_level = atoi(backend_level);
}
if (backend_option != NULL) {
b->option = strdup(backend_option);
if (b->option == NULL) {
return;
}
}
}
/*
* parse "backend1[:option1][@loglevel1] backend2[option2][@loglevel2] ... "
* and enable/disable backends accordingly
*/
static void debug_set_backends(const char *param)
{
size_t str_len = strlen(param);
char str[str_len+1];
char *tok, *saveptr;
unsigned i;
/*
* initialize new_log_level to detect backends that have been
* disabled
*/
for (i = 0; i < ARRAY_SIZE(debug_backends); i++) {
SAFE_FREE(debug_backends[i].option);
debug_backends[i].new_log_level = -1;
}
memcpy(str, param, str_len + 1);
tok = strtok_r(str, LIST_SEP, &saveptr);
if (tok == NULL) {
return;
}
while (tok != NULL) {
debug_backend_parse_token(tok);
tok = strtok_r(NULL, LIST_SEP, &saveptr);
}
/*
* Let backends react to config changes
*/
for (i = 0; i < ARRAY_SIZE(debug_backends); i++) {
struct debug_backend *b = &debug_backends[i];
if (b->reload) {
bool enabled = b->new_log_level > -1;
bool previously_enabled = b->log_level > -1;
b->reload(enabled, previously_enabled, state.prog_name,
b->option);
}
b->log_level = b->new_log_level;
}
}
static void debug_backends_log(const char *msg, size_t msg_len, int msg_level)
{
size_t i;
/*
* Some backends already add an extra newline, so initialize a
* buffer without the newline character. It will be filled by
* the first backend that needs it.
*/
state.msg_no_nl[0] = '\0';
for (i = 0; i < ARRAY_SIZE(debug_backends); i++) {
if (msg_level <= debug_backends[i].log_level) {
debug_backends[i].log(msg_level, msg, msg_len);
}
}
/* Only log the header once */
state.hs_len = 0;
}
int debuglevel_get_class(size_t idx)
{
return dbgc_config[idx].loglevel;
}
void debuglevel_set_class(size_t idx, int level)
{
dbgc_config[idx].loglevel = level;
}
/* -------------------------------------------------------------------------- **
* Internal variables.
*
* debug_count - Number of debug messages that have been output.
* Used to check log size.
*
* current_msg_level - Internal copy of the message debug level. Written by
* dbghdr() and read by Debug1().
*
* format_bufr - Used to format debug messages. The dbgtext() function
* prints debug messages to a string, and then passes the
* string to format_debug_text(), which uses format_bufr
* to build the formatted output.
*
* format_pos - Marks the first free byte of the format_bufr.
*
*
* log_overflow - When this variable is true, never attempt to check the
* size of the log. This is a hack, so that we can write
* a message using DEBUG, from open_logs() when we
* are unable to open a new log file for some reason.
*/
static int debug_count = 0;
static char format_bufr[FORMAT_BUFR_SIZE];
static size_t format_pos = 0;
static bool log_overflow = false;
/*
* Define all the debug class selection names here. Names *MUST NOT* contain
* white space. There must be one name for each DBGC_<class name>, and they
* must be in the table in the order of DBGC_<class name>..
*/
static char **classname_table = NULL;
/* -------------------------------------------------------------------------- **
* Functions...
*/
static void debug_init(void);
/***************************************************************************
Free memory pointed to by global pointers.
****************************************************************************/
void gfree_debugsyms(void)
{
unsigned i;
TALLOC_FREE(classname_table);
if ( dbgc_config != debug_class_list_initial ) {
TALLOC_FREE( dbgc_config );
dbgc_config = discard_const_p(struct debug_class,
debug_class_list_initial);
}
debug_num_classes = 0;
state.initialized = false;
for (i = 0; i < ARRAY_SIZE(debug_backends); i++) {
SAFE_FREE(debug_backends[i].option);
}
}
/****************************************************************************
utility lists registered debug class names's
****************************************************************************/
char *debug_list_class_names_and_levels(void)
{
char *buf = talloc_strdup(NULL, "");
size_t i;
/* prepare strings */
for (i = 0; i < debug_num_classes; i++) {
talloc_asprintf_addbuf(&buf,
"%s:%d%s",
classname_table[i],
dbgc_config[i].loglevel,
i == (debug_num_classes - 1) ? "\n" : " ");
}
return buf;
}
/****************************************************************************
Utility to translate names to debug class index's (internal version).
****************************************************************************/
static int debug_lookup_classname_int(const char* classname)
{
size_t i;
if (classname == NULL) {
return -1;
}
for (i=0; i < debug_num_classes; i++) {
char *entry = classname_table[i];
if (entry != NULL && strcmp(classname, entry)==0) {
return i;
}
}
return -1;
}
/****************************************************************************
Add a new debug class to the system.
****************************************************************************/
int debug_add_class(const char *classname)
{
int ndx;
struct debug_class *new_class_list = NULL;
char **new_name_list;
int default_level;
if (classname == NULL) {
return -1;
}
/* check the init has yet been called */
debug_init();
ndx = debug_lookup_classname_int(classname);
if (ndx >= 0) {
return ndx;
}
ndx = debug_num_classes;
if (dbgc_config == debug_class_list_initial) {
/* Initial loading... */
new_class_list = NULL;
} else {
new_class_list = dbgc_config;
}
default_level = dbgc_config[DBGC_ALL].loglevel;
new_class_list = talloc_realloc(NULL,
new_class_list,
struct debug_class,
ndx + 1);
if (new_class_list == NULL) {
return -1;
}
dbgc_config = new_class_list;
dbgc_config[ndx] = (struct debug_class) {
.loglevel = default_level,
.fd = -1,
};
new_name_list = talloc_realloc(NULL, classname_table, char *, ndx + 1);
if (new_name_list == NULL) {
return -1;
}
classname_table = new_name_list;
classname_table[ndx] = talloc_strdup(classname_table, classname);
if (classname_table[ndx] == NULL) {
return -1;
}
debug_num_classes = ndx + 1;
return ndx;
}
/****************************************************************************
Utility to translate names to debug class index's (public version).
****************************************************************************/
static int debug_lookup_classname(const char *classname)
{
int ndx;
if (classname == NULL || !*classname)
return -1;
ndx = debug_lookup_classname_int(classname);
if (ndx != -1)
return ndx;
DBG_WARNING("Unknown classname[%s] -> adding it...\n", classname);
return debug_add_class(classname);
}
/****************************************************************************
Dump the current registered debug levels.
****************************************************************************/
static void debug_dump_status(int level)
{
size_t q;
DEBUG(level, ("INFO: Current debug levels:\n"));
for (q = 0; q < debug_num_classes; q++) {
const char *classname = classname_table[q];
DEBUGADD(level, (" %s: %d\n",
classname,
dbgc_config[q].loglevel));
}
}
static bool debug_parse_param(char *param)
{
char *class_name;
char *class_file = NULL;
char *class_level;
char *saveptr = NULL;
int ndx;
class_name = strtok_r(param, ":", &saveptr);
if (class_name == NULL) {
return false;
}
class_level = strtok_r(NULL, "@\0", &saveptr);
if (class_level == NULL) {
return false;
}
class_file = strtok_r(NULL, "\0", &saveptr);
ndx = debug_lookup_classname(class_name);
if (ndx == -1) {
return false;
}
dbgc_config[ndx].loglevel = atoi(class_level);
if (class_file == NULL) {
return true;
}
TALLOC_FREE(dbgc_config[ndx].logfile);
dbgc_config[ndx].logfile = talloc_strdup(NULL, class_file);
if (dbgc_config[ndx].logfile == NULL) {
return false;
}
return true;
}
/****************************************************************************
Parse the debug levels from smb.conf. Example debug level string:
3 tdb:5 printdrivers:7
Note: the 1st param has no "name:" preceding it.
****************************************************************************/
bool debug_parse_levels(const char *params_str)
{
size_t str_len = strlen(params_str);
char str[str_len+1];
char *tok, *saveptr;
size_t i;
/* Just in case */
debug_init();
memcpy(str, params_str, str_len+1);
tok = strtok_r(str, LIST_SEP, &saveptr);
if (tok == NULL) {
return true;
}
/* Allow DBGC_ALL to be specified w/o requiring its class name e.g."10"
* v.s. "all:10", this is the traditional way to set DEBUGLEVEL
*/
if (isdigit(tok[0])) {
dbgc_config[DBGC_ALL].loglevel = atoi(tok);
tok = strtok_r(NULL, LIST_SEP, &saveptr);
} else {
dbgc_config[DBGC_ALL].loglevel = 0;
}
/* Array is debug_num_classes long */
for (i = DBGC_ALL+1; i < debug_num_classes; i++) {
dbgc_config[i].loglevel = dbgc_config[DBGC_ALL].loglevel;
TALLOC_FREE(dbgc_config[i].logfile);
}
while (tok != NULL) {
bool ok;
ok = debug_parse_param(tok);
if (!ok) {
DEBUG(0,("debug_parse_params: unrecognized debug "
"class name or format [%s]\n", tok));
return false;
}
tok = strtok_r(NULL, LIST_SEP, &saveptr);
}
debug_dump_status(5);
return true;
}
/* setup for logging of talloc warnings */
static void talloc_log_fn(const char *msg)
{
DEBUG(0,("%s", msg));
}
void debug_setup_talloc_log(void)
{
talloc_set_log_fn(talloc_log_fn);
}
/****************************************************************************
Init debugging (one time stuff)
****************************************************************************/
static void debug_init(void)
{
size_t i;
if (state.initialized)
return;
state.initialized = true;
debug_setup_talloc_log();
for (i = 0; i < ARRAY_SIZE(default_classname_table); i++) {
debug_add_class(default_classname_table[i]);
}
dbgc_config[DBGC_ALL].fd = 2;
for (i = 0; i < ARRAY_SIZE(debug_backends); i++) {
debug_backends[i].log_level = -1;
debug_backends[i].new_log_level = -1;
}
}
void debug_set_settings(struct debug_settings *settings,
const char *logging_param,
int syslog_level, bool syslog_only)
{
char fake_param[256];
size_t len = 0;
/*
* This forces in some smb.conf derived values into the debug
* system. There are no pointers in this structure, so we can
* just structure-assign it in
*/
state.settings = *settings;
/*
* If 'logging' is not set, create backend settings from
* deprecated 'syslog' and 'syslog only' parameters
*/
if (logging_param != NULL) {
len = strlen(logging_param);
}
if (len == 0) {
if (syslog_only) {
snprintf(fake_param, sizeof(fake_param),
"syslog@%d", syslog_level - 1);
} else {
snprintf(fake_param, sizeof(fake_param),
"syslog@%d file@%d", syslog_level -1,
MAX_DEBUG_LEVEL);
}
logging_param = fake_param;
}
debug_set_backends(logging_param);
}
static void ensure_hostname(void)
{
int ret;
if (state.hostname[0] != '\0') {
return;
}
ret = gethostname(state.hostname, sizeof(state.hostname));
if (ret != 0) {
strlcpy(state.hostname, "unknown", sizeof(state.hostname));
return;
}
/*
* Ensure NUL termination, since POSIX isn't clear about that.
*
* Don't worry about truncating at the first '.' or similar,
* since this is usually not fully qualified. Trying to
* truncate opens up the multibyte character gates of hell.
*/
state.hostname[sizeof(state.hostname) - 1] = '\0';
}
void debug_set_hostname(const char *name)
{
strlcpy(state.hostname, name, sizeof(state.hostname));
}
/**
* Ensure debug logs are initialised.
*
* setup_logging() is called to direct logging to the correct outputs, whether
* those be stderr, stdout, files, or syslog, and set the program name used in
* the logs. It can be called multiple times.
*
* There is an order of precedence to the log type. Once set to DEBUG_FILE, it
* cannot be reset DEFAULT_DEBUG_STDERR, but can be set to DEBUG_STDERR, after
* which DEBUG_FILE is unavailable). This makes it possible to override for
* debug to stderr on the command line, as the smb.conf cannot reset it back
* to file-based logging. See enum debug_logtype.
*
* @param prog_name the program name. Directory path component will be
* ignored.
*
* @param new_logtype the requested destination for the debug log,
* as an enum debug_logtype.
*/
void setup_logging(const char *prog_name, enum debug_logtype new_logtype)
{
debug_init();
if (state.logtype < new_logtype) {
state.logtype = new_logtype;
}
if (prog_name) {
const char *p = strrchr(prog_name, '/');
if (p) {
prog_name = p + 1;
}
strlcpy(state.prog_name, prog_name, sizeof(state.prog_name));
}
reopen_logs_internal();
}
/***************************************************************************
Set the logfile name.
**************************************************************************/
void debug_set_logfile(const char *name)
{
if (name == NULL || *name == 0) {
/* this copes with calls when smb.conf is not loaded yet */
return;
}
TALLOC_FREE(dbgc_config[DBGC_ALL].logfile);
dbgc_config[DBGC_ALL].logfile = talloc_strdup(NULL, name);
reopen_logs_internal();
}
static void debug_close_fd(int fd)
{
if (fd > 2) {
close(fd);
}
}
enum debug_logtype debug_get_log_type(void)
{
return state.logtype;
}
bool debug_get_output_is_stderr(void)
{
return (state.logtype == DEBUG_DEFAULT_STDERR) || (state.logtype == DEBUG_STDERR);
}
bool debug_get_output_is_stdout(void)
{
return (state.logtype == DEBUG_DEFAULT_STDOUT) || (state.logtype == DEBUG_STDOUT);
}
void debug_set_callback(void *private_ptr, debug_callback_fn fn)
{
debug_init();
if (fn) {
state.logtype = DEBUG_CALLBACK;
state.callback_private = private_ptr;
state.callback = fn;
} else {
state.logtype = DEBUG_DEFAULT_STDERR;
state.callback_private = NULL;
state.callback = NULL;
}
}
static void debug_callback_log(const char *msg, size_t msg_len, int msg_level)
{
char msg_copy[msg_len];
if ((msg_len > 0) && (msg[msg_len-1] == '\n')) {
memcpy(msg_copy, msg, msg_len-1);
msg_copy[msg_len-1] = '\0';
msg = msg_copy;
}
state.callback(state.callback_private, msg_level, msg);
}
/**************************************************************************
reopen the log files
note that we now do this unconditionally
We attempt to open the new debug fp before closing the old. This means
if we run out of fd's we just keep using the old fd rather than aborting.
Fix from dgibson@linuxcare.com.
**************************************************************************/
static bool reopen_one_log(struct debug_class *config)
{
int old_fd = config->fd;
const char *logfile = config->logfile;
struct stat st;
int new_fd;
int ret;
if (logfile == NULL) {
debug_close_fd(old_fd);
config->fd = -1;
return true;
}
new_fd = open(logfile, O_WRONLY|O_APPEND|O_CREAT, 0644);
if (new_fd == -1) {
log_overflow = true;
DBG_ERR("Unable to open new log file '%s': %s\n",
logfile, strerror(errno));
log_overflow = false;
return false;
}
debug_close_fd(old_fd);
smb_set_close_on_exec(new_fd);
config->fd = new_fd;
ret = fstat(new_fd, &st);
if (ret != 0) {
log_overflow = true;
DBG_ERR("Unable to fstat() new log file '%s': %s\n",
logfile, strerror(errno));
log_overflow = false;
return false;
}
config->ino = st.st_ino;
return true;
}
/**
reopen the log file (usually called because the log file name might have changed)
*/
bool reopen_logs_internal(void)
{
struct debug_backend *b = NULL;
mode_t oldumask;
size_t i;
bool ok;
if (state.reopening_logs) {
return true;
}
/* Now clear the SIGHUP induced flag */
state.schedule_reopen_logs = false;
switch (state.logtype) {
case DEBUG_CALLBACK:
return true;
case DEBUG_STDOUT:
case DEBUG_DEFAULT_STDOUT:
debug_close_fd(dbgc_config[DBGC_ALL].fd);
dbgc_config[DBGC_ALL].fd = 1;
return true;
case DEBUG_DEFAULT_STDERR:
case DEBUG_STDERR:
debug_close_fd(dbgc_config[DBGC_ALL].fd);
dbgc_config[DBGC_ALL].fd = 2;
return true;
case DEBUG_FILE:
b = debug_find_backend("file");
assert(b != NULL);
b->log_level = MAX_DEBUG_LEVEL;
break;
}
oldumask = umask( 022 );
for (i = DBGC_ALL; i < debug_num_classes; i++) {
if (dbgc_config[i].logfile != NULL) {
break;
}
}
if (i == debug_num_classes) {
return false;
}
state.reopening_logs = true;
for (i = DBGC_ALL; i < debug_num_classes; i++) {
ok = reopen_one_log(&dbgc_config[i]);
if (!ok) {
break;
}
}
/* Fix from klausr@ITAP.Physik.Uni-Stuttgart.De
* to fix problem where smbd's that generate less
* than 100 messages keep growing the log.
*/
force_check_log_size();
(void)umask(oldumask);
/*
* If log file was opened or created successfully, take over stderr to
* catch output into logs.
*/
if (!state.settings.debug_no_stderr_redirect &&
dbgc_config[DBGC_ALL].fd > 0) {
if (dup2(dbgc_config[DBGC_ALL].fd, 2) == -1) {
/* Close stderr too, if dup2 can't point it -
at the logfile. There really isn't much
that can be done on such a fundamental
failure... */
close_low_fd(2);
}
}
state.reopening_logs = false;
return ok;
}
/**************************************************************************
Force a check of the log size.
***************************************************************************/
void force_check_log_size( void )
{
debug_count = 100;
}
_PUBLIC_ void debug_schedule_reopen_logs(void)
{
state.schedule_reopen_logs = true;
}
/***************************************************************************
Check to see if there is any need to check if the logfile has grown too big.
**************************************************************************/
bool need_to_check_log_size(void)
{
int maxlog;
size_t i;
if (debug_count < 100) {
return false;
}
maxlog = state.settings.max_log_size * 1024;
if (maxlog <= 0) {
debug_count = 0;
return false;
}
if (dbgc_config[DBGC_ALL].fd > 2) {
return true;
}
for (i = DBGC_ALL + 1; i < debug_num_classes; i++) {
if (dbgc_config[i].fd != -1) {
return true;
}
}
debug_count = 0;
return false;
}
/**************************************************************************
Check to see if the log has grown to be too big.
**************************************************************************/
static void do_one_check_log_size(off_t maxlog, struct debug_class *config)
{
char name[strlen(config->logfile) + 5];
struct stat st;
int ret;
bool reopen = false;
bool ok;
if (maxlog == 0) {
return;
}
ret = stat(config->logfile, &st);
if (ret != 0) {
return;
}
if (st.st_size >= maxlog ) {
reopen = true;
}
if (st.st_ino != config->ino) {
reopen = true;
}
if (!reopen) {
return;
}
/* reopen_logs_internal() modifies *_fd */
(void)reopen_logs_internal();
if (config->fd <= 2) {
return;
}
ret = fstat(config->fd, &st);
if (ret != 0) {
config->ino = (ino_t)0;
return;
}
config->ino = st.st_ino;
if (st.st_size < maxlog) {
return;
}
snprintf(name, sizeof(name), "%s.old", config->logfile);
(void)rename(config->logfile, name);
ok = reopen_logs_internal();
if (ok) {
return;
}
/* We failed to reopen a log - continue using the old name. */
(void)rename(name, config->logfile);
}
static void do_check_log_size(off_t maxlog)
{
size_t i;
for (i = DBGC_ALL; i < debug_num_classes; i++) {
if (dbgc_config[i].fd == -1) {
continue;
}
if (dbgc_config[i].logfile == NULL) {
continue;
}
do_one_check_log_size(maxlog, &dbgc_config[i]);
}
}
void check_log_size( void )
{
off_t maxlog;
if (geteuid() != 0) {
/*
* We need to be root to change the log file (tests use a fake
* geteuid() from third_party/uid_wrapper). Otherwise we skip
* this and let the main smbd loop or some other process do
* the work.
*/
return;
}
if(log_overflow || (!state.schedule_reopen_logs && !need_to_check_log_size())) {
return;
}
maxlog = state.settings.max_log_size * 1024;
if (state.schedule_reopen_logs) {
(void)reopen_logs_internal();
}
do_check_log_size(maxlog);
/*
* Here's where we need to panic if dbgc_config[DBGC_ALL].fd == 0 or -1
* (invalid values)
*/
if (dbgc_config[DBGC_ALL].fd <= 0) {
/* This code should only be reached in very strange
* circumstances. If we merely fail to open the new log we
* should stick with the old one. ergo this should only be
* reached when opening the logs for the first time: at
* startup or when the log level is increased from zero.
* -dwg 6 June 2000
*/
int fd = open( "/dev/console", O_WRONLY, 0);
if (fd != -1) {
smb_set_close_on_exec(fd);
dbgc_config[DBGC_ALL].fd = fd;
DBG_ERR("check_log_size: open of debug file %s failed "
"- using console.\n",
dbgc_config[DBGC_ALL].logfile);
} else {
/*
* We cannot continue without a debug file handle.
*/
abort();
}
}
debug_count = 0;
}
/*************************************************************************
Write an debug message on the debugfile.
This is called by format_debug_text().
************************************************************************/
static void Debug1(const char *msg, size_t msg_len)
{
int old_errno = errno;
enum debug_logtype logtype = state.logtype;
debug_count++;
if (state.settings.debug_syslog_format == DEBUG_SYSLOG_FORMAT_ALWAYS) {
switch(state.logtype) {
case DEBUG_STDOUT:
case DEBUG_STDERR:
case DEBUG_DEFAULT_STDOUT:
case DEBUG_DEFAULT_STDERR:
/* Behave the same as logging to a file */
logtype = DEBUG_FILE;
break;
default:
break;
}
}
switch(logtype) {
case DEBUG_CALLBACK:
debug_callback_log(msg, msg_len, current_msg_level);
break;
case DEBUG_STDOUT:
case DEBUG_STDERR:
case DEBUG_DEFAULT_STDOUT:
case DEBUG_DEFAULT_STDERR:
if (dbgc_config[DBGC_ALL].fd > 0) {
ssize_t ret;
do {
ret = write(dbgc_config[DBGC_ALL].fd,
msg,
msg_len);
} while (ret == -1 && errno == EINTR);
}
break;
case DEBUG_FILE:
debug_backends_log(msg, msg_len, current_msg_level);
break;
};
errno = old_errno;
}
/**************************************************************************
Print the buffer content via Debug1(), then reset the buffer.
Input: none
Output: none
****************************************************************************/
static void bufr_print( void )
{
format_bufr[format_pos] = '\0';
(void)Debug1(format_bufr, format_pos);
format_pos = 0;
}
/*
* If set (by tevent_thread_call_depth_set()) to value > 0, debug code will use
* it for the trace indentation.
*/
static size_t debug_call_depth = 0;
size_t *debug_call_depth_addr(void)
{
return &debug_call_depth;
}
/***************************************************************************
Format the debug message text.
Input: msg - Text to be added to the "current" debug message text.
Output: none.
Notes: The purpose of this is two-fold. First, each call to syslog()
(used by Debug1(), see above) generates a new line of syslog
output. This is fixed by storing the partial lines until the
newline character is encountered. Second, printing the debug
message lines when a newline is encountered allows us to add
spaces, thus indenting the body of the message and making it
more readable.
**************************************************************************/
static void format_debug_text( const char *msg )
{
size_t i;
bool timestamp = (state.logtype == DEBUG_FILE && (state.settings.timestamp_logs));
debug_init();
for( i = 0; msg[i]; i++ ) {
/* Indent two spaces at each new line. */
if(timestamp && 0 == format_pos) {
/* Limit the maximum indentation to 20 levels */
size_t depth = MIN(20, debug_call_depth);
format_bufr[0] = format_bufr[1] = ' ';
format_pos = 2;
/*
* Indent by four spaces for each depth level,
* but only if the current debug level is >= 8.
*/
if (depth > 0 && debuglevel_get() >= 8 &&
format_pos + 4 * depth < FORMAT_BUFR_SIZE) {
memset(&format_bufr[format_pos],
' ',
4 * depth);
format_pos += 4 * depth;
}
}
/* If there's room, copy the character to the format buffer. */
if (format_pos < FORMAT_BUFR_SIZE - 1)
format_bufr[format_pos++] = msg[i];
/* If a newline is encountered, print & restart. */
if( '\n' == msg[i] )
bufr_print();
/* If the buffer is full dump it out, reset it, and put out a line
* continuation indicator.
*/
if (format_pos >= FORMAT_BUFR_SIZE - 1) {
const char cont[] = " +>\n";
bufr_print();
(void)Debug1(cont , sizeof(cont) - 1);
}
}
/* Just to be safe... */
format_bufr[format_pos] = '\0';
}
/***************************************************************************
Flush debug output, including the format buffer content.
Input: none
Output: none
***************************************************************************/
void dbgflush( void )
{
bufr_print();
}
bool dbgsetclass(int level, int cls)
{
/* Set current_msg_level. */
current_msg_level = level;
/* Set current message class */
current_msg_class = cls;
return true;
}
/***************************************************************************
Put a Debug Header into header_str.
Input: level - Debug level of the message (not the system-wide debug
level. )
cls - Debuglevel class of the calling module.
location - Pointer to a string containing the name of the file
from which this function was called, or an empty string
if the __FILE__ macro is not implemented.
func - Pointer to a string containing the name of the function
from which this function was called, or an empty string
if the __FUNCTION__ macro is not implemented.
Output: Always true. This makes it easy to fudge a call to dbghdr()
in a macro, since the function can be called as part of a test.
Eg: ( (level <= DEBUGLEVEL) && (dbghdr(level,"",line)) )
Notes: This function takes care of setting current_msg_level.
****************************************************************************/
bool dbghdrclass(int level, int cls, const char *location, const char *func)
{
/* Ensure we don't lose any real errno value. */
int old_errno = errno;
bool verbose = false;
struct timeval tv;
struct timeval_buf tvbuf;
/*
* This might be overkill, but if another early return is
* added later then initialising these avoids potential
* problems
*/
state.hs_len = 0;
state.header_str[0] = '\0';
if( format_pos ) {
/* This is a fudge. If there is stuff sitting in the format_bufr, then
* the *right* thing to do is to call
* format_debug_text( "\n" );
* to write the remainder, and then proceed with the new header.
* Unfortunately, there are several places in the code at which
* the DEBUG() macro is used to build partial lines. That in mind,
* we'll work under the assumption that an incomplete line indicates
* that a new header is *not* desired.
*/
return( true );
}
dbgsetclass(level, cls);
/*
* Don't print a header if we're logging to stdout,
* unless 'debug syslog format = always'
*/
if (state.logtype != DEBUG_FILE &&
state.settings.debug_syslog_format != DEBUG_SYSLOG_FORMAT_ALWAYS)
{
return true;
}
/*
* Print the header if timestamps (or debug syslog format) is
* turned on. If parameters are not yet loaded, then default
* to timestamps on.
*/
if (!(state.settings.timestamp_logs ||
state.settings.debug_prefix_timestamp ||
state.settings.debug_syslog_format != DEBUG_SYSLOG_FORMAT_NO))
{
return true;
}
GetTimeOfDay(&tv);
if (state.settings.debug_syslog_format != DEBUG_SYSLOG_FORMAT_NO) {
if (state.settings.debug_hires_timestamp) {
timeval_str_buf(&tv, true, true, &tvbuf);
} else {
time_t t;
struct tm *tm;
t = (time_t)tv.tv_sec;
tm = localtime(&t);
if (tm != NULL) {
size_t len;
len = strftime(tvbuf.buf,
sizeof(tvbuf.buf),
"%b %e %T",
tm);
if (len == 0) {
/* Trigger default time format below */
tm = NULL;
}
}
if (tm == NULL) {
snprintf(tvbuf.buf,
sizeof(tvbuf.buf),
"%ld seconds since the Epoch", (long)t);
}
}
ensure_hostname();
state.hs_len = snprintf(state.header_str,
sizeof(state.header_str),
"%s %.*s %s[%u]: ",
tvbuf.buf,
(int)(sizeof(state.hostname) - 1),
state.hostname,
state.prog_name,
(unsigned int) getpid());
goto full;
}
timeval_str_buf(&tv, false, state.settings.debug_hires_timestamp,
&tvbuf);
state.hs_len = snprintf(state.header_str,
sizeof(state.header_str),
"[%s, %2d",
tvbuf.buf,
level);
if (state.hs_len >= sizeof(state.header_str) - 1) {
goto full;
}
if (unlikely(dbgc_config[cls].loglevel >= 10)) {
verbose = true;
}
if (verbose || state.settings.debug_pid) {
state.hs_len += snprintf(state.header_str + state.hs_len,
sizeof(state.header_str) - state.hs_len,
", pid=%u",
(unsigned int)getpid());
if (state.hs_len >= sizeof(state.header_str) - 1) {
goto full;
}
}
if (verbose || state.settings.debug_uid) {
state.hs_len += snprintf(state.header_str + state.hs_len,
sizeof(state.header_str) - state.hs_len,
", effective(%u, %u), real(%u, %u)",
(unsigned int)geteuid(),
(unsigned int)getegid(),
(unsigned int)getuid(),
(unsigned int)getgid());
if (state.hs_len >= sizeof(state.header_str) - 1) {
goto full;
}
}
if ((verbose || state.settings.debug_class)
&& (cls != DBGC_ALL)) {
state.hs_len += snprintf(state.header_str + state.hs_len,
sizeof(state.header_str) - state.hs_len,
", class=%s",
classname_table[cls]);
if (state.hs_len >= sizeof(state.header_str) - 1) {
goto full;
}
}
if (debug_traceid_get() != 0) {
state.hs_len += snprintf(state.header_str + state.hs_len,
sizeof(state.header_str) - state.hs_len,
", traceid=%" PRIu64,
debug_traceid_get());
if (state.hs_len >= sizeof(state.header_str) - 1) {
goto full;
}
}
if (debug_call_depth > 0) {
state.hs_len += snprintf(state.header_str + state.hs_len,
sizeof(state.header_str) - state.hs_len,
", depth=%zu",
debug_call_depth);
if (state.hs_len >= sizeof(state.header_str) - 1) {
goto full;
}
}
state.header_str[state.hs_len] = ']';
state.hs_len++;
if (state.hs_len < sizeof(state.header_str) - 1) {
state.header_str[state.hs_len] = ' ';
state.hs_len++;
}
state.header_str[state.hs_len] = '\0';
if (!state.settings.debug_prefix_timestamp) {
state.hs_len += snprintf(state.header_str + state.hs_len,
sizeof(state.header_str) - state.hs_len,
"%s(%s)\n",
location,
func);
if (state.hs_len >= sizeof(state.header_str)) {
goto full;
}
}
full:
/*
* Above code never overflows state.header_str and always
* NUL-terminates correctly. However, state.hs_len can point
* past the end of the buffer to indicate that truncation
* occurred, so fix it if necessary, since state.hs_len is
* expected to be used after return.
*/
if (state.hs_len >= sizeof(state.header_str)) {
state.hs_len = sizeof(state.header_str) - 1;
}
state.header_str_no_nl[0] = '\0';
errno = old_errno;
return( true );
}
/***************************************************************************
Add text to the body of the "current" debug message via the format buffer.
Input: format_str - Format string, as used in printf(), et. al.
... - Variable argument list.
..or.. va_alist - Old style variable parameter list starting point.
Output: Always true. See dbghdr() for more info, though this is not
likely to be used in the same way.
***************************************************************************/
static inline bool __dbgtext_va(const char *format_str, va_list ap) PRINTF_ATTRIBUTE(1,0);
static inline bool __dbgtext_va(const char *format_str, va_list ap)
{
char *msgbuf = NULL;
bool ret = true;
int res;
res = vasprintf(&msgbuf, format_str, ap);
if (res != -1) {
format_debug_text(msgbuf);
} else {
ret = false;
}
SAFE_FREE(msgbuf);
return ret;
}
bool dbgtext_va(const char *format_str, va_list ap)
{
return __dbgtext_va(format_str, ap);
}
bool dbgtext(const char *format_str, ... )
{
va_list ap;
bool ret;
va_start(ap, format_str);
ret = __dbgtext_va(format_str, ap);
va_end(ap);
return ret;
}
static uint64_t debug_traceid = 0;
uint64_t debug_traceid_set(uint64_t id)
{
uint64_t old_id = debug_traceid;
debug_traceid = id;
return old_id;
}
uint64_t debug_traceid_get(void)
{
return debug_traceid;
}