diff options
| author | Cong Wang <cwang@twopensource.com> | 2014-04-25 13:54:06 -0700 | 
|---|---|---|
| committer | David S. Miller <davem@davemloft.net> | 2014-04-27 23:42:39 -0400 | 
| commit | 2f7ef2f8790f5bf53db4fc6b2310943139285827 (patch) | |
| tree | f2deb92bec03f3af01b9d7c1bf4dfb7b97760ecf /net/sched/cls_basic.c | |
| parent | 4940b8cd1bc6d452eaffa442d92a28534850ca78 (diff) | |
sched, cls: check if we could overwrite actions when changing a filter
When actions are attached to a filter, they are a part of the filter
itself, so when changing a filter we should allow to overwrite the actions
inside as well.
In my specific case, when I tried to _append_ a new action to an existing
filter which already has an action, I got EEXIST since kernel refused
to overwrite the existing one in kernel.
This patch checks if we are changing the filter checking NLM_F_CREATE flag
(Sigh, filters don't use NLM_F_REPLACE...) and then passes the boolean down
to actions. This fixes the problem above.
Cc: Jamal Hadi Salim <jhs@mojatatu.com>
Cc: David S. Miller <davem@davemloft.net>
Signed-off-by: Cong Wang <xiyou.wangcong@gmail.com>
Signed-off-by: Cong Wang <cwang@twopensource.com>
Signed-off-by: Jamal Hadi Salim <jhs@mojatatu.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
Diffstat (limited to 'net/sched/cls_basic.c')
| -rw-r--r-- | net/sched/cls_basic.c | 10 | 
1 files changed, 5 insertions, 5 deletions
| diff --git a/net/sched/cls_basic.c b/net/sched/cls_basic.c index e98ca99c202b..0ae1813e3e90 100644 --- a/net/sched/cls_basic.c +++ b/net/sched/cls_basic.c @@ -130,14 +130,14 @@ static const struct nla_policy basic_policy[TCA_BASIC_MAX + 1] = {  static int basic_set_parms(struct net *net, struct tcf_proto *tp,  			   struct basic_filter *f, unsigned long base,  			   struct nlattr **tb, -			   struct nlattr *est) +			   struct nlattr *est, bool ovr)  {  	int err;  	struct tcf_exts e;  	struct tcf_ematch_tree t;  	tcf_exts_init(&e, TCA_BASIC_ACT, TCA_BASIC_POLICE); -	err = tcf_exts_validate(net, tp, tb, est, &e); +	err = tcf_exts_validate(net, tp, tb, est, &e, ovr);  	if (err < 0)  		return err; @@ -161,7 +161,7 @@ errout:  static int basic_change(struct net *net, struct sk_buff *in_skb,  			struct tcf_proto *tp, unsigned long base, u32 handle, -			struct nlattr **tca, unsigned long *arg) +			struct nlattr **tca, unsigned long *arg, bool ovr)  {  	int err;  	struct basic_head *head = tp->root; @@ -179,7 +179,7 @@ static int basic_change(struct net *net, struct sk_buff *in_skb,  	if (f != NULL) {  		if (handle && f->handle != handle)  			return -EINVAL; -		return basic_set_parms(net, tp, f, base, tb, tca[TCA_RATE]); +		return basic_set_parms(net, tp, f, base, tb, tca[TCA_RATE], ovr);  	}  	err = -ENOBUFS; @@ -206,7 +206,7 @@ static int basic_change(struct net *net, struct sk_buff *in_skb,  		f->handle = head->hgenerator;  	} -	err = basic_set_parms(net, tp, f, base, tb, tca[TCA_RATE]); +	err = basic_set_parms(net, tp, f, base, tb, tca[TCA_RATE], ovr);  	if (err < 0)  		goto errout; | 
