Commit 18f9f0ac authored by Lendacky, Thomas's avatar Lendacky, Thomas Committed by David S. Miller

amd-xgbe: Add NUMA affinity support for memory allocations

Add support to perform memory allocations on the node of the device. The
original allocation or the ring structure and Tx/Rx queues allocated all
of the memory at once and then carved it up for each channel and queue.
To best ensure that we get as much memory from the NUMA node as we can,
break the channel and ring allocations into individual allocations.
Signed-off-by: default avatarTom Lendacky <thomas.lendacky@amd.com>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parent 85b85c85
...@@ -176,8 +176,8 @@ static void xgbe_free_ring_resources(struct xgbe_prv_data *pdata) ...@@ -176,8 +176,8 @@ static void xgbe_free_ring_resources(struct xgbe_prv_data *pdata)
DBGPR("-->xgbe_free_ring_resources\n"); DBGPR("-->xgbe_free_ring_resources\n");
channel = pdata->channel; for (i = 0; i < pdata->channel_count; i++) {
for (i = 0; i < pdata->channel_count; i++, channel++) { channel = pdata->channel[i];
xgbe_free_ring(pdata, channel->tx_ring); xgbe_free_ring(pdata, channel->tx_ring);
xgbe_free_ring(pdata, channel->rx_ring); xgbe_free_ring(pdata, channel->rx_ring);
} }
...@@ -185,34 +185,60 @@ static void xgbe_free_ring_resources(struct xgbe_prv_data *pdata) ...@@ -185,34 +185,60 @@ static void xgbe_free_ring_resources(struct xgbe_prv_data *pdata)
DBGPR("<--xgbe_free_ring_resources\n"); DBGPR("<--xgbe_free_ring_resources\n");
} }
static void *xgbe_alloc_node(size_t size, int node)
{
void *mem;
mem = kzalloc_node(size, GFP_KERNEL, node);
if (!mem)
mem = kzalloc(size, GFP_KERNEL);
return mem;
}
static void *xgbe_dma_alloc_node(struct device *dev, size_t size,
dma_addr_t *dma, int node)
{
void *mem;
int cur_node = dev_to_node(dev);
set_dev_node(dev, node);
mem = dma_alloc_coherent(dev, size, dma, GFP_KERNEL);
set_dev_node(dev, cur_node);
if (!mem)
mem = dma_alloc_coherent(dev, size, dma, GFP_KERNEL);
return mem;
}
static int xgbe_init_ring(struct xgbe_prv_data *pdata, static int xgbe_init_ring(struct xgbe_prv_data *pdata,
struct xgbe_ring *ring, unsigned int rdesc_count) struct xgbe_ring *ring, unsigned int rdesc_count)
{ {
DBGPR("-->xgbe_init_ring\n"); size_t size;
if (!ring) if (!ring)
return 0; return 0;
/* Descriptors */ /* Descriptors */
size = rdesc_count * sizeof(struct xgbe_ring_desc);
ring->rdesc_count = rdesc_count; ring->rdesc_count = rdesc_count;
ring->rdesc = dma_alloc_coherent(pdata->dev, ring->rdesc = xgbe_dma_alloc_node(pdata->dev, size, &ring->rdesc_dma,
(sizeof(struct xgbe_ring_desc) * ring->node);
rdesc_count), &ring->rdesc_dma,
GFP_KERNEL);
if (!ring->rdesc) if (!ring->rdesc)
return -ENOMEM; return -ENOMEM;
/* Descriptor information */ /* Descriptor information */
ring->rdata = kcalloc(rdesc_count, sizeof(struct xgbe_ring_data), size = rdesc_count * sizeof(struct xgbe_ring_data);
GFP_KERNEL);
ring->rdata = xgbe_alloc_node(size, ring->node);
if (!ring->rdata) if (!ring->rdata)
return -ENOMEM; return -ENOMEM;
netif_dbg(pdata, drv, pdata->netdev, netif_dbg(pdata, drv, pdata->netdev,
"rdesc=%p, rdesc_dma=%pad, rdata=%p\n", "rdesc=%p, rdesc_dma=%pad, rdata=%p, node=%d\n",
ring->rdesc, &ring->rdesc_dma, ring->rdata); ring->rdesc, &ring->rdesc_dma, ring->rdata, ring->node);
DBGPR("<--xgbe_init_ring\n");
return 0; return 0;
} }
...@@ -223,10 +249,8 @@ static int xgbe_alloc_ring_resources(struct xgbe_prv_data *pdata) ...@@ -223,10 +249,8 @@ static int xgbe_alloc_ring_resources(struct xgbe_prv_data *pdata)
unsigned int i; unsigned int i;
int ret; int ret;
DBGPR("-->xgbe_alloc_ring_resources\n"); for (i = 0; i < pdata->channel_count; i++) {
channel = pdata->channel[i];
channel = pdata->channel;
for (i = 0; i < pdata->channel_count; i++, channel++) {
netif_dbg(pdata, drv, pdata->netdev, "%s - Tx ring:\n", netif_dbg(pdata, drv, pdata->netdev, "%s - Tx ring:\n",
channel->name); channel->name);
...@@ -250,8 +274,6 @@ static int xgbe_alloc_ring_resources(struct xgbe_prv_data *pdata) ...@@ -250,8 +274,6 @@ static int xgbe_alloc_ring_resources(struct xgbe_prv_data *pdata)
} }
} }
DBGPR("<--xgbe_alloc_ring_resources\n");
return 0; return 0;
err_ring: err_ring:
...@@ -261,21 +283,33 @@ static int xgbe_alloc_ring_resources(struct xgbe_prv_data *pdata) ...@@ -261,21 +283,33 @@ static int xgbe_alloc_ring_resources(struct xgbe_prv_data *pdata)
} }
static int xgbe_alloc_pages(struct xgbe_prv_data *pdata, static int xgbe_alloc_pages(struct xgbe_prv_data *pdata,
struct xgbe_page_alloc *pa, gfp_t gfp, int order) struct xgbe_page_alloc *pa, int alloc_order,
int node)
{ {
struct page *pages = NULL; struct page *pages = NULL;
dma_addr_t pages_dma; dma_addr_t pages_dma;
int ret; gfp_t gfp;
int order, ret;
again:
order = alloc_order;
/* Try to obtain pages, decreasing order if necessary */ /* Try to obtain pages, decreasing order if necessary */
gfp |= __GFP_COLD | __GFP_COMP | __GFP_NOWARN; gfp = GFP_ATOMIC | __GFP_COLD | __GFP_COMP | __GFP_NOWARN;
while (order >= 0) { while (order >= 0) {
pages = alloc_pages(gfp, order); pages = alloc_pages_node(node, gfp, order);
if (pages) if (pages)
break; break;
order--; order--;
} }
/* If we couldn't get local pages, try getting from anywhere */
if (!pages && (node != NUMA_NO_NODE)) {
node = NUMA_NO_NODE;
goto again;
}
if (!pages) if (!pages)
return -ENOMEM; return -ENOMEM;
...@@ -327,14 +361,14 @@ static int xgbe_map_rx_buffer(struct xgbe_prv_data *pdata, ...@@ -327,14 +361,14 @@ static int xgbe_map_rx_buffer(struct xgbe_prv_data *pdata,
int ret; int ret;
if (!ring->rx_hdr_pa.pages) { if (!ring->rx_hdr_pa.pages) {
ret = xgbe_alloc_pages(pdata, &ring->rx_hdr_pa, GFP_ATOMIC, 0); ret = xgbe_alloc_pages(pdata, &ring->rx_hdr_pa, 0, ring->node);
if (ret) if (ret)
return ret; return ret;
} }
if (!ring->rx_buf_pa.pages) { if (!ring->rx_buf_pa.pages) {
ret = xgbe_alloc_pages(pdata, &ring->rx_buf_pa, GFP_ATOMIC, ret = xgbe_alloc_pages(pdata, &ring->rx_buf_pa,
PAGE_ALLOC_COSTLY_ORDER); PAGE_ALLOC_COSTLY_ORDER, ring->node);
if (ret) if (ret)
return ret; return ret;
} }
...@@ -362,8 +396,8 @@ static void xgbe_wrapper_tx_descriptor_init(struct xgbe_prv_data *pdata) ...@@ -362,8 +396,8 @@ static void xgbe_wrapper_tx_descriptor_init(struct xgbe_prv_data *pdata)
DBGPR("-->xgbe_wrapper_tx_descriptor_init\n"); DBGPR("-->xgbe_wrapper_tx_descriptor_init\n");
channel = pdata->channel; for (i = 0; i < pdata->channel_count; i++) {
for (i = 0; i < pdata->channel_count; i++, channel++) { channel = pdata->channel[i];
ring = channel->tx_ring; ring = channel->tx_ring;
if (!ring) if (!ring)
break; break;
...@@ -403,8 +437,8 @@ static void xgbe_wrapper_rx_descriptor_init(struct xgbe_prv_data *pdata) ...@@ -403,8 +437,8 @@ static void xgbe_wrapper_rx_descriptor_init(struct xgbe_prv_data *pdata)
DBGPR("-->xgbe_wrapper_rx_descriptor_init\n"); DBGPR("-->xgbe_wrapper_rx_descriptor_init\n");
channel = pdata->channel; for (i = 0; i < pdata->channel_count; i++) {
for (i = 0; i < pdata->channel_count; i++, channel++) { channel = pdata->channel[i];
ring = channel->rx_ring; ring = channel->rx_ring;
if (!ring) if (!ring)
break; break;
......
...@@ -176,12 +176,10 @@ static unsigned int xgbe_riwt_to_usec(struct xgbe_prv_data *pdata, ...@@ -176,12 +176,10 @@ static unsigned int xgbe_riwt_to_usec(struct xgbe_prv_data *pdata,
static int xgbe_config_pblx8(struct xgbe_prv_data *pdata) static int xgbe_config_pblx8(struct xgbe_prv_data *pdata)
{ {
struct xgbe_channel *channel;
unsigned int i; unsigned int i;
channel = pdata->channel; for (i = 0; i < pdata->channel_count; i++)
for (i = 0; i < pdata->channel_count; i++, channel++) XGMAC_DMA_IOWRITE_BITS(pdata->channel[i], DMA_CH_CR, PBLX8,
XGMAC_DMA_IOWRITE_BITS(channel, DMA_CH_CR, PBLX8,
pdata->pblx8); pdata->pblx8);
return 0; return 0;
...@@ -189,20 +187,18 @@ static int xgbe_config_pblx8(struct xgbe_prv_data *pdata) ...@@ -189,20 +187,18 @@ static int xgbe_config_pblx8(struct xgbe_prv_data *pdata)
static int xgbe_get_tx_pbl_val(struct xgbe_prv_data *pdata) static int xgbe_get_tx_pbl_val(struct xgbe_prv_data *pdata)
{ {
return XGMAC_DMA_IOREAD_BITS(pdata->channel, DMA_CH_TCR, PBL); return XGMAC_DMA_IOREAD_BITS(pdata->channel[0], DMA_CH_TCR, PBL);
} }
static int xgbe_config_tx_pbl_val(struct xgbe_prv_data *pdata) static int xgbe_config_tx_pbl_val(struct xgbe_prv_data *pdata)
{ {
struct xgbe_channel *channel;
unsigned int i; unsigned int i;
channel = pdata->channel; for (i = 0; i < pdata->channel_count; i++) {
for (i = 0; i < pdata->channel_count; i++, channel++) { if (!pdata->channel[i]->tx_ring)
if (!channel->tx_ring)
break; break;
XGMAC_DMA_IOWRITE_BITS(channel, DMA_CH_TCR, PBL, XGMAC_DMA_IOWRITE_BITS(pdata->channel[i], DMA_CH_TCR, PBL,
pdata->tx_pbl); pdata->tx_pbl);
} }
...@@ -211,20 +207,18 @@ static int xgbe_config_tx_pbl_val(struct xgbe_prv_data *pdata) ...@@ -211,20 +207,18 @@ static int xgbe_config_tx_pbl_val(struct xgbe_prv_data *pdata)
static int xgbe_get_rx_pbl_val(struct xgbe_prv_data *pdata) static int xgbe_get_rx_pbl_val(struct xgbe_prv_data *pdata)
{ {
return XGMAC_DMA_IOREAD_BITS(pdata->channel, DMA_CH_RCR, PBL); return XGMAC_DMA_IOREAD_BITS(pdata->channel[0], DMA_CH_RCR, PBL);
} }
static int xgbe_config_rx_pbl_val(struct xgbe_prv_data *pdata) static int xgbe_config_rx_pbl_val(struct xgbe_prv_data *pdata)
{ {
struct xgbe_channel *channel;
unsigned int i; unsigned int i;
channel = pdata->channel; for (i = 0; i < pdata->channel_count; i++) {
for (i = 0; i < pdata->channel_count; i++, channel++) { if (!pdata->channel[i]->rx_ring)
if (!channel->rx_ring)
break; break;
XGMAC_DMA_IOWRITE_BITS(channel, DMA_CH_RCR, PBL, XGMAC_DMA_IOWRITE_BITS(pdata->channel[i], DMA_CH_RCR, PBL,
pdata->rx_pbl); pdata->rx_pbl);
} }
...@@ -233,15 +227,13 @@ static int xgbe_config_rx_pbl_val(struct xgbe_prv_data *pdata) ...@@ -233,15 +227,13 @@ static int xgbe_config_rx_pbl_val(struct xgbe_prv_data *pdata)
static int xgbe_config_osp_mode(struct xgbe_prv_data *pdata) static int xgbe_config_osp_mode(struct xgbe_prv_data *pdata)
{ {
struct xgbe_channel *channel;
unsigned int i; unsigned int i;
channel = pdata->channel; for (i = 0; i < pdata->channel_count; i++) {
for (i = 0; i < pdata->channel_count; i++, channel++) { if (!pdata->channel[i]->tx_ring)
if (!channel->tx_ring)
break; break;
XGMAC_DMA_IOWRITE_BITS(channel, DMA_CH_TCR, OSP, XGMAC_DMA_IOWRITE_BITS(pdata->channel[i], DMA_CH_TCR, OSP,
pdata->tx_osp_mode); pdata->tx_osp_mode);
} }
...@@ -292,15 +284,13 @@ static int xgbe_config_tx_threshold(struct xgbe_prv_data *pdata, ...@@ -292,15 +284,13 @@ static int xgbe_config_tx_threshold(struct xgbe_prv_data *pdata,
static int xgbe_config_rx_coalesce(struct xgbe_prv_data *pdata) static int xgbe_config_rx_coalesce(struct xgbe_prv_data *pdata)
{ {
struct xgbe_channel *channel;
unsigned int i; unsigned int i;
channel = pdata->channel; for (i = 0; i < pdata->channel_count; i++) {
for (i = 0; i < pdata->channel_count; i++, channel++) { if (!pdata->channel[i]->rx_ring)
if (!channel->rx_ring)
break; break;
XGMAC_DMA_IOWRITE_BITS(channel, DMA_CH_RIWT, RWT, XGMAC_DMA_IOWRITE_BITS(pdata->channel[i], DMA_CH_RIWT, RWT,
pdata->rx_riwt); pdata->rx_riwt);
} }
...@@ -314,44 +304,38 @@ static int xgbe_config_tx_coalesce(struct xgbe_prv_data *pdata) ...@@ -314,44 +304,38 @@ static int xgbe_config_tx_coalesce(struct xgbe_prv_data *pdata)
static void xgbe_config_rx_buffer_size(struct xgbe_prv_data *pdata) static void xgbe_config_rx_buffer_size(struct xgbe_prv_data *pdata)
{ {
struct xgbe_channel *channel;
unsigned int i; unsigned int i;
channel = pdata->channel; for (i = 0; i < pdata->channel_count; i++) {
for (i = 0; i < pdata->channel_count; i++, channel++) { if (!pdata->channel[i]->rx_ring)
if (!channel->rx_ring)
break; break;
XGMAC_DMA_IOWRITE_BITS(channel, DMA_CH_RCR, RBSZ, XGMAC_DMA_IOWRITE_BITS(pdata->channel[i], DMA_CH_RCR, RBSZ,
pdata->rx_buf_size); pdata->rx_buf_size);
} }
} }
static void xgbe_config_tso_mode(struct xgbe_prv_data *pdata) static void xgbe_config_tso_mode(struct xgbe_prv_data *pdata)
{ {
struct xgbe_channel *channel;
unsigned int i; unsigned int i;
channel = pdata->channel; for (i = 0; i < pdata->channel_count; i++) {
for (i = 0; i < pdata->channel_count; i++, channel++) { if (!pdata->channel[i]->tx_ring)
if (!channel->tx_ring)
break; break;
XGMAC_DMA_IOWRITE_BITS(channel, DMA_CH_TCR, TSE, 1); XGMAC_DMA_IOWRITE_BITS(pdata->channel[i], DMA_CH_TCR, TSE, 1);
} }
} }
static void xgbe_config_sph_mode(struct xgbe_prv_data *pdata) static void xgbe_config_sph_mode(struct xgbe_prv_data *pdata)
{ {
struct xgbe_channel *channel;
unsigned int i; unsigned int i;
channel = pdata->channel; for (i = 0; i < pdata->channel_count; i++) {
for (i = 0; i < pdata->channel_count; i++, channel++) { if (!pdata->channel[i]->rx_ring)
if (!channel->rx_ring)
break; break;
XGMAC_DMA_IOWRITE_BITS(channel, DMA_CH_CR, SPH, 1); XGMAC_DMA_IOWRITE_BITS(pdata->channel[i], DMA_CH_CR, SPH, 1);
} }
XGMAC_IOWRITE_BITS(pdata, MAC_RCR, HDSMS, XGBE_SPH_HDSMS_SIZE); XGMAC_IOWRITE_BITS(pdata, MAC_RCR, HDSMS, XGBE_SPH_HDSMS_SIZE);
...@@ -651,8 +635,9 @@ static void xgbe_enable_dma_interrupts(struct xgbe_prv_data *pdata) ...@@ -651,8 +635,9 @@ static void xgbe_enable_dma_interrupts(struct xgbe_prv_data *pdata)
XGMAC_IOWRITE_BITS(pdata, DMA_MR, INTM, XGMAC_IOWRITE_BITS(pdata, DMA_MR, INTM,
pdata->channel_irq_mode); pdata->channel_irq_mode);
channel = pdata->channel; for (i = 0; i < pdata->channel_count; i++) {
for (i = 0; i < pdata->channel_count; i++, channel++) { channel = pdata->channel[i];
/* Clear all the interrupts which are set */ /* Clear all the interrupts which are set */
dma_ch_isr = XGMAC_DMA_IOREAD(channel, DMA_CH_SR); dma_ch_isr = XGMAC_DMA_IOREAD(channel, DMA_CH_SR);
XGMAC_DMA_IOWRITE(channel, DMA_CH_SR, dma_ch_isr); XGMAC_DMA_IOWRITE(channel, DMA_CH_SR, dma_ch_isr);
...@@ -3213,16 +3198,14 @@ static void xgbe_prepare_tx_stop(struct xgbe_prv_data *pdata, ...@@ -3213,16 +3198,14 @@ static void xgbe_prepare_tx_stop(struct xgbe_prv_data *pdata,
static void xgbe_enable_tx(struct xgbe_prv_data *pdata) static void xgbe_enable_tx(struct xgbe_prv_data *pdata)
{ {
struct xgbe_channel *channel;
unsigned int i; unsigned int i;
/* Enable each Tx DMA channel */ /* Enable each Tx DMA channel */
channel = pdata->channel; for (i = 0; i < pdata->channel_count; i++) {
for (i = 0; i < pdata->channel_count; i++, channel++) { if (!pdata->channel[i]->tx_ring)
if (!channel->tx_ring)
break; break;
XGMAC_DMA_IOWRITE_BITS(channel, DMA_CH_TCR, ST, 1); XGMAC_DMA_IOWRITE_BITS(pdata->channel[i], DMA_CH_TCR, ST, 1);
} }
/* Enable each Tx queue */ /* Enable each Tx queue */
...@@ -3236,7 +3219,6 @@ static void xgbe_enable_tx(struct xgbe_prv_data *pdata) ...@@ -3236,7 +3219,6 @@ static void xgbe_enable_tx(struct xgbe_prv_data *pdata)
static void xgbe_disable_tx(struct xgbe_prv_data *pdata) static void xgbe_disable_tx(struct xgbe_prv_data *pdata)
{ {
struct xgbe_channel *channel;
unsigned int i; unsigned int i;
/* Prepare for Tx DMA channel stop */ /* Prepare for Tx DMA channel stop */
...@@ -3251,12 +3233,11 @@ static void xgbe_disable_tx(struct xgbe_prv_data *pdata) ...@@ -3251,12 +3233,11 @@ static void xgbe_disable_tx(struct xgbe_prv_data *pdata)
XGMAC_MTL_IOWRITE_BITS(pdata, i, MTL_Q_TQOMR, TXQEN, 0); XGMAC_MTL_IOWRITE_BITS(pdata, i, MTL_Q_TQOMR, TXQEN, 0);
/* Disable each Tx DMA channel */ /* Disable each Tx DMA channel */
channel = pdata->channel; for (i = 0; i < pdata->channel_count; i++) {
for (i = 0; i < pdata->channel_count; i++, channel++) { if (!pdata->channel[i]->tx_ring)
if (!channel->tx_ring)
break; break;
XGMAC_DMA_IOWRITE_BITS(channel, DMA_CH_TCR, ST, 0); XGMAC_DMA_IOWRITE_BITS(pdata->channel[i], DMA_CH_TCR, ST, 0);
} }
} }
...@@ -3288,16 +3269,14 @@ static void xgbe_prepare_rx_stop(struct xgbe_prv_data *pdata, ...@@ -3288,16 +3269,14 @@ static void xgbe_prepare_rx_stop(struct xgbe_prv_data *pdata,
static void xgbe_enable_rx(struct xgbe_prv_data *pdata) static void xgbe_enable_rx(struct xgbe_prv_data *pdata)
{ {
struct xgbe_channel *channel;
unsigned int reg_val, i; unsigned int reg_val, i;
/* Enable each Rx DMA channel */ /* Enable each Rx DMA channel */
channel = pdata->channel; for (i = 0; i < pdata->channel_count; i++) {
for (i = 0; i < pdata->channel_count; i++, channel++) { if (!pdata->channel[i]->rx_ring)
if (!channel->rx_ring)
break; break;
XGMAC_DMA_IOWRITE_BITS(channel, DMA_CH_RCR, SR, 1); XGMAC_DMA_IOWRITE_BITS(pdata->channel[i], DMA_CH_RCR, SR, 1);
} }
/* Enable each Rx queue */ /* Enable each Rx queue */
...@@ -3315,7 +3294,6 @@ static void xgbe_enable_rx(struct xgbe_prv_data *pdata) ...@@ -3315,7 +3294,6 @@ static void xgbe_enable_rx(struct xgbe_prv_data *pdata)
static void xgbe_disable_rx(struct xgbe_prv_data *pdata) static void xgbe_disable_rx(struct xgbe_prv_data *pdata)
{ {
struct xgbe_channel *channel;
unsigned int i; unsigned int i;
/* Disable MAC Rx */ /* Disable MAC Rx */
...@@ -3332,27 +3310,24 @@ static void xgbe_disable_rx(struct xgbe_prv_data *pdata) ...@@ -3332,27 +3310,24 @@ static void xgbe_disable_rx(struct xgbe_prv_data *pdata)
XGMAC_IOWRITE(pdata, MAC_RQC0R, 0); XGMAC_IOWRITE(pdata, MAC_RQC0R, 0);
/* Disable each Rx DMA channel */ /* Disable each Rx DMA channel */
channel = pdata->channel; for (i = 0; i < pdata->channel_count; i++) {
for (i = 0; i < pdata->channel_count; i++, channel++) { if (!pdata->channel[i]->rx_ring)
if (!channel->rx_ring)
break; break;
XGMAC_DMA_IOWRITE_BITS(channel, DMA_CH_RCR, SR, 0); XGMAC_DMA_IOWRITE_BITS(pdata->channel[i], DMA_CH_RCR, SR, 0);
} }
} }
static void xgbe_powerup_tx(struct xgbe_prv_data *pdata) static void xgbe_powerup_tx(struct xgbe_prv_data *pdata)
{ {
struct xgbe_channel *channel;
unsigned int i; unsigned int i;
/* Enable each Tx DMA channel */ /* Enable each Tx DMA channel */
channel = pdata->channel; for (i = 0; i < pdata->channel_count; i++) {
for (i = 0; i < pdata->channel_count; i++, channel++) { if (!pdata->channel[i]->tx_ring)
if (!channel->tx_ring)
break; break;
XGMAC_DMA_IOWRITE_BITS(channel, DMA_CH_TCR, ST, 1); XGMAC_DMA_IOWRITE_BITS(pdata->channel[i], DMA_CH_TCR, ST, 1);
} }
/* Enable MAC Tx */ /* Enable MAC Tx */
...@@ -3361,7 +3336,6 @@ static void xgbe_powerup_tx(struct xgbe_prv_data *pdata) ...@@ -3361,7 +3336,6 @@ static void xgbe_powerup_tx(struct xgbe_prv_data *pdata)
static void xgbe_powerdown_tx(struct xgbe_prv_data *pdata) static void xgbe_powerdown_tx(struct xgbe_prv_data *pdata)
{ {
struct xgbe_channel *channel;
unsigned int i; unsigned int i;
/* Prepare for Tx DMA channel stop */ /* Prepare for Tx DMA channel stop */
...@@ -3372,42 +3346,37 @@ static void xgbe_powerdown_tx(struct xgbe_prv_data *pdata) ...@@ -3372,42 +3346,37 @@ static void xgbe_powerdown_tx(struct xgbe_prv_data *pdata)
XGMAC_IOWRITE_BITS(pdata, MAC_TCR, TE, 0); XGMAC_IOWRITE_BITS(pdata, MAC_TCR, TE, 0);
/* Disable each Tx DMA channel */ /* Disable each Tx DMA channel */
channel = pdata->channel; for (i = 0; i < pdata->channel_count; i++) {
for (i = 0; i < pdata->channel_count; i++, channel++) { if (!pdata->channel[i]->tx_ring)
if (!channel->tx_ring)
break; break;
XGMAC_DMA_IOWRITE_BITS(channel, DMA_CH_TCR, ST, 0); XGMAC_DMA_IOWRITE_BITS(pdata->channel[i], DMA_CH_TCR, ST, 0);
} }
} }
static void xgbe_powerup_rx(struct xgbe_prv_data *pdata) static void xgbe_powerup_rx(struct xgbe_prv_data *pdata)
{ {
struct xgbe_channel *channel;
unsigned int i; unsigned int i;
/* Enable each Rx DMA channel */ /* Enable each Rx DMA channel */
channel = pdata->channel; for (i = 0; i < pdata->channel_count; i++) {
for (i = 0; i < pdata->channel_count; i++, channel++) { if (!pdata->channel[i]->rx_ring)
if (!channel->rx_ring)
break; break;
XGMAC_DMA_IOWRITE_BITS(channel, DMA_CH_RCR, SR, 1); XGMAC_DMA_IOWRITE_BITS(pdata->channel[i], DMA_CH_RCR, SR, 1);
} }
} }
static void xgbe_powerdown_rx(struct xgbe_prv_data *pdata) static void xgbe_powerdown_rx(struct xgbe_prv_data *pdata)
{ {
struct xgbe_channel *channel;
unsigned int i; unsigned int i;
/* Disable each Rx DMA channel */ /* Disable each Rx DMA channel */
channel = pdata->channel; for (i = 0; i < pdata->channel_count; i++) {
for (i = 0; i < pdata->channel_count; i++, channel++) { if (!pdata->channel[i]->rx_ring)
if (!channel->rx_ring)
break; break;
XGMAC_DMA_IOWRITE_BITS(channel, DMA_CH_RCR, SR, 0); XGMAC_DMA_IOWRITE_BITS(pdata->channel[i], DMA_CH_RCR, SR, 0);
} }
} }
......
...@@ -158,81 +158,100 @@ static int xgbe_one_poll(struct napi_struct *, int); ...@@ -158,81 +158,100 @@ static int xgbe_one_poll(struct napi_struct *, int);
static int xgbe_all_poll(struct napi_struct *, int); static int xgbe_all_poll(struct napi_struct *, int);
static void xgbe_stop(struct xgbe_prv_data *); static void xgbe_stop(struct xgbe_prv_data *);
static int xgbe_alloc_channels(struct xgbe_prv_data *pdata) static void *xgbe_alloc_node(size_t size, int node)
{ {
struct xgbe_channel *channel_mem, *channel; void *mem;
struct xgbe_ring *tx_ring, *rx_ring;
unsigned int count, i;
int ret = -ENOMEM;
count = max_t(unsigned int, pdata->tx_ring_count, pdata->rx_ring_count); mem = kzalloc_node(size, GFP_KERNEL, node);
if (!mem)
mem = kzalloc(size, GFP_KERNEL);
channel_mem = kcalloc(count, sizeof(struct xgbe_channel), GFP_KERNEL); return mem;
if (!channel_mem) }
goto err_channel;
tx_ring = kcalloc(pdata->tx_ring_count, sizeof(struct xgbe_ring), static void xgbe_free_channels(struct xgbe_prv_data *pdata)
GFP_KERNEL); {
if (!tx_ring) unsigned int i;
goto err_tx_ring;
for (i = 0; i < ARRAY_SIZE(pdata->channel); i++) {
if (!pdata->channel[i])
continue;
rx_ring = kcalloc(pdata->rx_ring_count, sizeof(struct xgbe_ring), kfree(pdata->channel[i]->rx_ring);
GFP_KERNEL); kfree(pdata->channel[i]->tx_ring);
if (!rx_ring) kfree(pdata->channel[i]);
goto err_rx_ring;
pdata->channel[i] = NULL;
}
pdata->channel_count = 0;
}
static int xgbe_alloc_channels(struct xgbe_prv_data *pdata)
{
struct xgbe_channel *channel;
struct xgbe_ring *ring;
unsigned int count, i;
int node;
node = dev_to_node(pdata->dev);
count = max_t(unsigned int, pdata->tx_ring_count, pdata->rx_ring_count);
for (i = 0; i < count; i++) {
channel = xgbe_alloc_node(sizeof(*channel), node);
if (!channel)
goto err_mem;
pdata->channel[i] = channel;
for (i = 0, channel = channel_mem; i < count; i++, channel++) {
snprintf(channel->name, sizeof(channel->name), "channel-%u", i); snprintf(channel->name, sizeof(channel->name), "channel-%u", i);
channel->pdata = pdata; channel->pdata = pdata;
channel->queue_index = i; channel->queue_index = i;
channel->dma_regs = pdata->xgmac_regs + DMA_CH_BASE + channel->dma_regs = pdata->xgmac_regs + DMA_CH_BASE +
(DMA_CH_INC * i); (DMA_CH_INC * i);
channel->node = node;
if (pdata->per_channel_irq) if (pdata->per_channel_irq)
channel->dma_irq = pdata->channel_irq[i]; channel->dma_irq = pdata->channel_irq[i];
if (i < pdata->tx_ring_count) { if (i < pdata->tx_ring_count) {
spin_lock_init(&tx_ring->lock); ring = xgbe_alloc_node(sizeof(*ring), node);
channel->tx_ring = tx_ring++; if (!ring)
goto err_mem;
spin_lock_init(&ring->lock);
ring->node = node;
channel->tx_ring = ring;
} }
if (i < pdata->rx_ring_count) { if (i < pdata->rx_ring_count) {
spin_lock_init(&rx_ring->lock); ring = xgbe_alloc_node(sizeof(*ring), node);
channel->rx_ring = rx_ring++; if (!ring)
goto err_mem;
spin_lock_init(&ring->lock);
ring->node = node;
channel->rx_ring = ring;
} }
netif_dbg(pdata, drv, pdata->netdev,
"%s: node=%d\n", channel->name, node);
netif_dbg(pdata, drv, pdata->netdev, netif_dbg(pdata, drv, pdata->netdev,
"%s: dma_regs=%p, dma_irq=%d, tx=%p, rx=%p\n", "%s: dma_regs=%p, dma_irq=%d, tx=%p, rx=%p\n",
channel->name, channel->dma_regs, channel->dma_irq, channel->name, channel->dma_regs, channel->dma_irq,
channel->tx_ring, channel->rx_ring); channel->tx_ring, channel->rx_ring);
} }
pdata->channel = channel_mem;
pdata->channel_count = count; pdata->channel_count = count;
return 0; return 0;
err_rx_ring: err_mem:
kfree(tx_ring); xgbe_free_channels(pdata);
err_tx_ring:
kfree(channel_mem);
err_channel:
return ret;
}
static void xgbe_free_channels(struct xgbe_prv_data *pdata)
{
if (!pdata->channel)
return;
kfree(pdata->channel->rx_ring);
kfree(pdata->channel->tx_ring);
kfree(pdata->channel);
pdata->channel = NULL; return -ENOMEM;
pdata->channel_count = 0;
} }
static inline unsigned int xgbe_tx_avail_desc(struct xgbe_ring *ring) static inline unsigned int xgbe_tx_avail_desc(struct xgbe_ring *ring)
...@@ -301,12 +320,10 @@ static void xgbe_enable_rx_tx_int(struct xgbe_prv_data *pdata, ...@@ -301,12 +320,10 @@ static void xgbe_enable_rx_tx_int(struct xgbe_prv_data *pdata,
static void xgbe_enable_rx_tx_ints(struct xgbe_prv_data *pdata) static void xgbe_enable_rx_tx_ints(struct xgbe_prv_data *pdata)
{ {
struct xgbe_channel *channel;
unsigned int i; unsigned int i;
channel = pdata->channel; for (i = 0; i < pdata->channel_count; i++)
for (i = 0; i < pdata->channel_count; i++, channel++) xgbe_enable_rx_tx_int(pdata, pdata->channel[i]);
xgbe_enable_rx_tx_int(pdata, channel);
} }
static void xgbe_disable_rx_tx_int(struct xgbe_prv_data *pdata, static void xgbe_disable_rx_tx_int(struct xgbe_prv_data *pdata,
...@@ -329,12 +346,10 @@ static void xgbe_disable_rx_tx_int(struct xgbe_prv_data *pdata, ...@@ -329,12 +346,10 @@ static void xgbe_disable_rx_tx_int(struct xgbe_prv_data *pdata,
static void xgbe_disable_rx_tx_ints(struct xgbe_prv_data *pdata) static void xgbe_disable_rx_tx_ints(struct xgbe_prv_data *pdata)
{ {
struct xgbe_channel *channel;
unsigned int i; unsigned int i;
channel = pdata->channel; for (i = 0; i < pdata->channel_count; i++)
for (i = 0; i < pdata->channel_count; i++, channel++) xgbe_disable_rx_tx_int(pdata, pdata->channel[i]);
xgbe_disable_rx_tx_int(pdata, channel);
} }
static bool xgbe_ecc_sec(struct xgbe_prv_data *pdata, unsigned long *period, static bool xgbe_ecc_sec(struct xgbe_prv_data *pdata, unsigned long *period,
...@@ -475,7 +490,7 @@ static void xgbe_isr_task(unsigned long data) ...@@ -475,7 +490,7 @@ static void xgbe_isr_task(unsigned long data)
if (!(dma_isr & (1 << i))) if (!(dma_isr & (1 << i)))
continue; continue;
channel = pdata->channel + i; channel = pdata->channel[i];
dma_ch_isr = XGMAC_DMA_IOREAD(channel, DMA_CH_SR); dma_ch_isr = XGMAC_DMA_IOREAD(channel, DMA_CH_SR);
netif_dbg(pdata, intr, pdata->netdev, "DMA_CH%u_ISR=%#010x\n", netif_dbg(pdata, intr, pdata->netdev, "DMA_CH%u_ISR=%#010x\n",
...@@ -675,8 +690,8 @@ static void xgbe_init_timers(struct xgbe_prv_data *pdata) ...@@ -675,8 +690,8 @@ static void xgbe_init_timers(struct xgbe_prv_data *pdata)
setup_timer(&pdata->service_timer, xgbe_service_timer, setup_timer(&pdata->service_timer, xgbe_service_timer,
(unsigned long)pdata); (unsigned long)pdata);
channel = pdata->channel; for (i = 0; i < pdata->channel_count; i++) {
for (i = 0; i < pdata->channel_count; i++, channel++) { channel = pdata->channel[i];
if (!channel->tx_ring) if (!channel->tx_ring)
break; break;
...@@ -697,8 +712,8 @@ static void xgbe_stop_timers(struct xgbe_prv_data *pdata) ...@@ -697,8 +712,8 @@ static void xgbe_stop_timers(struct xgbe_prv_data *pdata)
del_timer_sync(&pdata->service_timer); del_timer_sync(&pdata->service_timer);
channel = pdata->channel; for (i = 0; i < pdata->channel_count; i++) {
for (i = 0; i < pdata->channel_count; i++, channel++) { channel = pdata->channel[i];
if (!channel->tx_ring) if (!channel->tx_ring)
break; break;
...@@ -816,8 +831,8 @@ static void xgbe_napi_enable(struct xgbe_prv_data *pdata, unsigned int add) ...@@ -816,8 +831,8 @@ static void xgbe_napi_enable(struct xgbe_prv_data *pdata, unsigned int add)
unsigned int i; unsigned int i;
if (pdata->per_channel_irq) { if (pdata->per_channel_irq) {
channel = pdata->channel; for (i = 0; i < pdata->channel_count; i++) {
for (i = 0; i < pdata->channel_count; i++, channel++) { channel = pdata->channel[i];
if (add) if (add)
netif_napi_add(pdata->netdev, &channel->napi, netif_napi_add(pdata->netdev, &channel->napi,
xgbe_one_poll, NAPI_POLL_WEIGHT); xgbe_one_poll, NAPI_POLL_WEIGHT);
...@@ -839,8 +854,8 @@ static void xgbe_napi_disable(struct xgbe_prv_data *pdata, unsigned int del) ...@@ -839,8 +854,8 @@ static void xgbe_napi_disable(struct xgbe_prv_data *pdata, unsigned int del)
unsigned int i; unsigned int i;
if (pdata->per_channel_irq) { if (pdata->per_channel_irq) {
channel = pdata->channel; for (i = 0; i < pdata->channel_count; i++) {
for (i = 0; i < pdata->channel_count; i++, channel++) { channel = pdata->channel[i];
napi_disable(&channel->napi); napi_disable(&channel->napi);
if (del) if (del)
...@@ -886,8 +901,8 @@ static int xgbe_request_irqs(struct xgbe_prv_data *pdata) ...@@ -886,8 +901,8 @@ static int xgbe_request_irqs(struct xgbe_prv_data *pdata)
if (!pdata->per_channel_irq) if (!pdata->per_channel_irq)
return 0; return 0;
channel = pdata->channel; for (i = 0; i < pdata->channel_count; i++) {
for (i = 0; i < pdata->channel_count; i++, channel++) { channel = pdata->channel[i];
snprintf(channel->dma_irq_name, snprintf(channel->dma_irq_name,
sizeof(channel->dma_irq_name) - 1, sizeof(channel->dma_irq_name) - 1,
"%s-TxRx-%u", netdev_name(netdev), "%s-TxRx-%u", netdev_name(netdev),
...@@ -907,8 +922,11 @@ static int xgbe_request_irqs(struct xgbe_prv_data *pdata) ...@@ -907,8 +922,11 @@ static int xgbe_request_irqs(struct xgbe_prv_data *pdata)
err_dma_irq: err_dma_irq:
/* Using an unsigned int, 'i' will go to UINT_MAX and exit */ /* Using an unsigned int, 'i' will go to UINT_MAX and exit */
for (i--, channel--; i < pdata->channel_count; i--, channel--) for (i--; i < pdata->channel_count; i--) {
channel = pdata->channel[i];
devm_free_irq(pdata->dev, channel->dma_irq, channel); devm_free_irq(pdata->dev, channel->dma_irq, channel);
}
if (pdata->vdata->ecc_support && (pdata->dev_irq != pdata->ecc_irq)) if (pdata->vdata->ecc_support && (pdata->dev_irq != pdata->ecc_irq))
devm_free_irq(pdata->dev, pdata->ecc_irq, pdata); devm_free_irq(pdata->dev, pdata->ecc_irq, pdata);
...@@ -932,9 +950,10 @@ static void xgbe_free_irqs(struct xgbe_prv_data *pdata) ...@@ -932,9 +950,10 @@ static void xgbe_free_irqs(struct xgbe_prv_data *pdata)
if (!pdata->per_channel_irq) if (!pdata->per_channel_irq)
return; return;
channel = pdata->channel; for (i = 0; i < pdata->channel_count; i++) {
for (i = 0; i < pdata->channel_count; i++, channel++) channel = pdata->channel[i];
devm_free_irq(pdata->dev, channel->dma_irq, channel); devm_free_irq(pdata->dev, channel->dma_irq, channel);
}
} }
void xgbe_init_tx_coalesce(struct xgbe_prv_data *pdata) void xgbe_init_tx_coalesce(struct xgbe_prv_data *pdata)
...@@ -969,16 +988,14 @@ void xgbe_init_rx_coalesce(struct xgbe_prv_data *pdata) ...@@ -969,16 +988,14 @@ void xgbe_init_rx_coalesce(struct xgbe_prv_data *pdata)
static void xgbe_free_tx_data(struct xgbe_prv_data *pdata) static void xgbe_free_tx_data(struct xgbe_prv_data *pdata)
{ {
struct xgbe_desc_if *desc_if = &pdata->desc_if; struct xgbe_desc_if *desc_if = &pdata->desc_if;
struct xgbe_channel *channel;
struct xgbe_ring *ring; struct xgbe_ring *ring;
struct xgbe_ring_data *rdata; struct xgbe_ring_data *rdata;
unsigned int i, j; unsigned int i, j;
DBGPR("-->xgbe_free_tx_data\n"); DBGPR("-->xgbe_free_tx_data\n");
channel = pdata->channel; for (i = 0; i < pdata->channel_count; i++) {
for (i = 0; i < pdata->channel_count; i++, channel++) { ring = pdata->channel[i]->tx_ring;
ring = channel->tx_ring;
if (!ring) if (!ring)
break; break;
...@@ -994,16 +1011,14 @@ static void xgbe_free_tx_data(struct xgbe_prv_data *pdata) ...@@ -994,16 +1011,14 @@ static void xgbe_free_tx_data(struct xgbe_prv_data *pdata)
static void xgbe_free_rx_data(struct xgbe_prv_data *pdata) static void xgbe_free_rx_data(struct xgbe_prv_data *pdata)
{ {
struct xgbe_desc_if *desc_if = &pdata->desc_if; struct xgbe_desc_if *desc_if = &pdata->desc_if;
struct xgbe_channel *channel;
struct xgbe_ring *ring; struct xgbe_ring *ring;
struct xgbe_ring_data *rdata; struct xgbe_ring_data *rdata;
unsigned int i, j; unsigned int i, j;
DBGPR("-->xgbe_free_rx_data\n"); DBGPR("-->xgbe_free_rx_data\n");
channel = pdata->channel; for (i = 0; i < pdata->channel_count; i++) {
for (i = 0; i < pdata->channel_count; i++, channel++) { ring = pdata->channel[i]->rx_ring;
ring = channel->rx_ring;
if (!ring) if (!ring)
break; break;
...@@ -1179,8 +1194,8 @@ static void xgbe_stop(struct xgbe_prv_data *pdata) ...@@ -1179,8 +1194,8 @@ static void xgbe_stop(struct xgbe_prv_data *pdata)
hw_if->exit(pdata); hw_if->exit(pdata);
channel = pdata->channel; for (i = 0; i < pdata->channel_count; i++) {
for (i = 0; i < pdata->channel_count; i++, channel++) { channel = pdata->channel[i];
if (!channel->tx_ring) if (!channel->tx_ring)
continue; continue;
...@@ -1667,7 +1682,7 @@ static int xgbe_xmit(struct sk_buff *skb, struct net_device *netdev) ...@@ -1667,7 +1682,7 @@ static int xgbe_xmit(struct sk_buff *skb, struct net_device *netdev)
DBGPR("-->xgbe_xmit: skb->len = %d\n", skb->len); DBGPR("-->xgbe_xmit: skb->len = %d\n", skb->len);
channel = pdata->channel + skb->queue_mapping; channel = pdata->channel[skb->queue_mapping];
txq = netdev_get_tx_queue(netdev, channel->queue_index); txq = netdev_get_tx_queue(netdev, channel->queue_index);
ring = channel->tx_ring; ring = channel->tx_ring;
packet = &ring->packet_data; packet = &ring->packet_data;
...@@ -1877,9 +1892,10 @@ static void xgbe_poll_controller(struct net_device *netdev) ...@@ -1877,9 +1892,10 @@ static void xgbe_poll_controller(struct net_device *netdev)
DBGPR("-->xgbe_poll_controller\n"); DBGPR("-->xgbe_poll_controller\n");
if (pdata->per_channel_irq) { if (pdata->per_channel_irq) {
channel = pdata->channel; for (i = 0; i < pdata->channel_count; i++) {
for (i = 0; i < pdata->channel_count; i++, channel++) channel = pdata->channel[i];
xgbe_dma_isr(channel->dma_irq, channel); xgbe_dma_isr(channel->dma_irq, channel);
}
} else { } else {
disable_irq(pdata->dev_irq); disable_irq(pdata->dev_irq);
xgbe_isr(pdata->dev_irq, pdata); xgbe_isr(pdata->dev_irq, pdata);
...@@ -2372,8 +2388,9 @@ static int xgbe_all_poll(struct napi_struct *napi, int budget) ...@@ -2372,8 +2388,9 @@ static int xgbe_all_poll(struct napi_struct *napi, int budget)
do { do {
last_processed = processed; last_processed = processed;
channel = pdata->channel; for (i = 0; i < pdata->channel_count; i++) {
for (i = 0; i < pdata->channel_count; i++, channel++) { channel = pdata->channel[i];
/* Cleanup Tx ring first */ /* Cleanup Tx ring first */
xgbe_tx_poll(channel); xgbe_tx_poll(channel);
......
...@@ -412,6 +412,7 @@ struct xgbe_ring { ...@@ -412,6 +412,7 @@ struct xgbe_ring {
/* Page allocation for RX buffers */ /* Page allocation for RX buffers */
struct xgbe_page_alloc rx_hdr_pa; struct xgbe_page_alloc rx_hdr_pa;
struct xgbe_page_alloc rx_buf_pa; struct xgbe_page_alloc rx_buf_pa;
int node;
/* Ring index values /* Ring index values
* cur - Tx: index of descriptor to be used for current transfer * cur - Tx: index of descriptor to be used for current transfer
...@@ -462,6 +463,8 @@ struct xgbe_channel { ...@@ -462,6 +463,8 @@ struct xgbe_channel {
struct xgbe_ring *tx_ring; struct xgbe_ring *tx_ring;
struct xgbe_ring *rx_ring; struct xgbe_ring *rx_ring;
int node;
} ____cacheline_aligned; } ____cacheline_aligned;
enum xgbe_state { enum xgbe_state {
...@@ -1012,7 +1015,7 @@ struct xgbe_prv_data { ...@@ -1012,7 +1015,7 @@ struct xgbe_prv_data {
struct timer_list service_timer; struct timer_list service_timer;
/* Rings for Tx/Rx on a DMA channel */ /* Rings for Tx/Rx on a DMA channel */
struct xgbe_channel *channel; struct xgbe_channel *channel[XGBE_MAX_DMA_CHANNELS];
unsigned int tx_max_channel_count; unsigned int tx_max_channel_count;
unsigned int rx_max_channel_count; unsigned int rx_max_channel_count;
unsigned int channel_count; unsigned int channel_count;
......
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