[PATCH 2/4] net/gve: remove variable length array in rx refill
Jasper Tran O'Leary
jtranoleary at google.com
Fri Mar 27 01:18:25 CET 2026
The new_bufs array in gve_rx_refill_dqo was being allocated on the
stack as a variable length array. With a large descriptor count, this
could lead to excessive stack usage or potential stack overflows, and
triggers -Wvla warnings during compilation.
To fix this, replace the stack array with a buffer allocated on the
heap during queue setup. This buffer, refill_bufs, is stored in the
rxq structure and is sized to the queue's descriptor count. This moves
the allocation from the datapath stack to the heap at initialization,
resolving the stack overflow risk while maintaining the efficiency of a
single bulk mbuf allocation.
Signed-off-by: Jasper Tran O'Leary <jtranoleary at google.com>
Reviewed-by: Joshua Washington <joshwash at google.com>
---
drivers/net/gve/gve_ethdev.h | 1 +
drivers/net/gve/gve_rx_dqo.c | 27 +++++++++++++++++++++------
2 files changed, 22 insertions(+), 6 deletions(-)
diff --git a/drivers/net/gve/gve_ethdev.h b/drivers/net/gve/gve_ethdev.h
index 2d570d0..8fd098c 100644
--- a/drivers/net/gve/gve_ethdev.h
+++ b/drivers/net/gve/gve_ethdev.h
@@ -250,6 +250,7 @@ struct gve_rx_queue {
/* Only valid for DQO_RDA queue format */
struct gve_rx_queue *bufq;
+ struct rte_mbuf **refill_bufs;
uint8_t is_gqi_qpl;
};
diff --git a/drivers/net/gve/gve_rx_dqo.c b/drivers/net/gve/gve_rx_dqo.c
index 3aa82b3..8035aee 100644
--- a/drivers/net/gve/gve_rx_dqo.c
+++ b/drivers/net/gve/gve_rx_dqo.c
@@ -40,12 +40,12 @@ static inline void
gve_rx_refill_dqo(struct gve_rx_queue *rxq)
{
volatile struct gve_rx_desc_dqo *rx_buf_desc;
- struct rte_mbuf *new_bufs[rxq->nb_rx_desc];
uint16_t rx_mask = rxq->nb_rx_desc - 1;
uint16_t next_avail = rxq->bufq_tail;
struct rte_eth_dev *dev;
uint16_t nb_refill;
uint64_t dma_addr;
+ rte_iova_t iova;
int16_t buf_id;
int diag;
int i;
@@ -54,7 +54,7 @@ gve_rx_refill_dqo(struct gve_rx_queue *rxq)
if (nb_refill < rxq->free_thresh)
return;
- diag = rte_pktmbuf_alloc_bulk(rxq->mpool, new_bufs, nb_refill);
+ diag = rte_pktmbuf_alloc_bulk(rxq->mpool, rxq->refill_bufs, nb_refill);
if (unlikely(diag < 0)) {
rxq->stats.no_mbufs_bulk++;
rxq->stats.no_mbufs += nb_refill;
@@ -76,12 +76,14 @@ gve_rx_refill_dqo(struct gve_rx_queue *rxq)
PMD_DRV_DP_LOG(ERR,
"No free entries in sw_ring for port %d, queue %d.",
rxq->port_id, rxq->queue_id);
- rte_pktmbuf_free_bulk(new_bufs + i, nb_refill - i);
+ rte_pktmbuf_free_bulk(rxq->refill_bufs + i,
+ nb_refill - i);
nb_refill = i;
break;
}
- rxq->sw_ring[buf_id] = new_bufs[i];
- dma_addr = rte_cpu_to_le_64(rte_mbuf_data_iova_default(new_bufs[i]));
+ rxq->sw_ring[buf_id] = rxq->refill_bufs[i];
+ iova = rte_mbuf_data_iova_default(rxq->refill_bufs[i]);
+ dma_addr = rte_cpu_to_le_64(iova);
rx_buf_desc->buf_id = buf_id;
rx_buf_desc->header_buf_addr = 0;
rx_buf_desc->buf_addr = dma_addr;
@@ -247,6 +249,7 @@ gve_rx_queue_release_dqo(struct rte_eth_dev *dev, uint16_t qid)
gve_release_rxq_mbufs_dqo(q);
rte_free(q->sw_ring);
rte_free(q->completed_buf_list);
+ rte_free(q->refill_bufs);
rte_memzone_free(q->compl_ring_mz);
rte_memzone_free(q->mz);
rte_memzone_free(q->qres_mz);
@@ -363,6 +366,16 @@ gve_rx_queue_setup_dqo(struct rte_eth_dev *dev, uint16_t queue_id,
goto free_rxq_sw_ring;
}
+ /* Allocate buffer for reallocating mbufs */
+ rxq->refill_bufs = rte_zmalloc_socket("gve rx refill bufs",
+ nb_desc * sizeof(*rxq->refill_bufs), RTE_CACHE_LINE_SIZE,
+ socket_id);
+ if (rxq->refill_bufs == NULL) {
+ PMD_DRV_LOG(ERR, "Failed to allocate rx refill bufs.");
+ err = -ENOMEM;
+ goto free_rxq_completed_buf_list;
+ }
+
/* Allocate RX buffer queue */
mz = rte_eth_dma_zone_reserve(dev, "rx_ring", queue_id,
nb_desc * sizeof(struct gve_rx_desc_dqo),
@@ -370,7 +383,7 @@ gve_rx_queue_setup_dqo(struct rte_eth_dev *dev, uint16_t queue_id,
if (mz == NULL) {
PMD_DRV_LOG(ERR, "Failed to reserve DMA memory for RX buffer queue");
err = -ENOMEM;
- goto free_rxq_completed_buf_list;
+ goto free_rxq_refill_bufs;
}
rxq->rx_ring = (struct gve_rx_desc_dqo *)mz->addr;
rxq->rx_ring_phys_addr = mz->iova;
@@ -412,6 +425,8 @@ free_rxq_cq_mz:
rte_memzone_free(rxq->compl_ring_mz);
free_rxq_mz:
rte_memzone_free(rxq->mz);
+free_rxq_refill_bufs:
+ rte_free(rxq->refill_bufs);
free_rxq_completed_buf_list:
rte_free(rxq->completed_buf_list);
free_rxq_sw_ring:
--
2.53.0.1118.gaef5881109-goog
More information about the dev
mailing list