mirror of
https://github.com/edk2-porting/linux-next.git
synced 2024-11-18 15:44:02 +08:00
enc28j60: Fix sporadic packet loss (corrected again)
Packet data read from the RX buffer the when the RSV is at the end of the RX buffer does not warp around. This causes packet loss, as the actual data is never read. Fix this by calculating the right packet data location. Thanks to Shachar Shemesh for suggesting the fix. Signed-off-by: Baruch Siach <baruch@tkos.co.il> Acked-by: Claudio Lanconelli <lanconelli.claudio@eptar.com> Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
parent
bd0914104c
commit
5176da7e53
@ -568,6 +568,17 @@ static u16 erxrdpt_workaround(u16 next_packet_ptr, u16 start, u16 end)
|
|||||||
return erxrdpt;
|
return erxrdpt;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Calculate wrap around when reading beyond the end of the RX buffer
|
||||||
|
*/
|
||||||
|
static u16 rx_packet_start(u16 ptr)
|
||||||
|
{
|
||||||
|
if (ptr + RSV_SIZE > RXEND_INIT)
|
||||||
|
return (ptr + RSV_SIZE) - (RXEND_INIT - RXSTART_INIT + 1);
|
||||||
|
else
|
||||||
|
return ptr + RSV_SIZE;
|
||||||
|
}
|
||||||
|
|
||||||
static void nolock_rxfifo_init(struct enc28j60_net *priv, u16 start, u16 end)
|
static void nolock_rxfifo_init(struct enc28j60_net *priv, u16 start, u16 end)
|
||||||
{
|
{
|
||||||
u16 erxrdpt;
|
u16 erxrdpt;
|
||||||
@ -938,8 +949,9 @@ static void enc28j60_hw_rx(struct net_device *ndev)
|
|||||||
skb->dev = ndev;
|
skb->dev = ndev;
|
||||||
skb_reserve(skb, NET_IP_ALIGN);
|
skb_reserve(skb, NET_IP_ALIGN);
|
||||||
/* copy the packet from the receive buffer */
|
/* copy the packet from the receive buffer */
|
||||||
enc28j60_mem_read(priv, priv->next_pk_ptr + sizeof(rsv),
|
enc28j60_mem_read(priv,
|
||||||
len, skb_put(skb, len));
|
rx_packet_start(priv->next_pk_ptr),
|
||||||
|
len, skb_put(skb, len));
|
||||||
if (netif_msg_pktdata(priv))
|
if (netif_msg_pktdata(priv))
|
||||||
dump_packet(__func__, skb->len, skb->data);
|
dump_packet(__func__, skb->len, skb->data);
|
||||||
skb->protocol = eth_type_trans(skb, ndev);
|
skb->protocol = eth_type_trans(skb, ndev);
|
||||||
|
Loading…
Reference in New Issue
Block a user