diff options
Diffstat (limited to 'include/net/xdp.h')
| -rw-r--r-- | include/net/xdp.h | 64 | 
1 files changed, 49 insertions, 15 deletions
| diff --git a/include/net/xdp.h b/include/net/xdp.h index f288c348a6c1..aa742f413c35 100644 --- a/include/net/xdp.h +++ b/include/net/xdp.h @@ -76,6 +76,11 @@ enum xdp_buff_flags {  	XDP_FLAGS_FRAGS_PF_MEMALLOC	= BIT(1), /* xdp paged memory is under  						   * pressure  						   */ +	/* frags have unreadable mem, this can't be true for real XDP packets, +	 * but drivers may use XDP helpers to construct Rx pkt state even when +	 * XDP program is not attached. +	 */ +	XDP_FLAGS_FRAGS_UNREADABLE	= BIT(2),  };  struct xdp_buff { @@ -85,8 +90,20 @@ struct xdp_buff {  	void *data_hard_start;  	struct xdp_rxq_info *rxq;  	struct xdp_txq_info *txq; -	u32 frame_sz; /* frame size to deduce data_hard_end/reserved tailroom*/ -	u32 flags; /* supported values defined in xdp_buff_flags */ + +	union { +		struct { +			/* frame size to deduce data_hard_end/tailroom */ +			u32 frame_sz; +			/* supported values defined in xdp_buff_flags */ +			u32 flags; +		}; + +#ifdef __LITTLE_ENDIAN +		/* Used to micro-optimize xdp_init_buff(), don't use directly */ +		u64 frame_sz_flags_init; +#endif +	};  };  static __always_inline bool xdp_buff_has_frags(const struct xdp_buff *xdp) @@ -104,15 +121,19 @@ static __always_inline void xdp_buff_clear_frags_flag(struct xdp_buff *xdp)  	xdp->flags &= ~XDP_FLAGS_HAS_FRAGS;  } -static __always_inline bool -xdp_buff_is_frag_pfmemalloc(const struct xdp_buff *xdp) +static __always_inline void xdp_buff_set_frag_pfmemalloc(struct xdp_buff *xdp)  { -	return !!(xdp->flags & XDP_FLAGS_FRAGS_PF_MEMALLOC); +	xdp->flags |= XDP_FLAGS_FRAGS_PF_MEMALLOC;  } -static __always_inline void xdp_buff_set_frag_pfmemalloc(struct xdp_buff *xdp) +static __always_inline void xdp_buff_set_frag_unreadable(struct xdp_buff *xdp)  { -	xdp->flags |= XDP_FLAGS_FRAGS_PF_MEMALLOC; +	xdp->flags |= XDP_FLAGS_FRAGS_UNREADABLE; +} + +static __always_inline u32 xdp_buff_get_skb_flags(const struct xdp_buff *xdp) +{ +	return xdp->flags;  }  static __always_inline void xdp_buff_clear_frag_pfmemalloc(struct xdp_buff *xdp) @@ -123,9 +144,19 @@ static __always_inline void xdp_buff_clear_frag_pfmemalloc(struct xdp_buff *xdp)  static __always_inline void  xdp_init_buff(struct xdp_buff *xdp, u32 frame_sz, struct xdp_rxq_info *rxq)  { -	xdp->frame_sz = frame_sz;  	xdp->rxq = rxq; + +#ifdef __LITTLE_ENDIAN +	/* +	 * Force the compilers to initialize ::flags and assign ::frame_sz with +	 * one write on 64-bit LE architectures as they're often unable to do +	 * it themselves. +	 */ +	xdp->frame_sz_flags_init = frame_sz; +#else +	xdp->frame_sz = frame_sz;  	xdp->flags = 0; +#endif  }  static __always_inline void @@ -254,6 +285,8 @@ static inline bool xdp_buff_add_frag(struct xdp_buff *xdp, netmem_ref netmem,  	if (unlikely(netmem_is_pfmemalloc(netmem)))  		xdp_buff_set_frag_pfmemalloc(xdp); +	if (unlikely(netmem_is_net_iov(netmem))) +		xdp_buff_set_frag_unreadable(xdp);  	return true;  } @@ -277,10 +310,10 @@ static __always_inline bool xdp_frame_has_frags(const struct xdp_frame *frame)  	return !!(frame->flags & XDP_FLAGS_HAS_FRAGS);  } -static __always_inline bool -xdp_frame_is_frag_pfmemalloc(const struct xdp_frame *frame) +static __always_inline u32 +xdp_frame_get_skb_flags(const struct xdp_frame *frame)  { -	return !!(frame->flags & XDP_FLAGS_FRAGS_PF_MEMALLOC); +	return frame->flags;  }  #define XDP_BULK_QUEUE_SIZE	16 @@ -317,9 +350,9 @@ static inline void xdp_scrub_frame(struct xdp_frame *frame)  }  static inline void -xdp_update_skb_shared_info(struct sk_buff *skb, u8 nr_frags, -			   unsigned int size, unsigned int truesize, -			   bool pfmemalloc) +xdp_update_skb_frags_info(struct sk_buff *skb, u8 nr_frags, +			  unsigned int size, unsigned int truesize, +			  u32 xdp_flags)  {  	struct skb_shared_info *sinfo = skb_shinfo(skb); @@ -333,7 +366,8 @@ xdp_update_skb_shared_info(struct sk_buff *skb, u8 nr_frags,  	skb->len += size;  	skb->data_len += size;  	skb->truesize += truesize; -	skb->pfmemalloc |= pfmemalloc; +	skb->pfmemalloc |= !!(xdp_flags & XDP_FLAGS_FRAGS_PF_MEMALLOC); +	skb->unreadable |= !!(xdp_flags & XDP_FLAGS_FRAGS_UNREADABLE);  }  /* Avoids inlining WARN macro in fast-path */ | 
