cxgbe(4): Tidy up PAUSE frame accounting.
Figure out if the chip is counting PAUSE frames in the "normal" stats and take them out if it is. This fixes a bug in the tx stats because the default hardware behavior is different for Tx and Rx but the driver was treating both the same way. The result was that OPACKETS, OBYTES, and OMCASTS were under-reported (if tx_pause > 0) before this change. Note that the mac_stats sysctl still gives you the raw value of these statistics straight from the device registers.
This commit is contained in:
parent
4215404617
commit
784a631dc5
@ -5615,6 +5615,7 @@ void t4_get_port_stats_offset(struct adapter *adap, int idx,
|
|||||||
void t4_get_port_stats(struct adapter *adap, int idx, struct port_stats *p)
|
void t4_get_port_stats(struct adapter *adap, int idx, struct port_stats *p)
|
||||||
{
|
{
|
||||||
u32 bgmap = t4_get_mps_bg_map(adap, idx);
|
u32 bgmap = t4_get_mps_bg_map(adap, idx);
|
||||||
|
u32 stat_ctl;
|
||||||
|
|
||||||
#define GET_STAT(name) \
|
#define GET_STAT(name) \
|
||||||
t4_read_reg64(adap, \
|
t4_read_reg64(adap, \
|
||||||
@ -5622,6 +5623,8 @@ void t4_get_port_stats(struct adapter *adap, int idx, struct port_stats *p)
|
|||||||
T5_PORT_REG(idx, A_MPS_PORT_STAT_##name##_L)))
|
T5_PORT_REG(idx, A_MPS_PORT_STAT_##name##_L)))
|
||||||
#define GET_STAT_COM(name) t4_read_reg64(adap, A_MPS_STAT_##name##_L)
|
#define GET_STAT_COM(name) t4_read_reg64(adap, A_MPS_STAT_##name##_L)
|
||||||
|
|
||||||
|
stat_ctl = t4_read_reg(adap, A_MPS_STAT_CTL);
|
||||||
|
|
||||||
p->tx_pause = GET_STAT(TX_PORT_PAUSE);
|
p->tx_pause = GET_STAT(TX_PORT_PAUSE);
|
||||||
p->tx_octets = GET_STAT(TX_PORT_BYTES);
|
p->tx_octets = GET_STAT(TX_PORT_BYTES);
|
||||||
p->tx_frames = GET_STAT(TX_PORT_FRAMES);
|
p->tx_frames = GET_STAT(TX_PORT_FRAMES);
|
||||||
@ -5646,6 +5649,12 @@ void t4_get_port_stats(struct adapter *adap, int idx, struct port_stats *p)
|
|||||||
p->tx_ppp6 = GET_STAT(TX_PORT_PPP6);
|
p->tx_ppp6 = GET_STAT(TX_PORT_PPP6);
|
||||||
p->tx_ppp7 = GET_STAT(TX_PORT_PPP7);
|
p->tx_ppp7 = GET_STAT(TX_PORT_PPP7);
|
||||||
|
|
||||||
|
if (stat_ctl & F_COUNTPAUSESTATTX) {
|
||||||
|
p->tx_frames -= p->tx_pause;
|
||||||
|
p->tx_octets -= p->tx_pause * 64;
|
||||||
|
p->tx_mcast_frames -= p->tx_pause;
|
||||||
|
}
|
||||||
|
|
||||||
p->rx_pause = GET_STAT(RX_PORT_PAUSE);
|
p->rx_pause = GET_STAT(RX_PORT_PAUSE);
|
||||||
p->rx_octets = GET_STAT(RX_PORT_BYTES);
|
p->rx_octets = GET_STAT(RX_PORT_BYTES);
|
||||||
p->rx_frames = GET_STAT(RX_PORT_FRAMES);
|
p->rx_frames = GET_STAT(RX_PORT_FRAMES);
|
||||||
@ -5674,6 +5683,12 @@ void t4_get_port_stats(struct adapter *adap, int idx, struct port_stats *p)
|
|||||||
p->rx_ppp6 = GET_STAT(RX_PORT_PPP6);
|
p->rx_ppp6 = GET_STAT(RX_PORT_PPP6);
|
||||||
p->rx_ppp7 = GET_STAT(RX_PORT_PPP7);
|
p->rx_ppp7 = GET_STAT(RX_PORT_PPP7);
|
||||||
|
|
||||||
|
if (stat_ctl & F_COUNTPAUSESTATRX) {
|
||||||
|
p->rx_frames -= p->rx_pause;
|
||||||
|
p->rx_octets -= p->rx_pause * 64;
|
||||||
|
p->rx_mcast_frames -= p->rx_pause;
|
||||||
|
}
|
||||||
|
|
||||||
p->rx_ovflow0 = (bgmap & 1) ? GET_STAT_COM(RX_BG_0_MAC_DROP_FRAME) : 0;
|
p->rx_ovflow0 = (bgmap & 1) ? GET_STAT_COM(RX_BG_0_MAC_DROP_FRAME) : 0;
|
||||||
p->rx_ovflow1 = (bgmap & 2) ? GET_STAT_COM(RX_BG_1_MAC_DROP_FRAME) : 0;
|
p->rx_ovflow1 = (bgmap & 2) ? GET_STAT_COM(RX_BG_1_MAC_DROP_FRAME) : 0;
|
||||||
p->rx_ovflow2 = (bgmap & 4) ? GET_STAT_COM(RX_BG_2_MAC_DROP_FRAME) : 0;
|
p->rx_ovflow2 = (bgmap & 4) ? GET_STAT_COM(RX_BG_2_MAC_DROP_FRAME) : 0;
|
||||||
|
@ -1732,29 +1732,29 @@ cxgbe_get_counter(struct ifnet *ifp, ift_counter c)
|
|||||||
|
|
||||||
switch (c) {
|
switch (c) {
|
||||||
case IFCOUNTER_IPACKETS:
|
case IFCOUNTER_IPACKETS:
|
||||||
return (s->rx_frames - s->rx_pause);
|
return (s->rx_frames);
|
||||||
|
|
||||||
case IFCOUNTER_IERRORS:
|
case IFCOUNTER_IERRORS:
|
||||||
return (s->rx_jabber + s->rx_runt + s->rx_too_long +
|
return (s->rx_jabber + s->rx_runt + s->rx_too_long +
|
||||||
s->rx_fcs_err + s->rx_len_err);
|
s->rx_fcs_err + s->rx_len_err);
|
||||||
|
|
||||||
case IFCOUNTER_OPACKETS:
|
case IFCOUNTER_OPACKETS:
|
||||||
return (s->tx_frames - s->tx_pause);
|
return (s->tx_frames);
|
||||||
|
|
||||||
case IFCOUNTER_OERRORS:
|
case IFCOUNTER_OERRORS:
|
||||||
return (s->tx_error_frames);
|
return (s->tx_error_frames);
|
||||||
|
|
||||||
case IFCOUNTER_IBYTES:
|
case IFCOUNTER_IBYTES:
|
||||||
return (s->rx_octets - s->rx_pause * 64);
|
return (s->rx_octets);
|
||||||
|
|
||||||
case IFCOUNTER_OBYTES:
|
case IFCOUNTER_OBYTES:
|
||||||
return (s->tx_octets - s->tx_pause * 64);
|
return (s->tx_octets);
|
||||||
|
|
||||||
case IFCOUNTER_IMCASTS:
|
case IFCOUNTER_IMCASTS:
|
||||||
return (s->rx_mcast_frames - s->rx_pause);
|
return (s->rx_mcast_frames);
|
||||||
|
|
||||||
case IFCOUNTER_OMCASTS:
|
case IFCOUNTER_OMCASTS:
|
||||||
return (s->tx_mcast_frames - s->tx_pause);
|
return (s->tx_mcast_frames);
|
||||||
|
|
||||||
case IFCOUNTER_IQDROPS:
|
case IFCOUNTER_IQDROPS:
|
||||||
return (s->rx_ovflow0 + s->rx_ovflow1 + s->rx_ovflow2 +
|
return (s->rx_ovflow0 + s->rx_ovflow1 + s->rx_ovflow2 +
|
||||||
|
Loading…
x
Reference in New Issue
Block a user