2013-04-17 00:00:36 -04:00
|
|
|
/* -*- mode: C; c-basic-offset: 4; indent-tabs-mode: nil -*- */
|
|
|
|
// vim: expandtab:ts=8:sw=4:softtabstop=4:
|
2013-04-17 00:00:05 -04:00
|
|
|
#ident "$Id: test-del-inorder.c 32975 2011-07-11 23:42:51Z leifwalsh $"
|
|
|
|
/* The goal of this test. Make sure that inserts stay behind deletes. */
|
|
|
|
|
|
|
|
|
|
|
|
#include "test.h"
|
|
|
|
#include "includes.h"
|
2013-04-17 00:00:35 -04:00
|
|
|
#include <ft-cachetable-wrappers.h>
|
|
|
|
#include "ft-flusher.h"
|
|
|
|
#include "ft-flusher-internal.h"
|
2013-04-17 00:00:05 -04:00
|
|
|
#include "checkpoint.h"
|
|
|
|
|
|
|
|
static TOKUTXN const null_txn = 0;
|
|
|
|
static DB * const null_db = 0;
|
|
|
|
|
|
|
|
enum { NODESIZE = 1024, KSIZE=NODESIZE-100, TOKU_PSIZE=20 };
|
|
|
|
|
|
|
|
CACHETABLE ct;
|
2013-04-17 00:00:35 -04:00
|
|
|
FT_HANDLE t;
|
2013-04-17 00:00:05 -04:00
|
|
|
|
|
|
|
BOOL checkpoint_called;
|
|
|
|
BOOL checkpoint_callback_called;
|
|
|
|
toku_pthread_t checkpoint_tid;
|
|
|
|
|
|
|
|
|
|
|
|
// callback functions for flush_some_child
|
|
|
|
static bool
|
|
|
|
dont_destroy_bn(void* UU(extra))
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
static void merge_should_not_happen(struct flusher_advice* UU(fa),
|
2013-04-17 00:00:35 -04:00
|
|
|
FT UU(h),
|
|
|
|
FTNODE UU(parent),
|
2013-04-17 00:00:05 -04:00
|
|
|
int UU(childnum),
|
2013-04-17 00:00:35 -04:00
|
|
|
FTNODE UU(child),
|
2013-04-17 00:00:05 -04:00
|
|
|
void* UU(extra))
|
|
|
|
{
|
|
|
|
assert(FALSE);
|
|
|
|
}
|
|
|
|
|
2013-04-17 00:00:35 -04:00
|
|
|
static bool recursively_flush_should_not_happen(FTNODE UU(child), void* UU(extra)) {
|
2013-04-17 00:00:05 -04:00
|
|
|
assert(FALSE);
|
|
|
|
}
|
|
|
|
|
2013-04-17 00:00:35 -04:00
|
|
|
static int child_to_flush(FT UU(h), FTNODE parent, void* UU(extra)) {
|
2013-04-17 00:00:05 -04:00
|
|
|
assert(parent->height == 1);
|
|
|
|
assert(parent->n_children == 1);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2013-04-17 00:00:35 -04:00
|
|
|
static void dummy_update_status(FTNODE UU(child), int UU(dirtied), void* UU(extra)) {
|
2013-04-17 00:00:05 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void checkpoint_callback(void* UU(extra)) {
|
|
|
|
usleep(1*1024*1024);
|
|
|
|
checkpoint_callback_called = TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void *do_checkpoint(void *arg) {
|
|
|
|
// first verify that checkpointed_data is correct;
|
|
|
|
if (verbose) printf("starting a checkpoint\n");
|
|
|
|
int r = toku_checkpoint(ct, NULL, checkpoint_callback, NULL, NULL, NULL, CLIENT_CHECKPOINT);
|
|
|
|
assert_zero(r);
|
|
|
|
if (verbose) printf("completed a checkpoint\n");
|
|
|
|
return arg;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void flusher_callback(int state, void* extra) {
|
|
|
|
BOOL after_child_pin = *(BOOL *)extra;
|
|
|
|
if (verbose) {
|
|
|
|
printf("state %d\n", state);
|
|
|
|
}
|
2013-04-17 00:00:35 -04:00
|
|
|
if ((state == flt_flush_before_child_pin && !after_child_pin) ||
|
|
|
|
(state == ft_flush_aflter_child_pin && after_child_pin)) {
|
2013-04-17 00:00:05 -04:00
|
|
|
checkpoint_called = TRUE;
|
|
|
|
int r = toku_pthread_create(&checkpoint_tid, NULL, do_checkpoint, NULL);
|
|
|
|
assert_zero(r);
|
|
|
|
while (!checkpoint_callback_called) {
|
|
|
|
usleep(1*1024*1024);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
doit (BOOL after_child_pin) {
|
|
|
|
BLOCKNUM node_leaf, node_root;
|
|
|
|
|
|
|
|
int r;
|
|
|
|
checkpoint_called = FALSE;
|
|
|
|
checkpoint_callback_called = FALSE;
|
|
|
|
|
|
|
|
toku_flusher_thread_set_callback(flusher_callback, &after_child_pin);
|
|
|
|
|
2013-04-17 00:00:35 -04:00
|
|
|
r = toku_create_cachetable(&ct, 500*1024*1024, ZERO_LSN, NULL_LOGGER); assert(r==0);
|
|
|
|
unlink("foo1.ft_handle");
|
|
|
|
r = toku_open_ft_handle("foo1.ft_handle", 1, &t, NODESIZE, NODESIZE/2, TOKU_DEFAULT_COMPRESSION_METHOD, ct, null_txn, toku_builtin_compare_fun);
|
2013-04-17 00:00:05 -04:00
|
|
|
assert(r==0);
|
|
|
|
|
|
|
|
toku_testsetup_initialize(); // must precede any other toku_testsetup calls
|
|
|
|
|
|
|
|
r = toku_testsetup_leaf(t, &node_leaf, 1, NULL, NULL);
|
|
|
|
assert(r==0);
|
|
|
|
|
|
|
|
r = toku_testsetup_nonleaf(t, 1, &node_root, 1, &node_leaf, 0, 0);
|
|
|
|
assert(r==0);
|
|
|
|
|
|
|
|
r = toku_testsetup_root(t, node_root);
|
|
|
|
assert(r==0);
|
|
|
|
|
|
|
|
|
|
|
|
r = toku_testsetup_insert_to_nonleaf(
|
|
|
|
t,
|
|
|
|
node_root,
|
2013-04-17 00:00:35 -04:00
|
|
|
FT_INSERT,
|
2013-04-17 00:00:05 -04:00
|
|
|
"a",
|
|
|
|
2,
|
|
|
|
NULL,
|
|
|
|
0
|
|
|
|
);
|
|
|
|
|
|
|
|
// at this point, we have inserted a message into
|
|
|
|
// the root, and we wish to flush it, the leaf
|
|
|
|
// should be empty
|
|
|
|
|
|
|
|
struct flusher_advice fa;
|
|
|
|
flusher_advice_init(
|
|
|
|
&fa,
|
|
|
|
child_to_flush,
|
|
|
|
dont_destroy_bn,
|
|
|
|
recursively_flush_should_not_happen,
|
|
|
|
merge_should_not_happen,
|
|
|
|
dummy_update_status,
|
|
|
|
default_pick_child_after_split,
|
|
|
|
NULL
|
|
|
|
);
|
|
|
|
|
2013-04-17 00:00:35 -04:00
|
|
|
FTNODE node = NULL;
|
|
|
|
struct ftnode_fetch_extra bfe;
|
2013-04-17 00:00:36 -04:00
|
|
|
fill_bfe_for_min_read(&bfe, t->ft);
|
2013-04-17 00:00:35 -04:00
|
|
|
toku_pin_ftnode_off_client_thread(
|
2013-04-17 00:00:36 -04:00
|
|
|
t->ft,
|
2013-04-17 00:00:05 -04:00
|
|
|
node_root,
|
2013-04-17 00:00:36 -04:00
|
|
|
toku_cachetable_hash(t->ft->cf, node_root),
|
2013-04-17 00:00:05 -04:00
|
|
|
&bfe,
|
2013-04-17 00:00:13 -04:00
|
|
|
TRUE,
|
2013-04-17 00:00:05 -04:00
|
|
|
0,
|
|
|
|
NULL,
|
|
|
|
&node
|
|
|
|
);
|
|
|
|
assert(node->height == 1);
|
|
|
|
assert(node->n_children == 1);
|
|
|
|
assert(toku_bnc_nbytesinbuf(BNC(node, 0)) > 0);
|
|
|
|
|
|
|
|
// do the flush
|
2013-04-17 00:00:36 -04:00
|
|
|
flush_some_child(t->ft, node, &fa);
|
2013-04-17 00:00:05 -04:00
|
|
|
assert(checkpoint_callback_called);
|
|
|
|
|
|
|
|
// now let's pin the root again and make sure it is flushed
|
2013-04-17 00:00:35 -04:00
|
|
|
toku_pin_ftnode_off_client_thread(
|
2013-04-17 00:00:36 -04:00
|
|
|
t->ft,
|
2013-04-17 00:00:05 -04:00
|
|
|
node_root,
|
2013-04-17 00:00:36 -04:00
|
|
|
toku_cachetable_hash(t->ft->cf, node_root),
|
2013-04-17 00:00:05 -04:00
|
|
|
&bfe,
|
2013-04-17 00:00:13 -04:00
|
|
|
TRUE,
|
2013-04-17 00:00:05 -04:00
|
|
|
0,
|
|
|
|
NULL,
|
|
|
|
&node
|
|
|
|
);
|
|
|
|
assert(node->height == 1);
|
|
|
|
assert(node->n_children == 1);
|
|
|
|
assert(toku_bnc_nbytesinbuf(BNC(node, 0)) == 0);
|
2013-04-17 00:00:36 -04:00
|
|
|
toku_unpin_ftnode(t->ft, node);
|
2013-04-17 00:00:05 -04:00
|
|
|
|
|
|
|
void *ret;
|
|
|
|
r = toku_pthread_join(checkpoint_tid, &ret);
|
|
|
|
assert_zero(r);
|
|
|
|
|
|
|
|
//
|
|
|
|
// now the dictionary has been checkpointed
|
|
|
|
// copy the file to something with a new name,
|
|
|
|
// open it, and verify that the state of what is
|
|
|
|
// checkpointed is what we expect
|
|
|
|
//
|
|
|
|
|
2013-04-17 00:00:35 -04:00
|
|
|
r = system("cp foo1.ft_handle bar1.ft_handle ");
|
2013-04-17 00:00:05 -04:00
|
|
|
assert_zero(r);
|
|
|
|
|
2013-04-17 00:00:35 -04:00
|
|
|
FT_HANDLE c_ft;
|
|
|
|
r = toku_open_ft_handle("bar1.ft_handle", 0, &c_ft, NODESIZE, NODESIZE/2, TOKU_DEFAULT_COMPRESSION_METHOD, ct, null_txn, toku_builtin_compare_fun);
|
2013-04-17 00:00:05 -04:00
|
|
|
assert(r==0);
|
|
|
|
|
|
|
|
//
|
|
|
|
// now pin the root, verify that we have a message in there, and that it is clean
|
|
|
|
//
|
2013-04-17 00:00:36 -04:00
|
|
|
fill_bfe_for_full_read(&bfe, c_ft->ft);
|
2013-04-17 00:00:35 -04:00
|
|
|
toku_pin_ftnode_off_client_thread(
|
2013-04-17 00:00:36 -04:00
|
|
|
c_ft->ft,
|
2013-04-17 00:00:05 -04:00
|
|
|
node_root,
|
2013-04-17 00:00:36 -04:00
|
|
|
toku_cachetable_hash(c_ft->ft->cf, node_root),
|
2013-04-17 00:00:05 -04:00
|
|
|
&bfe,
|
2013-04-17 00:00:13 -04:00
|
|
|
TRUE,
|
2013-04-17 00:00:05 -04:00
|
|
|
0,
|
|
|
|
NULL,
|
|
|
|
&node
|
|
|
|
);
|
|
|
|
assert(node->height == 1);
|
|
|
|
assert(!node->dirty);
|
|
|
|
assert(node->n_children == 1);
|
|
|
|
if (after_child_pin) {
|
|
|
|
assert(toku_bnc_nbytesinbuf(BNC(node, 0)) == 0);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
assert(toku_bnc_nbytesinbuf(BNC(node, 0)) > 0);
|
|
|
|
}
|
2013-04-17 00:00:36 -04:00
|
|
|
toku_unpin_ftnode_off_client_thread(c_ft->ft, node);
|
2013-04-17 00:00:05 -04:00
|
|
|
|
2013-04-17 00:00:35 -04:00
|
|
|
toku_pin_ftnode_off_client_thread(
|
2013-04-17 00:00:36 -04:00
|
|
|
c_ft->ft,
|
2013-04-17 00:00:05 -04:00
|
|
|
node_leaf,
|
2013-04-17 00:00:36 -04:00
|
|
|
toku_cachetable_hash(c_ft->ft->cf, node_root),
|
2013-04-17 00:00:05 -04:00
|
|
|
&bfe,
|
2013-04-17 00:00:13 -04:00
|
|
|
TRUE,
|
2013-04-17 00:00:05 -04:00
|
|
|
0,
|
|
|
|
NULL,
|
|
|
|
&node
|
|
|
|
);
|
|
|
|
assert(node->height == 0);
|
|
|
|
assert(!node->dirty);
|
|
|
|
assert(node->n_children == 1);
|
|
|
|
if (after_child_pin) {
|
|
|
|
assert(BLB_NBYTESINBUF(node,0) > 0);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
assert(BLB_NBYTESINBUF(node,0) == 0);
|
|
|
|
}
|
2013-04-17 00:00:36 -04:00
|
|
|
toku_unpin_ftnode_off_client_thread(c_ft->ft, node);
|
2013-04-17 00:00:05 -04:00
|
|
|
|
|
|
|
struct check_pair pair1 = {2, "a", 0, NULL, 0};
|
|
|
|
DBT k;
|
2013-04-17 00:00:35 -04:00
|
|
|
r = toku_ft_lookup(c_ft, toku_fill_dbt(&k, "a", 2), lookup_checkf, &pair1);
|
2013-04-17 00:00:05 -04:00
|
|
|
assert(r==0);
|
|
|
|
|
|
|
|
|
2013-04-17 00:00:35 -04:00
|
|
|
r = toku_close_ft_handle_nolsn(t, 0); assert(r==0);
|
|
|
|
r = toku_close_ft_handle_nolsn(c_ft, 0); assert(r==0);
|
2013-04-17 00:00:05 -04:00
|
|
|
r = toku_cachetable_close(&ct); assert(r==0);
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
test_main (int argc __attribute__((__unused__)), const char *argv[] __attribute__((__unused__))) {
|
2013-04-17 00:00:05 -04:00
|
|
|
default_parse_args(argc, argv);
|
2013-04-17 00:00:05 -04:00
|
|
|
doit(FALSE);
|
|
|
|
doit(TRUE);
|
|
|
|
return 0;
|
|
|
|
}
|