@@ -768,6 +768,13 @@ static __always_inline u32 bpf_prog_run_xdp(const struct bpf_prog *prog,
* already takes rcu_read_lock() when fetching the program, so
* it's not necessary here anymore.
*/
+ xdp->frame_length = xdp->data_end - xdp->data;
+ if (unlikely(xdp->mb)) {
+ struct xdp_shared_info *xdp_sinfo;
+
+ xdp_sinfo = xdp_get_shared_info_from_buff(xdp);
+ xdp->frame_length += xdp_sinfo->data_length;
+ }
return __BPF_PROG_RUN(prog, xdp, BPF_DISPATCHER_FUNC(xdp));
}
@@ -77,6 +77,13 @@ struct xdp_buff {
* tailroom
*/
u32 mb:1; /* xdp non-linear buffer */
+ u32 frame_length; /* Total frame length across all buffers. Only needs
+ * to be updated by helper functions, as it will be
+ * initialized at XDP program start. This field only
+ * needs 17-bits (128kB). In case the remaining bits
+ * need to be re-purposed, please make sure the
+ * xdp_convert_ctx_access() function gets updated.
+ */
};
static __always_inline void
@@ -237,6 +244,14 @@ void xdp_convert_frame_to_buff(struct xdp_frame *frame, struct xdp_buff *xdp)
xdp->data_meta = frame->data - frame->metasize;
xdp->frame_sz = frame->frame_sz;
xdp->mb = frame->mb;
+ xdp->frame_length = frame->len;
+
+ if (unlikely(xdp->mb)) {
+ struct xdp_shared_info *xdp_sinfo;
+
+ xdp_sinfo = xdp_get_shared_info_from_buff(xdp);
+ xdp->frame_length += xdp_sinfo->data_length;
+ }
}
static inline
@@ -5224,6 +5224,7 @@ struct xdp_md {
__u32 rx_queue_index; /* rxq->queue_index */
__u32 egress_ifindex; /* txq->dev->ifindex */
+ __u32 frame_length;
};
/* DEVMAP map-value layout
@@ -3873,6 +3873,7 @@ static int bpf_xdp_mb_adjust_tail(struct xdp_buff *xdp, int offset)
memset(xdp_get_frag_address(frag) + size, 0, offset);
xdp_set_frag_size(frag, size + offset);
xdp_sinfo->data_length += offset;
+ xdp->frame_length += offset;
} else {
int i, frags_to_free = 0;
@@ -3894,6 +3895,7 @@ static int bpf_xdp_mb_adjust_tail(struct xdp_buff *xdp, int offset)
* to adjust the data_length in line.
*/
xdp_sinfo->data_length -= shrink;
+ xdp->frame_length -= shrink;
xdp_set_frag_size(frag, size - shrink);
break;
}
@@ -9137,6 +9139,12 @@ static u32 xdp_convert_ctx_access(enum bpf_access_type type,
*insn++ = BPF_LDX_MEM(BPF_W, si->dst_reg, si->dst_reg,
offsetof(struct net_device, ifindex));
break;
+ case offsetof(struct xdp_md, frame_length):
+ *insn++ = BPF_LDX_MEM(BPF_FIELD_SIZEOF(struct xdp_buff,
+ frame_length),
+ si->dst_reg, si->src_reg,
+ offsetof(struct xdp_buff, frame_length));
+ break;
}
return insn - insn_buf;
@@ -510,6 +510,7 @@ void xdp_return_num_frags_from_buff(struct xdp_buff *xdp, u16 num_frags)
struct page *page = xdp_get_frag_page(frag);
xdp_sinfo->data_length -= xdp_get_frag_size(frag);
+ xdp->frame_length -= xdp_get_frag_size(frag);
__xdp_return(page_address(page), &xdp->rxq->mem, false, NULL);
}
xdp_sinfo->nr_frags -= num_frags;
@@ -5218,6 +5218,7 @@ struct xdp_md {
__u32 rx_queue_index; /* rxq->queue_index */
__u32 egress_ifindex; /* txq->dev->ifindex */
+ __u32 frame_length;
};
/* DEVMAP map-value layout