]> git.ipfire.org Git - thirdparty/linux.git/commitdiff
gve: update XDP allocation path support RX buffer posting
authorJoshua Washington <joshwash@google.com>
Fri, 21 Mar 2025 00:29:09 +0000 (00:29 +0000)
committerJakub Kicinski <kuba@kernel.org>
Tue, 25 Mar 2025 20:51:15 +0000 (13:51 -0700)
In order to support installing an XDP program on DQ, RX buffers need to
be reposted using 4K buffers, which is larger than the default packet
buffer size of 2K. This is needed to accommodate the extra head and tail
that accompanies the data portion of an XDP buffer. Continuing to use 2K
buffers would mean that the packet buffer size for the NIC would have to
be restricted to 2048 - 320 - 256 = 1472B. However, this is problematic
for two reasons: first, 1472 is not a packet buffer size accepted by
GVE; second, at least 1474B of buffer space is needed to accommodate an
MTU of 1460, which is the default on GCP. As such, we allocate 4K
buffers, and post a 2K section of those 4K buffers (offset relative to
the XDP headroom) to the NIC for DMA to avoid a potential extra copy.
Because the GQ-QPL datapath requires copies regardless, this change was
not needed to support XDP in that case.

To capture this subtlety, a new field, packet_buffer_truesize, has been
added to the rx ring struct to represent size of the allocated buffer,
while packet_buffer_size has been left to represent the portion of the
buffer posted to the NIC.

Reviewed-by: Willem de Bruijn <willemb@google.com>
Signed-off-by: Praveen Kaligineedi <pkaligineedi@google.com>
Signed-off-by: Joshua Washington <joshwash@google.com>
Signed-off-by: Harshitha Ramamurthy <hramamurthy@google.com>
Link: https://patch.msgid.link/20250321002910.1343422-6-hramamurthy@google.com
Signed-off-by: Jakub Kicinski <kuba@kernel.org>
drivers/net/ethernet/google/gve/gve.h
drivers/net/ethernet/google/gve/gve_buffer_mgmt_dqo.c
drivers/net/ethernet/google/gve/gve_main.c
drivers/net/ethernet/google/gve/gve_rx_dqo.c

index 9895541eddaeca8c6139b77c8188f04150bc4eb7..2fab38c8ee78a6f52f855934f2209252f78a4df8 100644 (file)
@@ -59,6 +59,8 @@
 
 #define GVE_MAX_RX_BUFFER_SIZE 4096
 
+#define GVE_XDP_RX_BUFFER_SIZE_DQO 4096
+
 #define GVE_DEFAULT_RX_BUFFER_OFFSET 2048
 
 #define GVE_PAGE_POOL_SIZE_MULTIPLIER 4
@@ -227,7 +229,11 @@ struct gve_rx_cnts {
 /* Contains datapath state used to represent an RX queue. */
 struct gve_rx_ring {
        struct gve_priv *gve;
-       u16 packet_buffer_size;
+
+       u16 packet_buffer_size;         /* Size of buffer posted to NIC */
+       u16 packet_buffer_truesize;     /* Total size of RX buffer */
+       u16 rx_headroom;
+
        union {
                /* GQI fields */
                struct {
@@ -688,6 +694,7 @@ struct gve_rx_alloc_rings_cfg {
        bool raw_addressing;
        bool enable_header_split;
        bool reset_rss;
+       bool xdp;
 
        /* Allocated resources are returned here */
        struct gve_rx_ring *rx;
@@ -1218,7 +1225,8 @@ void gve_free_buffer(struct gve_rx_ring *rx,
                     struct gve_rx_buf_state_dqo *buf_state);
 int gve_alloc_buffer(struct gve_rx_ring *rx, struct gve_rx_desc_dqo *desc);
 struct page_pool *gve_rx_create_page_pool(struct gve_priv *priv,
-                                         struct gve_rx_ring *rx);
+                                         struct gve_rx_ring *rx,
+                                         bool xdp);
 
 /* Reset */
 void gve_schedule_reset(struct gve_priv *priv);
index f9824664d04c53f045cf06e5c383bd644583a2f3..a71883e1d9202fd70d14e29e3b5b6e02351c4571 100644 (file)
@@ -139,7 +139,8 @@ int gve_alloc_qpl_page_dqo(struct gve_rx_ring *rx,
        buf_state->page_info.page_offset = 0;
        buf_state->page_info.page_address =
                page_address(buf_state->page_info.page);
-       buf_state->page_info.buf_size = rx->packet_buffer_size;
+       buf_state->page_info.buf_size = rx->packet_buffer_truesize;
+       buf_state->page_info.pad = rx->rx_headroom;
        buf_state->last_single_ref_offset = 0;
 
        /* The page already has 1 ref. */
@@ -162,7 +163,7 @@ void gve_free_qpl_page_dqo(struct gve_rx_buf_state_dqo *buf_state)
 void gve_try_recycle_buf(struct gve_priv *priv, struct gve_rx_ring *rx,
                         struct gve_rx_buf_state_dqo *buf_state)
 {
-       const u16 data_buffer_size = rx->packet_buffer_size;
+       const u16 data_buffer_size = rx->packet_buffer_truesize;
        int pagecount;
 
        /* Can't reuse if we only fit one buffer per page */
@@ -219,7 +220,7 @@ static int gve_alloc_from_page_pool(struct gve_rx_ring *rx,
 {
        netmem_ref netmem;
 
-       buf_state->page_info.buf_size = rx->packet_buffer_size;
+       buf_state->page_info.buf_size = rx->packet_buffer_truesize;
        netmem = page_pool_alloc_netmem(rx->dqo.page_pool,
                                        &buf_state->page_info.page_offset,
                                        &buf_state->page_info.buf_size,
@@ -231,12 +232,14 @@ static int gve_alloc_from_page_pool(struct gve_rx_ring *rx,
        buf_state->page_info.netmem = netmem;
        buf_state->page_info.page_address = netmem_address(netmem);
        buf_state->addr = page_pool_get_dma_addr_netmem(netmem);
+       buf_state->page_info.pad = rx->dqo.page_pool->p.offset;
 
        return 0;
 }
 
 struct page_pool *gve_rx_create_page_pool(struct gve_priv *priv,
-                                         struct gve_rx_ring *rx)
+                                         struct gve_rx_ring *rx,
+                                         bool xdp)
 {
        u32 ntfy_id = gve_rx_idx_to_ntfy(priv, rx->q_num);
        struct page_pool_params pp = {
@@ -247,7 +250,8 @@ struct page_pool *gve_rx_create_page_pool(struct gve_priv *priv,
                .netdev = priv->dev,
                .napi = &priv->ntfy_blocks[ntfy_id].napi,
                .max_len = PAGE_SIZE,
-               .dma_dir = DMA_FROM_DEVICE,
+               .dma_dir = xdp ? DMA_BIDIRECTIONAL : DMA_FROM_DEVICE,
+               .offset = xdp ? XDP_PACKET_HEADROOM : 0,
        };
 
        return page_pool_create(&pp);
@@ -301,7 +305,8 @@ int gve_alloc_buffer(struct gve_rx_ring *rx, struct gve_rx_desc_dqo *desc)
        }
        desc->buf_id = cpu_to_le16(buf_state - rx->dqo.buf_states);
        desc->buf_addr = cpu_to_le64(buf_state->addr +
-                                    buf_state->page_info.page_offset);
+                                    buf_state->page_info.page_offset +
+                                    buf_state->page_info.pad);
 
        return 0;
 
index 20aabbe0e5180f459eb5e4668a928457b9fa37ae..cb2f9978f45e2ce9dc8be309d7463bb344280f3a 100644 (file)
@@ -1149,8 +1149,14 @@ static int gve_reg_xdp_info(struct gve_priv *priv, struct net_device *dev)
                                       napi->napi_id);
                if (err)
                        goto err;
-               err = xdp_rxq_info_reg_mem_model(&rx->xdp_rxq,
-                                                MEM_TYPE_PAGE_SHARED, NULL);
+               if (gve_is_qpl(priv))
+                       err = xdp_rxq_info_reg_mem_model(&rx->xdp_rxq,
+                                                        MEM_TYPE_PAGE_SHARED,
+                                                        NULL);
+               else
+                       err = xdp_rxq_info_reg_mem_model(&rx->xdp_rxq,
+                                                        MEM_TYPE_PAGE_POOL,
+                                                        rx->dqo.page_pool);
                if (err)
                        goto err;
                rx->xsk_pool = xsk_get_pool_from_qid(dev, i);
@@ -1226,6 +1232,7 @@ static void gve_rx_get_curr_alloc_cfg(struct gve_priv *priv,
        cfg->ring_size = priv->rx_desc_cnt;
        cfg->packet_buffer_size = priv->rx_cfg.packet_buffer_size;
        cfg->rx = priv->rx;
+       cfg->xdp = !!cfg->qcfg_tx->num_xdp_queues;
 }
 
 void gve_get_curr_alloc_cfgs(struct gve_priv *priv,
@@ -1461,6 +1468,7 @@ static int gve_configure_rings_xdp(struct gve_priv *priv,
        gve_get_curr_alloc_cfgs(priv, &tx_alloc_cfg, &rx_alloc_cfg);
        tx_alloc_cfg.num_xdp_rings = num_xdp_rings;
 
+       rx_alloc_cfg.xdp = !!num_xdp_rings;
        return gve_adjust_config(priv, &tx_alloc_cfg, &rx_alloc_cfg);
 }
 
@@ -1629,6 +1637,7 @@ static int gve_xsk_wakeup(struct net_device *dev, u32 queue_id, u32 flags)
 static int verify_xdp_configuration(struct net_device *dev)
 {
        struct gve_priv *priv = netdev_priv(dev);
+       u16 max_xdp_mtu;
 
        if (dev->features & NETIF_F_LRO) {
                netdev_warn(dev, "XDP is not supported when LRO is on.\n");
@@ -1641,7 +1650,11 @@ static int verify_xdp_configuration(struct net_device *dev)
                return -EOPNOTSUPP;
        }
 
-       if (dev->mtu > GVE_DEFAULT_RX_BUFFER_SIZE - sizeof(struct ethhdr) - GVE_RX_PAD) {
+       max_xdp_mtu = priv->rx_cfg.packet_buffer_size - sizeof(struct ethhdr);
+       if (priv->queue_format == GVE_GQI_QPL_FORMAT)
+               max_xdp_mtu -= GVE_RX_PAD;
+
+       if (dev->mtu > max_xdp_mtu) {
                netdev_warn(dev, "XDP is not supported for mtu %d.\n",
                            dev->mtu);
                return -EOPNOTSUPP;
index 5fbcf93a54e0780b3bafa51f70e707b1dfc12aaa..2edf3c632cbddf612a5feb045dff88e8eace60d7 100644 (file)
@@ -225,6 +225,14 @@ int gve_rx_alloc_ring_dqo(struct gve_priv *priv,
        rx->q_num = idx;
        rx->packet_buffer_size = cfg->packet_buffer_size;
 
+       if (cfg->xdp) {
+               rx->packet_buffer_truesize = GVE_XDP_RX_BUFFER_SIZE_DQO;
+               rx->rx_headroom = XDP_PACKET_HEADROOM;
+       } else {
+               rx->packet_buffer_truesize = rx->packet_buffer_size;
+               rx->rx_headroom = 0;
+       }
+
        rx->dqo.num_buf_states = cfg->raw_addressing ? buffer_queue_slots :
                gve_get_rx_pages_per_qpl_dqo(cfg->ring_size);
        rx->dqo.buf_states = kvcalloc(rx->dqo.num_buf_states,
@@ -254,7 +262,7 @@ int gve_rx_alloc_ring_dqo(struct gve_priv *priv,
                goto err;
 
        if (cfg->raw_addressing) {
-               pool = gve_rx_create_page_pool(priv, rx);
+               pool = gve_rx_create_page_pool(priv, rx, cfg->xdp);
                if (IS_ERR(pool))
                        goto err;
 
@@ -484,14 +492,15 @@ static void gve_skb_add_rx_frag(struct gve_rx_ring *rx,
        if (rx->dqo.page_pool) {
                skb_add_rx_frag_netmem(rx->ctx.skb_tail, num_frags,
                                       buf_state->page_info.netmem,
-                                      buf_state->page_info.page_offset,
-                                      buf_len,
+                                      buf_state->page_info.page_offset +
+                                      buf_state->page_info.pad, buf_len,
                                       buf_state->page_info.buf_size);
        } else {
                skb_add_rx_frag(rx->ctx.skb_tail, num_frags,
                                buf_state->page_info.page,
-                               buf_state->page_info.page_offset,
-                               buf_len, buf_state->page_info.buf_size);
+                               buf_state->page_info.page_offset +
+                               buf_state->page_info.pad, buf_len,
+                               buf_state->page_info.buf_size);
        }
 }
 
@@ -611,7 +620,8 @@ static int gve_rx_dqo(struct napi_struct *napi, struct gve_rx_ring *rx,
 
        /* Sync the portion of dma buffer for CPU to read. */
        dma_sync_single_range_for_cpu(&priv->pdev->dev, buf_state->addr,
-                                     buf_state->page_info.page_offset,
+                                     buf_state->page_info.page_offset +
+                                     buf_state->page_info.pad,
                                      buf_len, DMA_FROM_DEVICE);
 
        /* Append to current skb if one exists. */