diff options
Diffstat (limited to 'net/core/skbuff.c')
| -rw-r--r-- | net/core/skbuff.c | 44 | 
1 files changed, 13 insertions, 31 deletions
diff --git a/net/core/skbuff.c b/net/core/skbuff.c index 3ab989b0de42..a9a2ae3e2213 100644 --- a/net/core/skbuff.c +++ b/net/core/skbuff.c @@ -1649,7 +1649,7 @@ static void sock_spd_release(struct splice_pipe_desc *spd, unsigned int i)  static struct page *linear_to_page(struct page *page, unsigned int *len,  				   unsigned int *offset, -				   struct sk_buff *skb, struct sock *sk) +				   struct sock *sk)  {  	struct page_frag *pfrag = sk_page_frag(sk); @@ -1682,14 +1682,14 @@ static bool spd_can_coalesce(const struct splice_pipe_desc *spd,  static bool spd_fill_page(struct splice_pipe_desc *spd,  			  struct pipe_inode_info *pipe, struct page *page,  			  unsigned int *len, unsigned int offset, -			  struct sk_buff *skb, bool linear, +			  bool linear,  			  struct sock *sk)  {  	if (unlikely(spd->nr_pages == MAX_SKB_FRAGS))  		return true;  	if (linear) { -		page = linear_to_page(page, len, &offset, skb, sk); +		page = linear_to_page(page, len, &offset, sk);  		if (!page)  			return true;  	} @@ -1706,23 +1706,9 @@ static bool spd_fill_page(struct splice_pipe_desc *spd,  	return false;  } -static inline void __segment_seek(struct page **page, unsigned int *poff, -				  unsigned int *plen, unsigned int off) -{ -	unsigned long n; - -	*poff += off; -	n = *poff / PAGE_SIZE; -	if (n) -		*page = nth_page(*page, n); - -	*poff = *poff % PAGE_SIZE; -	*plen -= off; -} -  static bool __splice_segment(struct page *page, unsigned int poff,  			     unsigned int plen, unsigned int *off, -			     unsigned int *len, struct sk_buff *skb, +			     unsigned int *len,  			     struct splice_pipe_desc *spd, bool linear,  			     struct sock *sk,  			     struct pipe_inode_info *pipe) @@ -1737,23 +1723,19 @@ static bool __splice_segment(struct page *page, unsigned int poff,  	}  	/* ignore any bits we already processed */ -	if (*off) { -		__segment_seek(&page, &poff, &plen, *off); -		*off = 0; -	} +	poff += *off; +	plen -= *off; +	*off = 0;  	do {  		unsigned int flen = min(*len, plen); -		/* the linear region may spread across several pages  */ -		flen = min_t(unsigned int, flen, PAGE_SIZE - poff); - -		if (spd_fill_page(spd, pipe, page, &flen, poff, skb, linear, sk)) +		if (spd_fill_page(spd, pipe, page, &flen, poff, +				  linear, sk))  			return true; - -		__segment_seek(&page, &poff, &plen, flen); +		poff += flen; +		plen -= flen;  		*len -= flen; -  	} while (*len && plen);  	return false; @@ -1777,7 +1759,7 @@ static bool __skb_splice_bits(struct sk_buff *skb, struct pipe_inode_info *pipe,  	if (__splice_segment(virt_to_page(skb->data),  			     (unsigned long) skb->data & (PAGE_SIZE - 1),  			     skb_headlen(skb), -			     offset, len, skb, spd, +			     offset, len, spd,  			     skb_head_is_locked(skb),  			     sk, pipe))  		return true; @@ -1790,7 +1772,7 @@ static bool __skb_splice_bits(struct sk_buff *skb, struct pipe_inode_info *pipe,  		if (__splice_segment(skb_frag_page(f),  				     f->page_offset, skb_frag_size(f), -				     offset, len, skb, spd, false, sk, pipe)) +				     offset, len, spd, false, sk, pipe))  			return true;  	}  | 
