/* -*- mode: C++; c-basic-offset: 4; indent-tabs-mode: nil -*- */ // vim: ft=cpp:expandtab:ts=8:sw=4:softtabstop=4: #ident "$Id$" /* COPYING CONDITIONS NOTICE: This program is free software; you can redistribute it and/or modify it under the terms of version 2 of the GNU General Public License as published by the Free Software Foundation, and provided that the following conditions are met: * Redistributions of source code must retain this COPYING CONDITIONS NOTICE, the COPYRIGHT NOTICE (below), the DISCLAIMER (below), the UNIVERSITY PATENT NOTICE (below), the PATENT MARKING NOTICE (below), and the PATENT RIGHTS GRANT (below). * Redistributions in binary form must reproduce this COPYING CONDITIONS NOTICE, the COPYRIGHT NOTICE (below), the DISCLAIMER (below), the UNIVERSITY PATENT NOTICE (below), the PATENT MARKING NOTICE (below), and the PATENT RIGHTS GRANT (below) in the documentation and/or other materials provided with the distribution. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. COPYRIGHT NOTICE: TokuDB, Tokutek Fractal Tree Indexing Library. Copyright (C) 2007-2013 Tokutek, Inc. DISCLAIMER: This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. UNIVERSITY PATENT NOTICE: 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. PATENT MARKING NOTICE: This software is covered by US Patent No. 8,185,551. PATENT RIGHTS GRANT: "THIS IMPLEMENTATION" means the copyrightable works distributed by Tokutek as part of the Fractal Tree project. "PATENT CLAIMS" means the claims of patents that are owned or licensable by Tokutek, both currently or in the future; and that in the absence of this license would be infringed by THIS IMPLEMENTATION or by using or running THIS IMPLEMENTATION. "PATENT CHALLENGE" shall mean a challenge to the validity, patentability, enforceability and/or non-infringement of any of the PATENT CLAIMS or otherwise opposing any of the PATENT CLAIMS. Tokutek hereby grants to you, for the term and geographical scope of the PATENT CLAIMS, a non-exclusive, no-charge, royalty-free, irrevocable (except as stated in this section) patent license to make, have made, use, offer to sell, sell, import, transfer, and otherwise run, modify, and propagate the contents of THIS IMPLEMENTATION, where such license applies only to the PATENT CLAIMS. This grant does not include claims that would be infringed only as a consequence of further modifications of THIS IMPLEMENTATION. If you or your agent or licensee institute or order or agree to the institution of patent litigation against any entity (including a cross-claim or counterclaim in a lawsuit) alleging that THIS IMPLEMENTATION constitutes direct or contributory patent infringement, or inducement of patent infringement, then any rights granted to you under this License shall terminate as of the date such litigation is filed. If you or your agent or exclusive licensee institute or order or agree to the institution of a PATENT CHALLENGE, then Tokutek may terminate any rights granted to you under this License. */ #ident "Copyright (c) 2007-2013 Tokutek Inc. All rights reserved." #include "test.h" #include "omt.h" #include typedef OMTVALUE TESTVALUE; static void parse_args (int argc, const char *argv[]) { const char *argv0=argv[0]; while (argc>1) { int resultcode=0; if (strcmp(argv[1], "-v")==0) { verbose++; } else if (strcmp(argv[1], "-q")==0) { verbose = 0; } else if (strcmp(argv[1], "-h")==0) { do_usage: fprintf(stderr, "Usage:\n%s [-v|-h]\n", argv0); exit(resultcode); } else { resultcode=1; goto do_usage; } argc--; argv++; } } /* End ".h like" stuff. */ struct value { uint32_t number; }; #define V(x) ((struct value *)(x)) enum rand_type { TEST_RANDOM, TEST_SORTED, TEST_IDENTITY }; enum close_when_done { CLOSE_WHEN_DONE, KEEP_WHEN_DONE }; enum create_type { STEAL_ARRAY, BATCH_INSERT, INSERT_AT, INSERT_AT_ALMOST_RANDOM, }; /* Globals */ OMT global_omt; TESTVALUE* values = NULL; struct value* nums = NULL; uint32_t length; static void cleanup_globals (void) { assert(values); toku_free(values); values = NULL; assert(nums); toku_free(nums); nums = NULL; } const unsigned int random_seed = 0xFEADACBA; static void init_init_values (unsigned int seed, uint32_t num_elements) { srandom(seed); cleanup_globals(); MALLOC_N(num_elements, values); assert(values); MALLOC_N(num_elements, nums); assert(nums); length = num_elements; } static void init_identity_values (unsigned int seed, uint32_t num_elements) { uint32_t i; init_init_values(seed, num_elements); for (i = 0; i < length; i++) { nums[i].number = i; values[i] = (TESTVALUE)&nums[i]; } } static void init_distinct_sorted_values (unsigned int seed, uint32_t num_elements) { uint32_t i; init_init_values(seed, num_elements); uint32_t number = 0; for (i = 0; i < length; i++) { number += (uint32_t)(random() % 32) + 1; nums[i].number = number; values[i] = (TESTVALUE)&nums[i]; } } static void init_distinct_random_values (unsigned int seed, uint32_t num_elements) { init_distinct_sorted_values(seed, num_elements); uint32_t i; uint32_t choice; uint32_t choices; struct value temp; for (i = 0; i < length - 1; i++) { choices = length - i; choice = random() % choices; if (choice != i) { temp = nums[i]; nums[i] = nums[choice]; nums[choice] = temp; } } } static void init_globals (void) { MALLOC_N(1, values); assert(values); MALLOC_N(1, nums); assert(nums); length = 1; } static void test_close (enum close_when_done do_close) { if (do_close == KEEP_WHEN_DONE) return; assert(do_close == CLOSE_WHEN_DONE); toku_omt_destroy(&global_omt); assert(global_omt==NULL); } static void test_create (enum close_when_done do_close) { int r; global_omt = NULL; r = toku_omt_create(&global_omt); CKERR(r); assert(global_omt!=NULL); test_close(do_close); } static void test_create_size (enum close_when_done do_close) { test_create(KEEP_WHEN_DONE); assert(toku_omt_size(global_omt) == 0); test_close(do_close); } static void test_create_insert_at_almost_random (enum close_when_done do_close) { uint32_t i; int r; uint32_t size = 0; test_create(KEEP_WHEN_DONE); r = toku_omt_insert_at(global_omt, values[0], toku_omt_size(global_omt)+1); CKERR2(r, EINVAL); r = toku_omt_insert_at(global_omt, values[0], toku_omt_size(global_omt)+2); CKERR2(r, EINVAL); for (i = 0; i < length/2; i++) { assert(size==toku_omt_size(global_omt)); r = toku_omt_insert_at(global_omt, values[i], i); CKERR(r); assert(++size==toku_omt_size(global_omt)); r = toku_omt_insert_at(global_omt, values[length-1-i], i+1); CKERR(r); assert(++size==toku_omt_size(global_omt)); } r = toku_omt_insert_at(global_omt, values[0], toku_omt_size(global_omt)+1); CKERR2(r, EINVAL); r = toku_omt_insert_at(global_omt, values[0], toku_omt_size(global_omt)+2); CKERR2(r, EINVAL); assert(size==toku_omt_size(global_omt)); test_close(do_close); } static void test_create_insert_at_sequential (enum close_when_done do_close) { uint32_t i; int r; uint32_t size = 0; test_create(KEEP_WHEN_DONE); r = toku_omt_insert_at(global_omt, values[0], toku_omt_size(global_omt)+1); CKERR2(r, EINVAL); r = toku_omt_insert_at(global_omt, values[0], toku_omt_size(global_omt)+2); CKERR2(r, EINVAL); for (i = 0; i < length; i++) { assert(size==toku_omt_size(global_omt)); r = toku_omt_insert_at(global_omt, values[i], i); CKERR(r); assert(++size==toku_omt_size(global_omt)); } r = toku_omt_insert_at(global_omt, values[0], toku_omt_size(global_omt)+1); CKERR2(r, EINVAL); r = toku_omt_insert_at(global_omt, values[0], toku_omt_size(global_omt)+2); CKERR2(r, EINVAL); assert(size==toku_omt_size(global_omt)); test_close(do_close); } static void test_create_from_sorted_array (enum create_type create_choice, enum close_when_done do_close) { int r; global_omt = NULL; if (create_choice == BATCH_INSERT) { r = toku_omt_create_from_sorted_array(&global_omt, values, length); CKERR(r); } else if (create_choice == STEAL_ARRAY) { TESTVALUE* MALLOC_N(length, values_copy); memcpy(values_copy, values, length*sizeof(*values)); r = toku_omt_create_steal_sorted_array(&global_omt, &values_copy, length, length); CKERR(r); assert(values_copy==NULL); } else if (create_choice == INSERT_AT) { test_create_insert_at_sequential(KEEP_WHEN_DONE); } else if (create_choice == INSERT_AT_ALMOST_RANDOM) { test_create_insert_at_almost_random(KEEP_WHEN_DONE); } else assert(false); assert(global_omt!=NULL); test_close(do_close); } static void test_create_from_sorted_array_size (enum create_type create_choice, enum close_when_done do_close) { test_create_from_sorted_array(create_choice, KEEP_WHEN_DONE); assert(toku_omt_size(global_omt)==length); test_close(do_close); } static void test_fetch_verify (OMT omtree, TESTVALUE* val, uint32_t len ) { uint32_t i; int r; TESTVALUE v = (TESTVALUE)&i; TESTVALUE oldv = v; assert(len == toku_omt_size(omtree)); for (i = 0; i < len; i++) { assert(oldv!=val[i]); v = NULL; r = toku_omt_fetch(omtree, i, &v); CKERR(r); assert(v != NULL); assert(v != oldv); assert(v == val[i]); assert(V(v)->number == V(val[i])->number); v = oldv; } for (i = len; i < len*2; i++) { v = oldv; r = toku_omt_fetch(omtree, i, &v); CKERR2(r, EINVAL); assert(v == oldv); } } static void test_create_fetch_verify (enum create_type create_choice, enum close_when_done do_close) { test_create_from_sorted_array(create_choice, KEEP_WHEN_DONE); test_fetch_verify(global_omt, values, length); test_close(do_close); } static int iterate_helper_error_return = 1; static int iterate_helper (TESTVALUE v, uint32_t idx, void* extra) { if (extra == NULL) return iterate_helper_error_return; TESTVALUE* vals = (TESTVALUE *)extra; assert(v != NULL); assert(v == vals[idx]); assert(V(v)->number == V(vals[idx])->number); return 0; } static void test_iterate_verify (OMT omtree, TESTVALUE* vals, uint32_t len) { int r; iterate_helper_error_return = 0; r = toku_omt_iterate(omtree, iterate_helper, (void*)vals); CKERR(r); iterate_helper_error_return = 0xFEEDABBA; r = toku_omt_iterate(omtree, iterate_helper, NULL); if (!len) { CKERR2(r, 0); } else { CKERR2(r, iterate_helper_error_return); } } static void test_create_iterate_verify (enum create_type create_choice, enum close_when_done do_close) { test_create_from_sorted_array(create_choice, KEEP_WHEN_DONE); test_iterate_verify(global_omt, values, length); test_close(do_close); } static void permute_array (uint32_t* arr, uint32_t len) { // // create a permutation of 0...size-1 // uint32_t i = 0; for (i = 0; i < len; i++) { arr[i] = i; } for (i = 0; i < len - 1; i++) { uint32_t choices = len - i; uint32_t choice = random() % choices; if (choice != i) { uint32_t temp = arr[i]; arr[i] = arr[choice]; arr[choice] = temp; } } } static void test_create_set_at (enum create_type create_choice, enum close_when_done do_close) { uint32_t i = 0; struct value* old_nums = NULL; MALLOC_N(length, old_nums); assert(nums); uint32_t* perm = NULL; MALLOC_N(length, perm); assert(perm); TESTVALUE* old_values = NULL; MALLOC_N(length, old_values); assert(old_values); permute_array(perm, length); // // These are going to be the new values // for (i = 0; i < length; i++) { old_nums[i] = nums[i]; old_values[i] = &old_nums[i]; values[i] = &old_nums[i]; } test_create_from_sorted_array(create_choice, KEEP_WHEN_DONE); int r; r = toku_omt_set_at (global_omt, values[0], length); CKERR2(r,EINVAL); r = toku_omt_set_at (global_omt, values[0], length+1); CKERR2(r,EINVAL); for (i = 0; i < length; i++) { uint32_t choice = perm[i]; values[choice] = &nums[choice]; nums[choice].number = (uint32_t)random(); r = toku_omt_set_at (global_omt, values[choice], choice); CKERR(r); test_iterate_verify(global_omt, values, length); test_fetch_verify(global_omt, values, length); } r = toku_omt_set_at (global_omt, values[0], length); CKERR2(r,EINVAL); r = toku_omt_set_at (global_omt, values[0], length+1); CKERR2(r,EINVAL); toku_free(perm); toku_free(old_values); toku_free(old_nums); test_close(do_close); } static int insert_helper (TESTVALUE value, void* extra_insert) { TESTVALUE to_insert = (OMTVALUE)extra_insert; assert(to_insert); if (V(value)->number < V(to_insert)->number) return -1; if (V(value)->number > V(to_insert)->number) return +1; return 0; } static void test_create_insert (enum close_when_done do_close) { uint32_t i = 0; uint32_t* perm = NULL; MALLOC_N(length, perm); assert(perm); permute_array(perm, length); test_create(KEEP_WHEN_DONE); int r; uint32_t size = length; length = 0; while (length < size) { uint32_t choice = perm[length]; TESTVALUE to_insert = &nums[choice]; uint32_t idx = UINT32_MAX; assert(length==toku_omt_size(global_omt)); r = toku_omt_insert(global_omt, to_insert, insert_helper, to_insert, &idx); CKERR(r); assert(idx <= length); if (idx > 0) { assert(V(to_insert)->number > V(values[idx-1])->number); } if (idx < length) { assert(V(to_insert)->number < V(values[idx])->number); } length++; assert(length==toku_omt_size(global_omt)); /* Make room */ for (i = length-1; i > idx; i--) { values[i] = values[i-1]; } values[idx] = to_insert; test_fetch_verify(global_omt, values, length); test_iterate_verify(global_omt, values, length); idx = UINT32_MAX; r = toku_omt_insert(global_omt, to_insert, insert_helper, to_insert, &idx); CKERR2(r, DB_KEYEXIST); assert(idx < length); assert(V(values[idx])->number == V(to_insert)->number); assert(length==toku_omt_size(global_omt)); test_iterate_verify(global_omt, values, length); test_fetch_verify(global_omt, values, length); } toku_free(perm); test_close(do_close); } static void test_create_delete_at (enum create_type create_choice, enum close_when_done do_close) { uint32_t i = 0; int r = ENOSYS; test_create_from_sorted_array(create_choice, KEEP_WHEN_DONE); assert(length == toku_omt_size(global_omt)); r = toku_omt_delete_at(global_omt, length); CKERR2(r,EINVAL); assert(length == toku_omt_size(global_omt)); r = toku_omt_delete_at(global_omt, length+1); CKERR2(r,EINVAL); while (length > 0) { assert(length == toku_omt_size(global_omt)); uint32_t index_to_delete = random()%length; r = toku_omt_delete_at(global_omt, index_to_delete); CKERR(r); for (i = index_to_delete+1; i < length; i++) { values[i-1] = values[i]; } length--; test_fetch_verify(global_omt, values, length); test_iterate_verify(global_omt, values, length); } assert(length == 0); assert(length == toku_omt_size(global_omt)); r = toku_omt_delete_at(global_omt, length); CKERR2(r, EINVAL); assert(length == toku_omt_size(global_omt)); r = toku_omt_delete_at(global_omt, length+1); CKERR2(r, EINVAL); test_close(do_close); } static void test_split_merge (enum create_type create_choice, enum close_when_done do_close) { int r = ENOSYS; uint32_t i = 0; OMT left_split = NULL; OMT right_split = NULL; test_create_from_sorted_array(create_choice, KEEP_WHEN_DONE); for (i = 0; i <= length; i++) { r = toku_omt_split_at(global_omt, &right_split, length+1); CKERR2(r,EINVAL); r = toku_omt_split_at(global_omt, &right_split, length+2); CKERR2(r,EINVAL); // // test successful split // r = toku_omt_split_at(global_omt, &right_split, i); CKERR(r); left_split = global_omt; global_omt = NULL; assert(toku_omt_size(left_split) == i); assert(toku_omt_size(right_split) == length - i); test_fetch_verify(left_split, values, i); test_iterate_verify(left_split, values, i); test_fetch_verify(right_split, &values[i], length - i); test_iterate_verify(right_split, &values[i], length - i); // // verify that new global_omt's cannot do bad splits // r = toku_omt_split_at(left_split, &global_omt, i+1); CKERR2(r,EINVAL); assert(toku_omt_size(left_split) == i); assert(toku_omt_size(right_split) == length - i); r = toku_omt_split_at(left_split, &global_omt, i+2); CKERR2(r,EINVAL); assert(toku_omt_size(left_split) == i); assert(toku_omt_size(right_split) == length - i); r = toku_omt_split_at(right_split, &global_omt, length - i + 1); CKERR2(r,EINVAL); assert(toku_omt_size(left_split) == i); assert(toku_omt_size(right_split) == length - i); r = toku_omt_split_at(right_split, &global_omt, length - i + 1); CKERR2(r,EINVAL); assert(toku_omt_size(left_split) == i); assert(toku_omt_size(right_split) == length - i); // // test merge // r = toku_omt_merge(left_split,right_split,&global_omt); CKERR(r); left_split = NULL; right_split = NULL; assert(toku_omt_size(global_omt) == length); test_fetch_verify(global_omt, values, length); test_iterate_verify(global_omt, values, length); } test_close(do_close); } static void init_values (enum rand_type rand_choice) { const uint32_t test_size = 100; if (rand_choice == TEST_RANDOM) { init_distinct_random_values(random_seed, test_size); } else if (rand_choice == TEST_SORTED) { init_distinct_sorted_values(random_seed, test_size); } else if (rand_choice == TEST_IDENTITY) { init_identity_values( random_seed, test_size); } else assert(false); } static void test_create_array (enum create_type create_choice, enum rand_type rand_choice) { /* ********************************************************************** */ init_values(rand_choice); test_create_from_sorted_array( create_choice, CLOSE_WHEN_DONE); test_create_from_sorted_array_size(create_choice, CLOSE_WHEN_DONE); /* ********************************************************************** */ init_values(rand_choice); test_create_fetch_verify( create_choice, CLOSE_WHEN_DONE); /* ********************************************************************** */ init_values(rand_choice); test_create_iterate_verify( create_choice, CLOSE_WHEN_DONE); /* ********************************************************************** */ init_values(rand_choice); test_create_set_at( create_choice, CLOSE_WHEN_DONE); /* ********************************************************************** */ init_values(rand_choice); test_create_delete_at( create_choice, CLOSE_WHEN_DONE); /* ********************************************************************** */ init_values(rand_choice); test_create_insert( CLOSE_WHEN_DONE); /* ********************************************************************** */ init_values(rand_choice); test_split_merge( create_choice, CLOSE_WHEN_DONE); } typedef struct { uint32_t first_zero; uint32_t first_pos; } h_extra; static int test_heaviside (OMTVALUE v_omt, void* x) { TESTVALUE v = (OMTVALUE) v_omt; h_extra* extra = (h_extra*)x; assert(v && x); assert(extra->first_zero <= extra->first_pos); uint32_t value = V(v)->number; if (value < extra->first_zero) return -1; if (value < extra->first_pos) return 0; return 1; } static void heavy_extra (h_extra* extra, uint32_t first_zero, uint32_t first_pos) { extra->first_zero = first_zero; extra->first_pos = first_pos; } static void test_find_dir (int dir, void* extra, int (*h)(OMTVALUE, void*), int r_expect, bool idx_will_change, uint32_t idx_expect, uint32_t number_expect, bool UU(cursor_valid)) { uint32_t idx = UINT32_MAX; uint32_t old_idx = idx; TESTVALUE omt_val; int r; omt_val = NULL; /* Verify we can pass NULL value. */ omt_val = NULL; idx = old_idx; if (dir == 0) { r = toku_omt_find_zero(global_omt, h, extra, NULL, &idx); } else { r = toku_omt_find( global_omt, h, extra, dir, NULL, &idx); } CKERR2(r, r_expect); if (idx_will_change) { assert(idx == idx_expect); } else { assert(idx == old_idx); } assert(omt_val == NULL); /* Verify we can pass NULL idx. */ omt_val = NULL; idx = old_idx; if (dir == 0) { r = toku_omt_find_zero(global_omt, h, extra, &omt_val, 0); } else { r = toku_omt_find( global_omt, h, extra, dir, &omt_val, 0); } CKERR2(r, r_expect); assert(idx == old_idx); if (r == DB_NOTFOUND) { assert(omt_val == NULL); } else { assert(V(omt_val)->number == number_expect); } /* Verify we can pass NULL both. */ omt_val = NULL; idx = old_idx; if (dir == 0) { r = toku_omt_find_zero(global_omt, h, extra, NULL, 0); } else { r = toku_omt_find( global_omt, h, extra, dir, NULL, 0); } CKERR2(r, r_expect); assert(idx == old_idx); assert(omt_val == NULL); } static void test_find (enum create_type create_choice, enum close_when_done do_close) { h_extra extra; init_identity_values(random_seed, 100); test_create_from_sorted_array(create_choice, KEEP_WHEN_DONE); /* -...- A */ heavy_extra(&extra, length, length); test_find_dir(-1, &extra, test_heaviside, 0, true, length-1, length-1, true); test_find_dir(+1, &extra, test_heaviside, DB_NOTFOUND, false, 0, 0, false); test_find_dir(0, &extra, test_heaviside, DB_NOTFOUND, true, length, length, false); /* +...+ B */ heavy_extra(&extra, 0, 0); test_find_dir(-1, &extra, test_heaviside, DB_NOTFOUND, false, 0, 0, false); test_find_dir(+1, &extra, test_heaviside, 0, true, 0, 0, true); test_find_dir(0, &extra, test_heaviside, DB_NOTFOUND, true, 0, 0, false); /* 0...0 C */ heavy_extra(&extra, 0, length); test_find_dir(-1, &extra, test_heaviside, DB_NOTFOUND, false, 0, 0, false); test_find_dir(+1, &extra, test_heaviside, DB_NOTFOUND, false, 0, 0, false); test_find_dir(0, &extra, test_heaviside, 0, true, 0, 0, true); /* -...-0...0 AC */ heavy_extra(&extra, length/2, length); test_find_dir(-1, &extra, test_heaviside, 0, true, length/2-1, length/2-1, true); test_find_dir(+1, &extra, test_heaviside, DB_NOTFOUND, false, 0, 0, false); test_find_dir(0, &extra, test_heaviside, 0, true, length/2, length/2, true); /* 0...0+...+ C B */ heavy_extra(&extra, 0, length/2); test_find_dir(-1, &extra, test_heaviside, DB_NOTFOUND, false, 0, 0, false); test_find_dir(+1, &extra, test_heaviside, 0, true, length/2, length/2, true); test_find_dir(0, &extra, test_heaviside, 0, true, 0, 0, true); /* -...-+...+ AB */ heavy_extra(&extra, length/2, length/2); test_find_dir(-1, &extra, test_heaviside, 0, true, length/2-1, length/2-1, true); test_find_dir(+1, &extra, test_heaviside, 0, true, length/2, length/2, true); test_find_dir(0, &extra, test_heaviside, DB_NOTFOUND, true, length/2, length/2, false); /* -...-0...0+...+ AC B */ heavy_extra(&extra, length/3, 2*length/3); test_find_dir(-1, &extra, test_heaviside, 0, true, length/3-1, length/3-1, true); test_find_dir(+1, &extra, test_heaviside, 0, true, 2*length/3, 2*length/3, true); test_find_dir(0, &extra, test_heaviside, 0, true, length/3, length/3, true); /* Cleanup */ test_close(do_close); } static void runtests_create_choice (enum create_type create_choice) { test_create_array(create_choice, TEST_SORTED); test_create_array(create_choice, TEST_RANDOM); test_create_array(create_choice, TEST_IDENTITY); test_find( create_choice, CLOSE_WHEN_DONE); } static void test_clone(uint32_t nelts) // Test that each clone operation gives the right data back. If nelts is // zero, also tests that you still get a valid OMT back and that the way // to deallocate it still works. { OMT src = NULL, dest = NULL; int r; r = toku_omt_create(&src); assert_zero(r); for (long i = 0; i < nelts; ++i) { r = toku_omt_insert_at(src, (OMTVALUE) i, i); assert_zero(r); } r = toku_omt_clone_noptr(&dest, src); assert_zero(r); assert(dest != NULL); assert(toku_omt_size(dest) == nelts); for (long i = 0; i < nelts; ++i) { OMTVALUE v; long l; r = toku_omt_fetch(dest, i, &v); assert_zero(r); l = (long) v; assert(l == i); } toku_omt_destroy(&dest); toku_omt_destroy(&src); } int test_main(int argc, const char *argv[]) { parse_args(argc, argv); init_globals(); test_create( CLOSE_WHEN_DONE); test_create_size( CLOSE_WHEN_DONE); runtests_create_choice(BATCH_INSERT); runtests_create_choice(STEAL_ARRAY); runtests_create_choice(INSERT_AT); runtests_create_choice(INSERT_AT_ALMOST_RANDOM); test_clone(0); test_clone(1); test_clone(1000); test_clone(10000); cleanup_globals(); return 0; }