staging/rdma/hfi1: Add coalescing support for SDMA TX descriptors

This fixes transmit errors when the number of scatter gather elements in the
request is more that the number of per packet descriptors supported by the
hardware, allocate and coalesce the extra scatter gather elements into a single
buffer. The last descriptor is reserved and used for this coalesced buffer.

Verbs potentially need this support when transferring small data chunks
involving different memory regions.

Reviewed-by: Mike Marciniszyn <mike.marciniszyn@intel.com>
Reviewed-by: Mitko Haralanov <mitko.haralanov@intel.com>
Signed-off-by: Niranjana Vishwanathapura <niranjana.vishwanathapura@intel.com>
Signed-off-by: Ira Weiny <ira.weiny@intel.com>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
This commit is contained in:
Niranjana Vishwanathapura 2015-10-26 10:28:32 -04:00 committed by Greg Kroah-Hartman
parent 3c2f85b8ce
commit f4d26d81ad
2 changed files with 168 additions and 30 deletions

View File

@ -55,6 +55,7 @@
#include <linux/bitops.h> #include <linux/bitops.h>
#include <linux/timer.h> #include <linux/timer.h>
#include <linux/vmalloc.h> #include <linux/vmalloc.h>
#include <linux/highmem.h>
#include "hfi.h" #include "hfi.h"
#include "common.h" #include "common.h"
@ -2705,26 +2706,133 @@ static void __sdma_process_event(struct sdma_engine *sde,
* *
* The code will bump the allocation up to the max * The code will bump the allocation up to the max
* of MAX_DESC (64) descriptors. There doesn't seem * of MAX_DESC (64) descriptors. There doesn't seem
* much point in an interim step. * much point in an interim step. The last descriptor
* is reserved for coalesce buffer in order to support
* cases where input packet has >MAX_DESC iovecs.
* *
*/ */
int _extend_sdma_tx_descs(struct hfi1_devdata *dd, struct sdma_txreq *tx) static int _extend_sdma_tx_descs(struct hfi1_devdata *dd, struct sdma_txreq *tx)
{ {
int i; int i;
/* Handle last descriptor */
if (unlikely((tx->num_desc == (MAX_DESC - 1)))) {
/* if tlen is 0, it is for padding, release last descriptor */
if (!tx->tlen) {
tx->desc_limit = MAX_DESC;
} else if (!tx->coalesce_buf) {
/* allocate coalesce buffer with space for padding */
tx->coalesce_buf = kmalloc(tx->tlen + sizeof(u32),
GFP_ATOMIC);
if (!tx->coalesce_buf)
return -ENOMEM;
tx->coalesce_idx = 0;
}
return 0;
}
if (unlikely(tx->num_desc == MAX_DESC))
return -ENOMEM;
tx->descp = kmalloc_array( tx->descp = kmalloc_array(
MAX_DESC, MAX_DESC,
sizeof(struct sdma_desc), sizeof(struct sdma_desc),
GFP_ATOMIC); GFP_ATOMIC);
if (!tx->descp) if (!tx->descp)
return -ENOMEM; return -ENOMEM;
tx->desc_limit = MAX_DESC;
/* reserve last descriptor for coalescing */
tx->desc_limit = MAX_DESC - 1;
/* copy ones already built */ /* copy ones already built */
for (i = 0; i < tx->num_desc; i++) for (i = 0; i < tx->num_desc; i++)
tx->descp[i] = tx->descs[i]; tx->descp[i] = tx->descs[i];
return 0; return 0;
} }
/*
* ext_coal_sdma_tx_descs() - extend or coalesce sdma tx descriptors
*
* This is called once the initial nominal allocation of descriptors
* in the sdma_txreq is exhausted.
*
* This function calls _extend_sdma_tx_descs to extend or allocate
* coalesce buffer. If there is a allocated coalesce buffer, it will
* copy the input packet data into the coalesce buffer. It also adds
* coalesce buffer descriptor once whe whole packet is received.
*
* Return:
* <0 - error
* 0 - coalescing, don't populate descriptor
* 1 - continue with populating descriptor
*/
int ext_coal_sdma_tx_descs(struct hfi1_devdata *dd, struct sdma_txreq *tx,
int type, void *kvaddr, struct page *page,
unsigned long offset, u16 len)
{
int pad_len, rval;
dma_addr_t addr;
rval = _extend_sdma_tx_descs(dd, tx);
if (rval) {
sdma_txclean(dd, tx);
return rval;
}
/* If coalesce buffer is allocated, copy data into it */
if (tx->coalesce_buf) {
if (type == SDMA_MAP_NONE) {
sdma_txclean(dd, tx);
return -EINVAL;
}
if (type == SDMA_MAP_PAGE) {
kvaddr = kmap(page);
kvaddr += offset;
} else if (WARN_ON(!kvaddr)) {
sdma_txclean(dd, tx);
return -EINVAL;
}
memcpy(tx->coalesce_buf + tx->coalesce_idx, kvaddr, len);
tx->coalesce_idx += len;
if (type == SDMA_MAP_PAGE)
kunmap(page);
/* If there is more data, return */
if (tx->tlen - tx->coalesce_idx)
return 0;
/* Whole packet is received; add any padding */
pad_len = tx->packet_len & (sizeof(u32) - 1);
if (pad_len) {
pad_len = sizeof(u32) - pad_len;
memset(tx->coalesce_buf + tx->coalesce_idx, 0, pad_len);
/* padding is taken care of for coalescing case */
tx->packet_len += pad_len;
tx->tlen += pad_len;
}
/* dma map the coalesce buffer */
addr = dma_map_single(&dd->pcidev->dev,
tx->coalesce_buf,
tx->tlen,
DMA_TO_DEVICE);
if (unlikely(dma_mapping_error(&dd->pcidev->dev, addr))) {
sdma_txclean(dd, tx);
return -ENOSPC;
}
/* Add descriptor for coalesce buffer */
tx->desc_limit = MAX_DESC;
return _sdma_txadd_daddr(dd, SDMA_MAP_SINGLE, tx,
addr, tx->tlen);
}
return 1;
}
/* Update sdes when the lmc changes */ /* Update sdes when the lmc changes */
void sdma_update_lmc(struct hfi1_devdata *dd, u64 mask, u32 lid) void sdma_update_lmc(struct hfi1_devdata *dd, u64 mask, u32 lid)
{ {
@ -2750,13 +2858,15 @@ int _pad_sdma_tx_descs(struct hfi1_devdata *dd, struct sdma_txreq *tx)
{ {
int rval = 0; int rval = 0;
tx->num_desc++;
if ((unlikely(tx->num_desc == tx->desc_limit))) { if ((unlikely(tx->num_desc == tx->desc_limit))) {
rval = _extend_sdma_tx_descs(dd, tx); rval = _extend_sdma_tx_descs(dd, tx);
if (rval) if (rval) {
sdma_txclean(dd, tx);
return rval; return rval;
} }
}
/* finish the one just added */ /* finish the one just added */
tx->num_desc++;
make_tx_sdma_desc( make_tx_sdma_desc(
tx, tx,
SDMA_MAP_NONE, SDMA_MAP_NONE,

View File

@ -352,6 +352,8 @@ struct sdma_txreq {
/* private: */ /* private: */
void *coalesce_buf; void *coalesce_buf;
/* private: */ /* private: */
u16 coalesce_idx;
/* private: */
struct iowait *wait; struct iowait *wait;
/* private: */ /* private: */
callback_t complete; callback_t complete;
@ -735,7 +737,9 @@ static inline void make_tx_sdma_desc(
} }
/* helper to extend txreq */ /* helper to extend txreq */
int _extend_sdma_tx_descs(struct hfi1_devdata *, struct sdma_txreq *); int ext_coal_sdma_tx_descs(struct hfi1_devdata *dd, struct sdma_txreq *tx,
int type, void *kvaddr, struct page *page,
unsigned long offset, u16 len);
int _pad_sdma_tx_descs(struct hfi1_devdata *, struct sdma_txreq *); int _pad_sdma_tx_descs(struct hfi1_devdata *, struct sdma_txreq *);
void sdma_txclean(struct hfi1_devdata *, struct sdma_txreq *); void sdma_txclean(struct hfi1_devdata *, struct sdma_txreq *);
@ -762,11 +766,6 @@ static inline int _sdma_txadd_daddr(
{ {
int rval = 0; int rval = 0;
if ((unlikely(tx->num_desc == tx->desc_limit))) {
rval = _extend_sdma_tx_descs(dd, tx);
if (rval)
return rval;
}
make_tx_sdma_desc( make_tx_sdma_desc(
tx, tx,
type, type,
@ -798,9 +797,7 @@ static inline int _sdma_txadd_daddr(
* *
* Return: * Return:
* 0 - success, -ENOSPC - mapping fail, -ENOMEM - couldn't * 0 - success, -ENOSPC - mapping fail, -ENOMEM - couldn't
* extend descriptor array or couldn't allocate coalesce * extend/coalesce descriptor array
* buffer.
*
*/ */
static inline int sdma_txadd_page( static inline int sdma_txadd_page(
struct hfi1_devdata *dd, struct hfi1_devdata *dd,
@ -809,17 +806,28 @@ static inline int sdma_txadd_page(
unsigned long offset, unsigned long offset,
u16 len) u16 len)
{ {
dma_addr_t addr = dma_addr_t addr;
dma_map_page( int rval;
if ((unlikely(tx->num_desc == tx->desc_limit))) {
rval = ext_coal_sdma_tx_descs(dd, tx, SDMA_MAP_PAGE,
NULL, page, offset, len);
if (rval <= 0)
return rval;
}
addr = dma_map_page(
&dd->pcidev->dev, &dd->pcidev->dev,
page, page,
offset, offset,
len, len,
DMA_TO_DEVICE); DMA_TO_DEVICE);
if (unlikely(dma_mapping_error(&dd->pcidev->dev, addr))) { if (unlikely(dma_mapping_error(&dd->pcidev->dev, addr))) {
sdma_txclean(dd, tx); sdma_txclean(dd, tx);
return -ENOSPC; return -ENOSPC;
} }
return _sdma_txadd_daddr( return _sdma_txadd_daddr(
dd, SDMA_MAP_PAGE, tx, addr, len); dd, SDMA_MAP_PAGE, tx, addr, len);
} }
@ -846,6 +854,15 @@ static inline int sdma_txadd_daddr(
dma_addr_t addr, dma_addr_t addr,
u16 len) u16 len)
{ {
int rval;
if ((unlikely(tx->num_desc == tx->desc_limit))) {
rval = ext_coal_sdma_tx_descs(dd, tx, SDMA_MAP_NONE,
NULL, NULL, 0, 0);
if (rval <= 0)
return rval;
}
return _sdma_txadd_daddr(dd, SDMA_MAP_NONE, tx, addr, len); return _sdma_txadd_daddr(dd, SDMA_MAP_NONE, tx, addr, len);
} }
@ -862,7 +879,7 @@ static inline int sdma_txadd_daddr(
* The mapping/unmapping of the kvaddr and len is automatically handled. * The mapping/unmapping of the kvaddr and len is automatically handled.
* *
* Return: * Return:
* 0 - success, -ENOSPC - mapping fail, -ENOMEM - couldn't extend * 0 - success, -ENOSPC - mapping fail, -ENOMEM - couldn't extend/coalesce
* descriptor array * descriptor array
*/ */
static inline int sdma_txadd_kvaddr( static inline int sdma_txadd_kvaddr(
@ -871,16 +888,27 @@ static inline int sdma_txadd_kvaddr(
void *kvaddr, void *kvaddr,
u16 len) u16 len)
{ {
dma_addr_t addr = dma_addr_t addr;
dma_map_single( int rval;
if ((unlikely(tx->num_desc == tx->desc_limit))) {
rval = ext_coal_sdma_tx_descs(dd, tx, SDMA_MAP_SINGLE,
kvaddr, NULL, 0, len);
if (rval <= 0)
return rval;
}
addr = dma_map_single(
&dd->pcidev->dev, &dd->pcidev->dev,
kvaddr, kvaddr,
len, len,
DMA_TO_DEVICE); DMA_TO_DEVICE);
if (unlikely(dma_mapping_error(&dd->pcidev->dev, addr))) { if (unlikely(dma_mapping_error(&dd->pcidev->dev, addr))) {
sdma_txclean(dd, tx); sdma_txclean(dd, tx);
return -ENOSPC; return -ENOSPC;
} }
return _sdma_txadd_daddr( return _sdma_txadd_daddr(
dd, SDMA_MAP_SINGLE, tx, addr, len); dd, SDMA_MAP_SINGLE, tx, addr, len);
} }