diff options
Diffstat (limited to 'drivers/net/xen-netback/netback.c')
| -rw-r--r-- | drivers/net/xen-netback/netback.c | 54 | 
1 files changed, 24 insertions, 30 deletions
| diff --git a/drivers/net/xen-netback/netback.c b/drivers/net/xen-netback/netback.c index e481f3710bd3..61b97c34bb3b 100644 --- a/drivers/net/xen-netback/netback.c +++ b/drivers/net/xen-netback/netback.c @@ -149,20 +149,19 @@ static inline pending_ring_idx_t pending_index(unsigned i)  	return i & (MAX_PENDING_REQS-1);  } -static int xenvif_rx_ring_slots_needed(struct xenvif *vif) -{ -	if (vif->gso_mask) -		return DIV_ROUND_UP(vif->dev->gso_max_size, XEN_PAGE_SIZE) + 1; -	else -		return DIV_ROUND_UP(vif->dev->mtu, XEN_PAGE_SIZE); -} -  static bool xenvif_rx_ring_slots_available(struct xenvif_queue *queue)  {  	RING_IDX prod, cons; +	struct sk_buff *skb;  	int needed; -	needed = xenvif_rx_ring_slots_needed(queue->vif); +	skb = skb_peek(&queue->rx_queue); +	if (!skb) +		return false; + +	needed = DIV_ROUND_UP(skb->len, XEN_PAGE_SIZE); +	if (skb_is_gso(skb)) +		needed++;  	do {  		prod = queue->rx.sring->req_prod; @@ -258,18 +257,18 @@ static struct xenvif_rx_meta *get_next_rx_buffer(struct xenvif_queue *queue,  						 struct netrx_pending_operations *npo)  {  	struct xenvif_rx_meta *meta; -	struct xen_netif_rx_request *req; +	struct xen_netif_rx_request req; -	req = RING_GET_REQUEST(&queue->rx, queue->rx.req_cons++); +	RING_COPY_REQUEST(&queue->rx, queue->rx.req_cons++, &req);  	meta = npo->meta + npo->meta_prod++;  	meta->gso_type = XEN_NETIF_GSO_TYPE_NONE;  	meta->gso_size = 0;  	meta->size = 0; -	meta->id = req->id; +	meta->id = req.id;  	npo->copy_off = 0; -	npo->copy_gref = req->gref; +	npo->copy_gref = req.gref;  	return meta;  } @@ -424,7 +423,7 @@ static int xenvif_gop_skb(struct sk_buff *skb,  	struct xenvif *vif = netdev_priv(skb->dev);  	int nr_frags = skb_shinfo(skb)->nr_frags;  	int i; -	struct xen_netif_rx_request *req; +	struct xen_netif_rx_request req;  	struct xenvif_rx_meta *meta;  	unsigned char *data;  	int head = 1; @@ -443,15 +442,15 @@ static int xenvif_gop_skb(struct sk_buff *skb,  	/* Set up a GSO prefix descriptor, if necessary */  	if ((1 << gso_type) & vif->gso_prefix_mask) { -		req = RING_GET_REQUEST(&queue->rx, queue->rx.req_cons++); +		RING_COPY_REQUEST(&queue->rx, queue->rx.req_cons++, &req);  		meta = npo->meta + npo->meta_prod++;  		meta->gso_type = gso_type;  		meta->gso_size = skb_shinfo(skb)->gso_size;  		meta->size = 0; -		meta->id = req->id; +		meta->id = req.id;  	} -	req = RING_GET_REQUEST(&queue->rx, queue->rx.req_cons++); +	RING_COPY_REQUEST(&queue->rx, queue->rx.req_cons++, &req);  	meta = npo->meta + npo->meta_prod++;  	if ((1 << gso_type) & vif->gso_mask) { @@ -463,9 +462,9 @@ static int xenvif_gop_skb(struct sk_buff *skb,  	}  	meta->size = 0; -	meta->id = req->id; +	meta->id = req.id;  	npo->copy_off = 0; -	npo->copy_gref = req->gref; +	npo->copy_gref = req.gref;  	data = skb->data;  	while (data < skb_tail_pointer(skb)) { @@ -679,9 +678,7 @@ static void tx_add_credit(struct xenvif_queue *queue)  	 * Allow a burst big enough to transmit a jumbo packet of up to 128kB.  	 * Otherwise the interface can seize up due to insufficient credit.  	 */ -	max_burst = RING_GET_REQUEST(&queue->tx, queue->tx.req_cons)->size; -	max_burst = min(max_burst, 131072UL); -	max_burst = max(max_burst, queue->credit_bytes); +	max_burst = max(131072UL, queue->credit_bytes);  	/* Take care that adding a new chunk of credit doesn't wrap to zero. */  	max_credit = queue->remaining_credit + queue->credit_bytes; @@ -711,7 +708,7 @@ static void xenvif_tx_err(struct xenvif_queue *queue,  		spin_unlock_irqrestore(&queue->response_lock, flags);  		if (cons == end)  			break; -		txp = RING_GET_REQUEST(&queue->tx, cons++); +		RING_COPY_REQUEST(&queue->tx, cons++, txp);  	} while (1);  	queue->tx.req_cons = cons;  } @@ -778,8 +775,7 @@ static int xenvif_count_requests(struct xenvif_queue *queue,  		if (drop_err)  			txp = &dropped_tx; -		memcpy(txp, RING_GET_REQUEST(&queue->tx, cons + slots), -		       sizeof(*txp)); +		RING_COPY_REQUEST(&queue->tx, cons + slots, txp);  		/* If the guest submitted a frame >= 64 KiB then  		 * first->size overflowed and following slots will @@ -1112,8 +1108,7 @@ static int xenvif_get_extras(struct xenvif_queue *queue,  			return -EBADR;  		} -		memcpy(&extra, RING_GET_REQUEST(&queue->tx, cons), -		       sizeof(extra)); +		RING_COPY_REQUEST(&queue->tx, cons, &extra);  		if (unlikely(!extra.type ||  			     extra.type >= XEN_NETIF_EXTRA_TYPE_MAX)) {  			queue->tx.req_cons = ++cons; @@ -1322,7 +1317,7 @@ static void xenvif_tx_build_gops(struct xenvif_queue *queue,  		idx = queue->tx.req_cons;  		rmb(); /* Ensure that we see the request before we copy it. */ -		memcpy(&txreq, RING_GET_REQUEST(&queue->tx, idx), sizeof(txreq)); +		RING_COPY_REQUEST(&queue->tx, idx, &txreq);  		/* Credit-based scheduling. */  		if (txreq.size > queue->remaining_credit && @@ -2009,8 +2004,7 @@ static bool xenvif_rx_queue_ready(struct xenvif_queue *queue)  static bool xenvif_have_rx_work(struct xenvif_queue *queue)  { -	return (!skb_queue_empty(&queue->rx_queue) -		&& xenvif_rx_ring_slots_available(queue)) +	return xenvif_rx_ring_slots_available(queue)  		|| (queue->vif->stall_timeout &&  		    (xenvif_rx_queue_stalled(queue)  		     || xenvif_rx_queue_ready(queue))) | 
