mirror of
https://codeberg.org/redict/redict.git
synced 2025-01-22 16:18:28 -05:00
better bug report info on crash
This commit is contained in:
parent
45e7a1ce00
commit
fa5af017d9
@ -297,10 +297,14 @@ void debugCommand(redisClient *c) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void _redisAssert(char *estr, char *file, int line) {
|
void _redisAssert(char *estr, char *file, int line) {
|
||||||
|
bugReportStart();
|
||||||
redisLog(REDIS_WARNING,"=== ASSERTION FAILED ===");
|
redisLog(REDIS_WARNING,"=== ASSERTION FAILED ===");
|
||||||
redisLog(REDIS_WARNING,"==> %s:%d '%s' is not true",file,line,estr);
|
redisLog(REDIS_WARNING,"==> %s:%d '%s' is not true",file,line,estr);
|
||||||
#ifdef HAVE_BACKTRACE
|
#ifdef HAVE_BACKTRACE
|
||||||
redisLog(REDIS_WARNING,"(forcing SIGSEGV in order to print the stack trace)");
|
server.assert_failed = estr;
|
||||||
|
server.assert_file = file;
|
||||||
|
server.assert_line = line;
|
||||||
|
redisLog(REDIS_WARNING,"(forcing SIGSEGV to print the bug report.)");
|
||||||
*((char*)-1) = 'x';
|
*((char*)-1) = 'x';
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
@ -308,6 +312,7 @@ void _redisAssert(char *estr, char *file, int line) {
|
|||||||
void _redisAssertPrintClientInfo(redisClient *c) {
|
void _redisAssertPrintClientInfo(redisClient *c) {
|
||||||
int j;
|
int j;
|
||||||
|
|
||||||
|
bugReportStart();
|
||||||
redisLog(REDIS_WARNING,"=== ASSERTION FAILED CLIENT CONTEXT ===");
|
redisLog(REDIS_WARNING,"=== ASSERTION FAILED CLIENT CONTEXT ===");
|
||||||
redisLog(REDIS_WARNING,"client->flags = %d", c->flags);
|
redisLog(REDIS_WARNING,"client->flags = %d", c->flags);
|
||||||
redisLog(REDIS_WARNING,"client->fd = %d", c->fd);
|
redisLog(REDIS_WARNING,"client->fd = %d", c->fd);
|
||||||
@ -331,6 +336,7 @@ void _redisAssertPrintClientInfo(redisClient *c) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void _redisAssertPrintObject(robj *o) {
|
void _redisAssertPrintObject(robj *o) {
|
||||||
|
bugReportStart();
|
||||||
redisLog(REDIS_WARNING,"=== ASSERTION FAILED OBJECT CONTEXT ===");
|
redisLog(REDIS_WARNING,"=== ASSERTION FAILED OBJECT CONTEXT ===");
|
||||||
redisLog(REDIS_WARNING,"Object type: %d", o->type);
|
redisLog(REDIS_WARNING,"Object type: %d", o->type);
|
||||||
redisLog(REDIS_WARNING,"Object encoding: %d", o->encoding);
|
redisLog(REDIS_WARNING,"Object encoding: %d", o->encoding);
|
||||||
@ -349,6 +355,7 @@ void _redisAssertWithInfo(redisClient *c, robj *o, char *estr, char *file, int l
|
|||||||
}
|
}
|
||||||
|
|
||||||
void _redisPanic(char *msg, char *file, int line) {
|
void _redisPanic(char *msg, char *file, int line) {
|
||||||
|
bugReportStart();
|
||||||
redisLog(REDIS_WARNING,"------------------------------------------------");
|
redisLog(REDIS_WARNING,"------------------------------------------------");
|
||||||
redisLog(REDIS_WARNING,"!!! Software Failure. Press left mouse button to continue");
|
redisLog(REDIS_WARNING,"!!! Software Failure. Press left mouse button to continue");
|
||||||
redisLog(REDIS_WARNING,"Guru Meditation: %s #%s:%d",msg,file,line);
|
redisLog(REDIS_WARNING,"Guru Meditation: %s #%s:%d",msg,file,line);
|
||||||
|
44
src/redis.c
44
src/redis.c
@ -946,6 +946,12 @@ void initServerConfig() {
|
|||||||
/* Slow log */
|
/* Slow log */
|
||||||
server.slowlog_log_slower_than = REDIS_SLOWLOG_LOG_SLOWER_THAN;
|
server.slowlog_log_slower_than = REDIS_SLOWLOG_LOG_SLOWER_THAN;
|
||||||
server.slowlog_max_len = REDIS_SLOWLOG_MAX_LEN;
|
server.slowlog_max_len = REDIS_SLOWLOG_MAX_LEN;
|
||||||
|
|
||||||
|
/* Assert */
|
||||||
|
server.assert_failed = "<no assertion failed>";
|
||||||
|
server.assert_file = "<no file>";
|
||||||
|
server.assert_line = 0;
|
||||||
|
server.bug_report_start = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
void initServer() {
|
void initServer() {
|
||||||
@ -1946,32 +1952,56 @@ static void *getMcontextEip(ucontext_t *uc) {
|
|||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void bugReportStart(void) {
|
||||||
|
if (server.bug_report_start == 0) {
|
||||||
|
redisLog(REDIS_WARNING,
|
||||||
|
"=== REDIS BUG REPORT START: Cut & paste starting from here ===");
|
||||||
|
server.bug_report_start = 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
static void sigsegvHandler(int sig, siginfo_t *info, void *secret) {
|
static void sigsegvHandler(int sig, siginfo_t *info, void *secret) {
|
||||||
void *trace[100];
|
void *trace[100];
|
||||||
char **messages = NULL;
|
char **messages = NULL;
|
||||||
int i, trace_size = 0;
|
int i, trace_size = 0;
|
||||||
ucontext_t *uc = (ucontext_t*) secret;
|
ucontext_t *uc = (ucontext_t*) secret;
|
||||||
sds infostring;
|
sds infostring, clients;
|
||||||
struct sigaction act;
|
struct sigaction act;
|
||||||
REDIS_NOTUSED(info);
|
REDIS_NOTUSED(info);
|
||||||
|
|
||||||
|
bugReportStart();
|
||||||
redisLog(REDIS_WARNING,
|
redisLog(REDIS_WARNING,
|
||||||
"======= Ooops! Redis %s got signal: -%d- =======", REDIS_VERSION, sig);
|
" Redis %s crashed by signal: %d", REDIS_VERSION, sig);
|
||||||
infostring = genRedisInfoString("all");
|
redisLog(REDIS_WARNING,
|
||||||
redisLogRaw(REDIS_WARNING, infostring);
|
" Failed assertion: %s (%s:%d)", server.assert_failed,
|
||||||
/* It's not safe to sdsfree() the returned string under memory
|
server.assert_file, server.assert_line);
|
||||||
* corruption conditions. Let it leak as we are going to abort */
|
|
||||||
|
|
||||||
|
/* Generate the stack trace */
|
||||||
trace_size = backtrace(trace, 100);
|
trace_size = backtrace(trace, 100);
|
||||||
|
|
||||||
/* overwrite sigaction with caller's address */
|
/* overwrite sigaction with caller's address */
|
||||||
if (getMcontextEip(uc) != NULL) {
|
if (getMcontextEip(uc) != NULL) {
|
||||||
trace[1] = getMcontextEip(uc);
|
trace[1] = getMcontextEip(uc);
|
||||||
}
|
}
|
||||||
messages = backtrace_symbols(trace, trace_size);
|
messages = backtrace_symbols(trace, trace_size);
|
||||||
|
redisLog(REDIS_WARNING, "--- STACK TRACE");
|
||||||
for (i=1; i<trace_size; ++i)
|
for (i=1; i<trace_size; ++i)
|
||||||
redisLog(REDIS_WARNING,"%s", messages[i]);
|
redisLog(REDIS_WARNING,"%s", messages[i]);
|
||||||
|
|
||||||
|
/* Log INFO and CLIENT LIST */
|
||||||
|
redisLog(REDIS_WARNING, "--- INFO OUTPUT");
|
||||||
|
infostring = genRedisInfoString("all");
|
||||||
|
redisLogRaw(REDIS_WARNING, infostring);
|
||||||
|
redisLog(REDIS_WARNING, "--- CLIENT LIST OUTPUT");
|
||||||
|
clients = getAllClientsInfoString();
|
||||||
|
redisLogRaw(REDIS_WARNING, clients);
|
||||||
|
/* Don't sdsfree() strings to avoid a crash. Memory may be corrupted. */
|
||||||
|
|
||||||
|
redisLog(REDIS_WARNING,
|
||||||
|
"=== REDIS BUG REPORT END. Make sure to include from START to END. ===\n\n"
|
||||||
|
" Please report the crash opening an issue on github:\n\n"
|
||||||
|
" http://github.com/antirez/redis/issues\n\n"
|
||||||
|
);
|
||||||
/* free(messages); Don't call free() with possibly corrupted memory. */
|
/* free(messages); Don't call free() with possibly corrupted memory. */
|
||||||
if (server.daemonize) unlink(server.pidfile);
|
if (server.daemonize) unlink(server.pidfile);
|
||||||
|
|
||||||
|
@ -634,6 +634,11 @@ struct redisServer {
|
|||||||
int lua_timedout; /* True if we reached the time limit for script
|
int lua_timedout; /* True if we reached the time limit for script
|
||||||
execution. */
|
execution. */
|
||||||
int lua_kill; /* Kill the script if true. */
|
int lua_kill; /* Kill the script if true. */
|
||||||
|
/* Assert & bug reportign */
|
||||||
|
char *assert_failed;
|
||||||
|
char *assert_file;
|
||||||
|
int assert_line;
|
||||||
|
int bug_report_start; /* True if bug report header already logged. */
|
||||||
};
|
};
|
||||||
|
|
||||||
typedef struct pubsubPattern {
|
typedef struct pubsubPattern {
|
||||||
@ -1148,5 +1153,6 @@ void *realloc(void *ptr, size_t size) __attribute__ ((deprecated));
|
|||||||
void _redisAssertWithInfo(redisClient *c, robj *o, char *estr, char *file, int line);
|
void _redisAssertWithInfo(redisClient *c, robj *o, char *estr, char *file, int line);
|
||||||
void _redisAssert(char *estr, char *file, int line);
|
void _redisAssert(char *estr, char *file, int line);
|
||||||
void _redisPanic(char *msg, char *file, int line);
|
void _redisPanic(char *msg, char *file, int line);
|
||||||
|
void bugReportStart(void);
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
Loading…
Reference in New Issue
Block a user