Commit c4405b71 authored by Bradley C. Kuszmaul's avatar Bradley C. Kuszmaul Committed by Yoni Fogel

Make the ...dbufio tests better. Refs #2633. [t:2633]

git-svn-id: file:///svn/toku/tokudb@20534 c7de825b-a66e-492c-adef-691d508d4ae1
parent 2ef2752c
...@@ -392,9 +392,6 @@ int toku_brt_loader_internal_init (/* out */ BRTLOADER *blp, ...@@ -392,9 +392,6 @@ int toku_brt_loader_internal_init (/* out */ BRTLOADER *blp,
BL_TRACE(blt_calibrate_done); BL_TRACE(blt_calibrate_done);
#endif #endif
bl->panic = FALSE;
bl->panic_errno = 0;
bl->generate_row_for_put = g; bl->generate_row_for_put = g;
bl->cachetable = cachetable; bl->cachetable = cachetable;
if (bl->cachetable) if (bl->cachetable)
...@@ -465,8 +462,6 @@ int toku_brt_loader_internal_init (/* out */ BRTLOADER *blp, ...@@ -465,8 +462,6 @@ int toku_brt_loader_internal_init (/* out */ BRTLOADER *blp,
if (r != 0) { toku_brtloader_internal_destroy(bl, TRUE); return r; } if (r != 0) { toku_brtloader_internal_destroy(bl, TRUE); return r; }
} }
bl->extractor_live = TRUE;
*blp = bl; *blp = bl;
return 0; return 0;
...@@ -509,7 +504,9 @@ int toku_brt_loader_open (/* out */ BRTLOADER *blp, ...@@ -509,7 +504,9 @@ int toku_brt_loader_open (/* out */ BRTLOADER *blp,
if (result==0) { if (result==0) {
BRTLOADER bl = *blp; BRTLOADER bl = *blp;
int r = toku_pthread_create(&bl->extractor_thread, NULL, extractor_thread, (void*)bl); int r = toku_pthread_create(&bl->extractor_thread, NULL, extractor_thread, (void*)bl);
if (r!=0) { if (r==0) {
bl->extractor_live = TRUE;
} else {
result = r; result = r;
toku_pthread_mutex_destroy(&bl->mutex); toku_pthread_mutex_destroy(&bl->mutex);
toku_brtloader_internal_destroy(bl, TRUE); toku_brtloader_internal_destroy(bl, TRUE);
...@@ -519,17 +516,10 @@ int toku_brt_loader_open (/* out */ BRTLOADER *blp, ...@@ -519,17 +516,10 @@ int toku_brt_loader_open (/* out */ BRTLOADER *blp,
return result; return result;
} }
static void brt_loader_set_panic(BRTLOADER bl, int error) { static void brt_loader_set_panic(BRTLOADER bl, int error, BOOL callback) {
int r = toku_pthread_mutex_lock(&bl->mutex); resource_assert(r == 0); int r = brt_loader_set_error(&bl->error_callback, error, NULL, 0, NULL, NULL);
BOOL is_panic = bl->panic; if (r == 0 && callback)
if (!is_panic) { brt_loader_call_error_function(&bl->error_callback);
bl->panic = TRUE;
bl->panic_errno = error;
}
r = toku_pthread_mutex_unlock(&bl->mutex); resource_assert(r == 0);
if (!is_panic) {
brt_loader_set_error(&bl->error_callback, error, NULL, 0, NULL, NULL);
}
} }
// One of the tests uses this. // One of the tests uses this.
...@@ -877,7 +867,7 @@ static void* extractor_thread (void *blv) { ...@@ -877,7 +867,7 @@ static void* extractor_thread (void *blv) {
{ {
r = process_primary_rows(bl, primary_rowset); r = process_primary_rows(bl, primary_rowset);
if (r) if (r)
brt_loader_set_panic(bl, r); brt_loader_set_panic(bl, r, FALSE);
} }
} }
...@@ -885,7 +875,7 @@ static void* extractor_thread (void *blv) { ...@@ -885,7 +875,7 @@ static void* extractor_thread (void *blv) {
if (r == 0) { if (r == 0) {
r = finish_primary_rows(bl); r = finish_primary_rows(bl);
if (r) if (r)
brt_loader_set_panic(bl, r); brt_loader_set_panic(bl, r, FALSE);
} }
BL_TRACE(blt_extractor); BL_TRACE(blt_extractor);
...@@ -1095,7 +1085,7 @@ int toku_brt_loader_put (BRTLOADER bl, DBT *key, DBT *val) ...@@ -1095,7 +1085,7 @@ int toku_brt_loader_put (BRTLOADER bl, DBT *key, DBT *val)
* Return value: 0 on success, an error number otherwise. * Return value: 0 on success, an error number otherwise.
*/ */
{ {
if (bl->panic || brt_loader_get_error(&bl->error_callback)) if (brt_loader_get_error(&bl->error_callback))
return EINVAL; // previous panic return EINVAL; // previous panic
bl->n_rows++; bl->n_rows++;
// return loader_write_row(key, val, bl->fprimary_rows, &bl->fprimary_offset, bl); // return loader_write_row(key, val, bl->fprimary_rows, &bl->fprimary_offset, bl);
...@@ -1470,53 +1460,51 @@ int toku_merge_some_files_using_dbufio (const BOOL to_q, FIDX dest_data, QUEUE q ...@@ -1470,53 +1460,51 @@ int toku_merge_some_files_using_dbufio (const BOOL to_q, FIDX dest_data, QUEUE q
DBT keys[n_sources]; DBT keys[n_sources];
DBT vals[n_sources]; DBT vals[n_sources];
u_int64_t dataoff[n_sources]; u_int64_t dataoff[n_sources];
DBT zero; memset(&zero, 0, sizeof zero); zero.data=0; zero.flags=DB_DBT_REALLOC; zero.size=0; zero.ulen=0; DBT zero = zero_dbt; zero.flags=DB_DBT_REALLOC;
for (int i=0; i<n_sources; i++) { for (int i=0; i<n_sources; i++) {
keys[i] = vals[i] = zero; // fill these all in with zero so we can delete stuff more reliably. keys[i] = vals[i] = zero; // fill these all in with zero so we can delete stuff more reliably.
} }
pqueue_t *pq; pqueue_t *pq = NULL;
pqueue_node_t *pq_nodes = (pqueue_node_t *)toku_malloc(n_sources * sizeof(pqueue_node_t)); // freed in cleanup pqueue_node_t *pq_nodes = (pqueue_node_t *)toku_malloc(n_sources * sizeof(pqueue_node_t)); // freed in cleanup
invariant(pq_nodes != NULL); if (pq_nodes == NULL) { result = errno; }
{ if (result==0) {
int r = pqueue_init(&pq, n_sources, which_db, dest_db, compare, &bl->error_callback); int r = pqueue_init(&pq, n_sources, which_db, dest_db, compare, &bl->error_callback);
lazy_assert(r == 0); if (r!=0) result = r;
result = r;
} }
u_int64_t n_rows = 0; u_int64_t n_rows = 0;
if ( result == 0 ) { if (result==0) {
// load pqueue with first value from each source
// load pqueue with first value from each source for (int i=0; i<n_sources; i++) {
for (int i=0; i<n_sources; i++) { BL_TRACE_QUIET(blt_do_i);
BL_TRACE_QUIET(blt_do_i); int r = loader_read_row_from_dbufio(bfs, i, &keys[i], &vals[i]);
int r = loader_read_row_from_dbufio(bfs, i, &keys[i], &vals[i]); BL_TRACE_QUIET(blt_read_row);
BL_TRACE_QUIET(blt_read_row); if (r==EOF) continue; // if the file is empty, don't initialize the pqueue.
if (r==EOF) continue; // if the file is empty, don't initialize the pqueue. lazy_assert(r == 0);
lazy_assert(r == 0); if (r!=0) {
if (r!=0) { result = r;
result = r; break;
break; }
}
pq_nodes[i].key = &keys[i]; pq_nodes[i].key = &keys[i];
pq_nodes[i].val = &vals[i]; pq_nodes[i].val = &vals[i];
pq_nodes[i].i = i; pq_nodes[i].i = i;
r = pqueue_insert(pq, &pq_nodes[i]); r = pqueue_insert(pq, &pq_nodes[i]);
if (r!=0) { if (r!=0) {
result = r; result = r;
// path tested by loader-dup-test5.tdbrun // path tested by loader-dup-test5.tdbrun
// printf("%s:%d returning\n", __FILE__, __LINE__); // printf("%s:%d returning\n", __FILE__, __LINE__);
break; break;
} }
dataoff[i] = 0; dataoff[i] = 0;
{ int r2 = toku_pthread_mutex_lock(&bl->file_infos.lock); resource_assert(r2==0); } { int r2 = toku_pthread_mutex_lock(&bl->file_infos.lock); resource_assert(r2==0); }
n_rows += bl->file_infos.file_infos[srcs_fidxs[i].idx].n_rows; n_rows += bl->file_infos.file_infos[srcs_fidxs[i].idx].n_rows;
{ int r2 = toku_pthread_mutex_unlock(&bl->file_infos.lock); resource_assert(r2==0); } { int r2 = toku_pthread_mutex_unlock(&bl->file_infos.lock); resource_assert(r2==0); }
} }
} }
u_int64_t n_rows_done = 0; u_int64_t n_rows_done = 0;
...@@ -1560,15 +1548,15 @@ int toku_merge_some_files_using_dbufio (const BOOL to_q, FIDX dest_data, QUEUE q ...@@ -1560,15 +1548,15 @@ int toku_merge_some_files_using_dbufio (const BOOL to_q, FIDX dest_data, QUEUE q
r = add_row(output_rowset, &keys[mini], &vals[mini]); r = add_row(output_rowset, &keys[mini], &vals[mini]);
lazy_assert(r == 0); lazy_assert(r == 0);
if (r!=0) { if (r!=0) {
return = r; result = r;
break; break;
} }
} else { } else {
// write it to the dest file // write it to the dest file
r = loader_write_row(&keys[mini], &vals[mini], dest_data, dest_stream, &dataoff[mini], bl); r = loader_write_row(&keys[mini], &vals[mini], dest_data, dest_stream, &dataoff[mini], bl);
lazy_assert(r==0); lazy_assert(r==0);
if (r!=0) { if (r!=0) {
return = r; result = r;
break; break;
} }
} }
...@@ -1640,7 +1628,7 @@ int toku_merge_some_files_using_dbufio (const BOOL to_q, FIDX dest_data, QUEUE q ...@@ -1640,7 +1628,7 @@ int toku_merge_some_files_using_dbufio (const BOOL to_q, FIDX dest_data, QUEUE q
destroy_rowset(output_rowset); destroy_rowset(output_rowset);
toku_free(output_rowset); toku_free(output_rowset);
} }
pqueue_free(pq); if (pq) { pqueue_free(pq); pq=NULL; }
toku_free(pq_nodes); toku_free(pq_nodes);
{ {
int r = update_progress(progress_allocation, bl, "end of merge_some_files"); int r = update_progress(progress_allocation, bl, "end of merge_some_files");
...@@ -1758,7 +1746,7 @@ int merge_files (struct merge_fileset *fs, ...@@ -1758,7 +1746,7 @@ int merge_files (struct merge_fileset *fs,
} }
if (result==0 && !to_queue) { if (result==0 && !to_queue) {
result = extend_fileset(bl, &next_file_set, &merged_data); result = extend_fileset(bl, &next_file_set, &merged_data);
if (result!=0) { printf("%s:%d r=%d\n", __FILE__, __LINE__, result); break; } if (result!=0) { break; }
} }
if (result==0) { if (result==0) {
...@@ -1805,7 +1793,7 @@ int merge_files (struct merge_fileset *fs, ...@@ -1805,7 +1793,7 @@ int merge_files (struct merge_fileset *fs,
if (result!=0) break; if (result!=0) break;
} }
if (result) brt_loader_set_panic(bl, result); if (result) brt_loader_set_panic(bl, result, TRUE);
{ {
int r = queue_eof(output_q); int r = queue_eof(output_q);
if (r!=0 && result==0) result = r; if (r!=0 && result==0) result = r;
...@@ -2191,7 +2179,7 @@ static int toku_loader_write_brt_from_q (BRTLOADER bl, ...@@ -2191,7 +2179,7 @@ static int toku_loader_write_brt_from_q (BRTLOADER bl,
BL_TRACE(blt_fractal_deq); BL_TRACE(blt_fractal_deq);
if (rr == EOF) break; if (rr == EOF) break;
if (rr != 0) { if (rr != 0) {
brt_loader_set_panic(bl, rr); // error after cilk sync brt_loader_set_panic(bl, rr, TRUE); // error after cilk sync
break; break;
} }
} }
...@@ -2227,7 +2215,7 @@ static int toku_loader_write_brt_from_q (BRTLOADER bl, ...@@ -2227,7 +2215,7 @@ static int toku_loader_write_brt_from_q (BRTLOADER bl,
n_pivots++; n_pivots++;
if ((r = bl_write_dbt(&key, pivots_stream, NULL, bl))) { if ((r = bl_write_dbt(&key, pivots_stream, NULL, bl))) {
brt_loader_set_panic(bl, r); // error after cilk sync brt_loader_set_panic(bl, r, TRUE); // error after cilk sync
if (result == 0) result = r; if (result == 0) result = r;
break; break;
} }
...@@ -2237,7 +2225,7 @@ static int toku_loader_write_brt_from_q (BRTLOADER bl, ...@@ -2237,7 +2225,7 @@ static int toku_loader_write_brt_from_q (BRTLOADER bl,
r = allocate_block(&out, &lblock); r = allocate_block(&out, &lblock);
if (r != 0) { if (r != 0) {
brt_loader_set_panic(bl, r); brt_loader_set_panic(bl, r, TRUE);
if (result == 0) result = r; if (result == 0) result = r;
break; break;
} }
...@@ -2266,8 +2254,9 @@ static int toku_loader_write_brt_from_q (BRTLOADER bl, ...@@ -2266,8 +2254,9 @@ static int toku_loader_write_brt_from_q (BRTLOADER bl,
cilk_sync; cilk_sync;
if (bl->panic) { // if there were any prior errors then exit if (result == 0) {
result = bl->panic_errno; goto error; result = brt_loader_get_error(&bl->error_callback); // if there were any prior errors then exit
if (result) goto error;
} }
// We haven't paniced, so the sum should add up. // We haven't paniced, so the sum should add up.
...@@ -2579,11 +2568,7 @@ int toku_brt_loader_abort(BRTLOADER bl, BOOL is_error) ...@@ -2579,11 +2568,7 @@ int toku_brt_loader_abort(BRTLOADER bl, BOOL is_error)
} }
int toku_brt_loader_get_error(BRTLOADER bl, int *error) { int toku_brt_loader_get_error(BRTLOADER bl, int *error) {
*error = 0; *error = brt_loader_get_error(&bl->error_callback);
if (bl->panic)
*error = bl->panic_errno;
else if (bl->error_callback.error)
*error = bl->error_callback.error;
return 0; return 0;
} }
...@@ -2730,12 +2715,10 @@ static void finish_leafnode (struct dbout *out, struct leaf_buf *lbuf, int progr ...@@ -2730,12 +2715,10 @@ static void finish_leafnode (struct dbout *out, struct leaf_buf *lbuf, int progr
//printf("Nodewrite %d (%.1f%%):", progress_allocation, 100.0*progress_allocation/PROGRESS_MAX); //printf("Nodewrite %d (%.1f%%):", progress_allocation, 100.0*progress_allocation/PROGRESS_MAX);
if (result == 0) { if (result == 0) {
result = update_progress(progress_allocation, bl, "wrote node"); result = update_progress(progress_allocation, bl, "wrote node");
if (result != 0)
bl->user_said_stop = result;
} }
if (result) if (result)
brt_loader_set_panic(bl, result); brt_loader_set_panic(bl, result, TRUE);
} }
CILK_END CILK_END
...@@ -2988,7 +2971,7 @@ static void write_nonleaf_node (BRTLOADER bl, struct dbout *out, int64_t blocknu ...@@ -2988,7 +2971,7 @@ static void write_nonleaf_node (BRTLOADER bl, struct dbout *out, int64_t blocknu
blocknum_of_new_node = blocknum_of_new_node; blocknum_of_new_node = blocknum_of_new_node;
if (result != 0) if (result != 0)
brt_loader_set_panic(bl, result); brt_loader_set_panic(bl, result, TRUE);
} }
static int write_nonleaves (BRTLOADER bl, FIDX pivots_fidx, struct dbout *out, struct subtrees_info *sts, const struct descriptor *descriptor) { static int write_nonleaves (BRTLOADER bl, FIDX pivots_fidx, struct dbout *out, struct subtrees_info *sts, const struct descriptor *descriptor) {
...@@ -3090,8 +3073,8 @@ static int write_nonleaves (BRTLOADER bl, FIDX pivots_fidx, struct dbout *out, s ...@@ -3090,8 +3073,8 @@ static int write_nonleaves (BRTLOADER bl, FIDX pivots_fidx, struct dbout *out, s
cilk_sync; cilk_sync;
if (result == 0 && bl->panic) // pick up write_nonleaf_node errors if (result == 0) // pick up write_nonleaf_node errors
result = bl->panic_errno; result = brt_loader_get_error(&bl->error_callback);
// Now set things up for the next iteration. // Now set things up for the next iteration.
int r = brtloader_fi_close(&bl->file_infos, pivots_fidx); if (r != 0 && result == 0) result = r; int r = brtloader_fi_close(&bl->file_infos, pivots_fidx); if (r != 0 && result == 0) result = r;
......
...@@ -15,6 +15,10 @@ C_BEGIN ...@@ -15,6 +15,10 @@ C_BEGIN
static int event_count, event_count_trigger; static int event_count, event_count_trigger;
static void my_assert_hook (void) {
fprintf(stderr, "event_count=%d\n", event_count);
}
static void reset_event_counts(void) { static void reset_event_counts(void) {
event_count = event_count_trigger = 0; event_count = event_count_trigger = 0;
} }
...@@ -76,19 +80,83 @@ static ssize_t bad_pwrite(int fd, const void * bp, size_t len, toku_off_t off) { ...@@ -76,19 +80,83 @@ static ssize_t bad_pwrite(int fd, const void * bp, size_t len, toku_off_t off) {
return r; return r;
} }
static int my_malloc_event = 0; static FILE *
bad_fdopen(int fd, const char * mode) {
FILE * rval;
event_count++;
if (event_count_trigger == event_count) {
event_hit();
errno = EINVAL;
rval = NULL;
} else {
rval = fdopen(fd, mode);
}
return rval;
}
static FILE *
bad_fopen(const char *filename, const char *mode) {
FILE * rval;
event_count++;
if (event_count_trigger == event_count) {
event_hit();
errno = EINVAL;
rval = NULL;
} else {
rval = fopen(filename, mode);
}
return rval;
}
static int
bad_open(const char *path, int oflag, int mode) {
int rval;
event_count++;
if (event_count_trigger == event_count) {
event_hit();
errno = EINVAL;
rval = -1;
} else {
rval = open(path, oflag, mode);
}
return rval;
}
static int
bad_fclose(FILE * stream) {
int rval;
event_count++;
// Must close the stream even in the "error case" because otherwise there is no way to get the memory back.
rval = fclose(stream);
if (rval==0) {
if (event_count_trigger == event_count) {
errno = ENOSPC;
rval = -1;
}
}
return rval;
}
static int my_malloc_event = 1;
static int my_malloc_count = 0, my_big_malloc_count = 0; static int my_malloc_count = 0, my_big_malloc_count = 0;
static void reset_my_malloc_counts(void) { static void reset_my_malloc_counts(void) {
my_malloc_count = my_big_malloc_count = 0; my_malloc_count = my_big_malloc_count = 0;
} }
size_t min_malloc_error_size = 0;
static void *my_malloc(size_t n) { static void *my_malloc(size_t n) {
void *caller = __builtin_return_address(0); void *caller = __builtin_return_address(0);
if (!((void*)toku_malloc <= caller && caller <= (void*)toku_free)) if (!((void*)toku_malloc <= caller && caller <= (void*)toku_free))
goto skip; goto skip;
my_malloc_count++; my_malloc_count++;
if (n >= 64*1024) { if (n >= min_malloc_error_size) {
my_big_malloc_count++; my_big_malloc_count++;
if (my_malloc_event) { if (my_malloc_event) {
caller = __builtin_return_address(1); caller = __builtin_return_address(1);
...@@ -194,11 +262,6 @@ static void test (const char *directory, BOOL is_error) { ...@@ -194,11 +262,6 @@ static void test (const char *directory, BOOL is_error) {
assert(r==0); assert(r==0);
} }
toku_set_func_malloc(my_malloc);
brtloader_set_os_fwrite(bad_fwrite);
toku_set_func_write(bad_write);
toku_set_func_pwrite(bad_pwrite);
BRTLOADER bl; BRTLOADER bl;
DB **XMALLOC_N(N_DEST_DBS, dbs); DB **XMALLOC_N(N_DEST_DBS, dbs);
const struct descriptor **XMALLOC_N(N_DEST_DBS, descriptors); const struct descriptor **XMALLOC_N(N_DEST_DBS, descriptors);
...@@ -238,6 +301,7 @@ static void test (const char *directory, BOOL is_error) { ...@@ -238,6 +301,7 @@ static void test (const char *directory, BOOL is_error) {
brt_loader_set_poll_function(&bl->poll_callback, loader_poll_callback, NULL); brt_loader_set_poll_function(&bl->poll_callback, loader_poll_callback, NULL);
QUEUE q; QUEUE q;
{ int r = queue_create(&q, 1000); assert(r==0); } { int r = queue_create(&q, 1000); assert(r==0); }
DBUFIO_FILESET bfs; DBUFIO_FILESET bfs;
...@@ -265,6 +329,17 @@ static void test (const char *directory, BOOL is_error) { ...@@ -265,6 +329,17 @@ static void test (const char *directory, BOOL is_error) {
int r = toku_pthread_create(&consumer, NULL, consumer_thread, (void*)&cthunk); int r = toku_pthread_create(&consumer, NULL, consumer_thread, (void*)&cthunk);
assert(r==0); assert(r==0);
} }
toku_set_func_malloc(my_malloc);
brtloader_set_os_fwrite(bad_fwrite);
toku_set_func_write(bad_write);
toku_set_func_pwrite(bad_pwrite);
toku_set_func_fdopen(bad_fdopen);
toku_set_func_fopen(bad_fopen);
toku_set_func_open(bad_open);
toku_set_func_fclose(bad_fclose);
int result = 0; int result = 0;
{ {
int r = toku_merge_some_files_using_dbufio(TRUE, FIDX_NULL, q, N_SOURCES, bfs, src_fidxs, bl, 0, (DB*)NULL, compare_ints, 10000); int r = toku_merge_some_files_using_dbufio(TRUE, FIDX_NULL, q, N_SOURCES, bfs, src_fidxs, bl, 0, (DB*)NULL, compare_ints, 10000);
...@@ -279,6 +354,17 @@ static void test (const char *directory, BOOL is_error) { ...@@ -279,6 +354,17 @@ static void test (const char *directory, BOOL is_error) {
int r = queue_eof(q); int r = queue_eof(q);
assert(r==0); assert(r==0);
} }
toku_set_func_malloc(NULL);
brtloader_set_os_fwrite(NULL);
toku_set_func_write(NULL);
toku_set_func_pwrite(NULL);
toku_set_func_fdopen(NULL);
toku_set_func_fopen(NULL);
toku_set_func_open(NULL);
toku_set_func_fclose(NULL);
do_assert_hook = my_assert_hook;
{ {
void *vresult; void *vresult;
int r = toku_pthread_join(consumer, &vresult); int r = toku_pthread_join(consumer, &vresult);
...@@ -323,16 +409,18 @@ static void test (const char *directory, BOOL is_error) { ...@@ -323,16 +409,18 @@ static void test (const char *directory, BOOL is_error) {
static int usage(const char *progname, int n) { static int usage(const char *progname, int n) {
fprintf(stderr, "Usage:\n %s [-v] [-q] [-r %d] [-s] [-m] directory\n", progname, n); fprintf(stderr, "Usage:\n %s [-v] [-q] [-r %d] [-s] [-m] [-tend NEVENTS] directory\n", progname, n);
fprintf(stderr, "[-v] turn on verbose\n"); fprintf(stderr, "[-v] turn on verbose\n");
fprintf(stderr, "[-q] turn off verbose\n"); fprintf(stderr, "[-q] turn off verbose\n");
fprintf(stderr, "[-r %d] set the number of rows\n", n); fprintf(stderr, "[-r %d] set the number of rows\n", n);
fprintf(stderr, "[-s] set the small loader size factor\n"); fprintf(stderr, "[-s] set the small loader size factor\n");
fprintf(stderr, "[-m] inject big malloc failures\n"); fprintf(stderr, "[-m] inject big malloc failures\n");
fprintf(stderr, "[-tend NEVENTS] stop testing after N events\n");
return 1; return 1;
} }
int test_main (int argc, const char *argv[]) { int test_main (int argc, const char *argv[]) {
int tend = -1;
const char *progname=argv[0]; const char *progname=argv[0];
argc--; argv++; argc--; argv++;
while (argc>0) { while (argc>0) {
...@@ -349,6 +437,9 @@ int test_main (int argc, const char *argv[]) { ...@@ -349,6 +437,9 @@ int test_main (int argc, const char *argv[]) {
toku_brtloader_set_size_factor(1); toku_brtloader_set_size_factor(1);
} else if (strcmp(argv[0],"-m") == 0) { } else if (strcmp(argv[0],"-m") == 0) {
my_malloc_event = 1; my_malloc_event = 1;
} else if (strcmp(argv[0],"-tend") == 0) {
argc--; argv++;
tend = atoi(argv[0]);
} else if (argc!=1) { } else if (argc!=1) {
return usage(progname, N_RECORDS); return usage(progname, N_RECORDS);
} }
...@@ -381,7 +472,8 @@ int test_main (int argc, const char *argv[]) { ...@@ -381,7 +472,8 @@ int test_main (int argc, const char *argv[]) {
{ {
int event_limit = event_count; int event_limit = event_count;
if (verbose) printf("event_limit=%d\n", event_limit); if (tend>0 && tend<event_limit) event_limit=tend;
if (verbose) printf("event_limit=%d\n", event_limit);
for (int i = 1; i <= event_limit; i++) { for (int i = 1; i <= event_limit; i++) {
reset_event_counts(); reset_event_counts();
......
...@@ -681,4 +681,4 @@ loader-cleanup-test.tdbrun: $(patsubst %,loader-cleanup-test%.tdbrun, 1) ...@@ -681,4 +681,4 @@ loader-cleanup-test.tdbrun: $(patsubst %,loader-cleanup-test%.tdbrun, 1)
# ./loader-cleanup-test.tdb $(SUMMARIZE_CMD) # ./loader-cleanup-test.tdb $(SUMMARIZE_CMD)
loader-cleanup-test1.tdbrun: loader-cleanup-test.tdb$(BINSUF) loader-cleanup-test1.tdbrun: loader-cleanup-test.tdb$(BINSUF)
$(VGRIND) ./loader-cleanup-test.tdb -r 1000 $(SUMMARIZE_CMD) $(VGRIND) ./loader-cleanup-test.tdb -r 4000 -s $(SUMMARIZE_CMD)
...@@ -62,14 +62,15 @@ int abort_on_poll = 0; // set when test_loader() called with test_type of abort ...@@ -62,14 +62,15 @@ int abort_on_poll = 0; // set when test_loader() called with test_type of abort
DB_ENV *env; DB_ENV *env;
enum {MAX_NAME=128}; enum {MAX_NAME=128};
enum {MAX_DBS=256}; enum {MAX_DBS=256};
int NUM_DBS=5; #define default_NUM_DBS 5
int NUM_ROWS=100000; int NUM_DBS=default_NUM_DBS;
#define default_NUM_ROWS 100000
int NUM_ROWS=default_NUM_ROWS;
//static int NUM_ROWS=50000000; //static int NUM_ROWS=50000000;
int CHECK_RESULTS=0; int CHECK_RESULTS=0;
int USE_PUTS=0; int USE_PUTS=0;
int event_trigger_lo=0; // what event triggers to use? int event_trigger_lo=0; // what event triggers to use?
int event_trigger_hi =0; // 0 and 0 mean none. int event_trigger_hi =0; // 0 and 0 mean none.
int assert_temp_files = 0;
enum {MAGIC=311}; enum {MAGIC=311};
...@@ -208,6 +209,7 @@ bad_fopen(const char *filename, const char *mode) { ...@@ -208,6 +209,7 @@ bad_fopen(const char *filename, const char *mode) {
fopen_count++; fopen_count++;
event_count++; event_count++;
if (fopen_count_trigger == fopen_count || event_count == event_count_trigger) { if (fopen_count_trigger == fopen_count || event_count == event_count_trigger) {
printf("Doing fopen_count=%d event_count=%" PRId64 "\n", fopen_count, event_count);
errno = EINVAL; errno = EINVAL;
rval = NULL; rval = NULL;
} else { } else {
...@@ -666,6 +668,7 @@ static void test_loader(enum test_type t, DB **dbs, int trigger) ...@@ -666,6 +668,7 @@ static void test_loader(enum test_type t, DB **dbs, int trigger)
printf(" done\n"); printf(" done\n");
if (t == commit) { if (t == commit) {
event_count_nominal = event_count;
fwrite_count_nominal = fwrite_count; // capture how many fwrites were required for normal operation fwrite_count_nominal = fwrite_count; // capture how many fwrites were required for normal operation
write_count_nominal = write_count; // capture how many writes were required for normal operation write_count_nominal = write_count; // capture how many writes were required for normal operation
pwrite_count_nominal = pwrite_count; // capture how many pwrites were required for normal operation pwrite_count_nominal = pwrite_count; // capture how many pwrites were required for normal operation
...@@ -676,6 +679,7 @@ static void test_loader(enum test_type t, DB **dbs, int trigger) ...@@ -676,6 +679,7 @@ static void test_loader(enum test_type t, DB **dbs, int trigger)
if (verbose) { if (verbose) {
printf("Nominal calls: function calls (number of calls for normal operation)\n"); printf("Nominal calls: function calls (number of calls for normal operation)\n");
printf(" events %" PRId64 "\n", event_count_nominal);
printf(" fwrite %d\n", fwrite_count_nominal); printf(" fwrite %d\n", fwrite_count_nominal);
printf(" write %d\n", write_count_nominal); printf(" write %d\n", write_count_nominal);
printf(" pwrite %d\n", pwrite_count_nominal); printf(" pwrite %d\n", pwrite_count_nominal);
...@@ -707,15 +711,20 @@ static void test_loader(enum test_type t, DB **dbs, int trigger) ...@@ -707,15 +711,20 @@ static void test_loader(enum test_type t, DB **dbs, int trigger)
} }
int run_test_count = 0;
static void run_test(enum test_type t, int trigger) static void run_test(enum test_type t, int trigger)
{ {
run_test_count++;
int r; int r;
if (verbose == 0) { // if no other ouput, give indication of progress if (verbose>0) { // Don't print anything if verbose is 0. Use "+" to indicate progress if verbose is positive
printf("."); printf("+");
fflush(stdout); fflush(stdout);
} }
r = system("rm -rf " ENVDIR); CKERR(r); r = system("rm -rf " ENVDIR); CKERR(r);
r = toku_os_mkdir(ENVDIR, S_IRWXU+S_IRWXG+S_IRWXO); CKERR(r); r = toku_os_mkdir(ENVDIR, S_IRWXU+S_IRWXG+S_IRWXO); CKERR(r);
...@@ -826,43 +835,45 @@ static void run_all_tests(void) { ...@@ -826,43 +835,45 @@ static void run_all_tests(void) {
run_test(abort_txn, 0); run_test(abort_txn, 0);
if (event_trigger_lo || event_trigger_hi) { if (event_trigger_lo || event_trigger_hi) {
if (verbose) printf("\n\nDoing events %d-%d\n", event_trigger_lo, event_trigger_hi); printf("\n\nDoing events %d-%d\n", event_trigger_lo, event_trigger_hi);
for (int i=event_trigger_lo; i<=event_trigger_hi; i++) for (int i=event_trigger_lo; i<=event_trigger_hi; i++) {
run_test(event, i); run_test(event, i);
return;
}
enum test_type et[NUM_ERR_TYPES] = {enospc_f, enospc_w, enospc_p, einval_fdo, einval_fo, einval_o, enospc_fc};
int * nomp[NUM_ERR_TYPES] = {&fwrite_count_nominal, &write_count_nominal, &pwrite_count_nominal,
&fdopen_count_nominal, &fopen_count_nominal, &open_count_nominal, &fclose_count_nominal};
int limit = NUM_DBS * 5;
int j;
for (j = 0; j<NUM_ERR_TYPES; j++) {
enum test_type t = et[j];
const char * write_type = err_type_str(t);
int nominal = *(nomp[j]);
if (verbose)
printf("\nNow test with induced ENOSPC/EINVAL errors returned from %s, nominal = %d\n", write_type, nominal);
int i;
// induce write error at beginning of process
for (i = 1; i < limit && i < nominal+1; i++) {
trigger = i;
if (verbose) printf("\n\nTesting loader with enospc/einval induced at %s count %d (of %d)\n", write_type, trigger, nominal);
run_test(t, trigger);
} }
if (nominal > limit) { // if we didn't already test every possible case } else {
// induce write error sprinkled through process
for (i = 2; i < 5; i++) { enum test_type et[NUM_ERR_TYPES] = {enospc_f, enospc_w, enospc_p, einval_fdo, einval_fo, einval_o, enospc_fc};
trigger = nominal / i; int * nomp[NUM_ERR_TYPES] = {&fwrite_count_nominal, &write_count_nominal, &pwrite_count_nominal,
&fdopen_count_nominal, &fopen_count_nominal, &open_count_nominal, &fclose_count_nominal};
int limit = NUM_DBS * 5;
int j;
for (j = 0; j<NUM_ERR_TYPES; j++) {
enum test_type t = et[j];
const char * write_type = err_type_str(t);
int nominal = *(nomp[j]);
printf("Test %s %d\n", write_type, nominal);
if (verbose)
printf("\nNow test with induced ENOSPC/EINVAL errors returned from %s, nominal = %d\n", write_type, nominal);
int i;
// induce write error at beginning of process
for (i = 1; i < limit && i < nominal+1; i++) {
trigger = i;
if (verbose) printf("\n\nTesting loader with enospc/einval induced at %s count %d (of %d)\n", write_type, trigger, nominal); if (verbose) printf("\n\nTesting loader with enospc/einval induced at %s count %d (of %d)\n", write_type, trigger, nominal);
run_test(t, trigger); run_test(t, trigger);
} }
// induce write error at end of process if (nominal > limit) { // if we didn't already test every possible case
for (i = 0; i < limit; i++) { // induce write error sprinkled through process
trigger = nominal - i; for (i = 2; i < 5; i++) {
assert(trigger > 0); trigger = nominal / i;
if (verbose) printf("\n\nTesting loader with enospc/einval induced at %s count %d (of %d)\n", write_type, trigger, nominal); if (verbose) printf("\n\nTesting loader with enospc/einval induced at %s count %d (of %d)\n", write_type, trigger, nominal);
run_test(t, trigger); run_test(t, trigger);
}
// induce write error at end of process
for (i = 0; i < limit; i++) {
trigger = nominal - i;
assert(trigger > 0);
if (verbose) printf("\n\nTesting loader with enospc/einval induced at %s count %d (of %d)\n", write_type, trigger, nominal);
run_test(t, trigger);
}
} }
} }
} }
...@@ -871,15 +882,9 @@ static void run_all_tests(void) { ...@@ -871,15 +882,9 @@ static void run_all_tests(void) {
int test_main(int argc, char * const *argv) { int test_main(int argc, char * const *argv) {
do_args(argc, argv); do_args(argc, argv);
printf("\nTesting loader with default size_factor\n");
assert_temp_files = 0;
run_all_tests(); run_all_tests();
printf("\nTesting loader with size_factor=1\n"); printf("run_test_count=%d\n", run_test_count);
db_env_set_loader_size_factor(1);
assert_temp_files = 1;
run_all_tests();
printf("\nTotal event_trigger count is %" PRId64 ". (Use with -t N M, where 1<=N<=M<=%" PRId64 "\n", event_count, event_count);
return 0; return 0;
} }
...@@ -897,7 +902,14 @@ static void do_args(int argc, char * const argv[]) { ...@@ -897,7 +902,14 @@ static void do_args(int argc, char * const argv[]) {
} else if (strcmp(argv[0], "-h")==0) { } else if (strcmp(argv[0], "-h")==0) {
resultcode=0; resultcode=0;
do_usage: do_usage:
fprintf(stderr, "Usage: -h -c -s -p -d <num_dbs> -r <num_rows>\n%s\n", cmd); fprintf(stderr, "Usage: -h -c -s -p -d <num_dbs> -r <num_rows> -t <elow> <ehi> \n%s\n", cmd);
fprintf(stderr, " where -h print this message.\n");
fprintf(stderr, " -c check the results.\n");
fprintf(stderr, " -p LOADER_USE_PUTS.\n");
fprintf(stderr, " -s size_factor=1.\n");
fprintf(stderr, " -d <num_dbs> Number of indexes to create (default=%d).\n", default_NUM_DBS);
fprintf(stderr, " -r <num_rows> Number of rows to put (default=%d).\n", default_NUM_ROWS);
fprintf(stderr, " -t <elo> <ehi> Instrument only events <elo> to <ehi> (default: instrument all).\n");
exit(resultcode); exit(resultcode);
} else if (strcmp(argv[0], "-d")==0) { } else if (strcmp(argv[0], "-d")==0) {
argc--; argv++; argc--; argv++;
...@@ -921,6 +933,8 @@ static void do_args(int argc, char * const argv[]) { ...@@ -921,6 +933,8 @@ static void do_args(int argc, char * const argv[]) {
event_trigger_lo = atoi(argv[0]); event_trigger_lo = atoi(argv[0]);
argc--; argv++; argc--; argv++;
event_trigger_hi = atoi(argv[0]); event_trigger_hi = atoi(argv[0]);
} else if (strcmp(argv[0], "-s")==0) {
db_env_set_loader_size_factor(1);
} else { } else {
fprintf(stderr, "Unknown arg: %s\n", argv[0]); fprintf(stderr, "Unknown arg: %s\n", argv[0]);
resultcode=1; resultcode=1;
......
...@@ -4,10 +4,9 @@ ...@@ -4,10 +4,9 @@
/* This version will complain if NDEBUG is set. */ /* This version will complain if NDEBUG is set. */
/* It evaluates the argument and then calls a function toku_do_assert() which takes all the hits for the branches not taken. */ /* It evaluates the argument and then calls a function toku_do_assert() which takes all the hits for the branches not taken. */
#if defined(__cplusplus) || defined(__cilkplusplus) #include "c_dialects.h"
extern "C" {
#endif
C_BEGIN
#ifdef NDEBUG #ifdef NDEBUG
#error NDEBUG should not be set #error NDEBUG should not be set
...@@ -19,6 +18,8 @@ void toku_do_assert(int,const char*/*expr_as_string*/,const char */*fun*/,const ...@@ -19,6 +18,8 @@ void toku_do_assert(int,const char*/*expr_as_string*/,const char */*fun*/,const
// Define GCOV if you want to get test-coverage information that ignores the assert statements. // Define GCOV if you want to get test-coverage information that ignores the assert statements.
// #define GCOV // #define GCOV
extern void (*do_assert_hook)(void); // Set this to a function you want called after printing the assertion failure message but before calling abort(). By default this is NULL.
#if defined(GCOV) || TOKU_WINDOWS #if defined(GCOV) || TOKU_WINDOWS
#define assert(expr) toku_do_assert((expr) != 0, #expr, __FUNCTION__, __FILE__, __LINE__) #define assert(expr) toku_do_assert((expr) != 0, #expr, __FUNCTION__, __FILE__, __LINE__)
#else #else
...@@ -37,8 +38,6 @@ void toku_do_assert(int,const char*/*expr_as_string*/,const char */*fun*/,const ...@@ -37,8 +38,6 @@ void toku_do_assert(int,const char*/*expr_as_string*/,const char */*fun*/,const
#define invariant(a) assert(a) // indicates a code invariant that must be true #define invariant(a) assert(a) // indicates a code invariant that must be true
#define resource_assert(a) assert(a) // indicates resource must be available, otherwise unrecoverable #define resource_assert(a) assert(a) // indicates resource must be available, otherwise unrecoverable
#if defined(__cplusplus) || defined(__cilkplusplus) C_END
}
#endif
#endif #endif
...@@ -16,6 +16,8 @@ ...@@ -16,6 +16,8 @@
static void *backtrace_pointers[N_POINTERS]; static void *backtrace_pointers[N_POINTERS];
#endif #endif
void (*do_assert_hook)(void) = NULL;
void toku_do_assert_fail (const char* expr_as_string,const char *function,const char*file,int line) void toku_do_assert_fail (const char* expr_as_string,const char *function,const char*file,int line)
{ {
fprintf(stderr, "%s:%d %s: Assertion `%s' failed\n", file,line,function,expr_as_string); fprintf(stderr, "%s:%d %s: Assertion `%s' failed\n", file,line,function,expr_as_string);
...@@ -44,6 +46,9 @@ void toku_do_assert_fail (const char* expr_as_string,const char *function,const ...@@ -44,6 +46,9 @@ void toku_do_assert_fail (const char* expr_as_string,const char *function,const
TerminateProcess(GetCurrentProcess(), 134); //Only way found so far to unconditionally TerminateProcess(GetCurrentProcess(), 134); //Only way found so far to unconditionally
//Terminate the process //Terminate the process
#endif #endif
if (do_assert_hook) do_assert_hook();
abort(); abort();
} }
......
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