00001
00047 #include <stdlib.h>
00048 #include <stdbool.h>
00049 #include <unistd.h>
00050 #include <errno.h>
00051 #include <time.h>
00052 #include <stdio.h>
00053 #include "../mkavl.h"
00054
00058 #define LOG_FAIL(fmt, args...) \
00059 do { \
00060 printf("FAILURE(%s:%u): " fmt "\n", __FUNCTION__, __LINE__, ##args); \
00061 } while (0)
00062
00066 #ifndef NELEMS
00067 #define NELEMS(x) (sizeof(x) / sizeof(x[0]))
00068 #endif
00069
00074 #ifndef CT_ASSERT
00075 #define CT_ASSERT(e) extern char (*CT_ASSERT(void)) [sizeof(char[1 - 2*!(e)])]
00076 #endif
00077
00081 #define MKAVL_TEST_RUNAWAY_SANITY 100000
00082
00084 static const uint32_t default_node_cnt = 15;
00086 static const uint32_t default_run_cnt = 15;
00088 static const uint8_t default_verbosity = 0;
00090 static const uint32_t default_range_start = 0;
00092 static const uint32_t default_range_end = 100;
00093
00097 typedef struct test_mkavl_opts_st_ {
00099 uint32_t node_cnt;
00101 uint32_t run_cnt;
00103 uint32_t seed;
00105 uint8_t verbosity;
00107 uint32_t range_start;
00109 uint32_t range_end;
00110 } test_mkavl_opts_st;
00111
00118 static void
00119 print_usage (bool do_exit, int32_t exit_val)
00120 {
00121 printf("\nTest the mkavl structure\n\n");
00122 printf("Usage:\n");
00123 printf("-s <seed>\n"
00124 " The starting seed for the RNG (default=seeded by time()).\n");
00125 printf("-n <nodes>\n"
00126 " The number of nodes to place in the trees (default=%u).\n",
00127 default_node_cnt);
00128 printf("-b <range beginning>\n"
00129 " The smallest (inclusive ) possible data value in the "
00130 "range of values (default=%u).\n", default_range_start);
00131 printf("-e <range ending>\n"
00132 " The largest (exclusive) possible data value in the "
00133 "range of values (default=%u).\n", default_range_end);
00134 printf("-r <runs>\n"
00135 " The number of runs to do (default=%u).\n",
00136 default_run_cnt);
00137 printf("-v <verbosity level>\n"
00138 " A higher number gives more output (default=%u).\n",
00139 default_verbosity);
00140 printf("-h\n"
00141 " Display this help message.\n");
00142 printf("\n");
00143
00144 if (do_exit) {
00145 exit(exit_val);
00146 }
00147 }
00148
00154 static void
00155 print_opts (test_mkavl_opts_st *opts)
00156 {
00157 if (NULL == opts) {
00158 return;
00159 }
00160
00161 printf("test_mkavl_opts: seed=%u, node_cnt=%u, run_cnt=%u,\n"
00162 " range=[%u,%u) verbosity=%u\n",
00163 opts->seed, opts->node_cnt, opts->run_cnt, opts->range_start,
00164 opts->range_end, opts->verbosity);
00165 }
00166
00174 static void
00175 parse_command_line (int argc, char **argv, test_mkavl_opts_st *opts)
00176 {
00177 int c;
00178 char *end_ptr;
00179 uint32_t val;
00180
00181 if (NULL == opts) {
00182 return;
00183 }
00184
00185 opts->node_cnt = default_node_cnt;
00186 opts->run_cnt = default_run_cnt;
00187 opts->verbosity = default_verbosity;
00188 opts->range_start = default_range_start;
00189 opts->range_end = default_range_end;
00190 opts->seed = (uint32_t) time(NULL);
00191
00192 while ((c = getopt(argc, argv, "n:r:v:s:b:e:h")) != -1) {
00193 switch (c) {
00194 case 'n':
00195 val = strtol(optarg, &end_ptr, 10);
00196 if ((end_ptr != optarg) && (0 == errno)) {
00197 opts->node_cnt = val;
00198 }
00199 break;
00200 case 'r':
00201 val = strtol(optarg, &end_ptr, 10);
00202 if ((end_ptr != optarg) && (0 == errno)) {
00203 opts->run_cnt = val;
00204 }
00205 break;
00206 case 'v':
00207 val = strtol(optarg, &end_ptr, 10);
00208 if ((end_ptr != optarg) && (0 == errno)) {
00209 opts->verbosity = val;
00210 }
00211 break;
00212 case 's':
00213 val = strtol(optarg, &end_ptr, 10);
00214 if ((end_ptr != optarg) && (0 == errno)) {
00215 opts->seed = val;
00216 }
00217 break;
00218 case 'b':
00219 val = strtol(optarg, &end_ptr, 10);
00220 if ((end_ptr != optarg) && (0 == errno)) {
00221 opts->range_start = val;
00222 }
00223 break;
00224 case 'e':
00225 val = strtol(optarg, &end_ptr, 10);
00226 if ((end_ptr != optarg) && (0 == errno)) {
00227 opts->range_end = val;
00228 }
00229 break;
00230 case 'h':
00231 case '?':
00232 default:
00233 print_usage(true, EXIT_SUCCESS);
00234 break;
00235 }
00236 }
00237
00238 if (optind < argc) {
00239 print_usage(true, EXIT_SUCCESS);
00240 }
00241
00242 if (opts->range_start >= opts->range_end) {
00243 printf("Error: range start(%u) must be strictly less than range "
00244 "end(%u)\n", opts->range_start, opts->range_end);
00245 print_usage(true, EXIT_SUCCESS);
00246 }
00247
00248 if (0 == opts->node_cnt) {
00249 printf("Error: node count(%u) must be non-zero\n",
00250 opts->node_cnt);
00251 print_usage(true, EXIT_SUCCESS);
00252 }
00253
00254 if (opts->verbosity >= 3) {
00255 print_opts(opts);
00256 }
00257 }
00258
00266 static void
00267 permute_array (const uint32_t *src_array, uint32_t *dst_array,
00268 size_t num_elem)
00269 {
00270 uint32_t i, j, swp_val;
00271
00272 if ((NULL == src_array) || (NULL == dst_array) || (0 == num_elem)) {
00273 return;
00274 }
00275
00276 memcpy(dst_array, src_array, (num_elem * sizeof(*dst_array)));
00277
00278 for (i = (num_elem - 1); i > 0; --i) {
00279 j = (rand() % (i + 1));
00280 swp_val = dst_array[j];
00281 dst_array[j] = dst_array[i];
00282 dst_array[i] = swp_val;
00283 }
00284 }
00285
00293 static int
00294 uint32_t_cmp (const void *a, const void *b)
00295 {
00296 const uint32_t i1 = *((const uint32_t *) a);
00297 const uint32_t i2 = *((const uint32_t *) b);
00298
00299 if (i1 < i2) {
00300 return (-1);
00301 } else if (i1 > i2) {
00302 return (1);
00303 }
00304
00305 return (0);
00306 }
00307
00316 static uint32_t
00317 get_unique_count (uint32_t *array, size_t num_elem)
00318 {
00319 uint32_t count = 0;
00320 uint32_t cur_val;
00321 uint32_t i;
00322
00323 if ((NULL == array) || (0 == num_elem)) {
00324 return (0);
00325 }
00326
00327 for (i = 0; i < num_elem; ++i) {
00328 if ((0 == i) || (array[i] != cur_val)) {
00329 cur_val = array[i];
00330 ++count;
00331 }
00332 }
00333
00334 return (count);
00335 }
00336
00340 typedef struct mkavl_test_input_st_ {
00342 uint32_t *insert_seq;
00344 uint32_t *delete_seq;
00346 uint32_t *sorted_seq;
00348 uint32_t uniq_cnt;
00350 uint32_t dup_cnt;
00352 const test_mkavl_opts_st *opts;
00354 mkavl_tree_handle tree_h;
00356 mkavl_tree_handle tree_copy_h;
00357 } mkavl_test_input_st;
00358
00359
00360
00361
00362
00363 static bool
00364 run_mkavl_test(mkavl_test_input_st *input);
00365
00369 int
00370 main (int argc, char *argv[])
00371 {
00372 test_mkavl_opts_st opts;
00373 bool was_success;
00374 uint32_t fail_count = 0;
00375 uint32_t i;
00376 uint32_t cur_run, cur_seed, uniq_cnt;
00377 mkavl_test_input_st test_input = {0};
00378
00379 parse_command_line(argc, argv, &opts);
00380
00381 printf("\n");
00382 cur_seed = opts.seed;
00383 for (cur_run = 0; cur_run < opts.run_cnt; ++cur_run) {
00384 uint32_t insert_seq[opts.node_cnt];
00385 uint32_t delete_seq[opts.node_cnt];
00386 uint32_t sorted_seq[opts.node_cnt];
00387
00388 printf("Doing run %u with seed %u\n", (cur_run + 1), cur_seed);
00389 srand(cur_seed);
00390
00391 for (i = 0; i < opts.node_cnt; ++i) {
00392 insert_seq[i] = ((rand() % opts.range_end) + opts.range_start);
00393 }
00394 permute_array(insert_seq, delete_seq, opts.node_cnt);
00395 memcpy(sorted_seq, insert_seq, sizeof(sorted_seq));
00396 qsort(sorted_seq, opts.node_cnt, sizeof(sorted_seq[0]), uint32_t_cmp);
00397 uniq_cnt = get_unique_count(sorted_seq, opts.node_cnt);
00398
00399 if (opts.verbosity >= 1) {
00400 printf("Unique count: %u\n", uniq_cnt);
00401 printf("Insertion sequence:\n ");
00402 for (i = 0; i < opts.node_cnt; ++i) {
00403 printf(" %u", insert_seq[i]);
00404 }
00405 printf("\n");
00406
00407 printf("Deletion sequence:\n ");
00408 for (i = 0; i < opts.node_cnt; ++i) {
00409 printf(" %u", delete_seq[i]);
00410 }
00411 printf("\n");
00412
00413 printf("Sorted sequence:\n ");
00414 for (i = 0; i < opts.node_cnt; ++i) {
00415 printf(" %u", sorted_seq[i]);
00416 }
00417 printf("\n");
00418 }
00419
00420 test_input.insert_seq = insert_seq;
00421 test_input.delete_seq = delete_seq;
00422 test_input.sorted_seq = sorted_seq;
00423 test_input.uniq_cnt = uniq_cnt;
00424 test_input.dup_cnt = (opts.node_cnt - uniq_cnt);
00425 test_input.opts = &opts;
00426 test_input.tree_h = NULL;
00427 test_input.tree_copy_h = NULL;
00428
00429 was_success = run_mkavl_test(&test_input);
00430 if (!was_success) {
00431 printf("FAILURE: the test has failed for seed %u!!!\n", cur_seed);
00432 ++fail_count;
00433 }
00434
00435 ++cur_seed;
00436 }
00437
00438 if (0 != fail_count) {
00439 printf("\n%u/%u TESTS FAILED\n", fail_count, opts.run_cnt);
00440 } else {
00441 printf("\nALL TESTS PASSED\n");
00442 }
00443 printf("\n");
00444
00445 return (0);
00446 }
00447
00448
00449
00451 #define MKAVL_TEST_MAGIC 0x1234ABCD
00452
00456 typedef struct mkavl_test_ctx_st_ {
00458 uint32_t magic;
00460 uint32_t copy_cnt;
00462 uint32_t item_fn_cnt;
00464 uint32_t copy_malloc_cnt;
00466 uint32_t copy_free_cnt;
00467 } mkavl_test_ctx_st;
00468
00476 static void *
00477 mkavl_test_copy_malloc (size_t size, void *context)
00478 {
00479 mkavl_test_ctx_st *ctx = (mkavl_test_ctx_st *) context;
00480
00481 if ((NULL == ctx) || (MKAVL_TEST_MAGIC != ctx->magic)) {
00482 abort();
00483 }
00484
00485 ++(ctx->copy_malloc_cnt);
00486
00487 return (malloc(size));
00488 }
00489
00490
00497 static void
00498 mkavl_test_copy_free (void *ptr, void *context)
00499 {
00500 mkavl_test_ctx_st *ctx = (mkavl_test_ctx_st *) context;
00501
00502 if ((NULL == ctx) || (MKAVL_TEST_MAGIC != ctx->magic)) {
00503 abort();
00504 }
00505
00506 ++(ctx->copy_free_cnt);
00507
00508 free(ptr);
00509 }
00510
00514 static mkavl_allocator_st copy_allocator = {
00515 mkavl_test_copy_malloc,
00516 mkavl_test_copy_free
00517 };
00518
00527 static int32_t
00528 mkavl_cmp_fn1 (const void *item1, const void *item2, void *context)
00529 {
00530 const uint32_t *i1 = item1;
00531 const uint32_t *i2 = item2;
00532 mkavl_test_ctx_st *ctx;
00533
00534 ctx = (mkavl_test_ctx_st *) context;
00535
00536 if ((NULL == ctx) || (MKAVL_TEST_MAGIC != ctx->magic)) {
00537 abort();
00538 }
00539
00540 if (*i1 < *i2) {
00541 return (-1);
00542 } else if (*i1 > *i2) {
00543 return (1);
00544 }
00545
00546 return (0);
00547 }
00548
00557 static int32_t
00558 mkavl_cmp_fn2 (const void *item1, const void *item2, void *context)
00559 {
00560 const uint32_t *i1 = item1;
00561 const uint32_t *i2 = item2;
00562 mkavl_test_ctx_st *ctx;
00563
00564 ctx = (mkavl_test_ctx_st *) context;
00565
00566 if ((NULL == ctx) || (MKAVL_TEST_MAGIC != ctx->magic)) {
00567 abort();
00568 }
00569
00570 if (*i1 > *i2) {
00571 return (-1);
00572 } else if (*i1 < *i2) {
00573 return (1);
00574 }
00575
00576 return (0);
00577 }
00578
00582 typedef enum mkavl_test_key_e_ {
00584 MKAVL_TEST_KEY_E_ASC,
00586 MKAVL_TEST_KEY_E_DESC,
00588 MKAVL_TEST_KEY_E_MAX,
00589 } mkavl_test_key_e;
00590
00592 static const mkavl_test_key_e mkavl_key_opposite[] = {
00593 MKAVL_TEST_KEY_E_DESC,
00594 MKAVL_TEST_KEY_E_ASC,
00595 };
00596
00598 static mkavl_compare_fn cmp_fn_array[] = { mkavl_cmp_fn1 , mkavl_cmp_fn2 };
00599
00601 CT_ASSERT(NELEMS(mkavl_key_opposite) == MKAVL_TEST_KEY_E_MAX);
00602 CT_ASSERT(NELEMS(cmp_fn_array) == MKAVL_TEST_KEY_E_MAX);
00608 static const mkavl_find_type_e
00609 mkavl_key_find_type[MKAVL_TEST_KEY_E_MAX][(MKAVL_FIND_TYPE_E_MAX + 1)] = {
00610 { MKAVL_FIND_TYPE_E_INVALID, MKAVL_FIND_TYPE_E_EQUAL,
00611 MKAVL_FIND_TYPE_E_GT, MKAVL_FIND_TYPE_E_LT,
00612 MKAVL_FIND_TYPE_E_GE, MKAVL_FIND_TYPE_E_LE,
00613 MKAVL_FIND_TYPE_E_MAX },
00614 { MKAVL_FIND_TYPE_E_INVALID, MKAVL_FIND_TYPE_E_EQUAL,
00615 MKAVL_FIND_TYPE_E_LT, MKAVL_FIND_TYPE_E_GT,
00616 MKAVL_FIND_TYPE_E_LE, MKAVL_FIND_TYPE_E_GE,
00617 MKAVL_FIND_TYPE_E_MAX }
00618 };
00619
00624 static bool
00625 mkavl_test_new_error (void)
00626 {
00627 mkavl_rc_e rc;
00628 mkavl_tree_handle tree_h = NULL;
00629
00630 if (0 != mkavl_count(NULL)) {
00631 LOG_FAIL("NULL mkavl_count failed, mkavl_count(%u)",
00632 mkavl_count(NULL));
00633 return (false);
00634 }
00635
00636 rc = mkavl_new(NULL, cmp_fn_array, NELEMS(cmp_fn_array), NULL, NULL);
00637 if (mkavl_rc_e_is_ok(rc)) {
00638 LOG_FAIL("NULL tree failed, rc(%s)", mkavl_rc_e_get_string(rc));
00639 return (false);
00640 }
00641
00642 rc = mkavl_new(&tree_h, NULL, NELEMS(cmp_fn_array), NULL, NULL);
00643 if (mkavl_rc_e_is_ok(rc)) {
00644 LOG_FAIL("NULL function failed, rc(%s)", mkavl_rc_e_get_string(rc));
00645 return (false);
00646 }
00647
00648 rc = mkavl_new(&tree_h, cmp_fn_array, 0, NULL, NULL);
00649 if (mkavl_rc_e_is_ok(rc)) {
00650 LOG_FAIL("zero size function failed, rc(%s)",
00651 mkavl_rc_e_get_string(rc));
00652 return (false);
00653 }
00654
00655 return (true);
00656 }
00657
00665 static bool
00666 mkavl_test_new (mkavl_test_input_st *input, mkavl_allocator_st *allocator)
00667 {
00668 mkavl_rc_e rc;
00669 mkavl_test_ctx_st *ctx;
00670
00671 ctx = calloc(1, sizeof(*ctx));
00672 if (NULL == ctx) {
00673 LOG_FAIL("calloc failed");
00674 return (false);
00675 }
00676 ctx->magic = MKAVL_TEST_MAGIC;
00677
00678 rc = mkavl_new(&(input->tree_h), cmp_fn_array, NELEMS(cmp_fn_array),
00679 ctx, allocator);
00680 if (mkavl_rc_e_is_notok(rc)) {
00681 LOG_FAIL("new failed, rc(%s)", mkavl_rc_e_get_string(rc));
00682 return (false);
00683 }
00684
00685 return (true);
00686 }
00687
00694 static mkavl_rc_e
00695 mkavl_test_delete_context (void *context)
00696 {
00697 mkavl_test_ctx_st *ctx = (mkavl_test_ctx_st *) context;
00698
00699 if ((NULL == ctx) || (MKAVL_TEST_MAGIC != ctx->magic)) {
00700 return (MKAVL_RC_E_EINVAL);
00701 }
00702
00703 ctx->magic = 0;
00704 free(context);
00705
00706 return (MKAVL_RC_E_SUCCESS);
00707 }
00708
00719 static bool
00720 mkavl_test_delete (mkavl_test_input_st *input, mkavl_item_fn item_fn,
00721 mkavl_delete_context_fn delete_context_fn,
00722 mkavl_delete_context_fn delete_copy_context_fn)
00723 {
00724 mkavl_rc_e rc;
00725
00726 if (NULL != input->tree_h) {
00727 rc = mkavl_delete(&(input->tree_h), item_fn, delete_context_fn);
00728 if (mkavl_rc_e_is_notok(rc)) {
00729 LOG_FAIL("delete failed, rc(%s)", mkavl_rc_e_get_string(rc));
00730 return (false);
00731 }
00732 }
00733
00734 if (NULL != input->tree_copy_h) {
00735 rc = mkavl_delete(&(input->tree_copy_h), item_fn,
00736 delete_copy_context_fn);
00737 if (mkavl_rc_e_is_notok(rc)) {
00738 LOG_FAIL("delete failed, rc(%s)", mkavl_rc_e_get_string(rc));
00739 return (false);
00740 }
00741 }
00742
00743 return (true);
00744 }
00745
00752 static bool
00753 mkavl_test_add_error (mkavl_test_input_st *input)
00754 {
00755 uint32_t *existing_item;
00756 mkavl_rc_e rc;
00757
00758 rc = mkavl_add(input->tree_h, &(input->insert_seq[0]), (void **) NULL);
00759 if (mkavl_rc_e_is_ok(rc)) {
00760 LOG_FAIL("NULL existing item failed, rc(%s)",
00761 mkavl_rc_e_get_string(rc));
00762 return (false);
00763 }
00764
00765 rc = mkavl_add(input->tree_h, NULL, (void **) &existing_item);
00766 if (mkavl_rc_e_is_ok(rc)) {
00767 LOG_FAIL("NULL item failed, rc(%s)",
00768 mkavl_rc_e_get_string(rc));
00769 return (false);
00770 }
00771
00772 rc = mkavl_add(NULL, &(input->insert_seq[0]), (void **) &existing_item);
00773 if (mkavl_rc_e_is_ok(rc)) {
00774 LOG_FAIL("NULL tree failed, rc(%s)", mkavl_rc_e_get_string(rc));
00775 return (false);
00776 }
00777
00778 return (true);
00779 }
00780
00787 static bool
00788 mkavl_test_add (mkavl_test_input_st *input)
00789 {
00790 uint32_t i;
00791 uint32_t non_null_cnt = 0;
00792 uint32_t *existing_item;
00793 mkavl_rc_e rc;
00794
00795 for (i = 0; i < input->opts->node_cnt; ++i) {
00796 rc = mkavl_add(input->tree_h, &(input->insert_seq[i]),
00797 (void **) &existing_item);
00798 if (mkavl_rc_e_is_notok(rc)) {
00799 LOG_FAIL("add failed, rc(%s)", mkavl_rc_e_get_string(rc));
00800 return (false);
00801 }
00802
00803 if (NULL != existing_item) {
00804 ++non_null_cnt;
00805 }
00806 }
00807
00808 if (non_null_cnt != input->dup_cnt) {
00809 LOG_FAIL("duplicate check failed, non_null_cnt(%u) dup_cnt(%u)",
00810 non_null_cnt, input->dup_cnt);
00811 return (false);
00812 }
00813
00814 if (mkavl_count(input->tree_h) != input->uniq_cnt) {
00815 LOG_FAIL("unique check failed, mkavl_count(%u) uniq_cnt(%u)",
00816 mkavl_count(input->tree_h), input->uniq_cnt);
00817 return (false);
00818 }
00819
00820 return (true);
00821 }
00822
00831 static uint32_t *
00832 mkavl_test_find_val (mkavl_test_input_st *input,
00833 uint32_t val, mkavl_find_type_e type)
00834 {
00835 int32_t lo, hi, mid;
00836 uint32_t runaway_counter = 0;
00837 bool is_equal_type, is_greater_type, is_less_type;
00838 size_t num_elems = input->opts->node_cnt;
00839
00840 if (0 == num_elems) {
00841 return (NULL);
00842 }
00843
00844 if (!mkavl_find_type_e_is_valid(type)) {
00845 return (NULL);
00846 }
00847
00848 is_equal_type = ((MKAVL_FIND_TYPE_E_EQUAL == type) ||
00849 (MKAVL_FIND_TYPE_E_GE == type) ||
00850 (MKAVL_FIND_TYPE_E_LE == type));
00851
00852 is_greater_type = ((MKAVL_FIND_TYPE_E_GT == type) ||
00853 (MKAVL_FIND_TYPE_E_GE == type));
00854
00855 is_less_type = ((MKAVL_FIND_TYPE_E_LT == type) ||
00856 (MKAVL_FIND_TYPE_E_LE == type));
00857
00858 lo = 0;
00859 hi = (num_elems - 1);
00860 while (lo <= hi) {
00861 mid = (lo + ((hi - lo) / 2));
00862 if (input->opts->verbosity >= 6) {
00863 printf("lo(%d) hi(%d) mid(%d) val(%u) arrayval(%u) type(%s)\n",
00864 lo, hi, mid, val, input->sorted_seq[mid],
00865 mkavl_find_type_e_get_string(type));
00866 }
00867 if (is_equal_type && (input->sorted_seq[mid] == val)) {
00868 return &(input->sorted_seq[mid]);
00869 }
00870
00871 if (is_greater_type && ((val >= input->sorted_seq[mid]) ||
00872 (0 == mid))) {
00873 if ((mid + 1) >= num_elems) {
00874 return (NULL);
00875 }
00876
00877 if ((0 == mid) && (val < input->sorted_seq[mid])) {
00878 return &(input->sorted_seq[mid]);
00879 }
00880
00881 if (val < input->sorted_seq[mid + 1]) {
00882 return &(input->sorted_seq[mid + 1]);
00883 }
00884 }
00885
00886 if (is_less_type && ((val <= input->sorted_seq[mid]) ||
00887 (mid == (num_elems - 1)))) {
00888 if (0 == mid) {
00889 return (NULL);
00890 }
00891
00892 if ((mid == (num_elems - 1)) && (val > input->sorted_seq[mid])) {
00893 return &(input->sorted_seq[mid]);
00894 }
00895
00896 if (val > input->sorted_seq[mid - 1]) {
00897 return &(input->sorted_seq[mid - 1]);
00898 }
00899 }
00900
00901 if (input->sorted_seq[mid] == val) {
00902 if (is_greater_type) {
00903 lo = (mid + 1);
00904 }
00905
00906 if (is_less_type) {
00907 hi = (mid - 1);
00908 }
00909 } else if (val > input->sorted_seq[mid]) {
00910 lo = (mid + 1);
00911 } else {
00912 hi = (mid - 1);
00913 if (is_greater_type) {
00914 hi = mid;
00915 }
00916 }
00917
00918 if (++runaway_counter > MKAVL_TEST_RUNAWAY_SANITY) {
00919 abort();
00920 }
00921 }
00922
00923 if (input->opts->verbosity >= 6) {
00924 printf("lo(%d) hi(%d)\n", lo, hi);
00925 }
00926
00927 return (NULL);
00928 }
00929
00937 static bool
00938 mkavl_test_find (mkavl_test_input_st *input, mkavl_find_type_e type)
00939 {
00940 uint32_t *existing_item, *array_item;
00941 uint32_t rand_lookup_val;
00942 uint32_t i, j;
00943 mkavl_rc_e rc;
00944 bool is_equal_type;
00945
00946 is_equal_type = ((MKAVL_FIND_TYPE_E_EQUAL == type) ||
00947 (MKAVL_FIND_TYPE_E_GE == type) ||
00948 (MKAVL_FIND_TYPE_E_LE == type));
00949
00950 for (i = 0; i < input->opts->node_cnt; ++i) {
00951 for (j = 0; j < MKAVL_TEST_KEY_E_MAX; ++j) {
00952
00953 rc = mkavl_find(input->tree_h, mkavl_key_find_type[j][type], j,
00954 &(input->insert_seq[i]), (void **) &existing_item);
00955 if (mkavl_rc_e_is_notok(rc)) {
00956 LOG_FAIL("find failed, rc(%s)", mkavl_rc_e_get_string(rc));
00957 return (false);
00958 }
00959
00960 if (is_equal_type) {
00961 if (NULL == existing_item) {
00962 LOG_FAIL("find failed for %u, type %s",
00963 input->insert_seq[i],
00964 mkavl_find_type_e_get_string(type));
00965 return (false);
00966 }
00967
00968 if (*existing_item != input->insert_seq[i]) {
00969 LOG_FAIL("find failed for %u, found %u type %s",
00970 input->insert_seq[i],
00971 *existing_item,
00972 mkavl_find_type_e_get_string(type));
00973 return (false);
00974 }
00975 }
00976
00977
00978
00979
00980
00981 array_item = mkavl_test_find_val(input, input->insert_seq[i], type);
00982 if (((NULL == existing_item) && (NULL != array_item)) ||
00983 ((NULL != existing_item) && (NULL == array_item)) ||
00984 ((NULL != existing_item) && (NULL != array_item) &&
00985 (*existing_item != *array_item))) {
00986 LOG_FAIL("mismatch in array and AVL find for %u, AVL(%p) %u "
00987 "array(%p) %u type %s key %u",
00988 input->insert_seq[i],
00989 existing_item,
00990 (NULL == existing_item) ? 0 : *existing_item,
00991 array_item,
00992 (NULL == array_item) ? 0 : *array_item,
00993 mkavl_find_type_e_get_string(type), j);
00994 return (false);
00995 }
00996
00997 if (input->opts->verbosity >= 5) {
00998 printf("find for type %s and key %u for %u, AVL(%p) %u "
00999 "array(%p) %u\n",
01000 mkavl_find_type_e_get_string(type),
01001 j, input->insert_seq[i],
01002 existing_item,
01003 (NULL == existing_item) ? 0 : *existing_item,
01004 array_item,
01005 (NULL == array_item) ? 0 : *array_item);
01006 }
01007
01008
01009 rand_lookup_val = ((rand() % input->opts->range_end) +
01010 input->opts->range_start);
01011 rc = mkavl_find(input->tree_h, mkavl_key_find_type[j][type], j,
01012 &rand_lookup_val, (void **) &existing_item);
01013 if (mkavl_rc_e_is_notok(rc)) {
01014 LOG_FAIL("find failed, rc(%s)", mkavl_rc_e_get_string(rc));
01015 return (false);
01016 }
01017
01018 array_item = mkavl_test_find_val(input, rand_lookup_val, type);
01019 if (((NULL == existing_item) && (NULL != array_item)) ||
01020 ((NULL != existing_item) && (NULL == array_item)) ||
01021 ((NULL != existing_item) && (NULL != array_item) &&
01022 (*existing_item != *array_item))) {
01023 LOG_FAIL("mismatch in array and AVL find for %u, AVL(%p) %u "
01024 "array(%p) %u type %s key %u",
01025 rand_lookup_val,
01026 existing_item,
01027 (NULL == existing_item) ? 0 : *existing_item,
01028 array_item,
01029 (NULL == array_item) ? 0 : *array_item,
01030 mkavl_find_type_e_get_string(type), j);
01031 return (false);
01032 }
01033
01034 if (input->opts->verbosity >= 5) {
01035 printf("find for type %s and key %u for %u, AVL(%p) %u "
01036 "array(%p) %u\n",
01037 mkavl_find_type_e_get_string(type),
01038 j, rand_lookup_val,
01039 existing_item,
01040 (NULL == existing_item) ? 0 : *existing_item,
01041 array_item,
01042 (NULL == array_item) ? 0 : *array_item);
01043 }
01044
01045 }
01046 }
01047
01048 return (true);
01049 }
01050
01057 static bool
01058 mkavl_test_find_error (mkavl_test_input_st *input)
01059 {
01060 uint32_t *existing_item;
01061 mkavl_rc_e rc;
01062
01063 rc = mkavl_find(NULL, MKAVL_FIND_TYPE_E_EQUAL, MKAVL_TEST_KEY_E_ASC,
01064 &(input->insert_seq[0]), (void **) &existing_item);
01065 if (mkavl_rc_e_is_ok(rc)) {
01066 LOG_FAIL("NULL tree failed, rc(%s)", mkavl_rc_e_get_string(rc));
01067 return (false);
01068 }
01069
01070 rc = mkavl_find(input->tree_h, (MKAVL_FIND_TYPE_E_MAX + 1),
01071 MKAVL_TEST_KEY_E_ASC,
01072 &(input->insert_seq[0]), (void **) &existing_item);
01073 if (mkavl_rc_e_is_ok(rc)) {
01074 LOG_FAIL("Invalid type failed, rc(%s)", mkavl_rc_e_get_string(rc));
01075 return (false);
01076 }
01077
01078 rc = mkavl_find(input->tree_h, MKAVL_FIND_TYPE_E_EQUAL,
01079 MKAVL_TEST_KEY_E_MAX,
01080 &(input->insert_seq[0]), (void **) &existing_item);
01081 if (mkavl_rc_e_is_ok(rc)) {
01082 LOG_FAIL("Invalid key index failed, rc(%s)", mkavl_rc_e_get_string(rc));
01083 return (false);
01084 }
01085
01086 rc = mkavl_find(input->tree_h, MKAVL_FIND_TYPE_E_EQUAL,
01087 MKAVL_TEST_KEY_E_ASC, NULL, (void **) &existing_item);
01088 if (mkavl_rc_e_is_ok(rc)) {
01089 LOG_FAIL("NULL item failed, rc(%s)", mkavl_rc_e_get_string(rc));
01090 return (false);
01091 }
01092
01093 rc = mkavl_find(input->tree_h, MKAVL_FIND_TYPE_E_EQUAL,
01094 MKAVL_TEST_KEY_E_ASC, &(input->insert_seq[0]), NULL);
01095 if (mkavl_rc_e_is_ok(rc)) {
01096 LOG_FAIL("NULL item failed, rc(%s)", mkavl_rc_e_get_string(rc));
01097 return (false);
01098 }
01099
01100 return (true);
01101 }
01102
01109 static bool
01110 mkavl_test_add_remove_key (mkavl_test_input_st *input)
01111 {
01112 uint32_t i, j;
01113 uint32_t non_null_cnt, null_cnt;
01114 uint32_t *existing_item;
01115 mkavl_rc_e rc;
01116
01117 for (i = 0; i < MKAVL_TEST_KEY_E_MAX; ++i) {
01118
01119 non_null_cnt = 0;
01120 for (j = 0; j < input->opts->node_cnt; ++j) {
01121 rc = mkavl_remove_key_idx(input->tree_h, i,
01122 &(input->delete_seq[j]),
01123 (void **) &existing_item);
01124 if (mkavl_rc_e_is_notok(rc)) {
01125 LOG_FAIL("remove key idx failed, rc(%s)",
01126 mkavl_rc_e_get_string(rc));
01127 return (false);
01128 }
01129
01130 if (NULL != existing_item) {
01131 ++non_null_cnt;
01132 }
01133
01134 rc = mkavl_find(input->tree_h, MKAVL_FIND_TYPE_E_EQUAL, i,
01135 &(input->delete_seq[j]), (void **) &existing_item);
01136 if (mkavl_rc_e_is_notok(rc)) {
01137 LOG_FAIL("find failed, rc(%s)", mkavl_rc_e_get_string(rc));
01138 return (false);
01139 }
01140
01141 if (NULL != existing_item) {
01142 LOG_FAIL("found item expected to be deleted, %u",
01143 input->delete_seq[j]);
01144 return (false);
01145 }
01146
01147 rc = mkavl_find(input->tree_h, MKAVL_FIND_TYPE_E_EQUAL,
01148 mkavl_key_opposite[i],
01149 &(input->delete_seq[j]), (void **) &existing_item);
01150 if (mkavl_rc_e_is_notok(rc)) {
01151 LOG_FAIL("find failed, rc(%s)", mkavl_rc_e_get_string(rc));
01152 return (false);
01153 }
01154
01155 if (NULL == existing_item) {
01156 LOG_FAIL("did not find item, %u", input->delete_seq[j]);
01157 return (false);
01158 }
01159 }
01160
01161 if (non_null_cnt != input->uniq_cnt) {
01162 LOG_FAIL("unique check failed, non_null_cnt(%u) uniq_cnt(%u)",
01163 non_null_cnt, input->uniq_cnt);
01164 return (false);
01165 }
01166
01167
01168 if (mkavl_count(input->tree_h) != input->uniq_cnt) {
01169 LOG_FAIL("unique check failed, mkavl_count(%u) uniq_cnt(%u)",
01170 mkavl_count(input->tree_h), input->uniq_cnt);
01171 return (false);
01172 }
01173
01174
01175 null_cnt = 0;
01176 for (j = 0; j < input->opts->node_cnt; ++j) {
01177 rc = mkavl_add_key_idx(input->tree_h, i, &(input->insert_seq[j]),
01178 (void **) &existing_item);
01179 if (mkavl_rc_e_is_notok(rc)) {
01180 LOG_FAIL("add key idx failed, rc(%s)",
01181 mkavl_rc_e_get_string(rc));
01182 return (false);
01183 }
01184
01185 if (NULL == existing_item) {
01186 ++null_cnt;
01187 }
01188 }
01189
01190 if (null_cnt != input->uniq_cnt) {
01191 LOG_FAIL("unique check failed, null_cnt(%u) uniq_cnt(%u)",
01192 null_cnt, input->uniq_cnt);
01193 return (false);
01194 }
01195
01196
01197 if (mkavl_count(input->tree_h) != input->uniq_cnt) {
01198 LOG_FAIL("unique check failed, mkavl_count(%u) uniq_cnt(%u)",
01199 mkavl_count(input->tree_h), input->uniq_cnt);
01200 return (false);
01201 }
01202 }
01203
01204 return (true);
01205 }
01206
01213 static bool
01214 mkavl_test_add_key_error (mkavl_test_input_st *input)
01215 {
01216 uint32_t *existing_item;
01217 mkavl_rc_e rc;
01218
01219 rc = mkavl_add_key_idx(NULL, MKAVL_TEST_KEY_E_ASC,
01220 &(input->insert_seq[0]), (void **) &existing_item);
01221 if (mkavl_rc_e_is_ok(rc)) {
01222 LOG_FAIL("Key index operation failed, rc(%s)",
01223 mkavl_rc_e_get_string(rc));
01224 return (false);
01225 }
01226
01227 rc = mkavl_add_key_idx(input->tree_h, MKAVL_TEST_KEY_E_MAX,
01228 &(input->insert_seq[0]), (void **) &existing_item);
01229 if (mkavl_rc_e_is_ok(rc)) {
01230 LOG_FAIL("Key index operation failed, rc(%s)",
01231 mkavl_rc_e_get_string(rc));
01232 return (false);
01233 }
01234
01235 rc = mkavl_add_key_idx(input->tree_h, MKAVL_TEST_KEY_E_ASC,
01236 NULL, (void **) &existing_item);
01237 if (mkavl_rc_e_is_ok(rc)) {
01238 LOG_FAIL("Key index operation failed, rc(%s)",
01239 mkavl_rc_e_get_string(rc));
01240 return (false);
01241 }
01242
01243 rc = mkavl_add_key_idx(input->tree_h, MKAVL_TEST_KEY_E_ASC,
01244 &(input->insert_seq[0]), NULL);
01245 if (mkavl_rc_e_is_ok(rc)) {
01246 LOG_FAIL("Key index operation failed, rc(%s)",
01247 mkavl_rc_e_get_string(rc));
01248 return (false);
01249 }
01250
01251 return (true);
01252 }
01253
01260 static bool
01261 mkavl_test_remove_key_error (mkavl_test_input_st *input)
01262 {
01263 uint32_t *existing_item;
01264 mkavl_rc_e rc;
01265
01266 rc = mkavl_remove_key_idx(NULL, MKAVL_TEST_KEY_E_ASC,
01267 &(input->insert_seq[0]), (void **) &existing_item);
01268 if (mkavl_rc_e_is_ok(rc)) {
01269 LOG_FAIL("Key index operation failed, rc(%s)",
01270 mkavl_rc_e_get_string(rc));
01271 return (false);
01272 }
01273
01274 rc = mkavl_remove_key_idx(input->tree_h, MKAVL_TEST_KEY_E_MAX,
01275 &(input->insert_seq[0]), (void **) &existing_item);
01276 if (mkavl_rc_e_is_ok(rc)) {
01277 LOG_FAIL("Key index operation failed, rc(%s)",
01278 mkavl_rc_e_get_string(rc));
01279 return (false);
01280 }
01281
01282 rc = mkavl_remove_key_idx(input->tree_h, MKAVL_TEST_KEY_E_ASC,
01283 NULL, (void **) &existing_item);
01284 if (mkavl_rc_e_is_ok(rc)) {
01285 LOG_FAIL("Key index operation failed, rc(%s)",
01286 mkavl_rc_e_get_string(rc));
01287 return (false);
01288 }
01289
01290 rc = mkavl_remove_key_idx(input->tree_h, MKAVL_TEST_KEY_E_ASC,
01291 &(input->insert_seq[0]), NULL);
01292 if (mkavl_rc_e_is_ok(rc)) {
01293 LOG_FAIL("Key index operation failed, rc(%s)",
01294 mkavl_rc_e_get_string(rc));
01295 return (false);
01296 }
01297
01298 return (true);
01299 }
01300
01308 static void *
01309 mkavl_test_copy_fn (void *item, void *context)
01310 {
01311 mkavl_test_ctx_st *ctx;
01312
01313 ctx = (mkavl_test_ctx_st *) context;
01314
01315 if ((NULL == ctx) || (MKAVL_TEST_MAGIC != ctx->magic)) {
01316 abort();
01317 }
01318 ++(ctx->copy_cnt);
01319
01320 return (item);
01321 }
01322
01329 static bool
01330 mkavl_test_copy (mkavl_test_input_st *input)
01331 {
01332 mkavl_rc_e rc;
01333 mkavl_test_ctx_st *ctx, *src_ctx;
01334
01335 src_ctx = mkavl_get_tree_context(input->tree_h);
01336 if (NULL == src_ctx) {
01337 LOG_FAIL("NULL context pointer");
01338 return (false);
01339 }
01340
01341 ctx = calloc(1, sizeof(*ctx));
01342 if (NULL == ctx) {
01343 LOG_FAIL("calloc failed");
01344 return (false);
01345 }
01346 ctx->magic = MKAVL_TEST_MAGIC;
01347
01348 rc = mkavl_copy(input->tree_h,
01349 &(input->tree_copy_h),
01350 mkavl_test_copy_fn, NULL, false, ctx,
01351 mkavl_test_delete_context,
01352 ©_allocator);
01353 if (mkavl_rc_e_is_notok(rc)) {
01354 LOG_FAIL("copy failed, rc(%s)", mkavl_rc_e_get_string(rc));
01355 return (false);
01356 }
01357
01358 if (src_ctx->copy_cnt != input->uniq_cnt) {
01359 LOG_FAIL("unexpected copy count, copy count %u "
01360 "unique count %u)", ctx->copy_cnt, input->uniq_cnt);
01361 return (false);
01362 }
01363
01364 if (mkavl_count(input->tree_h) != mkavl_count(input->tree_copy_h)) {
01365 LOG_FAIL("unequal count after copy, original %u copy %u)",
01366 mkavl_count(input->tree_h), mkavl_count(input->tree_copy_h));
01367 return (false);
01368 }
01369
01370 return (true);
01371 }
01372
01379 static bool
01380 mkavl_test_iterator (mkavl_test_input_st *input)
01381 {
01382 mkavl_iterator_handle iter1_h = NULL, iter2_h = NULL;
01383 mkavl_iterator_handle copy_iter1_h = NULL;
01384 uint32_t last_idx = (input->opts->node_cnt - 1);
01385 uint32_t *item, *copy_item, *cur_item, *prev_item, *found_item;
01386 uint32_t idx;
01387 mkavl_rc_e rc;
01388 bool retval = true;
01389
01390
01391
01392
01393
01394 rc = mkavl_iter_new(&iter1_h, input->tree_h, MKAVL_TEST_KEY_E_ASC);
01395 if (mkavl_rc_e_is_notok(rc)) {
01396 LOG_FAIL("new iterator failed, rc(%s)", mkavl_rc_e_get_string(rc));
01397 retval = false;
01398 goto cleanup;
01399 }
01400
01401 rc = mkavl_iter_new(&iter2_h, input->tree_h, MKAVL_TEST_KEY_E_DESC);
01402 if (mkavl_rc_e_is_notok(rc)) {
01403 LOG_FAIL("new iterator failed, rc(%s)", mkavl_rc_e_get_string(rc));
01404 retval = false;
01405 goto cleanup;
01406 }
01407
01408 rc = mkavl_iter_new(©_iter1_h, input->tree_copy_h,
01409 MKAVL_TEST_KEY_E_ASC);
01410 if (mkavl_rc_e_is_notok(rc)) {
01411 LOG_FAIL("new iterator failed, rc(%s)", mkavl_rc_e_get_string(rc));
01412 retval = false;
01413 goto cleanup;
01414 }
01415
01416 rc = mkavl_iter_last(iter1_h, (void **) &item);
01417 if (mkavl_rc_e_is_notok(rc)) {
01418 LOG_FAIL("iterator operation failed, rc(%s)",
01419 mkavl_rc_e_get_string(rc));
01420 retval = false;
01421 goto cleanup;
01422 }
01423
01424 if (*item != input->sorted_seq[last_idx]) {
01425 LOG_FAIL("iterator item value mismatch, item %u array val %u",
01426 *item, input->sorted_seq[last_idx]);
01427 retval = false;
01428 goto cleanup;
01429 }
01430
01431 rc = mkavl_iter_last(iter2_h, (void **) &item);
01432 if (mkavl_rc_e_is_notok(rc)) {
01433 LOG_FAIL("iterator operation failed, rc(%s)",
01434 mkavl_rc_e_get_string(rc));
01435 retval = false;
01436 goto cleanup;
01437 }
01438
01439 if (*item != input->sorted_seq[0]) {
01440 LOG_FAIL("iterator item value mismatch, item %u array val %u",
01441 *item, input->sorted_seq[0]);
01442 retval = false;
01443 goto cleanup;
01444 }
01445
01446 rc = mkavl_iter_first(iter2_h, (void **) &item);
01447 if (mkavl_rc_e_is_notok(rc)) {
01448 LOG_FAIL("iterator operation failed, rc(%s)",
01449 mkavl_rc_e_get_string(rc));
01450 retval = false;
01451 goto cleanup;
01452 }
01453
01454 if (*item != input->sorted_seq[last_idx]) {
01455 LOG_FAIL("iterator item value mismatch, item %u array val %u",
01456 *item, input->sorted_seq[last_idx]);
01457 retval = false;
01458 goto cleanup;
01459 }
01460
01461 rc = mkavl_iter_first(iter1_h, (void **) &item);
01462 if (mkavl_rc_e_is_notok(rc)) {
01463 LOG_FAIL("iterator operation failed, rc(%s)",
01464 mkavl_rc_e_get_string(rc));
01465 retval = false;
01466 goto cleanup;
01467 }
01468
01469 if (*item != input->sorted_seq[0]) {
01470 LOG_FAIL("iterator item value mismatch, item %u array val %u",
01471 *item, input->sorted_seq[0]);
01472 retval = false;
01473 goto cleanup;
01474 }
01475
01476 rc = mkavl_iter_first(copy_iter1_h, (void **) ©_item);
01477 if (mkavl_rc_e_is_notok(rc)) {
01478 LOG_FAIL("iterator operation failed, rc(%s)",
01479 mkavl_rc_e_get_string(rc));
01480 retval = false;
01481 goto cleanup;
01482 }
01483
01484 idx = 0;
01485 prev_item = NULL;
01486 while ((NULL != item) && (NULL != copy_item)) {
01487 if (idx >= input->opts->node_cnt) {
01488 LOG_FAIL("invalid idx(%u), node_cnt(%u)", idx,
01489 input->opts->node_cnt);
01490 retval = false;
01491 goto cleanup;
01492 }
01493
01494 if (*item != *copy_item) {
01495 LOG_FAIL("iterator has mismatch, item %u copy_item %u", *item,
01496 *copy_item);
01497 retval = false;
01498 goto cleanup;
01499 }
01500
01501 if (*item != input->sorted_seq[idx]) {
01502 LOG_FAIL("iterator has mismatch, item %u sorted_seq %u", *item,
01503 input->sorted_seq[idx]);
01504 retval = false;
01505 goto cleanup;
01506 }
01507
01508
01509 while ((idx < input->opts->node_cnt) &&
01510 (*item == input->sorted_seq[idx])) {
01511 ++idx;
01512 }
01513
01514
01515 rc = mkavl_iter_cur(iter1_h, (void **) &cur_item);
01516 if (mkavl_rc_e_is_notok(rc)) {
01517 LOG_FAIL("iterator operation failed, rc(%s)",
01518 mkavl_rc_e_get_string(rc));
01519 retval = false;
01520 goto cleanup;
01521 }
01522
01523 if (item != cur_item) {
01524 LOG_FAIL("iterator has mismatch, item %p cur_item %p", item,
01525 cur_item);
01526 retval = false;
01527 goto cleanup;
01528 }
01529
01530
01531 rc = mkavl_iter_prev(iter1_h, (void **) &item);
01532 if (mkavl_rc_e_is_notok(rc)) {
01533 LOG_FAIL("iterator operation failed, rc(%s)",
01534 mkavl_rc_e_get_string(rc));
01535 retval = false;
01536 goto cleanup;
01537 }
01538
01539 if (prev_item != item) {
01540 LOG_FAIL("iterator has mismatch, item %p prev_item %p", item,
01541 prev_item);
01542 retval = false;
01543 goto cleanup;
01544 }
01545
01546
01547 rc = mkavl_iter_find(iter1_h, cur_item, (void **) &found_item);
01548 if (mkavl_rc_e_is_notok(rc)) {
01549 LOG_FAIL("iterator operation failed, rc(%s)",
01550 mkavl_rc_e_get_string(rc));
01551 retval = false;
01552 goto cleanup;
01553 }
01554
01555 if (found_item != cur_item) {
01556 LOG_FAIL("iterator has mismatch, found_item %p cur_item %p",
01557 found_item, cur_item);
01558 retval = false;
01559 goto cleanup;
01560 }
01561
01562 rc = mkavl_iter_next(iter1_h, (void **) &item);
01563 if (mkavl_rc_e_is_notok(rc)) {
01564 LOG_FAIL("iterator operation failed, rc(%s)",
01565 mkavl_rc_e_get_string(rc));
01566 retval = false;
01567 goto cleanup;
01568 }
01569
01570 rc = mkavl_iter_next(copy_iter1_h, (void **) ©_item);
01571 if (mkavl_rc_e_is_notok(rc)) {
01572 LOG_FAIL("iterator operation failed, rc(%s)",
01573 mkavl_rc_e_get_string(rc));
01574 retval = false;
01575 goto cleanup;
01576 }
01577
01578 prev_item = cur_item;
01579 }
01580
01581 if (item != copy_item) {
01582 LOG_FAIL("iterator has mismatch, item %p copy_item %p", item,
01583 copy_item);
01584 retval = false;
01585 goto cleanup;
01586 }
01587
01588 cleanup:
01589
01590 if (NULL != iter1_h) {
01591 mkavl_iter_delete(&iter1_h);
01592 }
01593
01594 if (NULL != iter2_h) {
01595 mkavl_iter_delete(&iter2_h);
01596 }
01597
01598 if (NULL != copy_iter1_h) {
01599 mkavl_iter_delete(©_iter1_h);
01600 }
01601
01602 return (retval);
01603 }
01604
01606 typedef struct mkavl_test_walk_ctx_st_ {
01608 uint32_t magic;
01610 uint32_t walk_node_cnt;
01612 uint32_t walk_stop_cnt;
01613 } mkavl_test_walk_ctx_st;
01614
01624 static mkavl_rc_e
01625 mkavl_test_walk_cb (void *item, void *tree_context, void *walk_context,
01626 bool *stop_walk)
01627 {
01628 mkavl_test_walk_ctx_st *walk_ctx = (mkavl_test_walk_ctx_st *) walk_context;
01629 mkavl_test_ctx_st *tree_ctx = (mkavl_test_ctx_st *) tree_context;
01630
01631 if ((NULL == item) || (NULL == walk_ctx) || (NULL == tree_ctx) ||
01632 (NULL == stop_walk) || (MKAVL_TEST_MAGIC != walk_ctx->magic) ||
01633 (MKAVL_TEST_MAGIC != tree_ctx->magic)) {
01634 abort();
01635 }
01636
01637 if (walk_ctx->walk_stop_cnt == walk_ctx->walk_node_cnt) {
01638 *stop_walk = true;
01639 } else {
01640 ++(walk_ctx->walk_node_cnt);
01641 }
01642
01643 return (MKAVL_RC_E_SUCCESS);
01644 }
01645
01652 static bool
01653 mkavl_test_walk (mkavl_test_input_st *input)
01654 {
01655 mkavl_test_walk_ctx_st walk_ctx = {0};
01656 mkavl_rc_e rc;
01657
01658 walk_ctx.magic = MKAVL_TEST_MAGIC;
01659
01660 walk_ctx.walk_stop_cnt = input->uniq_cnt;
01661 rc = mkavl_walk(input->tree_h, mkavl_test_walk_cb, &walk_ctx);
01662 if (mkavl_rc_e_is_notok(rc)) {
01663 LOG_FAIL("walk failed, rc(%s)", mkavl_rc_e_get_string(rc));
01664 return (false);
01665 }
01666
01667 if (walk_ctx.walk_node_cnt != walk_ctx.walk_stop_cnt) {
01668 LOG_FAIL("unexpected walk count, walk_node_cnt(%u) stop_cnt(%u)",
01669 walk_ctx.walk_node_cnt, walk_ctx.walk_stop_cnt);
01670 return (false);
01671 }
01672
01673 walk_ctx.walk_node_cnt = 0;
01674
01675 walk_ctx.walk_stop_cnt = (rand() % input->uniq_cnt);
01676 rc = mkavl_walk(input->tree_copy_h, mkavl_test_walk_cb, &walk_ctx);
01677 if (mkavl_rc_e_is_notok(rc)) {
01678 LOG_FAIL("walk failed, rc(%s)", mkavl_rc_e_get_string(rc));
01679 return (false);
01680 }
01681
01682 if (walk_ctx.walk_node_cnt != walk_ctx.walk_stop_cnt) {
01683 LOG_FAIL("unexpected walk count, walk_node_cnt(%u) stop_cnt(%u)",
01684 walk_ctx.walk_node_cnt, walk_ctx.walk_stop_cnt);
01685 return (false);
01686 }
01687
01688 walk_ctx.magic = 0;
01689
01690 return (true);
01691 }
01692
01699 static bool
01700 mkavl_test_remove (mkavl_test_input_st *input)
01701 {
01702 uint32_t i, null_cnt = 0;
01703 mkavl_rc_e rc;
01704 uint32_t *found_item;
01705
01706 for (i = 0; i < input->opts->node_cnt; ++i) {
01707 rc = mkavl_remove(input->tree_h, &(input->delete_seq[i]),
01708 (void **) &found_item);
01709 if (mkavl_rc_e_is_notok(rc)) {
01710 LOG_FAIL("remove failed, rc(%s)",
01711 mkavl_rc_e_get_string(rc));
01712 return (false);
01713 }
01714
01715 if (NULL == found_item) {
01716 ++null_cnt;
01717 }
01718 }
01719
01720 if (null_cnt != input->dup_cnt) {
01721 LOG_FAIL("duplicate check failed, null_cnt(%u) dup_cnt(%u)",
01722 null_cnt, input->dup_cnt);
01723 return (false);
01724 }
01725
01726 if (0 != mkavl_count(input->tree_h)) {
01727 LOG_FAIL("remove count check failed, count(%u)",
01728 mkavl_count(input->tree_h));
01729 return (false);
01730 }
01731
01732 return (true);
01733 }
01734
01742 static mkavl_rc_e
01743 mkavl_test_item_fn (void *item, void *context)
01744 {
01745 mkavl_test_ctx_st *ctx = (mkavl_test_ctx_st *) context;
01746
01747 if ((NULL == item) || (NULL == ctx) ||
01748 (MKAVL_TEST_MAGIC != ctx->magic)) {
01749 abort();
01750 }
01751
01752 ++(ctx->item_fn_cnt);
01753
01754 return (MKAVL_RC_E_SUCCESS);
01755 }
01756
01763 static bool
01764 run_mkavl_test (mkavl_test_input_st *input)
01765 {
01766 mkavl_find_type_e find_type;
01767 mkavl_test_ctx_st *ctx;
01768 bool test_rc;
01769
01770 if (NULL == input) {
01771 LOG_FAIL("invalid input");
01772 return (false);
01773 }
01774
01775 test_rc = mkavl_test_new(input, NULL);
01776 if (!test_rc) {
01777 goto err_exit;
01778 }
01779
01780
01781 test_rc = mkavl_test_delete(input, mkavl_test_item_fn,
01782 mkavl_test_delete_context,
01783 mkavl_test_delete_context);
01784 if (!test_rc) {
01785 goto err_exit;
01786 }
01787
01788 test_rc = mkavl_test_new(input, NULL);
01789 if (!test_rc) {
01790 goto err_exit;
01791 }
01792
01793
01794 test_rc = mkavl_test_new_error();
01795 if (!test_rc) {
01796 goto err_exit;
01797 }
01798
01799
01800 test_rc = mkavl_test_add(input);
01801 if (!test_rc) {
01802 goto err_exit;
01803 }
01804
01805
01806 test_rc = mkavl_test_add_error(input);
01807 if (!test_rc) {
01808 goto err_exit;
01809 }
01810
01811
01812 for (find_type = MKAVL_FIND_TYPE_E_FIRST; find_type < MKAVL_FIND_TYPE_E_MAX;
01813 ++find_type) {
01814 test_rc = mkavl_test_find(input, find_type);
01815 if (!test_rc) {
01816 goto err_exit;
01817 }
01818 }
01819
01820
01821 test_rc = mkavl_test_find_error(input);
01822 if (!test_rc) {
01823 goto err_exit;
01824 }
01825
01826
01827 test_rc = mkavl_test_add_remove_key(input);
01828 if (!test_rc) {
01829 goto err_exit;
01830 }
01831
01832
01833 test_rc = mkavl_test_add_key_error(input);
01834 if (!test_rc) {
01835 goto err_exit;
01836 }
01837
01838 test_rc = mkavl_test_remove_key_error(input);
01839 if (!test_rc) {
01840 goto err_exit;
01841 }
01842
01843
01844 test_rc = mkavl_test_copy(input);
01845 if (!test_rc) {
01846 goto err_exit;
01847 }
01848
01849
01850 test_rc = mkavl_test_iterator(input);
01851 if (!test_rc) {
01852 goto err_exit;
01853 }
01854
01855
01856 test_rc = mkavl_test_walk(input);
01857 if (!test_rc) {
01858 goto err_exit;
01859 }
01860
01861
01862
01863
01864
01865 test_rc = mkavl_test_remove(input);
01866 if (!test_rc) {
01867 goto err_exit;
01868 }
01869
01870 ctx = mkavl_get_tree_context(input->tree_copy_h);
01871 if (NULL == ctx) {
01872 LOG_FAIL("NULL context");
01873 goto err_exit;
01874 }
01875
01876
01877
01878
01879
01880 test_rc = mkavl_test_delete(input, mkavl_test_item_fn,
01881 mkavl_test_delete_context, NULL);
01882 if (!test_rc) {
01883 goto err_exit;
01884 }
01885
01886 if (ctx->item_fn_cnt != input->uniq_cnt) {
01887 LOG_FAIL("item fn count(%u) != uniq count(%u)", ctx->item_fn_cnt,
01888 input->uniq_cnt);
01889 goto err_exit;
01890 }
01891
01892 if (ctx->copy_malloc_cnt != ctx->copy_free_cnt) {
01893 LOG_FAIL("malloc count(%u) != free count(%u)",
01894 ctx->copy_malloc_cnt, ctx->copy_free_cnt);
01895 goto err_exit;
01896 }
01897
01898 free(ctx);
01899 ctx = NULL;
01900
01901 return (true);
01902
01903 err_exit:
01904
01905 mkavl_test_delete(input, mkavl_test_item_fn, mkavl_test_delete_context,
01906 mkavl_test_delete_context);
01907
01908 return (false);
01909 }