Commit c4565538 authored by Florian Fainelli's avatar Florian Fainelli Committed by Jakub Kicinski
Browse files

net: dsa: b53: add support for FDB operations on 5325/5365



BCM5325 and BCM5365 are part of a much older generation of switches which,
due to their limited number of ports and VLAN entries (up to 256) allowed
a single 64-bit register to hold a full ARL entry.
This requires a little bit of massaging when reading, writing and
converting ARL entries in both directions.

Signed-off-by: default avatarFlorian Fainelli <f.fainelli@gmail.com>
Signed-off-by: default avatarÁlvaro Fernández Rojas <noltari@gmail.com>
Link: https://patch.msgid.link/20250614080000.1884236-6-noltari@gmail.com


Signed-off-by: default avatarJakub Kicinski <kuba@kernel.org>
parent 0cbec9ae
Loading
Loading
Loading
Loading
+82 −19
Original line number Diff line number Diff line
@@ -1764,6 +1764,45 @@ static int b53_arl_read(struct b53_device *dev, u64 mac,
	return *idx >= dev->num_arl_bins ? -ENOSPC : -ENOENT;
}

static int b53_arl_read_25(struct b53_device *dev, u64 mac,
			   u16 vid, struct b53_arl_entry *ent, u8 *idx)
{
	DECLARE_BITMAP(free_bins, B53_ARLTBL_MAX_BIN_ENTRIES);
	unsigned int i;
	int ret;

	ret = b53_arl_op_wait(dev);
	if (ret)
		return ret;

	bitmap_zero(free_bins, dev->num_arl_bins);

	/* Read the bins */
	for (i = 0; i < dev->num_arl_bins; i++) {
		u64 mac_vid;

		b53_read64(dev, B53_ARLIO_PAGE,
			   B53_ARLTBL_MAC_VID_ENTRY(i), &mac_vid);

		b53_arl_to_entry_25(ent, mac_vid);

		if (!(mac_vid & ARLTBL_VALID_25)) {
			set_bit(i, free_bins);
			continue;
		}
		if ((mac_vid & ARLTBL_MAC_MASK) != mac)
			continue;
		if (dev->vlan_enabled &&
		    ((mac_vid >> ARLTBL_VID_S_65) & ARLTBL_VID_MASK_25) != vid)
			continue;
		*idx = i;
		return 0;
	}

	*idx = find_first_bit(free_bins, dev->num_arl_bins);
	return *idx >= dev->num_arl_bins ? -ENOSPC : -ENOENT;
}

static int b53_arl_op(struct b53_device *dev, int op, int port,
		      const unsigned char *addr, u16 vid, bool is_valid)
{
@@ -1786,6 +1825,9 @@ static int b53_arl_op(struct b53_device *dev, int op, int port,
	if (ret)
		return ret;

	if (is5325(dev) || is5365(dev))
		ret = b53_arl_read_25(dev, mac, vid, &ent, &idx);
	else
		ret = b53_arl_read(dev, mac, vid, &ent, &idx);

	/* If this is a read, just finish now */
@@ -1830,10 +1872,15 @@ static int b53_arl_op(struct b53_device *dev, int op, int port,
	ent.is_static = true;
	ent.is_age = false;
	memcpy(ent.mac, addr, ETH_ALEN);
	if (is5325(dev) || is5365(dev))
		b53_arl_from_entry_25(&mac_vid, &ent);
	else
		b53_arl_from_entry(&mac_vid, &fwd_entry, &ent);

	b53_write64(dev, B53_ARLIO_PAGE,
		    B53_ARLTBL_MAC_VID_ENTRY(idx), mac_vid);

	if (!is5325(dev) && !is5365(dev))
		b53_write32(dev, B53_ARLIO_PAGE,
			    B53_ARLTBL_DATA_ENTRY(idx), fwd_entry);

@@ -1847,12 +1894,6 @@ int b53_fdb_add(struct dsa_switch *ds, int port,
	struct b53_device *priv = ds->priv;
	int ret;

	/* 5325 and 5365 require some more massaging, but could
	 * be supported eventually
	 */
	if (is5325(priv) || is5365(priv))
		return -EOPNOTSUPP;

	mutex_lock(&priv->arl_mutex);
	ret = b53_arl_op(priv, 0, port, addr, vid, true);
	mutex_unlock(&priv->arl_mutex);
@@ -1879,10 +1920,15 @@ EXPORT_SYMBOL(b53_fdb_del);
static int b53_arl_search_wait(struct b53_device *dev)
{
	unsigned int timeout = 1000;
	u8 reg;
	u8 reg, offset;

	if (is5325(dev) || is5365(dev))
		offset = B53_ARL_SRCH_CTL_25;
	else
		offset = B53_ARL_SRCH_CTL;

	do {
		b53_read8(dev, B53_ARLIO_PAGE, B53_ARL_SRCH_CTL, &reg);
		b53_read8(dev, B53_ARLIO_PAGE, offset, &reg);
		if (!(reg & ARL_SRCH_STDN))
			return 0;

@@ -1899,14 +1945,25 @@ static void b53_arl_search_rd(struct b53_device *dev, u8 idx,
			      struct b53_arl_entry *ent)
{
	u64 mac_vid;

	if (is5325(dev)) {
		b53_read64(dev, B53_ARLIO_PAGE, B53_ARL_SRCH_RSTL_0_MACVID_25,
			   &mac_vid);
		b53_arl_to_entry_25(ent, mac_vid);
	} else if (is5365(dev)) {
		b53_read64(dev, B53_ARLIO_PAGE, B53_ARL_SRCH_RSTL_0_MACVID_65,
			   &mac_vid);
		b53_arl_to_entry_25(ent, mac_vid);
	} else {
		u32 fwd_entry;

	b53_read64(dev, B53_ARLIO_PAGE,
		   B53_ARL_SRCH_RSTL_MACVID(idx), &mac_vid);
	b53_read32(dev, B53_ARLIO_PAGE,
		   B53_ARL_SRCH_RSTL(idx), &fwd_entry);
		b53_read64(dev, B53_ARLIO_PAGE, B53_ARL_SRCH_RSTL_MACVID(idx),
			   &mac_vid);
		b53_read32(dev, B53_ARLIO_PAGE, B53_ARL_SRCH_RSTL(idx),
			   &fwd_entry);
		b53_arl_to_entry(ent, mac_vid, fwd_entry);
	}
}

static int b53_fdb_copy(int port, const struct b53_arl_entry *ent,
			dsa_fdb_dump_cb_t *cb, void *data)
@@ -1926,14 +1983,20 @@ int b53_fdb_dump(struct dsa_switch *ds, int port,
	struct b53_device *priv = ds->priv;
	struct b53_arl_entry results[2];
	unsigned int count = 0;
	u8 offset;
	int ret;
	u8 reg;

	mutex_lock(&priv->arl_mutex);

	if (is5325(priv) || is5365(priv))
		offset = B53_ARL_SRCH_CTL_25;
	else
		offset = B53_ARL_SRCH_CTL;

	/* Start search operation */
	reg = ARL_SRCH_STDN;
	b53_write8(priv, B53_ARLIO_PAGE, B53_ARL_SRCH_CTL, reg);
	b53_write8(priv, offset, B53_ARL_SRCH_CTL, reg);

	do {
		ret = b53_arl_search_wait(priv);
+29 −0
Original line number Diff line number Diff line
@@ -317,6 +317,19 @@ static inline void b53_arl_to_entry(struct b53_arl_entry *ent,
	ent->vid = mac_vid >> ARLTBL_VID_S;
}

static inline void b53_arl_to_entry_25(struct b53_arl_entry *ent,
				       u64 mac_vid)
{
	memset(ent, 0, sizeof(*ent));
	ent->port = (mac_vid >> ARLTBL_DATA_PORT_ID_S_25) &
		     ARLTBL_DATA_PORT_ID_MASK_25;
	ent->is_valid = !!(mac_vid & ARLTBL_VALID_25);
	ent->is_age = !!(mac_vid & ARLTBL_AGE_25);
	ent->is_static = !!(mac_vid & ARLTBL_STATIC_25);
	u64_to_ether_addr(mac_vid, ent->mac);
	ent->vid = mac_vid >> ARLTBL_VID_S_65;
}

static inline void b53_arl_from_entry(u64 *mac_vid, u32 *fwd_entry,
				      const struct b53_arl_entry *ent)
{
@@ -331,6 +344,22 @@ static inline void b53_arl_from_entry(u64 *mac_vid, u32 *fwd_entry,
		*fwd_entry |= ARLTBL_AGE;
}

static inline void b53_arl_from_entry_25(u64 *mac_vid,
					 const struct b53_arl_entry *ent)
{
	*mac_vid = ether_addr_to_u64(ent->mac);
	*mac_vid |= (u64)(ent->port & ARLTBL_DATA_PORT_ID_MASK_25) <<
			  ARLTBL_DATA_PORT_ID_S_25;
	*mac_vid |= (u64)(ent->vid & ARLTBL_VID_MASK_25) <<
			  ARLTBL_VID_S_65;
	if (ent->is_valid)
		*mac_vid |= ARLTBL_VALID_25;
	if (ent->is_static)
		*mac_vid |= ARLTBL_STATIC_25;
	if (ent->is_age)
		*mac_vid |= ARLTBL_AGE_25;
}

#ifdef CONFIG_BCM47XX

#include <linux/bcm47xx_nvram.h>
+4 −3
Original line number Diff line number Diff line
@@ -324,9 +324,10 @@
#define   ARLTBL_VID_MASK		0xfff
#define   ARLTBL_DATA_PORT_ID_S_25	48
#define   ARLTBL_DATA_PORT_ID_MASK_25	0xf
#define   ARLTBL_AGE_25			BIT(61)
#define   ARLTBL_STATIC_25		BIT(62)
#define   ARLTBL_VALID_25		BIT(63)
#define   ARLTBL_VID_S_65		53
#define   ARLTBL_AGE_25			BIT_ULL(61)
#define   ARLTBL_STATIC_25		BIT_ULL(62)
#define   ARLTBL_VALID_25		BIT_ULL(63)

/* ARL Table Data Entry N Registers (32 bit) */
#define B53_ARLTBL_DATA_ENTRY(n)	((0x10 * (n)) + 0x18)