diff options
Diffstat (limited to 'net/sched/cls_u32.c')
| -rw-r--r-- | net/sched/cls_u32.c | 72 | 
1 files changed, 40 insertions, 32 deletions
| diff --git a/net/sched/cls_u32.c b/net/sched/cls_u32.c index 079b43b3c5d2..ffe593efe930 100644 --- a/net/sched/cls_u32.c +++ b/net/sched/cls_u32.c @@ -440,7 +440,7 @@ static void u32_remove_hw_knode(struct tcf_proto *tp, u32 handle)  	offload.type = TC_SETUP_CLSU32;  	offload.cls_u32 = &u32_offload; -	if (tc_should_offload(dev, 0)) { +	if (tc_should_offload(dev, tp, 0)) {  		offload.cls_u32->command = TC_CLSU32_DELETE_KNODE;  		offload.cls_u32->knode.handle = handle;  		dev->netdev_ops->ndo_setup_tc(dev, tp->q->handle, @@ -457,20 +457,21 @@ static int u32_replace_hw_hnode(struct tcf_proto *tp,  	struct tc_to_netdev offload;  	int err; +	if (!tc_should_offload(dev, tp, flags)) +		return tc_skip_sw(flags) ? -EINVAL : 0; +  	offload.type = TC_SETUP_CLSU32;  	offload.cls_u32 = &u32_offload; -	if (tc_should_offload(dev, flags)) { -		offload.cls_u32->command = TC_CLSU32_NEW_HNODE; -		offload.cls_u32->hnode.divisor = h->divisor; -		offload.cls_u32->hnode.handle = h->handle; -		offload.cls_u32->hnode.prio = h->prio; +	offload.cls_u32->command = TC_CLSU32_NEW_HNODE; +	offload.cls_u32->hnode.divisor = h->divisor; +	offload.cls_u32->hnode.handle = h->handle; +	offload.cls_u32->hnode.prio = h->prio; -		err = dev->netdev_ops->ndo_setup_tc(dev, tp->q->handle, -						    tp->protocol, &offload); -		if (tc_skip_sw(flags)) -			return err; -	} +	err = dev->netdev_ops->ndo_setup_tc(dev, tp->q->handle, +					    tp->protocol, &offload); +	if (tc_skip_sw(flags)) +		return err;  	return 0;  } @@ -484,7 +485,7 @@ static void u32_clear_hw_hnode(struct tcf_proto *tp, struct tc_u_hnode *h)  	offload.type = TC_SETUP_CLSU32;  	offload.cls_u32 = &u32_offload; -	if (tc_should_offload(dev, 0)) { +	if (tc_should_offload(dev, tp, 0)) {  		offload.cls_u32->command = TC_CLSU32_DELETE_HNODE;  		offload.cls_u32->hnode.divisor = h->divisor;  		offload.cls_u32->hnode.handle = h->handle; @@ -507,27 +508,28 @@ static int u32_replace_hw_knode(struct tcf_proto *tp,  	offload.type = TC_SETUP_CLSU32;  	offload.cls_u32 = &u32_offload; -	if (tc_should_offload(dev, flags)) { -		offload.cls_u32->command = TC_CLSU32_REPLACE_KNODE; -		offload.cls_u32->knode.handle = n->handle; -		offload.cls_u32->knode.fshift = n->fshift; +	if (!tc_should_offload(dev, tp, flags)) +		return tc_skip_sw(flags) ? -EINVAL : 0; + +	offload.cls_u32->command = TC_CLSU32_REPLACE_KNODE; +	offload.cls_u32->knode.handle = n->handle; +	offload.cls_u32->knode.fshift = n->fshift;  #ifdef CONFIG_CLS_U32_MARK -		offload.cls_u32->knode.val = n->val; -		offload.cls_u32->knode.mask = n->mask; +	offload.cls_u32->knode.val = n->val; +	offload.cls_u32->knode.mask = n->mask;  #else -		offload.cls_u32->knode.val = 0; -		offload.cls_u32->knode.mask = 0; +	offload.cls_u32->knode.val = 0; +	offload.cls_u32->knode.mask = 0;  #endif -		offload.cls_u32->knode.sel = &n->sel; -		offload.cls_u32->knode.exts = &n->exts; -		if (n->ht_down) -			offload.cls_u32->knode.link_handle = n->ht_down->handle; - -		err = dev->netdev_ops->ndo_setup_tc(dev, tp->q->handle, -						    tp->protocol, &offload); -		if (tc_skip_sw(flags)) -			return err; -	} +	offload.cls_u32->knode.sel = &n->sel; +	offload.cls_u32->knode.exts = &n->exts; +	if (n->ht_down) +		offload.cls_u32->knode.link_handle = n->ht_down->handle; + +	err = dev->netdev_ops->ndo_setup_tc(dev, tp->q->handle, +					    tp->protocol, &offload); +	if (tc_skip_sw(flags)) +		return err;  	return 0;  } @@ -863,7 +865,7 @@ static int u32_change(struct net *net, struct sk_buff *in_skb,  	if (tb[TCA_U32_FLAGS]) {  		flags = nla_get_u32(tb[TCA_U32_FLAGS]);  		if (!tc_flags_valid(flags)) -			return err; +			return -EINVAL;  	}  	n = (struct tc_u_knode *)*arg; @@ -921,11 +923,17 @@ static int u32_change(struct net *net, struct sk_buff *in_skb,  		ht->divisor = divisor;  		ht->handle = handle;  		ht->prio = tp->prio; + +		err = u32_replace_hw_hnode(tp, ht, flags); +		if (err) { +			kfree(ht); +			return err; +		} +  		RCU_INIT_POINTER(ht->next, tp_c->hlist);  		rcu_assign_pointer(tp_c->hlist, ht);  		*arg = (unsigned long)ht; -		u32_replace_hw_hnode(tp, ht, flags);  		return 0;  	} | 
