Commit 531cda20 authored by Rob Herring's avatar Rob Herring Committed by David S. Miller

net: calxedaxgmac: fix rx DMA mapping API size mismatches

Fix the mismatch in the DMA mapping and unmapping sizes for receive. The
unmap size must be equal to the map size and should not be the actual
received frame length. The map size should also be adjusted by the
NET_IP_ALIGN size since the h/w buffer size (dma_buf_sz) includes this
offset.

Also, add a missing dma_mapping_error check in xgmac_rx_refill.
Reported-by: default avatarLennert Buytenhek <buytenh@wantstofly.org>
Signed-off-by: default avatarRob Herring <rob.herring@calxeda.com>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parent d2a5128f
...@@ -695,9 +695,14 @@ static void xgmac_rx_refill(struct xgmac_priv *priv) ...@@ -695,9 +695,14 @@ static void xgmac_rx_refill(struct xgmac_priv *priv)
if (unlikely(skb == NULL)) if (unlikely(skb == NULL))
break; break;
priv->rx_skbuff[entry] = skb;
paddr = dma_map_single(priv->device, skb->data, paddr = dma_map_single(priv->device, skb->data,
bufsz, DMA_FROM_DEVICE); priv->dma_buf_sz - NET_IP_ALIGN,
DMA_FROM_DEVICE);
if (dma_mapping_error(priv->device, paddr)) {
dev_kfree_skb_any(skb);
break;
}
priv->rx_skbuff[entry] = skb;
desc_set_buf_addr(p, paddr, priv->dma_buf_sz); desc_set_buf_addr(p, paddr, priv->dma_buf_sz);
} }
...@@ -794,13 +799,14 @@ static void xgmac_free_rx_skbufs(struct xgmac_priv *priv) ...@@ -794,13 +799,14 @@ static void xgmac_free_rx_skbufs(struct xgmac_priv *priv)
return; return;
for (i = 0; i < DMA_RX_RING_SZ; i++) { for (i = 0; i < DMA_RX_RING_SZ; i++) {
if (priv->rx_skbuff[i] == NULL) struct sk_buff *skb = priv->rx_skbuff[i];
if (skb == NULL)
continue; continue;
p = priv->dma_rx + i; p = priv->dma_rx + i;
dma_unmap_single(priv->device, desc_get_buf_addr(p), dma_unmap_single(priv->device, desc_get_buf_addr(p),
priv->dma_buf_sz, DMA_FROM_DEVICE); priv->dma_buf_sz - NET_IP_ALIGN, DMA_FROM_DEVICE);
dev_kfree_skb_any(priv->rx_skbuff[i]); dev_kfree_skb_any(skb);
priv->rx_skbuff[i] = NULL; priv->rx_skbuff[i] = NULL;
} }
} }
...@@ -1187,7 +1193,7 @@ static int xgmac_rx(struct xgmac_priv *priv, int limit) ...@@ -1187,7 +1193,7 @@ static int xgmac_rx(struct xgmac_priv *priv, int limit)
skb_put(skb, frame_len); skb_put(skb, frame_len);
dma_unmap_single(priv->device, desc_get_buf_addr(p), dma_unmap_single(priv->device, desc_get_buf_addr(p),
frame_len, DMA_FROM_DEVICE); priv->dma_buf_sz - NET_IP_ALIGN, DMA_FROM_DEVICE);
skb->protocol = eth_type_trans(skb, priv->dev); skb->protocol = eth_type_trans(skb, priv->dev);
skb->ip_summed = ip_checksum; skb->ip_summed = ip_checksum;
......
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