ixgbe: combine ixgbe_add_rx_frag and ixgbe_can_reuse_page

This patch combines ixgbe_add_rx_frag and ixgbe_can_reuse_page into a
single function. The main motivation behind this is to make better use of
the values so that we don't have to load them from memory and into
registers twice.

Signed-off-by: Alexander Duyck <alexander.h.duyck@intel.com>
Tested-by: Phil Schmitt <phillip.j.schmitt@intel.com>
Signed-off-by: Peter P Waskiewicz Jr <peter.p.waskiewicz.jr@intel.com>
This commit is contained in:
Alexander Duyck 2012-07-20 08:08:18 +00:00 committed by Peter P Waskiewicz Jr
parent afaa9459de
commit 0549ae20b7
1 changed files with 34 additions and 39 deletions

View File

@ -1559,34 +1559,18 @@ static bool ixgbe_cleanup_headers(struct ixgbe_ring *rx_ring,
return false; return false;
} }
/**
* ixgbe_can_reuse_page - determine if we can reuse a page
* @rx_buffer: pointer to rx_buffer containing the page we want to reuse
*
* Returns true if page can be reused in another Rx buffer
**/
static inline bool ixgbe_can_reuse_page(struct ixgbe_rx_buffer *rx_buffer)
{
struct page *page = rx_buffer->page;
/* if we are only owner of page and it is local we can reuse it */
return likely(page_count(page) == 1) &&
likely(page_to_nid(page) == numa_node_id());
}
/** /**
* ixgbe_reuse_rx_page - page flip buffer and store it back on the ring * ixgbe_reuse_rx_page - page flip buffer and store it back on the ring
* @rx_ring: rx descriptor ring to store buffers on * @rx_ring: rx descriptor ring to store buffers on
* @old_buff: donor buffer to have page reused * @old_buff: donor buffer to have page reused
* *
* Syncronizes page for reuse by the adapter * Synchronizes page for reuse by the adapter
**/ **/
static void ixgbe_reuse_rx_page(struct ixgbe_ring *rx_ring, static void ixgbe_reuse_rx_page(struct ixgbe_ring *rx_ring,
struct ixgbe_rx_buffer *old_buff) struct ixgbe_rx_buffer *old_buff)
{ {
struct ixgbe_rx_buffer *new_buff; struct ixgbe_rx_buffer *new_buff;
u16 nta = rx_ring->next_to_alloc; u16 nta = rx_ring->next_to_alloc;
u16 bufsz = ixgbe_rx_bufsz(rx_ring);
new_buff = &rx_ring->rx_buffer_info[nta]; new_buff = &rx_ring->rx_buffer_info[nta];
@ -1597,17 +1581,13 @@ static void ixgbe_reuse_rx_page(struct ixgbe_ring *rx_ring,
/* transfer page from old buffer to new buffer */ /* transfer page from old buffer to new buffer */
new_buff->page = old_buff->page; new_buff->page = old_buff->page;
new_buff->dma = old_buff->dma; new_buff->dma = old_buff->dma;
new_buff->page_offset = old_buff->page_offset;
/* flip page offset to other buffer and store to new_buff */
new_buff->page_offset = old_buff->page_offset ^ bufsz;
/* sync the buffer for use by the device */ /* sync the buffer for use by the device */
dma_sync_single_range_for_device(rx_ring->dev, new_buff->dma, dma_sync_single_range_for_device(rx_ring->dev, new_buff->dma,
new_buff->page_offset, bufsz, new_buff->page_offset,
ixgbe_rx_bufsz(rx_ring),
DMA_FROM_DEVICE); DMA_FROM_DEVICE);
/* bump ref count on page before it is given to the stack */
get_page(new_buff->page);
} }
/** /**
@ -1617,20 +1597,38 @@ static void ixgbe_reuse_rx_page(struct ixgbe_ring *rx_ring,
* @rx_desc: descriptor containing length of buffer written by hardware * @rx_desc: descriptor containing length of buffer written by hardware
* @skb: sk_buff to place the data into * @skb: sk_buff to place the data into
* *
* This function is based on skb_add_rx_frag. I would have used that * This function will add the data contained in rx_buffer->page to the skb.
* function however it doesn't handle the truesize case correctly since we * This is done either through a direct copy if the data in the buffer is
* are allocating more memory than might be used for a single receive. * less than the skb header size, otherwise it will just attach the page as
* a frag to the skb.
*
* The function will then update the page offset if necessary and return
* true if the buffer can be reused by the adapter.
**/ **/
static void ixgbe_add_rx_frag(struct ixgbe_ring *rx_ring, static bool ixgbe_add_rx_frag(struct ixgbe_ring *rx_ring,
struct ixgbe_rx_buffer *rx_buffer, struct ixgbe_rx_buffer *rx_buffer,
struct sk_buff *skb, int size) union ixgbe_adv_rx_desc *rx_desc,
struct sk_buff *skb)
{ {
skb_fill_page_desc(skb, skb_shinfo(skb)->nr_frags, struct page *page = rx_buffer->page;
rx_buffer->page, rx_buffer->page_offset, unsigned int size = le16_to_cpu(rx_desc->wb.upper.length);
size); unsigned int truesize = ixgbe_rx_bufsz(rx_ring);
skb->len += size;
skb->data_len += size; skb_add_rx_frag(skb, skb_shinfo(skb)->nr_frags, page,
skb->truesize += ixgbe_rx_bufsz(rx_ring); rx_buffer->page_offset, size, truesize);
/* if we are only owner of page and it is local we can reuse it */
if (unlikely(page_count(page) != 1) ||
unlikely(page_to_nid(page) != numa_node_id()))
return false;
/* flip page offset to other buffer */
rx_buffer->page_offset ^= truesize;
/* bump ref count on page before it is given to the stack */
get_page(page);
return true;
} }
/** /**
@ -1731,10 +1729,7 @@ static bool ixgbe_clean_rx_irq(struct ixgbe_q_vector *q_vector,
} }
/* pull page into skb */ /* pull page into skb */
ixgbe_add_rx_frag(rx_ring, rx_buffer, skb, if (ixgbe_add_rx_frag(rx_ring, rx_buffer, rx_desc, skb)) {
le16_to_cpu(rx_desc->wb.upper.length));
if (ixgbe_can_reuse_page(rx_buffer)) {
/* hand second half of page back to the ring */ /* hand second half of page back to the ring */
ixgbe_reuse_rx_page(rx_ring, rx_buffer); ixgbe_reuse_rx_page(rx_ring, rx_buffer);
} else if (IXGBE_CB(skb)->dma == rx_buffer->dma) { } else if (IXGBE_CB(skb)->dma == rx_buffer->dma) {