/* -*- mode: C; c-basic-offset: 4 -*- */ #ident "$Id$" #ident "Copyright (c) 2011 Tokutek Inc. All rights reserved." // generate fractal trees with a given height, fanout, and number of leaf elements per leaf. // jam the child buffers with inserts. // this code can be used as a template to build broken trees // // This program (copied from make-tree.c) creates a tree with bad msns by commenting out // the setting of the msn: // // To correctly set msn per node: // - set in each non-leaf when message is injected into node (see insert_into_child_buffer()) // - set in each leaf node (see append_leaf()) // - set in root node (set test_make_tree()) #include "includes.h" #include "test.h" static BRTNODE make_node(BRT brt, int height) { BRTNODE node = NULL; int n_children = (height == 0) ? 1 : 0; toku_create_new_brtnode(brt, &node, height, n_children); if (n_children) BP_STATE(node,0) = PT_AVAIL; return node; } static void append_leaf(BRTNODE leafnode, void *key, size_t keylen, void *val, size_t vallen) { assert(leafnode->height == 0); DBT thekey; toku_fill_dbt(&thekey, key, keylen); DBT theval; toku_fill_dbt(&theval, val, vallen); // get an index that we can use to create a new leaf entry uint32_t idx = toku_omt_size(BLB_BUFFER(leafnode, 0)); MSN msn = next_dummymsn(); // apply an insert to the leaf node BRT_MSG_S cmd = { BRT_INSERT, msn, xids_get_root_xids(), .u.id = { &thekey, &theval } }; brt_leaf_apply_cmd_once(BLB(leafnode, 0), &BP_SUBTREE_EST(leafnode,0), &cmd, idx, NULL, NULL, NULL); // Create bad tree (don't do following): // leafnode->max_msn_applied_to_node = msn; // dont forget to dirty the node leafnode->dirty = 1; } static void populate_leaf(BRTNODE leafnode, int seq, int n, int *minkey, int *maxkey) { for (int i = 0; i < n; i++) { int k = htonl(seq + i); int v = seq + i; append_leaf(leafnode, &k, sizeof k, &v, sizeof v); } *minkey = htonl(seq); *maxkey = htonl(seq + n - 1); } static void insert_into_child_buffer(BRT brt, BRTNODE node, int childnum, int minkey, int maxkey) { for (unsigned int val = htonl(minkey); val <= htonl(maxkey); val++) { MSN msn = next_dummymsn(); unsigned int key = htonl(val); DBT thekey; toku_fill_dbt(&thekey, &key, sizeof key); DBT theval; toku_fill_dbt(&theval, &val, sizeof val); toku_brt_append_to_child_buffer(brt, node, childnum, BRT_INSERT, msn, xids_get_root_xids(), true, &thekey, &theval); // Create bad tree (don't do following): // node->max_msn_applied_to_node = msn; } } static BRTNODE make_tree(BRT brt, int height, int fanout, int nperleaf, int *seq, int *minkey, int *maxkey) { BRTNODE node; if (height == 0) { node = make_node(brt, 0); populate_leaf(node, *seq, nperleaf, minkey, maxkey); *seq += nperleaf; } else { node = make_node(brt, height); int minkeys[fanout], maxkeys[fanout]; for (int childnum = 0; childnum < fanout; childnum++) { BRTNODE child = make_tree(brt, height-1, fanout, nperleaf, seq, &minkeys[childnum], &maxkeys[childnum]); if (childnum == 0) toku_brt_nonleaf_append_child(node, child, NULL, 0); else { int k = maxkeys[childnum-1]; // use the max of the left tree struct kv_pair *pivotkey = kv_pair_malloc(&k, sizeof k, NULL, 0); toku_brt_nonleaf_append_child(node, child, pivotkey, sizeof k); } toku_unpin_brtnode(brt, child); insert_into_child_buffer(brt, node, childnum, minkeys[childnum], maxkeys[childnum]); } *minkey = minkeys[0]; *maxkey = maxkeys[0]; for (int i = 1; i < fanout; i++) { if (memcmp(minkey, &minkeys[i], sizeof minkeys[i]) > 0) *minkey = minkeys[i]; if (memcmp(maxkey, &maxkeys[i], sizeof maxkeys[i]) < 0) *maxkey = maxkeys[i]; } } return node; } static UU() void deleted_row(UU() DB *db, UU() DBT *key, UU() DBT *val) { } static void test_make_tree(int height, int fanout, int nperleaf, int do_verify) { int r; // cleanup char fname[]= __FILE__ ".brt"; r = unlink(fname); assert(r == 0 || (r == -1 && errno == ENOENT)); // create a cachetable CACHETABLE ct = NULL; r = toku_brt_create_cachetable(&ct, 0, ZERO_LSN, NULL_LOGGER); assert(r == 0); // create the brt TOKUTXN null_txn = NULL; DB *null_db = NULL; BRT brt = NULL; r = toku_open_brt(fname, 1, &brt, 1024, 256, ct, null_txn, toku_builtin_compare_fun, null_db); assert(r == 0); // make a tree int seq = 0, minkey, maxkey; BRTNODE newroot = make_tree(brt, height, fanout, nperleaf, &seq, &minkey, &maxkey); // discard the old root block u_int32_t fullhash = 0; CACHEKEY *rootp; rootp = toku_calculate_root_offset_pointer(brt, &fullhash); // set the new root to point to the new tree *rootp = newroot->thisnodename; // Create bad tree (don't do following): // newroot->max_msn_applied_to_node = last_dummymsn(); // capture msn of last message injected into tree // unpin the new root toku_unpin_brtnode(brt, newroot); if (do_verify) { r = toku_verify_brt(brt); assert(r != 0); } // flush to the file system r = toku_close_brt(brt, 0); assert(r == 0); // shutdown the cachetable r = toku_cachetable_close(&ct); assert(r == 0); } static int usage(void) { return 1; } int test_main (int argc , const char *argv[]) { int height = 1; int fanout = 2; int nperleaf = 8; int do_verify = 1; for (int i = 1; i < argc; i++) { const char *arg = argv[i]; if (strcmp(arg, "-v") == 0) { verbose++; continue; } if (strcmp(arg, "-q") == 0) { verbose = 0; continue; } if (strcmp(arg, "--height") == 0 && i+1 < argc) { height = atoi(argv[++i]); continue; } if (strcmp(arg, "--fanout") == 0 && i+1 < argc) { fanout = atoi(argv[++i]); continue; } if (strcmp(arg, "--nperleaf") == 0 && i+1 < argc) { nperleaf = atoi(argv[++i]); continue; } if (strcmp(arg, "--verify") == 0 && i+1 < argc) { do_verify = atoi(argv[++i]); continue; } return usage(); } test_make_tree(height, fanout, nperleaf, do_verify); return 0; }