Skip to content

Commit 7b70176

Browse files
edumazetdavem330
authored andcommitted
atl1c: Fix misuse of netdev_alloc_skb in refilling rx ring
On Mon, 2013-07-29 at 08:30 -0700, Eric Dumazet wrote: > On Mon, 2013-07-29 at 13:09 +0100, Luis Henriques wrote: > > > > > I confirm that I can't reproduce the issue using this patch. > > > > Thanks, I'll send a polished patch, as this one had an error if > build_skb() returns NULL (in case sk_buff allocation fails) Please try the following patch : It should use 2K frags instead of 4K for normal 1500 mtu Thanks ! [PATCH] atl1c: use custom skb allocator We had reports ( https://bugzilla.kernel.org/show_bug.cgi?id=54021 ) that using high order pages for skb allocations is problematic for atl1c We do not know exactly what the problem is, but we suspect that crossing 4K pages is not well supported by this hardware. Use a custom allocator, using page allocator and 2K fragments for optimal stack behavior. We might make this allocator generic in future kernels. Signed-off-by: Eric Dumazet <[email protected]> Cc: Luis Henriques <[email protected]> Cc: Neil Horman <[email protected]> Signed-off-by: David S. Miller <[email protected]>
1 parent e1ee367 commit 7b70176

File tree

2 files changed

+42
-1
lines changed

2 files changed

+42
-1
lines changed

drivers/net/ethernet/atheros/atl1c/atl1c.h

Lines changed: 3 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -520,6 +520,9 @@ struct atl1c_adapter {
520520
struct net_device *netdev;
521521
struct pci_dev *pdev;
522522
struct napi_struct napi;
523+
struct page *rx_page;
524+
unsigned int rx_page_offset;
525+
unsigned int rx_frag_size;
523526
struct atl1c_hw hw;
524527
struct atl1c_hw_stats hw_stats;
525528
struct mii_if_info mii; /* MII interface info */

drivers/net/ethernet/atheros/atl1c/atl1c_main.c

Lines changed: 39 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -481,10 +481,15 @@ static int atl1c_set_mac_addr(struct net_device *netdev, void *p)
481481
static void atl1c_set_rxbufsize(struct atl1c_adapter *adapter,
482482
struct net_device *dev)
483483
{
484+
unsigned int head_size;
484485
int mtu = dev->mtu;
485486

486487
adapter->rx_buffer_len = mtu > AT_RX_BUF_SIZE ?
487488
roundup(mtu + ETH_HLEN + ETH_FCS_LEN + VLAN_HLEN, 8) : AT_RX_BUF_SIZE;
489+
490+
head_size = SKB_DATA_ALIGN(adapter->rx_buffer_len + NET_SKB_PAD) +
491+
SKB_DATA_ALIGN(sizeof(struct skb_shared_info));
492+
adapter->rx_frag_size = roundup_pow_of_two(head_size);
488493
}
489494

490495
static netdev_features_t atl1c_fix_features(struct net_device *netdev,
@@ -952,6 +957,10 @@ static void atl1c_free_ring_resources(struct atl1c_adapter *adapter)
952957
kfree(adapter->tpd_ring[0].buffer_info);
953958
adapter->tpd_ring[0].buffer_info = NULL;
954959
}
960+
if (adapter->rx_page) {
961+
put_page(adapter->rx_page);
962+
adapter->rx_page = NULL;
963+
}
955964
}
956965

957966
/**
@@ -1639,6 +1648,35 @@ static inline void atl1c_rx_checksum(struct atl1c_adapter *adapter,
16391648
skb_checksum_none_assert(skb);
16401649
}
16411650

1651+
static struct sk_buff *atl1c_alloc_skb(struct atl1c_adapter *adapter)
1652+
{
1653+
struct sk_buff *skb;
1654+
struct page *page;
1655+
1656+
if (adapter->rx_frag_size > PAGE_SIZE)
1657+
return netdev_alloc_skb(adapter->netdev,
1658+
adapter->rx_buffer_len);
1659+
1660+
page = adapter->rx_page;
1661+
if (!page) {
1662+
adapter->rx_page = page = alloc_page(GFP_ATOMIC);
1663+
if (unlikely(!page))
1664+
return NULL;
1665+
adapter->rx_page_offset = 0;
1666+
}
1667+
1668+
skb = build_skb(page_address(page) + adapter->rx_page_offset,
1669+
adapter->rx_frag_size);
1670+
if (likely(skb)) {
1671+
adapter->rx_page_offset += adapter->rx_frag_size;
1672+
if (adapter->rx_page_offset >= PAGE_SIZE)
1673+
adapter->rx_page = NULL;
1674+
else
1675+
get_page(page);
1676+
}
1677+
return skb;
1678+
}
1679+
16421680
static int atl1c_alloc_rx_buffer(struct atl1c_adapter *adapter)
16431681
{
16441682
struct atl1c_rfd_ring *rfd_ring = &adapter->rfd_ring;
@@ -1660,7 +1698,7 @@ static int atl1c_alloc_rx_buffer(struct atl1c_adapter *adapter)
16601698
while (next_info->flags & ATL1C_BUFFER_FREE) {
16611699
rfd_desc = ATL1C_RFD_DESC(rfd_ring, rfd_next_to_use);
16621700

1663-
skb = netdev_alloc_skb(adapter->netdev, adapter->rx_buffer_len);
1701+
skb = atl1c_alloc_skb(adapter);
16641702
if (unlikely(!skb)) {
16651703
if (netif_msg_rx_err(adapter))
16661704
dev_warn(&pdev->dev, "alloc rx buffer failed\n");

0 commit comments

Comments
 (0)