1 // SPDX-License-Identifier: GPL-2.0
2 /*
3  * Broadcom STB ASP 2.0 Driver
4  *
5  * Copyright (c) 2023 Broadcom
6  */
7 #include <linux/etherdevice.h>
8 #include <linux/if_vlan.h>
9 #include <linux/init.h>
10 #include <linux/interrupt.h>
11 #include <linux/module.h>
12 #include <linux/kernel.h>
13 #include <linux/platform_device.h>
14 #include <linux/of.h>
15 #include <linux/of_address.h>
16 #include <linux/of_platform.h>
17 #include <linux/clk.h>
18 
19 #include "bcmasp.h"
20 #include "bcmasp_intf_defs.h"
21 
_intr2_mask_clear(struct bcmasp_priv * priv,u32 mask)22 static void _intr2_mask_clear(struct bcmasp_priv *priv, u32 mask)
23 {
24 	intr2_core_wl(priv, mask, ASP_INTR2_MASK_CLEAR);
25 	priv->irq_mask &= ~mask;
26 }
27 
_intr2_mask_set(struct bcmasp_priv * priv,u32 mask)28 static void _intr2_mask_set(struct bcmasp_priv *priv, u32 mask)
29 {
30 	intr2_core_wl(priv, mask, ASP_INTR2_MASK_SET);
31 	priv->irq_mask |= mask;
32 }
33 
bcmasp_enable_phy_irq(struct bcmasp_intf * intf,int en)34 void bcmasp_enable_phy_irq(struct bcmasp_intf *intf, int en)
35 {
36 	struct bcmasp_priv *priv = intf->parent;
37 
38 	/* Only supported with internal phys */
39 	if (!intf->internal_phy)
40 		return;
41 
42 	if (en)
43 		_intr2_mask_clear(priv, ASP_INTR2_PHY_EVENT(intf->channel));
44 	else
45 		_intr2_mask_set(priv, ASP_INTR2_PHY_EVENT(intf->channel));
46 }
47 
bcmasp_enable_tx_irq(struct bcmasp_intf * intf,int en)48 void bcmasp_enable_tx_irq(struct bcmasp_intf *intf, int en)
49 {
50 	struct bcmasp_priv *priv = intf->parent;
51 
52 	if (en)
53 		_intr2_mask_clear(priv, ASP_INTR2_TX_DESC(intf->channel));
54 	else
55 		_intr2_mask_set(priv, ASP_INTR2_TX_DESC(intf->channel));
56 }
57 EXPORT_SYMBOL_GPL(bcmasp_enable_tx_irq);
58 
bcmasp_enable_rx_irq(struct bcmasp_intf * intf,int en)59 void bcmasp_enable_rx_irq(struct bcmasp_intf *intf, int en)
60 {
61 	struct bcmasp_priv *priv = intf->parent;
62 
63 	if (en)
64 		_intr2_mask_clear(priv, ASP_INTR2_RX_ECH(intf->channel));
65 	else
66 		_intr2_mask_set(priv, ASP_INTR2_RX_ECH(intf->channel));
67 }
68 EXPORT_SYMBOL_GPL(bcmasp_enable_rx_irq);
69 
bcmasp_intr2_mask_set_all(struct bcmasp_priv * priv)70 static void bcmasp_intr2_mask_set_all(struct bcmasp_priv *priv)
71 {
72 	_intr2_mask_set(priv, 0xffffffff);
73 	priv->irq_mask = 0xffffffff;
74 }
75 
bcmasp_intr2_clear_all(struct bcmasp_priv * priv)76 static void bcmasp_intr2_clear_all(struct bcmasp_priv *priv)
77 {
78 	intr2_core_wl(priv, 0xffffffff, ASP_INTR2_CLEAR);
79 }
80 
bcmasp_intr2_handling(struct bcmasp_intf * intf,u32 status)81 static void bcmasp_intr2_handling(struct bcmasp_intf *intf, u32 status)
82 {
83 	if (status & ASP_INTR2_RX_ECH(intf->channel)) {
84 		if (likely(napi_schedule_prep(&intf->rx_napi))) {
85 			bcmasp_enable_rx_irq(intf, 0);
86 			__napi_schedule_irqoff(&intf->rx_napi);
87 		}
88 	}
89 
90 	if (status & ASP_INTR2_TX_DESC(intf->channel)) {
91 		if (likely(napi_schedule_prep(&intf->tx_napi))) {
92 			bcmasp_enable_tx_irq(intf, 0);
93 			__napi_schedule_irqoff(&intf->tx_napi);
94 		}
95 	}
96 
97 	if (status & ASP_INTR2_PHY_EVENT(intf->channel))
98 		phy_mac_interrupt(intf->ndev->phydev);
99 }
100 
bcmasp_isr(int irq,void * data)101 static irqreturn_t bcmasp_isr(int irq, void *data)
102 {
103 	struct bcmasp_priv *priv = data;
104 	struct bcmasp_intf *intf;
105 	u32 status;
106 
107 	status = intr2_core_rl(priv, ASP_INTR2_STATUS) &
108 		~intr2_core_rl(priv, ASP_INTR2_MASK_STATUS);
109 
110 	intr2_core_wl(priv, status, ASP_INTR2_CLEAR);
111 
112 	if (unlikely(status == 0)) {
113 		dev_warn(&priv->pdev->dev, "l2 spurious interrupt\n");
114 		return IRQ_NONE;
115 	}
116 
117 	/* Handle intferfaces */
118 	list_for_each_entry(intf, &priv->intfs, list)
119 		bcmasp_intr2_handling(intf, status);
120 
121 	return IRQ_HANDLED;
122 }
123 
bcmasp_flush_rx_port(struct bcmasp_intf * intf)124 void bcmasp_flush_rx_port(struct bcmasp_intf *intf)
125 {
126 	struct bcmasp_priv *priv = intf->parent;
127 	u32 mask;
128 
129 	switch (intf->port) {
130 	case 0:
131 		mask = ASP_CTRL_UMAC0_FLUSH_MASK;
132 		break;
133 	case 1:
134 		mask = ASP_CTRL_UMAC1_FLUSH_MASK;
135 		break;
136 	case 2:
137 		mask = ASP_CTRL_SPB_FLUSH_MASK;
138 		break;
139 	default:
140 		/* Not valid port */
141 		return;
142 	}
143 
144 	rx_ctrl_core_wl(priv, mask, priv->hw_info->rx_ctrl_flush);
145 }
146 
bcmasp_netfilt_hw_en_wake(struct bcmasp_priv * priv,struct bcmasp_net_filter * nfilt)147 static void bcmasp_netfilt_hw_en_wake(struct bcmasp_priv *priv,
148 				      struct bcmasp_net_filter *nfilt)
149 {
150 	rx_filter_core_wl(priv, ASP_RX_FILTER_NET_OFFSET_L3_1(64),
151 			  ASP_RX_FILTER_NET_OFFSET(nfilt->hw_index));
152 
153 	rx_filter_core_wl(priv, ASP_RX_FILTER_NET_OFFSET_L2(32) |
154 			  ASP_RX_FILTER_NET_OFFSET_L3_0(32) |
155 			  ASP_RX_FILTER_NET_OFFSET_L3_1(96) |
156 			  ASP_RX_FILTER_NET_OFFSET_L4(32),
157 			  ASP_RX_FILTER_NET_OFFSET(nfilt->hw_index + 1));
158 
159 	rx_filter_core_wl(priv, ASP_RX_FILTER_NET_CFG_CH(nfilt->port + 8) |
160 			  ASP_RX_FILTER_NET_CFG_EN |
161 			  ASP_RX_FILTER_NET_CFG_L2_EN |
162 			  ASP_RX_FILTER_NET_CFG_L3_EN |
163 			  ASP_RX_FILTER_NET_CFG_L4_EN |
164 			  ASP_RX_FILTER_NET_CFG_L3_FRM(2) |
165 			  ASP_RX_FILTER_NET_CFG_L4_FRM(2) |
166 			  ASP_RX_FILTER_NET_CFG_UMC(nfilt->port),
167 			  ASP_RX_FILTER_NET_CFG(nfilt->hw_index));
168 
169 	rx_filter_core_wl(priv, ASP_RX_FILTER_NET_CFG_CH(nfilt->port + 8) |
170 			  ASP_RX_FILTER_NET_CFG_EN |
171 			  ASP_RX_FILTER_NET_CFG_L2_EN |
172 			  ASP_RX_FILTER_NET_CFG_L3_EN |
173 			  ASP_RX_FILTER_NET_CFG_L4_EN |
174 			  ASP_RX_FILTER_NET_CFG_L3_FRM(2) |
175 			  ASP_RX_FILTER_NET_CFG_L4_FRM(2) |
176 			  ASP_RX_FILTER_NET_CFG_UMC(nfilt->port),
177 			  ASP_RX_FILTER_NET_CFG(nfilt->hw_index + 1));
178 }
179 
180 #define MAX_WAKE_FILTER_SIZE		256
181 enum asp_netfilt_reg_type {
182 	ASP_NETFILT_MATCH = 0,
183 	ASP_NETFILT_MASK,
184 	ASP_NETFILT_MAX
185 };
186 
bcmasp_netfilt_get_reg_offset(struct bcmasp_priv * priv,struct bcmasp_net_filter * nfilt,enum asp_netfilt_reg_type reg_type,u32 offset)187 static int bcmasp_netfilt_get_reg_offset(struct bcmasp_priv *priv,
188 					 struct bcmasp_net_filter *nfilt,
189 					 enum asp_netfilt_reg_type reg_type,
190 					 u32 offset)
191 {
192 	u32 block_index, filter_sel;
193 
194 	if (offset < 32) {
195 		block_index = ASP_RX_FILTER_NET_L2;
196 		filter_sel = nfilt->hw_index;
197 	} else if (offset < 64) {
198 		block_index = ASP_RX_FILTER_NET_L2;
199 		filter_sel = nfilt->hw_index + 1;
200 	} else if (offset < 96) {
201 		block_index = ASP_RX_FILTER_NET_L3_0;
202 		filter_sel = nfilt->hw_index;
203 	} else if (offset < 128) {
204 		block_index = ASP_RX_FILTER_NET_L3_0;
205 		filter_sel = nfilt->hw_index + 1;
206 	} else if (offset < 160) {
207 		block_index = ASP_RX_FILTER_NET_L3_1;
208 		filter_sel = nfilt->hw_index;
209 	} else if (offset < 192) {
210 		block_index = ASP_RX_FILTER_NET_L3_1;
211 		filter_sel = nfilt->hw_index + 1;
212 	} else if (offset < 224) {
213 		block_index = ASP_RX_FILTER_NET_L4;
214 		filter_sel = nfilt->hw_index;
215 	} else if (offset < 256) {
216 		block_index = ASP_RX_FILTER_NET_L4;
217 		filter_sel = nfilt->hw_index + 1;
218 	} else {
219 		return -EINVAL;
220 	}
221 
222 	switch (reg_type) {
223 	case ASP_NETFILT_MATCH:
224 		return ASP_RX_FILTER_NET_PAT(filter_sel, block_index,
225 					     (offset % 32));
226 	case ASP_NETFILT_MASK:
227 		return ASP_RX_FILTER_NET_MASK(filter_sel, block_index,
228 					      (offset % 32));
229 	default:
230 		return -EINVAL;
231 	}
232 }
233 
bcmasp_netfilt_wr(struct bcmasp_priv * priv,struct bcmasp_net_filter * nfilt,enum asp_netfilt_reg_type reg_type,u32 val,u32 offset)234 static void bcmasp_netfilt_wr(struct bcmasp_priv *priv,
235 			      struct bcmasp_net_filter *nfilt,
236 			      enum asp_netfilt_reg_type reg_type,
237 			      u32 val, u32 offset)
238 {
239 	int reg_offset;
240 
241 	/* HW only accepts 4 byte aligned writes */
242 	if (!IS_ALIGNED(offset, 4) || offset > MAX_WAKE_FILTER_SIZE)
243 		return;
244 
245 	reg_offset = bcmasp_netfilt_get_reg_offset(priv, nfilt, reg_type,
246 						   offset);
247 
248 	rx_filter_core_wl(priv, val, reg_offset);
249 }
250 
bcmasp_netfilt_rd(struct bcmasp_priv * priv,struct bcmasp_net_filter * nfilt,enum asp_netfilt_reg_type reg_type,u32 offset)251 static u32 bcmasp_netfilt_rd(struct bcmasp_priv *priv,
252 			     struct bcmasp_net_filter *nfilt,
253 			     enum asp_netfilt_reg_type reg_type,
254 			     u32 offset)
255 {
256 	int reg_offset;
257 
258 	/* HW only accepts 4 byte aligned writes */
259 	if (!IS_ALIGNED(offset, 4) || offset > MAX_WAKE_FILTER_SIZE)
260 		return 0;
261 
262 	reg_offset = bcmasp_netfilt_get_reg_offset(priv, nfilt, reg_type,
263 						   offset);
264 
265 	return rx_filter_core_rl(priv, reg_offset);
266 }
267 
bcmasp_netfilt_wr_m_wake(struct bcmasp_priv * priv,struct bcmasp_net_filter * nfilt,u32 offset,void * match,void * mask,size_t size)268 static int bcmasp_netfilt_wr_m_wake(struct bcmasp_priv *priv,
269 				    struct bcmasp_net_filter *nfilt,
270 				    u32 offset, void *match, void *mask,
271 				    size_t size)
272 {
273 	u32 shift, mask_val = 0, match_val = 0;
274 	bool first_byte = true;
275 
276 	if ((offset + size) > MAX_WAKE_FILTER_SIZE)
277 		return -EINVAL;
278 
279 	while (size--) {
280 		/* The HW only accepts 4 byte aligned writes, so if we
281 		 * begin unaligned or if remaining bytes less than 4,
282 		 * we need to read then write to avoid losing current
283 		 * register state
284 		 */
285 		if (first_byte && (!IS_ALIGNED(offset, 4) || size < 3)) {
286 			match_val = bcmasp_netfilt_rd(priv, nfilt,
287 						      ASP_NETFILT_MATCH,
288 						      ALIGN_DOWN(offset, 4));
289 			mask_val = bcmasp_netfilt_rd(priv, nfilt,
290 						     ASP_NETFILT_MASK,
291 						     ALIGN_DOWN(offset, 4));
292 		}
293 
294 		shift = (3 - (offset % 4)) * 8;
295 		match_val &= ~GENMASK(shift + 7, shift);
296 		mask_val &= ~GENMASK(shift + 7, shift);
297 		match_val |= (u32)(*((u8 *)match) << shift);
298 		mask_val |= (u32)(*((u8 *)mask) << shift);
299 
300 		/* If last byte or last byte of word, write to reg */
301 		if (!size || ((offset % 4) == 3)) {
302 			bcmasp_netfilt_wr(priv, nfilt, ASP_NETFILT_MATCH,
303 					  match_val, ALIGN_DOWN(offset, 4));
304 			bcmasp_netfilt_wr(priv, nfilt, ASP_NETFILT_MASK,
305 					  mask_val, ALIGN_DOWN(offset, 4));
306 			first_byte = true;
307 		} else {
308 			first_byte = false;
309 		}
310 
311 		offset++;
312 		match++;
313 		mask++;
314 	}
315 
316 	return 0;
317 }
318 
bcmasp_netfilt_reset_hw(struct bcmasp_priv * priv,struct bcmasp_net_filter * nfilt)319 static void bcmasp_netfilt_reset_hw(struct bcmasp_priv *priv,
320 				    struct bcmasp_net_filter *nfilt)
321 {
322 	int i;
323 
324 	for (i = 0; i < MAX_WAKE_FILTER_SIZE; i += 4) {
325 		bcmasp_netfilt_wr(priv, nfilt, ASP_NETFILT_MATCH, 0, i);
326 		bcmasp_netfilt_wr(priv, nfilt, ASP_NETFILT_MASK, 0, i);
327 	}
328 }
329 
bcmasp_netfilt_tcpip4_wr(struct bcmasp_priv * priv,struct bcmasp_net_filter * nfilt,struct ethtool_tcpip4_spec * match,struct ethtool_tcpip4_spec * mask,u32 offset)330 static void bcmasp_netfilt_tcpip4_wr(struct bcmasp_priv *priv,
331 				     struct bcmasp_net_filter *nfilt,
332 				     struct ethtool_tcpip4_spec *match,
333 				     struct ethtool_tcpip4_spec *mask,
334 				     u32 offset)
335 {
336 	__be16 val_16, mask_16;
337 
338 	val_16 = htons(ETH_P_IP);
339 	mask_16 = htons(0xFFFF);
340 	bcmasp_netfilt_wr_m_wake(priv, nfilt, (ETH_ALEN * 2) + offset,
341 				 &val_16, &mask_16, sizeof(val_16));
342 	bcmasp_netfilt_wr_m_wake(priv, nfilt, ETH_HLEN + offset + 1,
343 				 &match->tos, &mask->tos,
344 				 sizeof(match->tos));
345 	bcmasp_netfilt_wr_m_wake(priv, nfilt, ETH_HLEN + offset + 12,
346 				 &match->ip4src, &mask->ip4src,
347 				 sizeof(match->ip4src));
348 	bcmasp_netfilt_wr_m_wake(priv, nfilt, ETH_HLEN + offset + 16,
349 				 &match->ip4dst, &mask->ip4dst,
350 				 sizeof(match->ip4dst));
351 	bcmasp_netfilt_wr_m_wake(priv, nfilt, ETH_HLEN + offset + 20,
352 				 &match->psrc, &mask->psrc,
353 				 sizeof(match->psrc));
354 	bcmasp_netfilt_wr_m_wake(priv, nfilt, ETH_HLEN + offset + 22,
355 				 &match->pdst, &mask->pdst,
356 				 sizeof(match->pdst));
357 }
358 
bcmasp_netfilt_tcpip6_wr(struct bcmasp_priv * priv,struct bcmasp_net_filter * nfilt,struct ethtool_tcpip6_spec * match,struct ethtool_tcpip6_spec * mask,u32 offset)359 static void bcmasp_netfilt_tcpip6_wr(struct bcmasp_priv *priv,
360 				     struct bcmasp_net_filter *nfilt,
361 				     struct ethtool_tcpip6_spec *match,
362 				     struct ethtool_tcpip6_spec *mask,
363 				     u32 offset)
364 {
365 	__be16 val_16, mask_16;
366 
367 	val_16 = htons(ETH_P_IPV6);
368 	mask_16 = htons(0xFFFF);
369 	bcmasp_netfilt_wr_m_wake(priv, nfilt, (ETH_ALEN * 2) + offset,
370 				 &val_16, &mask_16, sizeof(val_16));
371 	val_16 = htons(match->tclass << 4);
372 	mask_16 = htons(mask->tclass << 4);
373 	bcmasp_netfilt_wr_m_wake(priv, nfilt, ETH_HLEN + offset,
374 				 &val_16, &mask_16, sizeof(val_16));
375 	bcmasp_netfilt_wr_m_wake(priv, nfilt, ETH_HLEN + offset + 8,
376 				 &match->ip6src, &mask->ip6src,
377 				 sizeof(match->ip6src));
378 	bcmasp_netfilt_wr_m_wake(priv, nfilt, ETH_HLEN + offset + 24,
379 				 &match->ip6dst, &mask->ip6dst,
380 				 sizeof(match->ip6dst));
381 	bcmasp_netfilt_wr_m_wake(priv, nfilt, ETH_HLEN + offset + 40,
382 				 &match->psrc, &mask->psrc,
383 				 sizeof(match->psrc));
384 	bcmasp_netfilt_wr_m_wake(priv, nfilt, ETH_HLEN + offset + 42,
385 				 &match->pdst, &mask->pdst,
386 				 sizeof(match->pdst));
387 }
388 
bcmasp_netfilt_wr_to_hw(struct bcmasp_priv * priv,struct bcmasp_net_filter * nfilt)389 static int bcmasp_netfilt_wr_to_hw(struct bcmasp_priv *priv,
390 				   struct bcmasp_net_filter *nfilt)
391 {
392 	struct ethtool_rx_flow_spec *fs = &nfilt->fs;
393 	unsigned int offset = 0;
394 	__be16 val_16, mask_16;
395 	u8 val_8, mask_8;
396 
397 	/* Currently only supports wake filters */
398 	if (!nfilt->wake_filter)
399 		return -EINVAL;
400 
401 	bcmasp_netfilt_reset_hw(priv, nfilt);
402 
403 	if (fs->flow_type & FLOW_MAC_EXT) {
404 		bcmasp_netfilt_wr_m_wake(priv, nfilt, 0, &fs->h_ext.h_dest,
405 					 &fs->m_ext.h_dest,
406 					 sizeof(fs->h_ext.h_dest));
407 	}
408 
409 	if ((fs->flow_type & FLOW_EXT) &&
410 	    (fs->m_ext.vlan_etype || fs->m_ext.vlan_tci)) {
411 		bcmasp_netfilt_wr_m_wake(priv, nfilt, (ETH_ALEN * 2),
412 					 &fs->h_ext.vlan_etype,
413 					 &fs->m_ext.vlan_etype,
414 					 sizeof(fs->h_ext.vlan_etype));
415 		bcmasp_netfilt_wr_m_wake(priv, nfilt, ((ETH_ALEN * 2) + 2),
416 					 &fs->h_ext.vlan_tci,
417 					 &fs->m_ext.vlan_tci,
418 					 sizeof(fs->h_ext.vlan_tci));
419 		offset += VLAN_HLEN;
420 	}
421 
422 	switch (fs->flow_type & ~(FLOW_EXT | FLOW_MAC_EXT)) {
423 	case ETHER_FLOW:
424 		bcmasp_netfilt_wr_m_wake(priv, nfilt, 0,
425 					 &fs->h_u.ether_spec.h_dest,
426 					 &fs->m_u.ether_spec.h_dest,
427 					 sizeof(fs->h_u.ether_spec.h_dest));
428 		bcmasp_netfilt_wr_m_wake(priv, nfilt, ETH_ALEN,
429 					 &fs->h_u.ether_spec.h_source,
430 					 &fs->m_u.ether_spec.h_source,
431 					 sizeof(fs->h_u.ether_spec.h_source));
432 		bcmasp_netfilt_wr_m_wake(priv, nfilt, (ETH_ALEN * 2) + offset,
433 					 &fs->h_u.ether_spec.h_proto,
434 					 &fs->m_u.ether_spec.h_proto,
435 					 sizeof(fs->h_u.ether_spec.h_proto));
436 
437 		break;
438 	case IP_USER_FLOW:
439 		val_16 = htons(ETH_P_IP);
440 		mask_16 = htons(0xFFFF);
441 		bcmasp_netfilt_wr_m_wake(priv, nfilt, (ETH_ALEN * 2) + offset,
442 					 &val_16, &mask_16, sizeof(val_16));
443 		bcmasp_netfilt_wr_m_wake(priv, nfilt, ETH_HLEN + offset + 1,
444 					 &fs->h_u.usr_ip4_spec.tos,
445 					 &fs->m_u.usr_ip4_spec.tos,
446 					 sizeof(fs->h_u.usr_ip4_spec.tos));
447 		bcmasp_netfilt_wr_m_wake(priv, nfilt, ETH_HLEN + offset + 9,
448 					 &fs->h_u.usr_ip4_spec.proto,
449 					 &fs->m_u.usr_ip4_spec.proto,
450 					 sizeof(fs->h_u.usr_ip4_spec.proto));
451 		bcmasp_netfilt_wr_m_wake(priv, nfilt, ETH_HLEN + offset + 12,
452 					 &fs->h_u.usr_ip4_spec.ip4src,
453 					 &fs->m_u.usr_ip4_spec.ip4src,
454 					 sizeof(fs->h_u.usr_ip4_spec.ip4src));
455 		bcmasp_netfilt_wr_m_wake(priv, nfilt, ETH_HLEN + offset + 16,
456 					 &fs->h_u.usr_ip4_spec.ip4dst,
457 					 &fs->m_u.usr_ip4_spec.ip4dst,
458 					 sizeof(fs->h_u.usr_ip4_spec.ip4dst));
459 		if (!fs->m_u.usr_ip4_spec.l4_4_bytes)
460 			break;
461 
462 		/* Only supports 20 byte IPv4 header */
463 		val_8 = 0x45;
464 		mask_8 = 0xFF;
465 		bcmasp_netfilt_wr_m_wake(priv, nfilt, ETH_HLEN + offset,
466 					 &val_8, &mask_8, sizeof(val_8));
467 		bcmasp_netfilt_wr_m_wake(priv, nfilt,
468 					 ETH_HLEN + 20 + offset,
469 					 &fs->h_u.usr_ip4_spec.l4_4_bytes,
470 					 &fs->m_u.usr_ip4_spec.l4_4_bytes,
471 					 sizeof(fs->h_u.usr_ip4_spec.l4_4_bytes)
472 					 );
473 		break;
474 	case TCP_V4_FLOW:
475 		val_8 = IPPROTO_TCP;
476 		mask_8 = 0xFF;
477 		bcmasp_netfilt_tcpip4_wr(priv, nfilt, &fs->h_u.tcp_ip4_spec,
478 					 &fs->m_u.tcp_ip4_spec, offset);
479 		bcmasp_netfilt_wr_m_wake(priv, nfilt, ETH_HLEN + offset + 9,
480 					 &val_8, &mask_8, sizeof(val_8));
481 		break;
482 	case UDP_V4_FLOW:
483 		val_8 = IPPROTO_UDP;
484 		mask_8 = 0xFF;
485 		bcmasp_netfilt_tcpip4_wr(priv, nfilt, &fs->h_u.udp_ip4_spec,
486 					 &fs->m_u.udp_ip4_spec, offset);
487 
488 		bcmasp_netfilt_wr_m_wake(priv, nfilt, ETH_HLEN + offset + 9,
489 					 &val_8, &mask_8, sizeof(val_8));
490 		break;
491 	case TCP_V6_FLOW:
492 		val_8 = IPPROTO_TCP;
493 		mask_8 = 0xFF;
494 		bcmasp_netfilt_tcpip6_wr(priv, nfilt, &fs->h_u.tcp_ip6_spec,
495 					 &fs->m_u.tcp_ip6_spec, offset);
496 		bcmasp_netfilt_wr_m_wake(priv, nfilt, ETH_HLEN + offset + 6,
497 					 &val_8, &mask_8, sizeof(val_8));
498 		break;
499 	case UDP_V6_FLOW:
500 		val_8 = IPPROTO_UDP;
501 		mask_8 = 0xFF;
502 		bcmasp_netfilt_tcpip6_wr(priv, nfilt, &fs->h_u.udp_ip6_spec,
503 					 &fs->m_u.udp_ip6_spec, offset);
504 		bcmasp_netfilt_wr_m_wake(priv, nfilt, ETH_HLEN + offset + 6,
505 					 &val_8, &mask_8, sizeof(val_8));
506 		break;
507 	}
508 
509 	bcmasp_netfilt_hw_en_wake(priv, nfilt);
510 
511 	return 0;
512 }
513 
bcmasp_netfilt_suspend(struct bcmasp_intf * intf)514 void bcmasp_netfilt_suspend(struct bcmasp_intf *intf)
515 {
516 	struct bcmasp_priv *priv = intf->parent;
517 	bool write = false;
518 	int ret, i;
519 
520 	/* Write all filters to HW */
521 	for (i = 0; i < NUM_NET_FILTERS; i++) {
522 		/* If the filter does not match the port, skip programming. */
523 		if (!priv->net_filters[i].claimed ||
524 		    priv->net_filters[i].port != intf->port)
525 			continue;
526 
527 		if (i > 0 && (i % 2) &&
528 		    priv->net_filters[i].wake_filter &&
529 		    priv->net_filters[i - 1].wake_filter)
530 			continue;
531 
532 		ret = bcmasp_netfilt_wr_to_hw(priv, &priv->net_filters[i]);
533 		if (!ret)
534 			write = true;
535 	}
536 
537 	/* Successfully programmed at least one wake filter
538 	 * so enable top level wake config
539 	 */
540 	if (write)
541 		rx_filter_core_wl(priv, (ASP_RX_FILTER_OPUT_EN |
542 				  ASP_RX_FILTER_LNR_MD |
543 				  ASP_RX_FILTER_GEN_WK_EN |
544 				  ASP_RX_FILTER_NT_FLT_EN),
545 				  ASP_RX_FILTER_BLK_CTRL);
546 }
547 
bcmasp_netfilt_get_all_active(struct bcmasp_intf * intf,u32 * rule_locs,u32 * rule_cnt)548 int bcmasp_netfilt_get_all_active(struct bcmasp_intf *intf, u32 *rule_locs,
549 				  u32 *rule_cnt)
550 {
551 	struct bcmasp_priv *priv = intf->parent;
552 	int j = 0, i;
553 
554 	for (i = 0; i < NUM_NET_FILTERS; i++) {
555 		if (!priv->net_filters[i].claimed ||
556 		    priv->net_filters[i].port != intf->port)
557 			continue;
558 
559 		if (i > 0 && (i % 2) &&
560 		    priv->net_filters[i].wake_filter &&
561 		    priv->net_filters[i - 1].wake_filter)
562 			continue;
563 
564 		if (j == *rule_cnt)
565 			return -EMSGSIZE;
566 
567 		rule_locs[j++] = priv->net_filters[i].fs.location;
568 	}
569 
570 	*rule_cnt = j;
571 
572 	return 0;
573 }
574 
bcmasp_netfilt_get_active(struct bcmasp_intf * intf)575 int bcmasp_netfilt_get_active(struct bcmasp_intf *intf)
576 {
577 	struct bcmasp_priv *priv = intf->parent;
578 	int cnt = 0, i;
579 
580 	for (i = 0; i < NUM_NET_FILTERS; i++) {
581 		if (!priv->net_filters[i].claimed ||
582 		    priv->net_filters[i].port != intf->port)
583 			continue;
584 
585 		/* Skip over a wake filter pair */
586 		if (i > 0 && (i % 2) &&
587 		    priv->net_filters[i].wake_filter &&
588 		    priv->net_filters[i - 1].wake_filter)
589 			continue;
590 
591 		cnt++;
592 	}
593 
594 	return cnt;
595 }
596 
bcmasp_netfilt_check_dup(struct bcmasp_intf * intf,struct ethtool_rx_flow_spec * fs)597 bool bcmasp_netfilt_check_dup(struct bcmasp_intf *intf,
598 			      struct ethtool_rx_flow_spec *fs)
599 {
600 	struct bcmasp_priv *priv = intf->parent;
601 	struct ethtool_rx_flow_spec *cur;
602 	size_t fs_size = 0;
603 	int i;
604 
605 	for (i = 0; i < NUM_NET_FILTERS; i++) {
606 		if (!priv->net_filters[i].claimed ||
607 		    priv->net_filters[i].port != intf->port)
608 			continue;
609 
610 		cur = &priv->net_filters[i].fs;
611 
612 		if (cur->flow_type != fs->flow_type ||
613 		    cur->ring_cookie != fs->ring_cookie)
614 			continue;
615 
616 		switch (fs->flow_type & ~(FLOW_EXT | FLOW_MAC_EXT)) {
617 		case ETHER_FLOW:
618 			fs_size = sizeof(struct ethhdr);
619 			break;
620 		case IP_USER_FLOW:
621 			fs_size = sizeof(struct ethtool_usrip4_spec);
622 			break;
623 		case TCP_V6_FLOW:
624 		case UDP_V6_FLOW:
625 			fs_size = sizeof(struct ethtool_tcpip6_spec);
626 			break;
627 		case TCP_V4_FLOW:
628 		case UDP_V4_FLOW:
629 			fs_size = sizeof(struct ethtool_tcpip4_spec);
630 			break;
631 		default:
632 			continue;
633 		}
634 
635 		if (memcmp(&cur->h_u, &fs->h_u, fs_size) ||
636 		    memcmp(&cur->m_u, &fs->m_u, fs_size))
637 			continue;
638 
639 		if (cur->flow_type & FLOW_EXT) {
640 			if (cur->h_ext.vlan_etype != fs->h_ext.vlan_etype ||
641 			    cur->m_ext.vlan_etype != fs->m_ext.vlan_etype ||
642 			    cur->h_ext.vlan_tci != fs->h_ext.vlan_tci ||
643 			    cur->m_ext.vlan_tci != fs->m_ext.vlan_tci ||
644 			    cur->h_ext.data[0] != fs->h_ext.data[0])
645 				continue;
646 		}
647 		if (cur->flow_type & FLOW_MAC_EXT) {
648 			if (memcmp(&cur->h_ext.h_dest,
649 				   &fs->h_ext.h_dest, ETH_ALEN) ||
650 			    memcmp(&cur->m_ext.h_dest,
651 				   &fs->m_ext.h_dest, ETH_ALEN))
652 				continue;
653 		}
654 
655 		return true;
656 	}
657 
658 	return false;
659 }
660 
661 /* If no network filter found, return open filter.
662  * If no more open filters return NULL
663  */
bcmasp_netfilt_get_init(struct bcmasp_intf * intf,u32 loc,bool wake_filter,bool init)664 struct bcmasp_net_filter *bcmasp_netfilt_get_init(struct bcmasp_intf *intf,
665 						  u32 loc, bool wake_filter,
666 						  bool init)
667 {
668 	struct bcmasp_net_filter *nfilter = NULL;
669 	struct bcmasp_priv *priv = intf->parent;
670 	int i, open_index = -1;
671 
672 	/* Check whether we exceed the filter table capacity */
673 	if (loc != RX_CLS_LOC_ANY && loc >= NUM_NET_FILTERS)
674 		return ERR_PTR(-EINVAL);
675 
676 	/* If the filter location is busy (already claimed) and we are initializing
677 	 * the filter (insertion), return a busy error code.
678 	 */
679 	if (loc != RX_CLS_LOC_ANY && init && priv->net_filters[loc].claimed)
680 		return ERR_PTR(-EBUSY);
681 
682 	/* We need two filters for wake-up, so we cannot use an odd filter */
683 	if (wake_filter && loc != RX_CLS_LOC_ANY && (loc % 2))
684 		return ERR_PTR(-EINVAL);
685 
686 	/* Initialize the loop index based on the desired location or from 0 */
687 	i = loc == RX_CLS_LOC_ANY ? 0 : loc;
688 
689 	for ( ; i < NUM_NET_FILTERS; i++) {
690 		/* Found matching network filter */
691 		if (!init &&
692 		    priv->net_filters[i].claimed &&
693 		    priv->net_filters[i].hw_index == i &&
694 		    priv->net_filters[i].port == intf->port)
695 			return &priv->net_filters[i];
696 
697 		/* If we don't need a new filter or new filter already found */
698 		if (!init || open_index >= 0)
699 			continue;
700 
701 		/* Wake filter conslidates two filters to cover more bytes
702 		 * Wake filter is open if...
703 		 * 1. It is an even filter
704 		 * 2. The current and next filter is not claimed
705 		 */
706 		if (wake_filter && !(i % 2) && !priv->net_filters[i].claimed &&
707 		    !priv->net_filters[i + 1].claimed)
708 			open_index = i;
709 		else if (!priv->net_filters[i].claimed)
710 			open_index = i;
711 	}
712 
713 	if (open_index >= 0) {
714 		nfilter = &priv->net_filters[open_index];
715 		nfilter->claimed = true;
716 		nfilter->port = intf->port;
717 		nfilter->hw_index = open_index;
718 	}
719 
720 	if (wake_filter && open_index >= 0) {
721 		/* Claim next filter */
722 		priv->net_filters[open_index + 1].claimed = true;
723 		priv->net_filters[open_index + 1].wake_filter = true;
724 		nfilter->wake_filter = true;
725 	}
726 
727 	return nfilter ? nfilter : ERR_PTR(-EINVAL);
728 }
729 
bcmasp_netfilt_release(struct bcmasp_intf * intf,struct bcmasp_net_filter * nfilt)730 void bcmasp_netfilt_release(struct bcmasp_intf *intf,
731 			    struct bcmasp_net_filter *nfilt)
732 {
733 	struct bcmasp_priv *priv = intf->parent;
734 
735 	if (nfilt->wake_filter) {
736 		memset(&priv->net_filters[nfilt->hw_index + 1], 0,
737 		       sizeof(struct bcmasp_net_filter));
738 	}
739 
740 	memset(nfilt, 0, sizeof(struct bcmasp_net_filter));
741 }
742 
bcmasp_addr_to_uint(unsigned char * addr,u32 * high,u32 * low)743 static void bcmasp_addr_to_uint(unsigned char *addr, u32 *high, u32 *low)
744 {
745 	*high = (u32)(addr[0] << 8 | addr[1]);
746 	*low = (u32)(addr[2] << 24 | addr[3] << 16 | addr[4] << 8 |
747 		     addr[5]);
748 }
749 
bcmasp_set_mda_filter(struct bcmasp_intf * intf,const unsigned char * addr,unsigned char * mask,unsigned int i)750 static void bcmasp_set_mda_filter(struct bcmasp_intf *intf,
751 				  const unsigned char *addr,
752 				  unsigned char *mask,
753 				  unsigned int i)
754 {
755 	struct bcmasp_priv *priv = intf->parent;
756 	u32 addr_h, addr_l, mask_h, mask_l;
757 
758 	/* Set local copy */
759 	ether_addr_copy(priv->mda_filters[i].mask, mask);
760 	ether_addr_copy(priv->mda_filters[i].addr, addr);
761 
762 	/* Write to HW */
763 	bcmasp_addr_to_uint(priv->mda_filters[i].mask, &mask_h, &mask_l);
764 	bcmasp_addr_to_uint(priv->mda_filters[i].addr, &addr_h, &addr_l);
765 	rx_filter_core_wl(priv, addr_h, ASP_RX_FILTER_MDA_PAT_H(i));
766 	rx_filter_core_wl(priv, addr_l, ASP_RX_FILTER_MDA_PAT_L(i));
767 	rx_filter_core_wl(priv, mask_h, ASP_RX_FILTER_MDA_MSK_H(i));
768 	rx_filter_core_wl(priv, mask_l, ASP_RX_FILTER_MDA_MSK_L(i));
769 }
770 
bcmasp_en_mda_filter(struct bcmasp_intf * intf,bool en,unsigned int i)771 static void bcmasp_en_mda_filter(struct bcmasp_intf *intf, bool en,
772 				 unsigned int i)
773 {
774 	struct bcmasp_priv *priv = intf->parent;
775 
776 	if (priv->mda_filters[i].en == en)
777 		return;
778 
779 	priv->mda_filters[i].en = en;
780 	priv->mda_filters[i].port = intf->port;
781 
782 	rx_filter_core_wl(priv, ((intf->channel + 8) |
783 			  (en << ASP_RX_FILTER_MDA_CFG_EN_SHIFT) |
784 			  ASP_RX_FILTER_MDA_CFG_UMC_SEL(intf->port)),
785 			  ASP_RX_FILTER_MDA_CFG(i));
786 }
787 
788 /* There are 32 MDA filters shared between all ports, we reserve 4 filters per
789  * port for the following.
790  * - Promisc: Filter to allow all packets when promisc is enabled
791  * - All Multicast
792  * - Broadcast
793  * - Own address
794  *
795  * The reserved filters are identified as so.
796  * - Promisc: (index * 4) + 0
797  * - All Multicast: (index * 4) + 1
798  * - Broadcast: (index * 4) + 2
799  * - Own address: (index * 4) + 3
800  */
801 enum asp_rx_filter_id {
802 	ASP_RX_FILTER_MDA_PROMISC = 0,
803 	ASP_RX_FILTER_MDA_ALLMULTI,
804 	ASP_RX_FILTER_MDA_BROADCAST,
805 	ASP_RX_FILTER_MDA_OWN_ADDR,
806 	ASP_RX_FILTER_MDA_RES_MAX,
807 };
808 
809 #define ASP_RX_FILT_MDA(intf, name)	(((intf)->index * \
810 					  ASP_RX_FILTER_MDA_RES_MAX) \
811 					 + ASP_RX_FILTER_MDA_##name)
812 
bcmasp_total_res_mda_cnt(struct bcmasp_priv * priv)813 static int bcmasp_total_res_mda_cnt(struct bcmasp_priv *priv)
814 {
815 	return list_count_nodes(&priv->intfs) * ASP_RX_FILTER_MDA_RES_MAX;
816 }
817 
bcmasp_set_promisc(struct bcmasp_intf * intf,bool en)818 void bcmasp_set_promisc(struct bcmasp_intf *intf, bool en)
819 {
820 	unsigned int i = ASP_RX_FILT_MDA(intf, PROMISC);
821 	unsigned char promisc[ETH_ALEN];
822 
823 	eth_zero_addr(promisc);
824 	/* Set mask to 00:00:00:00:00:00 to match all packets */
825 	bcmasp_set_mda_filter(intf, promisc, promisc, i);
826 	bcmasp_en_mda_filter(intf, en, i);
827 }
828 
bcmasp_set_allmulti(struct bcmasp_intf * intf,bool en)829 void bcmasp_set_allmulti(struct bcmasp_intf *intf, bool en)
830 {
831 	unsigned char allmulti[] = {0x01, 0x00, 0x00, 0x00, 0x00, 0x00};
832 	unsigned int i = ASP_RX_FILT_MDA(intf, ALLMULTI);
833 
834 	/* Set mask to 01:00:00:00:00:00 to match all multicast */
835 	bcmasp_set_mda_filter(intf, allmulti, allmulti, i);
836 	bcmasp_en_mda_filter(intf, en, i);
837 }
838 
bcmasp_set_broad(struct bcmasp_intf * intf,bool en)839 void bcmasp_set_broad(struct bcmasp_intf *intf, bool en)
840 {
841 	unsigned int i = ASP_RX_FILT_MDA(intf, BROADCAST);
842 	unsigned char addr[ETH_ALEN];
843 
844 	eth_broadcast_addr(addr);
845 	bcmasp_set_mda_filter(intf, addr, addr, i);
846 	bcmasp_en_mda_filter(intf, en, i);
847 }
848 
bcmasp_set_oaddr(struct bcmasp_intf * intf,const unsigned char * addr,bool en)849 void bcmasp_set_oaddr(struct bcmasp_intf *intf, const unsigned char *addr,
850 		      bool en)
851 {
852 	unsigned char mask[] = {0xff, 0xff, 0xff, 0xff, 0xff, 0xff};
853 	unsigned int i = ASP_RX_FILT_MDA(intf, OWN_ADDR);
854 
855 	bcmasp_set_mda_filter(intf, addr, mask, i);
856 	bcmasp_en_mda_filter(intf, en, i);
857 }
858 
bcmasp_disable_all_filters(struct bcmasp_intf * intf)859 void bcmasp_disable_all_filters(struct bcmasp_intf *intf)
860 {
861 	struct bcmasp_priv *priv = intf->parent;
862 	unsigned int i;
863 	int res_count;
864 
865 	res_count = bcmasp_total_res_mda_cnt(intf->parent);
866 
867 	/* Disable all filters held by this port */
868 	for (i = res_count; i < NUM_MDA_FILTERS; i++) {
869 		if (priv->mda_filters[i].en &&
870 		    priv->mda_filters[i].port == intf->port)
871 			bcmasp_en_mda_filter(intf, 0, i);
872 	}
873 }
874 
bcmasp_combine_set_filter(struct bcmasp_intf * intf,unsigned char * addr,unsigned char * mask,int i)875 static int bcmasp_combine_set_filter(struct bcmasp_intf *intf,
876 				     unsigned char *addr, unsigned char *mask,
877 				     int i)
878 {
879 	struct bcmasp_priv *priv = intf->parent;
880 	u64 addr1, addr2, mask1, mask2, mask3;
881 
882 	/* Switch to u64 to help with the calculations */
883 	addr1 = ether_addr_to_u64(priv->mda_filters[i].addr);
884 	mask1 = ether_addr_to_u64(priv->mda_filters[i].mask);
885 	addr2 = ether_addr_to_u64(addr);
886 	mask2 = ether_addr_to_u64(mask);
887 
888 	/* Check if one filter resides within the other */
889 	mask3 = mask1 & mask2;
890 	if (mask3 == mask1 && ((addr1 & mask1) == (addr2 & mask1))) {
891 		/* Filter 2 resides within filter 1, so everything is good */
892 		return 0;
893 	} else if (mask3 == mask2 && ((addr1 & mask2) == (addr2 & mask2))) {
894 		/* Filter 1 resides within filter 2, so swap filters */
895 		bcmasp_set_mda_filter(intf, addr, mask, i);
896 		return 0;
897 	}
898 
899 	/* Unable to combine */
900 	return -EINVAL;
901 }
902 
bcmasp_set_en_mda_filter(struct bcmasp_intf * intf,unsigned char * addr,unsigned char * mask)903 int bcmasp_set_en_mda_filter(struct bcmasp_intf *intf, unsigned char *addr,
904 			     unsigned char *mask)
905 {
906 	struct bcmasp_priv *priv = intf->parent;
907 	int ret, res_count;
908 	unsigned int i;
909 
910 	res_count = bcmasp_total_res_mda_cnt(intf->parent);
911 
912 	for (i = res_count; i < NUM_MDA_FILTERS; i++) {
913 		/* If filter not enabled or belongs to another port skip */
914 		if (!priv->mda_filters[i].en ||
915 		    priv->mda_filters[i].port != intf->port)
916 			continue;
917 
918 		/* Attempt to combine filters */
919 		ret = bcmasp_combine_set_filter(intf, addr, mask, i);
920 		if (!ret) {
921 			intf->mib.filters_combine_cnt++;
922 			return 0;
923 		}
924 	}
925 
926 	/* Create new filter if possible */
927 	for (i = res_count; i < NUM_MDA_FILTERS; i++) {
928 		if (priv->mda_filters[i].en)
929 			continue;
930 
931 		bcmasp_set_mda_filter(intf, addr, mask, i);
932 		bcmasp_en_mda_filter(intf, 1, i);
933 		return 0;
934 	}
935 
936 	/* No room for new filter */
937 	return -EINVAL;
938 }
939 
bcmasp_core_init_filters(struct bcmasp_priv * priv)940 static void bcmasp_core_init_filters(struct bcmasp_priv *priv)
941 {
942 	unsigned int i;
943 
944 	/* Disable all filters and reset software view since the HW
945 	 * can lose context while in deep sleep suspend states
946 	 */
947 	for (i = 0; i < NUM_MDA_FILTERS; i++) {
948 		rx_filter_core_wl(priv, 0x0, ASP_RX_FILTER_MDA_CFG(i));
949 		priv->mda_filters[i].en = 0;
950 	}
951 
952 	for (i = 0; i < NUM_NET_FILTERS; i++)
953 		rx_filter_core_wl(priv, 0x0, ASP_RX_FILTER_NET_CFG(i));
954 
955 	/* Top level filter enable bit should be enabled at all times, set
956 	 * GEN_WAKE_CLEAR to clear the network filter wake-up which would
957 	 * otherwise be sticky
958 	 */
959 	rx_filter_core_wl(priv, (ASP_RX_FILTER_OPUT_EN |
960 			  ASP_RX_FILTER_MDA_EN |
961 			  ASP_RX_FILTER_GEN_WK_CLR |
962 			  ASP_RX_FILTER_NT_FLT_EN),
963 			  ASP_RX_FILTER_BLK_CTRL);
964 }
965 
966 /* ASP core initialization */
bcmasp_core_init(struct bcmasp_priv * priv)967 static void bcmasp_core_init(struct bcmasp_priv *priv)
968 {
969 	tx_analytics_core_wl(priv, 0x0, ASP_TX_ANALYTICS_CTRL);
970 	rx_analytics_core_wl(priv, 0x4, ASP_RX_ANALYTICS_CTRL);
971 
972 	rx_edpkt_core_wl(priv, (ASP_EDPKT_HDR_SZ_128 << ASP_EDPKT_HDR_SZ_SHIFT),
973 			 ASP_EDPKT_HDR_CFG);
974 	rx_edpkt_core_wl(priv,
975 			 (ASP_EDPKT_ENDI_BT_SWP_WD << ASP_EDPKT_ENDI_DESC_SHIFT),
976 			 ASP_EDPKT_ENDI);
977 
978 	rx_edpkt_core_wl(priv, 0x1b, ASP_EDPKT_BURST_BUF_PSCAL_TOUT);
979 	rx_edpkt_core_wl(priv, 0x3e8, ASP_EDPKT_BURST_BUF_WRITE_TOUT);
980 	rx_edpkt_core_wl(priv, 0x3e8, ASP_EDPKT_BURST_BUF_READ_TOUT);
981 
982 	rx_edpkt_core_wl(priv, ASP_EDPKT_ENABLE_EN, ASP_EDPKT_ENABLE);
983 
984 	/* Disable and clear both UniMAC's wake-up interrupts to avoid
985 	 * sticky interrupts.
986 	 */
987 	_intr2_mask_set(priv, ASP_INTR2_UMC0_WAKE | ASP_INTR2_UMC1_WAKE);
988 	intr2_core_wl(priv, ASP_INTR2_UMC0_WAKE | ASP_INTR2_UMC1_WAKE,
989 		      ASP_INTR2_CLEAR);
990 }
991 
bcmasp_core_clock_select_many(struct bcmasp_priv * priv,bool slow)992 static void bcmasp_core_clock_select_many(struct bcmasp_priv *priv, bool slow)
993 {
994 	u32 reg;
995 
996 	reg = ctrl2_core_rl(priv, ASP_CTRL2_CORE_CLOCK_SELECT);
997 	if (slow)
998 		reg &= ~ASP_CTRL2_CORE_CLOCK_SELECT_MAIN;
999 	else
1000 		reg |= ASP_CTRL2_CORE_CLOCK_SELECT_MAIN;
1001 	ctrl2_core_wl(priv, reg, ASP_CTRL2_CORE_CLOCK_SELECT);
1002 
1003 	reg = ctrl2_core_rl(priv, ASP_CTRL2_CPU_CLOCK_SELECT);
1004 	if (slow)
1005 		reg &= ~ASP_CTRL2_CPU_CLOCK_SELECT_MAIN;
1006 	else
1007 		reg |= ASP_CTRL2_CPU_CLOCK_SELECT_MAIN;
1008 	ctrl2_core_wl(priv, reg, ASP_CTRL2_CPU_CLOCK_SELECT);
1009 }
1010 
bcmasp_core_clock_select_one(struct bcmasp_priv * priv,bool slow)1011 static void bcmasp_core_clock_select_one(struct bcmasp_priv *priv, bool slow)
1012 {
1013 	u32 reg;
1014 
1015 	reg = ctrl_core_rl(priv, ASP_CTRL_CORE_CLOCK_SELECT);
1016 	if (slow)
1017 		reg &= ~ASP_CTRL_CORE_CLOCK_SELECT_MAIN;
1018 	else
1019 		reg |= ASP_CTRL_CORE_CLOCK_SELECT_MAIN;
1020 	ctrl_core_wl(priv, reg, ASP_CTRL_CORE_CLOCK_SELECT);
1021 }
1022 
bcmasp_core_clock_set_ll(struct bcmasp_priv * priv,u32 clr,u32 set)1023 static void bcmasp_core_clock_set_ll(struct bcmasp_priv *priv, u32 clr, u32 set)
1024 {
1025 	u32 reg;
1026 
1027 	reg = ctrl_core_rl(priv, ASP_CTRL_CLOCK_CTRL);
1028 	reg &= ~clr;
1029 	reg |= set;
1030 	ctrl_core_wl(priv, reg, ASP_CTRL_CLOCK_CTRL);
1031 
1032 	reg = ctrl_core_rl(priv, ASP_CTRL_SCRATCH_0);
1033 	reg &= ~clr;
1034 	reg |= set;
1035 	ctrl_core_wl(priv, reg, ASP_CTRL_SCRATCH_0);
1036 }
1037 
bcmasp_core_clock_set(struct bcmasp_priv * priv,u32 clr,u32 set)1038 static void bcmasp_core_clock_set(struct bcmasp_priv *priv, u32 clr, u32 set)
1039 {
1040 	unsigned long flags;
1041 
1042 	spin_lock_irqsave(&priv->clk_lock, flags);
1043 	bcmasp_core_clock_set_ll(priv, clr, set);
1044 	spin_unlock_irqrestore(&priv->clk_lock, flags);
1045 }
1046 
bcmasp_core_clock_set_intf(struct bcmasp_intf * intf,bool en)1047 void bcmasp_core_clock_set_intf(struct bcmasp_intf *intf, bool en)
1048 {
1049 	u32 intf_mask = ASP_CTRL_CLOCK_CTRL_ASP_RGMII_DIS(intf->port);
1050 	struct bcmasp_priv *priv = intf->parent;
1051 	unsigned long flags;
1052 	u32 reg;
1053 
1054 	/* When enabling an interface, if the RX or TX clocks were not enabled,
1055 	 * enable them. Conversely, while disabling an interface, if this is
1056 	 * the last one enabled, we can turn off the shared RX and TX clocks as
1057 	 * well. We control enable bits which is why we test for equality on
1058 	 * the RGMII clock bit mask.
1059 	 */
1060 	spin_lock_irqsave(&priv->clk_lock, flags);
1061 	if (en) {
1062 		intf_mask |= ASP_CTRL_CLOCK_CTRL_ASP_TX_DISABLE |
1063 			     ASP_CTRL_CLOCK_CTRL_ASP_RX_DISABLE;
1064 		bcmasp_core_clock_set_ll(priv, intf_mask, 0);
1065 	} else {
1066 		reg = ctrl_core_rl(priv, ASP_CTRL_SCRATCH_0) | intf_mask;
1067 		if ((reg & ASP_CTRL_CLOCK_CTRL_ASP_RGMII_MASK) ==
1068 		    ASP_CTRL_CLOCK_CTRL_ASP_RGMII_MASK)
1069 			intf_mask |= ASP_CTRL_CLOCK_CTRL_ASP_TX_DISABLE |
1070 				     ASP_CTRL_CLOCK_CTRL_ASP_RX_DISABLE;
1071 		bcmasp_core_clock_set_ll(priv, 0, intf_mask);
1072 	}
1073 	spin_unlock_irqrestore(&priv->clk_lock, flags);
1074 }
1075 
bcmasp_isr_wol(int irq,void * data)1076 static irqreturn_t bcmasp_isr_wol(int irq, void *data)
1077 {
1078 	struct bcmasp_priv *priv = data;
1079 	u32 status;
1080 
1081 	/* No L3 IRQ, so we good */
1082 	if (priv->wol_irq <= 0)
1083 		goto irq_handled;
1084 
1085 	status = wakeup_intr2_core_rl(priv, ASP_WAKEUP_INTR2_STATUS) &
1086 		~wakeup_intr2_core_rl(priv, ASP_WAKEUP_INTR2_MASK_STATUS);
1087 	wakeup_intr2_core_wl(priv, status, ASP_WAKEUP_INTR2_CLEAR);
1088 
1089 irq_handled:
1090 	pm_wakeup_event(&priv->pdev->dev, 0);
1091 	return IRQ_HANDLED;
1092 }
1093 
bcmasp_get_and_request_irq(struct bcmasp_priv * priv,int i)1094 static int bcmasp_get_and_request_irq(struct bcmasp_priv *priv, int i)
1095 {
1096 	struct platform_device *pdev = priv->pdev;
1097 	int irq, ret;
1098 
1099 	irq = platform_get_irq_optional(pdev, i);
1100 	if (irq < 0)
1101 		return irq;
1102 
1103 	ret = devm_request_irq(&pdev->dev, irq, bcmasp_isr_wol, 0,
1104 			       pdev->name, priv);
1105 	if (ret)
1106 		return ret;
1107 
1108 	return irq;
1109 }
1110 
bcmasp_init_wol_shared(struct bcmasp_priv * priv)1111 static void bcmasp_init_wol_shared(struct bcmasp_priv *priv)
1112 {
1113 	struct platform_device *pdev = priv->pdev;
1114 	struct device *dev = &pdev->dev;
1115 	int irq;
1116 
1117 	irq = bcmasp_get_and_request_irq(priv, 1);
1118 	if (irq < 0) {
1119 		dev_warn(dev, "Failed to init WoL irq: %d\n", irq);
1120 		return;
1121 	}
1122 
1123 	priv->wol_irq = irq;
1124 	priv->wol_irq_enabled_mask = 0;
1125 	device_set_wakeup_capable(&pdev->dev, 1);
1126 }
1127 
bcmasp_enable_wol_shared(struct bcmasp_intf * intf,bool en)1128 static void bcmasp_enable_wol_shared(struct bcmasp_intf *intf, bool en)
1129 {
1130 	struct bcmasp_priv *priv = intf->parent;
1131 	struct device *dev = &priv->pdev->dev;
1132 
1133 	if (en) {
1134 		if (priv->wol_irq_enabled_mask) {
1135 			set_bit(intf->port, &priv->wol_irq_enabled_mask);
1136 			return;
1137 		}
1138 
1139 		/* First enable */
1140 		set_bit(intf->port, &priv->wol_irq_enabled_mask);
1141 		enable_irq_wake(priv->wol_irq);
1142 		device_set_wakeup_enable(dev, 1);
1143 	} else {
1144 		if (!priv->wol_irq_enabled_mask)
1145 			return;
1146 
1147 		clear_bit(intf->port, &priv->wol_irq_enabled_mask);
1148 		if (priv->wol_irq_enabled_mask)
1149 			return;
1150 
1151 		/* Last disable */
1152 		disable_irq_wake(priv->wol_irq);
1153 		device_set_wakeup_enable(dev, 0);
1154 	}
1155 }
1156 
bcmasp_wol_irq_destroy_shared(struct bcmasp_priv * priv)1157 static void bcmasp_wol_irq_destroy_shared(struct bcmasp_priv *priv)
1158 {
1159 	if (priv->wol_irq > 0)
1160 		free_irq(priv->wol_irq, priv);
1161 }
1162 
bcmasp_init_wol_per_intf(struct bcmasp_priv * priv)1163 static void bcmasp_init_wol_per_intf(struct bcmasp_priv *priv)
1164 {
1165 	struct platform_device *pdev = priv->pdev;
1166 	struct device *dev = &pdev->dev;
1167 	struct bcmasp_intf *intf;
1168 	int irq;
1169 
1170 	list_for_each_entry(intf, &priv->intfs, list) {
1171 		irq = bcmasp_get_and_request_irq(priv, intf->port + 1);
1172 		if (irq < 0) {
1173 			dev_warn(dev, "Failed to init WoL irq(port %d): %d\n",
1174 				 intf->port, irq);
1175 			continue;
1176 		}
1177 
1178 		intf->wol_irq = irq;
1179 		intf->wol_irq_enabled = false;
1180 		device_set_wakeup_capable(&pdev->dev, 1);
1181 	}
1182 }
1183 
bcmasp_enable_wol_per_intf(struct bcmasp_intf * intf,bool en)1184 static void bcmasp_enable_wol_per_intf(struct bcmasp_intf *intf, bool en)
1185 {
1186 	struct device *dev = &intf->parent->pdev->dev;
1187 
1188 	if (en ^ intf->wol_irq_enabled)
1189 		irq_set_irq_wake(intf->wol_irq, en);
1190 
1191 	intf->wol_irq_enabled = en;
1192 	device_set_wakeup_enable(dev, en);
1193 }
1194 
bcmasp_wol_irq_destroy_per_intf(struct bcmasp_priv * priv)1195 static void bcmasp_wol_irq_destroy_per_intf(struct bcmasp_priv *priv)
1196 {
1197 	struct bcmasp_intf *intf;
1198 
1199 	list_for_each_entry(intf, &priv->intfs, list) {
1200 		if (intf->wol_irq > 0)
1201 			free_irq(intf->wol_irq, priv);
1202 	}
1203 }
1204 
bcmasp_eee_fixup(struct bcmasp_intf * intf,bool en)1205 static void bcmasp_eee_fixup(struct bcmasp_intf *intf, bool en)
1206 {
1207 	u32 reg, phy_lpi_overwrite;
1208 
1209 	reg = rx_edpkt_core_rl(intf->parent, ASP_EDPKT_SPARE_REG);
1210 	phy_lpi_overwrite = intf->internal_phy ? ASP_EDPKT_SPARE_REG_EPHY_LPI :
1211 			    ASP_EDPKT_SPARE_REG_GPHY_LPI;
1212 
1213 	if (en)
1214 		reg |= phy_lpi_overwrite;
1215 	else
1216 		reg &= ~phy_lpi_overwrite;
1217 
1218 	rx_edpkt_core_wl(intf->parent, reg, ASP_EDPKT_SPARE_REG);
1219 
1220 	usleep_range(50, 100);
1221 }
1222 
1223 static struct bcmasp_hw_info v20_hw_info = {
1224 	.rx_ctrl_flush = ASP_RX_CTRL_FLUSH,
1225 	.umac2fb = UMAC2FB_OFFSET,
1226 	.rx_ctrl_fb_out_frame_count = ASP_RX_CTRL_FB_OUT_FRAME_COUNT,
1227 	.rx_ctrl_fb_filt_out_frame_count = ASP_RX_CTRL_FB_FILT_OUT_FRAME_COUNT,
1228 	.rx_ctrl_fb_rx_fifo_depth = ASP_RX_CTRL_FB_RX_FIFO_DEPTH,
1229 };
1230 
1231 static const struct bcmasp_plat_data v20_plat_data = {
1232 	.init_wol = bcmasp_init_wol_per_intf,
1233 	.enable_wol = bcmasp_enable_wol_per_intf,
1234 	.destroy_wol = bcmasp_wol_irq_destroy_per_intf,
1235 	.core_clock_select = bcmasp_core_clock_select_one,
1236 	.hw_info = &v20_hw_info,
1237 };
1238 
1239 static struct bcmasp_hw_info v21_hw_info = {
1240 	.rx_ctrl_flush = ASP_RX_CTRL_FLUSH_2_1,
1241 	.umac2fb = UMAC2FB_OFFSET_2_1,
1242 	.rx_ctrl_fb_out_frame_count = ASP_RX_CTRL_FB_OUT_FRAME_COUNT_2_1,
1243 	.rx_ctrl_fb_filt_out_frame_count =
1244 		ASP_RX_CTRL_FB_FILT_OUT_FRAME_COUNT_2_1,
1245 	.rx_ctrl_fb_rx_fifo_depth = ASP_RX_CTRL_FB_RX_FIFO_DEPTH_2_1,
1246 };
1247 
1248 static const struct bcmasp_plat_data v21_plat_data = {
1249 	.init_wol = bcmasp_init_wol_shared,
1250 	.enable_wol = bcmasp_enable_wol_shared,
1251 	.destroy_wol = bcmasp_wol_irq_destroy_shared,
1252 	.core_clock_select = bcmasp_core_clock_select_one,
1253 	.hw_info = &v21_hw_info,
1254 };
1255 
1256 static const struct bcmasp_plat_data v22_plat_data = {
1257 	.init_wol = bcmasp_init_wol_shared,
1258 	.enable_wol = bcmasp_enable_wol_shared,
1259 	.destroy_wol = bcmasp_wol_irq_destroy_shared,
1260 	.core_clock_select = bcmasp_core_clock_select_many,
1261 	.hw_info = &v21_hw_info,
1262 	.eee_fixup = bcmasp_eee_fixup,
1263 };
1264 
bcmasp_set_pdata(struct bcmasp_priv * priv,const struct bcmasp_plat_data * pdata)1265 static void bcmasp_set_pdata(struct bcmasp_priv *priv, const struct bcmasp_plat_data *pdata)
1266 {
1267 	priv->init_wol = pdata->init_wol;
1268 	priv->enable_wol = pdata->enable_wol;
1269 	priv->destroy_wol = pdata->destroy_wol;
1270 	priv->core_clock_select = pdata->core_clock_select;
1271 	priv->eee_fixup = pdata->eee_fixup;
1272 	priv->hw_info = pdata->hw_info;
1273 }
1274 
1275 static const struct of_device_id bcmasp_of_match[] = {
1276 	{ .compatible = "brcm,asp-v2.0", .data = &v20_plat_data },
1277 	{ .compatible = "brcm,asp-v2.1", .data = &v21_plat_data },
1278 	{ .compatible = "brcm,asp-v2.2", .data = &v22_plat_data },
1279 	{ /* sentinel */ },
1280 };
1281 MODULE_DEVICE_TABLE(of, bcmasp_of_match);
1282 
1283 static const struct of_device_id bcmasp_mdio_of_match[] = {
1284 	{ .compatible = "brcm,asp-v2.2-mdio", },
1285 	{ .compatible = "brcm,asp-v2.1-mdio", },
1286 	{ .compatible = "brcm,asp-v2.0-mdio", },
1287 	{ /* sentinel */ },
1288 };
1289 MODULE_DEVICE_TABLE(of, bcmasp_mdio_of_match);
1290 
bcmasp_remove_intfs(struct bcmasp_priv * priv)1291 static void bcmasp_remove_intfs(struct bcmasp_priv *priv)
1292 {
1293 	struct bcmasp_intf *intf, *n;
1294 
1295 	list_for_each_entry_safe(intf, n, &priv->intfs, list) {
1296 		list_del(&intf->list);
1297 		bcmasp_interface_destroy(intf);
1298 	}
1299 }
1300 
bcmasp_probe(struct platform_device * pdev)1301 static int bcmasp_probe(struct platform_device *pdev)
1302 {
1303 	const struct bcmasp_plat_data *pdata;
1304 	struct device *dev = &pdev->dev;
1305 	struct device_node *ports_node;
1306 	struct bcmasp_priv *priv;
1307 	struct bcmasp_intf *intf;
1308 	int ret = 0, count = 0;
1309 	unsigned int i;
1310 
1311 	priv = devm_kzalloc(dev, sizeof(*priv), GFP_KERNEL);
1312 	if (!priv)
1313 		return -ENOMEM;
1314 
1315 	priv->irq = platform_get_irq(pdev, 0);
1316 	if (priv->irq <= 0)
1317 		return -EINVAL;
1318 
1319 	priv->clk = devm_clk_get_optional_enabled(dev, "sw_asp");
1320 	if (IS_ERR(priv->clk))
1321 		return dev_err_probe(dev, PTR_ERR(priv->clk),
1322 				     "failed to request clock\n");
1323 
1324 	/* Base from parent node */
1325 	priv->base = devm_platform_ioremap_resource(pdev, 0);
1326 	if (IS_ERR(priv->base))
1327 		return dev_err_probe(dev, PTR_ERR(priv->base), "failed to iomap\n");
1328 
1329 	ret = dma_set_mask_and_coherent(&pdev->dev, DMA_BIT_MASK(40));
1330 	if (ret)
1331 		return dev_err_probe(dev, ret, "unable to set DMA mask: %d\n", ret);
1332 
1333 	dev_set_drvdata(&pdev->dev, priv);
1334 	priv->pdev = pdev;
1335 	spin_lock_init(&priv->mda_lock);
1336 	spin_lock_init(&priv->clk_lock);
1337 	mutex_init(&priv->wol_lock);
1338 	mutex_init(&priv->net_lock);
1339 	INIT_LIST_HEAD(&priv->intfs);
1340 
1341 	pdata = device_get_match_data(&pdev->dev);
1342 	if (!pdata)
1343 		return dev_err_probe(dev, -EINVAL, "unable to find platform data\n");
1344 
1345 	bcmasp_set_pdata(priv, pdata);
1346 
1347 	/* Enable all clocks to ensure successful probing */
1348 	bcmasp_core_clock_set(priv, ASP_CTRL_CLOCK_CTRL_ASP_ALL_DISABLE, 0);
1349 
1350 	/* Switch to the main clock */
1351 	priv->core_clock_select(priv, false);
1352 
1353 	bcmasp_intr2_mask_set_all(priv);
1354 	bcmasp_intr2_clear_all(priv);
1355 
1356 	ret = devm_request_irq(&pdev->dev, priv->irq, bcmasp_isr, 0,
1357 			       pdev->name, priv);
1358 	if (ret)
1359 		return dev_err_probe(dev, ret, "failed to request ASP interrupt: %d", ret);
1360 
1361 	/* Register mdio child nodes */
1362 	of_platform_populate(dev->of_node, bcmasp_mdio_of_match, NULL, dev);
1363 
1364 	/* ASP specific initialization, Needs to be done regardless of
1365 	 * how many interfaces come up.
1366 	 */
1367 	bcmasp_core_init(priv);
1368 	bcmasp_core_init_filters(priv);
1369 
1370 	ports_node = of_find_node_by_name(dev->of_node, "ethernet-ports");
1371 	if (!ports_node) {
1372 		dev_warn(dev, "No ports found\n");
1373 		return -EINVAL;
1374 	}
1375 
1376 	i = 0;
1377 	for_each_available_child_of_node_scoped(ports_node, intf_node) {
1378 		intf = bcmasp_interface_create(priv, intf_node, i);
1379 		if (!intf) {
1380 			dev_err(dev, "Cannot create eth interface %d\n", i);
1381 			bcmasp_remove_intfs(priv);
1382 			ret = -ENOMEM;
1383 			goto of_put_exit;
1384 		}
1385 		list_add_tail(&intf->list, &priv->intfs);
1386 		i++;
1387 	}
1388 
1389 	/* Check and enable WoL */
1390 	priv->init_wol(priv);
1391 
1392 	/* Drop the clock reference count now and let ndo_open()/ndo_close()
1393 	 * manage it for us from now on.
1394 	 */
1395 	bcmasp_core_clock_set(priv, 0, ASP_CTRL_CLOCK_CTRL_ASP_ALL_DISABLE);
1396 
1397 	clk_disable_unprepare(priv->clk);
1398 
1399 	/* Now do the registration of the network ports which will take care
1400 	 * of managing the clock properly.
1401 	 */
1402 	list_for_each_entry(intf, &priv->intfs, list) {
1403 		ret = register_netdev(intf->ndev);
1404 		if (ret) {
1405 			netdev_err(intf->ndev,
1406 				   "failed to register net_device: %d\n", ret);
1407 			priv->destroy_wol(priv);
1408 			bcmasp_remove_intfs(priv);
1409 			goto of_put_exit;
1410 		}
1411 		count++;
1412 	}
1413 
1414 	dev_info(dev, "Initialized %d port(s)\n", count);
1415 
1416 of_put_exit:
1417 	of_node_put(ports_node);
1418 	return ret;
1419 }
1420 
bcmasp_remove(struct platform_device * pdev)1421 static void bcmasp_remove(struct platform_device *pdev)
1422 {
1423 	struct bcmasp_priv *priv = dev_get_drvdata(&pdev->dev);
1424 
1425 	if (!priv)
1426 		return;
1427 
1428 	priv->destroy_wol(priv);
1429 	bcmasp_remove_intfs(priv);
1430 }
1431 
bcmasp_shutdown(struct platform_device * pdev)1432 static void bcmasp_shutdown(struct platform_device *pdev)
1433 {
1434 	bcmasp_remove(pdev);
1435 }
1436 
bcmasp_suspend(struct device * d)1437 static int __maybe_unused bcmasp_suspend(struct device *d)
1438 {
1439 	struct bcmasp_priv *priv = dev_get_drvdata(d);
1440 	struct bcmasp_intf *intf;
1441 	int ret;
1442 
1443 	list_for_each_entry(intf, &priv->intfs, list) {
1444 		ret = bcmasp_interface_suspend(intf);
1445 		if (ret)
1446 			break;
1447 	}
1448 
1449 	ret = clk_prepare_enable(priv->clk);
1450 	if (ret)
1451 		return ret;
1452 
1453 	/* Whether Wake-on-LAN is enabled or not, we can always disable
1454 	 * the shared TX clock
1455 	 */
1456 	bcmasp_core_clock_set(priv, 0, ASP_CTRL_CLOCK_CTRL_ASP_TX_DISABLE);
1457 
1458 	priv->core_clock_select(priv, true);
1459 
1460 	clk_disable_unprepare(priv->clk);
1461 
1462 	return ret;
1463 }
1464 
bcmasp_resume(struct device * d)1465 static int __maybe_unused bcmasp_resume(struct device *d)
1466 {
1467 	struct bcmasp_priv *priv = dev_get_drvdata(d);
1468 	struct bcmasp_intf *intf;
1469 	int ret;
1470 
1471 	ret = clk_prepare_enable(priv->clk);
1472 	if (ret)
1473 		return ret;
1474 
1475 	/* Switch to the main clock domain */
1476 	priv->core_clock_select(priv, false);
1477 
1478 	/* Re-enable all clocks for re-initialization */
1479 	bcmasp_core_clock_set(priv, ASP_CTRL_CLOCK_CTRL_ASP_ALL_DISABLE, 0);
1480 
1481 	bcmasp_core_init(priv);
1482 	bcmasp_core_init_filters(priv);
1483 
1484 	/* And disable them to let the network devices take care of them */
1485 	bcmasp_core_clock_set(priv, 0, ASP_CTRL_CLOCK_CTRL_ASP_ALL_DISABLE);
1486 
1487 	clk_disable_unprepare(priv->clk);
1488 
1489 	list_for_each_entry(intf, &priv->intfs, list) {
1490 		ret = bcmasp_interface_resume(intf);
1491 		if (ret)
1492 			break;
1493 	}
1494 
1495 	return ret;
1496 }
1497 
1498 static SIMPLE_DEV_PM_OPS(bcmasp_pm_ops,
1499 			 bcmasp_suspend, bcmasp_resume);
1500 
1501 static struct platform_driver bcmasp_driver = {
1502 	.probe = bcmasp_probe,
1503 	.remove_new = bcmasp_remove,
1504 	.shutdown = bcmasp_shutdown,
1505 	.driver = {
1506 		.name = "brcm,asp-v2",
1507 		.of_match_table = bcmasp_of_match,
1508 		.pm = &bcmasp_pm_ops,
1509 	},
1510 };
1511 module_platform_driver(bcmasp_driver);
1512 
1513 MODULE_DESCRIPTION("Broadcom ASP 2.0 Ethernet controller driver");
1514 MODULE_ALIAS("platform:brcm,asp-v2");
1515 MODULE_LICENSE("GPL");
1516