ibmvnic: Keep track of supplementary TX descriptors
authorThomas Falcon <tlfalcon@linux.vnet.ibm.com>
Sun, 18 Feb 2018 16:08:41 +0000 (10:08 -0600)
committerDavid S. Miller <davem@davemloft.net>
Tue, 20 Feb 2018 18:16:55 +0000 (13:16 -0500)
Supplementary TX descriptors were not being accounted for, which
was resulting in an overflow of the hardware device's transmit
queue. Keep track of those descriptors now when determining
how many entries remain on the TX queue.

Signed-off-by: Thomas Falcon <tlfalcon@linux.vnet.ibm.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
drivers/net/ethernet/ibm/ibmvnic.c
drivers/net/ethernet/ibm/ibmvnic.h

index 996f47568f9e3ba77f50668696cbd4e0979ceaa5..64770a1c406da0c0c5b69c5ec58ce6d1de7b7731 100644 (file)
@@ -1467,6 +1467,7 @@ static int ibmvnic_xmit(struct sk_buff *skb, struct net_device *netdev)
        if ((*hdrs >> 7) & 1) {
                build_hdr_descs_arr(tx_buff, &num_entries, *hdrs);
                tx_crq.v1.n_crq_elem = num_entries;
+               tx_buff->num_entries = num_entries;
                tx_buff->indir_arr[0] = tx_crq;
                tx_buff->indir_dma = dma_map_single(dev, tx_buff->indir_arr,
                                                    sizeof(tx_buff->indir_arr),
@@ -1515,7 +1516,7 @@ static int ibmvnic_xmit(struct sk_buff *skb, struct net_device *netdev)
                goto out;
        }
 
-       if (atomic_inc_return(&tx_scrq->used)
+       if (atomic_add_return(num_entries, &tx_scrq->used)
                                        >= adapter->req_tx_entries_per_subcrq) {
                netdev_info(netdev, "Stopping queue %d\n", queue_num);
                netif_stop_subqueue(netdev, queue_num);
@@ -2468,6 +2469,7 @@ static int ibmvnic_complete_tx(struct ibmvnic_adapter *adapter,
 restart_loop:
        while (pending_scrq(adapter, scrq)) {
                unsigned int pool = scrq->pool_index;
+               int num_entries = 0;
 
                next = ibmvnic_next_scrq(adapter, scrq);
                for (i = 0; i < next->tx_comp.num_comps; i++) {
@@ -2498,6 +2500,8 @@ restart_loop:
                                txbuff->skb = NULL;
                        }
 
+                       num_entries += txbuff->num_entries;
+
                        adapter->tx_pool[pool].free_map[adapter->tx_pool[pool].
                                                     producer_index] = index;
                        adapter->tx_pool[pool].producer_index =
@@ -2507,7 +2511,7 @@ restart_loop:
                /* remove tx_comp scrq*/
                next->tx_comp.first = 0;
 
-               if (atomic_sub_return(next->tx_comp.num_comps, &scrq->used) <=
+               if (atomic_sub_return(num_entries, &scrq->used) <=
                    (adapter->req_tx_entries_per_subcrq / 2) &&
                    __netif_subqueue_stopped(adapter->netdev,
                                             scrq->pool_index)) {
index fe21a6e2ddae0967fa5ff802209e8348b825da35..2f51458ccdc37f9069f58abf4a3cb98ec79ad1a7 100644 (file)
@@ -909,6 +909,7 @@ struct ibmvnic_tx_buff {
        union sub_crq indir_arr[6];
        u8 hdr_data[140];
        dma_addr_t indir_dma;
+       int num_entries;
 };
 
 struct ibmvnic_tx_pool {