Commit 1d5d4852 authored by David Vrabel's avatar David Vrabel Committed by David S. Miller

xen-netback: require fewer guest Rx slots when not using GSO

Commit f48da8b1 (xen-netback: fix
unlimited guest Rx internal queue and carrier flapping) introduced a
regression.

The PV frontend in IPXE only places 4 requests on the guest Rx ring.
Since netback required at least (MAX_SKB_FRAGS + 1) slots, IPXE could
not receive any packets.

a) If GSO is not enabled on the VIF, fewer guest Rx slots are required
   for the largest possible packet.  Calculate the required slots
   based on the maximum GSO size or the MTU.

   This calculation of the number of required slots relies on
   1650d545 (xen-netback: always fully coalesce guest Rx packets)
   which present in 4.0-rc1 and later.

b) Reduce the Rx stall detection to checking for at least one
   available Rx request.  This is fine since we're predominately
   concerned with detecting interfaces which are down and thus have
   zero available Rx requests.
Signed-off-by: default avatarDavid Vrabel <david.vrabel@citrix.com>
Reviewed-by: default avatarWei Liu <wei.liu2@citrix.com>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parent 9b57ab8b
...@@ -200,11 +200,6 @@ struct xenvif_queue { /* Per-queue data for xenvif */ ...@@ -200,11 +200,6 @@ struct xenvif_queue { /* Per-queue data for xenvif */
struct xenvif_stats stats; struct xenvif_stats stats;
}; };
/* Maximum number of Rx slots a to-guest packet may use, including the
* slot needed for GSO meta-data.
*/
#define XEN_NETBK_RX_SLOTS_MAX (MAX_SKB_FRAGS + 1)
enum state_bit_shift { enum state_bit_shift {
/* This bit marks that the vif is connected */ /* This bit marks that the vif is connected */
VIF_STATUS_CONNECTED, VIF_STATUS_CONNECTED,
...@@ -317,11 +312,6 @@ int xenvif_dealloc_kthread(void *data); ...@@ -317,11 +312,6 @@ int xenvif_dealloc_kthread(void *data);
void xenvif_rx_queue_tail(struct xenvif_queue *queue, struct sk_buff *skb); void xenvif_rx_queue_tail(struct xenvif_queue *queue, struct sk_buff *skb);
/* Determine whether the needed number of slots (req) are available,
* and set req_event if not.
*/
bool xenvif_rx_ring_slots_available(struct xenvif_queue *queue, int needed);
void xenvif_carrier_on(struct xenvif *vif); void xenvif_carrier_on(struct xenvif *vif);
/* Callback from stack when TX packet can be released */ /* Callback from stack when TX packet can be released */
......
...@@ -149,9 +149,20 @@ static inline pending_ring_idx_t pending_index(unsigned i) ...@@ -149,9 +149,20 @@ static inline pending_ring_idx_t pending_index(unsigned i)
return i & (MAX_PENDING_REQS-1); return i & (MAX_PENDING_REQS-1);
} }
bool xenvif_rx_ring_slots_available(struct xenvif_queue *queue, int needed) static int xenvif_rx_ring_slots_needed(struct xenvif *vif)
{
if (vif->gso_mask)
return DIV_ROUND_UP(vif->dev->gso_max_size, PAGE_SIZE) + 1;
else
return DIV_ROUND_UP(vif->dev->mtu, PAGE_SIZE);
}
static bool xenvif_rx_ring_slots_available(struct xenvif_queue *queue)
{ {
RING_IDX prod, cons; RING_IDX prod, cons;
int needed;
needed = xenvif_rx_ring_slots_needed(queue->vif);
do { do {
prod = queue->rx.sring->req_prod; prod = queue->rx.sring->req_prod;
...@@ -513,7 +524,7 @@ static void xenvif_rx_action(struct xenvif_queue *queue) ...@@ -513,7 +524,7 @@ static void xenvif_rx_action(struct xenvif_queue *queue)
skb_queue_head_init(&rxq); skb_queue_head_init(&rxq);
while (xenvif_rx_ring_slots_available(queue, XEN_NETBK_RX_SLOTS_MAX) while (xenvif_rx_ring_slots_available(queue)
&& (skb = xenvif_rx_dequeue(queue)) != NULL) { && (skb = xenvif_rx_dequeue(queue)) != NULL) {
queue->last_rx_time = jiffies; queue->last_rx_time = jiffies;
...@@ -1938,8 +1949,7 @@ static bool xenvif_rx_queue_stalled(struct xenvif_queue *queue) ...@@ -1938,8 +1949,7 @@ static bool xenvif_rx_queue_stalled(struct xenvif_queue *queue)
prod = queue->rx.sring->req_prod; prod = queue->rx.sring->req_prod;
cons = queue->rx.req_cons; cons = queue->rx.req_cons;
return !queue->stalled return !queue->stalled && prod - cons < 1
&& prod - cons < XEN_NETBK_RX_SLOTS_MAX
&& time_after(jiffies, && time_after(jiffies,
queue->last_rx_time + queue->vif->stall_timeout); queue->last_rx_time + queue->vif->stall_timeout);
} }
...@@ -1951,14 +1961,13 @@ static bool xenvif_rx_queue_ready(struct xenvif_queue *queue) ...@@ -1951,14 +1961,13 @@ static bool xenvif_rx_queue_ready(struct xenvif_queue *queue)
prod = queue->rx.sring->req_prod; prod = queue->rx.sring->req_prod;
cons = queue->rx.req_cons; cons = queue->rx.req_cons;
return queue->stalled return queue->stalled && prod - cons >= 1;
&& prod - cons >= XEN_NETBK_RX_SLOTS_MAX;
} }
static bool xenvif_have_rx_work(struct xenvif_queue *queue) static bool xenvif_have_rx_work(struct xenvif_queue *queue)
{ {
return (!skb_queue_empty(&queue->rx_queue) return (!skb_queue_empty(&queue->rx_queue)
&& xenvif_rx_ring_slots_available(queue, XEN_NETBK_RX_SLOTS_MAX)) && xenvif_rx_ring_slots_available(queue))
|| (queue->vif->stall_timeout && || (queue->vif->stall_timeout &&
(xenvif_rx_queue_stalled(queue) (xenvif_rx_queue_stalled(queue)
|| xenvif_rx_queue_ready(queue))) || xenvif_rx_queue_ready(queue)))
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment