net: mvpp2: debugfs: add classifier hit counters
authorMaxime Chevallier <maxime.chevallier@bootlin.com>
Sat, 14 Jul 2018 11:29:28 +0000 (13:29 +0200)
committerDavid S. Miller <davem@davemloft.net>
Mon, 16 Jul 2018 07:10:01 +0000 (00:10 -0700)
The classification operations that are used for RSS make use of several
lookup tables. Having hit counters for these tables is really helpful
to determine what flows were matched by ingress traffic, and see the
path of packets among all the classifier tables.

This commit adds hit counters for the 3 tables used at the moment :

 - The decoding table (also called lookup_id table), that links flows
   identified by the Header Parser to the flow table.

   There's one entry per flow, located at :
   .../mvpp2/<controller>/flows/XX/dec_hits

   Note that there are 21 flows in the decoding table, whereas there are
   52 flows in the Header Parser. That's because there are several kind
   of traffic that will match a given flow. Reading the hit counter from
   one sub-flow will clear all hit counter that have the same flow_id.

   This also applies to the flow_hits.

 - The flow table, that contains all the different lookups to be
   performed by the classifier for each packet of a given flow. The match
   is done on the first entry of the flow sequence.

 - The C2 engine entries, that are used to assign the default rx queue,
   and enable or disable RSS for a given port.

   There's one entry per flow, located at:
   .../mvpp2/<controller>/flows/XX/flow_hits

   There is one C2 entry per port, so the c2 hit counter is located at :
   .../mvpp2/<controller>/ethX/c2_hits

All hit counter values are 16-bits clear-on-read values.

Signed-off-by: Maxime Chevallier <maxime.chevallier@bootlin.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
drivers/net/ethernet/marvell/mvpp2/mvpp2.h
drivers/net/ethernet/marvell/mvpp2/mvpp2_cls.c
drivers/net/ethernet/marvell/mvpp2/mvpp2_cls.h
drivers/net/ethernet/marvell/mvpp2/mvpp2_debugfs.c

index af4968d7c007e8cdf295180dd89d9d154e042c91..67b9e81b7c0246435c26680e06939ef2e061bfd7 100644 (file)
 #define MVPP22_CLS_C2_TCAM_DATA3               0x1b1c
 #define MVPP22_CLS_C2_TCAM_DATA4               0x1b20
 #define     MVPP22_CLS_C2_PORT_ID(port)                ((port) << 8)
+#define MVPP22_CLS_C2_HIT_CTR                  0x1b50
 #define MVPP22_CLS_C2_ACT                      0x1b60
 #define     MVPP22_CLS_C2_ACT_RSS_EN(act)      (((act) & 0x3) << 19)
 #define     MVPP22_CLS_C2_ACT_FWD(act)         (((act) & 0x7) << 13)
 #define     MVPP22_BM_ADDR_HIGH_VIRT_RLS_MASK  0xff00
 #define     MVPP22_BM_ADDR_HIGH_VIRT_RLS_SHIFT 8
 
+/* Hit counters registers */
+#define MVPP2_CTRS_IDX                         0x7040
+#define MVPP2_CLS_DEC_TBL_HIT_CTR              0x7700
+#define MVPP2_CLS_FLOW_TBL_HIT_CTR             0x7704
+
 /* TX Scheduler registers */
 #define MVPP2_TXP_SCHED_PORT_INDEX_REG         0x8000
 #define MVPP2_TXP_SCHED_Q_CMD_REG              0x8004
index c5012fa390c8a0cdacdff58d80fd37a0ace0d581..efdb7a65683576a84806639630fce4d0928defcd 100644 (file)
@@ -322,6 +322,13 @@ static struct mvpp2_cls_flow cls_flows[MVPP2_N_FLOWS] = {
                       0, 0),
 };
 
+u32 mvpp2_cls_flow_hits(struct mvpp2 *priv, int index)
+{
+       mvpp2_write(priv, MVPP2_CTRS_IDX, index);
+
+       return mvpp2_read(priv, MVPP2_CLS_FLOW_TBL_HIT_CTR);
+}
+
 void mvpp2_cls_flow_read(struct mvpp2 *priv, int index,
                         struct mvpp2_cls_flow_entry *fe)
 {
@@ -342,6 +349,13 @@ static void mvpp2_cls_flow_write(struct mvpp2 *priv,
        mvpp2_write(priv, MVPP2_CLS_FLOW_TBL2_REG,  fe->data[2]);
 }
 
+u32 mvpp2_cls_lookup_hits(struct mvpp2 *priv, int index)
+{
+       mvpp2_write(priv, MVPP2_CTRS_IDX, index);
+
+       return mvpp2_read(priv, MVPP2_CLS_DEC_TBL_HIT_CTR);
+}
+
 void mvpp2_cls_lookup_read(struct mvpp2 *priv, int lkpid, int way,
                           struct mvpp2_cls_lookup_entry *le)
 {
@@ -859,6 +873,13 @@ void mvpp2_cls_port_config(struct mvpp2_port *port)
        mvpp2_port_c2_cls_init(port);
 }
 
+u32 mvpp2_cls_c2_hit_count(struct mvpp2 *priv, int c2_index)
+{
+       mvpp2_write(priv, MVPP22_CLS_C2_TCAM_IDX, c2_index);
+
+       return mvpp2_read(priv, MVPP22_CLS_C2_HIT_CTR);
+}
+
 static void mvpp2_rss_port_c2_enable(struct mvpp2_port *port)
 {
        struct mvpp2_cls_c2_entry c2;
index 13eae2d1b4bfe8c0a20f05c16a15ad2df3c0853c..089f05f298917ae394b9245cdb255259d269b3c7 100644 (file)
@@ -215,12 +215,18 @@ u16 mvpp2_flow_get_hek_fields(struct mvpp2_cls_flow_entry *fe);
 
 struct mvpp2_cls_flow *mvpp2_cls_flow_get(int flow);
 
+u32 mvpp2_cls_flow_hits(struct mvpp2 *priv, int index);
+
 void mvpp2_cls_flow_read(struct mvpp2 *priv, int index,
                         struct mvpp2_cls_flow_entry *fe);
 
+u32 mvpp2_cls_lookup_hits(struct mvpp2 *priv, int index);
+
 void mvpp2_cls_lookup_read(struct mvpp2 *priv, int lkpid, int way,
                           struct mvpp2_cls_lookup_entry *le);
 
+u32 mvpp2_cls_c2_hit_count(struct mvpp2 *priv, int c2_index);
+
 void mvpp2_cls_c2_read(struct mvpp2 *priv, int index,
                       struct mvpp2_cls_c2_entry *c2);
 
index fc46ec81249b66ccb4a63b1ab31ae194f0f7c7c7..02dfef13cccdd4f6c340f26e8a003edbaf3c0e71 100644 (file)
@@ -28,6 +28,33 @@ struct mvpp2_dbgfs_port_flow_entry {
        struct mvpp2_dbgfs_flow_entry *dbg_fe;
 };
 
+static int mvpp2_dbgfs_flow_flt_hits_show(struct seq_file *s, void *unused)
+{
+       struct mvpp2_dbgfs_flow_entry *entry = s->private;
+       int id = MVPP2_FLOW_C2_ENTRY(entry->flow);
+
+       u32 hits = mvpp2_cls_flow_hits(entry->priv, id);
+
+       seq_printf(s, "%u\n", hits);
+
+       return 0;
+}
+
+DEFINE_SHOW_ATTRIBUTE(mvpp2_dbgfs_flow_flt_hits);
+
+static int mvpp2_dbgfs_flow_dec_hits_show(struct seq_file *s, void *unused)
+{
+       struct mvpp2_dbgfs_flow_entry *entry = s->private;
+
+       u32 hits = mvpp2_cls_lookup_hits(entry->priv, entry->flow);
+
+       seq_printf(s, "%u\n", hits);
+
+       return 0;
+}
+
+DEFINE_SHOW_ATTRIBUTE(mvpp2_dbgfs_flow_dec_hits);
+
 static int mvpp2_dbgfs_flow_type_show(struct seq_file *s, void *unused)
 {
        struct mvpp2_dbgfs_flow_entry *entry = s->private;
@@ -174,6 +201,21 @@ static int mvpp2_dbgfs_port_flow_engine_show(struct seq_file *s, void *unused)
 
 DEFINE_SHOW_ATTRIBUTE(mvpp2_dbgfs_port_flow_engine);
 
+static int mvpp2_dbgfs_flow_c2_hits_show(struct seq_file *s, void *unused)
+{
+       struct mvpp2_port *port = s->private;
+       u32 hits;
+
+       hits = mvpp2_cls_c2_hit_count(port->priv,
+                                     MVPP22_CLS_C2_RSS_ENTRY(port->id));
+
+       seq_printf(s, "%u\n", hits);
+
+       return 0;
+}
+
+DEFINE_SHOW_ATTRIBUTE(mvpp2_dbgfs_flow_c2_hits);
+
 static int mvpp2_dbgfs_flow_c2_rxq_show(struct seq_file *s, void *unused)
 {
        struct mvpp2_port *port = s->private;
@@ -484,6 +526,12 @@ static int mvpp2_dbgfs_flow_entry_init(struct dentry *parent,
        entry->flow = flow;
        entry->priv = priv;
 
+       debugfs_create_file("flow_hits", 0444, flow_entry_dir, entry,
+                           &mvpp2_dbgfs_flow_flt_hits_fops);
+
+       debugfs_create_file("dec_hits", 0444, flow_entry_dir, entry,
+                           &mvpp2_dbgfs_flow_dec_hits_fops);
+
        debugfs_create_file("type", 0444, flow_entry_dir, entry,
                            &mvpp2_dbgfs_flow_type_fops);
 
@@ -600,6 +648,9 @@ static int mvpp2_dbgfs_port_init(struct dentry *parent,
        debugfs_create_file("vid_filter", 0444, port_dir, port,
                            &mvpp2_dbgfs_port_vid_fops);
 
+       debugfs_create_file("c2_hits", 0444, port_dir, port,
+                           &mvpp2_dbgfs_flow_c2_hits_fops);
+
        debugfs_create_file("default_rxq", 0444, port_dir, port,
                            &mvpp2_dbgfs_flow_c2_rxq_fops);