Commit bbda0f0d authored by David S. Miller's avatar David S. Miller

Merge branch 'dsa-trace-events'

Vladimir Oltean says:

====================
DSA trace events

This series introduces the "dsa" trace event class, with the following
events:

$ trace-cmd list | grep dsa
dsa
dsa:dsa_fdb_add_hw
dsa:dsa_mdb_add_hw
dsa:dsa_fdb_del_hw
dsa:dsa_mdb_del_hw
dsa:dsa_fdb_add_bump
dsa:dsa_mdb_add_bump
dsa:dsa_fdb_del_drop
dsa:dsa_mdb_del_drop
dsa:dsa_fdb_del_not_found
dsa:dsa_mdb_del_not_found
dsa:dsa_lag_fdb_add_hw
dsa:dsa_lag_fdb_add_bump
dsa:dsa_lag_fdb_del_hw
dsa:dsa_lag_fdb_del_drop
dsa:dsa_lag_fdb_del_not_found
dsa:dsa_vlan_add_hw
dsa:dsa_vlan_del_hw
dsa:dsa_vlan_add_bump
dsa:dsa_vlan_del_drop
dsa:dsa_vlan_del_not_found

These are useful to debug refcounting issues on CPU and DSA ports, where
entries may remain lingering, or may be removed too soon, depending on
bugs in higher layers of the network stack.
====================
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parents 18bb56ab 02020bd7
...@@ -16,7 +16,8 @@ dsa_core-y += \ ...@@ -16,7 +16,8 @@ dsa_core-y += \
slave.o \ slave.o \
switch.o \ switch.o \
tag.o \ tag.o \
tag_8021q.o tag_8021q.o \
trace.o
# tagging formats # tagging formats
obj-$(CONFIG_NET_DSA_TAG_AR9331) += tag_ar9331.o obj-$(CONFIG_NET_DSA_TAG_AR9331) += tag_ar9331.o
...@@ -37,3 +38,6 @@ obj-$(CONFIG_NET_DSA_TAG_RZN1_A5PSW) += tag_rzn1_a5psw.o ...@@ -37,3 +38,6 @@ obj-$(CONFIG_NET_DSA_TAG_RZN1_A5PSW) += tag_rzn1_a5psw.o
obj-$(CONFIG_NET_DSA_TAG_SJA1105) += tag_sja1105.o obj-$(CONFIG_NET_DSA_TAG_SJA1105) += tag_sja1105.o
obj-$(CONFIG_NET_DSA_TAG_TRAILER) += tag_trailer.o obj-$(CONFIG_NET_DSA_TAG_TRAILER) += tag_trailer.o
obj-$(CONFIG_NET_DSA_TAG_XRS700X) += tag_xrs700x.o obj-$(CONFIG_NET_DSA_TAG_XRS700X) += tag_xrs700x.o
# for tracing framework to find trace.h
CFLAGS_trace.o := -I$(src)
...@@ -18,6 +18,7 @@ ...@@ -18,6 +18,7 @@
#include "slave.h" #include "slave.h"
#include "switch.h" #include "switch.h"
#include "tag_8021q.h" #include "tag_8021q.h"
#include "trace.h"
static unsigned int dsa_switch_fastest_ageing_time(struct dsa_switch *ds, static unsigned int dsa_switch_fastest_ageing_time(struct dsa_switch *ds,
unsigned int ageing_time) unsigned int ageing_time)
...@@ -164,14 +165,20 @@ static int dsa_port_do_mdb_add(struct dsa_port *dp, ...@@ -164,14 +165,20 @@ static int dsa_port_do_mdb_add(struct dsa_port *dp,
int err = 0; int err = 0;
/* No need to bother with refcounting for user ports */ /* No need to bother with refcounting for user ports */
if (!(dsa_port_is_cpu(dp) || dsa_port_is_dsa(dp))) if (!(dsa_port_is_cpu(dp) || dsa_port_is_dsa(dp))) {
return ds->ops->port_mdb_add(ds, port, mdb, db); err = ds->ops->port_mdb_add(ds, port, mdb, db);
trace_dsa_mdb_add_hw(dp, mdb->addr, mdb->vid, &db, err);
return err;
}
mutex_lock(&dp->addr_lists_lock); mutex_lock(&dp->addr_lists_lock);
a = dsa_mac_addr_find(&dp->mdbs, mdb->addr, mdb->vid, db); a = dsa_mac_addr_find(&dp->mdbs, mdb->addr, mdb->vid, db);
if (a) { if (a) {
refcount_inc(&a->refcount); refcount_inc(&a->refcount);
trace_dsa_mdb_add_bump(dp, mdb->addr, mdb->vid, &db,
&a->refcount);
goto out; goto out;
} }
...@@ -182,6 +189,7 @@ static int dsa_port_do_mdb_add(struct dsa_port *dp, ...@@ -182,6 +189,7 @@ static int dsa_port_do_mdb_add(struct dsa_port *dp,
} }
err = ds->ops->port_mdb_add(ds, port, mdb, db); err = ds->ops->port_mdb_add(ds, port, mdb, db);
trace_dsa_mdb_add_hw(dp, mdb->addr, mdb->vid, &db, err);
if (err) { if (err) {
kfree(a); kfree(a);
goto out; goto out;
...@@ -209,21 +217,30 @@ static int dsa_port_do_mdb_del(struct dsa_port *dp, ...@@ -209,21 +217,30 @@ static int dsa_port_do_mdb_del(struct dsa_port *dp,
int err = 0; int err = 0;
/* No need to bother with refcounting for user ports */ /* No need to bother with refcounting for user ports */
if (!(dsa_port_is_cpu(dp) || dsa_port_is_dsa(dp))) if (!(dsa_port_is_cpu(dp) || dsa_port_is_dsa(dp))) {
return ds->ops->port_mdb_del(ds, port, mdb, db); err = ds->ops->port_mdb_del(ds, port, mdb, db);
trace_dsa_mdb_del_hw(dp, mdb->addr, mdb->vid, &db, err);
return err;
}
mutex_lock(&dp->addr_lists_lock); mutex_lock(&dp->addr_lists_lock);
a = dsa_mac_addr_find(&dp->mdbs, mdb->addr, mdb->vid, db); a = dsa_mac_addr_find(&dp->mdbs, mdb->addr, mdb->vid, db);
if (!a) { if (!a) {
trace_dsa_mdb_del_not_found(dp, mdb->addr, mdb->vid, &db);
err = -ENOENT; err = -ENOENT;
goto out; goto out;
} }
if (!refcount_dec_and_test(&a->refcount)) if (!refcount_dec_and_test(&a->refcount)) {
trace_dsa_mdb_del_drop(dp, mdb->addr, mdb->vid, &db,
&a->refcount);
goto out; goto out;
}
err = ds->ops->port_mdb_del(ds, port, mdb, db); err = ds->ops->port_mdb_del(ds, port, mdb, db);
trace_dsa_mdb_del_hw(dp, mdb->addr, mdb->vid, &db, err);
if (err) { if (err) {
refcount_set(&a->refcount, 1); refcount_set(&a->refcount, 1);
goto out; goto out;
...@@ -247,14 +264,19 @@ static int dsa_port_do_fdb_add(struct dsa_port *dp, const unsigned char *addr, ...@@ -247,14 +264,19 @@ static int dsa_port_do_fdb_add(struct dsa_port *dp, const unsigned char *addr,
int err = 0; int err = 0;
/* No need to bother with refcounting for user ports */ /* No need to bother with refcounting for user ports */
if (!(dsa_port_is_cpu(dp) || dsa_port_is_dsa(dp))) if (!(dsa_port_is_cpu(dp) || dsa_port_is_dsa(dp))) {
return ds->ops->port_fdb_add(ds, port, addr, vid, db); err = ds->ops->port_fdb_add(ds, port, addr, vid, db);
trace_dsa_fdb_add_hw(dp, addr, vid, &db, err);
return err;
}
mutex_lock(&dp->addr_lists_lock); mutex_lock(&dp->addr_lists_lock);
a = dsa_mac_addr_find(&dp->fdbs, addr, vid, db); a = dsa_mac_addr_find(&dp->fdbs, addr, vid, db);
if (a) { if (a) {
refcount_inc(&a->refcount); refcount_inc(&a->refcount);
trace_dsa_fdb_add_bump(dp, addr, vid, &db, &a->refcount);
goto out; goto out;
} }
...@@ -265,6 +287,7 @@ static int dsa_port_do_fdb_add(struct dsa_port *dp, const unsigned char *addr, ...@@ -265,6 +287,7 @@ static int dsa_port_do_fdb_add(struct dsa_port *dp, const unsigned char *addr,
} }
err = ds->ops->port_fdb_add(ds, port, addr, vid, db); err = ds->ops->port_fdb_add(ds, port, addr, vid, db);
trace_dsa_fdb_add_hw(dp, addr, vid, &db, err);
if (err) { if (err) {
kfree(a); kfree(a);
goto out; goto out;
...@@ -291,21 +314,29 @@ static int dsa_port_do_fdb_del(struct dsa_port *dp, const unsigned char *addr, ...@@ -291,21 +314,29 @@ static int dsa_port_do_fdb_del(struct dsa_port *dp, const unsigned char *addr,
int err = 0; int err = 0;
/* No need to bother with refcounting for user ports */ /* No need to bother with refcounting for user ports */
if (!(dsa_port_is_cpu(dp) || dsa_port_is_dsa(dp))) if (!(dsa_port_is_cpu(dp) || dsa_port_is_dsa(dp))) {
return ds->ops->port_fdb_del(ds, port, addr, vid, db); err = ds->ops->port_fdb_del(ds, port, addr, vid, db);
trace_dsa_fdb_del_hw(dp, addr, vid, &db, err);
return err;
}
mutex_lock(&dp->addr_lists_lock); mutex_lock(&dp->addr_lists_lock);
a = dsa_mac_addr_find(&dp->fdbs, addr, vid, db); a = dsa_mac_addr_find(&dp->fdbs, addr, vid, db);
if (!a) { if (!a) {
trace_dsa_fdb_del_not_found(dp, addr, vid, &db);
err = -ENOENT; err = -ENOENT;
goto out; goto out;
} }
if (!refcount_dec_and_test(&a->refcount)) if (!refcount_dec_and_test(&a->refcount)) {
trace_dsa_fdb_del_drop(dp, addr, vid, &db, &a->refcount);
goto out; goto out;
}
err = ds->ops->port_fdb_del(ds, port, addr, vid, db); err = ds->ops->port_fdb_del(ds, port, addr, vid, db);
trace_dsa_fdb_del_hw(dp, addr, vid, &db, err);
if (err) { if (err) {
refcount_set(&a->refcount, 1); refcount_set(&a->refcount, 1);
goto out; goto out;
...@@ -332,6 +363,8 @@ static int dsa_switch_do_lag_fdb_add(struct dsa_switch *ds, struct dsa_lag *lag, ...@@ -332,6 +363,8 @@ static int dsa_switch_do_lag_fdb_add(struct dsa_switch *ds, struct dsa_lag *lag,
a = dsa_mac_addr_find(&lag->fdbs, addr, vid, db); a = dsa_mac_addr_find(&lag->fdbs, addr, vid, db);
if (a) { if (a) {
refcount_inc(&a->refcount); refcount_inc(&a->refcount);
trace_dsa_lag_fdb_add_bump(lag->dev, addr, vid, &db,
&a->refcount);
goto out; goto out;
} }
...@@ -342,6 +375,7 @@ static int dsa_switch_do_lag_fdb_add(struct dsa_switch *ds, struct dsa_lag *lag, ...@@ -342,6 +375,7 @@ static int dsa_switch_do_lag_fdb_add(struct dsa_switch *ds, struct dsa_lag *lag,
} }
err = ds->ops->lag_fdb_add(ds, *lag, addr, vid, db); err = ds->ops->lag_fdb_add(ds, *lag, addr, vid, db);
trace_dsa_lag_fdb_add_hw(lag->dev, addr, vid, &db, err);
if (err) { if (err) {
kfree(a); kfree(a);
goto out; goto out;
...@@ -370,14 +404,19 @@ static int dsa_switch_do_lag_fdb_del(struct dsa_switch *ds, struct dsa_lag *lag, ...@@ -370,14 +404,19 @@ static int dsa_switch_do_lag_fdb_del(struct dsa_switch *ds, struct dsa_lag *lag,
a = dsa_mac_addr_find(&lag->fdbs, addr, vid, db); a = dsa_mac_addr_find(&lag->fdbs, addr, vid, db);
if (!a) { if (!a) {
trace_dsa_lag_fdb_del_not_found(lag->dev, addr, vid, &db);
err = -ENOENT; err = -ENOENT;
goto out; goto out;
} }
if (!refcount_dec_and_test(&a->refcount)) if (!refcount_dec_and_test(&a->refcount)) {
trace_dsa_lag_fdb_del_drop(lag->dev, addr, vid, &db,
&a->refcount);
goto out; goto out;
}
err = ds->ops->lag_fdb_del(ds, *lag, addr, vid, db); err = ds->ops->lag_fdb_del(ds, *lag, addr, vid, db);
trace_dsa_lag_fdb_del_hw(lag->dev, addr, vid, &db, err);
if (err) { if (err) {
refcount_set(&a->refcount, 1); refcount_set(&a->refcount, 1);
goto out; goto out;
...@@ -656,8 +695,12 @@ static int dsa_port_do_vlan_add(struct dsa_port *dp, ...@@ -656,8 +695,12 @@ static int dsa_port_do_vlan_add(struct dsa_port *dp,
int err = 0; int err = 0;
/* No need to bother with refcounting for user ports. */ /* No need to bother with refcounting for user ports. */
if (!(dsa_port_is_cpu(dp) || dsa_port_is_dsa(dp))) if (!(dsa_port_is_cpu(dp) || dsa_port_is_dsa(dp))) {
return ds->ops->port_vlan_add(ds, port, vlan, extack); err = ds->ops->port_vlan_add(ds, port, vlan, extack);
trace_dsa_vlan_add_hw(dp, vlan, err);
return err;
}
/* No need to propagate on shared ports the existing VLANs that were /* No need to propagate on shared ports the existing VLANs that were
* re-notified after just the flags have changed. This would cause a * re-notified after just the flags have changed. This would cause a
...@@ -672,6 +715,7 @@ static int dsa_port_do_vlan_add(struct dsa_port *dp, ...@@ -672,6 +715,7 @@ static int dsa_port_do_vlan_add(struct dsa_port *dp,
v = dsa_vlan_find(&dp->vlans, vlan); v = dsa_vlan_find(&dp->vlans, vlan);
if (v) { if (v) {
refcount_inc(&v->refcount); refcount_inc(&v->refcount);
trace_dsa_vlan_add_bump(dp, vlan, &v->refcount);
goto out; goto out;
} }
...@@ -682,6 +726,7 @@ static int dsa_port_do_vlan_add(struct dsa_port *dp, ...@@ -682,6 +726,7 @@ static int dsa_port_do_vlan_add(struct dsa_port *dp,
} }
err = ds->ops->port_vlan_add(ds, port, vlan, extack); err = ds->ops->port_vlan_add(ds, port, vlan, extack);
trace_dsa_vlan_add_hw(dp, vlan, err);
if (err) { if (err) {
kfree(v); kfree(v);
goto out; goto out;
...@@ -706,21 +751,29 @@ static int dsa_port_do_vlan_del(struct dsa_port *dp, ...@@ -706,21 +751,29 @@ static int dsa_port_do_vlan_del(struct dsa_port *dp,
int err = 0; int err = 0;
/* No need to bother with refcounting for user ports */ /* No need to bother with refcounting for user ports */
if (!(dsa_port_is_cpu(dp) || dsa_port_is_dsa(dp))) if (!(dsa_port_is_cpu(dp) || dsa_port_is_dsa(dp))) {
return ds->ops->port_vlan_del(ds, port, vlan); err = ds->ops->port_vlan_del(ds, port, vlan);
trace_dsa_vlan_del_hw(dp, vlan, err);
return err;
}
mutex_lock(&dp->vlans_lock); mutex_lock(&dp->vlans_lock);
v = dsa_vlan_find(&dp->vlans, vlan); v = dsa_vlan_find(&dp->vlans, vlan);
if (!v) { if (!v) {
trace_dsa_vlan_del_not_found(dp, vlan);
err = -ENOENT; err = -ENOENT;
goto out; goto out;
} }
if (!refcount_dec_and_test(&v->refcount)) if (!refcount_dec_and_test(&v->refcount)) {
trace_dsa_vlan_del_drop(dp, vlan, &v->refcount);
goto out; goto out;
}
err = ds->ops->port_vlan_del(ds, port, vlan); err = ds->ops->port_vlan_del(ds, port, vlan);
trace_dsa_vlan_del_hw(dp, vlan, err);
if (err) { if (err) {
refcount_set(&v->refcount, 1); refcount_set(&v->refcount, 1);
goto out; goto out;
......
// SPDX-License-Identifier: GPL-2.0-or-later
/* Copyright 2022-2023 NXP
*/
#define CREATE_TRACE_POINTS
#include "trace.h"
void dsa_db_print(const struct dsa_db *db, char buf[DSA_DB_BUFSIZ])
{
switch (db->type) {
case DSA_DB_PORT:
sprintf(buf, "port %s", db->dp->name);
break;
case DSA_DB_LAG:
sprintf(buf, "lag %s id %d", db->lag.dev->name, db->lag.id);
break;
case DSA_DB_BRIDGE:
sprintf(buf, "bridge %s num %d", db->bridge.dev->name,
db->bridge.num);
break;
default:
sprintf(buf, "unknown");
break;
}
}
const char *dsa_port_kind(const struct dsa_port *dp)
{
switch (dp->type) {
case DSA_PORT_TYPE_USER:
return "user";
case DSA_PORT_TYPE_CPU:
return "cpu";
case DSA_PORT_TYPE_DSA:
return "dsa";
default:
return "unused";
}
}
This diff is collapsed.
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