Commit 9916f14b authored by marko@hundin.mysql.fi's avatar marko@hundin.mysql.fi

InnoDB: Improved the handling of assertions.

parent 1f865775
...@@ -13,72 +13,79 @@ Created 1/30/1994 Heikki Tuuri ...@@ -13,72 +13,79 @@ Created 1/30/1994 Heikki Tuuri
#include <stdlib.h> #include <stdlib.h>
#include "os0thread.h" #include "os0thread.h"
extern ibool ut_dbg_stop_threads; #if defined(__GNUC__) && (__GNUC__ > 2)
# define UT_DBG_FAIL(EXPR) UNIV_UNLIKELY(!((ulint)(EXPR)))
#else
extern ulint ut_dbg_zero; /* This is used to eliminate
compiler warnings */
# define UT_DBG_FAIL(EXPR) !((ulint)(EXPR) + ut_dbg_zero)
#endif
extern ulint* ut_dbg_null_ptr; /*****************************************************************
Report a failed assertion. */
void
ut_dbg_assertion_failed(
/*====================*/
const char* expr, /* in: the failed assertion */
const char* file, /* in: source file containing the assertion */
ulint line); /* in: line number of the assertion */
extern const char* ut_dbg_msg_assert_fail;
extern const char* ut_dbg_msg_trap;
extern const char* ut_dbg_msg_stop;
/* Have a graceful exit on NetWare rather than a segfault to avoid abends */
#ifdef __NETWARE__ #ifdef __NETWARE__
/* Flag for ignoring further assertion failures.
On NetWare, have a graceful exit rather than a segfault to avoid abends. */
extern ibool panic_shutdown; extern ibool panic_shutdown;
#define ut_a(EXPR) do {\ /* Abort the execution. */
if (UNIV_UNLIKELY(!((ulint)(EXPR)))) {\ # define UT_DBG_PANIC \
ut_print_timestamp(stderr);\ if (!panic_shutdown){ \
fprintf(stderr, ut_dbg_msg_assert_fail,\ panic_shutdown = TRUE; \
os_thread_pf(os_thread_get_curr_id()), __FILE__,\ innobase_shutdown_for_mysql(); \
(ulint)__LINE__);\ } \
fputs("InnoDB: Failing assertion: " #EXPR "\n", stderr);\ exit(1)
fputs(ut_dbg_msg_trap, stderr);\ /* Stop threads in ut_a(). */
ut_dbg_stop_threads = TRUE;\ # define UT_DBG_STOP while (0) /* We do not do this on NetWare */
if (ut_dbg_stop_threads) {\ #else /* __NETWARE__ */
fprintf(stderr, ut_dbg_msg_stop,\ /* Flag for indicating that all threads should stop. This will be set
os_thread_pf(os_thread_get_curr_id()), __FILE__, (ulint)__LINE__);\ by ut_dbg_assertion_failed(). */
}\ extern ibool ut_dbg_stop_threads;
if(!panic_shutdown){\
panic_shutdown = TRUE;\ /* A null pointer that will be dereferenced to trigger a memory trap */
innobase_shutdown_for_mysql();}\ extern ulint* ut_dbg_null_ptr;
exit(1);\
}\ /*****************************************************************
} while (0) Stop a thread after assertion failure. */
#define ut_error do {\
ut_print_timestamp(stderr);\ void
fprintf(stderr, ut_dbg_msg_assert_fail,\ ut_dbg_stop_thread(
os_thread_pf(os_thread_get_curr_id()), __FILE__, (ulint)__LINE__);\ /*===============*/
fprintf(stderr, ut_dbg_msg_trap);\ const char* file,
ut_dbg_stop_threads = TRUE;\ ulint line);
if(!panic_shutdown){panic_shutdown = TRUE;\
innobase_shutdown_for_mysql();}\ /* Abort the execution. */
} while (0) # define UT_DBG_PANIC \
#else if (*(ut_dbg_null_ptr)) ut_dbg_null_ptr = NULL
#define ut_a(EXPR) do {\ /* Stop threads in ut_a(). */
if (UNIV_UNLIKELY(!((ulint)(EXPR)))) {\ # define UT_DBG_STOP do \
ut_print_timestamp(stderr);\ if (UNIV_UNLIKELY(ut_dbg_stop_threads)) { \
fprintf(stderr, ut_dbg_msg_assert_fail,\ ut_dbg_stop_thread(__FILE__, (ulint) __LINE__); \
os_thread_pf(os_thread_get_curr_id()), __FILE__,\ } while (0)
(ulint)__LINE__);\ #endif /* __NETWARE__ */
fputs("InnoDB: Failing assertion: " #EXPR "\n", stderr);\
fputs(ut_dbg_msg_trap, stderr);\ /* Abort execution if EXPR does not evaluate to nonzero. */
ut_dbg_stop_threads = TRUE;\ #define ut_a(EXPR) do { \
if (*(ut_dbg_null_ptr)) ut_dbg_null_ptr = NULL;\ if (UT_DBG_FAIL(EXPR)) { \
}\ ut_dbg_assertion_failed(#EXPR, \
if (ut_dbg_stop_threads) {\ __FILE__, (ulint) __LINE__); \
fprintf(stderr, ut_dbg_msg_stop,\ UT_DBG_PANIC; \
os_thread_pf(os_thread_get_curr_id()), __FILE__, (ulint)__LINE__);\ } \
os_thread_sleep(1000000000);\ UT_DBG_STOP; \
}\
} while (0) } while (0)
#define ut_error do {\ /* Abort execution. */
ut_print_timestamp(stderr);\ #define ut_error do { \
fprintf(stderr, ut_dbg_msg_assert_fail,\ ut_dbg_assertion_failed(0, __FILE__, (ulint) __LINE__); \
os_thread_pf(os_thread_get_curr_id()), __FILE__, (ulint)__LINE__);\ UT_DBG_PANIC; \
fprintf(stderr, ut_dbg_msg_trap);\
ut_dbg_stop_threads = TRUE;\
if (*(ut_dbg_null_ptr)) ut_dbg_null_ptr = NULL;\
} while (0) } while (0)
#endif
#ifdef UNIV_DEBUG #ifdef UNIV_DEBUG
#define ut_ad(EXPR) ut_a(EXPR) #define ut_ad(EXPR) ut_a(EXPR)
......
...@@ -8,6 +8,12 @@ Created 1/30/1994 Heikki Tuuri ...@@ -8,6 +8,12 @@ Created 1/30/1994 Heikki Tuuri
#include "univ.i" #include "univ.i"
#if defined(__GNUC__) && (__GNUC__ > 2)
#else
/* This is used to eliminate compiler warnings */
ulint ut_dbg_zero = 0;
#endif
/* If this is set to TRUE all threads will stop into the next assertion /* If this is set to TRUE all threads will stop into the next assertion
and assert */ and assert */
ibool ut_dbg_stop_threads = FALSE; ibool ut_dbg_stop_threads = FALSE;
...@@ -16,21 +22,55 @@ ibool panic_shutdown = FALSE; /* This is set to TRUE when on NetWare there ...@@ -16,21 +22,55 @@ ibool panic_shutdown = FALSE; /* This is set to TRUE when on NetWare there
happens an InnoDB assertion failure or other happens an InnoDB assertion failure or other
fatal error condition that requires an fatal error condition that requires an
immediate shutdown. */ immediate shutdown. */
#endif #else /* __NETWARE__ */
/* Null pointer used to generate memory trap */ /* Null pointer used to generate memory trap */
ulint* ut_dbg_null_ptr = NULL; ulint* ut_dbg_null_ptr = NULL;
#endif /* __NETWARE__ */
const char* ut_dbg_msg_assert_fail = /*****************************************************************
"InnoDB: Assertion failure in thread %lu in file %s line %lu\n"; Report a failed assertion. */
const char* ut_dbg_msg_trap =
void
ut_dbg_assertion_failed(
/*====================*/
const char* expr, /* in: the failed assertion (optional) */
const char* file, /* in: source file containing the assertion */
ulint line) /* in: line number of the assertion */
{
ut_print_timestamp(stderr);
fprintf(stderr,
"InnoDB: Assertion failure in thread %lu"
" in file %s line %lu\n",
os_thread_pf(os_thread_get_curr_id()), file, line);
if (expr) {
fprintf(stderr,
"InnoDB: Failing assertion: %s\n", expr);
}
fputs(
"InnoDB: We intentionally generate a memory trap.\n" "InnoDB: We intentionally generate a memory trap.\n"
"InnoDB: Submit a detailed bug report to http://bugs.mysql.com.\n" "InnoDB: Submit a detailed bug report to http://bugs.mysql.com.\n"
"InnoDB: If you get repeated assertion failures or crashes, even\n" "InnoDB: If you get repeated assertion failures or crashes, even\n"
"InnoDB: immediately after the mysqld startup, there may be\n" "InnoDB: immediately after the mysqld startup, there may be\n"
"InnoDB: corruption in the InnoDB tablespace. Please refer to\n" "InnoDB: corruption in the InnoDB tablespace. Please refer to\n"
"InnoDB: http://dev.mysql.com/doc/mysql/en/Forcing_recovery.html\n" "InnoDB: http://dev.mysql.com/doc/mysql/en/Forcing_recovery.html\n"
"InnoDB: about forcing recovery.\n"; "InnoDB: about forcing recovery.\n", stderr);
ut_dbg_stop_threads = TRUE;
}
#ifndef __NETWARE__
/*****************************************************************
Stop a thread after assertion failure. */
const char* ut_dbg_msg_stop = void
"InnoDB: Thread %lu stopped in file %s line %lu\n"; ut_dbg_stop_thread(
/*===============*/
const char* file,
ulint line)
{
fprintf(stderr, "InnoDB: Thread %lu stopped in file %s line %lu\n",
os_thread_pf(os_thread_get_curr_id()), file, line);
os_thread_sleep(1000000000);
}
#endif /* __NETWARE__ */
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment