forked from Minki/linux
Merge branch 'net-dsa-add-pause-stats-support'
Oleksij Rempel says: ==================== net: dsa: add pause stats support ==================== Link: https://lore.kernel.org/r/20220628085155.2591201-1-o.rempel@pengutronix.de Signed-off-by: Jakub Kicinski <kuba@kernel.org>
This commit is contained in:
commit
2a832912db
@ -671,18 +671,22 @@ static void ksz_phylink_get_caps(struct dsa_switch *ds, int port,
|
|||||||
|
|
||||||
void ksz_r_mib_stats64(struct ksz_device *dev, int port)
|
void ksz_r_mib_stats64(struct ksz_device *dev, int port)
|
||||||
{
|
{
|
||||||
|
struct ethtool_pause_stats *pstats;
|
||||||
struct rtnl_link_stats64 *stats;
|
struct rtnl_link_stats64 *stats;
|
||||||
struct ksz_stats_raw *raw;
|
struct ksz_stats_raw *raw;
|
||||||
struct ksz_port_mib *mib;
|
struct ksz_port_mib *mib;
|
||||||
|
|
||||||
mib = &dev->ports[port].mib;
|
mib = &dev->ports[port].mib;
|
||||||
stats = &mib->stats64;
|
stats = &mib->stats64;
|
||||||
|
pstats = &mib->pause_stats;
|
||||||
raw = (struct ksz_stats_raw *)mib->counters;
|
raw = (struct ksz_stats_raw *)mib->counters;
|
||||||
|
|
||||||
spin_lock(&mib->stats64_lock);
|
spin_lock(&mib->stats64_lock);
|
||||||
|
|
||||||
stats->rx_packets = raw->rx_bcast + raw->rx_mcast + raw->rx_ucast;
|
stats->rx_packets = raw->rx_bcast + raw->rx_mcast + raw->rx_ucast +
|
||||||
stats->tx_packets = raw->tx_bcast + raw->tx_mcast + raw->tx_ucast;
|
raw->rx_pause;
|
||||||
|
stats->tx_packets = raw->tx_bcast + raw->tx_mcast + raw->tx_ucast +
|
||||||
|
raw->tx_pause;
|
||||||
|
|
||||||
/* HW counters are counting bytes + FCS which is not acceptable
|
/* HW counters are counting bytes + FCS which is not acceptable
|
||||||
* for rtnl_link_stats64 interface
|
* for rtnl_link_stats64 interface
|
||||||
@ -708,6 +712,9 @@ void ksz_r_mib_stats64(struct ksz_device *dev, int port)
|
|||||||
stats->multicast = raw->rx_mcast;
|
stats->multicast = raw->rx_mcast;
|
||||||
stats->collisions = raw->tx_total_col;
|
stats->collisions = raw->tx_total_col;
|
||||||
|
|
||||||
|
pstats->tx_pause_frames = raw->tx_pause;
|
||||||
|
pstats->rx_pause_frames = raw->rx_pause;
|
||||||
|
|
||||||
spin_unlock(&mib->stats64_lock);
|
spin_unlock(&mib->stats64_lock);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -724,6 +731,19 @@ static void ksz_get_stats64(struct dsa_switch *ds, int port,
|
|||||||
spin_unlock(&mib->stats64_lock);
|
spin_unlock(&mib->stats64_lock);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void ksz_get_pause_stats(struct dsa_switch *ds, int port,
|
||||||
|
struct ethtool_pause_stats *pause_stats)
|
||||||
|
{
|
||||||
|
struct ksz_device *dev = ds->priv;
|
||||||
|
struct ksz_port_mib *mib;
|
||||||
|
|
||||||
|
mib = &dev->ports[port].mib;
|
||||||
|
|
||||||
|
spin_lock(&mib->stats64_lock);
|
||||||
|
memcpy(pause_stats, &mib->pause_stats, sizeof(*pause_stats));
|
||||||
|
spin_unlock(&mib->stats64_lock);
|
||||||
|
}
|
||||||
|
|
||||||
static void ksz_get_strings(struct dsa_switch *ds, int port,
|
static void ksz_get_strings(struct dsa_switch *ds, int port,
|
||||||
u32 stringset, uint8_t *buf)
|
u32 stringset, uint8_t *buf)
|
||||||
{
|
{
|
||||||
@ -1336,6 +1356,7 @@ static const struct dsa_switch_ops ksz_switch_ops = {
|
|||||||
.port_mirror_add = ksz_port_mirror_add,
|
.port_mirror_add = ksz_port_mirror_add,
|
||||||
.port_mirror_del = ksz_port_mirror_del,
|
.port_mirror_del = ksz_port_mirror_del,
|
||||||
.get_stats64 = ksz_get_stats64,
|
.get_stats64 = ksz_get_stats64,
|
||||||
|
.get_pause_stats = ksz_get_pause_stats,
|
||||||
.port_change_mtu = ksz_change_mtu,
|
.port_change_mtu = ksz_change_mtu,
|
||||||
.port_max_mtu = ksz_max_mtu,
|
.port_max_mtu = ksz_max_mtu,
|
||||||
};
|
};
|
||||||
|
@ -25,6 +25,7 @@ struct ksz_port_mib {
|
|||||||
u8 cnt_ptr;
|
u8 cnt_ptr;
|
||||||
u64 *counters;
|
u64 *counters;
|
||||||
struct rtnl_link_stats64 stats64;
|
struct rtnl_link_stats64 stats64;
|
||||||
|
struct ethtool_pause_stats pause_stats;
|
||||||
struct spinlock stats64_lock;
|
struct spinlock stats64_lock;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -231,6 +231,7 @@ struct ar9331_sw_port {
|
|||||||
int idx;
|
int idx;
|
||||||
struct delayed_work mib_read;
|
struct delayed_work mib_read;
|
||||||
struct rtnl_link_stats64 stats;
|
struct rtnl_link_stats64 stats;
|
||||||
|
struct ethtool_pause_stats pause_stats;
|
||||||
struct spinlock stats_lock;
|
struct spinlock stats_lock;
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -604,6 +605,7 @@ static void ar9331_sw_phylink_mac_link_up(struct dsa_switch *ds, int port,
|
|||||||
static void ar9331_read_stats(struct ar9331_sw_port *port)
|
static void ar9331_read_stats(struct ar9331_sw_port *port)
|
||||||
{
|
{
|
||||||
struct ar9331_sw_priv *priv = ar9331_sw_port_to_priv(port);
|
struct ar9331_sw_priv *priv = ar9331_sw_port_to_priv(port);
|
||||||
|
struct ethtool_pause_stats *pstats = &port->pause_stats;
|
||||||
struct rtnl_link_stats64 *stats = &port->stats;
|
struct rtnl_link_stats64 *stats = &port->stats;
|
||||||
struct ar9331_sw_stats_raw raw;
|
struct ar9331_sw_stats_raw raw;
|
||||||
int ret;
|
int ret;
|
||||||
@ -644,6 +646,9 @@ static void ar9331_read_stats(struct ar9331_sw_port *port)
|
|||||||
stats->multicast += raw.rxmulti;
|
stats->multicast += raw.rxmulti;
|
||||||
stats->collisions += raw.txcollision;
|
stats->collisions += raw.txcollision;
|
||||||
|
|
||||||
|
pstats->tx_pause_frames += raw.txpause;
|
||||||
|
pstats->rx_pause_frames += raw.rxpause;
|
||||||
|
|
||||||
spin_unlock(&port->stats_lock);
|
spin_unlock(&port->stats_lock);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -668,6 +673,17 @@ static void ar9331_get_stats64(struct dsa_switch *ds, int port,
|
|||||||
spin_unlock(&p->stats_lock);
|
spin_unlock(&p->stats_lock);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void ar9331_get_pause_stats(struct dsa_switch *ds, int port,
|
||||||
|
struct ethtool_pause_stats *pause_stats)
|
||||||
|
{
|
||||||
|
struct ar9331_sw_priv *priv = (struct ar9331_sw_priv *)ds->priv;
|
||||||
|
struct ar9331_sw_port *p = &priv->port[port];
|
||||||
|
|
||||||
|
spin_lock(&p->stats_lock);
|
||||||
|
memcpy(pause_stats, &p->pause_stats, sizeof(*pause_stats));
|
||||||
|
spin_unlock(&p->stats_lock);
|
||||||
|
}
|
||||||
|
|
||||||
static const struct dsa_switch_ops ar9331_sw_ops = {
|
static const struct dsa_switch_ops ar9331_sw_ops = {
|
||||||
.get_tag_protocol = ar9331_sw_get_tag_protocol,
|
.get_tag_protocol = ar9331_sw_get_tag_protocol,
|
||||||
.setup = ar9331_sw_setup,
|
.setup = ar9331_sw_setup,
|
||||||
@ -677,6 +693,7 @@ static const struct dsa_switch_ops ar9331_sw_ops = {
|
|||||||
.phylink_mac_link_down = ar9331_sw_phylink_mac_link_down,
|
.phylink_mac_link_down = ar9331_sw_phylink_mac_link_down,
|
||||||
.phylink_mac_link_up = ar9331_sw_phylink_mac_link_up,
|
.phylink_mac_link_up = ar9331_sw_phylink_mac_link_up,
|
||||||
.get_stats64 = ar9331_get_stats64,
|
.get_stats64 = ar9331_get_stats64,
|
||||||
|
.get_pause_stats = ar9331_get_pause_stats,
|
||||||
};
|
};
|
||||||
|
|
||||||
static irqreturn_t ar9331_sw_irq(int irq, void *data)
|
static irqreturn_t ar9331_sw_irq(int irq, void *data)
|
||||||
|
@ -895,6 +895,8 @@ struct dsa_switch_ops {
|
|||||||
const struct ethtool_rmon_hist_range **ranges);
|
const struct ethtool_rmon_hist_range **ranges);
|
||||||
void (*get_stats64)(struct dsa_switch *ds, int port,
|
void (*get_stats64)(struct dsa_switch *ds, int port,
|
||||||
struct rtnl_link_stats64 *s);
|
struct rtnl_link_stats64 *s);
|
||||||
|
void (*get_pause_stats)(struct dsa_switch *ds, int port,
|
||||||
|
struct ethtool_pause_stats *pause_stats);
|
||||||
void (*self_test)(struct dsa_switch *ds, int port,
|
void (*self_test)(struct dsa_switch *ds, int port,
|
||||||
struct ethtool_test *etest, u64 *data);
|
struct ethtool_test *etest, u64 *data);
|
||||||
|
|
||||||
|
@ -1109,6 +1109,16 @@ static int dsa_slave_set_link_ksettings(struct net_device *dev,
|
|||||||
return phylink_ethtool_ksettings_set(dp->pl, cmd);
|
return phylink_ethtool_ksettings_set(dp->pl, cmd);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void dsa_slave_get_pause_stats(struct net_device *dev,
|
||||||
|
struct ethtool_pause_stats *pause_stats)
|
||||||
|
{
|
||||||
|
struct dsa_port *dp = dsa_slave_to_port(dev);
|
||||||
|
struct dsa_switch *ds = dp->ds;
|
||||||
|
|
||||||
|
if (ds->ops->get_pause_stats)
|
||||||
|
ds->ops->get_pause_stats(ds, dp->index, pause_stats);
|
||||||
|
}
|
||||||
|
|
||||||
static void dsa_slave_get_pauseparam(struct net_device *dev,
|
static void dsa_slave_get_pauseparam(struct net_device *dev,
|
||||||
struct ethtool_pauseparam *pause)
|
struct ethtool_pauseparam *pause)
|
||||||
{
|
{
|
||||||
@ -2100,6 +2110,7 @@ static const struct ethtool_ops dsa_slave_ethtool_ops = {
|
|||||||
.get_eee = dsa_slave_get_eee,
|
.get_eee = dsa_slave_get_eee,
|
||||||
.get_link_ksettings = dsa_slave_get_link_ksettings,
|
.get_link_ksettings = dsa_slave_get_link_ksettings,
|
||||||
.set_link_ksettings = dsa_slave_set_link_ksettings,
|
.set_link_ksettings = dsa_slave_set_link_ksettings,
|
||||||
|
.get_pause_stats = dsa_slave_get_pause_stats,
|
||||||
.get_pauseparam = dsa_slave_get_pauseparam,
|
.get_pauseparam = dsa_slave_get_pauseparam,
|
||||||
.set_pauseparam = dsa_slave_set_pauseparam,
|
.set_pauseparam = dsa_slave_set_pauseparam,
|
||||||
.get_rxnfc = dsa_slave_get_rxnfc,
|
.get_rxnfc = dsa_slave_get_rxnfc,
|
||||||
|
Loading…
Reference in New Issue
Block a user