提交 d72e01a0 编写于 作者: B Benjamin Herrenschmidt 提交者: David S. Miller

ftgmac100: Use a scratch buffer for failed RX allocations

We can occasionally fail to allocate new RX buffers at
runtime or when starting the driver. At the moment the
latter just fails to open which is fine but the former
leaves stale DMA pointers in the ring.

Instead, use a scratch page and have all RX ring descriptors
point to it by default unless a proper buffer can be allocated.

It will help later on when re-initializing the whole ring
at runtime on link changes since there is no clean failure
path there unlike open().
Signed-off-by: NBenjamin Herrenschmidt <benh@kernel.crashing.org>
Signed-off-by: NDavid S. Miller <davem@davemloft.net>
上级 b1977bfb
...@@ -71,6 +71,10 @@ struct ftgmac100 { ...@@ -71,6 +71,10 @@ struct ftgmac100 {
u32 txdes0_edotr_mask; u32 txdes0_edotr_mask;
spinlock_t tx_lock; spinlock_t tx_lock;
/* Scratch page to use when rx skb alloc fails */
void *rx_scratch;
dma_addr_t rx_scratch_dma;
/* Component structures */ /* Component structures */
struct net_device *netdev; struct net_device *netdev;
struct device *dev; struct device *dev;
...@@ -404,12 +408,14 @@ static int ftgmac100_alloc_rx_page(struct ftgmac100 *priv, ...@@ -404,12 +408,14 @@ static int ftgmac100_alloc_rx_page(struct ftgmac100 *priv,
struct net_device *netdev = priv->netdev; struct net_device *netdev = priv->netdev;
struct page *page; struct page *page;
dma_addr_t map; dma_addr_t map;
int err;
page = alloc_page(gfp); page = alloc_page(gfp);
if (!page) { if (!page) {
if (net_ratelimit()) if (net_ratelimit())
netdev_err(netdev, "failed to allocate rx page\n"); netdev_err(netdev, "failed to allocate rx page\n");
return -ENOMEM; err = -ENOMEM;
map = priv->rx_scratch_dma;
} }
map = dma_map_page(priv->dev, page, 0, RX_BUF_SIZE, DMA_FROM_DEVICE); map = dma_map_page(priv->dev, page, 0, RX_BUF_SIZE, DMA_FROM_DEVICE);
...@@ -417,7 +423,9 @@ static int ftgmac100_alloc_rx_page(struct ftgmac100 *priv, ...@@ -417,7 +423,9 @@ static int ftgmac100_alloc_rx_page(struct ftgmac100 *priv,
if (net_ratelimit()) if (net_ratelimit())
netdev_err(netdev, "failed to map rx page\n"); netdev_err(netdev, "failed to map rx page\n");
__free_page(page); __free_page(page);
return -ENOMEM; err = -ENOMEM;
map = priv->rx_scratch_dma;
page = NULL;
} }
ftgmac100_rxdes_set_page(priv, rxdes, page); ftgmac100_rxdes_set_page(priv, rxdes, page);
...@@ -551,6 +559,16 @@ static bool ftgmac100_rx_packet(struct ftgmac100 *priv, int *processed) ...@@ -551,6 +559,16 @@ static bool ftgmac100_rx_packet(struct ftgmac100 *priv, int *processed)
return true; return true;
} }
/* If the packet had no buffer (failed to allocate earlier)
* then try to allocate one and skip
*/
page = ftgmac100_rxdes_get_page(priv, rxdes);
if (!page) {
ftgmac100_alloc_rx_page(priv, rxdes, GFP_ATOMIC);
ftgmac100_rx_pointer_advance(priv);
return true;
}
/* start processing */ /* start processing */
skb = netdev_alloc_skb_ip_align(netdev, 128); skb = netdev_alloc_skb_ip_align(netdev, 128);
if (unlikely(!skb)) { if (unlikely(!skb)) {
...@@ -854,6 +872,11 @@ static void ftgmac100_free_rings(struct ftgmac100 *priv) ...@@ -854,6 +872,11 @@ static void ftgmac100_free_rings(struct ftgmac100 *priv)
if (priv->descs) if (priv->descs)
dma_free_coherent(priv->dev, sizeof(struct ftgmac100_descs), dma_free_coherent(priv->dev, sizeof(struct ftgmac100_descs),
priv->descs, priv->descs_dma_addr); priv->descs, priv->descs_dma_addr);
/* Free scratch packet buffer */
if (priv->rx_scratch)
dma_free_coherent(priv->dev, RX_BUF_SIZE,
priv->rx_scratch, priv->rx_scratch_dma);
} }
static int ftgmac100_alloc_rings(struct ftgmac100 *priv) static int ftgmac100_alloc_rings(struct ftgmac100 *priv)
...@@ -865,6 +888,14 @@ static int ftgmac100_alloc_rings(struct ftgmac100 *priv) ...@@ -865,6 +888,14 @@ static int ftgmac100_alloc_rings(struct ftgmac100 *priv)
if (!priv->descs) if (!priv->descs)
return -ENOMEM; return -ENOMEM;
/* Allocate scratch packet buffer */
priv->rx_scratch = dma_alloc_coherent(priv->dev,
RX_BUF_SIZE,
&priv->rx_scratch_dma,
GFP_KERNEL);
if (!priv->rx_scratch)
return -ENOMEM;
return 0; return 0;
} }
...@@ -873,8 +904,11 @@ static void ftgmac100_init_rings(struct ftgmac100 *priv) ...@@ -873,8 +904,11 @@ static void ftgmac100_init_rings(struct ftgmac100 *priv)
int i; int i;
/* Initialize RX ring */ /* Initialize RX ring */
for (i = 0; i < RX_QUEUE_ENTRIES; i++) for (i = 0; i < RX_QUEUE_ENTRIES; i++) {
priv->descs->rxdes[i].rxdes0 = 0; struct ftgmac100_rxdes *rxdes = &priv->descs->rxdes[i];
ftgmac100_rxdes_set_dma_addr(rxdes, priv->rx_scratch_dma);
rxdes->rxdes0 = 0;
}
ftgmac100_rxdes_set_end_of_ring(priv, &priv->descs->rxdes[i - 1]); ftgmac100_rxdes_set_end_of_ring(priv, &priv->descs->rxdes[i - 1]);
/* Initialize TX ring */ /* Initialize TX ring */
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册