⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 radix.c

📁 早期freebsd实现
💻 C
📖 第 1 页 / 共 2 页
字号:
rn_addroute(v_arg, n_arg, head, treenodes)	void *v_arg, *n_arg;	struct radix_node_head *head;	struct radix_node treenodes[2];{	caddr_t v = (caddr_t)v_arg, netmask = (caddr_t)n_arg;	register struct radix_node *t, *x, *tt;	struct radix_node *saved_tt, *top = head->rnh_treetop;	short b = 0, b_leaf;	int mlen, keyduplicated;	caddr_t cplim;	struct radix_mask *m, **mp;	/*	 * In dealing with non-contiguous masks, there may be	 * many different routes which have the same mask.	 * We will find it useful to have a unique pointer to	 * the mask to speed avoiding duplicate references at	 * nodes and possibly save time in calculating indices.	 */	if (netmask)  {		x = rn_search(netmask, rn_masktop);		mlen = *(u_char *)netmask;		if (Bcmp(netmask, x->rn_key, mlen) != 0) {			x = rn_addmask(netmask, 0, top->rn_off);			if (x == 0)				return (0);		}		netmask = x->rn_key;		b = -1 - x->rn_b;	}	/*	 * Deal with duplicated keys: attach node to previous instance	 */	saved_tt = tt = rn_insert(v, head, &keyduplicated, treenodes);	if (keyduplicated) {		do {			if (tt->rn_mask == netmask)				return (0);			t = tt;			if (netmask == 0 ||			    (tt->rn_mask && rn_refines(netmask, tt->rn_mask)))				break;		} while (tt = tt->rn_dupedkey);		/*		 * If the mask is not duplicated, we wouldn't		 * find it among possible duplicate key entries		 * anyway, so the above test doesn't hurt.		 *		 * We sort the masks for a duplicated key the same way as		 * in a masklist -- most specific to least specific.		 * This may require the unfortunate nuisance of relocating		 * the head of the list.		 */		if (tt && t == saved_tt) {			struct	radix_node *xx = x;			/* link in at head of list */			(tt = treenodes)->rn_dupedkey = t;			tt->rn_flags = t->rn_flags;			tt->rn_p = x = t->rn_p;			if (x->rn_l == t) x->rn_l = tt; else x->rn_r = tt;			saved_tt = tt; x = xx;		} else {			(tt = treenodes)->rn_dupedkey = t->rn_dupedkey;			t->rn_dupedkey = tt;		}#ifdef RN_DEBUG		t=tt+1; tt->rn_info = rn_nodenum++; t->rn_info = rn_nodenum++;		tt->rn_twin = t; tt->rn_ybro = rn_clist; rn_clist = tt;#endif		t = saved_tt;		tt->rn_key = (caddr_t) v;		tt->rn_b = -1;		tt->rn_flags = t->rn_flags & ~RNF_ROOT;	}	/*	 * Put mask in tree.	 */	if (netmask) {		tt->rn_mask = netmask;		tt->rn_b = x->rn_b;	}	t = saved_tt->rn_p;	b_leaf = -1 - t->rn_b;	if (t->rn_r == saved_tt) x = t->rn_l; else x = t->rn_r;	/* Promote general routes from below */	if (x->rn_b < 0) { 		if (x->rn_mask && (x->rn_b >= b_leaf) && x->rn_mklist == 0) {			MKGet(m);			if (m) {				Bzero(m, sizeof *m);				m->rm_b = x->rn_b;				m->rm_mask = x->rn_mask;				x->rn_mklist = t->rn_mklist = m;			}		}	} else if (x->rn_mklist) {		/*		 * Skip over masks whose index is > that of new node		 */		for (mp = &x->rn_mklist; m = *mp; mp = &m->rm_mklist)			if (m->rm_b >= b_leaf)				break;		t->rn_mklist = m; *mp = 0;	}	/* Add new route to highest possible ancestor's list */	if ((netmask == 0) || (b > t->rn_b ))		return tt; /* can't lift at all */	b_leaf = tt->rn_b;	do {		x = t;		t = t->rn_p;	} while (b <= t->rn_b && x != top);	/*	 * Search through routes associated with node to	 * insert new route according to index.	 * For nodes of equal index, place more specific	 * masks first.	 */	cplim = netmask + mlen;	for (mp = &x->rn_mklist; m = *mp; mp = &m->rm_mklist) {		if (m->rm_b < b_leaf)			continue;		if (m->rm_b > b_leaf)			break;		if (m->rm_mask == netmask) {			m->rm_refs++;			tt->rn_mklist = m;			return tt;		}		if (rn_refines(netmask, m->rm_mask))			break;	}	MKGet(m);	if (m == 0) {		printf("Mask for route not entered\n");		return (tt);	}	Bzero(m, sizeof *m);	m->rm_b = b_leaf;	m->rm_mask = netmask;	m->rm_mklist = *mp;	*mp = m;	tt->rn_mklist = m;	return tt;}struct radix_node *rn_delete(v_arg, netmask_arg, head)	void *v_arg, *netmask_arg;	struct radix_node_head *head;{	register struct radix_node *t, *p, *x, *tt;	struct radix_mask *m, *saved_m, **mp;	struct radix_node *dupedkey, *saved_tt, *top;	caddr_t v, netmask;	int b, head_off, vlen;	v = v_arg;	netmask = netmask_arg;	x = head->rnh_treetop;	tt = rn_search(v, x);	head_off = x->rn_off;	vlen =  *(u_char *)v;	saved_tt = tt;	top = x;	if (tt == 0 ||	    Bcmp(v + head_off, tt->rn_key + head_off, vlen - head_off))		return (0);	/*	 * Delete our route from mask lists.	 */	if (dupedkey = tt->rn_dupedkey) {		if (netmask) 			netmask = rn_search(netmask, rn_masktop)->rn_key;		while (tt->rn_mask != netmask)			if ((tt = tt->rn_dupedkey) == 0)				return (0);	}	if (tt->rn_mask == 0 || (saved_m = m = tt->rn_mklist) == 0)		goto on1;	if (m->rm_mask != tt->rn_mask) {		printf("rn_delete: inconsistent annotation\n");		goto on1;	}	if (--m->rm_refs >= 0)		goto on1;	b = -1 - tt->rn_b;	t = saved_tt->rn_p;	if (b > t->rn_b)		goto on1; /* Wasn't lifted at all */	do {		x = t;		t = t->rn_p;	} while (b <= t->rn_b && x != top);	for (mp = &x->rn_mklist; m = *mp; mp = &m->rm_mklist)		if (m == saved_m) {			*mp = m->rm_mklist;			MKFree(m);			break;		}	if (m == 0)		printf("rn_delete: couldn't find our annotation\n");on1:	/*	 * Eliminate us from tree	 */	if (tt->rn_flags & RNF_ROOT)		return (0);#ifdef RN_DEBUG	/* Get us out of the creation list */	for (t = rn_clist; t && t->rn_ybro != tt; t = t->rn_ybro) {}	if (t) t->rn_ybro = tt->rn_ybro;#endif	t = tt->rn_p;	if (dupedkey) {		if (tt == saved_tt) {			x = dupedkey; x->rn_p = t;			if (t->rn_l == tt) t->rn_l = x; else t->rn_r = x;		} else {			for (x = p = saved_tt; p && p->rn_dupedkey != tt;)				p = p->rn_dupedkey;			if (p) p->rn_dupedkey = tt->rn_dupedkey;			else printf("rn_delete: couldn't find us\n");		}		t = tt + 1;		if  (t->rn_flags & RNF_ACTIVE) {#ifndef RN_DEBUG			*++x = *t; p = t->rn_p;#else			b = t->rn_info; *++x = *t; t->rn_info = b; p = t->rn_p;#endif			if (p->rn_l == t) p->rn_l = x; else p->rn_r = x;			x->rn_l->rn_p = x; x->rn_r->rn_p = x;		}		goto out;	}	if (t->rn_l == tt) x = t->rn_r; else x = t->rn_l;	p = t->rn_p;	if (p->rn_r == t) p->rn_r = x; else p->rn_l = x;	x->rn_p = p;	/*	 * Demote routes attached to us.	 */	if (t->rn_mklist) {		if (x->rn_b >= 0) {			for (mp = &x->rn_mklist; m = *mp;)				mp = &m->rm_mklist;			*mp = t->rn_mklist;		} else {			for (m = t->rn_mklist; m;) {				struct radix_mask *mm = m->rm_mklist;				if (m == x->rn_mklist && (--(m->rm_refs) < 0)) {					x->rn_mklist = 0;					MKFree(m);				} else 					printf("%s %x at %x\n",					    "rn_delete: Orphaned Mask", m, x);				m = mm;			}		}	}	/*	 * We may be holding an active internal node in the tree.	 */	x = tt + 1;	if (t != x) {#ifndef RN_DEBUG		*t = *x;#else		b = t->rn_info; *t = *x; t->rn_info = b;#endif		t->rn_l->rn_p = t; t->rn_r->rn_p = t;		p = x->rn_p;		if (p->rn_l == x) p->rn_l = t; else p->rn_r = t;	}out:	tt->rn_flags &= ~RNF_ACTIVE;	tt[1].rn_flags &= ~RNF_ACTIVE;	return (tt);}intrn_walktree(h, f, w)	struct radix_node_head *h;	register int (*f)();	void *w;{	int error;	struct radix_node *base, *next;	register struct radix_node *rn = h->rnh_treetop;	/*	 * This gets complicated because we may delete the node	 * while applying the function f to it, so we need to calculate	 * the successor node in advance.	 */	/* First time through node, go left */	while (rn->rn_b >= 0)		rn = rn->rn_l;	for (;;) {		base = rn;		/* If at right child go back up, otherwise, go right */		while (rn->rn_p->rn_r == rn && (rn->rn_flags & RNF_ROOT) == 0)			rn = rn->rn_p;		/* Find the next *leaf* since next node might vanish, too */		for (rn = rn->rn_p->rn_r; rn->rn_b >= 0;)			rn = rn->rn_l;		next = rn;		/* Process leaves */		while (rn = base) {			base = rn->rn_dupedkey;			if (!(rn->rn_flags & RNF_ROOT) && (error = (*f)(rn, w)))				return (error);		}		rn = next;		if (rn->rn_flags & RNF_ROOT)			return (0);	}	/* NOTREACHED */}intrn_inithead(head, off)	void **head;	int off;{	register struct radix_node_head *rnh;	register struct radix_node *t, *tt, *ttt;	if (*head)		return (1);	R_Malloc(rnh, struct radix_node_head *, sizeof (*rnh));	if (rnh == 0)		return (0);	Bzero(rnh, sizeof (*rnh));	*head = rnh;	t = rn_newpair(rn_zeros, off, rnh->rnh_nodes);	ttt = rnh->rnh_nodes + 2;	t->rn_r = ttt;	t->rn_p = t;	tt = t->rn_l;	tt->rn_flags = t->rn_flags = RNF_ROOT | RNF_ACTIVE;	tt->rn_b = -1 - off;	*ttt = *tt;	ttt->rn_key = rn_ones;	rnh->rnh_addaddr = rn_addroute;	rnh->rnh_deladdr = rn_delete;	rnh->rnh_matchaddr = rn_match;	rnh->rnh_walktree = rn_walktree;	rnh->rnh_treetop = t;	return (1);}voidrn_init(){	char *cp, *cplim;#ifdef KERNEL	struct domain *dom;	for (dom = domains; dom; dom = dom->dom_next)		if (dom->dom_maxrtkey > max_keylen)			max_keylen = dom->dom_maxrtkey;#endif	if (max_keylen == 0) {		printf("rn_init: radix functions require max_keylen be set\n");		return;	}	R_Malloc(rn_zeros, char *, 3 * max_keylen);	if (rn_zeros == NULL)		panic("rn_init");	Bzero(rn_zeros, 3 * max_keylen);	rn_ones = cp = rn_zeros + max_keylen;	maskedKey = cplim = rn_ones + max_keylen;	while (cp < cplim)		*cp++ = -1;	if (rn_inithead((void **)&mask_rnhead, 0) == 0)		panic("rn_init 2");}

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -