diff options
Diffstat (limited to 'drivers/net/xen-netback/netback.c')
| -rw-r--r-- | drivers/net/xen-netback/netback.c | 288 | 
1 files changed, 167 insertions, 121 deletions
| diff --git a/drivers/net/xen-netback/netback.c b/drivers/net/xen-netback/netback.c index 919b6509455c..78425554a537 100644 --- a/drivers/net/xen-netback/netback.c +++ b/drivers/net/xen-netback/netback.c @@ -39,6 +39,7 @@  #include <linux/udp.h>  #include <net/tcp.h> +#include <net/ip6_checksum.h>  #include <xen/xen.h>  #include <xen/events.h> @@ -451,7 +452,7 @@ static int xenvif_gop_skb(struct sk_buff *skb,  	}  	/* Set up a GSO prefix descriptor, if necessary */ -	if ((1 << skb_shinfo(skb)->gso_type) & vif->gso_prefix_mask) { +	if ((1 << gso_type) & vif->gso_prefix_mask) {  		req = RING_GET_REQUEST(&vif->rx, vif->rx.req_cons++);  		meta = npo->meta + npo->meta_prod++;  		meta->gso_type = gso_type; @@ -607,7 +608,7 @@ void xenvif_rx_action(struct xenvif *vif)  	if (!npo.copy_prod)  		return; -	BUG_ON(npo.copy_prod > ARRAY_SIZE(vif->grant_copy_op)); +	BUG_ON(npo.copy_prod > MAX_GRANT_COPY_OPS);  	gnttab_batch_copy(vif->grant_copy_op, npo.copy_prod);  	while ((skb = __skb_dequeue(&rxq)) != NULL) { @@ -1148,75 +1149,99 @@ static int xenvif_set_skb_gso(struct xenvif *vif,  	return 0;  } -static inline void maybe_pull_tail(struct sk_buff *skb, unsigned int len) +static inline int maybe_pull_tail(struct sk_buff *skb, unsigned int len, +				  unsigned int max)  { -	if (skb_is_nonlinear(skb) && skb_headlen(skb) < len) { -		/* If we need to pullup then pullup to the max, so we -		 * won't need to do it again. -		 */ -		int target = min_t(int, skb->len, MAX_TCP_HEADER); -		__pskb_pull_tail(skb, target - skb_headlen(skb)); -	} +	if (skb_headlen(skb) >= len) +		return 0; + +	/* If we need to pullup then pullup to the max, so we +	 * won't need to do it again. +	 */ +	if (max > skb->len) +		max = skb->len; + +	if (__pskb_pull_tail(skb, max - skb_headlen(skb)) == NULL) +		return -ENOMEM; + +	if (skb_headlen(skb) < len) +		return -EPROTO; + +	return 0;  } +/* This value should be large enough to cover a tagged ethernet header plus + * maximally sized IP and TCP or UDP headers. + */ +#define MAX_IP_HDR_LEN 128 +  static int checksum_setup_ip(struct xenvif *vif, struct sk_buff *skb,  			     int recalculate_partial_csum)  { -	struct iphdr *iph = (void *)skb->data; -	unsigned int header_size;  	unsigned int off; -	int err = -EPROTO; +	bool fragment; +	int err; -	off = sizeof(struct iphdr); +	fragment = false; -	header_size = skb->network_header + off + MAX_IPOPTLEN; -	maybe_pull_tail(skb, header_size); +	err = maybe_pull_tail(skb, +			      sizeof(struct iphdr), +			      MAX_IP_HDR_LEN); +	if (err < 0) +		goto out; -	off = iph->ihl * 4; +	if (ip_hdr(skb)->frag_off & htons(IP_OFFSET | IP_MF)) +		fragment = true; -	switch (iph->protocol) { -	case IPPROTO_TCP: -		if (!skb_partial_csum_set(skb, off, -					  offsetof(struct tcphdr, check))) -			goto out; +	off = ip_hdrlen(skb); -		if (recalculate_partial_csum) { -			struct tcphdr *tcph = tcp_hdr(skb); +	err = -EPROTO; + +	if (fragment) +		goto out; -			header_size = skb->network_header + -				off + -				sizeof(struct tcphdr); -			maybe_pull_tail(skb, header_size); +	switch (ip_hdr(skb)->protocol) { +	case IPPROTO_TCP: +		err = maybe_pull_tail(skb, +				      off + sizeof(struct tcphdr), +				      MAX_IP_HDR_LEN); +		if (err < 0) +			goto out; -			tcph->check = ~csum_tcpudp_magic(iph->saddr, iph->daddr, -							 skb->len - off, -							 IPPROTO_TCP, 0); +		if (!skb_partial_csum_set(skb, off, +					  offsetof(struct tcphdr, check))) { +			err = -EPROTO; +			goto out;  		} + +		if (recalculate_partial_csum) +			tcp_hdr(skb)->check = +				~csum_tcpudp_magic(ip_hdr(skb)->saddr, +						   ip_hdr(skb)->daddr, +						   skb->len - off, +						   IPPROTO_TCP, 0);  		break;  	case IPPROTO_UDP: -		if (!skb_partial_csum_set(skb, off, -					  offsetof(struct udphdr, check))) +		err = maybe_pull_tail(skb, +				      off + sizeof(struct udphdr), +				      MAX_IP_HDR_LEN); +		if (err < 0)  			goto out; -		if (recalculate_partial_csum) { -			struct udphdr *udph = udp_hdr(skb); - -			header_size = skb->network_header + -				off + -				sizeof(struct udphdr); -			maybe_pull_tail(skb, header_size); - -			udph->check = ~csum_tcpudp_magic(iph->saddr, iph->daddr, -							 skb->len - off, -							 IPPROTO_UDP, 0); +		if (!skb_partial_csum_set(skb, off, +					  offsetof(struct udphdr, check))) { +			err = -EPROTO; +			goto out;  		} + +		if (recalculate_partial_csum) +			udp_hdr(skb)->check = +				~csum_tcpudp_magic(ip_hdr(skb)->saddr, +						   ip_hdr(skb)->daddr, +						   skb->len - off, +						   IPPROTO_UDP, 0);  		break;  	default: -		if (net_ratelimit()) -			netdev_err(vif->dev, -				   "Attempting to checksum a non-TCP/UDP packet, " -				   "dropping a protocol %d packet\n", -				   iph->protocol);  		goto out;  	} @@ -1226,121 +1251,142 @@ out:  	return err;  } +/* This value should be large enough to cover a tagged ethernet header plus + * an IPv6 header, all options, and a maximal TCP or UDP header. + */ +#define MAX_IPV6_HDR_LEN 256 + +#define OPT_HDR(type, skb, off) \ +	(type *)(skb_network_header(skb) + (off)) +  static int checksum_setup_ipv6(struct xenvif *vif, struct sk_buff *skb,  			       int recalculate_partial_csum)  { -	int err = -EPROTO; -	struct ipv6hdr *ipv6h = (void *)skb->data; +	int err;  	u8 nexthdr; -	unsigned int header_size;  	unsigned int off; +	unsigned int len;  	bool fragment;  	bool done; +	fragment = false;  	done = false;  	off = sizeof(struct ipv6hdr); -	header_size = skb->network_header + off; -	maybe_pull_tail(skb, header_size); +	err = maybe_pull_tail(skb, off, MAX_IPV6_HDR_LEN); +	if (err < 0) +		goto out; -	nexthdr = ipv6h->nexthdr; +	nexthdr = ipv6_hdr(skb)->nexthdr; -	while ((off <= sizeof(struct ipv6hdr) + ntohs(ipv6h->payload_len)) && -	       !done) { +	len = sizeof(struct ipv6hdr) + ntohs(ipv6_hdr(skb)->payload_len); +	while (off <= len && !done) {  		switch (nexthdr) {  		case IPPROTO_DSTOPTS:  		case IPPROTO_HOPOPTS:  		case IPPROTO_ROUTING: { -			struct ipv6_opt_hdr *hp = (void *)(skb->data + off); +			struct ipv6_opt_hdr *hp; -			header_size = skb->network_header + -				off + -				sizeof(struct ipv6_opt_hdr); -			maybe_pull_tail(skb, header_size); +			err = maybe_pull_tail(skb, +					      off + +					      sizeof(struct ipv6_opt_hdr), +					      MAX_IPV6_HDR_LEN); +			if (err < 0) +				goto out; +			hp = OPT_HDR(struct ipv6_opt_hdr, skb, off);  			nexthdr = hp->nexthdr;  			off += ipv6_optlen(hp);  			break;  		}  		case IPPROTO_AH: { -			struct ip_auth_hdr *hp = (void *)(skb->data + off); +			struct ip_auth_hdr *hp; -			header_size = skb->network_header + -				off + -				sizeof(struct ip_auth_hdr); -			maybe_pull_tail(skb, header_size); +			err = maybe_pull_tail(skb, +					      off + +					      sizeof(struct ip_auth_hdr), +					      MAX_IPV6_HDR_LEN); +			if (err < 0) +				goto out; +			hp = OPT_HDR(struct ip_auth_hdr, skb, off);  			nexthdr = hp->nexthdr; -			off += (hp->hdrlen+2)<<2; +			off += ipv6_authlen(hp); +			break; +		} +		case IPPROTO_FRAGMENT: { +			struct frag_hdr *hp; + +			err = maybe_pull_tail(skb, +					      off + +					      sizeof(struct frag_hdr), +					      MAX_IPV6_HDR_LEN); +			if (err < 0) +				goto out; + +			hp = OPT_HDR(struct frag_hdr, skb, off); + +			if (hp->frag_off & htons(IP6_OFFSET | IP6_MF)) +				fragment = true; + +			nexthdr = hp->nexthdr; +			off += sizeof(struct frag_hdr);  			break;  		} -		case IPPROTO_FRAGMENT: -			fragment = true; -			/* fall through */  		default:  			done = true;  			break;  		}  	} -	if (!done) { -		if (net_ratelimit()) -			netdev_err(vif->dev, "Failed to parse packet header\n"); -		goto out; -	} +	err = -EPROTO; -	if (fragment) { -		if (net_ratelimit()) -			netdev_err(vif->dev, "Packet is a fragment!\n"); +	if (!done || fragment)  		goto out; -	}  	switch (nexthdr) {  	case IPPROTO_TCP: -		if (!skb_partial_csum_set(skb, off, -					  offsetof(struct tcphdr, check))) +		err = maybe_pull_tail(skb, +				      off + sizeof(struct tcphdr), +				      MAX_IPV6_HDR_LEN); +		if (err < 0)  			goto out; -		if (recalculate_partial_csum) { -			struct tcphdr *tcph = tcp_hdr(skb); - -			header_size = skb->network_header + -				off + -				sizeof(struct tcphdr); -			maybe_pull_tail(skb, header_size); - -			tcph->check = ~csum_ipv6_magic(&ipv6h->saddr, -						       &ipv6h->daddr, -						       skb->len - off, -						       IPPROTO_TCP, 0); +		if (!skb_partial_csum_set(skb, off, +					  offsetof(struct tcphdr, check))) { +			err = -EPROTO; +			goto out;  		} + +		if (recalculate_partial_csum) +			tcp_hdr(skb)->check = +				~csum_ipv6_magic(&ipv6_hdr(skb)->saddr, +						 &ipv6_hdr(skb)->daddr, +						 skb->len - off, +						 IPPROTO_TCP, 0);  		break;  	case IPPROTO_UDP: -		if (!skb_partial_csum_set(skb, off, -					  offsetof(struct udphdr, check))) +		err = maybe_pull_tail(skb, +				      off + sizeof(struct udphdr), +				      MAX_IPV6_HDR_LEN); +		if (err < 0)  			goto out; -		if (recalculate_partial_csum) { -			struct udphdr *udph = udp_hdr(skb); - -			header_size = skb->network_header + -				off + -				sizeof(struct udphdr); -			maybe_pull_tail(skb, header_size); - -			udph->check = ~csum_ipv6_magic(&ipv6h->saddr, -						       &ipv6h->daddr, -						       skb->len - off, -						       IPPROTO_UDP, 0); +		if (!skb_partial_csum_set(skb, off, +					  offsetof(struct udphdr, check))) { +			err = -EPROTO; +			goto out;  		} + +		if (recalculate_partial_csum) +			udp_hdr(skb)->check = +				~csum_ipv6_magic(&ipv6_hdr(skb)->saddr, +						 &ipv6_hdr(skb)->daddr, +						 skb->len - off, +						 IPPROTO_UDP, 0);  		break;  	default: -		if (net_ratelimit()) -			netdev_err(vif->dev, -				   "Attempting to checksum a non-TCP/UDP packet, " -				   "dropping a protocol %d packet\n", -				   nexthdr);  		goto out;  	} @@ -1410,14 +1456,15 @@ static bool tx_credit_exceeded(struct xenvif *vif, unsigned size)  	return false;  } -static unsigned xenvif_tx_build_gops(struct xenvif *vif) +static unsigned xenvif_tx_build_gops(struct xenvif *vif, int budget)  {  	struct gnttab_copy *gop = vif->tx_copy_ops, *request_gop;  	struct sk_buff *skb;  	int ret;  	while ((nr_pending_reqs(vif) + XEN_NETBK_LEGACY_SLOTS_MAX -		< MAX_PENDING_REQS)) { +		< MAX_PENDING_REQS) && +	       (skb_queue_len(&vif->tx_queue) < budget)) {  		struct xen_netif_tx_request txreq;  		struct xen_netif_tx_request txfrags[XEN_NETBK_LEGACY_SLOTS_MAX];  		struct page *page; @@ -1439,7 +1486,7 @@ static unsigned xenvif_tx_build_gops(struct xenvif *vif)  			continue;  		} -		RING_FINAL_CHECK_FOR_REQUESTS(&vif->tx, work_to_do); +		work_to_do = RING_HAS_UNCONSUMED_REQUESTS(&vif->tx);  		if (!work_to_do)  			break; @@ -1579,14 +1626,13 @@ static unsigned xenvif_tx_build_gops(struct xenvif *vif)  } -static int xenvif_tx_submit(struct xenvif *vif, int budget) +static int xenvif_tx_submit(struct xenvif *vif)  {  	struct gnttab_copy *gop = vif->tx_copy_ops;  	struct sk_buff *skb;  	int work_done = 0; -	while (work_done < budget && -	       (skb = __skb_dequeue(&vif->tx_queue)) != NULL) { +	while ((skb = __skb_dequeue(&vif->tx_queue)) != NULL) {  		struct xen_netif_tx_request *txp;  		u16 pending_idx;  		unsigned data_len; @@ -1661,14 +1707,14 @@ int xenvif_tx_action(struct xenvif *vif, int budget)  	if (unlikely(!tx_work_todo(vif)))  		return 0; -	nr_gops = xenvif_tx_build_gops(vif); +	nr_gops = xenvif_tx_build_gops(vif, budget);  	if (nr_gops == 0)  		return 0;  	gnttab_batch_copy(vif->tx_copy_ops, nr_gops); -	work_done = xenvif_tx_submit(vif, nr_gops); +	work_done = xenvif_tx_submit(vif);  	return work_done;  } | 
