mirror of
https://mirrors.bfsu.edu.cn/git/linux.git
synced 2024-12-21 01:54:51 +08:00
net: mvpp2: Add support for more ethtool counters
Besides the MIB counters, some other useful counters can be exposed to the user. This commit adds support for : - Per-port counters, that indicate FIFO drops and classifier drops, - Per-rxq counters, - Per-txq counters Signed-off-by: Maxime Chevallier <maxime.chevallier@bootlin.com> Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
parent
f9fa96b980
commit
9bea6897cc
@ -329,8 +329,26 @@
|
||||
#define MVPP22_BM_ADDR_HIGH_VIRT_RLS_MASK 0xff00
|
||||
#define MVPP22_BM_ADDR_HIGH_VIRT_RLS_SHIFT 8
|
||||
|
||||
/* Packet Processor per-port counters */
|
||||
#define MVPP2_OVERRUN_ETH_DROP 0x7000
|
||||
#define MVPP2_CLS_ETH_DROP 0x7020
|
||||
|
||||
/* Hit counters registers */
|
||||
#define MVPP2_CTRS_IDX 0x7040
|
||||
#define MVPP22_CTRS_TX_CTR(port, txq) ((txq) | ((port) << 3) | BIT(7))
|
||||
#define MVPP2_TX_DESC_ENQ_CTR 0x7100
|
||||
#define MVPP2_TX_DESC_ENQ_TO_DDR_CTR 0x7104
|
||||
#define MVPP2_TX_BUFF_ENQ_TO_DDR_CTR 0x7108
|
||||
#define MVPP2_TX_DESC_ENQ_HW_FWD_CTR 0x710c
|
||||
#define MVPP2_RX_DESC_ENQ_CTR 0x7120
|
||||
#define MVPP2_TX_PKTS_DEQ_CTR 0x7130
|
||||
#define MVPP2_TX_PKTS_FULL_QUEUE_DROP_CTR 0x7200
|
||||
#define MVPP2_TX_PKTS_EARLY_DROP_CTR 0x7204
|
||||
#define MVPP2_TX_PKTS_BM_DROP_CTR 0x7208
|
||||
#define MVPP2_TX_PKTS_BM_MC_DROP_CTR 0x720c
|
||||
#define MVPP2_RX_PKTS_FULL_QUEUE_DROP_CTR 0x7220
|
||||
#define MVPP2_RX_PKTS_EARLY_DROP_CTR 0x7224
|
||||
#define MVPP2_RX_PKTS_BM_DROP_CTR 0x7228
|
||||
#define MVPP2_CLS_DEC_TBL_HIT_CTR 0x7700
|
||||
#define MVPP2_CLS_FLOW_TBL_HIT_CTR 0x7704
|
||||
|
||||
|
@ -1258,6 +1258,17 @@ static u64 mvpp2_read_count(struct mvpp2_port *port,
|
||||
return val;
|
||||
}
|
||||
|
||||
/* Some counters are accessed indirectly by first writing an index to
|
||||
* MVPP2_CTRS_IDX. The index can represent various resources depending on the
|
||||
* register we access, it can be a hit counter for some classification tables,
|
||||
* a counter specific to a rxq, a txq or a buffer pool.
|
||||
*/
|
||||
static u32 mvpp2_read_index(struct mvpp2 *priv, u32 index, u32 reg)
|
||||
{
|
||||
mvpp2_write(priv, MVPP2_CTRS_IDX, index);
|
||||
return mvpp2_read(priv, reg);
|
||||
}
|
||||
|
||||
/* Due to the fact that software statistics and hardware statistics are, by
|
||||
* design, incremented at different moments in the chain of packet processing,
|
||||
* it is very likely that incoming packets could have been dropped after being
|
||||
@ -1297,17 +1308,103 @@ static const struct mvpp2_ethtool_counter mvpp2_ethtool_mib_regs[] = {
|
||||
{ MVPP2_MIB_LATE_COLLISION, "late_collision" },
|
||||
};
|
||||
|
||||
static const struct mvpp2_ethtool_counter mvpp2_ethtool_port_regs[] = {
|
||||
{ MVPP2_OVERRUN_ETH_DROP, "rx_fifo_or_parser_overrun_drops" },
|
||||
{ MVPP2_CLS_ETH_DROP, "rx_classifier_drops" },
|
||||
};
|
||||
|
||||
static const struct mvpp2_ethtool_counter mvpp2_ethtool_txq_regs[] = {
|
||||
{ MVPP2_TX_DESC_ENQ_CTR, "txq_%d_desc_enqueue" },
|
||||
{ MVPP2_TX_DESC_ENQ_TO_DDR_CTR, "txq_%d_desc_enqueue_to_ddr" },
|
||||
{ MVPP2_TX_BUFF_ENQ_TO_DDR_CTR, "txq_%d_buff_euqueue_to_ddr" },
|
||||
{ MVPP2_TX_DESC_ENQ_HW_FWD_CTR, "txq_%d_desc_hardware_forwarded" },
|
||||
{ MVPP2_TX_PKTS_DEQ_CTR, "txq_%d_packets_dequeued" },
|
||||
{ MVPP2_TX_PKTS_FULL_QUEUE_DROP_CTR, "txq_%d_queue_full_drops" },
|
||||
{ MVPP2_TX_PKTS_EARLY_DROP_CTR, "txq_%d_packets_early_drops" },
|
||||
{ MVPP2_TX_PKTS_BM_DROP_CTR, "txq_%d_packets_bm_drops" },
|
||||
{ MVPP2_TX_PKTS_BM_MC_DROP_CTR, "txq_%d_packets_rep_bm_drops" },
|
||||
};
|
||||
|
||||
static const struct mvpp2_ethtool_counter mvpp2_ethtool_rxq_regs[] = {
|
||||
{ MVPP2_RX_DESC_ENQ_CTR, "rxq_%d_desc_enqueue" },
|
||||
{ MVPP2_RX_PKTS_FULL_QUEUE_DROP_CTR, "rxq_%d_queue_full_drops" },
|
||||
{ MVPP2_RX_PKTS_EARLY_DROP_CTR, "rxq_%d_packets_early_drops" },
|
||||
{ MVPP2_RX_PKTS_BM_DROP_CTR, "rxq_%d_packets_bm_drops" },
|
||||
};
|
||||
|
||||
#define MVPP2_N_ETHTOOL_STATS(ntxqs, nrxqs) (ARRAY_SIZE(mvpp2_ethtool_mib_regs) + \
|
||||
ARRAY_SIZE(mvpp2_ethtool_port_regs) + \
|
||||
(ARRAY_SIZE(mvpp2_ethtool_txq_regs) * (ntxqs)) + \
|
||||
(ARRAY_SIZE(mvpp2_ethtool_rxq_regs) * (nrxqs)))
|
||||
|
||||
static void mvpp2_ethtool_get_strings(struct net_device *netdev, u32 sset,
|
||||
u8 *data)
|
||||
{
|
||||
if (sset == ETH_SS_STATS) {
|
||||
int i;
|
||||
struct mvpp2_port *port = netdev_priv(netdev);
|
||||
int i, q;
|
||||
|
||||
for (i = 0; i < ARRAY_SIZE(mvpp2_ethtool_mib_regs); i++)
|
||||
strscpy(data + i * ETH_GSTRING_LEN,
|
||||
mvpp2_ethtool_mib_regs[i].string,
|
||||
ETH_GSTRING_LEN);
|
||||
if (sset != ETH_SS_STATS)
|
||||
return;
|
||||
|
||||
for (i = 0; i < ARRAY_SIZE(mvpp2_ethtool_mib_regs); i++) {
|
||||
strscpy(data, mvpp2_ethtool_mib_regs[i].string,
|
||||
ETH_GSTRING_LEN);
|
||||
data += ETH_GSTRING_LEN;
|
||||
}
|
||||
|
||||
for (i = 0; i < ARRAY_SIZE(mvpp2_ethtool_port_regs); i++) {
|
||||
strscpy(data, mvpp2_ethtool_port_regs[i].string,
|
||||
ETH_GSTRING_LEN);
|
||||
data += ETH_GSTRING_LEN;
|
||||
}
|
||||
|
||||
for (q = 0; q < port->ntxqs; q++) {
|
||||
for (i = 0; i < ARRAY_SIZE(mvpp2_ethtool_txq_regs); i++) {
|
||||
snprintf(data, ETH_GSTRING_LEN,
|
||||
mvpp2_ethtool_txq_regs[i].string, q);
|
||||
data += ETH_GSTRING_LEN;
|
||||
}
|
||||
}
|
||||
|
||||
for (q = 0; q < port->nrxqs; q++) {
|
||||
for (i = 0; i < ARRAY_SIZE(mvpp2_ethtool_rxq_regs); i++) {
|
||||
snprintf(data, ETH_GSTRING_LEN,
|
||||
mvpp2_ethtool_rxq_regs[i].string,
|
||||
q);
|
||||
data += ETH_GSTRING_LEN;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
static void mvpp2_read_stats(struct mvpp2_port *port)
|
||||
{
|
||||
u64 *pstats;
|
||||
int i, q;
|
||||
|
||||
pstats = port->ethtool_stats;
|
||||
|
||||
for (i = 0; i < ARRAY_SIZE(mvpp2_ethtool_mib_regs); i++)
|
||||
*pstats++ += mvpp2_read_count(port, &mvpp2_ethtool_mib_regs[i]);
|
||||
|
||||
for (i = 0; i < ARRAY_SIZE(mvpp2_ethtool_port_regs); i++)
|
||||
*pstats++ += mvpp2_read(port->priv,
|
||||
mvpp2_ethtool_port_regs[i].offset +
|
||||
4 * port->id);
|
||||
|
||||
for (q = 0; q < port->ntxqs; q++)
|
||||
for (i = 0; i < ARRAY_SIZE(mvpp2_ethtool_txq_regs); i++)
|
||||
*pstats++ += mvpp2_read_index(port->priv,
|
||||
MVPP22_CTRS_TX_CTR(port->id, i),
|
||||
mvpp2_ethtool_txq_regs[i].offset);
|
||||
|
||||
/* Rxqs are numbered from 0 from the user standpoint, but not from the
|
||||
* driver's. We need to add the port->first_rxq offset.
|
||||
*/
|
||||
for (q = 0; q < port->nrxqs; q++)
|
||||
for (i = 0; i < ARRAY_SIZE(mvpp2_ethtool_rxq_regs); i++)
|
||||
*pstats++ += mvpp2_read_index(port->priv,
|
||||
port->first_rxq + i,
|
||||
mvpp2_ethtool_rxq_regs[i].offset);
|
||||
}
|
||||
|
||||
static void mvpp2_gather_hw_statistics(struct work_struct *work)
|
||||
@ -1315,14 +1412,10 @@ static void mvpp2_gather_hw_statistics(struct work_struct *work)
|
||||
struct delayed_work *del_work = to_delayed_work(work);
|
||||
struct mvpp2_port *port = container_of(del_work, struct mvpp2_port,
|
||||
stats_work);
|
||||
u64 *pstats;
|
||||
int i;
|
||||
|
||||
mutex_lock(&port->gather_stats_lock);
|
||||
|
||||
pstats = port->ethtool_stats;
|
||||
for (i = 0; i < ARRAY_SIZE(mvpp2_ethtool_mib_regs); i++)
|
||||
*pstats++ += mvpp2_read_count(port, &mvpp2_ethtool_mib_regs[i]);
|
||||
mvpp2_read_stats(port);
|
||||
|
||||
/* No need to read again the counters right after this function if it
|
||||
* was called asynchronously by the user (ie. use of ethtool).
|
||||
@ -1346,14 +1439,16 @@ static void mvpp2_ethtool_get_stats(struct net_device *dev,
|
||||
|
||||
mutex_lock(&port->gather_stats_lock);
|
||||
memcpy(data, port->ethtool_stats,
|
||||
sizeof(u64) * ARRAY_SIZE(mvpp2_ethtool_mib_regs));
|
||||
sizeof(u64) * MVPP2_N_ETHTOOL_STATS(port->ntxqs, port->nrxqs));
|
||||
mutex_unlock(&port->gather_stats_lock);
|
||||
}
|
||||
|
||||
static int mvpp2_ethtool_get_sset_count(struct net_device *dev, int sset)
|
||||
{
|
||||
struct mvpp2_port *port = netdev_priv(dev);
|
||||
|
||||
if (sset == ETH_SS_STATS)
|
||||
return ARRAY_SIZE(mvpp2_ethtool_mib_regs);
|
||||
return MVPP2_N_ETHTOOL_STATS(port->ntxqs, port->nrxqs);
|
||||
|
||||
return -EOPNOTSUPP;
|
||||
}
|
||||
@ -4261,7 +4356,7 @@ static int mvpp2_port_init(struct mvpp2_port *port)
|
||||
struct mvpp2 *priv = port->priv;
|
||||
struct mvpp2_txq_pcpu *txq_pcpu;
|
||||
unsigned int thread;
|
||||
int queue, err, i;
|
||||
int queue, err;
|
||||
|
||||
/* Checks for hardware constraints */
|
||||
if (port->first_rxq + port->nrxqs >
|
||||
@ -4368,9 +4463,10 @@ static int mvpp2_port_init(struct mvpp2_port *port)
|
||||
if (err)
|
||||
goto err_free_percpu;
|
||||
|
||||
/* Read the GOP statistics to reset the hardware counters */
|
||||
for (i = 0; i < ARRAY_SIZE(mvpp2_ethtool_mib_regs); i++)
|
||||
mvpp2_read_count(port, &mvpp2_ethtool_mib_regs[i]);
|
||||
/* Clear all port stats */
|
||||
mvpp2_read_stats(port);
|
||||
memset(port->ethtool_stats, 0,
|
||||
MVPP2_N_ETHTOOL_STATS(port->ntxqs, port->nrxqs) * sizeof(u64));
|
||||
|
||||
return 0;
|
||||
|
||||
@ -5053,7 +5149,7 @@ static int mvpp2_port_probe(struct platform_device *pdev,
|
||||
}
|
||||
|
||||
port->ethtool_stats = devm_kcalloc(&pdev->dev,
|
||||
ARRAY_SIZE(mvpp2_ethtool_mib_regs),
|
||||
MVPP2_N_ETHTOOL_STATS(ntxqs, nrxqs),
|
||||
sizeof(u64), GFP_KERNEL);
|
||||
if (!port->ethtool_stats) {
|
||||
err = -ENOMEM;
|
||||
|
Loading…
Reference in New Issue
Block a user