Commit 257310e9 authored by Przemek Kitszel's avatar Przemek Kitszel Committed by Tony Nguyen

ice: fix stats being updated by way too large values

Simplify stats accumulation logic to fix the case where we don't take
previous stat value into account, we should always respect it.

Main netdev stats of our PF (Tx/Rx packets/bytes) were reported orders of
magnitude too big during OpenStack reconfiguration events, possibly other
reconfiguration cases too.

The regression was reported to be between 6.1 and 6.2, so I was almost
certain that on of the two "preserve stats over reset" commits were the
culprit. While reading the code, it was found that in some cases we will
increase the stats by arbitrarily large number (thanks to ignoring "-prev"
part of condition, after zeroing it).

Note that this fixes also the case where we were around limits of u64, but
that was not the regression reported.

Full disclosure: I remember suggesting this particular piece of code to
Ben a few years ago, so blame on me.

Fixes: 2fd5e433 ("ice: Accumulate HW and Netdev statistics over reset")
Reported-by: default avatarNebojsa Stevanovic <nebojsa.stevanovic@gcore.com>
Link: https://lore.kernel.org/intel-wired-lan/VI1PR02MB439744DEDAA7B59B9A2833FE912EA@VI1PR02MB4397.eurprd02.prod.outlook.comReported-by: default avatarChristian Rohmann <christian.rohmann@inovex.de>
Link: https://lore.kernel.org/intel-wired-lan/f38a6ca4-af05-48b1-a3e6-17ef2054e525@inovex.deReviewed-by: default avatarJacob Keller <jacob.e.keller@intel.com>
Signed-off-by: default avatarPrzemek Kitszel <przemyslaw.kitszel@intel.com>
Reviewed-by: default avatarSimon Horman <horms@kernel.org>
Tested-by: Pucha Himasekhar Reddy <himasekharx.reddy.pucha@intel.com> (A Contingent worker at Intel)
Signed-off-by: default avatarTony Nguyen <anthony.l.nguyen@intel.com>
parent ee14cc9e
...@@ -6737,6 +6737,7 @@ static void ice_update_vsi_ring_stats(struct ice_vsi *vsi) ...@@ -6737,6 +6737,7 @@ static void ice_update_vsi_ring_stats(struct ice_vsi *vsi)
{ {
struct rtnl_link_stats64 *net_stats, *stats_prev; struct rtnl_link_stats64 *net_stats, *stats_prev;
struct rtnl_link_stats64 *vsi_stats; struct rtnl_link_stats64 *vsi_stats;
struct ice_pf *pf = vsi->back;
u64 pkts, bytes; u64 pkts, bytes;
int i; int i;
...@@ -6782,20 +6783,17 @@ static void ice_update_vsi_ring_stats(struct ice_vsi *vsi) ...@@ -6782,20 +6783,17 @@ static void ice_update_vsi_ring_stats(struct ice_vsi *vsi)
net_stats = &vsi->net_stats; net_stats = &vsi->net_stats;
stats_prev = &vsi->net_stats_prev; stats_prev = &vsi->net_stats_prev;
/* clear prev counters after reset */ /* Update netdev counters, but keep in mind that values could start at
if (vsi_stats->tx_packets < stats_prev->tx_packets || * random value after PF reset. And as we increase the reported stat by
vsi_stats->rx_packets < stats_prev->rx_packets) { * diff of Prev-Cur, we need to be sure that Prev is valid. If it's not,
stats_prev->tx_packets = 0; * let's skip this round.
stats_prev->tx_bytes = 0; */
stats_prev->rx_packets = 0; if (likely(pf->stat_prev_loaded)) {
stats_prev->rx_bytes = 0;
}
/* update netdev counters */
net_stats->tx_packets += vsi_stats->tx_packets - stats_prev->tx_packets; net_stats->tx_packets += vsi_stats->tx_packets - stats_prev->tx_packets;
net_stats->tx_bytes += vsi_stats->tx_bytes - stats_prev->tx_bytes; net_stats->tx_bytes += vsi_stats->tx_bytes - stats_prev->tx_bytes;
net_stats->rx_packets += vsi_stats->rx_packets - stats_prev->rx_packets; net_stats->rx_packets += vsi_stats->rx_packets - stats_prev->rx_packets;
net_stats->rx_bytes += vsi_stats->rx_bytes - stats_prev->rx_bytes; net_stats->rx_bytes += vsi_stats->rx_bytes - stats_prev->rx_bytes;
}
stats_prev->tx_packets = vsi_stats->tx_packets; stats_prev->tx_packets = vsi_stats->tx_packets;
stats_prev->tx_bytes = vsi_stats->tx_bytes; stats_prev->tx_bytes = vsi_stats->tx_bytes;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment