xref: /linux/drivers/net/ethernet/microchip/lan966x/lan966x_switchdev.c (revision 4f2c0a4acffbec01079c28f839422e64ddeff004)
1cf2f6089SHoratiu Vultur // SPDX-License-Identifier: GPL-2.0+
2cf2f6089SHoratiu Vultur 
3cf2f6089SHoratiu Vultur #include <linux/if_bridge.h>
4cf2f6089SHoratiu Vultur #include <net/switchdev.h>
5cf2f6089SHoratiu Vultur 
6cf2f6089SHoratiu Vultur #include "lan966x_main.h"
7cf2f6089SHoratiu Vultur 
8cf2f6089SHoratiu Vultur static struct notifier_block lan966x_netdevice_nb __read_mostly;
9cf2f6089SHoratiu Vultur 
lan966x_port_set_mcast_ip_flood(struct lan966x_port * port,u32 pgid_ip)1047aeea0dSHoratiu Vultur static void lan966x_port_set_mcast_ip_flood(struct lan966x_port *port,
1147aeea0dSHoratiu Vultur 					    u32 pgid_ip)
1247aeea0dSHoratiu Vultur {
1347aeea0dSHoratiu Vultur 	struct lan966x *lan966x = port->lan966x;
1447aeea0dSHoratiu Vultur 	u32 flood_mask_ip;
1547aeea0dSHoratiu Vultur 
1647aeea0dSHoratiu Vultur 	flood_mask_ip = lan_rd(lan966x, ANA_PGID(pgid_ip));
1747aeea0dSHoratiu Vultur 	flood_mask_ip = ANA_PGID_PGID_GET(flood_mask_ip);
1847aeea0dSHoratiu Vultur 
1947aeea0dSHoratiu Vultur 	/* If mcast snooping is not enabled then use mcast flood mask
2047aeea0dSHoratiu Vultur 	 * to decide to enable multicast flooding or not.
2147aeea0dSHoratiu Vultur 	 */
2247aeea0dSHoratiu Vultur 	if (!port->mcast_ena) {
2347aeea0dSHoratiu Vultur 		u32 flood_mask;
2447aeea0dSHoratiu Vultur 
2547aeea0dSHoratiu Vultur 		flood_mask = lan_rd(lan966x, ANA_PGID(PGID_MC));
2647aeea0dSHoratiu Vultur 		flood_mask = ANA_PGID_PGID_GET(flood_mask);
2747aeea0dSHoratiu Vultur 
2847aeea0dSHoratiu Vultur 		if (flood_mask & BIT(port->chip_port))
2947aeea0dSHoratiu Vultur 			flood_mask_ip |= BIT(port->chip_port);
3047aeea0dSHoratiu Vultur 		else
3147aeea0dSHoratiu Vultur 			flood_mask_ip &= ~BIT(port->chip_port);
3247aeea0dSHoratiu Vultur 	} else {
3347aeea0dSHoratiu Vultur 		flood_mask_ip &= ~BIT(port->chip_port);
3447aeea0dSHoratiu Vultur 	}
3547aeea0dSHoratiu Vultur 
3647aeea0dSHoratiu Vultur 	lan_rmw(ANA_PGID_PGID_SET(flood_mask_ip),
3747aeea0dSHoratiu Vultur 		ANA_PGID_PGID,
3847aeea0dSHoratiu Vultur 		lan966x, ANA_PGID(pgid_ip));
3947aeea0dSHoratiu Vultur }
4047aeea0dSHoratiu Vultur 
lan966x_port_set_mcast_flood(struct lan966x_port * port,bool enabled)41e14f7239SHoratiu Vultur static void lan966x_port_set_mcast_flood(struct lan966x_port *port,
42e14f7239SHoratiu Vultur 					 bool enabled)
43e14f7239SHoratiu Vultur {
44e14f7239SHoratiu Vultur 	u32 val = lan_rd(port->lan966x, ANA_PGID(PGID_MC));
45e14f7239SHoratiu Vultur 
46e14f7239SHoratiu Vultur 	val = ANA_PGID_PGID_GET(val);
47e14f7239SHoratiu Vultur 	if (enabled)
48e14f7239SHoratiu Vultur 		val |= BIT(port->chip_port);
49e14f7239SHoratiu Vultur 	else
50e14f7239SHoratiu Vultur 		val &= ~BIT(port->chip_port);
51e14f7239SHoratiu Vultur 
52e14f7239SHoratiu Vultur 	lan_rmw(ANA_PGID_PGID_SET(val),
53e14f7239SHoratiu Vultur 		ANA_PGID_PGID,
54e14f7239SHoratiu Vultur 		port->lan966x, ANA_PGID(PGID_MC));
5547aeea0dSHoratiu Vultur 
5647aeea0dSHoratiu Vultur 	if (!port->mcast_ena) {
5747aeea0dSHoratiu Vultur 		lan966x_port_set_mcast_ip_flood(port, PGID_MCIPV4);
5847aeea0dSHoratiu Vultur 		lan966x_port_set_mcast_ip_flood(port, PGID_MCIPV6);
5947aeea0dSHoratiu Vultur 	}
60e14f7239SHoratiu Vultur }
61e14f7239SHoratiu Vultur 
lan966x_port_set_ucast_flood(struct lan966x_port * port,bool enabled)622e49761eSHoratiu Vultur static void lan966x_port_set_ucast_flood(struct lan966x_port *port,
632e49761eSHoratiu Vultur 					 bool enabled)
642e49761eSHoratiu Vultur {
652e49761eSHoratiu Vultur 	u32 val = lan_rd(port->lan966x, ANA_PGID(PGID_UC));
662e49761eSHoratiu Vultur 
672e49761eSHoratiu Vultur 	val = ANA_PGID_PGID_GET(val);
682e49761eSHoratiu Vultur 	if (enabled)
692e49761eSHoratiu Vultur 		val |= BIT(port->chip_port);
702e49761eSHoratiu Vultur 	else
712e49761eSHoratiu Vultur 		val &= ~BIT(port->chip_port);
722e49761eSHoratiu Vultur 
732e49761eSHoratiu Vultur 	lan_rmw(ANA_PGID_PGID_SET(val),
742e49761eSHoratiu Vultur 		ANA_PGID_PGID,
752e49761eSHoratiu Vultur 		port->lan966x, ANA_PGID(PGID_UC));
762e49761eSHoratiu Vultur }
772e49761eSHoratiu Vultur 
lan966x_port_set_bcast_flood(struct lan966x_port * port,bool enabled)782e49761eSHoratiu Vultur static void lan966x_port_set_bcast_flood(struct lan966x_port *port,
792e49761eSHoratiu Vultur 					 bool enabled)
802e49761eSHoratiu Vultur {
812e49761eSHoratiu Vultur 	u32 val = lan_rd(port->lan966x, ANA_PGID(PGID_BC));
822e49761eSHoratiu Vultur 
832e49761eSHoratiu Vultur 	val = ANA_PGID_PGID_GET(val);
842e49761eSHoratiu Vultur 	if (enabled)
852e49761eSHoratiu Vultur 		val |= BIT(port->chip_port);
862e49761eSHoratiu Vultur 	else
872e49761eSHoratiu Vultur 		val &= ~BIT(port->chip_port);
882e49761eSHoratiu Vultur 
892e49761eSHoratiu Vultur 	lan_rmw(ANA_PGID_PGID_SET(val),
902e49761eSHoratiu Vultur 		ANA_PGID_PGID,
912e49761eSHoratiu Vultur 		port->lan966x, ANA_PGID(PGID_BC));
922e49761eSHoratiu Vultur }
932e49761eSHoratiu Vultur 
lan966x_port_set_learning(struct lan966x_port * port,bool enabled)942e49761eSHoratiu Vultur static void lan966x_port_set_learning(struct lan966x_port *port, bool enabled)
952e49761eSHoratiu Vultur {
962e49761eSHoratiu Vultur 	lan_rmw(ANA_PORT_CFG_LEARN_ENA_SET(enabled),
972e49761eSHoratiu Vultur 		ANA_PORT_CFG_LEARN_ENA,
982e49761eSHoratiu Vultur 		port->lan966x, ANA_PORT_CFG(port->chip_port));
992e49761eSHoratiu Vultur 
1002e49761eSHoratiu Vultur 	port->learn_ena = enabled;
1012e49761eSHoratiu Vultur }
1022e49761eSHoratiu Vultur 
lan966x_port_bridge_flags(struct lan966x_port * port,struct switchdev_brport_flags flags)103e14f7239SHoratiu Vultur static void lan966x_port_bridge_flags(struct lan966x_port *port,
104e14f7239SHoratiu Vultur 				      struct switchdev_brport_flags flags)
105e14f7239SHoratiu Vultur {
106e14f7239SHoratiu Vultur 	if (flags.mask & BR_MCAST_FLOOD)
107e14f7239SHoratiu Vultur 		lan966x_port_set_mcast_flood(port,
108e14f7239SHoratiu Vultur 					     !!(flags.val & BR_MCAST_FLOOD));
1092e49761eSHoratiu Vultur 
1102e49761eSHoratiu Vultur 	if (flags.mask & BR_FLOOD)
1112e49761eSHoratiu Vultur 		lan966x_port_set_ucast_flood(port,
1122e49761eSHoratiu Vultur 					     !!(flags.val & BR_FLOOD));
1132e49761eSHoratiu Vultur 
1142e49761eSHoratiu Vultur 	if (flags.mask & BR_BCAST_FLOOD)
1152e49761eSHoratiu Vultur 		lan966x_port_set_bcast_flood(port,
1162e49761eSHoratiu Vultur 					     !!(flags.val & BR_BCAST_FLOOD));
1172e49761eSHoratiu Vultur 
1182e49761eSHoratiu Vultur 	if (flags.mask & BR_LEARNING)
1192e49761eSHoratiu Vultur 		lan966x_port_set_learning(port,
1202e49761eSHoratiu Vultur 					  !!(flags.val & BR_LEARNING));
121e14f7239SHoratiu Vultur }
122e14f7239SHoratiu Vultur 
lan966x_port_pre_bridge_flags(struct lan966x_port * port,struct switchdev_brport_flags flags)123e14f7239SHoratiu Vultur static int lan966x_port_pre_bridge_flags(struct lan966x_port *port,
124e14f7239SHoratiu Vultur 					 struct switchdev_brport_flags flags)
125e14f7239SHoratiu Vultur {
1262e49761eSHoratiu Vultur 	if (flags.mask & ~(BR_MCAST_FLOOD | BR_FLOOD | BR_BCAST_FLOOD |
1272e49761eSHoratiu Vultur 			   BR_LEARNING))
128e14f7239SHoratiu Vultur 		return -EINVAL;
129e14f7239SHoratiu Vultur 
130e14f7239SHoratiu Vultur 	return 0;
131e14f7239SHoratiu Vultur }
132e14f7239SHoratiu Vultur 
lan966x_update_fwd_mask(struct lan966x * lan966x)133cabc9d49SHoratiu Vultur void lan966x_update_fwd_mask(struct lan966x *lan966x)
134cf2f6089SHoratiu Vultur {
135cf2f6089SHoratiu Vultur 	int i;
136cf2f6089SHoratiu Vultur 
137cf2f6089SHoratiu Vultur 	for (i = 0; i < lan966x->num_phys_ports; i++) {
138cf2f6089SHoratiu Vultur 		struct lan966x_port *port = lan966x->ports[i];
139cf2f6089SHoratiu Vultur 		unsigned long mask = 0;
140cf2f6089SHoratiu Vultur 
141cabc9d49SHoratiu Vultur 		if (port && lan966x->bridge_fwd_mask & BIT(i)) {
142cf2f6089SHoratiu Vultur 			mask = lan966x->bridge_fwd_mask & ~BIT(i);
143cf2f6089SHoratiu Vultur 
144cabc9d49SHoratiu Vultur 			if (port->bond)
145cabc9d49SHoratiu Vultur 				mask &= ~lan966x_lag_get_mask(lan966x,
146cabc9d49SHoratiu Vultur 							      port->bond);
147cabc9d49SHoratiu Vultur 		}
148cabc9d49SHoratiu Vultur 
149cf2f6089SHoratiu Vultur 		mask |= BIT(CPU_PORT);
150cf2f6089SHoratiu Vultur 
151cf2f6089SHoratiu Vultur 		lan_wr(ANA_PGID_PGID_SET(mask),
152cf2f6089SHoratiu Vultur 		       lan966x, ANA_PGID(PGID_SRC + i));
153cf2f6089SHoratiu Vultur 	}
154cf2f6089SHoratiu Vultur }
155cf2f6089SHoratiu Vultur 
lan966x_port_stp_state_set(struct lan966x_port * port,u8 state)156cabc9d49SHoratiu Vultur void lan966x_port_stp_state_set(struct lan966x_port *port, u8 state)
157cf2f6089SHoratiu Vultur {
158cf2f6089SHoratiu Vultur 	struct lan966x *lan966x = port->lan966x;
159cf2f6089SHoratiu Vultur 	bool learn_ena = false;
160cf2f6089SHoratiu Vultur 
1612e49761eSHoratiu Vultur 	if ((state == BR_STATE_FORWARDING || state == BR_STATE_LEARNING) &&
1622e49761eSHoratiu Vultur 	    port->learn_ena)
163cf2f6089SHoratiu Vultur 		learn_ena = true;
164cf2f6089SHoratiu Vultur 
165cf2f6089SHoratiu Vultur 	if (state == BR_STATE_FORWARDING)
166cf2f6089SHoratiu Vultur 		lan966x->bridge_fwd_mask |= BIT(port->chip_port);
167cf2f6089SHoratiu Vultur 	else
168cf2f6089SHoratiu Vultur 		lan966x->bridge_fwd_mask &= ~BIT(port->chip_port);
169cf2f6089SHoratiu Vultur 
170cf2f6089SHoratiu Vultur 	lan_rmw(ANA_PORT_CFG_LEARN_ENA_SET(learn_ena),
171cf2f6089SHoratiu Vultur 		ANA_PORT_CFG_LEARN_ENA,
172cf2f6089SHoratiu Vultur 		lan966x, ANA_PORT_CFG(port->chip_port));
173cf2f6089SHoratiu Vultur 
174cf2f6089SHoratiu Vultur 	lan966x_update_fwd_mask(lan966x);
175cf2f6089SHoratiu Vultur }
176cf2f6089SHoratiu Vultur 
lan966x_port_ageing_set(struct lan966x_port * port,unsigned long ageing_clock_t)177cabc9d49SHoratiu Vultur void lan966x_port_ageing_set(struct lan966x_port *port,
178cf2f6089SHoratiu Vultur 			     unsigned long ageing_clock_t)
179cf2f6089SHoratiu Vultur {
180cf2f6089SHoratiu Vultur 	unsigned long ageing_jiffies = clock_t_to_jiffies(ageing_clock_t);
181cf2f6089SHoratiu Vultur 	u32 ageing_time = jiffies_to_msecs(ageing_jiffies) / 1000;
182cf2f6089SHoratiu Vultur 
183cf2f6089SHoratiu Vultur 	lan966x_mac_set_ageing(port->lan966x, ageing_time);
184cf2f6089SHoratiu Vultur }
185cf2f6089SHoratiu Vultur 
lan966x_port_mc_set(struct lan966x_port * port,bool mcast_ena)18647aeea0dSHoratiu Vultur static void lan966x_port_mc_set(struct lan966x_port *port, bool mcast_ena)
18747aeea0dSHoratiu Vultur {
18847aeea0dSHoratiu Vultur 	struct lan966x *lan966x = port->lan966x;
18947aeea0dSHoratiu Vultur 
19047aeea0dSHoratiu Vultur 	port->mcast_ena = mcast_ena;
191add2c844SHoratiu Vultur 	if (mcast_ena)
192add2c844SHoratiu Vultur 		lan966x_mdb_restore_entries(lan966x);
193add2c844SHoratiu Vultur 	else
194add2c844SHoratiu Vultur 		lan966x_mdb_clear_entries(lan966x);
19547aeea0dSHoratiu Vultur 
19647aeea0dSHoratiu Vultur 	lan_rmw(ANA_CPU_FWD_CFG_IGMP_REDIR_ENA_SET(mcast_ena) |
19747aeea0dSHoratiu Vultur 		ANA_CPU_FWD_CFG_MLD_REDIR_ENA_SET(mcast_ena) |
19847aeea0dSHoratiu Vultur 		ANA_CPU_FWD_CFG_IPMC_CTRL_COPY_ENA_SET(mcast_ena),
19947aeea0dSHoratiu Vultur 		ANA_CPU_FWD_CFG_IGMP_REDIR_ENA |
20047aeea0dSHoratiu Vultur 		ANA_CPU_FWD_CFG_MLD_REDIR_ENA |
20147aeea0dSHoratiu Vultur 		ANA_CPU_FWD_CFG_IPMC_CTRL_COPY_ENA,
20247aeea0dSHoratiu Vultur 		lan966x, ANA_CPU_FWD_CFG(port->chip_port));
20347aeea0dSHoratiu Vultur 
20447aeea0dSHoratiu Vultur 	lan966x_port_set_mcast_ip_flood(port, PGID_MCIPV4);
20547aeea0dSHoratiu Vultur 	lan966x_port_set_mcast_ip_flood(port, PGID_MCIPV6);
20647aeea0dSHoratiu Vultur }
20747aeea0dSHoratiu Vultur 
lan966x_port_attr_set(struct net_device * dev,const void * ctx,const struct switchdev_attr * attr,struct netlink_ext_ack * extack)208cf2f6089SHoratiu Vultur static int lan966x_port_attr_set(struct net_device *dev, const void *ctx,
209cf2f6089SHoratiu Vultur 				 const struct switchdev_attr *attr,
210cf2f6089SHoratiu Vultur 				 struct netlink_ext_ack *extack)
211cf2f6089SHoratiu Vultur {
212cf2f6089SHoratiu Vultur 	struct lan966x_port *port = netdev_priv(dev);
213cf2f6089SHoratiu Vultur 	int err = 0;
214cf2f6089SHoratiu Vultur 
215cf2f6089SHoratiu Vultur 	if (ctx && ctx != port)
216cf2f6089SHoratiu Vultur 		return 0;
217cf2f6089SHoratiu Vultur 
218cf2f6089SHoratiu Vultur 	switch (attr->id) {
219e14f7239SHoratiu Vultur 	case SWITCHDEV_ATTR_ID_PORT_BRIDGE_FLAGS:
220e14f7239SHoratiu Vultur 		lan966x_port_bridge_flags(port, attr->u.brport_flags);
221e14f7239SHoratiu Vultur 		break;
222e14f7239SHoratiu Vultur 	case SWITCHDEV_ATTR_ID_PORT_PRE_BRIDGE_FLAGS:
223e14f7239SHoratiu Vultur 		err = lan966x_port_pre_bridge_flags(port, attr->u.brport_flags);
224e14f7239SHoratiu Vultur 		break;
225cf2f6089SHoratiu Vultur 	case SWITCHDEV_ATTR_ID_PORT_STP_STATE:
226cf2f6089SHoratiu Vultur 		lan966x_port_stp_state_set(port, attr->u.stp_state);
227cf2f6089SHoratiu Vultur 		break;
228cf2f6089SHoratiu Vultur 	case SWITCHDEV_ATTR_ID_BRIDGE_AGEING_TIME:
229cf2f6089SHoratiu Vultur 		lan966x_port_ageing_set(port, attr->u.ageing_time);
230cf2f6089SHoratiu Vultur 		break;
2316d2c186aSHoratiu Vultur 	case SWITCHDEV_ATTR_ID_BRIDGE_VLAN_FILTERING:
2326d2c186aSHoratiu Vultur 		lan966x_vlan_port_set_vlan_aware(port, attr->u.vlan_filtering);
2336d2c186aSHoratiu Vultur 		lan966x_vlan_port_apply(port);
2346d2c186aSHoratiu Vultur 		break;
23547aeea0dSHoratiu Vultur 	case SWITCHDEV_ATTR_ID_BRIDGE_MC_DISABLED:
23647aeea0dSHoratiu Vultur 		lan966x_port_mc_set(port, !attr->u.mc_disabled);
23747aeea0dSHoratiu Vultur 		break;
238cf2f6089SHoratiu Vultur 	default:
239cf2f6089SHoratiu Vultur 		err = -EOPNOTSUPP;
240cf2f6089SHoratiu Vultur 		break;
241cf2f6089SHoratiu Vultur 	}
242cf2f6089SHoratiu Vultur 
243cf2f6089SHoratiu Vultur 	return err;
244cf2f6089SHoratiu Vultur }
245cf2f6089SHoratiu Vultur 
lan966x_port_bridge_join(struct lan966x_port * port,struct net_device * brport_dev,struct net_device * bridge,struct netlink_ext_ack * extack)246cf2f6089SHoratiu Vultur static int lan966x_port_bridge_join(struct lan966x_port *port,
247cabc9d49SHoratiu Vultur 				    struct net_device *brport_dev,
248cf2f6089SHoratiu Vultur 				    struct net_device *bridge,
249cf2f6089SHoratiu Vultur 				    struct netlink_ext_ack *extack)
250cf2f6089SHoratiu Vultur {
2512e49761eSHoratiu Vultur 	struct switchdev_brport_flags flags = {0};
252cf2f6089SHoratiu Vultur 	struct lan966x *lan966x = port->lan966x;
253cf2f6089SHoratiu Vultur 	struct net_device *dev = port->dev;
254cf2f6089SHoratiu Vultur 	int err;
255cf2f6089SHoratiu Vultur 
256cf2f6089SHoratiu Vultur 	if (!lan966x->bridge_mask) {
257cf2f6089SHoratiu Vultur 		lan966x->bridge = bridge;
258cf2f6089SHoratiu Vultur 	} else {
259cf2f6089SHoratiu Vultur 		if (lan966x->bridge != bridge) {
260cf2f6089SHoratiu Vultur 			NL_SET_ERR_MSG_MOD(extack, "Not allow to add port to different bridge");
261cf2f6089SHoratiu Vultur 			return -ENODEV;
262cf2f6089SHoratiu Vultur 		}
263cf2f6089SHoratiu Vultur 	}
264cf2f6089SHoratiu Vultur 
265cabc9d49SHoratiu Vultur 	err = switchdev_bridge_port_offload(brport_dev, dev, port,
266cf2f6089SHoratiu Vultur 					    &lan966x_switchdev_nb,
267cf2f6089SHoratiu Vultur 					    &lan966x_switchdev_blocking_nb,
268cf2f6089SHoratiu Vultur 					    false, extack);
269cf2f6089SHoratiu Vultur 	if (err)
270cf2f6089SHoratiu Vultur 		return err;
271cf2f6089SHoratiu Vultur 
272cf2f6089SHoratiu Vultur 	lan966x->bridge_mask |= BIT(port->chip_port);
273cf2f6089SHoratiu Vultur 
2742e49761eSHoratiu Vultur 	flags.mask = BR_LEARNING | BR_FLOOD | BR_MCAST_FLOOD | BR_BCAST_FLOOD;
2752e49761eSHoratiu Vultur 	flags.val = flags.mask;
2762e49761eSHoratiu Vultur 	lan966x_port_bridge_flags(port, flags);
2772e49761eSHoratiu Vultur 
278cf2f6089SHoratiu Vultur 	return 0;
279cf2f6089SHoratiu Vultur }
280cf2f6089SHoratiu Vultur 
lan966x_port_bridge_leave(struct lan966x_port * port,struct net_device * bridge)281cf2f6089SHoratiu Vultur static void lan966x_port_bridge_leave(struct lan966x_port *port,
282cf2f6089SHoratiu Vultur 				      struct net_device *bridge)
283cf2f6089SHoratiu Vultur {
2842e49761eSHoratiu Vultur 	struct switchdev_brport_flags flags = {0};
285cf2f6089SHoratiu Vultur 	struct lan966x *lan966x = port->lan966x;
286cf2f6089SHoratiu Vultur 
2872e49761eSHoratiu Vultur 	flags.mask = BR_LEARNING | BR_FLOOD | BR_MCAST_FLOOD | BR_BCAST_FLOOD;
2882e49761eSHoratiu Vultur 	flags.val = flags.mask & ~BR_LEARNING;
2892e49761eSHoratiu Vultur 	lan966x_port_bridge_flags(port, flags);
2902e49761eSHoratiu Vultur 
291cf2f6089SHoratiu Vultur 	lan966x->bridge_mask &= ~BIT(port->chip_port);
292cf2f6089SHoratiu Vultur 
293cf2f6089SHoratiu Vultur 	if (!lan966x->bridge_mask)
294cf2f6089SHoratiu Vultur 		lan966x->bridge = NULL;
295cf2f6089SHoratiu Vultur 
2966d2c186aSHoratiu Vultur 	/* Set the port back to host mode */
2976d2c186aSHoratiu Vultur 	lan966x_vlan_port_set_vlan_aware(port, false);
2986d2c186aSHoratiu Vultur 	lan966x_vlan_port_set_vid(port, HOST_PVID, false, false);
2996d2c186aSHoratiu Vultur 	lan966x_vlan_port_apply(port);
300cf2f6089SHoratiu Vultur }
301cf2f6089SHoratiu Vultur 
lan966x_port_changeupper(struct net_device * dev,struct net_device * brport_dev,struct netdev_notifier_changeupper_info * info)302cabc9d49SHoratiu Vultur int lan966x_port_changeupper(struct net_device *dev,
303cabc9d49SHoratiu Vultur 			     struct net_device *brport_dev,
304cf2f6089SHoratiu Vultur 			     struct netdev_notifier_changeupper_info *info)
305cf2f6089SHoratiu Vultur {
306cf2f6089SHoratiu Vultur 	struct lan966x_port *port = netdev_priv(dev);
307cf2f6089SHoratiu Vultur 	struct netlink_ext_ack *extack;
308cf2f6089SHoratiu Vultur 	int err = 0;
309cf2f6089SHoratiu Vultur 
310cf2f6089SHoratiu Vultur 	extack = netdev_notifier_info_to_extack(&info->info);
311cf2f6089SHoratiu Vultur 
312cf2f6089SHoratiu Vultur 	if (netif_is_bridge_master(info->upper_dev)) {
313cf2f6089SHoratiu Vultur 		if (info->linking)
314cabc9d49SHoratiu Vultur 			err = lan966x_port_bridge_join(port, brport_dev,
315cabc9d49SHoratiu Vultur 						       info->upper_dev,
316cf2f6089SHoratiu Vultur 						       extack);
317cf2f6089SHoratiu Vultur 		else
318cf2f6089SHoratiu Vultur 			lan966x_port_bridge_leave(port, info->upper_dev);
319cf2f6089SHoratiu Vultur 	}
320cf2f6089SHoratiu Vultur 
321cabc9d49SHoratiu Vultur 	if (netif_is_lag_master(info->upper_dev)) {
322cabc9d49SHoratiu Vultur 		if (info->linking)
323cabc9d49SHoratiu Vultur 			err = lan966x_lag_port_join(port, info->upper_dev,
324cabc9d49SHoratiu Vultur 						    info->upper_dev,
325cabc9d49SHoratiu Vultur 						    extack);
326cabc9d49SHoratiu Vultur 		else
327cabc9d49SHoratiu Vultur 			lan966x_lag_port_leave(port, info->upper_dev);
328cabc9d49SHoratiu Vultur 	}
329cabc9d49SHoratiu Vultur 
330cf2f6089SHoratiu Vultur 	return err;
331cf2f6089SHoratiu Vultur }
332cf2f6089SHoratiu Vultur 
lan966x_port_prechangeupper(struct net_device * dev,struct net_device * brport_dev,struct netdev_notifier_changeupper_info * info)333cabc9d49SHoratiu Vultur int lan966x_port_prechangeupper(struct net_device *dev,
334cabc9d49SHoratiu Vultur 				struct net_device *brport_dev,
335cf2f6089SHoratiu Vultur 				struct netdev_notifier_changeupper_info *info)
336cf2f6089SHoratiu Vultur {
337cf2f6089SHoratiu Vultur 	struct lan966x_port *port = netdev_priv(dev);
338cabc9d49SHoratiu Vultur 	int err = NOTIFY_DONE;
339cf2f6089SHoratiu Vultur 
34086bac7f1SHoratiu Vultur 	if (netif_is_bridge_master(info->upper_dev) && !info->linking) {
34186bac7f1SHoratiu Vultur 		switchdev_bridge_port_unoffload(port->dev, port, NULL, NULL);
34286bac7f1SHoratiu Vultur 		lan966x_fdb_flush_workqueue(port->lan966x);
34386bac7f1SHoratiu Vultur 	}
344cf2f6089SHoratiu Vultur 
345cabc9d49SHoratiu Vultur 	if (netif_is_lag_master(info->upper_dev)) {
346cabc9d49SHoratiu Vultur 		err = lan966x_lag_port_prechangeupper(dev, info);
347cabc9d49SHoratiu Vultur 		if (err || info->linking)
348cabc9d49SHoratiu Vultur 			return err;
349cabc9d49SHoratiu Vultur 
350cabc9d49SHoratiu Vultur 		switchdev_bridge_port_unoffload(brport_dev, port, NULL, NULL);
351*9be99f2dSHoratiu Vultur 		lan966x_fdb_flush_workqueue(port->lan966x);
352cabc9d49SHoratiu Vultur 	}
353cabc9d49SHoratiu Vultur 
354cabc9d49SHoratiu Vultur 	return err;
355cf2f6089SHoratiu Vultur }
356cf2f6089SHoratiu Vultur 
lan966x_foreign_bridging_check(struct net_device * upper,bool * has_foreign,bool * seen_lan966x,struct netlink_ext_ack * extack)357a751ea4dSHoratiu Vultur static int lan966x_foreign_bridging_check(struct net_device *upper,
358a751ea4dSHoratiu Vultur 					  bool *has_foreign,
359a751ea4dSHoratiu Vultur 					  bool *seen_lan966x,
360cf2f6089SHoratiu Vultur 					  struct netlink_ext_ack *extack)
361cf2f6089SHoratiu Vultur {
362cf2f6089SHoratiu Vultur 	struct lan966x *lan966x = NULL;
363cf2f6089SHoratiu Vultur 	struct net_device *dev;
364cf2f6089SHoratiu Vultur 	struct list_head *iter;
365cf2f6089SHoratiu Vultur 
366a751ea4dSHoratiu Vultur 	if (!netif_is_bridge_master(upper) &&
367a751ea4dSHoratiu Vultur 	    !netif_is_lag_master(upper))
368cf2f6089SHoratiu Vultur 		return 0;
369cf2f6089SHoratiu Vultur 
370a751ea4dSHoratiu Vultur 	netdev_for_each_lower_dev(upper, dev, iter) {
371cf2f6089SHoratiu Vultur 		if (lan966x_netdevice_check(dev)) {
372cf2f6089SHoratiu Vultur 			struct lan966x_port *port = netdev_priv(dev);
373cf2f6089SHoratiu Vultur 
374cf2f6089SHoratiu Vultur 			if (lan966x) {
375a751ea4dSHoratiu Vultur 				/* Upper already has at least one port of a
376cf2f6089SHoratiu Vultur 				 * lan966x switch inside it, check that it's
377cf2f6089SHoratiu Vultur 				 * the same instance of the driver.
378cf2f6089SHoratiu Vultur 				 */
379cf2f6089SHoratiu Vultur 				if (port->lan966x != lan966x) {
380cf2f6089SHoratiu Vultur 					NL_SET_ERR_MSG_MOD(extack,
381cf2f6089SHoratiu Vultur 							   "Bridging between multiple lan966x switches disallowed");
382cf2f6089SHoratiu Vultur 					return -EINVAL;
383cf2f6089SHoratiu Vultur 				}
384cf2f6089SHoratiu Vultur 			} else {
385cf2f6089SHoratiu Vultur 				/* This is the first lan966x port inside this
386a751ea4dSHoratiu Vultur 				 * upper device
387cf2f6089SHoratiu Vultur 				 */
388cf2f6089SHoratiu Vultur 				lan966x = port->lan966x;
389a751ea4dSHoratiu Vultur 				*seen_lan966x = true;
390cf2f6089SHoratiu Vultur 			}
391a751ea4dSHoratiu Vultur 		} else if (netif_is_lag_master(dev)) {
392a751ea4dSHoratiu Vultur 			/* Allow to have bond interfaces that have only lan966x
393a751ea4dSHoratiu Vultur 			 * devices
394a751ea4dSHoratiu Vultur 			 */
395a751ea4dSHoratiu Vultur 			if (lan966x_foreign_bridging_check(dev, has_foreign,
396a751ea4dSHoratiu Vultur 							   seen_lan966x,
397a751ea4dSHoratiu Vultur 							   extack))
398a751ea4dSHoratiu Vultur 				return -EINVAL;
399cf2f6089SHoratiu Vultur 		} else {
400a751ea4dSHoratiu Vultur 			*has_foreign = true;
401cf2f6089SHoratiu Vultur 		}
402cf2f6089SHoratiu Vultur 
403a751ea4dSHoratiu Vultur 		if (*seen_lan966x && *has_foreign) {
404cf2f6089SHoratiu Vultur 			NL_SET_ERR_MSG_MOD(extack,
405cf2f6089SHoratiu Vultur 					   "Bridging lan966x ports with foreign interfaces disallowed");
406cf2f6089SHoratiu Vultur 			return -EINVAL;
407cf2f6089SHoratiu Vultur 		}
408cf2f6089SHoratiu Vultur 	}
409cf2f6089SHoratiu Vultur 
410cf2f6089SHoratiu Vultur 	return 0;
411cf2f6089SHoratiu Vultur }
412cf2f6089SHoratiu Vultur 
lan966x_bridge_check(struct net_device * dev,struct netdev_notifier_changeupper_info * info)413cf2f6089SHoratiu Vultur static int lan966x_bridge_check(struct net_device *dev,
414cf2f6089SHoratiu Vultur 				struct netdev_notifier_changeupper_info *info)
415cf2f6089SHoratiu Vultur {
416a751ea4dSHoratiu Vultur 	bool has_foreign = false;
417a751ea4dSHoratiu Vultur 	bool seen_lan966x = false;
418a751ea4dSHoratiu Vultur 
419cf2f6089SHoratiu Vultur 	return lan966x_foreign_bridging_check(info->upper_dev,
420a751ea4dSHoratiu Vultur 					      &has_foreign,
421a751ea4dSHoratiu Vultur 					      &seen_lan966x,
422cf2f6089SHoratiu Vultur 					      info->info.extack);
423cf2f6089SHoratiu Vultur }
424cf2f6089SHoratiu Vultur 
lan966x_netdevice_port_event(struct net_device * dev,struct notifier_block * nb,unsigned long event,void * ptr)425cf2f6089SHoratiu Vultur static int lan966x_netdevice_port_event(struct net_device *dev,
426cf2f6089SHoratiu Vultur 					struct notifier_block *nb,
427cf2f6089SHoratiu Vultur 					unsigned long event, void *ptr)
428cf2f6089SHoratiu Vultur {
429cf2f6089SHoratiu Vultur 	int err = 0;
430cf2f6089SHoratiu Vultur 
431cf2f6089SHoratiu Vultur 	if (!lan966x_netdevice_check(dev)) {
432cabc9d49SHoratiu Vultur 		switch (event) {
433cabc9d49SHoratiu Vultur 		case NETDEV_CHANGEUPPER:
434cabc9d49SHoratiu Vultur 		case NETDEV_PRECHANGEUPPER:
435cabc9d49SHoratiu Vultur 			err = lan966x_bridge_check(dev, ptr);
436cabc9d49SHoratiu Vultur 			if (err)
437cabc9d49SHoratiu Vultur 				return err;
438cabc9d49SHoratiu Vultur 
439cabc9d49SHoratiu Vultur 			if (netif_is_lag_master(dev)) {
440cf2f6089SHoratiu Vultur 				if (event == NETDEV_CHANGEUPPER)
441cabc9d49SHoratiu Vultur 					err = lan966x_lag_netdev_changeupper(dev,
442cabc9d49SHoratiu Vultur 									     ptr);
443cabc9d49SHoratiu Vultur 				else
444cabc9d49SHoratiu Vultur 					err = lan966x_lag_netdev_prechangeupper(dev,
445cabc9d49SHoratiu Vultur 										ptr);
446cabc9d49SHoratiu Vultur 
447cabc9d49SHoratiu Vultur 				return err;
448cabc9d49SHoratiu Vultur 			}
449cabc9d49SHoratiu Vultur 			break;
450cabc9d49SHoratiu Vultur 		default:
451cabc9d49SHoratiu Vultur 			return 0;
452cabc9d49SHoratiu Vultur 		}
453cabc9d49SHoratiu Vultur 
454cf2f6089SHoratiu Vultur 		return 0;
455cf2f6089SHoratiu Vultur 	}
456cf2f6089SHoratiu Vultur 
457cf2f6089SHoratiu Vultur 	switch (event) {
458cf2f6089SHoratiu Vultur 	case NETDEV_PRECHANGEUPPER:
459cabc9d49SHoratiu Vultur 		err = lan966x_port_prechangeupper(dev, dev, ptr);
460cf2f6089SHoratiu Vultur 		break;
461cf2f6089SHoratiu Vultur 	case NETDEV_CHANGEUPPER:
462cf2f6089SHoratiu Vultur 		err = lan966x_bridge_check(dev, ptr);
463cf2f6089SHoratiu Vultur 		if (err)
464cf2f6089SHoratiu Vultur 			return err;
465cf2f6089SHoratiu Vultur 
466cabc9d49SHoratiu Vultur 		err = lan966x_port_changeupper(dev, dev, ptr);
467cabc9d49SHoratiu Vultur 		break;
468cabc9d49SHoratiu Vultur 	case NETDEV_CHANGELOWERSTATE:
469cabc9d49SHoratiu Vultur 		err = lan966x_lag_port_changelowerstate(dev, ptr);
470cf2f6089SHoratiu Vultur 		break;
471cf2f6089SHoratiu Vultur 	}
472cf2f6089SHoratiu Vultur 
473cf2f6089SHoratiu Vultur 	return err;
474cf2f6089SHoratiu Vultur }
475cf2f6089SHoratiu Vultur 
lan966x_netdevice_event(struct notifier_block * nb,unsigned long event,void * ptr)476cf2f6089SHoratiu Vultur static int lan966x_netdevice_event(struct notifier_block *nb,
477cf2f6089SHoratiu Vultur 				   unsigned long event, void *ptr)
478cf2f6089SHoratiu Vultur {
479cf2f6089SHoratiu Vultur 	struct net_device *dev = netdev_notifier_info_to_dev(ptr);
480cf2f6089SHoratiu Vultur 	int ret;
481cf2f6089SHoratiu Vultur 
482cf2f6089SHoratiu Vultur 	ret = lan966x_netdevice_port_event(dev, nb, event, ptr);
483cf2f6089SHoratiu Vultur 
484cf2f6089SHoratiu Vultur 	return notifier_from_errno(ret);
485cf2f6089SHoratiu Vultur }
486cf2f6089SHoratiu Vultur 
lan966x_foreign_dev_check(const struct net_device * dev,const struct net_device * foreign_dev)487811ba277SHoratiu Vultur static bool lan966x_foreign_dev_check(const struct net_device *dev,
488811ba277SHoratiu Vultur 				      const struct net_device *foreign_dev)
489811ba277SHoratiu Vultur {
490811ba277SHoratiu Vultur 	struct lan966x_port *port = netdev_priv(dev);
491811ba277SHoratiu Vultur 	struct lan966x *lan966x = port->lan966x;
492cabc9d49SHoratiu Vultur 	int i;
493811ba277SHoratiu Vultur 
494811ba277SHoratiu Vultur 	if (netif_is_bridge_master(foreign_dev))
495ec638740SVladimir Oltean 		if (lan966x->bridge == foreign_dev)
496811ba277SHoratiu Vultur 			return false;
497ec638740SVladimir Oltean 
498cabc9d49SHoratiu Vultur 	if (netif_is_lag_master(foreign_dev))
499cabc9d49SHoratiu Vultur 		for (i = 0; i < lan966x->num_phys_ports; ++i)
500cabc9d49SHoratiu Vultur 			if (lan966x->ports[i] &&
501cabc9d49SHoratiu Vultur 			    lan966x->ports[i]->bond == foreign_dev)
502cabc9d49SHoratiu Vultur 				return false;
503cabc9d49SHoratiu Vultur 
504ec638740SVladimir Oltean 	return true;
505811ba277SHoratiu Vultur }
506811ba277SHoratiu Vultur 
lan966x_switchdev_event(struct notifier_block * nb,unsigned long event,void * ptr)507cf2f6089SHoratiu Vultur static int lan966x_switchdev_event(struct notifier_block *nb,
508cf2f6089SHoratiu Vultur 				   unsigned long event, void *ptr)
509cf2f6089SHoratiu Vultur {
510cf2f6089SHoratiu Vultur 	struct net_device *dev = switchdev_notifier_info_to_dev(ptr);
511cf2f6089SHoratiu Vultur 	int err;
512cf2f6089SHoratiu Vultur 
513cf2f6089SHoratiu Vultur 	switch (event) {
514cf2f6089SHoratiu Vultur 	case SWITCHDEV_PORT_ATTR_SET:
515cf2f6089SHoratiu Vultur 		err = switchdev_handle_port_attr_set(dev, ptr,
516cf2f6089SHoratiu Vultur 						     lan966x_netdevice_check,
517cf2f6089SHoratiu Vultur 						     lan966x_port_attr_set);
518cf2f6089SHoratiu Vultur 		return notifier_from_errno(err);
519811ba277SHoratiu Vultur 	case SWITCHDEV_FDB_ADD_TO_DEVICE:
520811ba277SHoratiu Vultur 	case SWITCHDEV_FDB_DEL_TO_DEVICE:
521811ba277SHoratiu Vultur 		err = switchdev_handle_fdb_event_to_device(dev, event, ptr,
522811ba277SHoratiu Vultur 							   lan966x_netdevice_check,
523811ba277SHoratiu Vultur 							   lan966x_foreign_dev_check,
524ec638740SVladimir Oltean 							   lan966x_handle_fdb);
525811ba277SHoratiu Vultur 		return notifier_from_errno(err);
526cf2f6089SHoratiu Vultur 	}
527cf2f6089SHoratiu Vultur 
528cf2f6089SHoratiu Vultur 	return NOTIFY_DONE;
529cf2f6089SHoratiu Vultur }
530cf2f6089SHoratiu Vultur 
lan966x_handle_port_vlan_add(struct lan966x_port * port,const struct switchdev_obj * obj)5316d2c186aSHoratiu Vultur static int lan966x_handle_port_vlan_add(struct lan966x_port *port,
5326d2c186aSHoratiu Vultur 					const struct switchdev_obj *obj)
5336d2c186aSHoratiu Vultur {
5346d2c186aSHoratiu Vultur 	const struct switchdev_obj_port_vlan *v = SWITCHDEV_OBJ_PORT_VLAN(obj);
5356d2c186aSHoratiu Vultur 	struct lan966x *lan966x = port->lan966x;
5366d2c186aSHoratiu Vultur 
5376d2c186aSHoratiu Vultur 	if (!netif_is_bridge_master(obj->orig_dev))
5386d2c186aSHoratiu Vultur 		lan966x_vlan_port_add_vlan(port, v->vid,
5396d2c186aSHoratiu Vultur 					   v->flags & BRIDGE_VLAN_INFO_PVID,
5406d2c186aSHoratiu Vultur 					   v->flags & BRIDGE_VLAN_INFO_UNTAGGED);
5416d2c186aSHoratiu Vultur 	else
5426d2c186aSHoratiu Vultur 		lan966x_vlan_cpu_add_vlan(lan966x, v->vid);
5436d2c186aSHoratiu Vultur 
5446d2c186aSHoratiu Vultur 	return 0;
5456d2c186aSHoratiu Vultur }
5466d2c186aSHoratiu Vultur 
lan966x_handle_port_obj_add(struct net_device * dev,const void * ctx,const struct switchdev_obj * obj,struct netlink_ext_ack * extack)5476d2c186aSHoratiu Vultur static int lan966x_handle_port_obj_add(struct net_device *dev, const void *ctx,
5486d2c186aSHoratiu Vultur 				       const struct switchdev_obj *obj,
5496d2c186aSHoratiu Vultur 				       struct netlink_ext_ack *extack)
5506d2c186aSHoratiu Vultur {
5516d2c186aSHoratiu Vultur 	struct lan966x_port *port = netdev_priv(dev);
5526d2c186aSHoratiu Vultur 	int err;
5536d2c186aSHoratiu Vultur 
5546d2c186aSHoratiu Vultur 	if (ctx && ctx != port)
5556d2c186aSHoratiu Vultur 		return 0;
5566d2c186aSHoratiu Vultur 
5576d2c186aSHoratiu Vultur 	switch (obj->id) {
5586d2c186aSHoratiu Vultur 	case SWITCHDEV_OBJ_ID_PORT_VLAN:
5596d2c186aSHoratiu Vultur 		err = lan966x_handle_port_vlan_add(port, obj);
5606d2c186aSHoratiu Vultur 		break;
5617aacb894SHoratiu Vultur 	case SWITCHDEV_OBJ_ID_PORT_MDB:
5627aacb894SHoratiu Vultur 	case SWITCHDEV_OBJ_ID_HOST_MDB:
5637aacb894SHoratiu Vultur 		err = lan966x_handle_port_mdb_add(port, obj);
5647aacb894SHoratiu Vultur 		break;
5656d2c186aSHoratiu Vultur 	default:
5666d2c186aSHoratiu Vultur 		err = -EOPNOTSUPP;
5676d2c186aSHoratiu Vultur 		break;
5686d2c186aSHoratiu Vultur 	}
5696d2c186aSHoratiu Vultur 
5706d2c186aSHoratiu Vultur 	return err;
5716d2c186aSHoratiu Vultur }
5726d2c186aSHoratiu Vultur 
lan966x_handle_port_vlan_del(struct lan966x_port * port,const struct switchdev_obj * obj)5736d2c186aSHoratiu Vultur static int lan966x_handle_port_vlan_del(struct lan966x_port *port,
5746d2c186aSHoratiu Vultur 					const struct switchdev_obj *obj)
5756d2c186aSHoratiu Vultur {
5766d2c186aSHoratiu Vultur 	const struct switchdev_obj_port_vlan *v = SWITCHDEV_OBJ_PORT_VLAN(obj);
5776d2c186aSHoratiu Vultur 	struct lan966x *lan966x = port->lan966x;
5786d2c186aSHoratiu Vultur 
5796d2c186aSHoratiu Vultur 	if (!netif_is_bridge_master(obj->orig_dev))
5806d2c186aSHoratiu Vultur 		lan966x_vlan_port_del_vlan(port, v->vid);
5816d2c186aSHoratiu Vultur 	else
5826d2c186aSHoratiu Vultur 		lan966x_vlan_cpu_del_vlan(lan966x, v->vid);
5836d2c186aSHoratiu Vultur 
5846d2c186aSHoratiu Vultur 	return 0;
5856d2c186aSHoratiu Vultur }
5866d2c186aSHoratiu Vultur 
lan966x_handle_port_obj_del(struct net_device * dev,const void * ctx,const struct switchdev_obj * obj)5876d2c186aSHoratiu Vultur static int lan966x_handle_port_obj_del(struct net_device *dev, const void *ctx,
5886d2c186aSHoratiu Vultur 				       const struct switchdev_obj *obj)
5896d2c186aSHoratiu Vultur {
5906d2c186aSHoratiu Vultur 	struct lan966x_port *port = netdev_priv(dev);
5916d2c186aSHoratiu Vultur 	int err;
5926d2c186aSHoratiu Vultur 
5936d2c186aSHoratiu Vultur 	if (ctx && ctx != port)
5946d2c186aSHoratiu Vultur 		return 0;
5956d2c186aSHoratiu Vultur 
5966d2c186aSHoratiu Vultur 	switch (obj->id) {
5976d2c186aSHoratiu Vultur 	case SWITCHDEV_OBJ_ID_PORT_VLAN:
5986d2c186aSHoratiu Vultur 		err = lan966x_handle_port_vlan_del(port, obj);
5996d2c186aSHoratiu Vultur 		break;
6007aacb894SHoratiu Vultur 	case SWITCHDEV_OBJ_ID_PORT_MDB:
6017aacb894SHoratiu Vultur 	case SWITCHDEV_OBJ_ID_HOST_MDB:
6027aacb894SHoratiu Vultur 		err = lan966x_handle_port_mdb_del(port, obj);
6037aacb894SHoratiu Vultur 		break;
6046d2c186aSHoratiu Vultur 	default:
6056d2c186aSHoratiu Vultur 		err = -EOPNOTSUPP;
6066d2c186aSHoratiu Vultur 		break;
6076d2c186aSHoratiu Vultur 	}
6086d2c186aSHoratiu Vultur 
6096d2c186aSHoratiu Vultur 	return err;
6106d2c186aSHoratiu Vultur }
6116d2c186aSHoratiu Vultur 
lan966x_switchdev_blocking_event(struct notifier_block * nb,unsigned long event,void * ptr)612cf2f6089SHoratiu Vultur static int lan966x_switchdev_blocking_event(struct notifier_block *nb,
613cf2f6089SHoratiu Vultur 					    unsigned long event,
614cf2f6089SHoratiu Vultur 					    void *ptr)
615cf2f6089SHoratiu Vultur {
616cf2f6089SHoratiu Vultur 	struct net_device *dev = switchdev_notifier_info_to_dev(ptr);
617cf2f6089SHoratiu Vultur 	int err;
618cf2f6089SHoratiu Vultur 
619cf2f6089SHoratiu Vultur 	switch (event) {
6206d2c186aSHoratiu Vultur 	case SWITCHDEV_PORT_OBJ_ADD:
6216d2c186aSHoratiu Vultur 		err = switchdev_handle_port_obj_add(dev, ptr,
6226d2c186aSHoratiu Vultur 						    lan966x_netdevice_check,
6236d2c186aSHoratiu Vultur 						    lan966x_handle_port_obj_add);
6246d2c186aSHoratiu Vultur 		return notifier_from_errno(err);
6256d2c186aSHoratiu Vultur 	case SWITCHDEV_PORT_OBJ_DEL:
6266d2c186aSHoratiu Vultur 		err = switchdev_handle_port_obj_del(dev, ptr,
6276d2c186aSHoratiu Vultur 						    lan966x_netdevice_check,
6286d2c186aSHoratiu Vultur 						    lan966x_handle_port_obj_del);
6296d2c186aSHoratiu Vultur 		return notifier_from_errno(err);
630cf2f6089SHoratiu Vultur 	case SWITCHDEV_PORT_ATTR_SET:
631cf2f6089SHoratiu Vultur 		err = switchdev_handle_port_attr_set(dev, ptr,
632cf2f6089SHoratiu Vultur 						     lan966x_netdevice_check,
633cf2f6089SHoratiu Vultur 						     lan966x_port_attr_set);
634cf2f6089SHoratiu Vultur 		return notifier_from_errno(err);
635cf2f6089SHoratiu Vultur 	}
636cf2f6089SHoratiu Vultur 
637cf2f6089SHoratiu Vultur 	return NOTIFY_DONE;
638cf2f6089SHoratiu Vultur }
639cf2f6089SHoratiu Vultur 
640cf2f6089SHoratiu Vultur static struct notifier_block lan966x_netdevice_nb __read_mostly = {
641cf2f6089SHoratiu Vultur 	.notifier_call = lan966x_netdevice_event,
642cf2f6089SHoratiu Vultur };
643cf2f6089SHoratiu Vultur 
644d6208adfSHoratiu Vultur struct notifier_block lan966x_switchdev_nb __read_mostly = {
645cf2f6089SHoratiu Vultur 	.notifier_call = lan966x_switchdev_event,
646cf2f6089SHoratiu Vultur };
647cf2f6089SHoratiu Vultur 
648d6208adfSHoratiu Vultur struct notifier_block lan966x_switchdev_blocking_nb __read_mostly = {
649cf2f6089SHoratiu Vultur 	.notifier_call = lan966x_switchdev_blocking_event,
650cf2f6089SHoratiu Vultur };
651cf2f6089SHoratiu Vultur 
lan966x_register_notifier_blocks(void)652cf2f6089SHoratiu Vultur void lan966x_register_notifier_blocks(void)
653cf2f6089SHoratiu Vultur {
654cf2f6089SHoratiu Vultur 	register_netdevice_notifier(&lan966x_netdevice_nb);
655cf2f6089SHoratiu Vultur 	register_switchdev_notifier(&lan966x_switchdev_nb);
656cf2f6089SHoratiu Vultur 	register_switchdev_blocking_notifier(&lan966x_switchdev_blocking_nb);
657cf2f6089SHoratiu Vultur }
658cf2f6089SHoratiu Vultur 
lan966x_unregister_notifier_blocks(void)659cf2f6089SHoratiu Vultur void lan966x_unregister_notifier_blocks(void)
660cf2f6089SHoratiu Vultur {
661cf2f6089SHoratiu Vultur 	unregister_switchdev_blocking_notifier(&lan966x_switchdev_blocking_nb);
662cf2f6089SHoratiu Vultur 	unregister_switchdev_notifier(&lan966x_switchdev_nb);
663cf2f6089SHoratiu Vultur 	unregister_netdevice_notifier(&lan966x_netdevice_nb);
664cf2f6089SHoratiu Vultur }
665