mariadb/ft/tests/test-oldest-referenced-xid-flush.cc
John Esmet 1c62ecd4bb refs #5762 add unit test for oldest referenced xid propogation from parent to child
git-svn-id: file:///svn/toku/tokudb@50832 c7de825b-a66e-492c-adef-691d508d4ae1
2013-04-17 00:01:21 -04:00

161 lines
5 KiB
C++

/* -*- mode: C++; c-basic-offset: 4; indent-tabs-mode: nil -*- */
// vim: ft=cpp:expandtab:ts=8:sw=4:softtabstop=4:
#ident "$Id: test-checkpoint-during-flush.cc 50812 2012-12-06 14:15:48Z esmet $"
#ident "Copyright (c) 2007-2012 Tokutek Inc. All rights reserved."
#ident "The technology is licensed by the Massachusetts Institute of Technology, Rutgers State University of New Jersey, and the Research Foundation of State University of New York at Stony Brook under United States of America Serial No. 11/760379 and to the patents and/or patent applications resulting from it."
#include "test.h"
#include "ft-cachetable-wrappers.h"
#include "ft-flusher.h"
#include "ft-flusher-internal.h"
static bool
dont_destroy_bn(void* UU(extra)) {
return false;
}
static void merge_should_not_happen(struct flusher_advice* UU(fa),
FT UU(h),
FTNODE UU(parent),
int UU(childnum),
FTNODE UU(child),
void* UU(extra))
{
assert(false);
}
static bool dont_recursively_flush(FTNODE UU(child), void* UU(extra)) {
return false;
}
static int child_to_flush(FT UU(h), FTNODE parent, void* UU(extra)) {
assert(parent->height == 2);
assert(parent->n_children == 1);
return 0;
}
static void dummy_update_status(FTNODE UU(child), int UU(dirtied), void* UU(extra)) {
}
enum { NODESIZE = 1024, KSIZE=NODESIZE-100, TOKU_PSIZE=20 };
static void test_oldest_referenced_xid_gets_propogated(void) {
int r;
CACHETABLE ct;
FT_HANDLE t;
BLOCKNUM grandchild_leaf_blocknum, child_nonleaf_blocknum, root_blocknum;
toku_cachetable_create(&ct, 500*1024*1024, ZERO_LSN, NULL_LOGGER);
unlink("foo1.ft_handle");
r = toku_open_ft_handle("foo1.ft_handle", 1, &t, NODESIZE, NODESIZE/2, TOKU_DEFAULT_COMPRESSION_METHOD, ct, nullptr, toku_builtin_compare_fun);
assert(r==0);
toku_testsetup_initialize(); // must precede any other toku_testsetup calls
// This test flushes from a nonleaf root to a nonleaf child, without any leaf nodes.
r = toku_testsetup_leaf(t, &grandchild_leaf_blocknum, 1, NULL, NULL);
assert(r==0);
r = toku_testsetup_nonleaf(t, 1, &child_nonleaf_blocknum, 1, &grandchild_leaf_blocknum, NULL, NULL);
assert(r==0);
r = toku_testsetup_nonleaf(t, 2, &root_blocknum, 1, &child_nonleaf_blocknum, NULL, NULL);
assert(r==0);
r = toku_testsetup_root(t, root_blocknum);
assert(r==0);
r = toku_testsetup_insert_to_nonleaf(
t,
root_blocknum,
FT_INSERT,
"a",
2,
NULL,
0
);
// Verify that both the root and its child start with TXNID_NONE
// for the oldest referenced xid
// first verify the child
FTNODE node = NULL;
struct ftnode_fetch_extra bfe;
fill_bfe_for_min_read(&bfe, t->ft);
toku_pin_ftnode_off_client_thread(
t->ft,
child_nonleaf_blocknum,
toku_cachetable_hash(t->ft->cf, child_nonleaf_blocknum),
&bfe,
PL_WRITE_EXPENSIVE,
0,
NULL,
&node
);
assert(node->height == 1);
assert(node->n_children == 1);
assert(BP_BLOCKNUM(node, 0).b == grandchild_leaf_blocknum.b);
assert(node->oldest_known_referenced_xid == TXNID_NONE);
toku_unpin_ftnode(t->ft, node);
// now verify the root - keep it pinned so we can flush it below
toku_pin_ftnode_off_client_thread(
t->ft,
root_blocknum,
toku_cachetable_hash(t->ft->cf, root_blocknum),
&bfe,
PL_WRITE_EXPENSIVE,
0,
NULL,
&node
);
assert(node->height == 2);
assert(node->n_children == 1);
assert(BP_BLOCKNUM(node, 0).b == child_nonleaf_blocknum.b);
assert(toku_bnc_nbytesinbuf(BNC(node, 0)) > 0);
assert(node->oldest_known_referenced_xid == TXNID_NONE);
// set the root's oldest referenced xid to something special
const TXNID flush_xid = 25000;
node->oldest_known_referenced_xid = flush_xid;
// do the flush
struct flusher_advice fa;
flusher_advice_init(
&fa,
child_to_flush,
dont_destroy_bn,
dont_recursively_flush,
merge_should_not_happen,
dummy_update_status,
default_pick_child_after_split,
NULL
);
toku_ft_flush_some_child(t->ft, node, &fa);
// pin the child, verify that oldest referenced xid was
// propogated from parent to child during the flush
toku_pin_ftnode_off_client_thread(
t->ft,
child_nonleaf_blocknum,
toku_cachetable_hash(t->ft->cf, child_nonleaf_blocknum),
&bfe,
PL_WRITE_EXPENSIVE,
0,
NULL,
&node
);
assert(node->oldest_known_referenced_xid == flush_xid);
toku_unpin_ftnode(t->ft, node);
r = toku_close_ft_handle_nolsn(t, 0); assert(r==0);
toku_cachetable_close(&ct);
}
int test_main(int argc __attribute__((__unused__)), const char *argv[] __attribute__((__unused__))) {
default_parse_args(argc, argv);
test_oldest_referenced_xid_gets_propogated();
return 0;
}