X-Git-Url: http://mj.ucw.cz/gitweb/?a=blobdiff_plain;f=lib%2Fredblack.h;h=a89149b20afedaea67ff28ae7550fb634afa5c73;hb=18fcca1138ad1b480854cb1e64fa10a84660dcba;hp=b6c37852c78adc559414b2e1f72e2ecdc6f36fd0;hpb=e7e89924c83ff16ebb4dd6b30b68b0292548a6ea;p=libucw.git diff --git a/lib/redblack.h b/lib/redblack.h index b6c37852..a89149b2 100644 --- a/lib/redblack.h +++ b/lib/redblack.h @@ -1,7 +1,7 @@ /* - * Red-black trees + * UCW Library -- Red-black trees * - * (c) 2002, Robert Spalek + * (c) 2002--2005, Robert Spalek * * Skeleton based on hash-tables by: * @@ -15,7 +15,7 @@ * A red-black tree is a binary search tree, where records are stored * in nodes (may be also leaves). Every node has a colour. The * following restrictions hold: - * + * * - a parent of a red node is black * - every path from the root to a node with less than 2 children * contains the same number of black nodes @@ -65,8 +65,14 @@ * key, return NULL if no such node exists. * TREE_WANT_FIND_NEXT node *find_next(node *start) -- find next node with the * specified key, return NULL if no such node exists. + * Implies TREE_DUPLICATES. * TREE_WANT_SEARCH node *search(key) -- find the node with the specified * or, if it does not exist, the nearest one. + * TREE_WANT_SEARCH_DOWN node *search_down(key) -- find either the node with + * specified value, or if it does not exist, the node + * with nearest smaller value. + * TREE_WANT_BOUNDARY node *boundary(uns direction) -- finds smallest + * (direction==0) or largest (direction==1) node. * TREE_WANT_ADJACENT node *adjacent(node *, uns direction) -- finds next * (direction==1) or previous (direction==0) node. * TREE_WANT_NEW node *new(key) -- create new node with given key. @@ -108,6 +114,7 @@ * TREE_GLOBAL Functions are exported (i.e., not static). * TREE_CONSERVE_SPACE Use as little space as possible at the price of a * little slowdown. + * TREE_DUPLICATES Records with duplicate keys are allowed. * TREE_MAX_DEPTH Maximal depth of a tree (for stack allocation). * * If you set TREE_WANT_ITERATOR, you also get a iterator macro at no @@ -129,6 +136,7 @@ * undef'd. */ +#include #include #if !defined(TREE_NODE) || !defined(TREE_PREFIX) @@ -256,13 +264,13 @@ typedef struct P(stack_entry) { #else /* Pointers are aligned, hence we can use lower bits. */ static inline uns P(red_flag) (P(bucket) *node) - { return ((long) node->son[0]) & 1L; } + { return ((uintptr_t) node->son[0]) & 1L; } static inline void P(set_red_flag) (P(bucket) *node, uns flag) - { (long) node->son[0] = (((long) node->son[0]) & ~1L) | (flag & 1L); } + { node->son[0] = (void*) ( (((uintptr_t) node->son[0]) & ~1L) | (flag & 1L) ); } static inline P(bucket) * P(tree_son) (P(bucket) *node, uns id) - { return (void *) (((long) node->son[id]) & ~1L); } + { return (void *) (((uintptr_t) node->son[id]) & ~1L); } static inline void P(set_tree_son) (P(bucket) *node, uns id, P(bucket) *son) - { node->son[id] = (void *) ((long) son | (((long) node->son[id]) & 1L) ); } + { node->son[id] = (void *) ((uintptr_t) son | (((uintptr_t) node->son[id]) & 1L) ); } #endif /* Defaults for missing parameters. */ @@ -323,6 +331,19 @@ static inline void P(init_data) (P(node) *n UNUSED) # define TREE_MAX_DEPTH 64 #endif +#if defined(TREE_WANT_FIND_NEXT) && !defined(TREE_DUPLICATES) +# define TREE_DUPLICATES +#endif + +#ifdef TREE_WANT_LOOKUP +#ifndef TREE_WANT_FIND +# define TREE_WANT_FIND +#endif +#ifndef TREE_WANT_NEW +# define TREE_WANT_NEW +#endif +#endif + /* Now the operations */ STATIC void P(init) (T *t) @@ -367,7 +388,7 @@ static uns P(fill_stack) (P(stack_entry) *stack, uns max_depth, P(bucket) *node, ASSERT(i+1 < max_depth); stack[i+1].buck = P(tree_son) (stack[i].buck, stack[i].son); } -#ifdef TREE_WANT_FIND_NEXT +#ifdef TREE_DUPLICATES if (stack[i].buck) { uns idx; @@ -385,7 +406,7 @@ static uns P(fill_stack) (P(stack_entry) *stack, uns max_depth, P(bucket) *node, return i; } -#if defined(TREE_WANT_FIND) || defined(TREE_WANT_LOOKUP) +#ifdef TREE_WANT_FIND STATIC P(node) * P(find) (T *t, TREE_KEY_DECL) { P(stack_entry) stack[TREE_MAX_DEPTH]; @@ -395,6 +416,45 @@ STATIC P(node) * P(find) (T *t, TREE_KEY_DECL) } #endif +#ifdef TREE_WANT_SEARCH_DOWN +STATIC P(node) * P(search_down) (T *t, TREE_KEY_DECL) +{ + P(node) *last_right=NULL; + P(bucket) *node=t->root; + while(node) + { + int cmp; + cmp = P(cmp) (TREE_KEY(), TREE_KEY(node->n.)); + if (cmp == 0) + return &node->n; + else if (cmp < 0) + node=P(tree_son) (node, 0); + else + { + last_right=&node->n; + node=P(tree_son) (node, 1); + } + } + return last_right; +} +#endif + +#ifdef TREE_WANT_BOUNDARY +STATIC P(node) * P(boundary) (T *t, uns direction) +{ + P(bucket) *n = t->root, *ns; + if (!n) + return NULL; + else + { + uns son = !!direction; + while ((ns = P(tree_son) (n, son))) + n = ns; + return &n->n; + } +} +#endif + #ifdef TREE_STORE_PARENT STATIC P(node) * P(adjacent) (P(node) *start, uns direction) { @@ -426,7 +486,7 @@ STATIC P(node) * P(adjacent) (P(node) *start, uns direction) } #endif -#if defined(TREE_WANT_FIND_NEXT) || defined(TREE_WANT_DELETE) || defined(TREE_WANT_REMOVE) +#if defined(TREE_DUPLICATES) || defined(TREE_WANT_DELETE) || defined(TREE_WANT_REMOVE) static int P(find_next_node) (P(stack_entry) *stack, uns max_depth, uns direction) { uns depth = 0; @@ -565,14 +625,14 @@ try_it_again: t->root = node; } -#if defined(TREE_WANT_NEW) || defined(TREE_WANT_LOOKUP) +#ifdef TREE_WANT_NEW STATIC P(node) * P(new) (T *t, TREE_KEY_DECL) { P(stack_entry) stack[TREE_MAX_DEPTH]; P(bucket) *added; uns depth; depth = P(fill_stack) (stack, TREE_MAX_DEPTH, t->root, TREE_KEY(), 1); -#ifdef TREE_WANT_FIND_NEXT +#ifdef TREE_DUPLICATES /* It is the last found value, hence everything in the right subtree is * strongly _bigger_. */ depth += P(find_next_node) (stack+depth, TREE_MAX_DEPTH-depth, 1); @@ -868,7 +928,7 @@ static void P(dump_subtree) (struct fastbuf *fb, T *t, P(bucket) *node, P(bucket { ASSERT(!flag || !P(red_flag) (parent)); cmp_res *= P(cmp) (TREE_KEY(node->n.), TREE_KEY(parent->n.)); -#ifdef TREE_WANT_FIND_NEXT +#ifdef TREE_DUPLICATES ASSERT(cmp_res >= 0); #else ASSERT(cmp_res > 0); @@ -925,13 +985,12 @@ static P(node) * P(first_node) (T *t, uns direction) #define TREE_FOR_ALL(t_px, t_ptr, t_var) \ do \ { \ - TREE_GLUE(t_px,node) *t_var = TREE_GLUE(t_px,first_node)(t_ptr, 0); \ - for (; t_var; t_var = TREE_GLUE(t_px,adjacent)(t_var, 1)) \ + GLUE_(t_px,node) *t_var = GLUE_(t_px,first_node)(t_ptr, 0); \ + for (; t_var; t_var = GLUE_(t_px,adjacent)(t_var, 1)) \ { #define TREE_END_FOR } } while(0) #define TREE_BREAK break #define TREE_CONTINUE continue -#define TREE_GLUE(x,y) x##_##y #endif #endif @@ -952,6 +1011,8 @@ do \ #undef TREE_WANT_FIND #undef TREE_WANT_FIND_NEXT #undef TREE_WANT_SEARCH +#undef TREE_WANT_SEARCH_DOWN +#undef TREE_WANT_BOUNDARY #undef TREE_WANT_ADJACENT #undef TREE_WANT_NEW #undef TREE_WANT_LOOKUP @@ -969,6 +1030,7 @@ do \ #undef TREE_USE_POOL #undef TREE_STATIC #undef TREE_CONSERVE_SPACE +#undef TREE_DUPLICATES #undef TREE_MAX_DEPTH #undef TREE_STORE_PARENT #undef TREE_KEY