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

📄 act_api.c

📁 Linux Kernel 2.6.9 for OMAP1710
💻 C
📖 第 1 页 / 共 2 页
字号:
/* * net/sched/act_api.c	Packet action API. * *		This program is free software; you can redistribute it and/or *		modify it under the terms of the GNU General Public License *		as published by the Free Software Foundation; either version *		2 of the License, or (at your option) any later version. * * Author:	Jamal Hadi Salim * * */#include <asm/uaccess.h>#include <asm/system.h>#include <asm/bitops.h>#include <linux/config.h>#include <linux/types.h>#include <linux/kernel.h>#include <linux/sched.h>#include <linux/string.h>#include <linux/mm.h>#include <linux/socket.h>#include <linux/sockios.h>#include <linux/in.h>#include <linux/errno.h>#include <linux/interrupt.h>#include <linux/netdevice.h>#include <linux/skbuff.h>#include <linux/rtnetlink.h>#include <linux/init.h>#include <linux/kmod.h>#include <net/sock.h>#include <net/pkt_sched.h>#if 1 /* control */#define DPRINTK(format,args...) printk(KERN_DEBUG format,##args)#else#define DPRINTK(format,args...)#endif#if 0 /* data */#define D2PRINTK(format,args...) printk(KERN_DEBUG format,##args)#else#define D2PRINTK(format,args...)#endifstatic struct tc_action_ops *act_base = NULL;static rwlock_t act_mod_lock = RW_LOCK_UNLOCKED;int tcf_register_action(struct tc_action_ops *act){	struct tc_action_ops *a, **ap;	write_lock(&act_mod_lock);	for (ap = &act_base; (a=*ap)!=NULL; ap = &a->next) {		if (act->type == a->type || (strcmp(act->kind, a->kind) == 0)) {			write_unlock(&act_mod_lock);			return -EEXIST;		}	}        act->next = NULL;	*ap = act;	write_unlock(&act_mod_lock);	return 0;}int tcf_unregister_action(struct tc_action_ops *act){	struct tc_action_ops *a, **ap;	int err = -ENOENT;	write_lock(&act_mod_lock);	for (ap = &act_base; (a=*ap)!=NULL; ap = &a->next) 		if(a == act)			break;	if (a) {		*ap = a->next;		a->next = NULL;		err = 0;	}	write_unlock(&act_mod_lock);	return err;}/* lookup by name */struct tc_action_ops *tc_lookup_action_n(char *kind){	struct tc_action_ops *a = NULL;	if (kind) {		read_lock(&act_mod_lock);		for (a = act_base; a; a = a->next) {			if (strcmp(kind,a->kind) == 0) {				if (!try_module_get(a->owner)) {					read_unlock(&act_mod_lock);					return NULL;				} 				break;			}		}		read_unlock(&act_mod_lock);	}	return a;}/* lookup by rtattr */struct tc_action_ops *tc_lookup_action(struct rtattr *kind){	struct tc_action_ops *a = NULL;	if (kind) {		read_lock(&act_mod_lock);		for (a = act_base; a; a = a->next) {			if (strcmp((char*)RTA_DATA(kind),a->kind) == 0){				if (!try_module_get(a->owner)) {					read_unlock(&act_mod_lock);					return NULL;				} 				break;			}		}		read_unlock(&act_mod_lock);	}	return a;}/* lookup by id */struct tc_action_ops *tc_lookup_action_id(u32 type){	struct tc_action_ops *a = NULL;	if (type) {		read_lock(&act_mod_lock);		for (a = act_base; a; a = a->next) {			if (a->type == type) {				if (!try_module_get(a->owner)) {					read_unlock(&act_mod_lock);					return NULL;				} 				break;			}		}		read_unlock(&act_mod_lock);	}	return a;}int tcf_action_exec(struct sk_buff *skb,struct tc_action *act, struct tcf_result *res){	struct tc_action *a;	int ret = -1; 	if (skb->tc_verd & TC_NCLS) {		skb->tc_verd = CLR_TC_NCLS(skb->tc_verd);		D2PRINTK("(%p)tcf_action_exec: cleared TC_NCLS in %s out %s\n",skb,skb->input_dev?skb->input_dev->name:"xxx",skb->dev->name);		ret = TC_ACT_OK;		goto exec_done;	}	while ((a = act) != NULL) {repeat:		if (a->ops && a->ops->act) {			ret = a->ops->act(&skb,a);				if (TC_MUNGED & skb->tc_verd) {					/* copied already, allow trampling */					skb->tc_verd = SET_TC_OK2MUNGE(skb->tc_verd);					skb->tc_verd = CLR_TC_MUNGED(skb->tc_verd);				}			if (ret != TC_ACT_PIPE)				goto exec_done;			if (ret == TC_ACT_REPEAT)				goto repeat;	/* we need a ttl - JHS */		}		act = a->next;	}exec_done:	if (skb->tc_classid > 0) {		res->classid = skb->tc_classid;		res->class = 0;		skb->tc_classid = 0;	}	return ret;}void tcf_action_destroy(struct tc_action *act, int bind){	struct tc_action *a;	for (a = act; act; a = act) {		if (a && a->ops && a->ops->cleanup) {			DPRINTK("tcf_action_destroy destroying %p next %p\n", a,a->next?a->next:NULL);			act = act->next;			if (ACT_P_DELETED == a->ops->cleanup(a, bind)) {				module_put(a->ops->owner);			}						a->ops = NULL;  			kfree(a);		} else { /*FIXME: Remove later - catch insertion bugs*/			printk("tcf_action_destroy: BUG? destroying NULL ops \n");			if (a) {				act = act->next;				kfree(a);			} else {				printk("tcf_action_destroy: BUG? destroying NULL action! \n");				break;			}		}	}}int tcf_action_dump_old(struct sk_buff *skb, struct tc_action *a, int bind, int ref){	int err = -EINVAL;	if ( (NULL == a) || (NULL == a->ops)	   || (NULL == a->ops->dump) )		return err;	return a->ops->dump(skb, a, bind, ref);}int tcf_action_dump_1(struct sk_buff *skb, struct tc_action *a, int bind, int ref){	int err = -EINVAL;	unsigned char    *b = skb->tail;	struct rtattr *r;	if ( (NULL == a) || (NULL == a->ops)	   || (NULL == a->ops->dump) || (NULL == a->ops->kind))		return err;	RTA_PUT(skb, TCA_KIND, IFNAMSIZ, a->ops->kind);	if (tcf_action_copy_stats(skb,a))		goto rtattr_failure;	r = (struct rtattr*) skb->tail;	RTA_PUT(skb, TCA_OPTIONS, 0, NULL);	if ((err = tcf_action_dump_old(skb, a, bind, ref)) > 0) {		r->rta_len = skb->tail - (u8*)r;		return err;	}rtattr_failure:	skb_trim(skb, b - skb->data);	return -1;}int tcf_action_dump(struct sk_buff *skb, struct tc_action *act, int bind, int ref){	struct tc_action *a;	int err = -EINVAL;	unsigned char    *b = skb->tail;	struct rtattr *r ;	while ((a = act) != NULL) {		r = (struct rtattr*) skb->tail;		act = a->next;		RTA_PUT(skb, a->order, 0, NULL);		err = tcf_action_dump_1(skb, a, bind, ref);		if (0 > err) 			goto rtattr_failure;		r->rta_len = skb->tail - (u8*)r;	}	return 0;rtattr_failure:	skb_trim(skb, b - skb->data);	return -err;	}int tcf_action_init_1(struct rtattr *rta, struct rtattr *est, struct tc_action *a, char *name, int ovr, int bind ){	struct tc_action_ops *a_o;	char act_name[4 + IFNAMSIZ + 1];	struct rtattr *tb[TCA_ACT_MAX+1];	struct rtattr *kind = NULL;	int err = -EINVAL;	if (NULL == name) {		if (rtattr_parse(tb, TCA_ACT_MAX, RTA_DATA(rta), RTA_PAYLOAD(rta))<0)			goto err_out;		kind = tb[TCA_ACT_KIND-1];		if (NULL != kind) {			sprintf(act_name, "%s", (char*)RTA_DATA(kind));			if (RTA_PAYLOAD(kind) >= IFNAMSIZ) {				printk(" Action %s bad\n", (char*)RTA_DATA(kind));				goto err_out;			}		} else {			printk("Action bad kind\n");			goto err_out;		}		a_o = tc_lookup_action(kind);	} else {		sprintf(act_name, "%s", name);		DPRINTK("tcf_action_init_1: finding  %s\n",act_name);		a_o = tc_lookup_action_n(name);	}#ifdef CONFIG_KMOD	if (NULL == a_o) {		DPRINTK("tcf_action_init_1: trying to load module %s\n",act_name);		request_module (act_name);		a_o = tc_lookup_action_n(act_name);	}#endif	if (NULL == a_o) {		printk("failed to find %s\n",act_name);		goto err_out;	}	if (NULL == a) {		goto err_mod;	}	/* backward compatibility for policer */	if (NULL == name) {		err = a_o->init(tb[TCA_ACT_OPTIONS-1], est, a, ovr, bind);		if (0 > err ) {			err = -EINVAL;			goto err_mod;		}	} else {		err = a_o->init(rta, est, a, ovr, bind);		if (0 > err ) {			err = -EINVAL;			goto err_mod;		}	}	/* module count goes up only when brand new policy is created	   if it exists and is only bound to in a_o->init() then           ACT_P_CREATED is not returned (a zero is).        */	if (ACT_P_CREATED != err) {		module_put(a_o->owner);	} 	a->ops = a_o;	DPRINTK("tcf_action_init_1: successfull %s \n",act_name);	return 0;err_mod:	module_put(a_o->owner);err_out:	return err;}int tcf_action_init(struct rtattr *rta, struct rtattr *est, struct tc_action *a, char *name, int ovr , int bind){	struct rtattr *tb[TCA_ACT_MAX_PRIO+1];	int i;	struct tc_action *act = a, *a_s = a;	int err = -EINVAL;	if (rtattr_parse(tb, TCA_ACT_MAX_PRIO, RTA_DATA(rta), RTA_PAYLOAD(rta))<0)		return err;	for (i=0; i < TCA_ACT_MAX_PRIO ; i++) {		if (tb[i]) {			if (NULL == act) {				act = kmalloc(sizeof(*act),GFP_KERNEL);				if (NULL == act) {					err = -ENOMEM;					goto bad_ret;				}				memset(act, 0,sizeof(*act));			}			act->next = NULL;			if (0 > tcf_action_init_1(tb[i],est,act,name,ovr,bind)) {				printk("Error processing action order %d\n",i);				return err;			}			act->order = i+1;			if (a_s != act) {				a_s->next = act;				a_s = act;			}			act = NULL;		}	}	return 0;bad_ret:	tcf_action_destroy(a, bind);	return err;}int tcf_action_copy_stats (struct sk_buff *skb,struct tc_action *a){#ifdef CONFIG_KMOD	/* place holder */#endif	if (NULL == a->ops || NULL == a->ops->get_stats)		return 1;	return a->ops->get_stats(skb,a);}static inttca_get_fill(struct sk_buff *skb,  struct tc_action *a,	      u32 pid, u32 seq, unsigned flags, int event, int bind, int ref){	struct tcamsg *t;	struct nlmsghdr  *nlh;	unsigned char	 *b = skb->tail;	struct rtattr *x;	nlh = NLMSG_PUT(skb, pid, seq, event, sizeof(*t));	nlh->nlmsg_flags = flags;	t = NLMSG_DATA(nlh);	t->tca_family = AF_UNSPEC;		x = (struct rtattr*) skb->tail;	RTA_PUT(skb, TCA_ACT_TAB, 0, NULL);	if (0 > tcf_action_dump(skb, a, bind, ref)) {		goto rtattr_failure;	}	x->rta_len = skb->tail - (u8*)x;		nlh->nlmsg_len = skb->tail - b;	return skb->len;rtattr_failure:nlmsg_failure:	skb_trim(skb, b - skb->data);	return -1;}static int act_get_notify(u32 pid, struct nlmsghdr *n,			   struct tc_action *a, int event){	struct sk_buff *skb;	int err = 0;	skb = alloc_skb(NLMSG_GOODSIZE, GFP_KERNEL);	if (!skb)		return -ENOBUFS;	if (tca_get_fill(skb, a,  pid, n->nlmsg_seq, 0, event, 0, 0) <= 0) {		kfree_skb(skb);		return -EINVAL;	}	err =  netlink_unicast(rtnl,skb, pid, MSG_DONTWAIT);	if (err > 0)		err = 0;	return err;}int tcf_action_get_1(struct rtattr *rta, struct tc_action *a, struct nlmsghdr *n, u32 pid){	struct tc_action_ops *a_o;	char act_name[4 + IFNAMSIZ + 1];	struct rtattr *tb[TCA_ACT_MAX+1];	struct rtattr *kind = NULL;	int index;	int err = -EINVAL;	if (rtattr_parse(tb, TCA_ACT_MAX, RTA_DATA(rta), RTA_PAYLOAD(rta))<0)		goto err_out;	kind = tb[TCA_ACT_KIND-1];	if (NULL != kind) {		sprintf(act_name, "%s", (char*)RTA_DATA(kind));		if (RTA_PAYLOAD(kind) >= IFNAMSIZ) {			printk("tcf_action_get_1: action %s bad\n", (char*)RTA_DATA(kind));			goto err_out;		}	} else {		printk("tcf_action_get_1: action bad kind\n");		goto err_out;	}	if (tb[TCA_ACT_INDEX - 1]) {		index = *(int *)RTA_DATA(tb[TCA_ACT_INDEX - 1]);	} else {

⌨️ 快捷键说明

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