[openwrt/openwrt] realtek: rtl931x: Fix l2 fdb entry handling
LEDE Commits
lede-commits at lists.infradead.org
Sun Aug 10 12:59:06 PDT 2025
robimarko pushed a commit to openwrt/openwrt.git, branch main:
https://git.openwrt.org/522294eeef54017b496a5f324961ccbdefb921d4
commit 522294eeef54017b496a5f324961ccbdefb921d4
Author: Harshal Gohel <hg at simonwunderlich.de>
AuthorDate: Tue Jul 15 20:21:20 2025 +0200
realtek: rtl931x: Fix l2 fdb entry handling
Previous implementation was directly copied from rtl930x and was not
working. Table field offsets are different between rlt931x and rtl930x
Signed-off-by: Harshal Gohel <hg at simonwunderlich.de>
Signed-off-by: Sharadanand Karanjkar <sk at simonwunderlich.de>
Link: https://github.com/openwrt/openwrt/pull/19580
Signed-off-by: Robert Marko <robimarko at gmail.com>
---
.../files-6.12/drivers/net/dsa/rtl83xx/rtl931x.c | 98 +++++++++++++---------
1 file changed, 58 insertions(+), 40 deletions(-)
diff --git a/target/linux/realtek/files-6.12/drivers/net/dsa/rtl83xx/rtl931x.c b/target/linux/realtek/files-6.12/drivers/net/dsa/rtl83xx/rtl931x.c
index 7d625e0104..36103b43dd 100644
--- a/target/linux/realtek/files-6.12/drivers/net/dsa/rtl83xx/rtl931x.c
+++ b/target/linux/realtek/files-6.12/drivers/net/dsa/rtl83xx/rtl931x.c
@@ -657,8 +657,7 @@ static void rtl931x_fill_l2_entry(u32 r[], struct rtl838x_l2_entry *e)
/* Check for trunk port */
if (r[2] & BIT(29)) {
e->is_trunk = true;
- e->stack_dev = (e->port >> 9) & 1;
- e->trunk = e->port & 0x3f;
+ e->trunk = e->port & 0xff;
} else {
e->is_trunk = false;
e->stack_dev = (e->port >> 6) & 0xf;
@@ -668,16 +667,11 @@ static void rtl931x_fill_l2_entry(u32 r[], struct rtl838x_l2_entry *e)
e->block_da = !!(r[2] & BIT(14));
e->block_sa = !!(r[2] & BIT(15));
e->suspended = !!(r[2] & BIT(12));
- e->age = (r[2] >> 16) & 3;
+ e->age = (r[2] >> 16) & 7;
+
+ /* HW doesn't use VID but FID for as key */
+ e->vid = (r[0] >> 16) & 0xfff;
- /* the UC_VID field in hardware is used for the VID or for the route id */
- if (e->next_hop) {
- e->nh_route_id = r[2] & 0x7ff;
- e->vid = 0;
- } else {
- e->vid = r[2] & 0xfff;
- e->nh_route_id = 0;
- }
if (e->is_l2_tunnel)
e->l2_tunnel_id = ((r[2] & 0xff) << 4) | (r[3] >> 28);
/* TODO: Implement VLAN conversion */
@@ -696,45 +690,49 @@ static void rtl931x_fill_l2_row(u32 r[], struct rtl838x_l2_entry *e)
u32 port;
if (!e->valid) {
- r[0] = r[1] = r[2] = 0;
+ r[0] = r[1] = r[2] = r[3] = 0;
return;
}
- r[2] = BIT(31); /* Set valid bit */
+ r[3] = 0;
+
+ r[0] = BIT(31); /* Set valid bit */
- r[0] = ((u32)e->mac[0]) << 24 |
- ((u32)e->mac[1]) << 16 |
- ((u32)e->mac[2]) << 8 |
- ((u32)e->mac[3]);
- r[1] = ((u32)e->mac[4]) << 24 |
- ((u32)e->mac[5]) << 16;
+ r[0] |= ((u32)e->mac[0]) << 8 |
+ ((u32)e->mac[1]);
+ r[1] = ((u32)e->mac[2]) << 24 |
+ ((u32)e->mac[3]) << 16 |
+ ((u32)e->mac[4]) << 8 |
+ ((u32)e->mac[5]);
- r[2] |= e->next_hop ? BIT(12) : 0;
+ r[0] |= e->is_open_flow ? BIT(30) : 0;
+ r[0] |= e->is_pe_forward ? BIT(29) : 0;
+ r[2] = e->next_hop ? BIT(30) : 0;
+ r[0] |= (e->rvid & 0xfff) << 16;
if (e->type == L2_UNICAST) {
- r[2] |= e->is_static ? BIT(14) : 0;
- r[1] |= e->rvid & 0xfff;
- r[2] |= (e->port & 0x3ff) << 20;
+ r[2] |= e->is_l2_tunnel ? BIT(31) : 0;
+ r[2] |= e->is_static ? BIT(13) : 0;
+
if (e->is_trunk) {
- r[2] |= BIT(30);
- port = e->stack_dev << 9 | (e->port & 0x3f);
+ r[2] |= BIT(29);
+ port = e->trunk & 0xff;
} else {
- port = (e->stack_dev & 0xf) << 6;
- port |= e->port & 0x3f;
+ port = e->port & 0x3f;
+ port |= (e->stack_dev & 0xf) << 6;
+ }
+
+ r[2] |= (port & 0x3ff) << 19;
+ r[2] |= e->block_da ? BIT(14) : 0;
+ r[2] |= e->block_sa ? BIT(15) : 0;
+ r[2] |= e->suspended ? BIT(12) : 0;
+ r[2] |= (e->age & 0x7) << 16;
+ if (e->is_l2_tunnel) {
+ r[2] |= (e->l2_tunnel_id >> 4) & 0xff;
+ r[3] |= (e->l2_tunnel_id & 0xf) << 28;
}
- r[2] |= port << 20;
- r[2] |= e->block_da ? BIT(15) : 0;
- r[2] |= e->block_sa ? BIT(17) : 0;
- r[2] |= e->suspended ? BIT(13) : 0;
- r[2] |= (e->age & 0x3) << 17;
- /* the UC_VID field in hardware is used for the VID or for the route id */
- if (e->next_hop)
- r[2] |= e->nh_route_id & 0x7ff;
- else
- r[2] |= e->vid & 0xfff;
} else { /* L2_MULTICAST */
- r[2] |= (e->mc_portmask_index & 0x3ff) << 16;
- r[2] |= e->mc_mac_index & 0x7ff;
+ r[2] |= (e->mc_portmask_index & 0xfff) << 18;
}
}
@@ -794,11 +792,31 @@ static u64 rtl931x_read_l2_entry_using_hash(u32 hash, u32 pos, struct rtl838x_l2
static u64 rtl931x_read_cam(int idx, struct rtl838x_l2_entry *e)
{
- return 0;
+ u32 r[4];
+ struct table_reg *q = rtl_table_get(RTL9310_TBL_0, 1);
+ rtl_table_read(q, idx);
+ for ( int i = 0; i < 4; i++)
+ r[i] = sw_r32(rtl_table_data(q, i));
+
+ rtl_table_release(q);
+ rtl931x_fill_l2_entry(r, e);
+ if (!e->valid)
+ return 0;
+
+ /* return mac with concatenated fid as unique id */
+ return ((((u64)(r[0] & 0xffff) << 32) | (u64)r[1]) << 12) | e->vid;
}
static void rtl931x_write_cam(int idx, struct rtl838x_l2_entry *e)
{
+ u32 r[4];
+ struct table_reg *q = rtl_table_get(RTL9310_TBL_0, 1);
+ rtl931x_fill_l2_row(r, e);
+
+ for (int i = 0; i < 4; i++)
+ sw_w32(r[i], rtl_table_data(q, i));
+ rtl_table_write(q, idx);
+ rtl_table_release(q);
}
static void rtl931x_write_l2_entry_using_hash(u32 hash, u32 pos, struct rtl838x_l2_entry *e)
More information about the lede-commits
mailing list