> From: Feifei Wang [mailto:feifei.wa...@arm.com]
> Sent: Thursday, 30 March 2023 08.30
> 

[...]

> +/**
> + * @internal
> + * Rx routine for rte_eth_dev_buf_recycle().
> + * Refill Rx descriptors in buffer recycle mode.
> + *
> + * @note
> + * This API can only be called by rte_eth_dev_buf_recycle().
> + * Before calling this API, rte_eth_tx_buf_stash() should be
> + * called to stash Tx used buffers into Rx buffer ring.
> + *
> + * When this functionality is not implemented in the driver, the return
> + * buffer number is 0.
> + *
> + * @param port_id
> + *   The port identifier of the Ethernet device.
> + * @param queue_id
> + *   The index of the receive queue.
> + *   The value must be in the range [0, nb_rx_queue - 1] previously supplied
> + *   to rte_eth_dev_configure().
> + *@param nb
> + *   The number of Rx descriptors to be refilled.
> + * @return
> + *   The number Rx descriptors correct to be refilled.
> + *   - ENODEV: bad port or queue (only if compiled with debug).

If you want errors reported by the return value, the function return type 
cannot be uint16_t.

> + */
> +static inline uint16_t rte_eth_rx_descriptors_refill(uint16_t port_id,
> +             uint16_t queue_id, uint16_t nb)
> +{
> +     struct rte_eth_fp_ops *p;
> +     void *qd;
> +
> +#ifdef RTE_ETHDEV_DEBUG_RX
> +     if (port_id >= RTE_MAX_ETHPORTS ||
> +                     queue_id >= RTE_MAX_QUEUES_PER_PORT) {
> +             RTE_ETHDEV_LOG(ERR,
> +                     "Invalid port_id=%u or queue_id=%u\n",
> +                     port_id, queue_id);
> +             rte_errno = ENODEV;
> +             return 0;

If p->rx_descriptors_refill() is likely to return 0, this function should not 
use 0 as return value to indicate errors.

> +     }
> +#endif
> +
> +     p = &rte_eth_fp_ops[port_id];
> +     qd = p->rxq.data[queue_id];
> +
> +#ifdef RTE_ETHDEV_DEBUG_RX
> +     if (!rte_eth_dev_is_valid_port(port_id)) {
> +             RTE_ETHDEV_LOG(ERR, "Invalid Rx port_id=%u\n", port_id);
> +             rte_errno = ENODEV;
> +             return 0;
> +
> +     if (qd == NULL) {
> +             RTE_ETHDEV_LOG(ERR, "Invalid Rx queue_id=%u for port_id=%u\n",
> +                     queue_id, port_id);
> +             rte_errno = ENODEV;
> +             return 0;
> +     }
> +#endif
> +
> +     if (p->rx_descriptors_refill == NULL)
> +             return 0;
> +
> +     return p->rx_descriptors_refill(qd, nb);
> +}
> +
>  /**@{@name Rx hardware descriptor states
>   * @see rte_eth_rx_descriptor_status
>   */
> @@ -6483,6 +6597,122 @@ rte_eth_tx_buffer(uint16_t port_id, uint16_t queue_id,
>       return rte_eth_tx_buffer_flush(port_id, queue_id, buffer);
>  }
> 
> +/**
> + * @internal
> + * Tx routine for rte_eth_dev_buf_recycle().
> + * Stash Tx used buffers into Rx buffer ring in buffer recycle mode.
> + *
> + * @note
> + * This API can only be called by rte_eth_dev_buf_recycle().
> + * After calling this API, rte_eth_rx_descriptors_refill() should be
> + * called to refill Rx ring descriptors.
> + *
> + * When this functionality is not implemented in the driver, the return
> + * buffer number is 0.
> + *
> + * @param port_id
> + *   The port identifier of the Ethernet device.
> + * @param queue_id
> + *   The index of the transmit queue.
> + *   The value must be in the range [0, nb_tx_queue - 1] previously supplied
> + *   to rte_eth_dev_configure().
> + * @param rxq_buf_recycle_info
> + *   A pointer to a structure of Rx queue buffer ring information in buffer
> + *   recycle mode.
> + *
> + * @return
> + *   The number buffers correct to be filled in the Rx buffer ring.
> + *   - ENODEV: bad port or queue (only if compiled with debug).

If you want errors reported by the return value, the function return type 
cannot be uint16_t.

> + */
> +static inline uint16_t rte_eth_tx_buf_stash(uint16_t port_id, uint16_t
> queue_id,
> +             struct rte_eth_rxq_buf_recycle_info *rxq_buf_recycle_info)
> +{
> +     struct rte_eth_fp_ops *p;
> +     void *qd;
> +
> +#ifdef RTE_ETHDEV_DEBUG_TX
> +     if (port_id >= RTE_MAX_ETHPORTS ||
> +                     queue_id >= RTE_MAX_QUEUES_PER_PORT) {
> +             RTE_ETHDEV_LOG(ERR,
> +                     "Invalid port_id=%u or queue_id=%u\n",
> +                     port_id, queue_id);
> +             rte_errno = ENODEV;
> +             return 0;

If p->tx_buf_stash() is likely to return 0, this function should not use 0 as 
return value to indicate errors.

> +     }
> +#endif
> +
> +     p = &rte_eth_fp_ops[port_id];
> +     qd = p->txq.data[queue_id];
> +
> +#ifdef RTE_ETHDEV_DEBUG_TX
> +     if (!rte_eth_dev_is_valid_port(port_id)) {
> +             RTE_ETHDEV_LOG(ERR, "Invalid Tx port_id=%u\n", port_id);
> +             rte_errno = ENODEV;
> +             return 0;
> +
> +     if (qd == NULL) {
> +             RTE_ETHDEV_LOG(ERR, "Invalid Tx queue_id=%u for port_id=%u\n",
> +                     queue_id, port_id);
> +             rte_erno = ENODEV;
> +             return 0;
> +     }
> +#endif
> +
> +     if (p->tx_buf_stash == NULL)
> +             return 0;
> +
> +     return p->tx_buf_stash(qd, rxq_buf_recycle_info);
> +}
> +
> +/**
> + * @warning
> + * @b EXPERIMENTAL: this API may change, or be removed, without prior notice
> + *
> + * Buffer recycle mode can let Tx queue directly put used buffers into Rx
> buffer
> + * ring. This avoids freeing buffers into mempool and allocating buffers from
> + * mempool.

This function description generally describes the buffer recycle mode.

Please update it to describe what this specific function does.

> + *
> + * @param rx_port_id
> + *   Port identifying the receive side.
> + * @param rx_queue_id
> + *   The index of the receive queue identifying the receive side.
> + *   The value must be in the range [0, nb_rx_queue - 1] previously supplied
> + *   to rte_eth_dev_configure().
> + * @param tx_port_id
> + *   Port identifying the transmit side.
> + * @param tx_queue_id
> + *   The index of the transmit queue identifying the transmit side.
> + *   The value must be in the range [0, nb_tx_queue - 1] previously supplied
> + *   to rte_eth_dev_configure().
> + * @param rxq_recycle_info
> + *   A pointer to a structure of type *rte_eth_txq_rearm_data* to be filled.
> + * @return
> + *   - (0) on success or no recycling buffer.

Why not return the return value of rte_eth_rx_descriptors_refill() instead of 0 
on success? (This is a question, not a suggestion.)

Or, if rxq_buf_recycle_info must be valid, the function return type could be 
void instead of int.

> + *   - (-EINVAL) rxq_recycle_info is NULL.
> + */
> +__rte_experimental
> +static inline int
> +rte_eth_dev_buf_recycle(uint16_t rx_port_id, uint16_t rx_queue_id,
> +             uint16_t tx_port_id, uint16_t tx_queue_id,
> +             struct rte_eth_rxq_buf_recycle_info *rxq_buf_recycle_info)
> +{
> +     /* The number of recycling buffers. */
> +     uint16_t nb_buf;
> +
> +     if (!rxq_buf_recycle_info)
> +             return -EINVAL;

This is a fast path function. In which situation is this function called with 
rxq_buf_recycle_info == NULL?

If this function can genuinely be called with rxq_buf_recycle_info == NULL, you 
should test for (rxq_buf_recycle_info == NULL), not (! rxq_buf_recycle_info). 
Otherwise, I think RTE_ASSERT(rxq_buf_recycle_info != NULL) is more appropriate.

> +
> +     /* Stash Tx used buffers into Rx buffer ring */
> +     nb_buf = rte_eth_tx_buf_stash(tx_port_id, tx_queue_id,
> +                             rxq_buf_recycle_info);
> +     /* If there are recycling buffers, refill Rx queue descriptors. */
> +     if (nb_buf)
> +             rte_eth_rx_descriptors_refill(rx_port_id, rx_queue_id,
> +                                     nb_buf);
> +
> +     return 0;
> +}
> +
>  /**
>   * @warning
>   * @b EXPERIMENTAL: this API may change without prior notice
> diff --git a/lib/ethdev/rte_ethdev_core.h b/lib/ethdev/rte_ethdev_core.h
> index dcf8adab92..a138fd4dbc 100644
> --- a/lib/ethdev/rte_ethdev_core.h
> +++ b/lib/ethdev/rte_ethdev_core.h
> @@ -56,6 +56,13 @@ typedef int (*eth_rx_descriptor_status_t)(void *rxq,
> uint16_t offset);
>  /** @internal Check the status of a Tx descriptor */
>  typedef int (*eth_tx_descriptor_status_t)(void *txq, uint16_t offset);
> 
> +/** @internal Stash Tx used buffers into RX ring in buffer recycle mode */
> +typedef uint16_t (*eth_tx_buf_stash_t)(void *txq,
> +             struct rte_eth_rxq_buf_recycle_info *rxq_buf_recycle_info);
> +
> +/** @internal Refill Rx descriptors in buffer recycle mode */
> +typedef uint16_t (*eth_rx_descriptors_refill_t)(void *rxq, uint16_t nb);

Please add proper function descriptions for the two callbacks above.

> +
>  /**
>   * @internal
>   * Structure used to hold opaque pointers to internal ethdev Rx/Tx
> @@ -90,9 +97,11 @@ struct rte_eth_fp_ops {
>       eth_rx_queue_count_t rx_queue_count;
>       /** Check the status of a Rx descriptor. */
>       eth_rx_descriptor_status_t rx_descriptor_status;
> +     /** Refill Rx descriptors in buffer recycle mode */
> +     eth_rx_descriptors_refill_t rx_descriptors_refill;
>       /** Rx queues data. */
>       struct rte_ethdev_qdata rxq;
> -     uintptr_t reserved1[3];
> +     uintptr_t reserved1[4];

You added a function pointer above, so to keep the structure alignment, you 
must remove one here, not add one:

-       uintptr_t reserved1[3];
+       uintptr_t reserved1[2];

>       /**@}*/
> 
>       /**@{*/
> @@ -106,9 +115,11 @@ struct rte_eth_fp_ops {
>       eth_tx_prep_t tx_pkt_prepare;
>       /** Check the status of a Tx descriptor. */
>       eth_tx_descriptor_status_t tx_descriptor_status;
> +     /** Stash Tx used buffers into RX ring in buffer recycle mode */
> +     eth_tx_buf_stash_t tx_buf_stash;
>       /** Tx queues data. */
>       struct rte_ethdev_qdata txq;
> -     uintptr_t reserved2[3];
> +     uintptr_t reserved2[4];

You added a function pointer above, so to keep the structure alignment, you 
must remove one here, not add one:

-       uintptr_t reserved1[3];
+       uintptr_t reserved1[2];

>       /**@}*/
> 
>  } __rte_cache_aligned;

Reply via email to