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 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 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 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 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 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 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 76 static void bcmasp_intr2_clear_all(struct bcmasp_priv *priv) 77 { 78 intr2_core_wl(priv, 0xffffffff, ASP_INTR2_CLEAR); 79 } 80 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 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 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 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 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 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 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 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 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 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 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 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 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 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 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 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 */ 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 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 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 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 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 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 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 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 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 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 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 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 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 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 */ 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 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 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 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 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 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 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 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 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 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 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 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 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 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 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 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 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 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 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 1432 static void bcmasp_shutdown(struct platform_device *pdev) 1433 { 1434 bcmasp_remove(pdev); 1435 } 1436 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 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 = 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