| 
									
										
										
										
											1998-05-15 07:54:32 +00:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  *	BIRD -- Forwarding Information Base -- Data Structures | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *	(c) 1998 Martin Mares <mj@ucw.cz> | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *	Can be freely distributed and used under the terms of the GNU GPL. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define LOCAL_DEBUG
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <string.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "nest/bird.h"
 | 
					
						
							|  |  |  | #include "nest/route.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-12-20 14:01:20 +00:00
										 |  |  | #define HASH_DEF_ORDER 10
 | 
					
						
							|  |  |  | #define HASH_HI_MARK *4
 | 
					
						
							|  |  |  | #define HASH_HI_STEP 2
 | 
					
						
							|  |  |  | #define HASH_HI_MAX 16			/* Must be at most 16 */
 | 
					
						
							|  |  |  | #define HASH_LO_MARK /5
 | 
					
						
							|  |  |  | #define HASH_LO_STEP 2
 | 
					
						
							|  |  |  | #define HASH_LO_MIN 10
 | 
					
						
							| 
									
										
										
										
											1998-05-15 07:54:32 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | fib_ht_alloc(struct fib *f) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											1998-12-20 14:01:20 +00:00
										 |  |  |   f->hash_size = 1 << f->hash_order; | 
					
						
							|  |  |  |   f->hash_shift = 16 - f->hash_order; | 
					
						
							|  |  |  |   if (f->hash_order > HASH_HI_MAX - HASH_HI_STEP) | 
					
						
							|  |  |  |     f->entries_max = ~0; | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  |     f->entries_max = f->hash_size HASH_HI_MARK; | 
					
						
							|  |  |  |   if (f->hash_order < HASH_LO_MIN + HASH_LO_STEP) | 
					
						
							| 
									
										
										
										
											1998-05-15 07:54:32 +00:00
										 |  |  |     f->entries_min = 0; | 
					
						
							| 
									
										
										
										
											1998-12-20 14:01:20 +00:00
										 |  |  |   else | 
					
						
							|  |  |  |     f->entries_min = f->hash_size HASH_LO_MARK; | 
					
						
							|  |  |  |   DBG("Allocating FIB hash of order %d: %d entries, %d low, %d high\n", | 
					
						
							|  |  |  |       f->hash_order, f->hash_size, f->entries_min, f->entries_max); | 
					
						
							| 
									
										
										
										
											1998-05-15 07:54:32 +00:00
										 |  |  |   f->hash_table = mb_alloc(f->fib_pool, f->hash_size * sizeof(struct fib_node *)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline void | 
					
						
							|  |  |  | fib_ht_free(struct fib_node **h) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   mb_free(h); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline unsigned | 
					
						
							|  |  |  | fib_hash(struct fib *f, ip_addr *a) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											1998-12-20 14:01:20 +00:00
										 |  |  |   return ipa_hash(*a) >> f->hash_shift; | 
					
						
							| 
									
										
										
										
											1998-05-15 07:54:32 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-12-22 19:20:43 +00:00
										 |  |  | static void | 
					
						
							|  |  |  | fib_dummy_init(struct fib_node *dummy) | 
					
						
							| 
									
										
										
										
											1998-12-22 18:55:49 +00:00
										 |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-05-15 07:54:32 +00:00
										 |  |  | void | 
					
						
							| 
									
										
										
										
											1998-12-20 14:01:20 +00:00
										 |  |  | fib_init(struct fib *f, pool *p, unsigned node_size, unsigned hash_order, void (*init)(struct fib_node *)) | 
					
						
							| 
									
										
										
										
											1998-05-15 07:54:32 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											1998-12-20 14:01:20 +00:00
										 |  |  |   if (!hash_order) | 
					
						
							|  |  |  |     hash_order = HASH_DEF_ORDER; | 
					
						
							| 
									
										
										
										
											1998-05-15 07:54:32 +00:00
										 |  |  |   f->fib_pool = p; | 
					
						
							|  |  |  |   f->fib_slab = sl_new(p, node_size); | 
					
						
							| 
									
										
										
										
											1998-12-20 14:01:20 +00:00
										 |  |  |   f->hash_order = hash_order; | 
					
						
							| 
									
										
										
										
											1998-05-15 07:54:32 +00:00
										 |  |  |   fib_ht_alloc(f); | 
					
						
							| 
									
										
										
										
											1998-12-20 14:01:20 +00:00
										 |  |  |   bzero(f->hash_table, f->hash_size * sizeof(struct fib_node *)); | 
					
						
							| 
									
										
										
										
											1998-05-15 07:54:32 +00:00
										 |  |  |   f->entries = 0; | 
					
						
							|  |  |  |   f->entries_min = 0; | 
					
						
							| 
									
										
										
										
											1998-12-22 18:55:49 +00:00
										 |  |  |   f->init = init ? : fib_dummy_init; | 
					
						
							| 
									
										
										
										
											1998-05-15 07:54:32 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							| 
									
										
										
										
											1998-12-20 14:01:20 +00:00
										 |  |  | fib_rehash(struct fib *f, int step) | 
					
						
							| 
									
										
										
										
											1998-05-15 07:54:32 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											1998-12-20 14:01:20 +00:00
										 |  |  |   unsigned old, new, oldn, newn, ni, nh; | 
					
						
							| 
									
										
										
										
											1998-05-15 07:54:32 +00:00
										 |  |  |   struct fib_node **n, *e, *x, **t, **m, **h; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-12-20 14:01:20 +00:00
										 |  |  |   old = f->hash_order; | 
					
						
							|  |  |  |   oldn = f->hash_size; | 
					
						
							|  |  |  |   new = old + step; | 
					
						
							| 
									
										
										
										
											1998-05-15 07:54:32 +00:00
										 |  |  |   m = h = f->hash_table; | 
					
						
							| 
									
										
										
										
											1998-12-20 14:01:20 +00:00
										 |  |  |   DBG("Re-hashing FIB from order %d to %d\n", old, new); | 
					
						
							|  |  |  |   f->hash_order = new; | 
					
						
							| 
									
										
										
										
											1998-05-15 07:54:32 +00:00
										 |  |  |   fib_ht_alloc(f); | 
					
						
							| 
									
										
										
										
											1998-12-20 14:01:20 +00:00
										 |  |  |   t = n = f->hash_table; | 
					
						
							|  |  |  |   newn = f->hash_size; | 
					
						
							|  |  |  |   ni = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-05-15 07:54:32 +00:00
										 |  |  |   while (old--) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       x = *h++; | 
					
						
							|  |  |  |       while (e = x) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  x = e->next; | 
					
						
							| 
									
										
										
										
											1998-12-20 14:01:20 +00:00
										 |  |  | 	  nh = fib_hash(f, &e->prefix); | 
					
						
							|  |  |  | 	  while (nh > ni) | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	      *t = NULL; | 
					
						
							|  |  |  | 	      ni++; | 
					
						
							|  |  |  | 	      t = ++n; | 
					
						
							|  |  |  | 	    } | 
					
						
							| 
									
										
										
										
											1998-05-15 07:54:32 +00:00
										 |  |  | 	  *t = e; | 
					
						
							| 
									
										
										
										
											1998-12-20 14:01:20 +00:00
										 |  |  | 	  t = &e->next; | 
					
						
							| 
									
										
										
										
											1998-05-15 07:54:32 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											1998-12-20 14:01:20 +00:00
										 |  |  |   while (ni < newn) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       *t = NULL; | 
					
						
							|  |  |  |       ni++; | 
					
						
							|  |  |  |       t = ++n; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											1998-05-15 07:54:32 +00:00
										 |  |  |   fib_ht_free(m); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void * | 
					
						
							|  |  |  | fib_find(struct fib *f, ip_addr *a, int len) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   struct fib_node *e = f->hash_table[fib_hash(f, a)]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   while (e && (e->pxlen != len || !ipa_equal(*a, e->prefix))) | 
					
						
							|  |  |  |     e = e->next; | 
					
						
							|  |  |  |   return e; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void * | 
					
						
							|  |  |  | fib_get(struct fib *f, ip_addr *a, int len) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											1998-12-20 14:01:20 +00:00
										 |  |  |   unsigned int h = ipa_hash(*a); | 
					
						
							|  |  |  |   struct fib_node **ee = f->hash_table + (h >> f->hash_shift); | 
					
						
							|  |  |  |   struct fib_node *g, *e = *ee; | 
					
						
							| 
									
										
										
										
											1998-05-15 07:54:32 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   while (e && (e->pxlen != len || !ipa_equal(*a, e->prefix))) | 
					
						
							|  |  |  |     e = e->next; | 
					
						
							|  |  |  |   if (e) | 
					
						
							|  |  |  |     return e; | 
					
						
							| 
									
										
										
										
											1998-11-29 14:38:34 +00:00
										 |  |  | #ifdef DEBUGGING
 | 
					
						
							| 
									
										
										
										
											1998-05-15 07:54:32 +00:00
										 |  |  |   if (len < 0 || len > BITS_PER_IP_ADDRESS || !ip_is_prefix(*a,len)) | 
					
						
							| 
									
										
										
										
											1998-12-20 14:27:37 +00:00
										 |  |  |     bug("fib_get() called for invalid address"); | 
					
						
							| 
									
										
										
										
											1998-05-15 07:54:32 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  |   e = sl_alloc(f->fib_slab); | 
					
						
							|  |  |  |   e->prefix = *a; | 
					
						
							|  |  |  |   e->pxlen = len; | 
					
						
							| 
									
										
										
										
											1998-12-20 14:01:20 +00:00
										 |  |  |   while ((g = *ee) && ipa_hash(g->prefix) < h) | 
					
						
							|  |  |  |     ee = &g->next; | 
					
						
							| 
									
										
										
										
											1998-05-15 07:54:32 +00:00
										 |  |  |   e->next = *ee; | 
					
						
							|  |  |  |   *ee = e; | 
					
						
							| 
									
										
										
										
											1998-12-20 14:01:20 +00:00
										 |  |  |   e->readers = NULL; | 
					
						
							| 
									
										
										
										
											1998-05-15 07:54:32 +00:00
										 |  |  |   f->init(e); | 
					
						
							|  |  |  |   if (f->entries++ > f->entries_max) | 
					
						
							| 
									
										
										
										
											1998-12-20 14:01:20 +00:00
										 |  |  |     fib_rehash(f, HASH_HI_STEP); | 
					
						
							| 
									
										
										
										
											1998-05-15 07:54:32 +00:00
										 |  |  |   return e; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-12-20 14:01:20 +00:00
										 |  |  | static inline void | 
					
						
							|  |  |  | fib_merge_readers(struct fib_iterator *i, struct fib_node *to) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   if (to) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       struct fib_iterator *j = to->readers; | 
					
						
							|  |  |  |       if (!j) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  /* Fast path */ | 
					
						
							|  |  |  | 	  to->readers = i; | 
					
						
							|  |  |  | 	  i->prev = (struct fib_iterator *) to; | 
					
						
							|  |  |  | 	fixup: | 
					
						
							|  |  |  | 	  while (i && i->node) | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	      i->node = NULL; | 
					
						
							|  |  |  | 	      i = i->next; | 
					
						
							|  |  |  | 	    } | 
					
						
							|  |  |  | 	  return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       /* Really merging */ | 
					
						
							|  |  |  |       while (j->next) | 
					
						
							|  |  |  | 	j = j->next; | 
					
						
							|  |  |  |       j->next = i; | 
					
						
							|  |  |  |       i->prev = j; | 
					
						
							|  |  |  |       goto fixup; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   else					/* No more nodes */ | 
					
						
							|  |  |  |     while (i) | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	i->prev = NULL; | 
					
						
							|  |  |  | 	i = i->next; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-05-15 07:54:32 +00:00
										 |  |  | void | 
					
						
							|  |  |  | fib_delete(struct fib *f, void *E) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   struct fib_node *e = E; | 
					
						
							| 
									
										
										
										
											1998-12-20 14:01:20 +00:00
										 |  |  |   unsigned int h = fib_hash(f, &e->prefix); | 
					
						
							|  |  |  |   struct fib_node **ee = f->hash_table + h; | 
					
						
							|  |  |  |   struct fib_iterator *it; | 
					
						
							| 
									
										
										
										
											1998-05-15 07:54:32 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   while (*ee) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       if (*ee == e) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  *ee = e->next; | 
					
						
							| 
									
										
										
										
											1998-12-20 14:01:20 +00:00
										 |  |  | 	  if (it = e->readers) | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	      struct fib_node *l = e->next; | 
					
						
							|  |  |  | 	      while (!l) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 		  h++; | 
					
						
							|  |  |  | 		  if (h >= f->hash_size) | 
					
						
							|  |  |  | 		    break; | 
					
						
							|  |  |  | 		  else | 
					
						
							|  |  |  | 		    l = f->hash_table[h]; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	      fib_merge_readers(it, l); | 
					
						
							|  |  |  | 	    } | 
					
						
							| 
									
										
										
										
											1998-05-15 07:54:32 +00:00
										 |  |  | 	  sl_free(f->fib_slab, e); | 
					
						
							|  |  |  | 	  if (f->entries-- < f->entries_min) | 
					
						
							| 
									
										
										
										
											1998-12-20 14:01:20 +00:00
										 |  |  | 	    fib_rehash(f, -HASH_LO_STEP); | 
					
						
							| 
									
										
										
										
											1998-05-15 07:54:32 +00:00
										 |  |  | 	  return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       ee = &((*ee)->next); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											1998-12-20 14:27:37 +00:00
										 |  |  |   bug("fib_delete() called for invalid node"); | 
					
						
							| 
									
										
										
										
											1998-05-15 07:54:32 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void | 
					
						
							|  |  |  | fib_free(struct fib *f) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   fib_ht_free(f->hash_table); | 
					
						
							|  |  |  |   rfree(f->fib_slab); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											1998-12-20 14:01:20 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | void | 
					
						
							|  |  |  | fit_init(struct fib_iterator *i, struct fib *f) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   unsigned h; | 
					
						
							|  |  |  |   struct fib_node *n; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   i->efef = 0xff; | 
					
						
							|  |  |  |   for(h=0; h<f->hash_size; h++) | 
					
						
							|  |  |  |     if (n = f->hash_table[h]) | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	i->prev = (struct fib_iterator *) n; | 
					
						
							|  |  |  | 	if (i->next = n->readers) | 
					
						
							|  |  |  | 	  i->next->prev = i; | 
					
						
							|  |  |  | 	n->readers = i; | 
					
						
							|  |  |  | 	i->node = n; | 
					
						
							|  |  |  | 	return; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |   /* The fib is empty, nothing to do */ | 
					
						
							|  |  |  |   i->prev = i->next = NULL; | 
					
						
							|  |  |  |   i->node = NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct fib_node * | 
					
						
							|  |  |  | fit_get(struct fib *f, struct fib_iterator *i) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   struct fib_node *n; | 
					
						
							|  |  |  |   struct fib_iterator *j, *k; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (!i->prev) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       /* We are at the end */ | 
					
						
							|  |  |  |       i->hash = f->hash_size; | 
					
						
							|  |  |  |       return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   if (!(n = i->node)) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       /* No node info available, we are a victim of merging. Try harder. */ | 
					
						
							|  |  |  |       j = i; | 
					
						
							|  |  |  |       while (j->efef == 0xff) | 
					
						
							|  |  |  | 	j = j->prev; | 
					
						
							|  |  |  |       n = (struct fib_node *) j; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   j = i->prev; | 
					
						
							|  |  |  |   if (k = i->next) | 
					
						
							|  |  |  |     k->prev = j; | 
					
						
							|  |  |  |   j->next = k; | 
					
						
							|  |  |  |   i->hash = fib_hash(f, &n->prefix); | 
					
						
							|  |  |  |   return n; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void | 
					
						
							|  |  |  | fit_put(struct fib_iterator *i, struct fib_node *n) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   struct fib_iterator *j; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   i->node = n; | 
					
						
							|  |  |  |   if (j = n->readers) | 
					
						
							|  |  |  |     j->prev = i; | 
					
						
							|  |  |  |   i->next = j; | 
					
						
							|  |  |  |   n->readers = i; | 
					
						
							|  |  |  |   i->prev = (struct fib_iterator *) n; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef DEBUGGING
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void | 
					
						
							|  |  |  | fib_check(struct fib *f) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   unsigned int i, ec, lo, nulls; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   ec = 0; | 
					
						
							|  |  |  |   for(i=0; i<f->hash_size; i++) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       struct fib_node *n; | 
					
						
							|  |  |  |       lo = 0; | 
					
						
							|  |  |  |       for(n=f->hash_table[i]; n; n=n->next) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  struct fib_iterator *j, *j0; | 
					
						
							|  |  |  | 	  unsigned int h0 = ipa_hash(n->prefix); | 
					
						
							|  |  |  | 	  if (h0 < lo) | 
					
						
							| 
									
										
										
										
											1998-12-20 14:27:37 +00:00
										 |  |  | 	    bug("fib_check: discord in hash chains"); | 
					
						
							| 
									
										
										
										
											1998-12-20 14:01:20 +00:00
										 |  |  | 	  lo = h0; | 
					
						
							|  |  |  | 	  if ((h0 >> f->hash_shift) != i) | 
					
						
							| 
									
										
										
										
											1998-12-20 14:27:37 +00:00
										 |  |  | 	    bug("fib_check: mishashed %x->%x (order %d)", h0, i, f->hash_order); | 
					
						
							| 
									
										
										
										
											1998-12-20 14:01:20 +00:00
										 |  |  | 	  j0 = (struct fib_iterator *) n; | 
					
						
							|  |  |  | 	  nulls = 0; | 
					
						
							|  |  |  | 	  for(j=n->readers; j; j=j->next) | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	      if (j->prev != j0) | 
					
						
							| 
									
										
										
										
											1998-12-20 14:27:37 +00:00
										 |  |  | 		bug("fib_check: iterator->prev mismatch"); | 
					
						
							| 
									
										
										
										
											1998-12-20 14:01:20 +00:00
										 |  |  | 	      j0 = j; | 
					
						
							|  |  |  | 	      if (!j->node) | 
					
						
							|  |  |  | 		nulls++; | 
					
						
							|  |  |  | 	      else if (nulls) | 
					
						
							| 
									
										
										
										
											1998-12-20 14:27:37 +00:00
										 |  |  | 		bug("fib_check: iterator nullified"); | 
					
						
							| 
									
										
										
										
											1998-12-20 14:01:20 +00:00
										 |  |  | 	      else if (j->node != n) | 
					
						
							| 
									
										
										
										
											1998-12-20 14:27:37 +00:00
										 |  |  | 		bug("fib_check: iterator->node mismatch"); | 
					
						
							| 
									
										
										
										
											1998-12-20 14:01:20 +00:00
										 |  |  | 	    } | 
					
						
							|  |  |  | 	  ec++; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   if (ec != f->entries) | 
					
						
							| 
									
										
										
										
											1998-12-20 14:27:37 +00:00
										 |  |  |     bug("fib_check: invalid entry count (%d != %d)", ec, f->entries); | 
					
						
							| 
									
										
										
										
											1998-12-20 14:01:20 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef TEST
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "lib/resource.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct fib f; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void dump(char *m) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   unsigned int i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   debug("%s ... order=%d, size=%d, entries=%d\n", m, f.hash_order, f.hash_size, f.hash_size); | 
					
						
							|  |  |  |   for(i=0; i<f.hash_size; i++) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       struct fib_node *n; | 
					
						
							|  |  |  |       struct fib_iterator *j; | 
					
						
							|  |  |  |       for(n=f.hash_table[i]; n; n=n->next) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  debug("%04x %04x %p %I/%2d", i, ipa_hash(n->prefix), n, n->prefix, n->pxlen); | 
					
						
							|  |  |  | 	  for(j=n->readers; j; j=j->next) | 
					
						
							|  |  |  | 	    debug(" %p[%p]", j, j->node); | 
					
						
							|  |  |  | 	  debug("\n"); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   fib_check(&f); | 
					
						
							|  |  |  |   debug("-----\n"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void init(struct fib_node *n) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int main(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   struct fib_node *n; | 
					
						
							|  |  |  |   struct fib_iterator i, j; | 
					
						
							|  |  |  |   ip_addr a; | 
					
						
							|  |  |  |   int c; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   log_init_debug(NULL); | 
					
						
							|  |  |  |   resource_init(); | 
					
						
							|  |  |  |   fib_init(&f, &root_pool, sizeof(struct fib_node), 4, init); | 
					
						
							|  |  |  |   dump("init"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   a = ipa_from_u32(0x01020304); n = fib_get(&f, &a, 32); | 
					
						
							|  |  |  |   a = ipa_from_u32(0x02030405); n = fib_get(&f, &a, 32); | 
					
						
							|  |  |  |   a = ipa_from_u32(0x03040506); n = fib_get(&f, &a, 32); | 
					
						
							|  |  |  |   a = ipa_from_u32(0x00000000); n = fib_get(&f, &a, 32); | 
					
						
							|  |  |  |   a = ipa_from_u32(0x00000c01); n = fib_get(&f, &a, 32); | 
					
						
							|  |  |  |   a = ipa_from_u32(0xffffffff); n = fib_get(&f, &a, 32); | 
					
						
							|  |  |  |   dump("fill"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   fit_init(&i, &f); | 
					
						
							|  |  |  |   dump("iter init"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   fib_rehash(&f, 1); | 
					
						
							|  |  |  |   dump("rehash up"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   fib_rehash(&f, -1); | 
					
						
							|  |  |  |   dump("rehash down"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | next: | 
					
						
							|  |  |  |   c = 0; | 
					
						
							|  |  |  |   FIB_ITERATE_START(&f, &i, z) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       if (c) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  FIB_ITERATE_PUT(&i, z); | 
					
						
							|  |  |  | 	  dump("iter"); | 
					
						
							|  |  |  | 	  goto next; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       c = 1; | 
					
						
							|  |  |  |       debug("got %p\n", z); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   FIB_ITERATE_END; | 
					
						
							|  |  |  |   dump("iter end"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   fit_init(&i, &f); | 
					
						
							|  |  |  |   fit_init(&j, &f); | 
					
						
							|  |  |  |   dump("iter init 2"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   n = fit_get(&f, &i); | 
					
						
							|  |  |  |   dump("iter step 2"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   fit_put(&i, n->next); | 
					
						
							|  |  |  |   dump("iter step 3"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   a = ipa_from_u32(0xffffffff); n = fib_get(&f, &a, 32); | 
					
						
							|  |  |  |   fib_delete(&f, n); | 
					
						
							|  |  |  |   dump("iter step 3"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #endif
 |