Commit d569899a authored by Dave Wells's avatar Dave Wells Committed by Yoni Fogel

now randomly aborts before, during, or after the last checkpoint

git-svn-id: file:///svn/toku/tokudb@16217 c7de825b-a66e-492c-adef-691d508d4ae1
parent 79e7c8a3
/* -*- mode: C; c-basic-offset: 4 -*_ */ /* -*- mode: C; c-basic-offset: 4 -*- */
#ident "Copyright (c) 2009 Tokutek Inc. All rights reserved." #ident "Copyright (c) 2009 Tokutek Inc. All rights reserved."
#ident "$Id$" #ident "$Id$"
...@@ -7,23 +7,24 @@ ...@@ -7,23 +7,24 @@
#include <db.h> #include <db.h>
#include <sys/stat.h> #include <sys/stat.h>
#include <stdlib.h> #include <stdlib.h>
//#include "checkpoint_test.h"
static const int OPER_PER_STEP = 3; static const int OPER_PER_STEP = 3;
//static const int NUM_DICTIONARIES = 100;
static const int NUM_DICTIONARIES = 3; static const int NUM_DICTIONARIES = 3;
static const char *table = "tbl"; static const char *table = "tbl";
static const int ROWS_PER_TABLE = 10; static const int ROWS_PER_TABLE = 10;
DB_ENV *env; DB_ENV *env;
DB** db_array;
DB* states;
static const int percent_do_op = 20;
char *state_db_name="states.db";
#define CREATED 0 #define CREATED 0
#define OPEN 1 #define OPEN 1
#define CLOSED 2 #define CLOSED 2
#define DELETED 3 #define DELETED 3
DB* db_array;
DB* states;
static void put_state(int db_num, int state) { static void put_state(int db_num, int state) {
int r; int r;
DB_TXN* txn; DB_TXN* txn;
...@@ -41,11 +42,9 @@ static void put_state(int db_num, int state) { ...@@ -41,11 +42,9 @@ static void put_state(int db_num, int state) {
static int get_state(int db_num) { static int get_state(int db_num) {
int r; int r;
DBT key, val; DBT key, val;
// int key_data = db_num;
memset(&val, 0, sizeof(val)); memset(&val, 0, sizeof(val));
r = states->get(states, 0, r = states->get(states, 0,
// dbt_init(&key, &key_data, sizeof(key_data)),
dbt_init(&key, &db_num, sizeof(db_num)), dbt_init(&key, &db_num, sizeof(db_num)),
&val, &val,
0); 0);
...@@ -54,59 +53,76 @@ static int get_state(int db_num) { ...@@ -54,59 +53,76 @@ static int get_state(int db_num) {
return state; return state;
} }
static int crash_timer;
static void crash_it(void);
static void set_crash_timer(void) {
crash_timer = random() % (3 * NUM_DICTIONARIES);
}
static void update_crash_timer(void) {
if ( --crash_timer == 0 ) {
if ( verbose ) {
printf("%s : crash\n", __FILE__);
fflush(stdout);
}
crash_it();
}
}
static void env_startup(int recovery_flags); static void env_startup(int recovery_flags);
static int64_t generate_val(int64_t key); static int64_t generate_val(int64_t key);
static void insert_n(DB *db, DB_TXN *txn, int firstkey, int n); static void insert_n(DB *db, DB_TXN *txn, int firstkey, int n);
static void verify_dbremove(DB* db);
static int verify_identical_dbts(const DBT *dbt1, const DBT *dbt2); static int verify_identical_dbts(const DBT *dbt1, const DBT *dbt2);
static void verify_sequential_rows(DB* compare_db, int64_t firstkey, int64_t numkeys); static void verify_sequential_rows(DB* compare_db, int64_t firstkey, int64_t numkeys);
static void crash_it(void);
static void do_create(DB* db, char* name, int* next_state) { static DB* do_create(char* name, int* next_state) {
DB* db;
if ( verbose ) printf("%s : do_create(%s)\n", __FILE__, name); if ( verbose ) printf("%s : do_create(%s)\n", __FILE__, name);
int r; int r;
r = db_create(&db, env, 0); DB_TXN* txn;
r = db_create(&db, env, 0);
CKERR(r); CKERR(r);
r = db->open(db, NULL, name, NULL, DB_BTREE, DB_AUTO_COMMIT | DB_CREATE, 0666); r = db->open(db, NULL, name, NULL, DB_BTREE, DB_CREATE, 0666);
CKERR(r); CKERR(r);
DB_TXN* txn; r = env->txn_begin(env, NULL, &txn, 0);
r = env->txn_begin(env, NULL, &txn, 0);
CKERR(r); CKERR(r);
insert_n(db, txn, 0, ROWS_PER_TABLE); insert_n(db, txn, 0, ROWS_PER_TABLE);
r = txn->commit(txn, 0); r = txn->commit(txn, 0);
CKERR(r); CKERR(r);
*next_state = CREATED; *next_state = CREATED;
return db;
} }
static void do_open(DB* db, char* name, int* next_state) { static DB* do_open(char* name, int* next_state) {
DB* db;
if ( verbose ) printf("%s : do_open(%s)\n", __FILE__, name); if ( verbose ) printf("%s : do_open(%s)\n", __FILE__, name);
int r; int r;
r = db_create(&db, env, 0); r = db_create(&db, env, 0);
CKERR(r); CKERR(r);
r = db->open(db, NULL, name, NULL, DB_UNKNOWN, DB_AUTO_COMMIT, 0666); r = db->open(db, NULL, name, NULL, DB_UNKNOWN, 0, 0666);
CKERR(r); CKERR(r);
*next_state = OPEN; *next_state = OPEN;
return db;
} }
static void do_close(DB* db, char *name UU(), int* next_state) { static void do_close(DB* db, char* name, int* next_state) {
if ( verbose ) printf("%s : do_close(%s)\n", __FILE__, name); if ( verbose ) printf("%s : do_close(%s)\n", __FILE__, name);
if (!db) printf("db == NULL\n"); if (!db) printf("db == NULL\n");
int r = db->close(db, 0); int r = db->close(db, 0);
printf("do_closed\n");
CKERR(r); CKERR(r);
*next_state = CLOSED; *next_state = CLOSED;
} }
static void do_delete(DB* db UU(), char* name, int* next_state) { static void do_delete(char* name, int* next_state) {
if ( verbose ) printf("%s : do_delete(%s)\n", __FILE__, name); if ( verbose ) printf("%s : do_delete(%s)\n", __FILE__, name);
int r = env->dbremove(env, NULL, name, NULL, 0); int r = env->dbremove(env, NULL, name, NULL, 0);
CKERR(r); CKERR(r);
*next_state = DELETED; *next_state = DELETED;
} }
static const int percent_do_op = 10; static int do_random_fileop(int i, int state) {
static int do_random_fileop(DB* db, int i, int state) { DB* db = db_array[i];
int rval = random() % 100; int rval = random() % 100;
// if ( verbose ) printf("%s : %s : DB '%d', state '%d, rval '%d'\n", __FILE__, __FUNCTION__, i, state, rval); // if ( verbose ) printf("%s : %s : DB '%d', state '%d, rval '%d'\n", __FILE__, __FUNCTION__, i, state, rval);
...@@ -117,37 +133,52 @@ static int do_random_fileop(DB* db, int i, int state) { ...@@ -117,37 +133,52 @@ static int do_random_fileop(DB* db, int i, int state) {
sprintf(fname, "%s%d.db", table, i); sprintf(fname, "%s%d.db", table, i);
if ( rval < percent_do_op ) { if ( rval < percent_do_op ) {
// if ( verbose ) printf("%s : rval = %d\n", __FILE__, rval);
switch ( state ) { switch ( state ) {
case CREATED: case CREATED:
do_close(db, fname, &next_state); do_close(db, fname, &next_state);
db_array[i] = db = 0;
if ( r < (percent_do_op / 2) ) { if ( r < (percent_do_op / 2) ) {
do_delete(db, fname, &next_state); do_delete(fname, &next_state);
} }
break; break;
case OPEN: case OPEN:
do_close(db, fname, &next_state); do_close(db, fname, &next_state);
db_array[i] = db = 0;
if ( rval < (percent_do_op / 2) ) { if ( rval < (percent_do_op / 2) ) {
do_delete(db, fname, &next_state); do_delete(fname, &next_state);
} }
break; break;
case CLOSED: case CLOSED:
if ( rval < (percent_do_op / 2) ) { if ( rval < (percent_do_op / 2) ) {
do_open(db, fname, &next_state); db = do_open(fname, &next_state);
db_array[i] = db;
} }
else { else {
do_delete(db, fname, &next_state); do_delete(fname, &next_state);
} }
break; break;
case DELETED: case DELETED:
do_create(db, fname, &next_state); db = do_create(fname, &next_state);
db_array[i] = db;
break; break;
} }
} }
return next_state; return next_state;
} }
char *state_db_name="states.db"; static void do_random_fileops(void)
{
int r, i, state, next_state;
DB_TXN *txn;
for (i=0;i<NUM_DICTIONARIES;i++) {
r = env->txn_begin(env, NULL, &txn, 0);
state = get_state(i);
next_state = do_random_fileop(i, state);
put_state(i, next_state);
r = txn->commit(txn, 0);
update_crash_timer();
}
}
static void run_test(int iter, int crash){ static void run_test(int iter, int crash){
u_int32_t recovery_flags = DB_INIT_LOG | DB_INIT_TXN; u_int32_t recovery_flags = DB_INIT_LOG | DB_INIT_TXN;
...@@ -185,14 +216,15 @@ static void run_test(int iter, int crash){ ...@@ -185,14 +216,15 @@ static void run_test(int iter, int crash){
r = db_create(&states, env, 0); CKERR(r); r = db_create(&states, env, 0); CKERR(r);
r = states->open(states, NULL, state_db_name, NULL, DB_UNKNOWN, 0, 0666); CKERR(r); r = states->open(states, NULL, state_db_name, NULL, DB_UNKNOWN, 0, 0666); CKERR(r);
if ( verbose ) printf("%s : === ITERATION %6d ===\n", __FILE__, iter);
// verify previous results // verify previous results
if ( verbose ) printf("%s : verify previous results\n", __FILE__); if ( verbose ) printf("%s : verify previous results\n", __FILE__);
int state = DELETED, next_state; int state = DELETED;
DB* db; DB* db;
char fname[100]; char fname[100];
if ( iter > 0 ) { if ( iter > 0 ) {
for (i=0;i<NUM_DICTIONARIES;i++) { for (i=0;i<NUM_DICTIONARIES;i++) {
db = &db_array[i];
sprintf(fname, "%s%d.db", table, i); sprintf(fname, "%s%d.db", table, i);
state = get_state(i); state = get_state(i);
switch (state) { switch (state) {
...@@ -201,6 +233,7 @@ static void run_test(int iter, int crash){ ...@@ -201,6 +233,7 @@ static void run_test(int iter, int crash){
// open the table // open the table
r = db_create(&db, env, 0); CKERR(r); r = db_create(&db, env, 0); CKERR(r);
r = db->open(db, NULL, fname, NULL, DB_UNKNOWN, 0, 0666); CKERR(r); r = db->open(db, NULL, fname, NULL, DB_UNKNOWN, 0, 0666); CKERR(r);
db_array[i] = db;
verify_sequential_rows(db, 0, ROWS_PER_TABLE); verify_sequential_rows(db, 0, ROWS_PER_TABLE);
// leave table open // leave table open
if (verbose) printf("%s : verified open/created db[%d]\n", __FILE__, i); if (verbose) printf("%s : verified open/created db[%d]\n", __FILE__, i);
...@@ -212,10 +245,14 @@ static void run_test(int iter, int crash){ ...@@ -212,10 +245,14 @@ static void run_test(int iter, int crash){
verify_sequential_rows(db, 0, ROWS_PER_TABLE); verify_sequential_rows(db, 0, ROWS_PER_TABLE);
// close table // close table
r = db->close(db, 0); CKERR(r); r = db->close(db, 0); CKERR(r);
db_array[i] = db = NULL;
if (verbose) printf("%s : verified closed db[%d]\n", __FILE__, i); if (verbose) printf("%s : verified closed db[%d]\n", __FILE__, i);
break; break;
case DELETED: case DELETED:
verify_dbremove(db); r = db_create(&db, env, 0); CKERR(r);
r = db->open(db, NULL, fname, NULL, DB_UNKNOWN, 0, 0666);
if ( r == 0 ) assert(1);
db_array[i] = db = NULL;
if (verbose) printf("%s : verified db[%d] removed\n", __FILE__, i); if (verbose) printf("%s : verified db[%d] removed\n", __FILE__, i);
break; break;
default: default:
...@@ -227,56 +264,71 @@ static void run_test(int iter, int crash){ ...@@ -227,56 +264,71 @@ static void run_test(int iter, int crash){
if ( verbose ) printf("%s : previous results verified\n", __FILE__); if ( verbose ) printf("%s : previous results verified\n", __FILE__);
// for each of the dictionaries, perform a fileop some percentage of time (set in do_random_fileop). // for each of the dictionaries, perform a fileop some percentage of time (set in do_random_fileop).
DB_TXN* txn; // before checkpoint #1
// before checkpoint if ( verbose ) printf("%s : before checkpoint #1\n", __FILE__);
if ( verbose ) printf("%s : before checkpoint\n", __FILE__); crash_timer = NUM_DICTIONARIES + 1; // won't go off
for (i=0;i<NUM_DICTIONARIES;i++) { do_random_fileops();
db = &db_array[i];
r = env->txn_begin(env, NULL, &txn, 0); // during checkpoint #1
state = get_state(i); if ( verbose ) printf("%s : during checkpoint #1\n", __FILE__);
next_state = do_random_fileop(db, i, state); crash_timer = NUM_DICTIONARIES + 1; // won't go off
put_state(i, next_state);
// TODO : change to mix of commit and abort if ( iter & 1 )
r = txn->commit(txn, 0); db_env_set_checkpoint_callback((void (*)(void*))do_random_fileops, NULL);
else
db_env_set_checkpoint_callback2((void (*)(void*))do_random_fileops, NULL);
// checkpoint
r = env->txn_checkpoint(env, 0, 0, 0); CKERR(r);
db_env_set_checkpoint_callback(NULL, NULL);
db_env_set_checkpoint_callback2(NULL, NULL);
// randomly fail sometime during the next 3 phases
// 1) before the next checkpoint
// 2) during the next checkpoint
// 3) after the next (final) checkpoint
if ( iter > 10 ) {
set_crash_timer();
}
else {
crash_timer = ( 3 * NUM_DICTIONARIES ) + 1; // won't go off
} }
// before checkpoint #2
if ( verbose ) printf("%s : before checkpoint #2\n", __FILE__);
do_random_fileops();
// during checkpoint // during checkpoint
if ( verbose ) printf("%s : during checkpoint\n", __FILE__); if ( verbose ) printf("%s : during checkpoint #2\n", __FILE__);
for (i=0;i<NUM_DICTIONARIES;i++) {
db = &db_array[i];
r = env->txn_begin(env, NULL, &txn, 0);
state = get_state(i);
next_state = do_random_fileop(db, i, state);
put_state(i, next_state);
// TODO : change to mix of commit and abort
r = txn->commit(txn, 0);
}
if ( iter & 1 )
db_env_set_checkpoint_callback((void (*)(void*))do_random_fileops, NULL);
else
db_env_set_checkpoint_callback2((void (*)(void*))do_random_fileops, NULL);
// checkpoint // checkpoint
r = env->txn_checkpoint(env, 0, 0, 0); CKERR(r); r = env->txn_checkpoint(env, 0, 0, 0); CKERR(r);
db_env_set_checkpoint_callback(NULL, NULL);
db_env_set_checkpoint_callback2(NULL, NULL);
// after checkpoint // after checkpoint
if ( verbose ) printf("%s : after checkpoint\n", __FILE__); if ( verbose ) printf("%s : after checkpoint #2\n", __FILE__);
for (i=0;i<NUM_DICTIONARIES;i++) { do_random_fileops();
db = &db_array[i];
r = env->txn_begin(env, NULL, &txn, 0);
state = get_state(i);
next_state = do_random_fileop(db, i, state);
put_state(i, next_state);
// TODO : change to mix of commit and abort
r = txn->commit(txn, 0);
}
// close the states table before we crash // close the states table before we crash
r = states->close(states, 0); CKERR(r); r = states->close(states, 0); CKERR(r);
crash = crash; crash = crash;
if (iter > 100 ) if (iter > 10 ) {
if ( verbose ) printf("%s : crash\n", __FILE__);
crash_it(); crash_it();
}
if ( verbose ) printf("%s : done\n", __FILE__);
r = env->txn_checkpoint(env, 0, 0, 0); CKERR(r); r = env->txn_checkpoint(env, 0, 0, 0); CKERR(r);
r = env->close(env, 0); r = env->close(env, 0);
assert((r == 0) || (r == EINVAL)); // OK to have open transactions prior to close assert((r == 0) || (r == EINVAL)); // OK to have open transactions prior to close
} }
// ------------ infrastructure ---------- // ------------ infrastructure ----------
...@@ -352,15 +404,6 @@ static void insert_n(DB *db, DB_TXN *txn, int firstkey, int n) { ...@@ -352,15 +404,6 @@ static void insert_n(DB *db, DB_TXN *txn, int firstkey, int n) {
} }
} }
static void verify_dbremove(DB* db) {
// DBC *cursor;
// DB_TXN *txn;
// int r;
// r = db->cursor(db, txn, &cursor, 0);
// assert(r == EINVAL);
db = db;
}
static int verify_identical_dbts(const DBT *dbt1, const DBT *dbt2) { static int verify_identical_dbts(const DBT *dbt1, const DBT *dbt2) {
int r = 0; int r = 0;
if (dbt1->size != dbt2->size) r = 1; if (dbt1->size != dbt2->size) r = 1;
......
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