staging/rdma/hfi1: fix sdma build failures to always clean up
authorMike Marciniszyn <mike.marciniszyn@intel.com>
Thu, 3 Dec 2015 21:41:05 +0000 (16:41 -0500)
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>
Mon, 21 Dec 2015 22:00:17 +0000 (14:00 -0800)
There are holes in the sdma build support routines that do
not clean any partially built sdma descriptors after mapping or
allocate failures.

This patch corrects these issues.

Reviewed-by: Dennis Dalessandro <dennis.dalessandro@intel.com>
Signed-off-by: Mike Marciniszyn <mike.marciniszyn@intel.com>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
drivers/staging/rdma/hfi1/sdma.c
drivers/staging/rdma/hfi1/sdma.h

index 0710e2a..9a15f1f 100644 (file)
@@ -2731,22 +2731,21 @@ static int _extend_sdma_tx_descs(struct hfi1_devdata *dd, struct sdma_txreq *tx)
                        tx->coalesce_buf = kmalloc(tx->tlen + sizeof(u32),
                                                   GFP_ATOMIC);
                        if (!tx->coalesce_buf)
-                               return -ENOMEM;
-
+                               goto enomem;
                        tx->coalesce_idx = 0;
                }
                return 0;
        }
 
        if (unlikely(tx->num_desc == MAX_DESC))
-               return -ENOMEM;
+               goto enomem;
 
        tx->descp = kmalloc_array(
                        MAX_DESC,
                        sizeof(struct sdma_desc),
                        GFP_ATOMIC);
        if (!tx->descp)
-               return -ENOMEM;
+               goto enomem;
 
        /* reserve last descriptor for coalescing */
        tx->desc_limit = MAX_DESC - 1;
@@ -2754,6 +2753,9 @@ static int _extend_sdma_tx_descs(struct hfi1_devdata *dd, struct sdma_txreq *tx)
        for (i = 0; i < tx->num_desc; i++)
                tx->descp[i] = tx->descs[i];
        return 0;
+enomem:
+       sdma_txclean(dd, tx);
+       return -ENOMEM;
 }
 
 /*
index 85701ee..da89e64 100644 (file)
@@ -774,10 +774,13 @@ static inline int _sdma_txadd_daddr(
        tx->tlen -= len;
        /* special cases for last */
        if (!tx->tlen) {
-               if (tx->packet_len & (sizeof(u32) - 1))
+               if (tx->packet_len & (sizeof(u32) - 1)) {
                        rval = _pad_sdma_tx_descs(dd, tx);
-               else
+                       if (rval)
+                               return rval;
+               } else {
                        _sdma_close_tx(dd, tx);
+               }
        }
        tx->num_desc++;
        return rval;