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

📄 dn_neigh.c

📁 Linux内核源代码 为压缩文件 是<<Linux内核>>一书中的源代码
💻 C
📖 第 1 页 / 共 2 页
字号:
	sp->msgflg     = DN_RT_PKT_SHORT|(cb->rt_flags&(DN_RT_F_RQR|DN_RT_F_RTS));	sp->dstnode    = cb->dst;	sp->srcnode    = cb->src;	sp->forward    = cb->hops & 0x3f;	skb->nh.raw = skb->data;	return NF_HOOK(PF_DECnet, NF_DN_POST_ROUTING, skb, NULL, neigh->dev, dn_neigh_output_packet);}/* * Phase 3 output is the same is short output, execpt that * it clears the area bits before transmission. */static int dn_phase3_output(struct sk_buff *skb){	struct dst_entry *dst = skb->dst;	struct neighbour *neigh = dst->neighbour;	struct net_device *dev = neigh->dev;	int headroom = dev->hard_header_len + sizeof(struct dn_short_packet) + 2;	struct dn_short_packet *sp;	unsigned char *data;	struct dn_skb_cb *cb = (struct dn_skb_cb *)skb->cb;	if (skb_headroom(skb) < headroom) {		struct sk_buff *skb2 = skb_realloc_headroom(skb, headroom);		if (skb2 == NULL) {			if (net_ratelimit())				printk(KERN_CRIT "dn_phase3_output: no memory\n");			kfree_skb(skb);			return -ENOBUFS;		}		kfree_skb(skb);		skb = skb2;		if (net_ratelimit())			printk(KERN_INFO "dn_phase3_output: Increasing headroom\n");	}	data = skb_push(skb, sizeof(struct dn_short_packet) + 2);	((unsigned short *)data) = dn_htons(skb->len - 2);	sp = (struct dn_short_packet *)(data + 2);	sp->msgflg   = DN_RT_PKT_SHORT|(cb->rt_flags&(DN_RT_F_RQR|DN_RT_F_RTS));	sp->dstnode  = cb->dst & dn_htons(0x03ff);	sp->srcnode  = cb->src & dn_htons(0x03ff);	sp->forward  = cb->hops & 0x3f;	skb->nh.raw = skb->data;	return NF_HOOK(PF_DECnet, NF_DN_POST_ROUTING, skb, NULL, neigh->dev, dn_neigh_output_packet);}/* * Unfortunately, the neighbour code uses the device in its hash * function, so we don't get any advantage from it. This function * basically does a neigh_lookup(), but without comparing the device * field. This is required for the On-Ethernet cache */struct neighbour *dn_neigh_lookup(struct neigh_table *tbl, void *ptr){	struct neighbour *neigh;	u32 hash_val;	hash_val = tbl->hash(ptr, NULL);	read_lock_bh(&tbl->lock);	for(neigh = tbl->hash_buckets[hash_val]; neigh != NULL; neigh = neigh->next) {		if (memcmp(neigh->primary_key, ptr, tbl->key_len) == 0) {			atomic_inc(&neigh->refcnt);			read_unlock_bh(&tbl->lock);			return neigh;		}	}	read_unlock_bh(&tbl->lock);	return NULL;}/* * Any traffic on a pointopoint link causes the timer to be reset * for the entry in the neighbour table. */void dn_neigh_pointopoint_notify(struct sk_buff *skb){	return;}/* * Pointopoint link receives a hello message */void dn_neigh_pointopoint_hello(struct sk_buff *skb){	kfree_skb(skb);}/* * Ethernet router hello message received */int dn_neigh_router_hello(struct sk_buff *skb){	struct rtnode_hello_message *msg = (struct rtnode_hello_message *)skb->data;	struct neighbour *neigh;	struct dn_neigh *dn;	struct dn_dev *dn_db;	dn_address src;	src = dn_htons(dn_eth2dn(msg->id));	neigh = __neigh_lookup(&dn_neigh_table, &src, skb->dev, 1);	dn = (struct dn_neigh *)neigh;	if (neigh) {		write_lock(&neigh->lock);		neigh->used = jiffies;		dn_db = (struct dn_dev *)neigh->dev->dn_ptr;		if (!(neigh->nud_state & NUD_PERMANENT)) {			neigh->updated = jiffies;			if (neigh->dev->type == ARPHRD_ETHER)				memcpy(neigh->ha, &skb->mac.ethernet->h_source, ETH_ALEN);			dn->blksize  = dn_ntohs(msg->blksize);			dn->priority = msg->priority;			dn->flags &= ~DN_NDFLAG_P3;			switch(msg->iinfo & DN_RT_INFO_TYPE) {				case DN_RT_INFO_L1RT:					dn->flags &=~DN_NDFLAG_R2;					dn->flags |= DN_NDFLAG_R1;					break;				case DN_RT_INFO_L2RT:					dn->flags |= DN_NDFLAG_R2;			}		}		if (!dn_db->router) {			dn_db->router = neigh_clone(neigh);		} else {			if (msg->priority > ((struct dn_neigh *)dn_db->router)->priority)				neigh_release(xchg(&dn_db->router, neigh_clone(neigh)));		}		write_unlock(&neigh->lock);		neigh_release(neigh);	}	kfree_skb(skb);	return 0;}/* * Endnode hello message received */int dn_neigh_endnode_hello(struct sk_buff *skb){	struct endnode_hello_message *msg = (struct endnode_hello_message *)skb->data;	struct neighbour *neigh;	struct dn_neigh *dn;	dn_address src;	src = dn_htons(dn_eth2dn(msg->id));	neigh = __neigh_lookup(&dn_neigh_table, &src, skb->dev, 1);	dn = (struct dn_neigh *)neigh;	if (neigh) {		write_lock(&neigh->lock);		neigh->used = jiffies;		if (!(neigh->nud_state & NUD_PERMANENT)) {			neigh->updated = jiffies;			if (neigh->dev->type == ARPHRD_ETHER)				memcpy(neigh->ha, &skb->mac.ethernet->h_source, ETH_ALEN);			dn->flags   &= ~(DN_NDFLAG_R1 | DN_NDFLAG_R2);			dn->blksize  = dn_ntohs(msg->blksize);			dn->priority = 0;		}		write_unlock(&neigh->lock);		neigh_release(neigh);	}	kfree_skb(skb);	return 0;}#ifdef CONFIG_DECNET_ROUTERstatic char *dn_find_slot(char *base, int max, int priority){	int i;	unsigned char *min = NULL;	base += 6; /* skip first id */	for(i = 0; i < max; i++) {		if (!min || (*base < *min))			min = base;		base += 7; /* find next priority */	}	if (!min)		return NULL;	return (*min < priority) ? (min - 6) : NULL;}int dn_neigh_elist(struct net_device *dev, unsigned char *ptr, int n){	int t = 0;	int i;	struct neighbour *neigh;	struct dn_neigh *dn;	struct neigh_table *tbl = &dn_neigh_table;	unsigned char *rs = ptr;	struct dn_dev *dn_db = (struct dn_dev *)dev->dn_ptr;	read_lock_bh(&tbl->lock);	for(i = 0; i < NEIGH_HASHMASK; i++) {		for(neigh = tbl->hash_buckets[i]; neigh != NULL; neigh = neigh->next) {			if (neigh->dev != dev)				continue;			dn = (struct dn_neigh *)neigh;			if (!(dn->flags & (DN_NDFLAG_R1|DN_NDFLAG_R2)))				continue;			if (dn_db->parms.forwarding == 1 && (dn->flags & DN_NDFLAG_R2))				continue;			if (t == n)				rs = dn_find_slot(ptr, n, dn->priority);			else				t++;			if (rs == NULL)				continue;			dn_dn2eth(rs, dn->addr);			rs += 6;			*rs = neigh->nud_state & NUD_CONNECTED ? 0x80 : 0x0;			*rs |= dn->priority;			rs++;		}	}	read_unlock_bh(&tbl->lock);	return t;}#endif /* CONFIG_DECNET_ROUTER */#ifdef CONFIG_PROC_FSstatic int dn_neigh_get_info(char *buffer, char **start, off_t offset, int length){        int len     = 0;        off_t pos   = 0;        off_t begin = 0;	struct neighbour *n;	int i;	char buf[DN_ASCBUF_LEN];	len += sprintf(buffer + len, "Addr    Flags State Use Blksize Dev\n");	for(i=0;i <= NEIGH_HASHMASK; i++) {		read_lock_bh(&dn_neigh_table.lock);		n = dn_neigh_table.hash_buckets[i];		for(; n != NULL; n = n->next) {			struct dn_neigh *dn = (struct dn_neigh *)n;			read_lock(&n->lock);			len += sprintf(buffer+len, "%-7s %s%s%s   %02x    %02d  %07ld %-8s\n",					dn_addr2asc(dn_ntohs(dn->addr), buf),					(dn->flags&DN_NDFLAG_R1) ? "1" : "-",					(dn->flags&DN_NDFLAG_R2) ? "2" : "-",					(dn->flags&DN_NDFLAG_P3) ? "3" : "-",					dn->n.nud_state,					atomic_read(&dn->n.refcnt),					dn->blksize,					(dn->n.dev) ? dn->n.dev->name : "?");			read_unlock(&n->lock);			pos = begin + len;                	if (pos < offset) {                        	len = 0;                        	begin = pos;                	}                	if (pos > offset + length) {				read_unlock_bh(&dn_neigh_table.lock);                       		goto done;			}		}		read_unlock_bh(&dn_neigh_table.lock);	}done:        *start = buffer + (offset - begin);        len   -= offset - begin;        if (len > length) len = length;        return len;}#endifvoid __init dn_neigh_init(void){	neigh_table_init(&dn_neigh_table);#ifdef CONFIG_PROC_FS	proc_net_create("decnet_neigh",0,dn_neigh_get_info);#endif /* CONFIG_PROC_FS */}void __exit dn_neigh_cleanup(void){#ifdef CONFIG_PROC_FS	proc_net_remove("decnet_neigh");#endif /* CONFIG_PROC_FS */	neigh_table_clear(&dn_neigh_table);}

⌨️ 快捷键说明

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