xref: /openbmc/linux/drivers/net/ethernet/microchip/lan966x/lan966x_mac.c (revision 4f2c0a4acffbec01079c28f839422e64ddeff004)
1e18aba89SHoratiu Vultur // SPDX-License-Identifier: GPL-2.0+
2e18aba89SHoratiu Vultur 
35ccd66e0SHoratiu Vultur #include <net/switchdev.h>
4e18aba89SHoratiu Vultur #include "lan966x_main.h"
5e18aba89SHoratiu Vultur 
6e18aba89SHoratiu Vultur #define LAN966X_MAC_COLUMNS		4
7e18aba89SHoratiu Vultur #define MACACCESS_CMD_IDLE		0
8e18aba89SHoratiu Vultur #define MACACCESS_CMD_LEARN		1
9e18aba89SHoratiu Vultur #define MACACCESS_CMD_FORGET		2
10e18aba89SHoratiu Vultur #define MACACCESS_CMD_AGE		3
11e18aba89SHoratiu Vultur #define MACACCESS_CMD_GET_NEXT		4
12e18aba89SHoratiu Vultur #define MACACCESS_CMD_INIT		5
13e18aba89SHoratiu Vultur #define MACACCESS_CMD_READ		6
14e18aba89SHoratiu Vultur #define MACACCESS_CMD_WRITE		7
15e18aba89SHoratiu Vultur #define MACACCESS_CMD_SYNC_GET_NEXT	8
16e18aba89SHoratiu Vultur 
175ccd66e0SHoratiu Vultur #define LAN966X_MAC_INVALID_ROW		-1
185ccd66e0SHoratiu Vultur 
195ccd66e0SHoratiu Vultur struct lan966x_mac_entry {
205ccd66e0SHoratiu Vultur 	struct list_head list;
215ccd66e0SHoratiu Vultur 	unsigned char mac[ETH_ALEN] __aligned(2);
225ccd66e0SHoratiu Vultur 	u16 vid;
235ccd66e0SHoratiu Vultur 	u16 port_index;
245ccd66e0SHoratiu Vultur 	int row;
25*e09ce977SHoratiu Vultur 	bool lag;
265ccd66e0SHoratiu Vultur };
275ccd66e0SHoratiu Vultur 
285ccd66e0SHoratiu Vultur struct lan966x_mac_raw_entry {
295ccd66e0SHoratiu Vultur 	u32 mach;
305ccd66e0SHoratiu Vultur 	u32 macl;
315ccd66e0SHoratiu Vultur 	u32 maca;
325ccd66e0SHoratiu Vultur 	bool processed;
335ccd66e0SHoratiu Vultur };
345ccd66e0SHoratiu Vultur 
lan966x_mac_get_status(struct lan966x * lan966x)35e18aba89SHoratiu Vultur static int lan966x_mac_get_status(struct lan966x *lan966x)
36e18aba89SHoratiu Vultur {
37e18aba89SHoratiu Vultur 	return lan_rd(lan966x, ANA_MACACCESS);
38e18aba89SHoratiu Vultur }
39e18aba89SHoratiu Vultur 
lan966x_mac_wait_for_completion(struct lan966x * lan966x)40e18aba89SHoratiu Vultur static int lan966x_mac_wait_for_completion(struct lan966x *lan966x)
41e18aba89SHoratiu Vultur {
42e18aba89SHoratiu Vultur 	u32 val;
43e18aba89SHoratiu Vultur 
4477bdaf39SHoratiu Vultur 	return readx_poll_timeout_atomic(lan966x_mac_get_status,
45e18aba89SHoratiu Vultur 					 lan966x, val,
46e18aba89SHoratiu Vultur 					 (ANA_MACACCESS_MAC_TABLE_CMD_GET(val)) ==
47e18aba89SHoratiu Vultur 					 MACACCESS_CMD_IDLE,
4877bdaf39SHoratiu Vultur 					 TABLE_UPDATE_SLEEP_US,
4977bdaf39SHoratiu Vultur 					 TABLE_UPDATE_TIMEOUT_US);
50e18aba89SHoratiu Vultur }
51e18aba89SHoratiu Vultur 
lan966x_mac_select(struct lan966x * lan966x,const unsigned char mac[ETH_ALEN],unsigned int vid)52e18aba89SHoratiu Vultur static void lan966x_mac_select(struct lan966x *lan966x,
53e18aba89SHoratiu Vultur 			       const unsigned char mac[ETH_ALEN],
54e18aba89SHoratiu Vultur 			       unsigned int vid)
55e18aba89SHoratiu Vultur {
56e18aba89SHoratiu Vultur 	u32 macl = 0, mach = 0;
57e18aba89SHoratiu Vultur 
58e18aba89SHoratiu Vultur 	/* Set the MAC address to handle and the vlan associated in a format
59e18aba89SHoratiu Vultur 	 * understood by the hardware.
60e18aba89SHoratiu Vultur 	 */
61e18aba89SHoratiu Vultur 	mach |= vid    << 16;
62e18aba89SHoratiu Vultur 	mach |= mac[0] << 8;
63e18aba89SHoratiu Vultur 	mach |= mac[1] << 0;
64e18aba89SHoratiu Vultur 	macl |= mac[2] << 24;
65e18aba89SHoratiu Vultur 	macl |= mac[3] << 16;
66e18aba89SHoratiu Vultur 	macl |= mac[4] << 8;
67e18aba89SHoratiu Vultur 	macl |= mac[5] << 0;
68e18aba89SHoratiu Vultur 
69e18aba89SHoratiu Vultur 	lan_wr(macl, lan966x, ANA_MACLDATA);
70e18aba89SHoratiu Vultur 	lan_wr(mach, lan966x, ANA_MACHDATA);
71e18aba89SHoratiu Vultur }
72e18aba89SHoratiu Vultur 
__lan966x_mac_learn_locked(struct lan966x * lan966x,int pgid,bool cpu_copy,const unsigned char mac[ETH_ALEN],unsigned int vid,enum macaccess_entry_type type)73*e09ce977SHoratiu Vultur static int __lan966x_mac_learn_locked(struct lan966x *lan966x, int pgid,
74fc0c3fe7SHoratiu Vultur 				      bool cpu_copy,
75e18aba89SHoratiu Vultur 				      const unsigned char mac[ETH_ALEN],
76e18aba89SHoratiu Vultur 				      unsigned int vid,
77e18aba89SHoratiu Vultur 				      enum macaccess_entry_type type)
78e18aba89SHoratiu Vultur {
79*e09ce977SHoratiu Vultur 	lockdep_assert_held(&lan966x->mac_lock);
8043243bb3SHoratiu Vultur 
81e18aba89SHoratiu Vultur 	lan966x_mac_select(lan966x, mac, vid);
82e18aba89SHoratiu Vultur 
83e18aba89SHoratiu Vultur 	/* Issue a write command */
84e18aba89SHoratiu Vultur 	lan_wr(ANA_MACACCESS_VALID_SET(1) |
85e18aba89SHoratiu Vultur 	       ANA_MACACCESS_CHANGE2SW_SET(0) |
86fc0c3fe7SHoratiu Vultur 	       ANA_MACACCESS_MAC_CPU_COPY_SET(cpu_copy) |
87fc0c3fe7SHoratiu Vultur 	       ANA_MACACCESS_DEST_IDX_SET(pgid) |
88e18aba89SHoratiu Vultur 	       ANA_MACACCESS_ENTRYTYPE_SET(type) |
89e18aba89SHoratiu Vultur 	       ANA_MACACCESS_MAC_TABLE_CMD_SET(MACACCESS_CMD_LEARN),
90e18aba89SHoratiu Vultur 	       lan966x, ANA_MACACCESS);
91e18aba89SHoratiu Vultur 
92*e09ce977SHoratiu Vultur 	return lan966x_mac_wait_for_completion(lan966x);
93*e09ce977SHoratiu Vultur }
94*e09ce977SHoratiu Vultur 
__lan966x_mac_learn(struct lan966x * lan966x,int pgid,bool cpu_copy,const unsigned char mac[ETH_ALEN],unsigned int vid,enum macaccess_entry_type type)95*e09ce977SHoratiu Vultur static int __lan966x_mac_learn(struct lan966x *lan966x, int pgid,
96*e09ce977SHoratiu Vultur 			       bool cpu_copy,
97*e09ce977SHoratiu Vultur 			       const unsigned char mac[ETH_ALEN],
98*e09ce977SHoratiu Vultur 			       unsigned int vid,
99*e09ce977SHoratiu Vultur 			       enum macaccess_entry_type type)
100*e09ce977SHoratiu Vultur {
101*e09ce977SHoratiu Vultur 	int ret;
102*e09ce977SHoratiu Vultur 
103*e09ce977SHoratiu Vultur 	spin_lock(&lan966x->mac_lock);
104*e09ce977SHoratiu Vultur 	ret = __lan966x_mac_learn_locked(lan966x, pgid, cpu_copy, mac, vid, type);
10543243bb3SHoratiu Vultur 	spin_unlock(&lan966x->mac_lock);
10643243bb3SHoratiu Vultur 
10743243bb3SHoratiu Vultur 	return ret;
108e18aba89SHoratiu Vultur }
109e18aba89SHoratiu Vultur 
110fc0c3fe7SHoratiu Vultur /* The mask of the front ports is encoded inside the mac parameter via a call
111fc0c3fe7SHoratiu Vultur  * to lan966x_mdb_encode_mac().
112fc0c3fe7SHoratiu Vultur  */
lan966x_mac_ip_learn(struct lan966x * lan966x,bool cpu_copy,const unsigned char mac[ETH_ALEN],unsigned int vid,enum macaccess_entry_type type)113fc0c3fe7SHoratiu Vultur int lan966x_mac_ip_learn(struct lan966x *lan966x,
114fc0c3fe7SHoratiu Vultur 			 bool cpu_copy,
115fc0c3fe7SHoratiu Vultur 			 const unsigned char mac[ETH_ALEN],
116fc0c3fe7SHoratiu Vultur 			 unsigned int vid,
117fc0c3fe7SHoratiu Vultur 			 enum macaccess_entry_type type)
118fc0c3fe7SHoratiu Vultur {
119fc0c3fe7SHoratiu Vultur 	WARN_ON(type != ENTRYTYPE_MACV4 && type != ENTRYTYPE_MACV6);
120fc0c3fe7SHoratiu Vultur 
121fc0c3fe7SHoratiu Vultur 	return __lan966x_mac_learn(lan966x, 0, cpu_copy, mac, vid, type);
122fc0c3fe7SHoratiu Vultur }
123fc0c3fe7SHoratiu Vultur 
lan966x_mac_learn(struct lan966x * lan966x,int port,const unsigned char mac[ETH_ALEN],unsigned int vid,enum macaccess_entry_type type)124fc0c3fe7SHoratiu Vultur int lan966x_mac_learn(struct lan966x *lan966x, int port,
125fc0c3fe7SHoratiu Vultur 		      const unsigned char mac[ETH_ALEN],
126fc0c3fe7SHoratiu Vultur 		      unsigned int vid,
127fc0c3fe7SHoratiu Vultur 		      enum macaccess_entry_type type)
128fc0c3fe7SHoratiu Vultur {
129fc0c3fe7SHoratiu Vultur 	WARN_ON(type != ENTRYTYPE_NORMAL && type != ENTRYTYPE_LOCKED);
130fc0c3fe7SHoratiu Vultur 
131fc0c3fe7SHoratiu Vultur 	return __lan966x_mac_learn(lan966x, port, false, mac, vid, type);
132fc0c3fe7SHoratiu Vultur }
133fc0c3fe7SHoratiu Vultur 
lan966x_mac_learn_locked(struct lan966x * lan966x,int port,const unsigned char mac[ETH_ALEN],unsigned int vid,enum macaccess_entry_type type)134*e09ce977SHoratiu Vultur static int lan966x_mac_learn_locked(struct lan966x *lan966x, int port,
135*e09ce977SHoratiu Vultur 				    const unsigned char mac[ETH_ALEN],
136*e09ce977SHoratiu Vultur 				    unsigned int vid,
137*e09ce977SHoratiu Vultur 				    enum macaccess_entry_type type)
138*e09ce977SHoratiu Vultur {
139*e09ce977SHoratiu Vultur 	WARN_ON(type != ENTRYTYPE_NORMAL && type != ENTRYTYPE_LOCKED);
140*e09ce977SHoratiu Vultur 
141*e09ce977SHoratiu Vultur 	return __lan966x_mac_learn_locked(lan966x, port, false, mac, vid, type);
142*e09ce977SHoratiu Vultur }
143*e09ce977SHoratiu Vultur 
lan966x_mac_forget_locked(struct lan966x * lan966x,const unsigned char mac[ETH_ALEN],unsigned int vid,enum macaccess_entry_type type)14499343cfaSHoratiu Vultur static int lan966x_mac_forget_locked(struct lan966x *lan966x,
145e18aba89SHoratiu Vultur 				     const unsigned char mac[ETH_ALEN],
146e18aba89SHoratiu Vultur 				     unsigned int vid,
147e18aba89SHoratiu Vultur 				     enum macaccess_entry_type type)
148e18aba89SHoratiu Vultur {
14999343cfaSHoratiu Vultur 	lockdep_assert_held(&lan966x->mac_lock);
15099343cfaSHoratiu Vultur 
151e18aba89SHoratiu Vultur 	lan966x_mac_select(lan966x, mac, vid);
152e18aba89SHoratiu Vultur 
153e18aba89SHoratiu Vultur 	/* Issue a forget command */
154e18aba89SHoratiu Vultur 	lan_wr(ANA_MACACCESS_ENTRYTYPE_SET(type) |
155e18aba89SHoratiu Vultur 	       ANA_MACACCESS_MAC_TABLE_CMD_SET(MACACCESS_CMD_FORGET),
156e18aba89SHoratiu Vultur 	       lan966x, ANA_MACACCESS);
157e18aba89SHoratiu Vultur 
158e18aba89SHoratiu Vultur 	return lan966x_mac_wait_for_completion(lan966x);
159e18aba89SHoratiu Vultur }
160e18aba89SHoratiu Vultur 
lan966x_mac_forget(struct lan966x * lan966x,const unsigned char mac[ETH_ALEN],unsigned int vid,enum macaccess_entry_type type)16199343cfaSHoratiu Vultur int lan966x_mac_forget(struct lan966x *lan966x,
16299343cfaSHoratiu Vultur 		       const unsigned char mac[ETH_ALEN],
16399343cfaSHoratiu Vultur 		       unsigned int vid,
16499343cfaSHoratiu Vultur 		       enum macaccess_entry_type type)
16599343cfaSHoratiu Vultur {
16699343cfaSHoratiu Vultur 	int ret;
16799343cfaSHoratiu Vultur 
16899343cfaSHoratiu Vultur 	spin_lock(&lan966x->mac_lock);
16999343cfaSHoratiu Vultur 	ret = lan966x_mac_forget_locked(lan966x, mac, vid, type);
17099343cfaSHoratiu Vultur 	spin_unlock(&lan966x->mac_lock);
17199343cfaSHoratiu Vultur 
17299343cfaSHoratiu Vultur 	return ret;
17399343cfaSHoratiu Vultur }
17499343cfaSHoratiu Vultur 
lan966x_mac_cpu_learn(struct lan966x * lan966x,const char * addr,u16 vid)175e18aba89SHoratiu Vultur int lan966x_mac_cpu_learn(struct lan966x *lan966x, const char *addr, u16 vid)
176e18aba89SHoratiu Vultur {
177e18aba89SHoratiu Vultur 	return lan966x_mac_learn(lan966x, PGID_CPU, addr, vid, ENTRYTYPE_LOCKED);
178e18aba89SHoratiu Vultur }
179e18aba89SHoratiu Vultur 
lan966x_mac_cpu_forget(struct lan966x * lan966x,const char * addr,u16 vid)180e18aba89SHoratiu Vultur int lan966x_mac_cpu_forget(struct lan966x *lan966x, const char *addr, u16 vid)
181e18aba89SHoratiu Vultur {
182e18aba89SHoratiu Vultur 	return lan966x_mac_forget(lan966x, addr, vid, ENTRYTYPE_LOCKED);
183e18aba89SHoratiu Vultur }
184e18aba89SHoratiu Vultur 
lan966x_mac_set_ageing(struct lan966x * lan966x,u32 ageing)18525ee9561SHoratiu Vultur void lan966x_mac_set_ageing(struct lan966x *lan966x,
18625ee9561SHoratiu Vultur 			    u32 ageing)
18725ee9561SHoratiu Vultur {
18825ee9561SHoratiu Vultur 	lan_rmw(ANA_AUTOAGE_AGE_PERIOD_SET(ageing / 2),
18925ee9561SHoratiu Vultur 		ANA_AUTOAGE_AGE_PERIOD,
19025ee9561SHoratiu Vultur 		lan966x, ANA_AUTOAGE);
19125ee9561SHoratiu Vultur }
19225ee9561SHoratiu Vultur 
lan966x_mac_init(struct lan966x * lan966x)193e18aba89SHoratiu Vultur void lan966x_mac_init(struct lan966x *lan966x)
194e18aba89SHoratiu Vultur {
195e18aba89SHoratiu Vultur 	/* Clear the MAC table */
196e18aba89SHoratiu Vultur 	lan_wr(MACACCESS_CMD_INIT, lan966x, ANA_MACACCESS);
197e18aba89SHoratiu Vultur 	lan966x_mac_wait_for_completion(lan966x);
1985ccd66e0SHoratiu Vultur 
1995ccd66e0SHoratiu Vultur 	spin_lock_init(&lan966x->mac_lock);
2005ccd66e0SHoratiu Vultur 	INIT_LIST_HEAD(&lan966x->mac_entries);
2015ccd66e0SHoratiu Vultur }
2025ccd66e0SHoratiu Vultur 
lan966x_mac_alloc_entry(struct lan966x_port * port,const unsigned char * mac,u16 vid)203*e09ce977SHoratiu Vultur static struct lan966x_mac_entry *lan966x_mac_alloc_entry(struct lan966x_port *port,
204*e09ce977SHoratiu Vultur 							 const unsigned char *mac,
205*e09ce977SHoratiu Vultur 							 u16 vid)
2065ccd66e0SHoratiu Vultur {
2075ccd66e0SHoratiu Vultur 	struct lan966x_mac_entry *mac_entry;
2085ccd66e0SHoratiu Vultur 
209c1924684SHoratiu Vultur 	mac_entry = kzalloc(sizeof(*mac_entry), GFP_ATOMIC);
2105ccd66e0SHoratiu Vultur 	if (!mac_entry)
2115ccd66e0SHoratiu Vultur 		return NULL;
2125ccd66e0SHoratiu Vultur 
2135ccd66e0SHoratiu Vultur 	memcpy(mac_entry->mac, mac, ETH_ALEN);
2145ccd66e0SHoratiu Vultur 	mac_entry->vid = vid;
215*e09ce977SHoratiu Vultur 	mac_entry->port_index = port->chip_port;
2165ccd66e0SHoratiu Vultur 	mac_entry->row = LAN966X_MAC_INVALID_ROW;
217*e09ce977SHoratiu Vultur 	mac_entry->lag = port->bond ? true : false;
2185ccd66e0SHoratiu Vultur 	return mac_entry;
2195ccd66e0SHoratiu Vultur }
2205ccd66e0SHoratiu Vultur 
lan966x_mac_find_entry(struct lan966x * lan966x,const unsigned char * mac,u16 vid,u16 port_index)22125ee9561SHoratiu Vultur static struct lan966x_mac_entry *lan966x_mac_find_entry(struct lan966x *lan966x,
22225ee9561SHoratiu Vultur 							const unsigned char *mac,
22325ee9561SHoratiu Vultur 							u16 vid, u16 port_index)
22425ee9561SHoratiu Vultur {
22525ee9561SHoratiu Vultur 	struct lan966x_mac_entry *res = NULL;
22625ee9561SHoratiu Vultur 	struct lan966x_mac_entry *mac_entry;
22725ee9561SHoratiu Vultur 
22825ee9561SHoratiu Vultur 	list_for_each_entry(mac_entry, &lan966x->mac_entries, list) {
22925ee9561SHoratiu Vultur 		if (mac_entry->vid == vid &&
23025ee9561SHoratiu Vultur 		    ether_addr_equal(mac, mac_entry->mac) &&
23125ee9561SHoratiu Vultur 		    mac_entry->port_index == port_index) {
23225ee9561SHoratiu Vultur 			res = mac_entry;
23325ee9561SHoratiu Vultur 			break;
23425ee9561SHoratiu Vultur 		}
23525ee9561SHoratiu Vultur 	}
23625ee9561SHoratiu Vultur 
23725ee9561SHoratiu Vultur 	return res;
23825ee9561SHoratiu Vultur }
23925ee9561SHoratiu Vultur 
lan966x_mac_lookup(struct lan966x * lan966x,const unsigned char mac[ETH_ALEN],unsigned int vid,enum macaccess_entry_type type)24025ee9561SHoratiu Vultur static int lan966x_mac_lookup(struct lan966x *lan966x,
24125ee9561SHoratiu Vultur 			      const unsigned char mac[ETH_ALEN],
24225ee9561SHoratiu Vultur 			      unsigned int vid, enum macaccess_entry_type type)
24325ee9561SHoratiu Vultur {
24425ee9561SHoratiu Vultur 	int ret;
24525ee9561SHoratiu Vultur 
24625ee9561SHoratiu Vultur 	lan966x_mac_select(lan966x, mac, vid);
24725ee9561SHoratiu Vultur 
24825ee9561SHoratiu Vultur 	/* Issue a read command */
24925ee9561SHoratiu Vultur 	lan_wr(ANA_MACACCESS_ENTRYTYPE_SET(type) |
25025ee9561SHoratiu Vultur 	       ANA_MACACCESS_VALID_SET(1) |
25125ee9561SHoratiu Vultur 	       ANA_MACACCESS_MAC_TABLE_CMD_SET(MACACCESS_CMD_READ),
25225ee9561SHoratiu Vultur 	       lan966x, ANA_MACACCESS);
25325ee9561SHoratiu Vultur 
25425ee9561SHoratiu Vultur 	ret = lan966x_mac_wait_for_completion(lan966x);
25525ee9561SHoratiu Vultur 	if (ret)
25625ee9561SHoratiu Vultur 		return ret;
25725ee9561SHoratiu Vultur 
25825ee9561SHoratiu Vultur 	return ANA_MACACCESS_VALID_GET(lan_rd(lan966x, ANA_MACACCESS));
25925ee9561SHoratiu Vultur }
26025ee9561SHoratiu Vultur 
lan966x_fdb_call_notifiers(enum switchdev_notifier_type type,const char * mac,u16 vid,struct net_device * dev)2615ccd66e0SHoratiu Vultur static void lan966x_fdb_call_notifiers(enum switchdev_notifier_type type,
2625ccd66e0SHoratiu Vultur 				       const char *mac, u16 vid,
2635ccd66e0SHoratiu Vultur 				       struct net_device *dev)
2645ccd66e0SHoratiu Vultur {
2655ccd66e0SHoratiu Vultur 	struct switchdev_notifier_fdb_info info = { 0 };
2665ccd66e0SHoratiu Vultur 
2675ccd66e0SHoratiu Vultur 	info.addr = mac;
2685ccd66e0SHoratiu Vultur 	info.vid = vid;
2695ccd66e0SHoratiu Vultur 	info.offloaded = true;
2705ccd66e0SHoratiu Vultur 	call_switchdev_notifiers(type, dev, &info.info, NULL);
2715ccd66e0SHoratiu Vultur }
2725ccd66e0SHoratiu Vultur 
lan966x_mac_add_entry(struct lan966x * lan966x,struct lan966x_port * port,const unsigned char * addr,u16 vid)27325ee9561SHoratiu Vultur int lan966x_mac_add_entry(struct lan966x *lan966x, struct lan966x_port *port,
27425ee9561SHoratiu Vultur 			  const unsigned char *addr, u16 vid)
27525ee9561SHoratiu Vultur {
27625ee9561SHoratiu Vultur 	struct lan966x_mac_entry *mac_entry;
27725ee9561SHoratiu Vultur 
278675c807aSHoratiu Vultur 	spin_lock(&lan966x->mac_lock);
279675c807aSHoratiu Vultur 	if (lan966x_mac_lookup(lan966x, addr, vid, ENTRYTYPE_NORMAL)) {
280675c807aSHoratiu Vultur 		spin_unlock(&lan966x->mac_lock);
28125ee9561SHoratiu Vultur 		return 0;
282675c807aSHoratiu Vultur 	}
28325ee9561SHoratiu Vultur 
28425ee9561SHoratiu Vultur 	/* In case the entry already exists, don't add it again to SW,
28525ee9561SHoratiu Vultur 	 * just update HW, but we need to look in the actual HW because
28625ee9561SHoratiu Vultur 	 * it is possible for an entry to be learn by HW and before we
28725ee9561SHoratiu Vultur 	 * get the interrupt the frame will reach CPU and the CPU will
28825ee9561SHoratiu Vultur 	 * add the entry but without the extern_learn flag.
28925ee9561SHoratiu Vultur 	 */
29025ee9561SHoratiu Vultur 	mac_entry = lan966x_mac_find_entry(lan966x, addr, vid, port->chip_port);
291675c807aSHoratiu Vultur 	if (mac_entry) {
292675c807aSHoratiu Vultur 		spin_unlock(&lan966x->mac_lock);
293675c807aSHoratiu Vultur 		goto mac_learn;
294675c807aSHoratiu Vultur 	}
29525ee9561SHoratiu Vultur 
296*e09ce977SHoratiu Vultur 	mac_entry = lan966x_mac_alloc_entry(port, addr, vid);
297675c807aSHoratiu Vultur 	if (!mac_entry) {
298675c807aSHoratiu Vultur 		spin_unlock(&lan966x->mac_lock);
29925ee9561SHoratiu Vultur 		return -ENOMEM;
300675c807aSHoratiu Vultur 	}
30125ee9561SHoratiu Vultur 
30225ee9561SHoratiu Vultur 	list_add_tail(&mac_entry->list, &lan966x->mac_entries);
30325ee9561SHoratiu Vultur 	spin_unlock(&lan966x->mac_lock);
30425ee9561SHoratiu Vultur 
305*e09ce977SHoratiu Vultur 	lan966x_fdb_call_notifiers(SWITCHDEV_FDB_OFFLOADED, addr, vid,
306*e09ce977SHoratiu Vultur 				   port->bond ?: port->dev);
30725ee9561SHoratiu Vultur 
308675c807aSHoratiu Vultur mac_learn:
309675c807aSHoratiu Vultur 	lan966x_mac_learn(lan966x, port->chip_port, addr, vid, ENTRYTYPE_LOCKED);
310675c807aSHoratiu Vultur 
31125ee9561SHoratiu Vultur 	return 0;
31225ee9561SHoratiu Vultur }
31325ee9561SHoratiu Vultur 
lan966x_mac_del_entry(struct lan966x * lan966x,const unsigned char * addr,u16 vid)31425ee9561SHoratiu Vultur int lan966x_mac_del_entry(struct lan966x *lan966x, const unsigned char *addr,
31525ee9561SHoratiu Vultur 			  u16 vid)
31625ee9561SHoratiu Vultur {
31725ee9561SHoratiu Vultur 	struct lan966x_mac_entry *mac_entry, *tmp;
31825ee9561SHoratiu Vultur 
31925ee9561SHoratiu Vultur 	spin_lock(&lan966x->mac_lock);
32025ee9561SHoratiu Vultur 	list_for_each_entry_safe(mac_entry, tmp, &lan966x->mac_entries,
32125ee9561SHoratiu Vultur 				 list) {
32225ee9561SHoratiu Vultur 		if (mac_entry->vid == vid &&
32325ee9561SHoratiu Vultur 		    ether_addr_equal(addr, mac_entry->mac)) {
324675c807aSHoratiu Vultur 			lan966x_mac_forget_locked(lan966x, mac_entry->mac,
325675c807aSHoratiu Vultur 						  mac_entry->vid,
32625ee9561SHoratiu Vultur 						  ENTRYTYPE_LOCKED);
32725ee9561SHoratiu Vultur 
32825ee9561SHoratiu Vultur 			list_del(&mac_entry->list);
32925ee9561SHoratiu Vultur 			kfree(mac_entry);
33025ee9561SHoratiu Vultur 		}
33125ee9561SHoratiu Vultur 	}
33225ee9561SHoratiu Vultur 	spin_unlock(&lan966x->mac_lock);
33325ee9561SHoratiu Vultur 
33425ee9561SHoratiu Vultur 	return 0;
33525ee9561SHoratiu Vultur }
33625ee9561SHoratiu Vultur 
lan966x_mac_lag_replace_port_entry(struct lan966x * lan966x,struct lan966x_port * src,struct lan966x_port * dst)337*e09ce977SHoratiu Vultur void lan966x_mac_lag_replace_port_entry(struct lan966x *lan966x,
338*e09ce977SHoratiu Vultur 					struct lan966x_port *src,
339*e09ce977SHoratiu Vultur 					struct lan966x_port *dst)
340*e09ce977SHoratiu Vultur {
341*e09ce977SHoratiu Vultur 	struct lan966x_mac_entry *mac_entry;
342*e09ce977SHoratiu Vultur 
343*e09ce977SHoratiu Vultur 	spin_lock(&lan966x->mac_lock);
344*e09ce977SHoratiu Vultur 	list_for_each_entry(mac_entry, &lan966x->mac_entries, list) {
345*e09ce977SHoratiu Vultur 		if (mac_entry->port_index == src->chip_port &&
346*e09ce977SHoratiu Vultur 		    mac_entry->lag) {
347*e09ce977SHoratiu Vultur 			lan966x_mac_forget_locked(lan966x, mac_entry->mac,
348*e09ce977SHoratiu Vultur 						  mac_entry->vid,
349*e09ce977SHoratiu Vultur 						  ENTRYTYPE_LOCKED);
350*e09ce977SHoratiu Vultur 
351*e09ce977SHoratiu Vultur 			lan966x_mac_learn_locked(lan966x, dst->chip_port,
352*e09ce977SHoratiu Vultur 						 mac_entry->mac, mac_entry->vid,
353*e09ce977SHoratiu Vultur 						 ENTRYTYPE_LOCKED);
354*e09ce977SHoratiu Vultur 			mac_entry->port_index = dst->chip_port;
355*e09ce977SHoratiu Vultur 		}
356*e09ce977SHoratiu Vultur 	}
357*e09ce977SHoratiu Vultur 	spin_unlock(&lan966x->mac_lock);
358*e09ce977SHoratiu Vultur }
359*e09ce977SHoratiu Vultur 
lan966x_mac_lag_remove_port_entry(struct lan966x * lan966x,struct lan966x_port * src)360*e09ce977SHoratiu Vultur void lan966x_mac_lag_remove_port_entry(struct lan966x *lan966x,
361*e09ce977SHoratiu Vultur 				       struct lan966x_port *src)
362*e09ce977SHoratiu Vultur {
363*e09ce977SHoratiu Vultur 	struct lan966x_mac_entry *mac_entry, *tmp;
364*e09ce977SHoratiu Vultur 
365*e09ce977SHoratiu Vultur 	spin_lock(&lan966x->mac_lock);
366*e09ce977SHoratiu Vultur 	list_for_each_entry_safe(mac_entry, tmp, &lan966x->mac_entries,
367*e09ce977SHoratiu Vultur 				 list) {
368*e09ce977SHoratiu Vultur 		if (mac_entry->port_index == src->chip_port &&
369*e09ce977SHoratiu Vultur 		    mac_entry->lag) {
370*e09ce977SHoratiu Vultur 			lan966x_mac_forget_locked(lan966x, mac_entry->mac,
371*e09ce977SHoratiu Vultur 						  mac_entry->vid,
372*e09ce977SHoratiu Vultur 						  ENTRYTYPE_LOCKED);
373*e09ce977SHoratiu Vultur 
374*e09ce977SHoratiu Vultur 			list_del(&mac_entry->list);
375*e09ce977SHoratiu Vultur 			kfree(mac_entry);
376*e09ce977SHoratiu Vultur 		}
377*e09ce977SHoratiu Vultur 	}
378*e09ce977SHoratiu Vultur 	spin_unlock(&lan966x->mac_lock);
379*e09ce977SHoratiu Vultur }
380*e09ce977SHoratiu Vultur 
lan966x_mac_purge_entries(struct lan966x * lan966x)3815ccd66e0SHoratiu Vultur void lan966x_mac_purge_entries(struct lan966x *lan966x)
3825ccd66e0SHoratiu Vultur {
3835ccd66e0SHoratiu Vultur 	struct lan966x_mac_entry *mac_entry, *tmp;
3845ccd66e0SHoratiu Vultur 
3855ccd66e0SHoratiu Vultur 	spin_lock(&lan966x->mac_lock);
3865ccd66e0SHoratiu Vultur 	list_for_each_entry_safe(mac_entry, tmp, &lan966x->mac_entries,
3875ccd66e0SHoratiu Vultur 				 list) {
388c1924684SHoratiu Vultur 		lan966x_mac_forget_locked(lan966x, mac_entry->mac,
389c1924684SHoratiu Vultur 					  mac_entry->vid, ENTRYTYPE_LOCKED);
3905ccd66e0SHoratiu Vultur 
3915ccd66e0SHoratiu Vultur 		list_del(&mac_entry->list);
3925ccd66e0SHoratiu Vultur 		kfree(mac_entry);
3935ccd66e0SHoratiu Vultur 	}
3945ccd66e0SHoratiu Vultur 	spin_unlock(&lan966x->mac_lock);
3955ccd66e0SHoratiu Vultur }
3965ccd66e0SHoratiu Vultur 
lan966x_mac_notifiers(enum switchdev_notifier_type type,unsigned char * mac,u32 vid,struct net_device * dev)3975ccd66e0SHoratiu Vultur static void lan966x_mac_notifiers(enum switchdev_notifier_type type,
3985ccd66e0SHoratiu Vultur 				  unsigned char *mac, u32 vid,
3995ccd66e0SHoratiu Vultur 				  struct net_device *dev)
4005ccd66e0SHoratiu Vultur {
4015ccd66e0SHoratiu Vultur 	rtnl_lock();
4025ccd66e0SHoratiu Vultur 	lan966x_fdb_call_notifiers(type, mac, vid, dev);
4035ccd66e0SHoratiu Vultur 	rtnl_unlock();
4045ccd66e0SHoratiu Vultur }
4055ccd66e0SHoratiu Vultur 
lan966x_mac_process_raw_entry(struct lan966x_mac_raw_entry * raw_entry,u8 * mac,u16 * vid,u32 * dest_idx)4065ccd66e0SHoratiu Vultur static void lan966x_mac_process_raw_entry(struct lan966x_mac_raw_entry *raw_entry,
4075ccd66e0SHoratiu Vultur 					  u8 *mac, u16 *vid, u32 *dest_idx)
4085ccd66e0SHoratiu Vultur {
4095ccd66e0SHoratiu Vultur 	mac[0] = (raw_entry->mach >> 8)  & 0xff;
4105ccd66e0SHoratiu Vultur 	mac[1] = (raw_entry->mach >> 0)  & 0xff;
4115ccd66e0SHoratiu Vultur 	mac[2] = (raw_entry->macl >> 24) & 0xff;
4125ccd66e0SHoratiu Vultur 	mac[3] = (raw_entry->macl >> 16) & 0xff;
4135ccd66e0SHoratiu Vultur 	mac[4] = (raw_entry->macl >> 8)  & 0xff;
4145ccd66e0SHoratiu Vultur 	mac[5] = (raw_entry->macl >> 0)  & 0xff;
4155ccd66e0SHoratiu Vultur 
4165ccd66e0SHoratiu Vultur 	*vid = (raw_entry->mach >> 16) & 0xfff;
4175ccd66e0SHoratiu Vultur 	*dest_idx = ANA_MACACCESS_DEST_IDX_GET(raw_entry->maca);
4185ccd66e0SHoratiu Vultur }
4195ccd66e0SHoratiu Vultur 
lan966x_mac_irq_process(struct lan966x * lan966x,u32 row,struct lan966x_mac_raw_entry * raw_entries)4205ccd66e0SHoratiu Vultur static void lan966x_mac_irq_process(struct lan966x *lan966x, u32 row,
4215ccd66e0SHoratiu Vultur 				    struct lan966x_mac_raw_entry *raw_entries)
4225ccd66e0SHoratiu Vultur {
4235ccd66e0SHoratiu Vultur 	struct lan966x_mac_entry *mac_entry, *tmp;
4245ccd66e0SHoratiu Vultur 	unsigned char mac[ETH_ALEN] __aligned(2);
42545533a53SHoratiu Vultur 	struct list_head mac_deleted_entries;
426*e09ce977SHoratiu Vultur 	struct lan966x_port *port;
4275ccd66e0SHoratiu Vultur 	u32 dest_idx;
4285ccd66e0SHoratiu Vultur 	u32 column;
4295ccd66e0SHoratiu Vultur 	u16 vid;
4305ccd66e0SHoratiu Vultur 
43145533a53SHoratiu Vultur 	INIT_LIST_HEAD(&mac_deleted_entries);
43245533a53SHoratiu Vultur 
4335ccd66e0SHoratiu Vultur 	spin_lock(&lan966x->mac_lock);
4345ccd66e0SHoratiu Vultur 	list_for_each_entry_safe(mac_entry, tmp, &lan966x->mac_entries, list) {
4355ccd66e0SHoratiu Vultur 		bool found = false;
4365ccd66e0SHoratiu Vultur 
4375ccd66e0SHoratiu Vultur 		if (mac_entry->row != row)
4385ccd66e0SHoratiu Vultur 			continue;
4395ccd66e0SHoratiu Vultur 
4405ccd66e0SHoratiu Vultur 		for (column = 0; column < LAN966X_MAC_COLUMNS; ++column) {
4415ccd66e0SHoratiu Vultur 			/* All the valid entries are at the start of the row,
4425ccd66e0SHoratiu Vultur 			 * so when get one invalid entry it can just skip the
4435ccd66e0SHoratiu Vultur 			 * rest of the columns
4445ccd66e0SHoratiu Vultur 			 */
4455ccd66e0SHoratiu Vultur 			if (!ANA_MACACCESS_VALID_GET(raw_entries[column].maca))
4465ccd66e0SHoratiu Vultur 				break;
4475ccd66e0SHoratiu Vultur 
4485ccd66e0SHoratiu Vultur 			lan966x_mac_process_raw_entry(&raw_entries[column],
4495ccd66e0SHoratiu Vultur 						      mac, &vid, &dest_idx);
4509810c58cSDan Carpenter 			if (WARN_ON(dest_idx >= lan966x->num_phys_ports))
45126921932SHoratiu Vultur 				continue;
4525ccd66e0SHoratiu Vultur 
4535ccd66e0SHoratiu Vultur 			/* If the entry in SW is found, then there is nothing
4545ccd66e0SHoratiu Vultur 			 * to do
4555ccd66e0SHoratiu Vultur 			 */
4565ccd66e0SHoratiu Vultur 			if (mac_entry->vid == vid &&
4575ccd66e0SHoratiu Vultur 			    ether_addr_equal(mac_entry->mac, mac) &&
4585ccd66e0SHoratiu Vultur 			    mac_entry->port_index == dest_idx) {
4595ccd66e0SHoratiu Vultur 				raw_entries[column].processed = true;
4605ccd66e0SHoratiu Vultur 				found = true;
4615ccd66e0SHoratiu Vultur 				break;
4625ccd66e0SHoratiu Vultur 			}
4635ccd66e0SHoratiu Vultur 		}
4645ccd66e0SHoratiu Vultur 
4655ccd66e0SHoratiu Vultur 		if (!found) {
46645533a53SHoratiu Vultur 			list_del(&mac_entry->list);
46745533a53SHoratiu Vultur 			/* Move the entry from SW list to a tmp list such that
46845533a53SHoratiu Vultur 			 * it would be deleted later
46945533a53SHoratiu Vultur 			 */
47045533a53SHoratiu Vultur 			list_add_tail(&mac_entry->list, &mac_deleted_entries);
47145533a53SHoratiu Vultur 		}
47245533a53SHoratiu Vultur 	}
47345533a53SHoratiu Vultur 	spin_unlock(&lan966x->mac_lock);
47445533a53SHoratiu Vultur 
47545533a53SHoratiu Vultur 	list_for_each_entry_safe(mac_entry, tmp, &mac_deleted_entries, list) {
4765ccd66e0SHoratiu Vultur 		/* Notify the bridge that the entry doesn't exist
47745533a53SHoratiu Vultur 		 * anymore in the HW
4785ccd66e0SHoratiu Vultur 		 */
479*e09ce977SHoratiu Vultur 		port = lan966x->ports[mac_entry->port_index];
4805ccd66e0SHoratiu Vultur 		lan966x_mac_notifiers(SWITCHDEV_FDB_DEL_TO_BRIDGE,
4815ccd66e0SHoratiu Vultur 				      mac_entry->mac, mac_entry->vid,
482*e09ce977SHoratiu Vultur 				      port->bond ?: port->dev);
4835ccd66e0SHoratiu Vultur 		list_del(&mac_entry->list);
4845ccd66e0SHoratiu Vultur 		kfree(mac_entry);
4855ccd66e0SHoratiu Vultur 	}
4865ccd66e0SHoratiu Vultur 
4875ccd66e0SHoratiu Vultur 	/* Now go to the list of columns and see if any entry was not in the SW
4885ccd66e0SHoratiu Vultur 	 * list, then that means that the entry is new so it needs to notify the
4895ccd66e0SHoratiu Vultur 	 * bridge.
4905ccd66e0SHoratiu Vultur 	 */
4915ccd66e0SHoratiu Vultur 	for (column = 0; column < LAN966X_MAC_COLUMNS; ++column) {
4925ccd66e0SHoratiu Vultur 		/* All the valid entries are at the start of the row, so when
4935ccd66e0SHoratiu Vultur 		 * get one invalid entry it can just skip the rest of the columns
4945ccd66e0SHoratiu Vultur 		 */
4955ccd66e0SHoratiu Vultur 		if (!ANA_MACACCESS_VALID_GET(raw_entries[column].maca))
4965ccd66e0SHoratiu Vultur 			break;
4975ccd66e0SHoratiu Vultur 
4985ccd66e0SHoratiu Vultur 		/* If the entry already exists then don't do anything */
4995ccd66e0SHoratiu Vultur 		if (raw_entries[column].processed)
5005ccd66e0SHoratiu Vultur 			continue;
5015ccd66e0SHoratiu Vultur 
5025ccd66e0SHoratiu Vultur 		lan966x_mac_process_raw_entry(&raw_entries[column],
5035ccd66e0SHoratiu Vultur 					      mac, &vid, &dest_idx);
5049810c58cSDan Carpenter 		if (WARN_ON(dest_idx >= lan966x->num_phys_ports))
50526921932SHoratiu Vultur 			continue;
5065ccd66e0SHoratiu Vultur 
507c1924684SHoratiu Vultur 		spin_lock(&lan966x->mac_lock);
508675c807aSHoratiu Vultur 		mac_entry = lan966x_mac_find_entry(lan966x, mac, vid, dest_idx);
509675c807aSHoratiu Vultur 		if (mac_entry) {
510675c807aSHoratiu Vultur 			spin_unlock(&lan966x->mac_lock);
511675c807aSHoratiu Vultur 			continue;
512675c807aSHoratiu Vultur 		}
513675c807aSHoratiu Vultur 
514*e09ce977SHoratiu Vultur 		port = lan966x->ports[dest_idx];
515*e09ce977SHoratiu Vultur 		mac_entry = lan966x_mac_alloc_entry(port, mac, vid);
516c1924684SHoratiu Vultur 		if (!mac_entry) {
517c1924684SHoratiu Vultur 			spin_unlock(&lan966x->mac_lock);
5185ccd66e0SHoratiu Vultur 			return;
519c1924684SHoratiu Vultur 		}
5205ccd66e0SHoratiu Vultur 
5215ccd66e0SHoratiu Vultur 		mac_entry->row = row;
5225ccd66e0SHoratiu Vultur 		list_add_tail(&mac_entry->list, &lan966x->mac_entries);
5235ccd66e0SHoratiu Vultur 		spin_unlock(&lan966x->mac_lock);
5245ccd66e0SHoratiu Vultur 
5255ccd66e0SHoratiu Vultur 		lan966x_mac_notifiers(SWITCHDEV_FDB_ADD_TO_BRIDGE,
526*e09ce977SHoratiu Vultur 				      mac, vid, port->bond ?: port->dev);
5275ccd66e0SHoratiu Vultur 	}
5285ccd66e0SHoratiu Vultur }
5295ccd66e0SHoratiu Vultur 
lan966x_mac_irq_handler(struct lan966x * lan966x)5305ccd66e0SHoratiu Vultur irqreturn_t lan966x_mac_irq_handler(struct lan966x *lan966x)
5315ccd66e0SHoratiu Vultur {
5325ccd66e0SHoratiu Vultur 	struct lan966x_mac_raw_entry entry[LAN966X_MAC_COLUMNS] = { 0 };
5335ccd66e0SHoratiu Vultur 	u32 index, column;
5345ccd66e0SHoratiu Vultur 	bool stop = true;
5355ccd66e0SHoratiu Vultur 	u32 val;
5365ccd66e0SHoratiu Vultur 
5375ccd66e0SHoratiu Vultur 	/* Start the scan from 0, 0 */
5385ccd66e0SHoratiu Vultur 	lan_wr(ANA_MACTINDX_M_INDEX_SET(0) |
5395ccd66e0SHoratiu Vultur 	       ANA_MACTINDX_BUCKET_SET(0),
5405ccd66e0SHoratiu Vultur 	       lan966x, ANA_MACTINDX);
5415ccd66e0SHoratiu Vultur 
5425ccd66e0SHoratiu Vultur 	while (1) {
543c1924684SHoratiu Vultur 		spin_lock(&lan966x->mac_lock);
5445ccd66e0SHoratiu Vultur 		lan_rmw(ANA_MACACCESS_MAC_TABLE_CMD_SET(MACACCESS_CMD_SYNC_GET_NEXT),
5455ccd66e0SHoratiu Vultur 			ANA_MACACCESS_MAC_TABLE_CMD,
5465ccd66e0SHoratiu Vultur 			lan966x, ANA_MACACCESS);
5475ccd66e0SHoratiu Vultur 		lan966x_mac_wait_for_completion(lan966x);
5485ccd66e0SHoratiu Vultur 
5495ccd66e0SHoratiu Vultur 		val = lan_rd(lan966x, ANA_MACTINDX);
5505ccd66e0SHoratiu Vultur 		index = ANA_MACTINDX_M_INDEX_GET(val);
5515ccd66e0SHoratiu Vultur 		column = ANA_MACTINDX_BUCKET_GET(val);
5525ccd66e0SHoratiu Vultur 
5535ccd66e0SHoratiu Vultur 		/* The SYNC-GET-NEXT returns all the entries(4) in a row in
5545ccd66e0SHoratiu Vultur 		 * which is suffered a change. By change it means that new entry
5555ccd66e0SHoratiu Vultur 		 * was added or an entry was removed because of ageing.
5565ccd66e0SHoratiu Vultur 		 * It would return all the columns for that row. And after that
5575ccd66e0SHoratiu Vultur 		 * it would return the next row The stop conditions of the
5585ccd66e0SHoratiu Vultur 		 * SYNC-GET-NEXT is when it reaches 'directly' to row 0
5595ccd66e0SHoratiu Vultur 		 * column 3. So if SYNC-GET-NEXT returns row 0 and column 0
5605ccd66e0SHoratiu Vultur 		 * then it is required to continue to read more even if it
5615ccd66e0SHoratiu Vultur 		 * reaches row 0 and column 3.
5625ccd66e0SHoratiu Vultur 		 */
5635ccd66e0SHoratiu Vultur 		if (index == 0 && column == 0)
5645ccd66e0SHoratiu Vultur 			stop = false;
5655ccd66e0SHoratiu Vultur 
5665ccd66e0SHoratiu Vultur 		if (column == LAN966X_MAC_COLUMNS - 1 &&
567c1924684SHoratiu Vultur 		    index == 0 && stop) {
568c1924684SHoratiu Vultur 			spin_unlock(&lan966x->mac_lock);
5695ccd66e0SHoratiu Vultur 			break;
570c1924684SHoratiu Vultur 		}
5715ccd66e0SHoratiu Vultur 
5725ccd66e0SHoratiu Vultur 		entry[column].mach = lan_rd(lan966x, ANA_MACHDATA);
5735ccd66e0SHoratiu Vultur 		entry[column].macl = lan_rd(lan966x, ANA_MACLDATA);
5745ccd66e0SHoratiu Vultur 		entry[column].maca = lan_rd(lan966x, ANA_MACACCESS);
575c1924684SHoratiu Vultur 		spin_unlock(&lan966x->mac_lock);
5765ccd66e0SHoratiu Vultur 
5775ccd66e0SHoratiu Vultur 		/* Once all the columns are read process them */
5785ccd66e0SHoratiu Vultur 		if (column == LAN966X_MAC_COLUMNS - 1) {
5795ccd66e0SHoratiu Vultur 			lan966x_mac_irq_process(lan966x, index, entry);
5805ccd66e0SHoratiu Vultur 			/* A row was processed so it is safe to assume that the
5815ccd66e0SHoratiu Vultur 			 * next row/column can be the stop condition
5825ccd66e0SHoratiu Vultur 			 */
5835ccd66e0SHoratiu Vultur 			stop = true;
5845ccd66e0SHoratiu Vultur 		}
5855ccd66e0SHoratiu Vultur 	}
5865ccd66e0SHoratiu Vultur 
5875ccd66e0SHoratiu Vultur 	lan_rmw(ANA_ANAINTR_INTR_SET(0),
5885ccd66e0SHoratiu Vultur 		ANA_ANAINTR_INTR,
5895ccd66e0SHoratiu Vultur 		lan966x, ANA_ANAINTR);
5905ccd66e0SHoratiu Vultur 
5915ccd66e0SHoratiu Vultur 	return IRQ_HANDLED;
592e18aba89SHoratiu Vultur }
593