1cabc9d49SHoratiu Vultur // SPDX-License-Identifier: GPL-2.0+
2cabc9d49SHoratiu Vultur 
3cabc9d49SHoratiu Vultur #include <linux/if_bridge.h>
4cabc9d49SHoratiu Vultur 
5cabc9d49SHoratiu Vultur #include "lan966x_main.h"
6cabc9d49SHoratiu Vultur 
lan966x_lag_set_aggr_pgids(struct lan966x * lan966x)7cabc9d49SHoratiu Vultur static void lan966x_lag_set_aggr_pgids(struct lan966x *lan966x)
8cabc9d49SHoratiu Vultur {
9cabc9d49SHoratiu Vultur 	u32 visited = GENMASK(lan966x->num_phys_ports - 1, 0);
10cabc9d49SHoratiu Vultur 	int p, lag, i;
11cabc9d49SHoratiu Vultur 
12cabc9d49SHoratiu Vultur 	/* Reset destination and aggregation PGIDS */
13cabc9d49SHoratiu Vultur 	for (p = 0; p < lan966x->num_phys_ports; ++p)
14cabc9d49SHoratiu Vultur 		lan_wr(ANA_PGID_PGID_SET(BIT(p)),
15cabc9d49SHoratiu Vultur 		       lan966x, ANA_PGID(p));
16cabc9d49SHoratiu Vultur 
17cabc9d49SHoratiu Vultur 	for (p = PGID_AGGR; p < PGID_SRC; ++p)
18cabc9d49SHoratiu Vultur 		lan_wr(ANA_PGID_PGID_SET(visited),
19cabc9d49SHoratiu Vultur 		       lan966x, ANA_PGID(p));
20cabc9d49SHoratiu Vultur 
21cabc9d49SHoratiu Vultur 	/* The visited ports bitmask holds the list of ports offloading any
22cabc9d49SHoratiu Vultur 	 * bonding interface. Initially we mark all these ports as unvisited,
23cabc9d49SHoratiu Vultur 	 * then every time we visit a port in this bitmask, we know that it is
24cabc9d49SHoratiu Vultur 	 * the lowest numbered port, i.e. the one whose logical ID == physical
25cabc9d49SHoratiu Vultur 	 * port ID == LAG ID. So we mark as visited all further ports in the
26cabc9d49SHoratiu Vultur 	 * bitmask that are offloading the same bonding interface. This way,
27cabc9d49SHoratiu Vultur 	 * we set up the aggregation PGIDs only once per bonding interface.
28cabc9d49SHoratiu Vultur 	 */
29cabc9d49SHoratiu Vultur 	for (p = 0; p < lan966x->num_phys_ports; ++p) {
30cabc9d49SHoratiu Vultur 		struct lan966x_port *port = lan966x->ports[p];
31cabc9d49SHoratiu Vultur 
32cabc9d49SHoratiu Vultur 		if (!port || !port->bond)
33cabc9d49SHoratiu Vultur 			continue;
34cabc9d49SHoratiu Vultur 
35cabc9d49SHoratiu Vultur 		visited &= ~BIT(p);
36cabc9d49SHoratiu Vultur 	}
37cabc9d49SHoratiu Vultur 
38cabc9d49SHoratiu Vultur 	/* Now, set PGIDs for each active LAG */
39cabc9d49SHoratiu Vultur 	for (lag = 0; lag < lan966x->num_phys_ports; ++lag) {
40*15faa1f6SHoratiu Vultur 		struct lan966x_port *port = lan966x->ports[lag];
41cabc9d49SHoratiu Vultur 		int num_active_ports = 0;
42*15faa1f6SHoratiu Vultur 		struct net_device *bond;
43cabc9d49SHoratiu Vultur 		unsigned long bond_mask;
44cabc9d49SHoratiu Vultur 		u8 aggr_idx[16];
45cabc9d49SHoratiu Vultur 
46*15faa1f6SHoratiu Vultur 		if (!port || !port->bond || (visited & BIT(lag)))
47cabc9d49SHoratiu Vultur 			continue;
48cabc9d49SHoratiu Vultur 
49*15faa1f6SHoratiu Vultur 		bond = port->bond;
50cabc9d49SHoratiu Vultur 		bond_mask = lan966x_lag_get_mask(lan966x, bond);
51cabc9d49SHoratiu Vultur 
52cabc9d49SHoratiu Vultur 		for_each_set_bit(p, &bond_mask, lan966x->num_phys_ports) {
53cabc9d49SHoratiu Vultur 			struct lan966x_port *port = lan966x->ports[p];
54cabc9d49SHoratiu Vultur 
55*15faa1f6SHoratiu Vultur 			if (!port)
56*15faa1f6SHoratiu Vultur 				continue;
57*15faa1f6SHoratiu Vultur 
58cabc9d49SHoratiu Vultur 			lan_wr(ANA_PGID_PGID_SET(bond_mask),
59cabc9d49SHoratiu Vultur 			       lan966x, ANA_PGID(p));
60cabc9d49SHoratiu Vultur 			if (port->lag_tx_active)
61cabc9d49SHoratiu Vultur 				aggr_idx[num_active_ports++] = p;
62cabc9d49SHoratiu Vultur 		}
63cabc9d49SHoratiu Vultur 
64cabc9d49SHoratiu Vultur 		for (i = PGID_AGGR; i < PGID_SRC; ++i) {
65cabc9d49SHoratiu Vultur 			u32 ac;
66cabc9d49SHoratiu Vultur 
67cabc9d49SHoratiu Vultur 			ac = lan_rd(lan966x, ANA_PGID(i));
68cabc9d49SHoratiu Vultur 			ac &= ~bond_mask;
69cabc9d49SHoratiu Vultur 			/* Don't do division by zero if there was no active
70cabc9d49SHoratiu Vultur 			 * port. Just make all aggregation codes zero.
71cabc9d49SHoratiu Vultur 			 */
72cabc9d49SHoratiu Vultur 			if (num_active_ports)
73cabc9d49SHoratiu Vultur 				ac |= BIT(aggr_idx[i % num_active_ports]);
74cabc9d49SHoratiu Vultur 			lan_wr(ANA_PGID_PGID_SET(ac),
75cabc9d49SHoratiu Vultur 			       lan966x, ANA_PGID(i));
76cabc9d49SHoratiu Vultur 		}
77cabc9d49SHoratiu Vultur 
78cabc9d49SHoratiu Vultur 		/* Mark all ports in the same LAG as visited to avoid applying
79cabc9d49SHoratiu Vultur 		 * the same config again.
80cabc9d49SHoratiu Vultur 		 */
81cabc9d49SHoratiu Vultur 		for (p = lag; p < lan966x->num_phys_ports; p++) {
82cabc9d49SHoratiu Vultur 			struct lan966x_port *port = lan966x->ports[p];
83cabc9d49SHoratiu Vultur 
84cabc9d49SHoratiu Vultur 			if (!port)
85cabc9d49SHoratiu Vultur 				continue;
86cabc9d49SHoratiu Vultur 
87cabc9d49SHoratiu Vultur 			if (port->bond == bond)
88cabc9d49SHoratiu Vultur 				visited |= BIT(p);
89cabc9d49SHoratiu Vultur 		}
90cabc9d49SHoratiu Vultur 	}
91cabc9d49SHoratiu Vultur }
92cabc9d49SHoratiu Vultur 
lan966x_lag_set_port_ids(struct lan966x * lan966x)93cabc9d49SHoratiu Vultur static void lan966x_lag_set_port_ids(struct lan966x *lan966x)
94cabc9d49SHoratiu Vultur {
95cabc9d49SHoratiu Vultur 	struct lan966x_port *port;
96cabc9d49SHoratiu Vultur 	u32 bond_mask;
97cabc9d49SHoratiu Vultur 	u32 lag_id;
98cabc9d49SHoratiu Vultur 	int p;
99cabc9d49SHoratiu Vultur 
100cabc9d49SHoratiu Vultur 	for (p = 0; p < lan966x->num_phys_ports; ++p) {
101cabc9d49SHoratiu Vultur 		port = lan966x->ports[p];
102cabc9d49SHoratiu Vultur 		if (!port)
103cabc9d49SHoratiu Vultur 			continue;
104cabc9d49SHoratiu Vultur 
105cabc9d49SHoratiu Vultur 		lag_id = port->chip_port;
106cabc9d49SHoratiu Vultur 
107cabc9d49SHoratiu Vultur 		bond_mask = lan966x_lag_get_mask(lan966x, port->bond);
108cabc9d49SHoratiu Vultur 		if (bond_mask)
109cabc9d49SHoratiu Vultur 			lag_id = __ffs(bond_mask);
110cabc9d49SHoratiu Vultur 
111cabc9d49SHoratiu Vultur 		lan_rmw(ANA_PORT_CFG_PORTID_VAL_SET(lag_id),
112cabc9d49SHoratiu Vultur 			ANA_PORT_CFG_PORTID_VAL,
113cabc9d49SHoratiu Vultur 			lan966x, ANA_PORT_CFG(port->chip_port));
114cabc9d49SHoratiu Vultur 	}
115cabc9d49SHoratiu Vultur }
116cabc9d49SHoratiu Vultur 
lan966x_lag_update_ids(struct lan966x * lan966x)117cabc9d49SHoratiu Vultur static void lan966x_lag_update_ids(struct lan966x *lan966x)
118cabc9d49SHoratiu Vultur {
119cabc9d49SHoratiu Vultur 	lan966x_lag_set_port_ids(lan966x);
120cabc9d49SHoratiu Vultur 	lan966x_update_fwd_mask(lan966x);
121cabc9d49SHoratiu Vultur 	lan966x_lag_set_aggr_pgids(lan966x);
122cabc9d49SHoratiu Vultur }
123cabc9d49SHoratiu Vultur 
lan966x_lag_port_join(struct lan966x_port * port,struct net_device * brport_dev,struct net_device * bond,struct netlink_ext_ack * extack)124cabc9d49SHoratiu Vultur int lan966x_lag_port_join(struct lan966x_port *port,
125cabc9d49SHoratiu Vultur 			  struct net_device *brport_dev,
126cabc9d49SHoratiu Vultur 			  struct net_device *bond,
127cabc9d49SHoratiu Vultur 			  struct netlink_ext_ack *extack)
128cabc9d49SHoratiu Vultur {
129cabc9d49SHoratiu Vultur 	struct lan966x *lan966x = port->lan966x;
130cabc9d49SHoratiu Vultur 	struct net_device *dev = port->dev;
131e09ce977SHoratiu Vultur 	u32 lag_id = -1;
132e09ce977SHoratiu Vultur 	u32 bond_mask;
133cabc9d49SHoratiu Vultur 	int err;
134cabc9d49SHoratiu Vultur 
135e09ce977SHoratiu Vultur 	bond_mask = lan966x_lag_get_mask(lan966x, bond);
136e09ce977SHoratiu Vultur 	if (bond_mask)
137e09ce977SHoratiu Vultur 		lag_id = __ffs(bond_mask);
138e09ce977SHoratiu Vultur 
139cabc9d49SHoratiu Vultur 	port->bond = bond;
140cabc9d49SHoratiu Vultur 	lan966x_lag_update_ids(lan966x);
141cabc9d49SHoratiu Vultur 
142cabc9d49SHoratiu Vultur 	err = switchdev_bridge_port_offload(brport_dev, dev, port,
143cabc9d49SHoratiu Vultur 					    &lan966x_switchdev_nb,
144cabc9d49SHoratiu Vultur 					    &lan966x_switchdev_blocking_nb,
145cabc9d49SHoratiu Vultur 					    false, extack);
146cabc9d49SHoratiu Vultur 	if (err)
147cabc9d49SHoratiu Vultur 		goto out;
148cabc9d49SHoratiu Vultur 
149cabc9d49SHoratiu Vultur 	lan966x_port_stp_state_set(port, br_port_get_stp_state(brport_dev));
150cabc9d49SHoratiu Vultur 
151e09ce977SHoratiu Vultur 	if (lan966x_lag_first_port(port->bond, port->dev) &&
152e09ce977SHoratiu Vultur 	    lag_id != -1)
153e09ce977SHoratiu Vultur 		lan966x_mac_lag_replace_port_entry(lan966x,
154e09ce977SHoratiu Vultur 						   lan966x->ports[lag_id],
155e09ce977SHoratiu Vultur 						   port);
156e09ce977SHoratiu Vultur 
157cabc9d49SHoratiu Vultur 	return 0;
158cabc9d49SHoratiu Vultur 
159cabc9d49SHoratiu Vultur out:
160cabc9d49SHoratiu Vultur 	port->bond = NULL;
161cabc9d49SHoratiu Vultur 	lan966x_lag_update_ids(lan966x);
162cabc9d49SHoratiu Vultur 
163cabc9d49SHoratiu Vultur 	return err;
164cabc9d49SHoratiu Vultur }
165cabc9d49SHoratiu Vultur 
lan966x_lag_port_leave(struct lan966x_port * port,struct net_device * bond)166cabc9d49SHoratiu Vultur void lan966x_lag_port_leave(struct lan966x_port *port, struct net_device *bond)
167cabc9d49SHoratiu Vultur {
168cabc9d49SHoratiu Vultur 	struct lan966x *lan966x = port->lan966x;
169e09ce977SHoratiu Vultur 	u32 bond_mask;
170e09ce977SHoratiu Vultur 	u32 lag_id;
171e09ce977SHoratiu Vultur 
172e09ce977SHoratiu Vultur 	if (lan966x_lag_first_port(port->bond, port->dev)) {
173e09ce977SHoratiu Vultur 		bond_mask = lan966x_lag_get_mask(lan966x, port->bond);
174e09ce977SHoratiu Vultur 		bond_mask &= ~BIT(port->chip_port);
175e09ce977SHoratiu Vultur 		if (bond_mask) {
176e09ce977SHoratiu Vultur 			lag_id = __ffs(bond_mask);
177e09ce977SHoratiu Vultur 			lan966x_mac_lag_replace_port_entry(lan966x, port,
178e09ce977SHoratiu Vultur 							   lan966x->ports[lag_id]);
179e09ce977SHoratiu Vultur 		} else {
180e09ce977SHoratiu Vultur 			lan966x_mac_lag_remove_port_entry(lan966x, port);
181e09ce977SHoratiu Vultur 		}
182e09ce977SHoratiu Vultur 	}
183cabc9d49SHoratiu Vultur 
184cabc9d49SHoratiu Vultur 	port->bond = NULL;
185cabc9d49SHoratiu Vultur 	lan966x_lag_update_ids(lan966x);
186cabc9d49SHoratiu Vultur 	lan966x_port_stp_state_set(port, BR_STATE_FORWARDING);
187cabc9d49SHoratiu Vultur }
188cabc9d49SHoratiu Vultur 
lan966x_lag_port_check_hash_types(struct lan966x * lan966x,enum netdev_lag_hash hash_type)189cabc9d49SHoratiu Vultur static bool lan966x_lag_port_check_hash_types(struct lan966x *lan966x,
190cabc9d49SHoratiu Vultur 					      enum netdev_lag_hash hash_type)
191cabc9d49SHoratiu Vultur {
192cabc9d49SHoratiu Vultur 	int p;
193cabc9d49SHoratiu Vultur 
194cabc9d49SHoratiu Vultur 	for (p = 0; p < lan966x->num_phys_ports; ++p) {
195cabc9d49SHoratiu Vultur 		struct lan966x_port *port = lan966x->ports[p];
196cabc9d49SHoratiu Vultur 
197cabc9d49SHoratiu Vultur 		if (!port || !port->bond)
198cabc9d49SHoratiu Vultur 			continue;
199cabc9d49SHoratiu Vultur 
200cabc9d49SHoratiu Vultur 		if (port->hash_type != hash_type)
201cabc9d49SHoratiu Vultur 			return false;
202cabc9d49SHoratiu Vultur 	}
203cabc9d49SHoratiu Vultur 
204cabc9d49SHoratiu Vultur 	return true;
205cabc9d49SHoratiu Vultur }
206cabc9d49SHoratiu Vultur 
lan966x_lag_port_prechangeupper(struct net_device * dev,struct netdev_notifier_changeupper_info * info)207cabc9d49SHoratiu Vultur int lan966x_lag_port_prechangeupper(struct net_device *dev,
208cabc9d49SHoratiu Vultur 				    struct netdev_notifier_changeupper_info *info)
209cabc9d49SHoratiu Vultur {
210cabc9d49SHoratiu Vultur 	struct lan966x_port *port = netdev_priv(dev);
211cabc9d49SHoratiu Vultur 	struct lan966x *lan966x = port->lan966x;
212cabc9d49SHoratiu Vultur 	struct netdev_lag_upper_info *lui;
213cabc9d49SHoratiu Vultur 	struct netlink_ext_ack *extack;
214cabc9d49SHoratiu Vultur 
215cabc9d49SHoratiu Vultur 	extack = netdev_notifier_info_to_extack(&info->info);
216cabc9d49SHoratiu Vultur 	lui = info->upper_info;
217cabc9d49SHoratiu Vultur 	if (!lui) {
218cabc9d49SHoratiu Vultur 		port->hash_type = NETDEV_LAG_HASH_NONE;
219cabc9d49SHoratiu Vultur 		return NOTIFY_DONE;
220cabc9d49SHoratiu Vultur 	}
221cabc9d49SHoratiu Vultur 
222cabc9d49SHoratiu Vultur 	if (lui->tx_type != NETDEV_LAG_TX_TYPE_HASH) {
223cabc9d49SHoratiu Vultur 		NL_SET_ERR_MSG_MOD(extack,
224cabc9d49SHoratiu Vultur 				   "LAG device using unsupported Tx type");
225cabc9d49SHoratiu Vultur 		return -EINVAL;
226cabc9d49SHoratiu Vultur 	}
227cabc9d49SHoratiu Vultur 
228cabc9d49SHoratiu Vultur 	if (!lan966x_lag_port_check_hash_types(lan966x, lui->hash_type)) {
229cabc9d49SHoratiu Vultur 		NL_SET_ERR_MSG_MOD(extack,
230cabc9d49SHoratiu Vultur 				   "LAG devices can have only the same hash_type");
231cabc9d49SHoratiu Vultur 		return -EINVAL;
232cabc9d49SHoratiu Vultur 	}
233cabc9d49SHoratiu Vultur 
234cabc9d49SHoratiu Vultur 	switch (lui->hash_type) {
235cabc9d49SHoratiu Vultur 	case NETDEV_LAG_HASH_L2:
236cabc9d49SHoratiu Vultur 		lan_wr(ANA_AGGR_CFG_AC_DMAC_ENA_SET(1) |
237cabc9d49SHoratiu Vultur 		       ANA_AGGR_CFG_AC_SMAC_ENA_SET(1),
238cabc9d49SHoratiu Vultur 		       lan966x, ANA_AGGR_CFG);
239cabc9d49SHoratiu Vultur 		break;
240cabc9d49SHoratiu Vultur 	case NETDEV_LAG_HASH_L34:
241cabc9d49SHoratiu Vultur 		lan_wr(ANA_AGGR_CFG_AC_IP6_TCPUDP_ENA_SET(1) |
242cabc9d49SHoratiu Vultur 		       ANA_AGGR_CFG_AC_IP4_TCPUDP_ENA_SET(1) |
243cabc9d49SHoratiu Vultur 		       ANA_AGGR_CFG_AC_IP4_SIPDIP_ENA_SET(1),
244cabc9d49SHoratiu Vultur 		       lan966x, ANA_AGGR_CFG);
245cabc9d49SHoratiu Vultur 		break;
246cabc9d49SHoratiu Vultur 	case NETDEV_LAG_HASH_L23:
247cabc9d49SHoratiu Vultur 		lan_wr(ANA_AGGR_CFG_AC_DMAC_ENA_SET(1) |
248cabc9d49SHoratiu Vultur 		       ANA_AGGR_CFG_AC_SMAC_ENA_SET(1) |
249cabc9d49SHoratiu Vultur 		       ANA_AGGR_CFG_AC_IP6_TCPUDP_ENA_SET(1) |
250cabc9d49SHoratiu Vultur 		       ANA_AGGR_CFG_AC_IP4_TCPUDP_ENA_SET(1),
251cabc9d49SHoratiu Vultur 		       lan966x, ANA_AGGR_CFG);
252cabc9d49SHoratiu Vultur 		break;
253cabc9d49SHoratiu Vultur 	default:
254cabc9d49SHoratiu Vultur 		NL_SET_ERR_MSG_MOD(extack,
255cabc9d49SHoratiu Vultur 				   "LAG device using unsupported hash type");
256cabc9d49SHoratiu Vultur 		return -EINVAL;
257cabc9d49SHoratiu Vultur 	}
258cabc9d49SHoratiu Vultur 
259cabc9d49SHoratiu Vultur 	port->hash_type = lui->hash_type;
260cabc9d49SHoratiu Vultur 
261cabc9d49SHoratiu Vultur 	return NOTIFY_OK;
262cabc9d49SHoratiu Vultur }
263cabc9d49SHoratiu Vultur 
lan966x_lag_port_changelowerstate(struct net_device * dev,struct netdev_notifier_changelowerstate_info * info)264cabc9d49SHoratiu Vultur int lan966x_lag_port_changelowerstate(struct net_device *dev,
265cabc9d49SHoratiu Vultur 				      struct netdev_notifier_changelowerstate_info *info)
266cabc9d49SHoratiu Vultur {
267cabc9d49SHoratiu Vultur 	struct netdev_lag_lower_state_info *lag = info->lower_state_info;
268cabc9d49SHoratiu Vultur 	struct lan966x_port *port = netdev_priv(dev);
269cabc9d49SHoratiu Vultur 	struct lan966x *lan966x = port->lan966x;
270cabc9d49SHoratiu Vultur 	bool is_active;
271cabc9d49SHoratiu Vultur 
272cabc9d49SHoratiu Vultur 	if (!port->bond)
273cabc9d49SHoratiu Vultur 		return NOTIFY_DONE;
274cabc9d49SHoratiu Vultur 
275cabc9d49SHoratiu Vultur 	is_active = lag->link_up && lag->tx_enabled;
276cabc9d49SHoratiu Vultur 	if (port->lag_tx_active == is_active)
277cabc9d49SHoratiu Vultur 		return NOTIFY_DONE;
278cabc9d49SHoratiu Vultur 
279cabc9d49SHoratiu Vultur 	port->lag_tx_active = is_active;
280cabc9d49SHoratiu Vultur 	lan966x_lag_set_aggr_pgids(lan966x);
281cabc9d49SHoratiu Vultur 
282cabc9d49SHoratiu Vultur 	return NOTIFY_OK;
283cabc9d49SHoratiu Vultur }
284cabc9d49SHoratiu Vultur 
lan966x_lag_netdev_prechangeupper(struct net_device * dev,struct netdev_notifier_changeupper_info * info)285cabc9d49SHoratiu Vultur int lan966x_lag_netdev_prechangeupper(struct net_device *dev,
286cabc9d49SHoratiu Vultur 				      struct netdev_notifier_changeupper_info *info)
287cabc9d49SHoratiu Vultur {
288cabc9d49SHoratiu Vultur 	struct lan966x_port *port;
289cabc9d49SHoratiu Vultur 	struct net_device *lower;
290cabc9d49SHoratiu Vultur 	struct list_head *iter;
291cabc9d49SHoratiu Vultur 	int err;
292cabc9d49SHoratiu Vultur 
293cabc9d49SHoratiu Vultur 	netdev_for_each_lower_dev(dev, lower, iter) {
294cabc9d49SHoratiu Vultur 		if (!lan966x_netdevice_check(lower))
295cabc9d49SHoratiu Vultur 			continue;
296cabc9d49SHoratiu Vultur 
297cabc9d49SHoratiu Vultur 		port = netdev_priv(lower);
298cabc9d49SHoratiu Vultur 		if (port->bond != dev)
299cabc9d49SHoratiu Vultur 			continue;
300cabc9d49SHoratiu Vultur 
301cabc9d49SHoratiu Vultur 		err = lan966x_port_prechangeupper(lower, dev, info);
302cabc9d49SHoratiu Vultur 		if (err)
303cabc9d49SHoratiu Vultur 			return err;
304cabc9d49SHoratiu Vultur 	}
305cabc9d49SHoratiu Vultur 
306cabc9d49SHoratiu Vultur 	return NOTIFY_DONE;
307cabc9d49SHoratiu Vultur }
308cabc9d49SHoratiu Vultur 
lan966x_lag_netdev_changeupper(struct net_device * dev,struct netdev_notifier_changeupper_info * info)309cabc9d49SHoratiu Vultur int lan966x_lag_netdev_changeupper(struct net_device *dev,
310cabc9d49SHoratiu Vultur 				   struct netdev_notifier_changeupper_info *info)
311cabc9d49SHoratiu Vultur {
312cabc9d49SHoratiu Vultur 	struct lan966x_port *port;
313cabc9d49SHoratiu Vultur 	struct net_device *lower;
314cabc9d49SHoratiu Vultur 	struct list_head *iter;
315cabc9d49SHoratiu Vultur 	int err;
316cabc9d49SHoratiu Vultur 
317cabc9d49SHoratiu Vultur 	netdev_for_each_lower_dev(dev, lower, iter) {
318cabc9d49SHoratiu Vultur 		if (!lan966x_netdevice_check(lower))
319cabc9d49SHoratiu Vultur 			continue;
320cabc9d49SHoratiu Vultur 
321cabc9d49SHoratiu Vultur 		port = netdev_priv(lower);
322cabc9d49SHoratiu Vultur 		if (port->bond != dev)
323cabc9d49SHoratiu Vultur 			continue;
324cabc9d49SHoratiu Vultur 
325cabc9d49SHoratiu Vultur 		err = lan966x_port_changeupper(lower, dev, info);
326cabc9d49SHoratiu Vultur 		if (err)
327cabc9d49SHoratiu Vultur 			return err;
328cabc9d49SHoratiu Vultur 	}
329cabc9d49SHoratiu Vultur 
330cabc9d49SHoratiu Vultur 	return NOTIFY_DONE;
331cabc9d49SHoratiu Vultur }
332cabc9d49SHoratiu Vultur 
lan966x_lag_first_port(struct net_device * lag,struct net_device * dev)333cabc9d49SHoratiu Vultur bool lan966x_lag_first_port(struct net_device *lag, struct net_device *dev)
334cabc9d49SHoratiu Vultur {
335cabc9d49SHoratiu Vultur 	struct lan966x_port *port = netdev_priv(dev);
336cabc9d49SHoratiu Vultur 	struct lan966x *lan966x = port->lan966x;
337cabc9d49SHoratiu Vultur 	unsigned long bond_mask;
338cabc9d49SHoratiu Vultur 
339cabc9d49SHoratiu Vultur 	if (port->bond != lag)
340cabc9d49SHoratiu Vultur 		return false;
341cabc9d49SHoratiu Vultur 
342cabc9d49SHoratiu Vultur 	bond_mask = lan966x_lag_get_mask(lan966x, lag);
343cabc9d49SHoratiu Vultur 	if (bond_mask && port->chip_port == __ffs(bond_mask))
344cabc9d49SHoratiu Vultur 		return true;
345cabc9d49SHoratiu Vultur 
346cabc9d49SHoratiu Vultur 	return false;
347cabc9d49SHoratiu Vultur }
348cabc9d49SHoratiu Vultur 
lan966x_lag_get_mask(struct lan966x * lan966x,struct net_device * bond)349cabc9d49SHoratiu Vultur u32 lan966x_lag_get_mask(struct lan966x *lan966x, struct net_device *bond)
350cabc9d49SHoratiu Vultur {
351cabc9d49SHoratiu Vultur 	struct lan966x_port *port;
352cabc9d49SHoratiu Vultur 	u32 mask = 0;
353cabc9d49SHoratiu Vultur 	int p;
354cabc9d49SHoratiu Vultur 
355cabc9d49SHoratiu Vultur 	if (!bond)
356cabc9d49SHoratiu Vultur 		return mask;
357cabc9d49SHoratiu Vultur 
358cabc9d49SHoratiu Vultur 	for (p = 0; p < lan966x->num_phys_ports; p++) {
359cabc9d49SHoratiu Vultur 		port = lan966x->ports[p];
360cabc9d49SHoratiu Vultur 		if (!port)
361cabc9d49SHoratiu Vultur 			continue;
362cabc9d49SHoratiu Vultur 
363cabc9d49SHoratiu Vultur 		if (port->bond == bond)
364cabc9d49SHoratiu Vultur 			mask |= BIT(p);
365cabc9d49SHoratiu Vultur 	}
366cabc9d49SHoratiu Vultur 
367cabc9d49SHoratiu Vultur 	return mask;
368cabc9d49SHoratiu Vultur }
369