|
@@ -2093,9 +2093,9 @@ static int _mv88e6xxx_atu_load(struct mv88e6xxx_chip *chip,
|
|
return _mv88e6xxx_atu_cmd(chip, entry->fid, GLOBAL_ATU_OP_LOAD_DB);
|
|
return _mv88e6xxx_atu_cmd(chip, entry->fid, GLOBAL_ATU_OP_LOAD_DB);
|
|
}
|
|
}
|
|
|
|
|
|
-static int _mv88e6xxx_port_fdb_load(struct mv88e6xxx_chip *chip, int port,
|
|
|
|
- const unsigned char *addr, u16 vid,
|
|
|
|
- u8 state)
|
|
|
|
|
|
+static int mv88e6xxx_port_db_load_purge(struct mv88e6xxx_chip *chip, int port,
|
|
|
|
+ const unsigned char *addr, u16 vid,
|
|
|
|
+ u8 state)
|
|
{
|
|
{
|
|
struct mv88e6xxx_atu_entry entry = { 0 };
|
|
struct mv88e6xxx_atu_entry entry = { 0 };
|
|
struct mv88e6xxx_vtu_stu_entry vlan;
|
|
struct mv88e6xxx_vtu_stu_entry vlan;
|
|
@@ -2134,15 +2134,12 @@ static void mv88e6xxx_port_fdb_add(struct dsa_switch *ds, int port,
|
|
const struct switchdev_obj_port_fdb *fdb,
|
|
const struct switchdev_obj_port_fdb *fdb,
|
|
struct switchdev_trans *trans)
|
|
struct switchdev_trans *trans)
|
|
{
|
|
{
|
|
- int state = is_multicast_ether_addr(fdb->addr) ?
|
|
|
|
- GLOBAL_ATU_DATA_STATE_MC_STATIC :
|
|
|
|
- GLOBAL_ATU_DATA_STATE_UC_STATIC;
|
|
|
|
struct mv88e6xxx_chip *chip = ds_to_priv(ds);
|
|
struct mv88e6xxx_chip *chip = ds_to_priv(ds);
|
|
|
|
|
|
mutex_lock(&chip->reg_lock);
|
|
mutex_lock(&chip->reg_lock);
|
|
- if (_mv88e6xxx_port_fdb_load(chip, port, fdb->addr, fdb->vid, state))
|
|
|
|
- netdev_err(ds->ports[port].netdev,
|
|
|
|
- "failed to load MAC address\n");
|
|
|
|
|
|
+ if (mv88e6xxx_port_db_load_purge(chip, port, fdb->addr, fdb->vid,
|
|
|
|
+ GLOBAL_ATU_DATA_STATE_UC_STATIC))
|
|
|
|
+ netdev_err(ds->ports[port].netdev, "failed to load unicast MAC address\n");
|
|
mutex_unlock(&chip->reg_lock);
|
|
mutex_unlock(&chip->reg_lock);
|
|
}
|
|
}
|
|
|
|
|
|
@@ -2150,14 +2147,14 @@ static int mv88e6xxx_port_fdb_del(struct dsa_switch *ds, int port,
|
|
const struct switchdev_obj_port_fdb *fdb)
|
|
const struct switchdev_obj_port_fdb *fdb)
|
|
{
|
|
{
|
|
struct mv88e6xxx_chip *chip = ds_to_priv(ds);
|
|
struct mv88e6xxx_chip *chip = ds_to_priv(ds);
|
|
- int ret;
|
|
|
|
|
|
+ int err;
|
|
|
|
|
|
mutex_lock(&chip->reg_lock);
|
|
mutex_lock(&chip->reg_lock);
|
|
- ret = _mv88e6xxx_port_fdb_load(chip, port, fdb->addr, fdb->vid,
|
|
|
|
- GLOBAL_ATU_DATA_STATE_UNUSED);
|
|
|
|
|
|
+ err = mv88e6xxx_port_db_load_purge(chip, port, fdb->addr, fdb->vid,
|
|
|
|
+ GLOBAL_ATU_DATA_STATE_UNUSED);
|
|
mutex_unlock(&chip->reg_lock);
|
|
mutex_unlock(&chip->reg_lock);
|
|
|
|
|
|
- return ret;
|
|
|
|
|
|
+ return err;
|
|
}
|
|
}
|
|
|
|
|
|
static int _mv88e6xxx_atu_getnext(struct mv88e6xxx_chip *chip, u16 fid,
|
|
static int _mv88e6xxx_atu_getnext(struct mv88e6xxx_chip *chip, u16 fid,
|
|
@@ -2205,10 +2202,10 @@ static int _mv88e6xxx_atu_getnext(struct mv88e6xxx_chip *chip, u16 fid,
|
|
return 0;
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
|
|
-static int _mv88e6xxx_port_fdb_dump_one(struct mv88e6xxx_chip *chip,
|
|
|
|
- u16 fid, u16 vid, int port,
|
|
|
|
- struct switchdev_obj_port_fdb *fdb,
|
|
|
|
- int (*cb)(struct switchdev_obj *obj))
|
|
|
|
|
|
+static int mv88e6xxx_port_db_dump_fid(struct mv88e6xxx_chip *chip,
|
|
|
|
+ u16 fid, u16 vid, int port,
|
|
|
|
+ struct switchdev_obj *obj,
|
|
|
|
+ int (*cb)(struct switchdev_obj *obj))
|
|
{
|
|
{
|
|
struct mv88e6xxx_atu_entry addr = {
|
|
struct mv88e6xxx_atu_entry addr = {
|
|
.mac = { 0xff, 0xff, 0xff, 0xff, 0xff, 0xff },
|
|
.mac = { 0xff, 0xff, 0xff, 0xff, 0xff, 0xff },
|
|
@@ -2222,72 +2219,98 @@ static int _mv88e6xxx_port_fdb_dump_one(struct mv88e6xxx_chip *chip,
|
|
do {
|
|
do {
|
|
err = _mv88e6xxx_atu_getnext(chip, fid, &addr);
|
|
err = _mv88e6xxx_atu_getnext(chip, fid, &addr);
|
|
if (err)
|
|
if (err)
|
|
- break;
|
|
|
|
|
|
+ return err;
|
|
|
|
|
|
if (addr.state == GLOBAL_ATU_DATA_STATE_UNUSED)
|
|
if (addr.state == GLOBAL_ATU_DATA_STATE_UNUSED)
|
|
break;
|
|
break;
|
|
|
|
|
|
- if (!addr.trunk && addr.portv_trunkid & BIT(port)) {
|
|
|
|
- bool is_static = addr.state ==
|
|
|
|
- (is_multicast_ether_addr(addr.mac) ?
|
|
|
|
- GLOBAL_ATU_DATA_STATE_MC_STATIC :
|
|
|
|
- GLOBAL_ATU_DATA_STATE_UC_STATIC);
|
|
|
|
|
|
+ if (addr.trunk || (addr.portv_trunkid & BIT(port)) == 0)
|
|
|
|
+ continue;
|
|
|
|
+
|
|
|
|
+ if (obj->id == SWITCHDEV_OBJ_ID_PORT_FDB) {
|
|
|
|
+ struct switchdev_obj_port_fdb *fdb;
|
|
|
|
+
|
|
|
|
+ if (!is_unicast_ether_addr(addr.mac))
|
|
|
|
+ continue;
|
|
|
|
|
|
|
|
+ fdb = SWITCHDEV_OBJ_PORT_FDB(obj);
|
|
fdb->vid = vid;
|
|
fdb->vid = vid;
|
|
ether_addr_copy(fdb->addr, addr.mac);
|
|
ether_addr_copy(fdb->addr, addr.mac);
|
|
- fdb->ndm_state = is_static ? NUD_NOARP : NUD_REACHABLE;
|
|
|
|
|
|
+ if (addr.state == GLOBAL_ATU_DATA_STATE_UC_STATIC)
|
|
|
|
+ fdb->ndm_state = NUD_NOARP;
|
|
|
|
+ else
|
|
|
|
+ fdb->ndm_state = NUD_REACHABLE;
|
|
|
|
+ } else if (obj->id == SWITCHDEV_OBJ_ID_PORT_MDB) {
|
|
|
|
+ struct switchdev_obj_port_mdb *mdb;
|
|
|
|
|
|
- err = cb(&fdb->obj);
|
|
|
|
- if (err)
|
|
|
|
- break;
|
|
|
|
|
|
+ if (!is_multicast_ether_addr(addr.mac))
|
|
|
|
+ continue;
|
|
|
|
+
|
|
|
|
+ mdb = SWITCHDEV_OBJ_PORT_MDB(obj);
|
|
|
|
+ mdb->vid = vid;
|
|
|
|
+ ether_addr_copy(mdb->addr, addr.mac);
|
|
|
|
+ } else {
|
|
|
|
+ return -EOPNOTSUPP;
|
|
}
|
|
}
|
|
|
|
+
|
|
|
|
+ err = cb(obj);
|
|
|
|
+ if (err)
|
|
|
|
+ return err;
|
|
} while (!is_broadcast_ether_addr(addr.mac));
|
|
} while (!is_broadcast_ether_addr(addr.mac));
|
|
|
|
|
|
return err;
|
|
return err;
|
|
}
|
|
}
|
|
|
|
|
|
-static int mv88e6xxx_port_fdb_dump(struct dsa_switch *ds, int port,
|
|
|
|
- struct switchdev_obj_port_fdb *fdb,
|
|
|
|
- int (*cb)(struct switchdev_obj *obj))
|
|
|
|
|
|
+static int mv88e6xxx_port_db_dump(struct mv88e6xxx_chip *chip, int port,
|
|
|
|
+ struct switchdev_obj *obj,
|
|
|
|
+ int (*cb)(struct switchdev_obj *obj))
|
|
{
|
|
{
|
|
- struct mv88e6xxx_chip *chip = ds_to_priv(ds);
|
|
|
|
struct mv88e6xxx_vtu_stu_entry vlan = {
|
|
struct mv88e6xxx_vtu_stu_entry vlan = {
|
|
.vid = GLOBAL_VTU_VID_MASK, /* all ones */
|
|
.vid = GLOBAL_VTU_VID_MASK, /* all ones */
|
|
};
|
|
};
|
|
u16 fid;
|
|
u16 fid;
|
|
int err;
|
|
int err;
|
|
|
|
|
|
- mutex_lock(&chip->reg_lock);
|
|
|
|
-
|
|
|
|
/* Dump port's default Filtering Information Database (VLAN ID 0) */
|
|
/* Dump port's default Filtering Information Database (VLAN ID 0) */
|
|
err = _mv88e6xxx_port_fid_get(chip, port, &fid);
|
|
err = _mv88e6xxx_port_fid_get(chip, port, &fid);
|
|
if (err)
|
|
if (err)
|
|
- goto unlock;
|
|
|
|
|
|
+ return err;
|
|
|
|
|
|
- err = _mv88e6xxx_port_fdb_dump_one(chip, fid, 0, port, fdb, cb);
|
|
|
|
|
|
+ err = mv88e6xxx_port_db_dump_fid(chip, fid, 0, port, obj, cb);
|
|
if (err)
|
|
if (err)
|
|
- goto unlock;
|
|
|
|
|
|
+ return err;
|
|
|
|
|
|
/* Dump VLANs' Filtering Information Databases */
|
|
/* Dump VLANs' Filtering Information Databases */
|
|
err = _mv88e6xxx_vtu_vid_write(chip, vlan.vid);
|
|
err = _mv88e6xxx_vtu_vid_write(chip, vlan.vid);
|
|
if (err)
|
|
if (err)
|
|
- goto unlock;
|
|
|
|
|
|
+ return err;
|
|
|
|
|
|
do {
|
|
do {
|
|
err = _mv88e6xxx_vtu_getnext(chip, &vlan);
|
|
err = _mv88e6xxx_vtu_getnext(chip, &vlan);
|
|
if (err)
|
|
if (err)
|
|
- break;
|
|
|
|
|
|
+ return err;
|
|
|
|
|
|
if (!vlan.valid)
|
|
if (!vlan.valid)
|
|
break;
|
|
break;
|
|
|
|
|
|
- err = _mv88e6xxx_port_fdb_dump_one(chip, vlan.fid, vlan.vid,
|
|
|
|
- port, fdb, cb);
|
|
|
|
|
|
+ err = mv88e6xxx_port_db_dump_fid(chip, vlan.fid, vlan.vid, port,
|
|
|
|
+ obj, cb);
|
|
if (err)
|
|
if (err)
|
|
- break;
|
|
|
|
|
|
+ return err;
|
|
} while (vlan.vid < GLOBAL_VTU_VID_MASK);
|
|
} while (vlan.vid < GLOBAL_VTU_VID_MASK);
|
|
|
|
|
|
-unlock:
|
|
|
|
|
|
+ return err;
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+static int mv88e6xxx_port_fdb_dump(struct dsa_switch *ds, int port,
|
|
|
|
+ struct switchdev_obj_port_fdb *fdb,
|
|
|
|
+ int (*cb)(struct switchdev_obj *obj))
|
|
|
|
+{
|
|
|
|
+ struct mv88e6xxx_chip *chip = ds_to_priv(ds);
|
|
|
|
+ int err;
|
|
|
|
+
|
|
|
|
+ mutex_lock(&chip->reg_lock);
|
|
|
|
+ err = mv88e6xxx_port_db_dump(chip, port, &fdb->obj, cb);
|
|
mutex_unlock(&chip->reg_lock);
|
|
mutex_unlock(&chip->reg_lock);
|
|
|
|
|
|
return err;
|
|
return err;
|
|
@@ -3980,6 +4003,58 @@ free:
|
|
return NULL;
|
|
return NULL;
|
|
}
|
|
}
|
|
|
|
|
|
|
|
+static int mv88e6xxx_port_mdb_prepare(struct dsa_switch *ds, int port,
|
|
|
|
+ const struct switchdev_obj_port_mdb *mdb,
|
|
|
|
+ struct switchdev_trans *trans)
|
|
|
|
+{
|
|
|
|
+ /* We don't need any dynamic resource from the kernel (yet),
|
|
|
|
+ * so skip the prepare phase.
|
|
|
|
+ */
|
|
|
|
+
|
|
|
|
+ return 0;
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+static void mv88e6xxx_port_mdb_add(struct dsa_switch *ds, int port,
|
|
|
|
+ const struct switchdev_obj_port_mdb *mdb,
|
|
|
|
+ struct switchdev_trans *trans)
|
|
|
|
+{
|
|
|
|
+ struct mv88e6xxx_chip *chip = ds_to_priv(ds);
|
|
|
|
+
|
|
|
|
+ mutex_lock(&chip->reg_lock);
|
|
|
|
+ if (mv88e6xxx_port_db_load_purge(chip, port, mdb->addr, mdb->vid,
|
|
|
|
+ GLOBAL_ATU_DATA_STATE_MC_STATIC))
|
|
|
|
+ netdev_err(ds->ports[port].netdev, "failed to load multicast MAC address\n");
|
|
|
|
+ mutex_unlock(&chip->reg_lock);
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+static int mv88e6xxx_port_mdb_del(struct dsa_switch *ds, int port,
|
|
|
|
+ const struct switchdev_obj_port_mdb *mdb)
|
|
|
|
+{
|
|
|
|
+ struct mv88e6xxx_chip *chip = ds_to_priv(ds);
|
|
|
|
+ int err;
|
|
|
|
+
|
|
|
|
+ mutex_lock(&chip->reg_lock);
|
|
|
|
+ err = mv88e6xxx_port_db_load_purge(chip, port, mdb->addr, mdb->vid,
|
|
|
|
+ GLOBAL_ATU_DATA_STATE_UNUSED);
|
|
|
|
+ mutex_unlock(&chip->reg_lock);
|
|
|
|
+
|
|
|
|
+ return err;
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+static int mv88e6xxx_port_mdb_dump(struct dsa_switch *ds, int port,
|
|
|
|
+ struct switchdev_obj_port_mdb *mdb,
|
|
|
|
+ int (*cb)(struct switchdev_obj *obj))
|
|
|
|
+{
|
|
|
|
+ struct mv88e6xxx_chip *chip = ds_to_priv(ds);
|
|
|
|
+ int err;
|
|
|
|
+
|
|
|
|
+ mutex_lock(&chip->reg_lock);
|
|
|
|
+ err = mv88e6xxx_port_db_dump(chip, port, &mdb->obj, cb);
|
|
|
|
+ mutex_unlock(&chip->reg_lock);
|
|
|
|
+
|
|
|
|
+ return err;
|
|
|
|
+}
|
|
|
|
+
|
|
static struct dsa_switch_ops mv88e6xxx_switch_ops = {
|
|
static struct dsa_switch_ops mv88e6xxx_switch_ops = {
|
|
.probe = mv88e6xxx_drv_probe,
|
|
.probe = mv88e6xxx_drv_probe,
|
|
.get_tag_protocol = mv88e6xxx_get_tag_protocol,
|
|
.get_tag_protocol = mv88e6xxx_get_tag_protocol,
|
|
@@ -4015,6 +4090,10 @@ static struct dsa_switch_ops mv88e6xxx_switch_ops = {
|
|
.port_fdb_add = mv88e6xxx_port_fdb_add,
|
|
.port_fdb_add = mv88e6xxx_port_fdb_add,
|
|
.port_fdb_del = mv88e6xxx_port_fdb_del,
|
|
.port_fdb_del = mv88e6xxx_port_fdb_del,
|
|
.port_fdb_dump = mv88e6xxx_port_fdb_dump,
|
|
.port_fdb_dump = mv88e6xxx_port_fdb_dump,
|
|
|
|
+ .port_mdb_prepare = mv88e6xxx_port_mdb_prepare,
|
|
|
|
+ .port_mdb_add = mv88e6xxx_port_mdb_add,
|
|
|
|
+ .port_mdb_del = mv88e6xxx_port_mdb_del,
|
|
|
|
+ .port_mdb_dump = mv88e6xxx_port_mdb_dump,
|
|
};
|
|
};
|
|
|
|
|
|
static int mv88e6xxx_register_switch(struct mv88e6xxx_chip *chip,
|
|
static int mv88e6xxx_register_switch(struct mv88e6xxx_chip *chip,
|