* [PATCH] net/af_packet: provide packet drop stats
@ 2025-01-16 16:17 Stefan Laesser
2025-01-16 16:24 ` Stephen Hemminger
2025-01-21 12:26 ` [PATCH v2] " Stefan Laesser
0 siblings, 2 replies; 9+ messages in thread
From: Stefan Laesser @ 2025-01-16 16:17 UTC (permalink / raw)
To: John W. Linville; +Cc: dev, Stefan Laesser
The Linux kernel provides the ability to query the packet drop counter
of a socket. This information can be provided when the user requests
stats.
It is important to note that each call to getsockopt with
PACKET_STATISTICS resets the internal counters. So the caller needs to
keep track of the total count on its own.
Next, I have added a counter for the case when mbuf couldn't be
allocated.
Signed-off-by: Stefan Laesser <stefan.laesser@omicronenergy.com>
---
drivers/net/af_packet/rte_eth_af_packet.c | 32 +++++++++++++++++++++--
1 file changed, 30 insertions(+), 2 deletions(-)
diff --git a/drivers/net/af_packet/rte_eth_af_packet.c b/drivers/net/af_packet/rte_eth_af_packet.c
index ceb8d9356a..a771dd854d 100644
--- a/drivers/net/af_packet/rte_eth_af_packet.c
+++ b/drivers/net/af_packet/rte_eth_af_packet.c
@@ -58,6 +58,8 @@ struct __rte_cache_aligned pkt_rx_queue {
volatile unsigned long rx_pkts;
volatile unsigned long rx_bytes;
+ volatile unsigned long rx_nombuf;
+ volatile unsigned long rx_dropped_pkts;
};
struct __rte_cache_aligned pkt_tx_queue {
@@ -145,8 +147,10 @@ eth_af_packet_rx(void *queue, struct rte_mbuf **bufs, uint16_t nb_pkts)
/* allocate the next mbuf */
mbuf = rte_pktmbuf_alloc(pkt_q->mb_pool);
- if (unlikely(mbuf == NULL))
+ if (unlikely(mbuf == NULL)) {
+ pkt_q->rx_nombuf++;
break;
+ }
/* packet will fit in the mbuf, go ahead and receive it */
rte_pktmbuf_pkt_len(mbuf) = rte_pktmbuf_data_len(mbuf) = ppd->tp_snaplen;
@@ -417,17 +421,37 @@ static int
eth_stats_get(struct rte_eth_dev *dev, struct rte_eth_stats *igb_stats)
{
unsigned i, imax;
- unsigned long rx_total = 0, tx_total = 0, tx_err_total = 0;
+ unsigned long rx_total = 0, rx_dropped_total = 0, rx_nombuf_total = 0;
+ unsigned long tx_total = 0, tx_err_total = 0;
unsigned long rx_bytes_total = 0, tx_bytes_total = 0;
const struct pmd_internals *internal = dev->data->dev_private;
+ struct tpacket_stats iface_stats;
+ socklen_t iface_stats_len = sizeof(struct tpacket_stats);
+
imax = (internal->nb_queues < RTE_ETHDEV_QUEUE_STAT_CNTRS ?
internal->nb_queues : RTE_ETHDEV_QUEUE_STAT_CNTRS);
for (i = 0; i < imax; i++) {
+ /* query dropped packets counter from socket */
+ if (internal->rx_queue[i].sockfd != -1 &&
+ getsockopt(internal->rx_queue[i].sockfd, SOL_PACKET,
+ PACKET_STATISTICS, &iface_stats,
+ &iface_stats_len) > -1) {
+ /*
+ * keep total because each call to getsocketopt with PACKET_STATISTICS
+ * reset the counter of the socket
+ */
+ internal->rx_queue[i].rx_dropped_pkts += iface_stats.tp_drops;
+ }
+
igb_stats->q_ipackets[i] = internal->rx_queue[i].rx_pkts;
igb_stats->q_ibytes[i] = internal->rx_queue[i].rx_bytes;
+ igb_stats->q_errors[i] = internal->rx_queue[i].rx_dropped_pkts;
+
rx_total += igb_stats->q_ipackets[i];
rx_bytes_total += igb_stats->q_ibytes[i];
+ rx_dropped_total += igb_stats->q_errors[i];
+ rx_nombuf_total += internal->rx_queue[i].rx_nombuf;
}
imax = (internal->nb_queues < RTE_ETHDEV_QUEUE_STAT_CNTRS ?
@@ -442,6 +466,8 @@ eth_stats_get(struct rte_eth_dev *dev, struct rte_eth_stats *igb_stats)
igb_stats->ipackets = rx_total;
igb_stats->ibytes = rx_bytes_total;
+ igb_stats->imissed = rx_dropped_total;
+ igb_stats->rx_nombuf = rx_nombuf_total;
igb_stats->opackets = tx_total;
igb_stats->oerrors = tx_err_total;
igb_stats->obytes = tx_bytes_total;
@@ -457,6 +483,8 @@ eth_stats_reset(struct rte_eth_dev *dev)
for (i = 0; i < internal->nb_queues; i++) {
internal->rx_queue[i].rx_pkts = 0;
internal->rx_queue[i].rx_bytes = 0;
+ internal->rx_queue[i].rx_nombuf = 0;
+ internal->rx_queue[i].rx_dropped_pkts = 0;
}
for (i = 0; i < internal->nb_queues; i++) {
--
2.34.1
^ permalink raw reply [flat|nested] 9+ messages in thread
* Re: [PATCH] net/af_packet: provide packet drop stats
2025-01-16 16:17 [PATCH] net/af_packet: provide packet drop stats Stefan Laesser
@ 2025-01-16 16:24 ` Stephen Hemminger
2025-01-17 7:20 ` Stefan Lässer
2025-01-21 12:26 ` [PATCH v2] " Stefan Laesser
1 sibling, 1 reply; 9+ messages in thread
From: Stephen Hemminger @ 2025-01-16 16:24 UTC (permalink / raw)
To: Stefan Laesser; +Cc: John W. Linville, dev
On Thu, 16 Jan 2025 17:17:03 +0100
Stefan Laesser <stefan.laesser@omicronenergy.com> wrote:
> The Linux kernel provides the ability to query the packet drop counter
> of a socket. This information can be provided when the user requests
> stats.
>
> It is important to note that each call to getsockopt with
> PACKET_STATISTICS resets the internal counters. So the caller needs to
> keep track of the total count on its own.
>
> Next, I have added a counter for the case when mbuf couldn't be
> allocated.
>
> Signed-off-by: Stefan Laesser <stefan.laesser@omicronenergy.com>
> ---
> drivers/net/af_packet/rte_eth_af_packet.c | 32 +++++++++++++++++++++--
> 1 file changed, 30 insertions(+), 2 deletions(-)
>
> diff --git a/drivers/net/af_packet/rte_eth_af_packet.c b/drivers/net/af_packet/rte_eth_af_packet.c
> index ceb8d9356a..a771dd854d 100644
> --- a/drivers/net/af_packet/rte_eth_af_packet.c
> +++ b/drivers/net/af_packet/rte_eth_af_packet.c
> @@ -58,6 +58,8 @@ struct __rte_cache_aligned pkt_rx_queue {
>
> volatile unsigned long rx_pkts;
> volatile unsigned long rx_bytes;
> + volatile unsigned long rx_nombuf;
> + volatile unsigned long rx_dropped_pkts;
> };
>
> struct __rte_cache_aligned pkt_tx_queue {
> @@ -145,8 +147,10 @@ eth_af_packet_rx(void *queue, struct rte_mbuf **bufs, uint16_t nb_pkts)
>
> /* allocate the next mbuf */
> mbuf = rte_pktmbuf_alloc(pkt_q->mb_pool);
> - if (unlikely(mbuf == NULL))
> + if (unlikely(mbuf == NULL)) {
> + pkt_q->rx_nombuf++;
> break;
> + }
>
> /* packet will fit in the mbuf, go ahead and receive it */
> rte_pktmbuf_pkt_len(mbuf) = rte_pktmbuf_data_len(mbuf) = ppd->tp_snaplen;
> @@ -417,17 +421,37 @@ static int
> eth_stats_get(struct rte_eth_dev *dev, struct rte_eth_stats *igb_stats)
> {
> unsigned i, imax;
> - unsigned long rx_total = 0, tx_total = 0, tx_err_total = 0;
> + unsigned long rx_total = 0, rx_dropped_total = 0, rx_nombuf_total = 0;
> + unsigned long tx_total = 0, tx_err_total = 0;
> unsigned long rx_bytes_total = 0, tx_bytes_total = 0;
> const struct pmd_internals *internal = dev->data->dev_private;
>
> + struct tpacket_stats iface_stats;
> + socklen_t iface_stats_len = sizeof(struct tpacket_stats);
This declaration could be moved inside the loop.
> +
> imax = (internal->nb_queues < RTE_ETHDEV_QUEUE_STAT_CNTRS ?
> internal->nb_queues : RTE_ETHDEV_QUEUE_STAT_CNTRS);
> for (i = 0; i < imax; i++) {
> + /* query dropped packets counter from socket */
> + if (internal->rx_queue[i].sockfd != -1 &&
> + getsockopt(internal->rx_queue[i].sockfd, SOL_PACKET,
> + PACKET_STATISTICS, &iface_stats,
> + &iface_stats_len) > -1) {
> + /*
> + * keep total because each call to getsocketopt with PACKET_STATISTICS
> + * reset the counter of the socket
> + */
> + internal->rx_queue[i].rx_dropped_pkts += iface_stats.tp_drops;
> + }
> +
> igb_stats->q_ipackets[i] = internal->rx_queue[i].rx_pkts;
> igb_stats->q_ibytes[i] = internal->rx_queue[i].rx_bytes;
> + igb_stats->q_errors[i] = internal->rx_queue[i].rx_dropped_pkts;
Dropped packets are not errors; at least most other drivers do not report missed
packets as errors. Should be imissed statistic.
> +
> rx_total += igb_stats->q_ipackets[i];
> rx_bytes_total += igb_stats->q_ibytes[i];
> + rx_dropped_total += igb_stats->q_errors[i];
> + rx_nombuf_total += internal->rx_queue[i].rx_nombuf;
> }
>
> imax = (internal->nb_queues < RTE_ETHDEV_QUEUE_STAT_CNTRS ?
> @@ -442,6 +466,8 @@ eth_stats_get(struct rte_eth_dev *dev, struct rte_eth_stats *igb_stats)
>
> igb_stats->ipackets = rx_total;
> igb_stats->ibytes = rx_bytes_total;
> + igb_stats->imissed = rx_dropped_total;
> + igb_stats->rx_nombuf = rx_nombuf_total;
> igb_stats->opackets = tx_total;
> igb_stats->oerrors = tx_err_total;
> igb_stats->obytes = tx_bytes_total;
> @@ -457,6 +483,8 @@ eth_stats_reset(struct rte_eth_dev *dev)
> for (i = 0; i < internal->nb_queues; i++) {
> internal->rx_queue[i].rx_pkts = 0;
> internal->rx_queue[i].rx_bytes = 0;
> + internal->rx_queue[i].rx_nombuf = 0;
> + internal->rx_queue[i].rx_dropped_pkts = 0;
> }
>
> for (i = 0; i < internal->nb_queues; i++) {
What about stats reset?
^ permalink raw reply [flat|nested] 9+ messages in thread
* RE: [PATCH] net/af_packet: provide packet drop stats
2025-01-16 16:24 ` Stephen Hemminger
@ 2025-01-17 7:20 ` Stefan Lässer
2025-01-17 16:43 ` Stephen Hemminger
0 siblings, 1 reply; 9+ messages in thread
From: Stefan Lässer @ 2025-01-17 7:20 UTC (permalink / raw)
To: Stephen Hemminger; +Cc: John W. Linville, dev
> -----Original Message-----
> From: Stephen Hemminger <stephen@networkplumber.org>
> Sent: Thursday, January 16, 2025 5:25 PM
> To: Stefan Lässer <stefan.laesser@omicronenergy.com>
> Cc: John W. Linville <linville@tuxdriver.com>; dev@dpdk.org
> Subject: Re: [PATCH] net/af_packet: provide packet drop stats
>
> On Thu, 16 Jan 2025 17:17:03 +0100
> Stefan Laesser <stefan.laesser@omicronenergy.com> wrote:
>
> > The Linux kernel provides the ability to query the packet drop counter
> > of a socket. This information can be provided when the user requests
> > stats.
> >
> > It is important to note that each call to getsockopt with
> > PACKET_STATISTICS resets the internal counters. So the caller needs to
> > keep track of the total count on its own.
> >
> > Next, I have added a counter for the case when mbuf couldn't be
> > allocated.
> >
> > Signed-off-by: Stefan Laesser <stefan.laesser@omicronenergy.com>
> > ---
> > drivers/net/af_packet/rte_eth_af_packet.c | 32
> > +++++++++++++++++++++--
> > 1 file changed, 30 insertions(+), 2 deletions(-)
> >
> > diff --git a/drivers/net/af_packet/rte_eth_af_packet.c
> > b/drivers/net/af_packet/rte_eth_af_packet.c
> > index ceb8d9356a..a771dd854d 100644
> > --- a/drivers/net/af_packet/rte_eth_af_packet.c
> > +++ b/drivers/net/af_packet/rte_eth_af_packet.c
> > @@ -58,6 +58,8 @@ struct __rte_cache_aligned pkt_rx_queue {
> >
> > volatile unsigned long rx_pkts;
> > volatile unsigned long rx_bytes;
> > + volatile unsigned long rx_nombuf;
> > + volatile unsigned long rx_dropped_pkts;
> > };
> >
> > struct __rte_cache_aligned pkt_tx_queue { @@ -145,8 +147,10 @@
> > eth_af_packet_rx(void *queue, struct rte_mbuf **bufs, uint16_t
> > nb_pkts)
> >
> > /* allocate the next mbuf */
> > mbuf = rte_pktmbuf_alloc(pkt_q->mb_pool);
> > - if (unlikely(mbuf == NULL))
> > + if (unlikely(mbuf == NULL)) {
> > + pkt_q->rx_nombuf++;
> > break;
> > + }
> >
> > /* packet will fit in the mbuf, go ahead and receive it */
> > rte_pktmbuf_pkt_len(mbuf) = rte_pktmbuf_data_len(mbuf)
> =
> > ppd->tp_snaplen; @@ -417,17 +421,37 @@ static int
> > eth_stats_get(struct rte_eth_dev *dev, struct rte_eth_stats
> > *igb_stats) {
> > unsigned i, imax;
> > - unsigned long rx_total = 0, tx_total = 0, tx_err_total = 0;
> > + unsigned long rx_total = 0, rx_dropped_total = 0, rx_nombuf_total =
> 0;
> > + unsigned long tx_total = 0, tx_err_total = 0;
> > unsigned long rx_bytes_total = 0, tx_bytes_total = 0;
> > const struct pmd_internals *internal = dev->data->dev_private;
> >
> > + struct tpacket_stats iface_stats;
> > + socklen_t iface_stats_len = sizeof(struct tpacket_stats);
>
> This declaration could be moved inside the loop.
Yes, you are right - I will move it inside the loop.
> > +
> > imax = (internal->nb_queues < RTE_ETHDEV_QUEUE_STAT_CNTRS ?
> > internal->nb_queues : RTE_ETHDEV_QUEUE_STAT_CNTRS);
> > for (i = 0; i < imax; i++) {
> > + /* query dropped packets counter from socket */
> > + if (internal->rx_queue[i].sockfd != -1 &&
> > + getsockopt(internal->rx_queue[i].sockfd,
> SOL_PACKET,
> > + PACKET_STATISTICS,
> &iface_stats,
> > + &iface_stats_len) > -1) {
> > + /*
> > + * keep total because each call to getsocketopt with
> PACKET_STATISTICS
> > + * reset the counter of the socket
> > + */
> > + internal->rx_queue[i].rx_dropped_pkts +=
> iface_stats.tp_drops;
> > + }
> > +
> > igb_stats->q_ipackets[i] = internal->rx_queue[i].rx_pkts;
> > igb_stats->q_ibytes[i] = internal->rx_queue[i].rx_bytes;
> > + igb_stats->q_errors[i] = internal-
> >rx_queue[i].rx_dropped_pkts;
>
> Dropped packets are not errors; at least most other drivers do not report
> missed packets as errors. Should be imissed statistic.
The struct rte_eth_stats currently does not contain q_imissed. It only has
q_ipackets, q_opackets, q_ibytes, q_obytes and q_errors. The latter is described as
"Total number of queue packets received that are dropped.". This is why I have choosen
q_errors because the field comment sounds like a good match to me.
As there is no q_imissed, I suggest removing this line from my patch and just adding the imissed total counter:
igb_stats->q_ipackets[i] = internal->rx_queue[i].rx_pkts;
igb_stats->q_ibytes[i] = internal->rx_queue[i].rx_bytes;
- igb_stats->q_errors[i] = internal->rx_queue[i].rx_dropped_pkts;
rx_total += igb_stats->q_ipackets[i];
rx_bytes_total += igb_stats->q_ibytes[i];
- rx_dropped_total += igb_stats->q_errors[i];
+ rx_dropped_total += internal->rx_queue[i].rx_dropped_pkts;
rx_nombuf_total += internal->rx_queue[i].rx_nombuf;
Do you agree with that?
> > +
> > rx_total += igb_stats->q_ipackets[i];
> > rx_bytes_total += igb_stats->q_ibytes[i];
> > + rx_dropped_total += igb_stats->q_errors[i];
> > + rx_nombuf_total += internal->rx_queue[i].rx_nombuf;
> > }
> >
> > imax = (internal->nb_queues < RTE_ETHDEV_QUEUE_STAT_CNTRS ?
> > @@ -442,6 +466,8 @@ eth_stats_get(struct rte_eth_dev *dev, struct
> > rte_eth_stats *igb_stats)
> >
> > igb_stats->ipackets = rx_total;
> > igb_stats->ibytes = rx_bytes_total;
> > + igb_stats->imissed = rx_dropped_total;
> > + igb_stats->rx_nombuf = rx_nombuf_total;
> > igb_stats->opackets = tx_total;
> > igb_stats->oerrors = tx_err_total;
> > igb_stats->obytes = tx_bytes_total;
> > @@ -457,6 +483,8 @@ eth_stats_reset(struct rte_eth_dev *dev)
> > for (i = 0; i < internal->nb_queues; i++) {
> > internal->rx_queue[i].rx_pkts = 0;
> > internal->rx_queue[i].rx_bytes = 0;
> > + internal->rx_queue[i].rx_nombuf = 0;
> > + internal->rx_queue[i].rx_dropped_pkts = 0;
> > }
> >
> > for (i = 0; i < internal->nb_queues; i++) {
>
> What about stats reset?
I'm sorry, I don't understand what you mean. The two newly added queue counters are reset in the eth_stats_reset() function.
Isn't that the right place to do it?
Thanks for your feedback!
^ permalink raw reply [flat|nested] 9+ messages in thread
* Re: [PATCH] net/af_packet: provide packet drop stats
2025-01-17 7:20 ` Stefan Lässer
@ 2025-01-17 16:43 ` Stephen Hemminger
0 siblings, 0 replies; 9+ messages in thread
From: Stephen Hemminger @ 2025-01-17 16:43 UTC (permalink / raw)
To: Stefan Lässer; +Cc: John W. Linville, dev
On Fri, 17 Jan 2025 07:20:05 +0000
Stefan Lässer <stefan.laesser@omicronenergy.com> wrote:
> > -----Original Message-----
> > From: Stephen Hemminger <stephen@networkplumber.org>
> > Sent: Thursday, January 16, 2025 5:25 PM
> > To: Stefan Lässer <stefan.laesser@omicronenergy.com>
> > Cc: John W. Linville <linville@tuxdriver.com>; dev@dpdk.org
> > Subject: Re: [PATCH] net/af_packet: provide packet drop stats
> >
> > On Thu, 16 Jan 2025 17:17:03 +0100
> > Stefan Laesser <stefan.laesser@omicronenergy.com> wrote:
> >
> > > The Linux kernel provides the ability to query the packet drop counter
> > > of a socket. This information can be provided when the user requests
> > > stats.
> > >
> > > It is important to note that each call to getsockopt with
> > > PACKET_STATISTICS resets the internal counters. So the caller needs to
> > > keep track of the total count on its own.
> > >
> > > Next, I have added a counter for the case when mbuf couldn't be
> > > allocated.
> > >
> > > Signed-off-by: Stefan Laesser <stefan.laesser@omicronenergy.com>
> > > ---
> > > drivers/net/af_packet/rte_eth_af_packet.c | 32
> > > +++++++++++++++++++++--
> > > 1 file changed, 30 insertions(+), 2 deletions(-)
> > >
> > > diff --git a/drivers/net/af_packet/rte_eth_af_packet.c
> > > b/drivers/net/af_packet/rte_eth_af_packet.c
> > > index ceb8d9356a..a771dd854d 100644
> > > --- a/drivers/net/af_packet/rte_eth_af_packet.c
> > > +++ b/drivers/net/af_packet/rte_eth_af_packet.c
> > > @@ -58,6 +58,8 @@ struct __rte_cache_aligned pkt_rx_queue {
> > >
> > > volatile unsigned long rx_pkts;
> > > volatile unsigned long rx_bytes;
> > > + volatile unsigned long rx_nombuf;
> > > + volatile unsigned long rx_dropped_pkts;
> > > };
> > >
> > > struct __rte_cache_aligned pkt_tx_queue { @@ -145,8 +147,10 @@
> > > eth_af_packet_rx(void *queue, struct rte_mbuf **bufs, uint16_t
> > > nb_pkts)
> > >
> > > /* allocate the next mbuf */
> > > mbuf = rte_pktmbuf_alloc(pkt_q->mb_pool);
> > > - if (unlikely(mbuf == NULL))
> > > + if (unlikely(mbuf == NULL)) {
> > > + pkt_q->rx_nombuf++;
> > > break;
> > > + }
> > >
> > > /* packet will fit in the mbuf, go ahead and receive it */
> > > rte_pktmbuf_pkt_len(mbuf) = rte_pktmbuf_data_len(mbuf)
> > =
> > > ppd->tp_snaplen; @@ -417,17 +421,37 @@ static int
> > > eth_stats_get(struct rte_eth_dev *dev, struct rte_eth_stats
> > > *igb_stats) {
> > > unsigned i, imax;
> > > - unsigned long rx_total = 0, tx_total = 0, tx_err_total = 0;
> > > + unsigned long rx_total = 0, rx_dropped_total = 0, rx_nombuf_total =
> > 0;
> > > + unsigned long tx_total = 0, tx_err_total = 0;
> > > unsigned long rx_bytes_total = 0, tx_bytes_total = 0;
> > > const struct pmd_internals *internal = dev->data->dev_private;
> > >
> > > + struct tpacket_stats iface_stats;
> > > + socklen_t iface_stats_len = sizeof(struct tpacket_stats);
> >
> > This declaration could be moved inside the loop.
>
> Yes, you are right - I will move it inside the loop.
>
> > > +
> > > imax = (internal->nb_queues < RTE_ETHDEV_QUEUE_STAT_CNTRS ?
> > > internal->nb_queues : RTE_ETHDEV_QUEUE_STAT_CNTRS);
> > > for (i = 0; i < imax; i++) {
> > > + /* query dropped packets counter from socket */
> > > + if (internal->rx_queue[i].sockfd != -1 &&
> > > + getsockopt(internal->rx_queue[i].sockfd,
> > SOL_PACKET,
> > > + PACKET_STATISTICS,
> > &iface_stats,
> > > + &iface_stats_len) > -1) {
> > > + /*
> > > + * keep total because each call to getsocketopt with
> > PACKET_STATISTICS
> > > + * reset the counter of the socket
> > > + */
> > > + internal->rx_queue[i].rx_dropped_pkts +=
> > iface_stats.tp_drops;
> > > + }
> > > +
> > > igb_stats->q_ipackets[i] = internal->rx_queue[i].rx_pkts;
> > > igb_stats->q_ibytes[i] = internal->rx_queue[i].rx_bytes;
> > > + igb_stats->q_errors[i] = internal-
> > >rx_queue[i].rx_dropped_pkts;
> >
> > Dropped packets are not errors; at least most other drivers do not report
> > missed packets as errors. Should be imissed statistic.
>
> The struct rte_eth_stats currently does not contain q_imissed. It only has
> q_ipackets, q_opackets, q_ibytes, q_obytes and q_errors. The latter is described as
> "Total number of queue packets received that are dropped.". This is why I have choosen
> q_errors because the field comment sounds like a good match to me.
The comment is open to interpretation.
My preference is that all drivers behave the same, and the reference is one of
the original Intel drivers (ixge, e1000, etc) since they were the first.
>
> As there is no q_imissed, I suggest removing this line from my patch and just adding the imissed total counter:
>
> igb_stats->q_ipackets[i] = internal->rx_queue[i].rx_pkts;
> igb_stats->q_ibytes[i] = internal->rx_queue[i].rx_bytes;
> - igb_stats->q_errors[i] = internal->rx_queue[i].rx_dropped_pkts;
>
> rx_total += igb_stats->q_ipackets[i];
> rx_bytes_total += igb_stats->q_ibytes[i];
> - rx_dropped_total += igb_stats->q_errors[i];
> + rx_dropped_total += internal->rx_queue[i].rx_dropped_pkts;
> rx_nombuf_total += internal->rx_queue[i].rx_nombuf;
>
> Do you agree with that?
Yes, that would be good.
Probably worth renaming igb_stats variable since it is a left over copy/paste from somewhere.
^ permalink raw reply [flat|nested] 9+ messages in thread
* [PATCH v2] net/af_packet: provide packet drop stats
2025-01-16 16:17 [PATCH] net/af_packet: provide packet drop stats Stefan Laesser
2025-01-16 16:24 ` Stephen Hemminger
@ 2025-01-21 12:26 ` Stefan Laesser
2025-01-21 15:48 ` Stephen Hemminger
2025-01-22 7:28 ` [PATCH v3] " Stefan Laesser
1 sibling, 2 replies; 9+ messages in thread
From: Stefan Laesser @ 2025-01-21 12:26 UTC (permalink / raw)
To: dev; +Cc: stephen, Stefan Laesser
The Linux kernel provides the ability to query the packet drop counter
of a socket. This information can be provided when the user requests
stats.
It is important to note that each call to getsockopt with
PACKET_STATISTICS resets the internal counters. So the caller needs to
keep track of the total count on its own.
Next, I have added a counter for the case when mbuf could not be
allocated.
Signed-off-by: Stefan Laesser <stefan.laesser@omicronenergy.com>
---
v2:
* Limited scope of tpacket_stats and socklen_t variables and moved them
inside the for-loop in function eth_stats_get.
* Removed q_errors counting as dropped packets should not be counted
as errors.
* Renamed parameter igb_stats of function eth_stats_get to stats since
it was a copy/paste issue.
drivers/net/af_packet/rte_eth_af_packet.c | 59 +++++++++++++++++------
1 file changed, 43 insertions(+), 16 deletions(-)
diff --git a/drivers/net/af_packet/rte_eth_af_packet.c b/drivers/net/af_packet/rte_eth_af_packet.c
index ceb8d9356a..7033910df8 100644
--- a/drivers/net/af_packet/rte_eth_af_packet.c
+++ b/drivers/net/af_packet/rte_eth_af_packet.c
@@ -58,6 +58,8 @@ struct __rte_cache_aligned pkt_rx_queue {
volatile unsigned long rx_pkts;
volatile unsigned long rx_bytes;
+ volatile unsigned long rx_nombuf;
+ volatile unsigned long rx_dropped_pkts;
};
struct __rte_cache_aligned pkt_tx_queue {
@@ -145,8 +147,10 @@ eth_af_packet_rx(void *queue, struct rte_mbuf **bufs, uint16_t nb_pkts)
/* allocate the next mbuf */
mbuf = rte_pktmbuf_alloc(pkt_q->mb_pool);
- if (unlikely(mbuf == NULL))
+ if (unlikely(mbuf == NULL)) {
+ pkt_q->rx_nombuf++;
break;
+ }
/* packet will fit in the mbuf, go ahead and receive it */
rte_pktmbuf_pkt_len(mbuf) = rte_pktmbuf_data_len(mbuf) = ppd->tp_snaplen;
@@ -414,37 +418,58 @@ eth_dev_info(struct rte_eth_dev *dev, struct rte_eth_dev_info *dev_info)
}
static int
-eth_stats_get(struct rte_eth_dev *dev, struct rte_eth_stats *igb_stats)
+eth_stats_get(struct rte_eth_dev *dev, struct rte_eth_stats *stats)
{
unsigned i, imax;
- unsigned long rx_total = 0, tx_total = 0, tx_err_total = 0;
+ unsigned long rx_total = 0, rx_dropped_total = 0, rx_nombuf_total = 0;
+ unsigned long tx_total = 0, tx_err_total = 0;
unsigned long rx_bytes_total = 0, tx_bytes_total = 0;
const struct pmd_internals *internal = dev->data->dev_private;
imax = (internal->nb_queues < RTE_ETHDEV_QUEUE_STAT_CNTRS ?
internal->nb_queues : RTE_ETHDEV_QUEUE_STAT_CNTRS);
for (i = 0; i < imax; i++) {
- igb_stats->q_ipackets[i] = internal->rx_queue[i].rx_pkts;
- igb_stats->q_ibytes[i] = internal->rx_queue[i].rx_bytes;
- rx_total += igb_stats->q_ipackets[i];
- rx_bytes_total += igb_stats->q_ibytes[i];
+ struct tpacket_stats iface_stats;
+ socklen_t iface_stats_len = sizeof(struct tpacket_stats);
+
+ /* query dropped packets counter from socket */
+ if (internal->rx_queue[i].sockfd != -1 &&
+ getsockopt(internal->rx_queue[i].sockfd, SOL_PACKET,
+ PACKET_STATISTICS, &iface_stats,
+ &iface_stats_len) > -1) {
+ /*
+ * keep total because each call to getsocketopt with PACKET_STATISTICS
+ * reset the counter of the socket
+ */
+ internal->rx_queue[i].rx_dropped_pkts += iface_stats.tp_drops;
+ }
+
+ stats->q_ipackets[i] = internal->rx_queue[i].rx_pkts;
+ stats->q_ibytes[i] = internal->rx_queue[i].rx_bytes;
+
+ rx_total += stats->q_ipackets[i];
+ rx_bytes_total += stats->q_ibytes[i];
+ rx_dropped_total += internal->rx_queue[i].rx_dropped_pkts;
+ rx_nombuf_total += internal->rx_queue[i].rx_nombuf;
}
imax = (internal->nb_queues < RTE_ETHDEV_QUEUE_STAT_CNTRS ?
internal->nb_queues : RTE_ETHDEV_QUEUE_STAT_CNTRS);
for (i = 0; i < imax; i++) {
- igb_stats->q_opackets[i] = internal->tx_queue[i].tx_pkts;
- igb_stats->q_obytes[i] = internal->tx_queue[i].tx_bytes;
- tx_total += igb_stats->q_opackets[i];
+ stats->q_opackets[i] = internal->tx_queue[i].tx_pkts;
+ stats->q_obytes[i] = internal->tx_queue[i].tx_bytes;
+ tx_total += stats->q_opackets[i];
tx_err_total += internal->tx_queue[i].err_pkts;
- tx_bytes_total += igb_stats->q_obytes[i];
+ tx_bytes_total += stats->q_obytes[i];
}
- igb_stats->ipackets = rx_total;
- igb_stats->ibytes = rx_bytes_total;
- igb_stats->opackets = tx_total;
- igb_stats->oerrors = tx_err_total;
- igb_stats->obytes = tx_bytes_total;
+ stats->ipackets = rx_total;
+ stats->ibytes = rx_bytes_total;
+ stats->imissed = rx_dropped_total;
+ stats->rx_nombuf = rx_nombuf_total;
+ stats->opackets = tx_total;
+ stats->oerrors = tx_err_total;
+ stats->obytes = tx_bytes_total;
return 0;
}
@@ -457,6 +482,8 @@ eth_stats_reset(struct rte_eth_dev *dev)
for (i = 0; i < internal->nb_queues; i++) {
internal->rx_queue[i].rx_pkts = 0;
internal->rx_queue[i].rx_bytes = 0;
+ internal->rx_queue[i].rx_nombuf = 0;
+ internal->rx_queue[i].rx_dropped_pkts = 0;
}
for (i = 0; i < internal->nb_queues; i++) {
--
2.34.1
^ permalink raw reply [flat|nested] 9+ messages in thread
* Re: [PATCH v2] net/af_packet: provide packet drop stats
2025-01-21 12:26 ` [PATCH v2] " Stefan Laesser
@ 2025-01-21 15:48 ` Stephen Hemminger
2025-01-22 7:03 ` Stefan Lässer
2025-01-22 7:28 ` [PATCH v3] " Stefan Laesser
1 sibling, 1 reply; 9+ messages in thread
From: Stephen Hemminger @ 2025-01-21 15:48 UTC (permalink / raw)
To: Stefan Laesser; +Cc: dev
On Tue, 21 Jan 2025 13:26:16 +0100
Stefan Laesser <stefan.laesser@omicronenergy.com> wrote:
> + /* query dropped packets counter from socket */
> + if (internal->rx_queue[i].sockfd != -1 &&
> + getsockopt(internal->rx_queue[i].sockfd, SOL_PACKET,
> + PACKET_STATISTICS, &iface_stats,
> + &iface_stats_len) > -1) {
> + /*
> + * keep total because each call to getsocketopt with PACKET_STATISTICS
> + * reset the counter of the socket
> + */
> + internal->rx_queue[i].rx_dropped_pkts += iface_stats.tp_drops;
> + }
Since reading the value clears it, maybe stats_reset should call as well?
Also, queues greater that the RTE_ETHDEV_STAT_CNTRS should still count
against the total. And the two loops could be combined.
Maybe something like this?
diff --git a/drivers/net/af_packet/rte_eth_af_packet.c b/drivers/net/af_packet/rte_eth_af_packet.c
index 7033910df8..af08e893f1 100644
--- a/drivers/net/af_packet/rte_eth_af_packet.c
+++ b/drivers/net/af_packet/rte_eth_af_packet.c
@@ -417,50 +417,51 @@ eth_dev_info(struct rte_eth_dev *dev, struct rte_eth_dev_info *dev_info)
return 0;
}
+/* query dropped packets counter from socket */
+static unsigned int
+packet_drop_count(int sockfd)
+{
+ struct tpacket_stats pkt_stats;
+ socklen_t pkt_stats_len = sizeof(struct tpacket_stats);
+
+ if (sockfd == -1)
+ return 0;
+
+ if (getsockopt(sockfd, SOL_PACKET, PACKET_STATISTICS, &pkt_stats, &pkt_stats_len) < -1)
+ return 0;
+
+ return pkt_stats.tp_drops;
+}
+
static int
eth_stats_get(struct rte_eth_dev *dev, struct rte_eth_stats *stats)
{
- unsigned i, imax;
+ unsigned int i;
unsigned long rx_total = 0, rx_dropped_total = 0, rx_nombuf_total = 0;
unsigned long tx_total = 0, tx_err_total = 0;
unsigned long rx_bytes_total = 0, tx_bytes_total = 0;
const struct pmd_internals *internal = dev->data->dev_private;
- imax = (internal->nb_queues < RTE_ETHDEV_QUEUE_STAT_CNTRS ?
- internal->nb_queues : RTE_ETHDEV_QUEUE_STAT_CNTRS);
- for (i = 0; i < imax; i++) {
- struct tpacket_stats iface_stats;
- socklen_t iface_stats_len = sizeof(struct tpacket_stats);
-
- /* query dropped packets counter from socket */
- if (internal->rx_queue[i].sockfd != -1 &&
- getsockopt(internal->rx_queue[i].sockfd, SOL_PACKET,
- PACKET_STATISTICS, &iface_stats,
- &iface_stats_len) > -1) {
- /*
- * keep total because each call to getsocketopt with PACKET_STATISTICS
- * reset the counter of the socket
- */
- internal->rx_queue[i].rx_dropped_pkts += iface_stats.tp_drops;
- }
-
- stats->q_ipackets[i] = internal->rx_queue[i].rx_pkts;
- stats->q_ibytes[i] = internal->rx_queue[i].rx_bytes;
+ for (i = 0; i < internal->nb_queues; i++) {
+ /* reading drop count clears the value */
+ internal->rx_queue[i].rx_dropped_pkts +=
+ packet_drop_count(internal->rx_queue[i].sockfd);
rx_total += stats->q_ipackets[i];
rx_bytes_total += stats->q_ibytes[i];
rx_dropped_total += internal->rx_queue[i].rx_dropped_pkts;
rx_nombuf_total += internal->rx_queue[i].rx_nombuf;
- }
- imax = (internal->nb_queues < RTE_ETHDEV_QUEUE_STAT_CNTRS ?
- internal->nb_queues : RTE_ETHDEV_QUEUE_STAT_CNTRS);
- for (i = 0; i < imax; i++) {
- stats->q_opackets[i] = internal->tx_queue[i].tx_pkts;
- stats->q_obytes[i] = internal->tx_queue[i].tx_bytes;
tx_total += stats->q_opackets[i];
tx_err_total += internal->tx_queue[i].err_pkts;
tx_bytes_total += stats->q_obytes[i];
+
+ if (i < RTE_ETHDEV_QUEUE_STAT_CNTRS) {
+ stats->q_ipackets[i] = internal->rx_queue[i].rx_pkts;
+ stats->q_ibytes[i] = internal->rx_queue[i].rx_bytes;
+ stats->q_opackets[i] = internal->tx_queue[i].tx_pkts;
+ stats->q_obytes[i] = internal->tx_queue[i].tx_bytes;
+ }
}
stats->ipackets = rx_total;
@@ -480,6 +481,7 @@ eth_stats_reset(struct rte_eth_dev *dev)
struct pmd_internals *internal = dev->data->dev_private;
for (i = 0; i < internal->nb_queues; i++) {
+ packet_drop_count(internal->rx_queue[i].sockfd);
internal->rx_queue[i].rx_pkts = 0;
internal->rx_queue[i].rx_bytes = 0;
internal->rx_queue[i].rx_nombuf = 0;
^ permalink raw reply [flat|nested] 9+ messages in thread
* RE: [PATCH v2] net/af_packet: provide packet drop stats
2025-01-21 15:48 ` Stephen Hemminger
@ 2025-01-22 7:03 ` Stefan Lässer
0 siblings, 0 replies; 9+ messages in thread
From: Stefan Lässer @ 2025-01-22 7:03 UTC (permalink / raw)
To: Stephen Hemminger; +Cc: dev
> -----Original Message-----
> From: Stephen Hemminger <stephen@networkplumber.org>
> Sent: Tuesday, January 21, 2025 4:48 PM
> To: Stefan Lässer <stefan.laesser@omicronenergy.com>
> Cc: dev@dpdk.org
> Subject: Re: [PATCH v2] net/af_packet: provide packet drop stats
>
> On Tue, 21 Jan 2025 13:26:16 +0100
> Stefan Laesser <stefan.laesser@omicronenergy.com> wrote:
>
> > + /* query dropped packets counter from socket */
> > + if (internal->rx_queue[i].sockfd != -1 &&
> > + getsockopt(internal->rx_queue[i].sockfd, SOL_PACKET,
> > + PACKET_STATISTICS,
> &iface_stats,
> > + &iface_stats_len) > -1) {
> > + /*
> > + * keep total because each call to getsocketopt with
> PACKET_STATISTICS
> > + * reset the counter of the socket
> > + */
> > + internal->rx_queue[i].rx_dropped_pkts +=
> iface_stats.tp_drops;
> > + }
>
> Since reading the value clears it, maybe stats_reset should call as well?
>
> Also, queues greater that the RTE_ETHDEV_STAT_CNTRS should still count
> against the total. And the two loops could be combined.
>
> Maybe something like this?
That looks very reasonable - I will update my patch.
>
> diff --git a/drivers/net/af_packet/rte_eth_af_packet.c
> b/drivers/net/af_packet/rte_eth_af_packet.c
> index 7033910df8..af08e893f1 100644
> --- a/drivers/net/af_packet/rte_eth_af_packet.c
> +++ b/drivers/net/af_packet/rte_eth_af_packet.c
> @@ -417,50 +417,51 @@ eth_dev_info(struct rte_eth_dev *dev, struct
> rte_eth_dev_info *dev_info)
> return 0;
> }
>
> +/* query dropped packets counter from socket */ static unsigned int
> +packet_drop_count(int sockfd) {
> + struct tpacket_stats pkt_stats;
> + socklen_t pkt_stats_len = sizeof(struct tpacket_stats);
> +
> + if (sockfd == -1)
> + return 0;
> +
> + if (getsockopt(sockfd, SOL_PACKET, PACKET_STATISTICS, &pkt_stats,
> &pkt_stats_len) < -1)
> + return 0;
> +
> + return pkt_stats.tp_drops;
> +}
> +
> static int
> eth_stats_get(struct rte_eth_dev *dev, struct rte_eth_stats *stats) {
> - unsigned i, imax;
> + unsigned int i;
> unsigned long rx_total = 0, rx_dropped_total = 0, rx_nombuf_total = 0;
> unsigned long tx_total = 0, tx_err_total = 0;
> unsigned long rx_bytes_total = 0, tx_bytes_total = 0;
> const struct pmd_internals *internal = dev->data->dev_private;
>
> - imax = (internal->nb_queues < RTE_ETHDEV_QUEUE_STAT_CNTRS ?
> - internal->nb_queues : RTE_ETHDEV_QUEUE_STAT_CNTRS);
> - for (i = 0; i < imax; i++) {
> - struct tpacket_stats iface_stats;
> - socklen_t iface_stats_len = sizeof(struct tpacket_stats);
> -
> - /* query dropped packets counter from socket */
> - if (internal->rx_queue[i].sockfd != -1 &&
> - getsockopt(internal->rx_queue[i].sockfd, SOL_PACKET,
> - PACKET_STATISTICS,
> &iface_stats,
> - &iface_stats_len) > -1) {
> - /*
> - * keep total because each call to getsocketopt with
> PACKET_STATISTICS
> - * reset the counter of the socket
> - */
> - internal->rx_queue[i].rx_dropped_pkts +=
> iface_stats.tp_drops;
> - }
> -
> - stats->q_ipackets[i] = internal->rx_queue[i].rx_pkts;
> - stats->q_ibytes[i] = internal->rx_queue[i].rx_bytes;
> + for (i = 0; i < internal->nb_queues; i++) {
> + /* reading drop count clears the value */
> + internal->rx_queue[i].rx_dropped_pkts +=
> + packet_drop_count(internal->rx_queue[i].sockfd);
>
> rx_total += stats->q_ipackets[i];
> rx_bytes_total += stats->q_ibytes[i];
> rx_dropped_total += internal->rx_queue[i].rx_dropped_pkts;
> rx_nombuf_total += internal->rx_queue[i].rx_nombuf;
> - }
>
> - imax = (internal->nb_queues < RTE_ETHDEV_QUEUE_STAT_CNTRS ?
> - internal->nb_queues : RTE_ETHDEV_QUEUE_STAT_CNTRS);
> - for (i = 0; i < imax; i++) {
> - stats->q_opackets[i] = internal->tx_queue[i].tx_pkts;
> - stats->q_obytes[i] = internal->tx_queue[i].tx_bytes;
> tx_total += stats->q_opackets[i];
> tx_err_total += internal->tx_queue[i].err_pkts;
> tx_bytes_total += stats->q_obytes[i];
> +
> + if (i < RTE_ETHDEV_QUEUE_STAT_CNTRS) {
> + stats->q_ipackets[i] = internal->rx_queue[i].rx_pkts;
> + stats->q_ibytes[i] = internal->rx_queue[i].rx_bytes;
> + stats->q_opackets[i] = internal->tx_queue[i].tx_pkts;
> + stats->q_obytes[i] = internal->tx_queue[i].tx_bytes;
> + }
> }
>
> stats->ipackets = rx_total;
> @@ -480,6 +481,7 @@ eth_stats_reset(struct rte_eth_dev *dev)
> struct pmd_internals *internal = dev->data->dev_private;
>
> for (i = 0; i < internal->nb_queues; i++) {
> + packet_drop_count(internal->rx_queue[i].sockfd);
> internal->rx_queue[i].rx_pkts = 0;
> internal->rx_queue[i].rx_bytes = 0;
> internal->rx_queue[i].rx_nombuf = 0;
^ permalink raw reply [flat|nested] 9+ messages in thread
* [PATCH v3] net/af_packet: provide packet drop stats
2025-01-21 12:26 ` [PATCH v2] " Stefan Laesser
2025-01-21 15:48 ` Stephen Hemminger
@ 2025-01-22 7:28 ` Stefan Laesser
2025-01-22 15:56 ` [PATCH] doc: add release not for af_packet Stephen Hemminger
1 sibling, 1 reply; 9+ messages in thread
From: Stefan Laesser @ 2025-01-22 7:28 UTC (permalink / raw)
To: dev; +Cc: stephen, Stefan Laesser
The Linux kernel provides the ability to query the packet drop counter
of a socket. This information can be provided when the user requests
stats.
It is important to note that each call to getsockopt with
PACKET_STATISTICS resets the internal counters. So the caller needs to
keep track of the total count on its own.
Next, I have added a counter for the case when mbuf could not be
allocated.
Signed-off-by: Stefan Laesser <stefan.laesser@omicronenergy.com>
---
v3:
* Introduced function for querying socket dropped counter and use it also
to reset internal state when stats are reset.
* Combined the for-loops in in eth_stats_get and eth_stats_reset.
* Count all queues to total counters, but only add up to RTE_ETHDEV_QUEUE_STAT_CNTRS
values to queue stats.
v2:
* Limited scope of tpacket_stats and socklen_t variables and moved them
inside the for-loop in function eth_stats_get.
* Removed q_errors counting as dropped packets should not be counted
as errors.
* Renamed parameter igb_stats of function eth_stats_get to stats since
it was a copy/paste issue.
drivers/net/af_packet/rte_eth_af_packet.c | 86 ++++++++++++++++-------
1 file changed, 60 insertions(+), 26 deletions(-)
diff --git a/drivers/net/af_packet/rte_eth_af_packet.c b/drivers/net/af_packet/rte_eth_af_packet.c
index ceb8d9356a..c088deb587 100644
--- a/drivers/net/af_packet/rte_eth_af_packet.c
+++ b/drivers/net/af_packet/rte_eth_af_packet.c
@@ -58,6 +58,8 @@ struct __rte_cache_aligned pkt_rx_queue {
volatile unsigned long rx_pkts;
volatile unsigned long rx_bytes;
+ volatile unsigned long rx_nombuf;
+ volatile unsigned long rx_dropped_pkts;
};
struct __rte_cache_aligned pkt_tx_queue {
@@ -145,8 +147,10 @@ eth_af_packet_rx(void *queue, struct rte_mbuf **bufs, uint16_t nb_pkts)
/* allocate the next mbuf */
mbuf = rte_pktmbuf_alloc(pkt_q->mb_pool);
- if (unlikely(mbuf == NULL))
+ if (unlikely(mbuf == NULL)) {
+ pkt_q->rx_nombuf++;
break;
+ }
/* packet will fit in the mbuf, go ahead and receive it */
rte_pktmbuf_pkt_len(mbuf) = rte_pktmbuf_data_len(mbuf) = ppd->tp_snaplen;
@@ -413,38 +417,65 @@ eth_dev_info(struct rte_eth_dev *dev, struct rte_eth_dev_info *dev_info)
return 0;
}
+
+/*
+ * Query dropped packets counter from socket.
+ * Reading drop count clears the value of the socket!
+ */
+static unsigned int
+packet_drop_count(int sockfd)
+{
+ struct tpacket_stats pkt_stats;
+ socklen_t pkt_stats_len = sizeof(struct tpacket_stats);
+
+ if (sockfd == -1)
+ return 0;
+
+ if (getsockopt(sockfd, SOL_PACKET, PACKET_STATISTICS, &pkt_stats,
+ &pkt_stats_len) < -1)
+ return 0;
+
+ return pkt_stats.tp_drops;
+}
+
static int
-eth_stats_get(struct rte_eth_dev *dev, struct rte_eth_stats *igb_stats)
+eth_stats_get(struct rte_eth_dev *dev, struct rte_eth_stats *stats)
{
- unsigned i, imax;
- unsigned long rx_total = 0, tx_total = 0, tx_err_total = 0;
+ unsigned int i;
+ unsigned long rx_total = 0, rx_dropped_total = 0, rx_nombuf_total = 0;
+ unsigned long tx_total = 0, tx_err_total = 0;
unsigned long rx_bytes_total = 0, tx_bytes_total = 0;
const struct pmd_internals *internal = dev->data->dev_private;
- imax = (internal->nb_queues < RTE_ETHDEV_QUEUE_STAT_CNTRS ?
- internal->nb_queues : RTE_ETHDEV_QUEUE_STAT_CNTRS);
- for (i = 0; i < imax; i++) {
- igb_stats->q_ipackets[i] = internal->rx_queue[i].rx_pkts;
- igb_stats->q_ibytes[i] = internal->rx_queue[i].rx_bytes;
- rx_total += igb_stats->q_ipackets[i];
- rx_bytes_total += igb_stats->q_ibytes[i];
- }
+ for (i = 0; i < internal->nb_queues; i++) {
+ /* reading drop count clears the value, therefore keep total value */
+ internal->rx_queue[i].rx_dropped_pkts +=
+ packet_drop_count(internal->rx_queue[i].sockfd);
+
+ rx_total += internal->rx_queue[i].rx_pkts;
+ rx_bytes_total += internal->rx_queue[i].rx_bytes;
+ rx_dropped_total += internal->rx_queue[i].rx_dropped_pkts;
+ rx_nombuf_total += internal->rx_queue[i].rx_nombuf;
- imax = (internal->nb_queues < RTE_ETHDEV_QUEUE_STAT_CNTRS ?
- internal->nb_queues : RTE_ETHDEV_QUEUE_STAT_CNTRS);
- for (i = 0; i < imax; i++) {
- igb_stats->q_opackets[i] = internal->tx_queue[i].tx_pkts;
- igb_stats->q_obytes[i] = internal->tx_queue[i].tx_bytes;
- tx_total += igb_stats->q_opackets[i];
+ tx_total += internal->tx_queue[i].tx_pkts;
tx_err_total += internal->tx_queue[i].err_pkts;
- tx_bytes_total += igb_stats->q_obytes[i];
+ tx_bytes_total += internal->tx_queue[i].tx_bytes;
+
+ if (i < RTE_ETHDEV_QUEUE_STAT_CNTRS) {
+ stats->q_ipackets[i] = internal->rx_queue[i].rx_pkts;
+ stats->q_ibytes[i] = internal->rx_queue[i].rx_bytes;
+ stats->q_opackets[i] = internal->tx_queue[i].tx_pkts;
+ stats->q_obytes[i] = internal->tx_queue[i].tx_bytes;
+ }
}
- igb_stats->ipackets = rx_total;
- igb_stats->ibytes = rx_bytes_total;
- igb_stats->opackets = tx_total;
- igb_stats->oerrors = tx_err_total;
- igb_stats->obytes = tx_bytes_total;
+ stats->ipackets = rx_total;
+ stats->ibytes = rx_bytes_total;
+ stats->imissed = rx_dropped_total;
+ stats->rx_nombuf = rx_nombuf_total;
+ stats->opackets = tx_total;
+ stats->oerrors = tx_err_total;
+ stats->obytes = tx_bytes_total;
return 0;
}
@@ -455,11 +486,14 @@ eth_stats_reset(struct rte_eth_dev *dev)
struct pmd_internals *internal = dev->data->dev_private;
for (i = 0; i < internal->nb_queues; i++) {
+ /* clear socket counter */
+ packet_drop_count(internal->rx_queue[i].sockfd);
+
internal->rx_queue[i].rx_pkts = 0;
internal->rx_queue[i].rx_bytes = 0;
- }
+ internal->rx_queue[i].rx_nombuf = 0;
+ internal->rx_queue[i].rx_dropped_pkts = 0;
- for (i = 0; i < internal->nb_queues; i++) {
internal->tx_queue[i].tx_pkts = 0;
internal->tx_queue[i].err_pkts = 0;
internal->tx_queue[i].tx_bytes = 0;
--
2.34.1
^ permalink raw reply [flat|nested] 9+ messages in thread
* [PATCH] doc: add release not for af_packet
2025-01-22 7:28 ` [PATCH v3] " Stefan Laesser
@ 2025-01-22 15:56 ` Stephen Hemminger
0 siblings, 0 replies; 9+ messages in thread
From: Stephen Hemminger @ 2025-01-22 15:56 UTC (permalink / raw)
To: dev; +Cc: Stephen Hemminger
Add short release not about enhancements to af_packet.
Signed-off-by: Stephen Hemminger <stephen@networkplumber.org>
---
doc/guides/rel_notes/release_25_03.rst | 5 +++++
1 file changed, 5 insertions(+)
diff --git a/doc/guides/rel_notes/release_25_03.rst b/doc/guides/rel_notes/release_25_03.rst
index 85986ffa61..21ec44f0ab 100644
--- a/doc/guides/rel_notes/release_25_03.rst
+++ b/doc/guides/rel_notes/release_25_03.rst
@@ -63,6 +63,11 @@ New Features
and even substantial part of its code.
It can be viewed as an extension of rte_ring functionality.
+* **Enhancements to af_packet net driver.**
+
+ Added ability to option to configure receive packet fanout mode.
+ Improved statistics including failed receive buffer allocation
+ and missed packets.
Removed Items
-------------
--
2.45.2
^ permalink raw reply [flat|nested] 9+ messages in thread
end of thread, other threads:[~2025-01-22 15:57 UTC | newest]
Thread overview: 9+ messages (download: mbox.gz / follow: Atom feed)
-- links below jump to the message on this page --
2025-01-16 16:17 [PATCH] net/af_packet: provide packet drop stats Stefan Laesser
2025-01-16 16:24 ` Stephen Hemminger
2025-01-17 7:20 ` Stefan Lässer
2025-01-17 16:43 ` Stephen Hemminger
2025-01-21 12:26 ` [PATCH v2] " Stefan Laesser
2025-01-21 15:48 ` Stephen Hemminger
2025-01-22 7:03 ` Stefan Lässer
2025-01-22 7:28 ` [PATCH v3] " Stefan Laesser
2025-01-22 15:56 ` [PATCH] doc: add release not for af_packet Stephen Hemminger
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).