net: dsa: rzn1-a5psw: add FDB support
This commits add forwarding database support to the driver. It implements fdb_add(), fdb_del() and fdb_dump(). Signed-off-by: Clément Léger <clement.leger@bootlin.com> Reviewed-by: Vladimir Oltean <olteanv@gmail.com> Reviewed-by: Florian Fainelli <f.fainelli@gmail.com> Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
Родитель
c7243fd4a6
Коммит
5edf246c68
|
@ -375,6 +375,171 @@ static void a5psw_port_fast_age(struct dsa_switch *ds, int port)
|
|||
a5psw_port_fdb_flush(a5psw, port);
|
||||
}
|
||||
|
||||
static int a5psw_lk_execute_lookup(struct a5psw *a5psw, union lk_data *lk_data,
|
||||
u16 *entry)
|
||||
{
|
||||
u32 ctrl;
|
||||
int ret;
|
||||
|
||||
a5psw_reg_writel(a5psw, A5PSW_LK_DATA_LO, lk_data->lo);
|
||||
a5psw_reg_writel(a5psw, A5PSW_LK_DATA_HI, lk_data->hi);
|
||||
|
||||
ctrl = A5PSW_LK_ADDR_CTRL_LOOKUP;
|
||||
ret = a5psw_lk_execute_ctrl(a5psw, &ctrl);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
*entry = ctrl & A5PSW_LK_ADDR_CTRL_ADDRESS;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int a5psw_port_fdb_add(struct dsa_switch *ds, int port,
|
||||
const unsigned char *addr, u16 vid,
|
||||
struct dsa_db db)
|
||||
{
|
||||
struct a5psw *a5psw = ds->priv;
|
||||
union lk_data lk_data = {0};
|
||||
bool inc_learncount = false;
|
||||
int ret = 0;
|
||||
u16 entry;
|
||||
u32 reg;
|
||||
|
||||
ether_addr_copy(lk_data.entry.mac, addr);
|
||||
lk_data.entry.port_mask = BIT(port);
|
||||
|
||||
mutex_lock(&a5psw->lk_lock);
|
||||
|
||||
/* Set the value to be written in the lookup table */
|
||||
ret = a5psw_lk_execute_lookup(a5psw, &lk_data, &entry);
|
||||
if (ret)
|
||||
goto lk_unlock;
|
||||
|
||||
lk_data.hi = a5psw_reg_readl(a5psw, A5PSW_LK_DATA_HI);
|
||||
if (!lk_data.entry.valid) {
|
||||
inc_learncount = true;
|
||||
/* port_mask set to 0x1f when entry is not valid, clear it */
|
||||
lk_data.entry.port_mask = 0;
|
||||
lk_data.entry.prio = 0;
|
||||
}
|
||||
|
||||
lk_data.entry.port_mask |= BIT(port);
|
||||
lk_data.entry.is_static = 1;
|
||||
lk_data.entry.valid = 1;
|
||||
|
||||
a5psw_reg_writel(a5psw, A5PSW_LK_DATA_HI, lk_data.hi);
|
||||
|
||||
reg = A5PSW_LK_ADDR_CTRL_WRITE | entry;
|
||||
ret = a5psw_lk_execute_ctrl(a5psw, ®);
|
||||
if (ret)
|
||||
goto lk_unlock;
|
||||
|
||||
if (inc_learncount) {
|
||||
reg = A5PSW_LK_LEARNCOUNT_MODE_INC;
|
||||
a5psw_reg_writel(a5psw, A5PSW_LK_LEARNCOUNT, reg);
|
||||
}
|
||||
|
||||
lk_unlock:
|
||||
mutex_unlock(&a5psw->lk_lock);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int a5psw_port_fdb_del(struct dsa_switch *ds, int port,
|
||||
const unsigned char *addr, u16 vid,
|
||||
struct dsa_db db)
|
||||
{
|
||||
struct a5psw *a5psw = ds->priv;
|
||||
union lk_data lk_data = {0};
|
||||
bool clear = false;
|
||||
u16 entry;
|
||||
u32 reg;
|
||||
int ret;
|
||||
|
||||
ether_addr_copy(lk_data.entry.mac, addr);
|
||||
|
||||
mutex_lock(&a5psw->lk_lock);
|
||||
|
||||
ret = a5psw_lk_execute_lookup(a5psw, &lk_data, &entry);
|
||||
if (ret)
|
||||
goto lk_unlock;
|
||||
|
||||
lk_data.hi = a5psw_reg_readl(a5psw, A5PSW_LK_DATA_HI);
|
||||
|
||||
/* Our hardware does not associate any VID to the FDB entries so this
|
||||
* means that if two entries were added for the same mac but for
|
||||
* different VID, then, on the deletion of the first one, we would also
|
||||
* delete the second one. Since there is unfortunately nothing we can do
|
||||
* about that, do not return an error...
|
||||
*/
|
||||
if (!lk_data.entry.valid)
|
||||
goto lk_unlock;
|
||||
|
||||
lk_data.entry.port_mask &= ~BIT(port);
|
||||
/* If there is no more port in the mask, clear the entry */
|
||||
if (lk_data.entry.port_mask == 0)
|
||||
clear = true;
|
||||
|
||||
a5psw_reg_writel(a5psw, A5PSW_LK_DATA_HI, lk_data.hi);
|
||||
|
||||
reg = entry;
|
||||
if (clear)
|
||||
reg |= A5PSW_LK_ADDR_CTRL_CLEAR;
|
||||
else
|
||||
reg |= A5PSW_LK_ADDR_CTRL_WRITE;
|
||||
|
||||
ret = a5psw_lk_execute_ctrl(a5psw, ®);
|
||||
if (ret)
|
||||
goto lk_unlock;
|
||||
|
||||
/* Decrement LEARNCOUNT */
|
||||
if (clear) {
|
||||
reg = A5PSW_LK_LEARNCOUNT_MODE_DEC;
|
||||
a5psw_reg_writel(a5psw, A5PSW_LK_LEARNCOUNT, reg);
|
||||
}
|
||||
|
||||
lk_unlock:
|
||||
mutex_unlock(&a5psw->lk_lock);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int a5psw_port_fdb_dump(struct dsa_switch *ds, int port,
|
||||
dsa_fdb_dump_cb_t *cb, void *data)
|
||||
{
|
||||
struct a5psw *a5psw = ds->priv;
|
||||
union lk_data lk_data;
|
||||
int i = 0, ret = 0;
|
||||
u32 reg;
|
||||
|
||||
mutex_lock(&a5psw->lk_lock);
|
||||
|
||||
for (i = 0; i < A5PSW_TABLE_ENTRIES; i++) {
|
||||
reg = A5PSW_LK_ADDR_CTRL_READ | A5PSW_LK_ADDR_CTRL_WAIT | i;
|
||||
|
||||
ret = a5psw_lk_execute_ctrl(a5psw, ®);
|
||||
if (ret)
|
||||
goto out_unlock;
|
||||
|
||||
lk_data.hi = a5psw_reg_readl(a5psw, A5PSW_LK_DATA_HI);
|
||||
/* If entry is not valid or does not contain the port, skip */
|
||||
if (!lk_data.entry.valid ||
|
||||
!(lk_data.entry.port_mask & BIT(port)))
|
||||
continue;
|
||||
|
||||
lk_data.lo = a5psw_reg_readl(a5psw, A5PSW_LK_DATA_LO);
|
||||
|
||||
ret = cb(lk_data.entry.mac, 0, lk_data.entry.is_static, data);
|
||||
if (ret)
|
||||
goto out_unlock;
|
||||
}
|
||||
|
||||
out_unlock:
|
||||
mutex_unlock(&a5psw->lk_lock);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static u64 a5psw_read_stat(struct a5psw *a5psw, u32 offset, int port)
|
||||
{
|
||||
u32 reg_lo, reg_hi;
|
||||
|
@ -591,6 +756,9 @@ static const struct dsa_switch_ops a5psw_switch_ops = {
|
|||
.port_bridge_leave = a5psw_port_bridge_leave,
|
||||
.port_stp_state_set = a5psw_port_stp_state_set,
|
||||
.port_fast_age = a5psw_port_fast_age,
|
||||
.port_fdb_add = a5psw_port_fdb_add,
|
||||
.port_fdb_del = a5psw_port_fdb_del,
|
||||
.port_fdb_dump = a5psw_port_fdb_dump,
|
||||
};
|
||||
|
||||
static int a5psw_mdio_wait_busy(struct a5psw *a5psw)
|
||||
|
|
|
@ -211,6 +211,23 @@
|
|||
#define A5PSW_CTRL_TIMEOUT 1000
|
||||
#define A5PSW_TABLE_ENTRIES 8192
|
||||
|
||||
struct fdb_entry {
|
||||
u8 mac[ETH_ALEN];
|
||||
u16 valid:1;
|
||||
u16 is_static:1;
|
||||
u16 prio:3;
|
||||
u16 port_mask:5;
|
||||
u16 reserved:6;
|
||||
} __packed;
|
||||
|
||||
union lk_data {
|
||||
struct {
|
||||
u32 lo;
|
||||
u32 hi;
|
||||
};
|
||||
struct fdb_entry entry;
|
||||
};
|
||||
|
||||
/**
|
||||
* struct a5psw - switch struct
|
||||
* @base: Base address of the switch
|
||||
|
|
Загрузка…
Ссылка в новой задаче