mlxsw: spectrum: Remove support for bridge bypass FDB add/del
The FDB add/del are now done through the notification chain. The FDBs are synced with the bridge and there is no need for extra dumping. Signed-off-by: Arkadi Sharshevsky <arkadis@mellanox.com> Reviewed-by: Ido Schimmel <idosch@mellanox.com> Signed-off-by: Jiri Pirko <jiri@mellanox.com> Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
parent
af06137892
commit
be7432b952
@ -1756,9 +1756,6 @@ static const struct net_device_ops mlxsw_sp_port_netdev_ops = {
|
|||||||
.ndo_get_offload_stats = mlxsw_sp_port_get_offload_stats,
|
.ndo_get_offload_stats = mlxsw_sp_port_get_offload_stats,
|
||||||
.ndo_vlan_rx_add_vid = mlxsw_sp_port_add_vid,
|
.ndo_vlan_rx_add_vid = mlxsw_sp_port_add_vid,
|
||||||
.ndo_vlan_rx_kill_vid = mlxsw_sp_port_kill_vid,
|
.ndo_vlan_rx_kill_vid = mlxsw_sp_port_kill_vid,
|
||||||
.ndo_fdb_add = switchdev_port_fdb_add,
|
|
||||||
.ndo_fdb_del = switchdev_port_fdb_del,
|
|
||||||
.ndo_fdb_dump = switchdev_port_fdb_dump,
|
|
||||||
.ndo_get_phys_port_name = mlxsw_sp_port_get_phys_port_name,
|
.ndo_get_phys_port_name = mlxsw_sp_port_get_phys_port_name,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -1140,47 +1140,6 @@ static int mlxsw_sp_port_fdb_uc_lag_op(struct mlxsw_sp *mlxsw_sp, u16 lag_id,
|
|||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int
|
|
||||||
mlxsw_sp_port_fdb_static_add(struct mlxsw_sp_port *mlxsw_sp_port,
|
|
||||||
const struct switchdev_obj_port_fdb *fdb,
|
|
||||||
struct switchdev_trans *trans)
|
|
||||||
{
|
|
||||||
struct mlxsw_sp *mlxsw_sp = mlxsw_sp_port->mlxsw_sp;
|
|
||||||
struct net_device *orig_dev = fdb->obj.orig_dev;
|
|
||||||
struct mlxsw_sp_port_vlan *mlxsw_sp_port_vlan;
|
|
||||||
struct mlxsw_sp_bridge_device *bridge_device;
|
|
||||||
struct mlxsw_sp_bridge_port *bridge_port;
|
|
||||||
u16 fid_index, vid;
|
|
||||||
|
|
||||||
if (switchdev_trans_ph_prepare(trans))
|
|
||||||
return 0;
|
|
||||||
|
|
||||||
bridge_port = mlxsw_sp_bridge_port_find(mlxsw_sp->bridge, orig_dev);
|
|
||||||
if (WARN_ON(!bridge_port))
|
|
||||||
return -EINVAL;
|
|
||||||
|
|
||||||
bridge_device = bridge_port->bridge_device;
|
|
||||||
mlxsw_sp_port_vlan = mlxsw_sp_port_vlan_find_by_bridge(mlxsw_sp_port,
|
|
||||||
bridge_device,
|
|
||||||
fdb->vid);
|
|
||||||
if (!mlxsw_sp_port_vlan)
|
|
||||||
return 0;
|
|
||||||
|
|
||||||
fid_index = mlxsw_sp_fid_index(mlxsw_sp_port_vlan->fid);
|
|
||||||
vid = mlxsw_sp_port_vlan->vid;
|
|
||||||
|
|
||||||
if (!mlxsw_sp_port->lagged)
|
|
||||||
return mlxsw_sp_port_fdb_uc_op(mlxsw_sp,
|
|
||||||
mlxsw_sp_port->local_port,
|
|
||||||
fdb->addr, fid_index, true,
|
|
||||||
false);
|
|
||||||
else
|
|
||||||
return mlxsw_sp_port_fdb_uc_lag_op(mlxsw_sp,
|
|
||||||
mlxsw_sp_port->lag_id,
|
|
||||||
fdb->addr, fid_index, vid,
|
|
||||||
true, false);
|
|
||||||
}
|
|
||||||
|
|
||||||
static int
|
static int
|
||||||
mlxsw_sp_port_fdb_set(struct mlxsw_sp_port *mlxsw_sp_port,
|
mlxsw_sp_port_fdb_set(struct mlxsw_sp_port *mlxsw_sp_port,
|
||||||
struct switchdev_notifier_fdb_info *fdb_info, bool adding)
|
struct switchdev_notifier_fdb_info *fdb_info, bool adding)
|
||||||
@ -1385,11 +1344,6 @@ static int mlxsw_sp_port_obj_add(struct net_device *dev,
|
|||||||
SWITCHDEV_OBJ_PORT_VLAN(obj),
|
SWITCHDEV_OBJ_PORT_VLAN(obj),
|
||||||
trans);
|
trans);
|
||||||
break;
|
break;
|
||||||
case SWITCHDEV_OBJ_ID_PORT_FDB:
|
|
||||||
err = mlxsw_sp_port_fdb_static_add(mlxsw_sp_port,
|
|
||||||
SWITCHDEV_OBJ_PORT_FDB(obj),
|
|
||||||
trans);
|
|
||||||
break;
|
|
||||||
case SWITCHDEV_OBJ_ID_PORT_MDB:
|
case SWITCHDEV_OBJ_ID_PORT_MDB:
|
||||||
err = mlxsw_sp_port_mdb_add(mlxsw_sp_port,
|
err = mlxsw_sp_port_mdb_add(mlxsw_sp_port,
|
||||||
SWITCHDEV_OBJ_PORT_MDB(obj),
|
SWITCHDEV_OBJ_PORT_MDB(obj),
|
||||||
@ -1441,43 +1395,6 @@ static int mlxsw_sp_port_vlans_del(struct mlxsw_sp_port *mlxsw_sp_port,
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int
|
|
||||||
mlxsw_sp_port_fdb_static_del(struct mlxsw_sp_port *mlxsw_sp_port,
|
|
||||||
const struct switchdev_obj_port_fdb *fdb)
|
|
||||||
{
|
|
||||||
struct mlxsw_sp *mlxsw_sp = mlxsw_sp_port->mlxsw_sp;
|
|
||||||
struct net_device *orig_dev = fdb->obj.orig_dev;
|
|
||||||
struct mlxsw_sp_port_vlan *mlxsw_sp_port_vlan;
|
|
||||||
struct mlxsw_sp_bridge_device *bridge_device;
|
|
||||||
struct mlxsw_sp_bridge_port *bridge_port;
|
|
||||||
u16 fid_index, vid;
|
|
||||||
|
|
||||||
bridge_port = mlxsw_sp_bridge_port_find(mlxsw_sp->bridge, orig_dev);
|
|
||||||
if (WARN_ON(!bridge_port))
|
|
||||||
return -EINVAL;
|
|
||||||
|
|
||||||
bridge_device = bridge_port->bridge_device;
|
|
||||||
mlxsw_sp_port_vlan = mlxsw_sp_port_vlan_find_by_bridge(mlxsw_sp_port,
|
|
||||||
bridge_device,
|
|
||||||
fdb->vid);
|
|
||||||
if (!mlxsw_sp_port_vlan)
|
|
||||||
return 0;
|
|
||||||
|
|
||||||
fid_index = mlxsw_sp_fid_index(mlxsw_sp_port_vlan->fid);
|
|
||||||
vid = mlxsw_sp_port_vlan->vid;
|
|
||||||
|
|
||||||
if (!mlxsw_sp_port->lagged)
|
|
||||||
return mlxsw_sp_port_fdb_uc_op(mlxsw_sp,
|
|
||||||
mlxsw_sp_port->local_port,
|
|
||||||
fdb->addr, fid_index, false,
|
|
||||||
false);
|
|
||||||
else
|
|
||||||
return mlxsw_sp_port_fdb_uc_lag_op(mlxsw_sp,
|
|
||||||
mlxsw_sp_port->lag_id,
|
|
||||||
fdb->addr, fid_index, vid,
|
|
||||||
false, false);
|
|
||||||
}
|
|
||||||
|
|
||||||
static int mlxsw_sp_port_mdb_del(struct mlxsw_sp_port *mlxsw_sp_port,
|
static int mlxsw_sp_port_mdb_del(struct mlxsw_sp_port *mlxsw_sp_port,
|
||||||
const struct switchdev_obj_port_mdb *mdb)
|
const struct switchdev_obj_port_mdb *mdb)
|
||||||
{
|
{
|
||||||
@ -1537,10 +1454,6 @@ static int mlxsw_sp_port_obj_del(struct net_device *dev,
|
|||||||
err = mlxsw_sp_port_vlans_del(mlxsw_sp_port,
|
err = mlxsw_sp_port_vlans_del(mlxsw_sp_port,
|
||||||
SWITCHDEV_OBJ_PORT_VLAN(obj));
|
SWITCHDEV_OBJ_PORT_VLAN(obj));
|
||||||
break;
|
break;
|
||||||
case SWITCHDEV_OBJ_ID_PORT_FDB:
|
|
||||||
err = mlxsw_sp_port_fdb_static_del(mlxsw_sp_port,
|
|
||||||
SWITCHDEV_OBJ_PORT_FDB(obj));
|
|
||||||
break;
|
|
||||||
case SWITCHDEV_OBJ_ID_PORT_MDB:
|
case SWITCHDEV_OBJ_ID_PORT_MDB:
|
||||||
err = mlxsw_sp_port_mdb_del(mlxsw_sp_port,
|
err = mlxsw_sp_port_mdb_del(mlxsw_sp_port,
|
||||||
SWITCHDEV_OBJ_PORT_MDB(obj));
|
SWITCHDEV_OBJ_PORT_MDB(obj));
|
||||||
@ -1570,124 +1483,11 @@ static struct mlxsw_sp_port *mlxsw_sp_lag_rep_port(struct mlxsw_sp *mlxsw_sp,
|
|||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int mlxsw_sp_port_fdb_dump(struct mlxsw_sp_port *mlxsw_sp_port,
|
|
||||||
struct switchdev_obj_port_fdb *fdb,
|
|
||||||
switchdev_obj_dump_cb_t *cb)
|
|
||||||
{
|
|
||||||
struct mlxsw_sp *mlxsw_sp = mlxsw_sp_port->mlxsw_sp;
|
|
||||||
struct net_device *orig_dev = fdb->obj.orig_dev;
|
|
||||||
struct mlxsw_sp_bridge_port *bridge_port;
|
|
||||||
u16 lag_id, fid_index;
|
|
||||||
char mac[ETH_ALEN];
|
|
||||||
int stored_err = 0;
|
|
||||||
char *sfd_pl;
|
|
||||||
u8 num_rec;
|
|
||||||
int err;
|
|
||||||
|
|
||||||
bridge_port = mlxsw_sp_bridge_port_find(mlxsw_sp->bridge, orig_dev);
|
|
||||||
if (!bridge_port)
|
|
||||||
return 0;
|
|
||||||
|
|
||||||
sfd_pl = kmalloc(MLXSW_REG_SFD_LEN, GFP_KERNEL);
|
|
||||||
if (!sfd_pl)
|
|
||||||
return -ENOMEM;
|
|
||||||
|
|
||||||
mlxsw_reg_sfd_pack(sfd_pl, MLXSW_REG_SFD_OP_QUERY_DUMP, 0);
|
|
||||||
do {
|
|
||||||
struct mlxsw_sp_port *tmp;
|
|
||||||
u8 local_port;
|
|
||||||
int i;
|
|
||||||
|
|
||||||
mlxsw_reg_sfd_num_rec_set(sfd_pl, MLXSW_REG_SFD_REC_MAX_COUNT);
|
|
||||||
err = mlxsw_reg_query(mlxsw_sp->core, MLXSW_REG(sfd), sfd_pl);
|
|
||||||
if (err)
|
|
||||||
goto out;
|
|
||||||
|
|
||||||
num_rec = mlxsw_reg_sfd_num_rec_get(sfd_pl);
|
|
||||||
|
|
||||||
/* Even in case of error, we have to run the dump to the end
|
|
||||||
* so the session in firmware is finished.
|
|
||||||
*/
|
|
||||||
if (stored_err)
|
|
||||||
continue;
|
|
||||||
|
|
||||||
for (i = 0; i < num_rec; i++) {
|
|
||||||
switch (mlxsw_reg_sfd_rec_type_get(sfd_pl, i)) {
|
|
||||||
case MLXSW_REG_SFD_REC_TYPE_UNICAST:
|
|
||||||
mlxsw_reg_sfd_uc_unpack(sfd_pl, i, mac,
|
|
||||||
&fid_index,
|
|
||||||
&local_port);
|
|
||||||
if (bridge_port->lagged)
|
|
||||||
continue;
|
|
||||||
if (bridge_port->system_port != local_port)
|
|
||||||
continue;
|
|
||||||
if (bridge_port->bridge_device->vlan_enabled)
|
|
||||||
fdb->vid = fid_index;
|
|
||||||
else
|
|
||||||
fdb->vid = 0;
|
|
||||||
ether_addr_copy(fdb->addr, mac);
|
|
||||||
fdb->ndm_state = NUD_REACHABLE;
|
|
||||||
err = cb(&fdb->obj);
|
|
||||||
if (err)
|
|
||||||
stored_err = err;
|
|
||||||
break;
|
|
||||||
case MLXSW_REG_SFD_REC_TYPE_UNICAST_LAG:
|
|
||||||
mlxsw_reg_sfd_uc_lag_unpack(sfd_pl, i,
|
|
||||||
mac, &fid_index,
|
|
||||||
&lag_id);
|
|
||||||
if (!bridge_port->lagged)
|
|
||||||
continue;
|
|
||||||
if (bridge_port->lag_id != lag_id)
|
|
||||||
continue;
|
|
||||||
tmp = mlxsw_sp_lag_rep_port(mlxsw_sp, lag_id);
|
|
||||||
if (tmp->local_port !=
|
|
||||||
mlxsw_sp_port->local_port)
|
|
||||||
continue;
|
|
||||||
if (bridge_port->bridge_device->vlan_enabled)
|
|
||||||
fdb->vid = fid_index;
|
|
||||||
else
|
|
||||||
fdb->vid = 0;
|
|
||||||
ether_addr_copy(fdb->addr, mac);
|
|
||||||
fdb->ndm_state = NUD_REACHABLE;
|
|
||||||
err = cb(&fdb->obj);
|
|
||||||
if (err)
|
|
||||||
stored_err = err;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} while (num_rec == MLXSW_REG_SFD_REC_MAX_COUNT);
|
|
||||||
|
|
||||||
out:
|
|
||||||
kfree(sfd_pl);
|
|
||||||
return stored_err ? stored_err : err;
|
|
||||||
}
|
|
||||||
|
|
||||||
static int mlxsw_sp_port_obj_dump(struct net_device *dev,
|
|
||||||
struct switchdev_obj *obj,
|
|
||||||
switchdev_obj_dump_cb_t *cb)
|
|
||||||
{
|
|
||||||
struct mlxsw_sp_port *mlxsw_sp_port = netdev_priv(dev);
|
|
||||||
int err = 0;
|
|
||||||
|
|
||||||
switch (obj->id) {
|
|
||||||
case SWITCHDEV_OBJ_ID_PORT_FDB:
|
|
||||||
err = mlxsw_sp_port_fdb_dump(mlxsw_sp_port,
|
|
||||||
SWITCHDEV_OBJ_PORT_FDB(obj), cb);
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
err = -EOPNOTSUPP;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
return err;
|
|
||||||
}
|
|
||||||
|
|
||||||
static const struct switchdev_ops mlxsw_sp_port_switchdev_ops = {
|
static const struct switchdev_ops mlxsw_sp_port_switchdev_ops = {
|
||||||
.switchdev_port_attr_get = mlxsw_sp_port_attr_get,
|
.switchdev_port_attr_get = mlxsw_sp_port_attr_get,
|
||||||
.switchdev_port_attr_set = mlxsw_sp_port_attr_set,
|
.switchdev_port_attr_set = mlxsw_sp_port_attr_set,
|
||||||
.switchdev_port_obj_add = mlxsw_sp_port_obj_add,
|
.switchdev_port_obj_add = mlxsw_sp_port_obj_add,
|
||||||
.switchdev_port_obj_del = mlxsw_sp_port_obj_del,
|
.switchdev_port_obj_del = mlxsw_sp_port_obj_del,
|
||||||
.switchdev_port_obj_dump = mlxsw_sp_port_obj_dump,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
static int
|
static int
|
||||||
|
Loading…
Reference in New Issue
Block a user