diff options
Diffstat (limited to 'net/dsa')
| -rw-r--r-- | net/dsa/dsa2.c | 15 | ||||
| -rw-r--r-- | net/dsa/tag_ksz.c | 17 | ||||
| -rw-r--r-- | net/dsa/tag_trailer.c | 2 | 
3 files changed, 21 insertions, 13 deletions
diff --git a/net/dsa/dsa2.c b/net/dsa/dsa2.c index 56e46090526b..20bc9c56fca0 100644 --- a/net/dsa/dsa2.c +++ b/net/dsa/dsa2.c @@ -509,21 +509,22 @@ static int dsa_cpu_parse(struct dsa_port *port, u32 index,  		dst->cpu_dp->netdev = ethernet_dev;  	} +	/* Initialize cpu_port_mask now for drv->setup() +	 * to have access to a correct value, just like what +	 * net/dsa/dsa.c::dsa_switch_setup_one does. +	 */ +	ds->cpu_port_mask |= BIT(index); +  	tag_protocol = ds->ops->get_tag_protocol(ds);  	dst->tag_ops = dsa_resolve_tag_protocol(tag_protocol);  	if (IS_ERR(dst->tag_ops)) {  		dev_warn(ds->dev, "No tagger for this switch\n"); +		ds->cpu_port_mask &= ~BIT(index);  		return PTR_ERR(dst->tag_ops);  	}  	dst->rcv = dst->tag_ops->rcv; -	/* Initialize cpu_port_mask now for drv->setup() -	 * to have access to a correct value, just like what -	 * net/dsa/dsa.c::dsa_switch_setup_one does. -	 */ -	ds->cpu_port_mask |= BIT(index); -  	return 0;  } @@ -576,7 +577,7 @@ static int dsa_dst_parse(struct dsa_switch_tree *dst)  			return err;  	} -	if (!dst->cpu_dp->netdev) { +	if (!dst->cpu_dp) {  		pr_warn("Tree has no master device\n");  		return -EINVAL;  	} diff --git a/net/dsa/tag_ksz.c b/net/dsa/tag_ksz.c index fab41de8e983..fcd90f79458e 100644 --- a/net/dsa/tag_ksz.c +++ b/net/dsa/tag_ksz.c @@ -42,6 +42,10 @@ static struct sk_buff *ksz_xmit(struct sk_buff *skb, struct net_device *dev)  	padlen = (skb->len >= ETH_ZLEN) ? 0 : ETH_ZLEN - skb->len;  	if (skb_tailroom(skb) >= padlen + KSZ_INGRESS_TAG_LEN) { +		/* Let dsa_slave_xmit() free skb */ +		if (__skb_put_padto(skb, skb->len + padlen, false)) +			return NULL; +  		nskb = skb;  	} else {  		nskb = alloc_skb(NET_IP_ALIGN + skb->len + @@ -56,12 +60,15 @@ static struct sk_buff *ksz_xmit(struct sk_buff *skb, struct net_device *dev)  		skb_set_transport_header(nskb,  					 skb_transport_header(skb) - skb->head);  		skb_copy_and_csum_dev(skb, skb_put(nskb, skb->len)); -		kfree_skb(skb); -	} -	/* skb is freed when it fails */ -	if (skb_put_padto(nskb, nskb->len + padlen)) -		return NULL; +		/* Let skb_put_padto() free nskb, and let dsa_slave_xmit() free +		 * skb +		 */ +		if (skb_put_padto(nskb, nskb->len + padlen)) +			return NULL; + +		consume_skb(skb); +	}  	tag = skb_put(nskb, KSZ_INGRESS_TAG_LEN);  	tag[0] = 0; diff --git a/net/dsa/tag_trailer.c b/net/dsa/tag_trailer.c index b09e56214005..9c7b1d74a5c6 100644 --- a/net/dsa/tag_trailer.c +++ b/net/dsa/tag_trailer.c @@ -40,7 +40,7 @@ static struct sk_buff *trailer_xmit(struct sk_buff *skb, struct net_device *dev)  	skb_set_network_header(nskb, skb_network_header(skb) - skb->head);  	skb_set_transport_header(nskb, skb_transport_header(skb) - skb->head);  	skb_copy_and_csum_dev(skb, skb_put(nskb, skb->len)); -	kfree_skb(skb); +	consume_skb(skb);  	if (padlen) {  		skb_put_zero(nskb, padlen);  | 
