test/test_mkavl.c
Go to the documentation of this file.
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  * Do a forward declaration so we can keep all the AVL setup stuff separate
00361  * below main().
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 /* AVL operation functions */
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             /* Do the operation on an existing item */
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              * Make sure what we founds matches a binary search on
00979              * the sorted array.
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             /* Do the operation on a (potentially) non-existing item */
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         /* Take them all out for one key */
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         /* Tree count should remain unchanged */
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         /* Put them all back in for the key */
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         /* Tree count should remain unchanged */
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                     &copy_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      * Iterate over both trees, make sure order is the same, everything is less
01392      * than in one iterator and greater than in the other.
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(&copy_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 **) &copy_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         /* Go to the next unique value in the sorted array */
01509         while ((idx < input->opts->node_cnt) &&
01510                (*item == input->sorted_seq[idx])) {
01511             ++idx;
01512         }
01513 
01514         /* Test the current function */
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         /* Test previous */
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         /* Test find */
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 **) &copy_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(&copy_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     /* Set it high enough that this walk will go all the way */
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     /* Set it high enough that this walk will go all the way */
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     /* Destroy an empty tree */
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     /* Test new error input */
01794     test_rc = mkavl_test_new_error();
01795     if (!test_rc) {
01796         goto err_exit;
01797     }
01798 
01799     /* Add in all the items */
01800     test_rc = mkavl_test_add(input);
01801     if (!test_rc) {
01802         goto err_exit;
01803     }
01804 
01805     /* Test add error input */
01806     test_rc = mkavl_test_add_error(input);
01807     if (!test_rc) {
01808         goto err_exit;
01809     }
01810 
01811     /* Test all types of find */
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     /* Test find error input */
01821     test_rc = mkavl_test_find_error(input);
01822     if (!test_rc) {
01823         goto err_exit;
01824     }
01825 
01826     /* Test find and add/remove from key */
01827     test_rc = mkavl_test_add_remove_key(input);
01828     if (!test_rc) {
01829         goto err_exit;
01830     }
01831 
01832     /* Test add/remove idx error conditions */
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     /* Test copying a tree */
01844     test_rc = mkavl_test_copy(input);
01845     if (!test_rc) {
01846         goto err_exit;
01847     }
01848 
01849     /* Test iterators */
01850     test_rc = mkavl_test_iterator(input);
01851     if (!test_rc) {
01852         goto err_exit;
01853     }
01854 
01855     /* Do walk over trees */
01856     test_rc = mkavl_test_walk(input);
01857     if (!test_rc) {
01858         goto err_exit;
01859     }
01860 
01861     /* 
01862      * Remove items from the original tree, let the items remain in the copied
01863      * tree so mkavl_delete handles them.
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      * Destroy both trees: make sure the delete function is called as expected
01878      * for the copied tree.
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 }
 All Classes Files Functions Variables Typedefs Enumerations Enumerator Defines