Merge branch 'errfunc-engine-status'

This commit is contained in:
Leif Walsh 2014-03-25 11:22:08 -04:00
commit 6cfd541797
3 changed files with 100 additions and 8 deletions

View file

@ -120,12 +120,15 @@ toku_assert_init(void)
// Function pointers are zero by default so asserts can be used by ft-layer tests without an environment.
static int (*toku_maybe_get_engine_status_text_p)(char* buff, int buffsize) = 0;
static int (*toku_maybe_err_engine_status_p)(void) = 0;
static void (*toku_maybe_set_env_panic_p)(int code, const char* msg) = 0;
void toku_assert_set_fpointers(int (*toku_maybe_get_engine_status_text_pointer)(char*, int),
void toku_assert_set_fpointers(int (*toku_maybe_get_engine_status_text_pointer)(char*, int),
int (*toku_maybe_err_engine_status_pointer)(void),
void (*toku_maybe_set_env_panic_pointer)(int, const char*),
uint64_t num_rows) {
toku_maybe_get_engine_status_text_p = toku_maybe_get_engine_status_text_pointer;
toku_maybe_err_engine_status_p = toku_maybe_err_engine_status_pointer;
toku_maybe_set_env_panic_p = toku_maybe_set_env_panic_pointer;
engine_status_num_rows = num_rows;
}
@ -145,11 +148,8 @@ void db_env_do_backtrace_errfunc(toku_env_err_func errfunc, const void *env) {
free(syms);
}
if (engine_status_num_rows && toku_maybe_get_engine_status_text_p) {
int buffsize = engine_status_num_rows * 128; // assume 128 characters per row (gross overestimate, should be safe)
char buff[buffsize];
toku_maybe_get_engine_status_text_p(buff, buffsize);
errfunc(env, 0, "Engine status:\n%s\n", buff);
if (engine_status_num_rows && toku_maybe_err_engine_status_p) {
toku_maybe_err_engine_status_p();
} else {
errfunc(env, 0, "Engine status function not available\n");
}

View file

@ -120,7 +120,8 @@ set_errno(int new_errno)
void toku_assert_init(void) __attribute__((constructor));
void toku_assert_set_fpointers(int (*toku_maybe_get_engine_status_text_pointer)(char*, int),
void toku_assert_set_fpointers(int (*toku_maybe_get_engine_status_text_pointer)(char*, int),
int (*toku_maybe_err_engine_status_pointer)(void),
void (*toku_maybe_set_env_panic_pointer)(int, const char*),
uint64_t num_rows);

View file

@ -233,6 +233,7 @@ static DB_ENV * volatile most_recent_env; // most recently opened env, used fo
static int env_get_iname(DB_ENV* env, DBT* dname_dbt, DBT* iname_dbt);
static int toku_maybe_get_engine_status_text (char* buff, int buffsize); // for use by toku_assert
static int toku_maybe_err_engine_status (void);
static void toku_maybe_set_env_panic(int code, const char * msg); // for use by toku_assert
int
@ -1114,7 +1115,7 @@ cleanup:
most_recent_env = env;
uint64_t num_rows;
env_get_engine_status_num_rows(env, &num_rows);
toku_assert_set_fpointers(toku_maybe_get_engine_status_text, toku_maybe_set_env_panic, num_rows);
toku_assert_set_fpointers(toku_maybe_get_engine_status_text, toku_maybe_err_engine_status, toku_maybe_set_env_panic, num_rows);
}
return r;
}
@ -2235,6 +2236,83 @@ env_get_engine_status_text(DB_ENV * env, char * buff, int bufsiz) {
return r;
}
// prints engine status using toku_env_err line-by-line
static int
env_err_engine_status(DB_ENV * env) {
uint32_t stringsize = 1024;
uint64_t panic;
char panicstring[stringsize];
uint64_t num_rows;
uint64_t max_rows;
fs_redzone_state redzone_state;
toku_env_err(env, 0, "BUILD_ID = %d", BUILD_ID);
(void) env_get_engine_status_num_rows (env, &max_rows);
TOKU_ENGINE_STATUS_ROW_S mystat[max_rows];
int r = env->get_engine_status (env, mystat, max_rows, &num_rows, &redzone_state, &panic, panicstring, stringsize, TOKU_ENGINE_STATUS);
if (r) {
toku_env_err(env, 0, "Engine status not available: ");
if (!env) {
toku_env_err(env, 0, "no environment");
}
else if (!(env->i)) {
toku_env_err(env, 0, "environment internal struct is null");
}
else if (!env_opened(env)) {
toku_env_err(env, 0, "environment is not open");
}
}
else {
if (panic) {
toku_env_err(env, 0, "Env panic code: %" PRIu64, panic);
if (strlen(panicstring)) {
invariant(strlen(panicstring) <= stringsize);
toku_env_err(env, 0, "Env panic string: %s", panicstring);
}
}
for (uint64_t row = 0; row < num_rows; row++) {
switch (mystat[row].type) {
case FS_STATE:
toku_env_err(env, 0, "%s: %" PRIu64, mystat[row].legend, mystat[row].value.num);
break;
case UINT64:
toku_env_err(env, 0, "%s: %" PRIu64, mystat[row].legend, mystat[row].value.num);
break;
case CHARSTR:
toku_env_err(env, 0, "%s: %s", mystat[row].legend, mystat[row].value.str);
break;
case UNIXTIME:
{
char tbuf[26];
format_time((time_t*)&mystat[row].value.num, tbuf);
toku_env_err(env, 0, "%s: %s", mystat[row].legend, tbuf);
}
break;
case TOKUTIME:
{
double t = tokutime_to_seconds(mystat[row].value.num);
toku_env_err(env, 0, "%s: %.6f", mystat[row].legend, t);
}
break;
case PARCOUNT:
{
uint64_t v = read_partitioned_counter(mystat[row].value.parcount);
toku_env_err(env, 0, "%s: %" PRIu64, mystat[row].legend, v);
}
break;
default:
toku_env_err(env, 0, "%s: UNKNOWN STATUS TYPE: %d", mystat[row].legend, mystat[row].type);
break;
}
}
}
return r;
}
// intended for use by toku_assert logic, when env is not known
static int
toku_maybe_get_engine_status_text (char * buff, int buffsize) {
@ -2250,6 +2328,19 @@ toku_maybe_get_engine_status_text (char * buff, int buffsize) {
return r;
}
static int
toku_maybe_err_engine_status (void) {
DB_ENV * env = most_recent_env;
int r;
if (engine_status_enable && env != NULL) {
r = env_err_engine_status(env);
}
else {
r = EOPNOTSUPP;
}
return r;
}
// Set panic code and panic string if not already panicked,
// intended for use by toku_assert when about to abort().
static void