1 // SPDX-License-Identifier: GPL-2.0+ 2 /* Microchip Sparx5 Switch driver 3 * 4 * Copyright (c) 2021 Microchip Technology Inc. and its subsidiaries. 5 */ 6 7 #include <linux/module.h> 8 #include <linux/phy/phy.h> 9 #include <net/dcbnl.h> 10 11 #include "sparx5_main_regs.h" 12 #include "sparx5_main.h" 13 #include "sparx5_port.h" 14 15 #define SPX5_ETYPE_TAG_C 0x8100 16 #define SPX5_ETYPE_TAG_S 0x88a8 17 18 #define SPX5_WAIT_US 1000 19 #define SPX5_WAIT_MAX_US 2000 20 21 enum port_error { 22 SPX5_PERR_SPEED, 23 SPX5_PERR_IFTYPE, 24 }; 25 26 #define PAUSE_DISCARD 0xC 27 #define ETH_MAXLEN (ETH_DATA_LEN + ETH_HLEN + ETH_FCS_LEN) 28 29 static void decode_sgmii_word(u16 lp_abil, struct sparx5_port_status *status) 30 { 31 status->an_complete = true; 32 if (!(lp_abil & LPA_SGMII_LINK)) { 33 status->link = false; 34 return; 35 } 36 37 switch (lp_abil & LPA_SGMII_SPD_MASK) { 38 case LPA_SGMII_10: 39 status->speed = SPEED_10; 40 break; 41 case LPA_SGMII_100: 42 status->speed = SPEED_100; 43 break; 44 case LPA_SGMII_1000: 45 status->speed = SPEED_1000; 46 break; 47 default: 48 status->link = false; 49 return; 50 } 51 if (lp_abil & LPA_SGMII_FULL_DUPLEX) 52 status->duplex = DUPLEX_FULL; 53 else 54 status->duplex = DUPLEX_HALF; 55 } 56 57 static void decode_cl37_word(u16 lp_abil, uint16_t ld_abil, struct sparx5_port_status *status) 58 { 59 status->link = !(lp_abil & ADVERTISE_RFAULT) && status->link; 60 status->an_complete = true; 61 status->duplex = (ADVERTISE_1000XFULL & lp_abil) ? 62 DUPLEX_FULL : DUPLEX_UNKNOWN; // 1G HDX not supported 63 64 if ((ld_abil & ADVERTISE_1000XPAUSE) && 65 (lp_abil & ADVERTISE_1000XPAUSE)) { 66 status->pause = MLO_PAUSE_RX | MLO_PAUSE_TX; 67 } else if ((ld_abil & ADVERTISE_1000XPSE_ASYM) && 68 (lp_abil & ADVERTISE_1000XPSE_ASYM)) { 69 status->pause |= (lp_abil & ADVERTISE_1000XPAUSE) ? 70 MLO_PAUSE_TX : 0; 71 status->pause |= (ld_abil & ADVERTISE_1000XPAUSE) ? 72 MLO_PAUSE_RX : 0; 73 } else { 74 status->pause = MLO_PAUSE_NONE; 75 } 76 } 77 78 static int sparx5_get_dev2g5_status(struct sparx5 *sparx5, 79 struct sparx5_port *port, 80 struct sparx5_port_status *status) 81 { 82 u32 portno = port->portno; 83 u16 lp_adv, ld_adv; 84 u32 value; 85 86 /* Get PCS Link down sticky */ 87 value = spx5_rd(sparx5, DEV2G5_PCS1G_STICKY(portno)); 88 status->link_down = DEV2G5_PCS1G_STICKY_LINK_DOWN_STICKY_GET(value); 89 if (status->link_down) /* Clear the sticky */ 90 spx5_wr(value, sparx5, DEV2G5_PCS1G_STICKY(portno)); 91 92 /* Get both current Link and Sync status */ 93 value = spx5_rd(sparx5, DEV2G5_PCS1G_LINK_STATUS(portno)); 94 status->link = DEV2G5_PCS1G_LINK_STATUS_LINK_STATUS_GET(value) && 95 DEV2G5_PCS1G_LINK_STATUS_SYNC_STATUS_GET(value); 96 97 if (port->conf.portmode == PHY_INTERFACE_MODE_1000BASEX) 98 status->speed = SPEED_1000; 99 else if (port->conf.portmode == PHY_INTERFACE_MODE_2500BASEX) 100 status->speed = SPEED_2500; 101 102 status->duplex = DUPLEX_FULL; 103 104 /* Get PCS ANEG status register */ 105 value = spx5_rd(sparx5, DEV2G5_PCS1G_ANEG_STATUS(portno)); 106 107 /* Aneg complete provides more information */ 108 if (DEV2G5_PCS1G_ANEG_STATUS_ANEG_COMPLETE_GET(value)) { 109 lp_adv = DEV2G5_PCS1G_ANEG_STATUS_LP_ADV_ABILITY_GET(value); 110 if (port->conf.portmode == PHY_INTERFACE_MODE_SGMII) { 111 decode_sgmii_word(lp_adv, status); 112 } else { 113 value = spx5_rd(sparx5, DEV2G5_PCS1G_ANEG_CFG(portno)); 114 ld_adv = DEV2G5_PCS1G_ANEG_CFG_ADV_ABILITY_GET(value); 115 decode_cl37_word(lp_adv, ld_adv, status); 116 } 117 } 118 return 0; 119 } 120 121 static int sparx5_get_sfi_status(struct sparx5 *sparx5, 122 struct sparx5_port *port, 123 struct sparx5_port_status *status) 124 { 125 bool high_speed_dev = sparx5_is_baser(port->conf.portmode); 126 u32 portno = port->portno; 127 u32 value, dev, tinst; 128 void __iomem *inst; 129 130 if (!high_speed_dev) { 131 netdev_err(port->ndev, "error: low speed and SFI mode\n"); 132 return -EINVAL; 133 } 134 135 dev = sparx5_to_high_dev(portno); 136 tinst = sparx5_port_dev_index(portno); 137 inst = spx5_inst_get(sparx5, dev, tinst); 138 139 value = spx5_inst_rd(inst, DEV10G_MAC_TX_MONITOR_STICKY(0)); 140 if (value != DEV10G_MAC_TX_MONITOR_STICKY_IDLE_STATE_STICKY) { 141 /* The link is or has been down. Clear the sticky bit */ 142 status->link_down = 1; 143 spx5_inst_wr(0xffffffff, inst, DEV10G_MAC_TX_MONITOR_STICKY(0)); 144 value = spx5_inst_rd(inst, DEV10G_MAC_TX_MONITOR_STICKY(0)); 145 } 146 status->link = (value == DEV10G_MAC_TX_MONITOR_STICKY_IDLE_STATE_STICKY); 147 status->duplex = DUPLEX_FULL; 148 if (port->conf.portmode == PHY_INTERFACE_MODE_5GBASER) 149 status->speed = SPEED_5000; 150 else if (port->conf.portmode == PHY_INTERFACE_MODE_10GBASER) 151 status->speed = SPEED_10000; 152 else 153 status->speed = SPEED_25000; 154 155 return 0; 156 } 157 158 /* Get link status of 1000Base-X/in-band and SFI ports. 159 */ 160 int sparx5_get_port_status(struct sparx5 *sparx5, 161 struct sparx5_port *port, 162 struct sparx5_port_status *status) 163 { 164 memset(status, 0, sizeof(*status)); 165 status->speed = port->conf.speed; 166 if (port->conf.power_down) { 167 status->link = false; 168 return 0; 169 } 170 switch (port->conf.portmode) { 171 case PHY_INTERFACE_MODE_SGMII: 172 case PHY_INTERFACE_MODE_QSGMII: 173 case PHY_INTERFACE_MODE_1000BASEX: 174 case PHY_INTERFACE_MODE_2500BASEX: 175 return sparx5_get_dev2g5_status(sparx5, port, status); 176 case PHY_INTERFACE_MODE_5GBASER: 177 case PHY_INTERFACE_MODE_10GBASER: 178 case PHY_INTERFACE_MODE_25GBASER: 179 return sparx5_get_sfi_status(sparx5, port, status); 180 case PHY_INTERFACE_MODE_NA: 181 return 0; 182 default: 183 netdev_err(port->ndev, "Status not supported"); 184 return -ENODEV; 185 } 186 return 0; 187 } 188 189 static int sparx5_port_error(struct sparx5_port *port, 190 struct sparx5_port_config *conf, 191 enum port_error errtype) 192 { 193 switch (errtype) { 194 case SPX5_PERR_SPEED: 195 netdev_err(port->ndev, 196 "Interface does not support speed: %u: for %s\n", 197 conf->speed, phy_modes(conf->portmode)); 198 break; 199 case SPX5_PERR_IFTYPE: 200 netdev_err(port->ndev, 201 "Switch port does not support interface type: %s\n", 202 phy_modes(conf->portmode)); 203 break; 204 default: 205 netdev_err(port->ndev, 206 "Interface configuration error\n"); 207 } 208 209 return -EINVAL; 210 } 211 212 static int sparx5_port_verify_speed(struct sparx5 *sparx5, 213 struct sparx5_port *port, 214 struct sparx5_port_config *conf) 215 { 216 if ((sparx5_port_is_2g5(port->portno) && 217 conf->speed > SPEED_2500) || 218 (sparx5_port_is_5g(port->portno) && 219 conf->speed > SPEED_5000) || 220 (sparx5_port_is_10g(port->portno) && 221 conf->speed > SPEED_10000)) 222 return sparx5_port_error(port, conf, SPX5_PERR_SPEED); 223 224 switch (conf->portmode) { 225 case PHY_INTERFACE_MODE_NA: 226 return -EINVAL; 227 case PHY_INTERFACE_MODE_1000BASEX: 228 if (conf->speed != SPEED_1000 || 229 sparx5_port_is_2g5(port->portno)) 230 return sparx5_port_error(port, conf, SPX5_PERR_SPEED); 231 if (sparx5_port_is_2g5(port->portno)) 232 return sparx5_port_error(port, conf, SPX5_PERR_IFTYPE); 233 break; 234 case PHY_INTERFACE_MODE_2500BASEX: 235 if (conf->speed != SPEED_2500 || 236 sparx5_port_is_2g5(port->portno)) 237 return sparx5_port_error(port, conf, SPX5_PERR_SPEED); 238 break; 239 case PHY_INTERFACE_MODE_QSGMII: 240 if (port->portno > 47) 241 return sparx5_port_error(port, conf, SPX5_PERR_IFTYPE); 242 fallthrough; 243 case PHY_INTERFACE_MODE_SGMII: 244 if (conf->speed != SPEED_1000 && 245 conf->speed != SPEED_100 && 246 conf->speed != SPEED_10 && 247 conf->speed != SPEED_2500) 248 return sparx5_port_error(port, conf, SPX5_PERR_SPEED); 249 break; 250 case PHY_INTERFACE_MODE_5GBASER: 251 case PHY_INTERFACE_MODE_10GBASER: 252 case PHY_INTERFACE_MODE_25GBASER: 253 if ((conf->speed != SPEED_5000 && 254 conf->speed != SPEED_10000 && 255 conf->speed != SPEED_25000)) 256 return sparx5_port_error(port, conf, SPX5_PERR_SPEED); 257 break; 258 default: 259 return sparx5_port_error(port, conf, SPX5_PERR_IFTYPE); 260 } 261 return 0; 262 } 263 264 static bool sparx5_dev_change(struct sparx5 *sparx5, 265 struct sparx5_port *port, 266 struct sparx5_port_config *conf) 267 { 268 return sparx5_is_baser(port->conf.portmode) ^ 269 sparx5_is_baser(conf->portmode); 270 } 271 272 static int sparx5_port_flush_poll(struct sparx5 *sparx5, u32 portno) 273 { 274 u32 value, resource, prio, delay_cnt = 0; 275 bool poll_src = true; 276 char *mem = ""; 277 278 /* Resource == 0: Memory tracked per source (SRC-MEM) 279 * Resource == 1: Frame references tracked per source (SRC-REF) 280 * Resource == 2: Memory tracked per destination (DST-MEM) 281 * Resource == 3: Frame references tracked per destination. (DST-REF) 282 */ 283 while (1) { 284 bool empty = true; 285 286 for (resource = 0; resource < (poll_src ? 2 : 1); resource++) { 287 u32 base; 288 289 base = (resource == 0 ? 2048 : 0) + SPX5_PRIOS * portno; 290 for (prio = 0; prio < SPX5_PRIOS; prio++) { 291 value = spx5_rd(sparx5, 292 QRES_RES_STAT(base + prio)); 293 if (value) { 294 mem = resource == 0 ? 295 "DST-MEM" : "SRC-MEM"; 296 empty = false; 297 } 298 } 299 } 300 301 if (empty) 302 break; 303 304 if (delay_cnt++ == 2000) { 305 dev_err(sparx5->dev, 306 "Flush timeout port %u. %s queue not empty\n", 307 portno, mem); 308 return -EINVAL; 309 } 310 311 usleep_range(SPX5_WAIT_US, SPX5_WAIT_MAX_US); 312 } 313 return 0; 314 } 315 316 static int sparx5_port_disable(struct sparx5 *sparx5, struct sparx5_port *port, bool high_spd_dev) 317 { 318 u32 tinst = high_spd_dev ? 319 sparx5_port_dev_index(port->portno) : port->portno; 320 u32 dev = high_spd_dev ? 321 sparx5_to_high_dev(port->portno) : TARGET_DEV2G5; 322 void __iomem *devinst = spx5_inst_get(sparx5, dev, tinst); 323 u32 spd = port->conf.speed; 324 u32 spd_prm; 325 int err; 326 327 if (high_spd_dev) { 328 /* 1: Reset the PCS Rx clock domain */ 329 spx5_inst_rmw(DEV10G_DEV_RST_CTRL_PCS_RX_RST, 330 DEV10G_DEV_RST_CTRL_PCS_RX_RST, 331 devinst, 332 DEV10G_DEV_RST_CTRL(0)); 333 334 /* 2: Disable MAC frame reception */ 335 spx5_inst_rmw(0, 336 DEV10G_MAC_ENA_CFG_RX_ENA, 337 devinst, 338 DEV10G_MAC_ENA_CFG(0)); 339 } else { 340 /* 1: Reset the PCS Rx clock domain */ 341 spx5_inst_rmw(DEV2G5_DEV_RST_CTRL_PCS_RX_RST, 342 DEV2G5_DEV_RST_CTRL_PCS_RX_RST, 343 devinst, 344 DEV2G5_DEV_RST_CTRL(0)); 345 /* 2: Disable MAC frame reception */ 346 spx5_inst_rmw(0, 347 DEV2G5_MAC_ENA_CFG_RX_ENA, 348 devinst, 349 DEV2G5_MAC_ENA_CFG(0)); 350 } 351 /* 3: Disable traffic being sent to or from switch port->portno */ 352 spx5_rmw(0, 353 QFWD_SWITCH_PORT_MODE_PORT_ENA, 354 sparx5, 355 QFWD_SWITCH_PORT_MODE(port->portno)); 356 357 /* 4: Disable dequeuing from the egress queues */ 358 spx5_rmw(HSCH_PORT_MODE_DEQUEUE_DIS, 359 HSCH_PORT_MODE_DEQUEUE_DIS, 360 sparx5, 361 HSCH_PORT_MODE(port->portno)); 362 363 /* 5: Disable Flowcontrol */ 364 spx5_rmw(QSYS_PAUSE_CFG_PAUSE_STOP_SET(0xFFF - 1), 365 QSYS_PAUSE_CFG_PAUSE_STOP, 366 sparx5, 367 QSYS_PAUSE_CFG(port->portno)); 368 369 spd_prm = spd == SPEED_10 ? 1000 : spd == SPEED_100 ? 100 : 10; 370 /* 6: Wait while the last frame is exiting the queues */ 371 usleep_range(8 * spd_prm, 10 * spd_prm); 372 373 /* 7: Flush the queues accociated with the port->portno */ 374 spx5_rmw(HSCH_FLUSH_CTRL_FLUSH_PORT_SET(port->portno) | 375 HSCH_FLUSH_CTRL_FLUSH_DST_SET(1) | 376 HSCH_FLUSH_CTRL_FLUSH_SRC_SET(1) | 377 HSCH_FLUSH_CTRL_FLUSH_ENA_SET(1), 378 HSCH_FLUSH_CTRL_FLUSH_PORT | 379 HSCH_FLUSH_CTRL_FLUSH_DST | 380 HSCH_FLUSH_CTRL_FLUSH_SRC | 381 HSCH_FLUSH_CTRL_FLUSH_ENA, 382 sparx5, 383 HSCH_FLUSH_CTRL); 384 385 /* 8: Enable dequeuing from the egress queues */ 386 spx5_rmw(0, 387 HSCH_PORT_MODE_DEQUEUE_DIS, 388 sparx5, 389 HSCH_PORT_MODE(port->portno)); 390 391 /* 9: Wait until flushing is complete */ 392 err = sparx5_port_flush_poll(sparx5, port->portno); 393 if (err) 394 return err; 395 396 /* 10: Reset the MAC clock domain */ 397 if (high_spd_dev) { 398 spx5_inst_rmw(DEV10G_DEV_RST_CTRL_PCS_TX_RST_SET(1) | 399 DEV10G_DEV_RST_CTRL_MAC_RX_RST_SET(1) | 400 DEV10G_DEV_RST_CTRL_MAC_TX_RST_SET(1), 401 DEV10G_DEV_RST_CTRL_PCS_TX_RST | 402 DEV10G_DEV_RST_CTRL_MAC_RX_RST | 403 DEV10G_DEV_RST_CTRL_MAC_TX_RST, 404 devinst, 405 DEV10G_DEV_RST_CTRL(0)); 406 407 } else { 408 spx5_inst_rmw(DEV2G5_DEV_RST_CTRL_SPEED_SEL_SET(3) | 409 DEV2G5_DEV_RST_CTRL_PCS_TX_RST_SET(1) | 410 DEV2G5_DEV_RST_CTRL_PCS_RX_RST_SET(1) | 411 DEV2G5_DEV_RST_CTRL_MAC_TX_RST_SET(1) | 412 DEV2G5_DEV_RST_CTRL_MAC_RX_RST_SET(1), 413 DEV2G5_DEV_RST_CTRL_SPEED_SEL | 414 DEV2G5_DEV_RST_CTRL_PCS_TX_RST | 415 DEV2G5_DEV_RST_CTRL_PCS_RX_RST | 416 DEV2G5_DEV_RST_CTRL_MAC_TX_RST | 417 DEV2G5_DEV_RST_CTRL_MAC_RX_RST, 418 devinst, 419 DEV2G5_DEV_RST_CTRL(0)); 420 } 421 /* 11: Clear flushing */ 422 spx5_rmw(HSCH_FLUSH_CTRL_FLUSH_PORT_SET(port->portno) | 423 HSCH_FLUSH_CTRL_FLUSH_ENA_SET(0), 424 HSCH_FLUSH_CTRL_FLUSH_PORT | 425 HSCH_FLUSH_CTRL_FLUSH_ENA, 426 sparx5, 427 HSCH_FLUSH_CTRL); 428 429 if (high_spd_dev) { 430 u32 pcs = sparx5_to_pcs_dev(port->portno); 431 void __iomem *pcsinst = spx5_inst_get(sparx5, pcs, tinst); 432 433 /* 12: Disable 5G/10G/25 BaseR PCS */ 434 spx5_inst_rmw(PCS10G_BR_PCS_CFG_PCS_ENA_SET(0), 435 PCS10G_BR_PCS_CFG_PCS_ENA, 436 pcsinst, 437 PCS10G_BR_PCS_CFG(0)); 438 439 if (sparx5_port_is_25g(port->portno)) 440 /* Disable 25G PCS */ 441 spx5_rmw(DEV25G_PCS25G_CFG_PCS25G_ENA_SET(0), 442 DEV25G_PCS25G_CFG_PCS25G_ENA, 443 sparx5, 444 DEV25G_PCS25G_CFG(tinst)); 445 } else { 446 /* 12: Disable 1G PCS */ 447 spx5_rmw(DEV2G5_PCS1G_CFG_PCS_ENA_SET(0), 448 DEV2G5_PCS1G_CFG_PCS_ENA, 449 sparx5, 450 DEV2G5_PCS1G_CFG(port->portno)); 451 } 452 453 /* The port is now flushed and disabled */ 454 return 0; 455 } 456 457 static int sparx5_port_fifo_sz(struct sparx5 *sparx5, 458 u32 portno, u32 speed) 459 { 460 u32 sys_clk = sparx5_clk_period(sparx5->coreclock); 461 const u32 taxi_dist[SPX5_PORTS_ALL] = { 462 6, 8, 10, 6, 8, 10, 6, 8, 10, 6, 8, 10, 463 4, 4, 4, 4, 464 11, 12, 13, 14, 15, 16, 17, 18, 465 11, 12, 13, 14, 15, 16, 17, 18, 466 11, 12, 13, 14, 15, 16, 17, 18, 467 11, 12, 13, 14, 15, 16, 17, 18, 468 4, 6, 8, 4, 6, 8, 6, 8, 469 2, 2, 2, 2, 2, 2, 2, 4, 2 470 }; 471 u32 mac_per = 6400, tmp1, tmp2, tmp3; 472 u32 fifo_width = 16; 473 u32 mac_width = 8; 474 u32 addition = 0; 475 476 switch (speed) { 477 case SPEED_25000: 478 return 0; 479 case SPEED_10000: 480 mac_per = 6400; 481 mac_width = 8; 482 addition = 1; 483 break; 484 case SPEED_5000: 485 mac_per = 12800; 486 mac_width = 8; 487 addition = 0; 488 break; 489 case SPEED_2500: 490 mac_per = 3200; 491 mac_width = 1; 492 addition = 0; 493 break; 494 case SPEED_1000: 495 mac_per = 8000; 496 mac_width = 1; 497 addition = 0; 498 break; 499 case SPEED_100: 500 case SPEED_10: 501 return 1; 502 default: 503 break; 504 } 505 506 tmp1 = 1000 * mac_width / fifo_width; 507 tmp2 = 3000 + ((12000 + 2 * taxi_dist[portno] * 1000) 508 * sys_clk / mac_per); 509 tmp3 = tmp1 * tmp2 / 1000; 510 return (tmp3 + 2000 + 999) / 1000 + addition; 511 } 512 513 /* Configure port muxing: 514 * QSGMII: 4x2G5 devices 515 */ 516 static int sparx5_port_mux_set(struct sparx5 *sparx5, 517 struct sparx5_port *port, 518 struct sparx5_port_config *conf) 519 { 520 u32 portno = port->portno; 521 u32 inst; 522 523 if (port->conf.portmode == conf->portmode) 524 return 0; /* Nothing to do */ 525 526 switch (conf->portmode) { 527 case PHY_INTERFACE_MODE_QSGMII: /* QSGMII: 4x2G5 devices. Mode Q' */ 528 inst = (portno - portno % 4) / 4; 529 spx5_rmw(BIT(inst), 530 BIT(inst), 531 sparx5, 532 PORT_CONF_QSGMII_ENA); 533 534 if ((portno / 4 % 2) == 0) { 535 /* Affects d0-d3,d8-d11..d40-d43 */ 536 spx5_rmw(PORT_CONF_USGMII_CFG_BYPASS_SCRAM_SET(1) | 537 PORT_CONF_USGMII_CFG_BYPASS_DESCRAM_SET(1) | 538 PORT_CONF_USGMII_CFG_QUAD_MODE_SET(1), 539 PORT_CONF_USGMII_CFG_BYPASS_SCRAM | 540 PORT_CONF_USGMII_CFG_BYPASS_DESCRAM | 541 PORT_CONF_USGMII_CFG_QUAD_MODE, 542 sparx5, 543 PORT_CONF_USGMII_CFG((portno / 8))); 544 } 545 break; 546 default: 547 break; 548 } 549 return 0; 550 } 551 552 static int sparx5_port_max_tags_set(struct sparx5 *sparx5, 553 struct sparx5_port *port) 554 { 555 enum sparx5_port_max_tags max_tags = port->max_vlan_tags; 556 int tag_ct = max_tags == SPX5_PORT_MAX_TAGS_ONE ? 1 : 557 max_tags == SPX5_PORT_MAX_TAGS_TWO ? 2 : 0; 558 bool dtag = max_tags == SPX5_PORT_MAX_TAGS_TWO; 559 enum sparx5_vlan_port_type vlan_type = port->vlan_type; 560 bool dotag = max_tags != SPX5_PORT_MAX_TAGS_NONE; 561 u32 dev = sparx5_to_high_dev(port->portno); 562 u32 tinst = sparx5_port_dev_index(port->portno); 563 void __iomem *inst = spx5_inst_get(sparx5, dev, tinst); 564 u32 etype; 565 566 etype = (vlan_type == SPX5_VLAN_PORT_TYPE_S_CUSTOM ? 567 port->custom_etype : 568 vlan_type == SPX5_VLAN_PORT_TYPE_C ? 569 SPX5_ETYPE_TAG_C : SPX5_ETYPE_TAG_S); 570 571 spx5_wr(DEV2G5_MAC_TAGS_CFG_TAG_ID_SET(etype) | 572 DEV2G5_MAC_TAGS_CFG_PB_ENA_SET(dtag) | 573 DEV2G5_MAC_TAGS_CFG_VLAN_AWR_ENA_SET(dotag) | 574 DEV2G5_MAC_TAGS_CFG_VLAN_LEN_AWR_ENA_SET(dotag), 575 sparx5, 576 DEV2G5_MAC_TAGS_CFG(port->portno)); 577 578 if (sparx5_port_is_2g5(port->portno)) 579 return 0; 580 581 spx5_inst_rmw(DEV10G_MAC_TAGS_CFG_TAG_ID_SET(etype) | 582 DEV10G_MAC_TAGS_CFG_TAG_ENA_SET(dotag), 583 DEV10G_MAC_TAGS_CFG_TAG_ID | 584 DEV10G_MAC_TAGS_CFG_TAG_ENA, 585 inst, 586 DEV10G_MAC_TAGS_CFG(0, 0)); 587 588 spx5_inst_rmw(DEV10G_MAC_NUM_TAGS_CFG_NUM_TAGS_SET(tag_ct), 589 DEV10G_MAC_NUM_TAGS_CFG_NUM_TAGS, 590 inst, 591 DEV10G_MAC_NUM_TAGS_CFG(0)); 592 593 spx5_inst_rmw(DEV10G_MAC_MAXLEN_CFG_MAX_LEN_TAG_CHK_SET(dotag), 594 DEV10G_MAC_MAXLEN_CFG_MAX_LEN_TAG_CHK, 595 inst, 596 DEV10G_MAC_MAXLEN_CFG(0)); 597 return 0; 598 } 599 600 int sparx5_port_fwd_urg(struct sparx5 *sparx5, u32 speed) 601 { 602 u32 clk_period_ps = 1600; /* 625Mhz for now */ 603 u32 urg = 672000; 604 605 switch (speed) { 606 case SPEED_10: 607 case SPEED_100: 608 case SPEED_1000: 609 urg = 672000; 610 break; 611 case SPEED_2500: 612 urg = 270000; 613 break; 614 case SPEED_5000: 615 urg = 135000; 616 break; 617 case SPEED_10000: 618 urg = 67200; 619 break; 620 case SPEED_25000: 621 urg = 27000; 622 break; 623 } 624 return urg / clk_period_ps - 1; 625 } 626 627 static u16 sparx5_wm_enc(u16 value) 628 { 629 if (value >= 2048) 630 return 2048 + value / 16; 631 632 return value; 633 } 634 635 static int sparx5_port_fc_setup(struct sparx5 *sparx5, 636 struct sparx5_port *port, 637 struct sparx5_port_config *conf) 638 { 639 bool fc_obey = conf->pause & MLO_PAUSE_RX ? 1 : 0; 640 u32 pause_stop = 0xFFF - 1; /* FC gen disabled */ 641 642 if (conf->pause & MLO_PAUSE_TX) 643 pause_stop = sparx5_wm_enc(4 * (ETH_MAXLEN / 644 SPX5_BUFFER_CELL_SZ)); 645 646 /* Set HDX flowcontrol */ 647 spx5_rmw(DSM_MAC_CFG_HDX_BACKPREASSURE_SET(conf->duplex == DUPLEX_HALF), 648 DSM_MAC_CFG_HDX_BACKPREASSURE, 649 sparx5, 650 DSM_MAC_CFG(port->portno)); 651 652 /* Obey flowcontrol */ 653 spx5_rmw(DSM_RX_PAUSE_CFG_RX_PAUSE_EN_SET(fc_obey), 654 DSM_RX_PAUSE_CFG_RX_PAUSE_EN, 655 sparx5, 656 DSM_RX_PAUSE_CFG(port->portno)); 657 658 /* Disable forward pressure */ 659 spx5_rmw(QSYS_FWD_PRESSURE_FWD_PRESSURE_DIS_SET(fc_obey), 660 QSYS_FWD_PRESSURE_FWD_PRESSURE_DIS, 661 sparx5, 662 QSYS_FWD_PRESSURE(port->portno)); 663 664 /* Generate pause frames */ 665 spx5_rmw(QSYS_PAUSE_CFG_PAUSE_STOP_SET(pause_stop), 666 QSYS_PAUSE_CFG_PAUSE_STOP, 667 sparx5, 668 QSYS_PAUSE_CFG(port->portno)); 669 670 return 0; 671 } 672 673 static u16 sparx5_get_aneg_word(struct sparx5_port_config *conf) 674 { 675 if (conf->portmode == PHY_INTERFACE_MODE_1000BASEX) /* cl-37 aneg */ 676 return (conf->pause_adv | ADVERTISE_LPACK | ADVERTISE_1000XFULL); 677 else 678 return 1; /* Enable SGMII Aneg */ 679 } 680 681 int sparx5_serdes_set(struct sparx5 *sparx5, 682 struct sparx5_port *port, 683 struct sparx5_port_config *conf) 684 { 685 int portmode, err, speed = conf->speed; 686 687 if (conf->portmode == PHY_INTERFACE_MODE_QSGMII && 688 ((port->portno % 4) != 0)) { 689 return 0; 690 } 691 if (sparx5_is_baser(conf->portmode)) { 692 if (conf->portmode == PHY_INTERFACE_MODE_25GBASER) 693 speed = SPEED_25000; 694 else if (conf->portmode == PHY_INTERFACE_MODE_10GBASER) 695 speed = SPEED_10000; 696 else 697 speed = SPEED_5000; 698 } 699 700 err = phy_set_media(port->serdes, conf->media); 701 if (err) 702 return err; 703 if (speed > 0) { 704 err = phy_set_speed(port->serdes, speed); 705 if (err) 706 return err; 707 } 708 if (conf->serdes_reset) { 709 err = phy_reset(port->serdes); 710 if (err) 711 return err; 712 } 713 714 /* Configure SerDes with port parameters 715 * For BaseR, the serdes driver supports 10GGBASE-R and speed 5G/10G/25G 716 */ 717 portmode = conf->portmode; 718 if (sparx5_is_baser(conf->portmode)) 719 portmode = PHY_INTERFACE_MODE_10GBASER; 720 err = phy_set_mode_ext(port->serdes, PHY_MODE_ETHERNET, portmode); 721 if (err) 722 return err; 723 conf->serdes_reset = false; 724 return err; 725 } 726 727 static int sparx5_port_pcs_low_set(struct sparx5 *sparx5, 728 struct sparx5_port *port, 729 struct sparx5_port_config *conf) 730 { 731 bool sgmii = false, inband_aneg = false; 732 int err; 733 734 if (port->conf.inband) { 735 if (conf->portmode == PHY_INTERFACE_MODE_SGMII || 736 conf->portmode == PHY_INTERFACE_MODE_QSGMII) 737 inband_aneg = true; /* Cisco-SGMII in-band-aneg */ 738 else if (conf->portmode == PHY_INTERFACE_MODE_1000BASEX && 739 conf->autoneg) 740 inband_aneg = true; /* Clause-37 in-band-aneg */ 741 742 err = sparx5_serdes_set(sparx5, port, conf); 743 if (err) 744 return -EINVAL; 745 } else { 746 sgmii = true; /* Phy is connected to the MAC */ 747 } 748 749 /* Choose SGMII or 1000BaseX/2500BaseX PCS mode */ 750 spx5_rmw(DEV2G5_PCS1G_MODE_CFG_SGMII_MODE_ENA_SET(sgmii), 751 DEV2G5_PCS1G_MODE_CFG_SGMII_MODE_ENA, 752 sparx5, 753 DEV2G5_PCS1G_MODE_CFG(port->portno)); 754 755 /* Enable PCS */ 756 spx5_wr(DEV2G5_PCS1G_CFG_PCS_ENA_SET(1), 757 sparx5, 758 DEV2G5_PCS1G_CFG(port->portno)); 759 760 if (inband_aneg) { 761 u16 abil = sparx5_get_aneg_word(conf); 762 763 /* Enable in-band aneg */ 764 spx5_wr(DEV2G5_PCS1G_ANEG_CFG_ADV_ABILITY_SET(abil) | 765 DEV2G5_PCS1G_ANEG_CFG_SW_RESOLVE_ENA_SET(1) | 766 DEV2G5_PCS1G_ANEG_CFG_ANEG_ENA_SET(1) | 767 DEV2G5_PCS1G_ANEG_CFG_ANEG_RESTART_ONE_SHOT_SET(1), 768 sparx5, 769 DEV2G5_PCS1G_ANEG_CFG(port->portno)); 770 } else { 771 spx5_wr(0, sparx5, DEV2G5_PCS1G_ANEG_CFG(port->portno)); 772 } 773 774 /* Take PCS out of reset */ 775 spx5_rmw(DEV2G5_DEV_RST_CTRL_SPEED_SEL_SET(2) | 776 DEV2G5_DEV_RST_CTRL_PCS_TX_RST_SET(0) | 777 DEV2G5_DEV_RST_CTRL_PCS_RX_RST_SET(0), 778 DEV2G5_DEV_RST_CTRL_SPEED_SEL | 779 DEV2G5_DEV_RST_CTRL_PCS_TX_RST | 780 DEV2G5_DEV_RST_CTRL_PCS_RX_RST, 781 sparx5, 782 DEV2G5_DEV_RST_CTRL(port->portno)); 783 784 return 0; 785 } 786 787 static int sparx5_port_pcs_high_set(struct sparx5 *sparx5, 788 struct sparx5_port *port, 789 struct sparx5_port_config *conf) 790 { 791 u32 clk_spd = conf->portmode == PHY_INTERFACE_MODE_5GBASER ? 1 : 0; 792 u32 pix = sparx5_port_dev_index(port->portno); 793 u32 dev = sparx5_to_high_dev(port->portno); 794 u32 pcs = sparx5_to_pcs_dev(port->portno); 795 void __iomem *devinst; 796 void __iomem *pcsinst; 797 int err; 798 799 devinst = spx5_inst_get(sparx5, dev, pix); 800 pcsinst = spx5_inst_get(sparx5, pcs, pix); 801 802 /* SFI : No in-band-aneg. Speeds 5G/10G/25G */ 803 err = sparx5_serdes_set(sparx5, port, conf); 804 if (err) 805 return -EINVAL; 806 if (conf->portmode == PHY_INTERFACE_MODE_25GBASER) { 807 /* Enable PCS for 25G device, speed 25G */ 808 spx5_rmw(DEV25G_PCS25G_CFG_PCS25G_ENA_SET(1), 809 DEV25G_PCS25G_CFG_PCS25G_ENA, 810 sparx5, 811 DEV25G_PCS25G_CFG(pix)); 812 } else { 813 /* Enable PCS for 5G/10G/25G devices, speed 5G/10G */ 814 spx5_inst_rmw(PCS10G_BR_PCS_CFG_PCS_ENA_SET(1), 815 PCS10G_BR_PCS_CFG_PCS_ENA, 816 pcsinst, 817 PCS10G_BR_PCS_CFG(0)); 818 } 819 820 /* Enable 5G/10G/25G MAC module */ 821 spx5_inst_wr(DEV10G_MAC_ENA_CFG_RX_ENA_SET(1) | 822 DEV10G_MAC_ENA_CFG_TX_ENA_SET(1), 823 devinst, 824 DEV10G_MAC_ENA_CFG(0)); 825 826 /* Take the device out of reset */ 827 spx5_inst_rmw(DEV10G_DEV_RST_CTRL_PCS_RX_RST_SET(0) | 828 DEV10G_DEV_RST_CTRL_PCS_TX_RST_SET(0) | 829 DEV10G_DEV_RST_CTRL_MAC_RX_RST_SET(0) | 830 DEV10G_DEV_RST_CTRL_MAC_TX_RST_SET(0) | 831 DEV10G_DEV_RST_CTRL_SPEED_SEL_SET(clk_spd), 832 DEV10G_DEV_RST_CTRL_PCS_RX_RST | 833 DEV10G_DEV_RST_CTRL_PCS_TX_RST | 834 DEV10G_DEV_RST_CTRL_MAC_RX_RST | 835 DEV10G_DEV_RST_CTRL_MAC_TX_RST | 836 DEV10G_DEV_RST_CTRL_SPEED_SEL, 837 devinst, 838 DEV10G_DEV_RST_CTRL(0)); 839 840 return 0; 841 } 842 843 /* Switch between 1G/2500 and 5G/10G/25G devices */ 844 static void sparx5_dev_switch(struct sparx5 *sparx5, int port, bool hsd) 845 { 846 int bt_indx = BIT(sparx5_port_dev_index(port)); 847 848 if (sparx5_port_is_5g(port)) { 849 spx5_rmw(hsd ? 0 : bt_indx, 850 bt_indx, 851 sparx5, 852 PORT_CONF_DEV5G_MODES); 853 } else if (sparx5_port_is_10g(port)) { 854 spx5_rmw(hsd ? 0 : bt_indx, 855 bt_indx, 856 sparx5, 857 PORT_CONF_DEV10G_MODES); 858 } else if (sparx5_port_is_25g(port)) { 859 spx5_rmw(hsd ? 0 : bt_indx, 860 bt_indx, 861 sparx5, 862 PORT_CONF_DEV25G_MODES); 863 } 864 } 865 866 /* Configure speed/duplex dependent registers */ 867 static int sparx5_port_config_low_set(struct sparx5 *sparx5, 868 struct sparx5_port *port, 869 struct sparx5_port_config *conf) 870 { 871 u32 clk_spd, gig_mode, tx_gap, hdx_gap_1, hdx_gap_2; 872 bool fdx = conf->duplex == DUPLEX_FULL; 873 int spd = conf->speed; 874 875 clk_spd = spd == SPEED_10 ? 0 : spd == SPEED_100 ? 1 : 2; 876 gig_mode = spd == SPEED_1000 || spd == SPEED_2500; 877 tx_gap = spd == SPEED_1000 ? 4 : fdx ? 6 : 5; 878 hdx_gap_1 = spd == SPEED_1000 ? 0 : spd == SPEED_100 ? 1 : 2; 879 hdx_gap_2 = spd == SPEED_1000 ? 0 : spd == SPEED_100 ? 4 : 1; 880 881 /* GIG/FDX mode */ 882 spx5_rmw(DEV2G5_MAC_MODE_CFG_GIGA_MODE_ENA_SET(gig_mode) | 883 DEV2G5_MAC_MODE_CFG_FDX_ENA_SET(fdx), 884 DEV2G5_MAC_MODE_CFG_GIGA_MODE_ENA | 885 DEV2G5_MAC_MODE_CFG_FDX_ENA, 886 sparx5, 887 DEV2G5_MAC_MODE_CFG(port->portno)); 888 889 /* Set MAC IFG Gaps */ 890 spx5_wr(DEV2G5_MAC_IFG_CFG_TX_IFG_SET(tx_gap) | 891 DEV2G5_MAC_IFG_CFG_RX_IFG1_SET(hdx_gap_1) | 892 DEV2G5_MAC_IFG_CFG_RX_IFG2_SET(hdx_gap_2), 893 sparx5, 894 DEV2G5_MAC_IFG_CFG(port->portno)); 895 896 /* Disabling frame aging when in HDX (due to HDX issue) */ 897 spx5_rmw(HSCH_PORT_MODE_AGE_DIS_SET(fdx == 0), 898 HSCH_PORT_MODE_AGE_DIS, 899 sparx5, 900 HSCH_PORT_MODE(port->portno)); 901 902 /* Enable MAC module */ 903 spx5_wr(DEV2G5_MAC_ENA_CFG_RX_ENA | 904 DEV2G5_MAC_ENA_CFG_TX_ENA, 905 sparx5, 906 DEV2G5_MAC_ENA_CFG(port->portno)); 907 908 /* Select speed and take MAC out of reset */ 909 spx5_rmw(DEV2G5_DEV_RST_CTRL_SPEED_SEL_SET(clk_spd) | 910 DEV2G5_DEV_RST_CTRL_MAC_TX_RST_SET(0) | 911 DEV2G5_DEV_RST_CTRL_MAC_RX_RST_SET(0), 912 DEV2G5_DEV_RST_CTRL_SPEED_SEL | 913 DEV2G5_DEV_RST_CTRL_MAC_TX_RST | 914 DEV2G5_DEV_RST_CTRL_MAC_RX_RST, 915 sparx5, 916 DEV2G5_DEV_RST_CTRL(port->portno)); 917 918 return 0; 919 } 920 921 int sparx5_port_pcs_set(struct sparx5 *sparx5, 922 struct sparx5_port *port, 923 struct sparx5_port_config *conf) 924 925 { 926 bool high_speed_dev = sparx5_is_baser(conf->portmode); 927 int err; 928 929 if (sparx5_dev_change(sparx5, port, conf)) { 930 /* switch device */ 931 sparx5_dev_switch(sparx5, port->portno, high_speed_dev); 932 933 /* Disable the not-in-use device */ 934 err = sparx5_port_disable(sparx5, port, !high_speed_dev); 935 if (err) 936 return err; 937 } 938 /* Disable the port before re-configuring */ 939 err = sparx5_port_disable(sparx5, port, high_speed_dev); 940 if (err) 941 return -EINVAL; 942 943 if (high_speed_dev) 944 err = sparx5_port_pcs_high_set(sparx5, port, conf); 945 else 946 err = sparx5_port_pcs_low_set(sparx5, port, conf); 947 948 if (err) 949 return -EINVAL; 950 951 if (port->conf.inband) { 952 /* Enable/disable 1G counters in ASM */ 953 spx5_rmw(ASM_PORT_CFG_CSC_STAT_DIS_SET(high_speed_dev), 954 ASM_PORT_CFG_CSC_STAT_DIS, 955 sparx5, 956 ASM_PORT_CFG(port->portno)); 957 958 /* Enable/disable 1G counters in DSM */ 959 spx5_rmw(DSM_BUF_CFG_CSC_STAT_DIS_SET(high_speed_dev), 960 DSM_BUF_CFG_CSC_STAT_DIS, 961 sparx5, 962 DSM_BUF_CFG(port->portno)); 963 } 964 965 port->conf = *conf; 966 967 return 0; 968 } 969 970 int sparx5_port_config(struct sparx5 *sparx5, 971 struct sparx5_port *port, 972 struct sparx5_port_config *conf) 973 { 974 bool high_speed_dev = sparx5_is_baser(conf->portmode); 975 int err, urgency, stop_wm; 976 977 err = sparx5_port_verify_speed(sparx5, port, conf); 978 if (err) 979 return err; 980 981 /* high speed device is already configured */ 982 if (!high_speed_dev) 983 sparx5_port_config_low_set(sparx5, port, conf); 984 985 /* Configure flow control */ 986 err = sparx5_port_fc_setup(sparx5, port, conf); 987 if (err) 988 return err; 989 990 /* Set the DSM stop watermark */ 991 stop_wm = sparx5_port_fifo_sz(sparx5, port->portno, conf->speed); 992 spx5_rmw(DSM_DEV_TX_STOP_WM_CFG_DEV_TX_STOP_WM_SET(stop_wm), 993 DSM_DEV_TX_STOP_WM_CFG_DEV_TX_STOP_WM, 994 sparx5, 995 DSM_DEV_TX_STOP_WM_CFG(port->portno)); 996 997 /* Enable port in queue system */ 998 urgency = sparx5_port_fwd_urg(sparx5, conf->speed); 999 spx5_rmw(QFWD_SWITCH_PORT_MODE_PORT_ENA_SET(1) | 1000 QFWD_SWITCH_PORT_MODE_FWD_URGENCY_SET(urgency), 1001 QFWD_SWITCH_PORT_MODE_PORT_ENA | 1002 QFWD_SWITCH_PORT_MODE_FWD_URGENCY, 1003 sparx5, 1004 QFWD_SWITCH_PORT_MODE(port->portno)); 1005 1006 /* Save the new values */ 1007 port->conf = *conf; 1008 1009 return 0; 1010 } 1011 1012 /* Initialize port config to default */ 1013 int sparx5_port_init(struct sparx5 *sparx5, 1014 struct sparx5_port *port, 1015 struct sparx5_port_config *conf) 1016 { 1017 u32 pause_start = sparx5_wm_enc(6 * (ETH_MAXLEN / SPX5_BUFFER_CELL_SZ)); 1018 u32 atop = sparx5_wm_enc(20 * (ETH_MAXLEN / SPX5_BUFFER_CELL_SZ)); 1019 u32 devhigh = sparx5_to_high_dev(port->portno); 1020 u32 pix = sparx5_port_dev_index(port->portno); 1021 u32 pcs = sparx5_to_pcs_dev(port->portno); 1022 bool sd_pol = port->signd_active_high; 1023 bool sd_sel = !port->signd_internal; 1024 bool sd_ena = port->signd_enable; 1025 u32 pause_stop = 0xFFF - 1; /* FC generate disabled */ 1026 void __iomem *devinst; 1027 void __iomem *pcsinst; 1028 int err; 1029 1030 devinst = spx5_inst_get(sparx5, devhigh, pix); 1031 pcsinst = spx5_inst_get(sparx5, pcs, pix); 1032 1033 /* Set the mux port mode */ 1034 err = sparx5_port_mux_set(sparx5, port, conf); 1035 if (err) 1036 return err; 1037 1038 /* Configure MAC vlan awareness */ 1039 err = sparx5_port_max_tags_set(sparx5, port); 1040 if (err) 1041 return err; 1042 1043 /* Set Max Length */ 1044 spx5_rmw(DEV2G5_MAC_MAXLEN_CFG_MAX_LEN_SET(ETH_MAXLEN), 1045 DEV2G5_MAC_MAXLEN_CFG_MAX_LEN, 1046 sparx5, 1047 DEV2G5_MAC_MAXLEN_CFG(port->portno)); 1048 1049 /* 1G/2G5: Signal Detect configuration */ 1050 spx5_wr(DEV2G5_PCS1G_SD_CFG_SD_POL_SET(sd_pol) | 1051 DEV2G5_PCS1G_SD_CFG_SD_SEL_SET(sd_sel) | 1052 DEV2G5_PCS1G_SD_CFG_SD_ENA_SET(sd_ena), 1053 sparx5, 1054 DEV2G5_PCS1G_SD_CFG(port->portno)); 1055 1056 /* Set Pause WM hysteresis */ 1057 spx5_rmw(QSYS_PAUSE_CFG_PAUSE_START_SET(pause_start) | 1058 QSYS_PAUSE_CFG_PAUSE_STOP_SET(pause_stop) | 1059 QSYS_PAUSE_CFG_PAUSE_ENA_SET(1), 1060 QSYS_PAUSE_CFG_PAUSE_START | 1061 QSYS_PAUSE_CFG_PAUSE_STOP | 1062 QSYS_PAUSE_CFG_PAUSE_ENA, 1063 sparx5, 1064 QSYS_PAUSE_CFG(port->portno)); 1065 1066 /* Port ATOP. Frames are tail dropped when this WM is hit */ 1067 spx5_wr(QSYS_ATOP_ATOP_SET(atop), 1068 sparx5, 1069 QSYS_ATOP(port->portno)); 1070 1071 /* Discard pause frame 01-80-C2-00-00-01 */ 1072 spx5_wr(PAUSE_DISCARD, sparx5, ANA_CL_CAPTURE_BPDU_CFG(port->portno)); 1073 1074 if (conf->portmode == PHY_INTERFACE_MODE_QSGMII || 1075 conf->portmode == PHY_INTERFACE_MODE_SGMII) { 1076 err = sparx5_serdes_set(sparx5, port, conf); 1077 if (err) 1078 return err; 1079 1080 if (!sparx5_port_is_2g5(port->portno)) 1081 /* Enable shadow device */ 1082 spx5_rmw(DSM_DEV_TX_STOP_WM_CFG_DEV10G_SHADOW_ENA_SET(1), 1083 DSM_DEV_TX_STOP_WM_CFG_DEV10G_SHADOW_ENA, 1084 sparx5, 1085 DSM_DEV_TX_STOP_WM_CFG(port->portno)); 1086 1087 sparx5_dev_switch(sparx5, port->portno, false); 1088 } 1089 if (conf->portmode == PHY_INTERFACE_MODE_QSGMII) { 1090 // All ports must be PCS enabled in QSGMII mode 1091 spx5_rmw(DEV2G5_DEV_RST_CTRL_PCS_TX_RST_SET(0), 1092 DEV2G5_DEV_RST_CTRL_PCS_TX_RST, 1093 sparx5, 1094 DEV2G5_DEV_RST_CTRL(port->portno)); 1095 } 1096 /* Default IFGs for 1G */ 1097 spx5_wr(DEV2G5_MAC_IFG_CFG_TX_IFG_SET(6) | 1098 DEV2G5_MAC_IFG_CFG_RX_IFG1_SET(0) | 1099 DEV2G5_MAC_IFG_CFG_RX_IFG2_SET(0), 1100 sparx5, 1101 DEV2G5_MAC_IFG_CFG(port->portno)); 1102 1103 if (sparx5_port_is_2g5(port->portno)) 1104 return 0; /* Low speed device only - return */ 1105 1106 /* Now setup the high speed device */ 1107 if (conf->portmode == PHY_INTERFACE_MODE_NA) 1108 conf->portmode = PHY_INTERFACE_MODE_10GBASER; 1109 1110 if (sparx5_is_baser(conf->portmode)) 1111 sparx5_dev_switch(sparx5, port->portno, true); 1112 1113 /* Set Max Length */ 1114 spx5_inst_rmw(DEV10G_MAC_MAXLEN_CFG_MAX_LEN_SET(ETH_MAXLEN), 1115 DEV10G_MAC_MAXLEN_CFG_MAX_LEN, 1116 devinst, 1117 DEV10G_MAC_ENA_CFG(0)); 1118 1119 /* Handle Signal Detect in 10G PCS */ 1120 spx5_inst_wr(PCS10G_BR_PCS_SD_CFG_SD_POL_SET(sd_pol) | 1121 PCS10G_BR_PCS_SD_CFG_SD_SEL_SET(sd_sel) | 1122 PCS10G_BR_PCS_SD_CFG_SD_ENA_SET(sd_ena), 1123 pcsinst, 1124 PCS10G_BR_PCS_SD_CFG(0)); 1125 1126 if (sparx5_port_is_25g(port->portno)) { 1127 /* Handle Signal Detect in 25G PCS */ 1128 spx5_wr(DEV25G_PCS25G_SD_CFG_SD_POL_SET(sd_pol) | 1129 DEV25G_PCS25G_SD_CFG_SD_SEL_SET(sd_sel) | 1130 DEV25G_PCS25G_SD_CFG_SD_ENA_SET(sd_ena), 1131 sparx5, 1132 DEV25G_PCS25G_SD_CFG(pix)); 1133 } 1134 1135 return 0; 1136 } 1137 1138 void sparx5_port_enable(struct sparx5_port *port, bool enable) 1139 { 1140 struct sparx5 *sparx5 = port->sparx5; 1141 1142 /* Enable port for frame transfer? */ 1143 spx5_rmw(QFWD_SWITCH_PORT_MODE_PORT_ENA_SET(enable), 1144 QFWD_SWITCH_PORT_MODE_PORT_ENA, 1145 sparx5, 1146 QFWD_SWITCH_PORT_MODE(port->portno)); 1147 } 1148 1149 int sparx5_port_qos_set(struct sparx5_port *port, 1150 struct sparx5_port_qos *qos) 1151 { 1152 sparx5_port_qos_dscp_set(port, &qos->dscp); 1153 sparx5_port_qos_pcp_set(port, &qos->pcp); 1154 sparx5_port_qos_pcp_rewr_set(port, &qos->pcp_rewr); 1155 sparx5_port_qos_dscp_rewr_set(port, &qos->dscp_rewr); 1156 sparx5_port_qos_default_set(port, qos); 1157 1158 return 0; 1159 } 1160 1161 int sparx5_port_qos_pcp_rewr_set(const struct sparx5_port *port, 1162 struct sparx5_port_qos_pcp_rewr *qos) 1163 { 1164 int i, mode = SPARX5_PORT_REW_TAG_CTRL_CLASSIFIED; 1165 struct sparx5 *sparx5 = port->sparx5; 1166 u8 pcp, dei; 1167 1168 /* Use mapping table, with classified QoS as index, to map QoS and DP 1169 * to tagged PCP and DEI, if PCP is trusted. Otherwise use classified 1170 * PCP. Classified PCP equals frame PCP. 1171 */ 1172 if (qos->enable) 1173 mode = SPARX5_PORT_REW_TAG_CTRL_MAPPED; 1174 1175 spx5_rmw(REW_TAG_CTRL_TAG_PCP_CFG_SET(mode) | 1176 REW_TAG_CTRL_TAG_DEI_CFG_SET(mode), 1177 REW_TAG_CTRL_TAG_PCP_CFG | REW_TAG_CTRL_TAG_DEI_CFG, 1178 port->sparx5, REW_TAG_CTRL(port->portno)); 1179 1180 for (i = 0; i < ARRAY_SIZE(qos->map.map); i++) { 1181 /* Extract PCP and DEI */ 1182 pcp = qos->map.map[i]; 1183 if (pcp > SPARX5_PORT_QOS_PCP_COUNT) 1184 dei = 1; 1185 else 1186 dei = 0; 1187 1188 /* Rewrite PCP and DEI, for each classified QoS class and DP 1189 * level. This table is only used if tag ctrl mode is set to 1190 * 'mapped'. 1191 * 1192 * 0:0nd - prio=0 and dp:0 => pcp=0 and dei=0 1193 * 0:0de - prio=0 and dp:1 => pcp=0 and dei=1 1194 */ 1195 if (dei) { 1196 spx5_rmw(REW_PCP_MAP_DE1_PCP_DE1_SET(pcp), 1197 REW_PCP_MAP_DE1_PCP_DE1, sparx5, 1198 REW_PCP_MAP_DE1(port->portno, i)); 1199 1200 spx5_rmw(REW_DEI_MAP_DE1_DEI_DE1_SET(dei), 1201 REW_DEI_MAP_DE1_DEI_DE1, port->sparx5, 1202 REW_DEI_MAP_DE1(port->portno, i)); 1203 } else { 1204 spx5_rmw(REW_PCP_MAP_DE0_PCP_DE0_SET(pcp), 1205 REW_PCP_MAP_DE0_PCP_DE0, sparx5, 1206 REW_PCP_MAP_DE0(port->portno, i)); 1207 1208 spx5_rmw(REW_DEI_MAP_DE0_DEI_DE0_SET(dei), 1209 REW_DEI_MAP_DE0_DEI_DE0, port->sparx5, 1210 REW_DEI_MAP_DE0(port->portno, i)); 1211 } 1212 } 1213 1214 return 0; 1215 } 1216 1217 int sparx5_port_qos_pcp_set(const struct sparx5_port *port, 1218 struct sparx5_port_qos_pcp *qos) 1219 { 1220 struct sparx5 *sparx5 = port->sparx5; 1221 u8 *pcp_itr = qos->map.map; 1222 u8 pcp, dp; 1223 int i; 1224 1225 /* Enable/disable pcp and dp for qos classification. */ 1226 spx5_rmw(ANA_CL_QOS_CFG_PCP_DEI_QOS_ENA_SET(qos->qos_enable) | 1227 ANA_CL_QOS_CFG_PCP_DEI_DP_ENA_SET(qos->dp_enable), 1228 ANA_CL_QOS_CFG_PCP_DEI_QOS_ENA | ANA_CL_QOS_CFG_PCP_DEI_DP_ENA, 1229 sparx5, ANA_CL_QOS_CFG(port->portno)); 1230 1231 /* Map each pcp and dei value to priority and dp */ 1232 for (i = 0; i < ARRAY_SIZE(qos->map.map); i++) { 1233 pcp = *(pcp_itr + i); 1234 dp = (i < SPARX5_PORT_QOS_PCP_COUNT) ? 0 : 1; 1235 spx5_rmw(ANA_CL_PCP_DEI_MAP_CFG_PCP_DEI_QOS_VAL_SET(pcp) | 1236 ANA_CL_PCP_DEI_MAP_CFG_PCP_DEI_DP_VAL_SET(dp), 1237 ANA_CL_PCP_DEI_MAP_CFG_PCP_DEI_QOS_VAL | 1238 ANA_CL_PCP_DEI_MAP_CFG_PCP_DEI_DP_VAL, sparx5, 1239 ANA_CL_PCP_DEI_MAP_CFG(port->portno, i)); 1240 } 1241 1242 return 0; 1243 } 1244 1245 void sparx5_port_qos_dscp_rewr_mode_set(const struct sparx5_port *port, 1246 int mode) 1247 { 1248 spx5_rmw(ANA_CL_QOS_CFG_DSCP_REWR_MODE_SEL_SET(mode), 1249 ANA_CL_QOS_CFG_DSCP_REWR_MODE_SEL, port->sparx5, 1250 ANA_CL_QOS_CFG(port->portno)); 1251 } 1252 1253 int sparx5_port_qos_dscp_rewr_set(const struct sparx5_port *port, 1254 struct sparx5_port_qos_dscp_rewr *qos) 1255 { 1256 struct sparx5 *sparx5 = port->sparx5; 1257 bool rewr = false; 1258 u16 dscp; 1259 int i; 1260 1261 /* On egress, rewrite DSCP value to either classified DSCP or frame 1262 * DSCP. If enabled; classified DSCP, if disabled; frame DSCP. 1263 */ 1264 if (qos->enable) 1265 rewr = true; 1266 1267 spx5_rmw(REW_DSCP_MAP_DSCP_UPDATE_ENA_SET(rewr), 1268 REW_DSCP_MAP_DSCP_UPDATE_ENA, sparx5, 1269 REW_DSCP_MAP(port->portno)); 1270 1271 /* On ingress, map each classified QoS class and DP to classified DSCP 1272 * value. This mapping table is global for all ports. 1273 */ 1274 for (i = 0; i < ARRAY_SIZE(qos->map.map); i++) { 1275 dscp = qos->map.map[i]; 1276 spx5_rmw(ANA_CL_QOS_MAP_CFG_DSCP_REWR_VAL_SET(dscp), 1277 ANA_CL_QOS_MAP_CFG_DSCP_REWR_VAL, sparx5, 1278 ANA_CL_QOS_MAP_CFG(i)); 1279 } 1280 1281 return 0; 1282 } 1283 1284 int sparx5_port_qos_dscp_set(const struct sparx5_port *port, 1285 struct sparx5_port_qos_dscp *qos) 1286 { 1287 struct sparx5 *sparx5 = port->sparx5; 1288 u8 *dscp = qos->map.map; 1289 int i; 1290 1291 /* Enable/disable dscp and dp for qos classification. 1292 * Disable rewrite of dscp values for now. 1293 */ 1294 spx5_rmw(ANA_CL_QOS_CFG_DSCP_QOS_ENA_SET(qos->qos_enable) | 1295 ANA_CL_QOS_CFG_DSCP_DP_ENA_SET(qos->dp_enable) | 1296 ANA_CL_QOS_CFG_DSCP_KEEP_ENA_SET(1), 1297 ANA_CL_QOS_CFG_DSCP_QOS_ENA | ANA_CL_QOS_CFG_DSCP_DP_ENA | 1298 ANA_CL_QOS_CFG_DSCP_KEEP_ENA, sparx5, 1299 ANA_CL_QOS_CFG(port->portno)); 1300 1301 /* Map each dscp value to priority and dp */ 1302 for (i = 0; i < ARRAY_SIZE(qos->map.map); i++) { 1303 spx5_rmw(ANA_CL_DSCP_CFG_DSCP_QOS_VAL_SET(*(dscp + i)) | 1304 ANA_CL_DSCP_CFG_DSCP_DP_VAL_SET(0), 1305 ANA_CL_DSCP_CFG_DSCP_QOS_VAL | 1306 ANA_CL_DSCP_CFG_DSCP_DP_VAL, sparx5, 1307 ANA_CL_DSCP_CFG(i)); 1308 } 1309 1310 /* Set per-dscp trust */ 1311 for (i = 0; i < ARRAY_SIZE(qos->map.map); i++) { 1312 if (qos->qos_enable) { 1313 spx5_rmw(ANA_CL_DSCP_CFG_DSCP_TRUST_ENA_SET(1), 1314 ANA_CL_DSCP_CFG_DSCP_TRUST_ENA, sparx5, 1315 ANA_CL_DSCP_CFG(i)); 1316 } 1317 } 1318 1319 return 0; 1320 } 1321 1322 int sparx5_port_qos_default_set(const struct sparx5_port *port, 1323 const struct sparx5_port_qos *qos) 1324 { 1325 struct sparx5 *sparx5 = port->sparx5; 1326 1327 /* Set default prio and dp level */ 1328 spx5_rmw(ANA_CL_QOS_CFG_DEFAULT_QOS_VAL_SET(qos->default_prio) | 1329 ANA_CL_QOS_CFG_DEFAULT_DP_VAL_SET(0), 1330 ANA_CL_QOS_CFG_DEFAULT_QOS_VAL | 1331 ANA_CL_QOS_CFG_DEFAULT_DP_VAL, 1332 sparx5, ANA_CL_QOS_CFG(port->portno)); 1333 1334 /* Set default pcp and dei for untagged frames */ 1335 spx5_rmw(ANA_CL_VLAN_CTRL_PORT_PCP_SET(0) | 1336 ANA_CL_VLAN_CTRL_PORT_DEI_SET(0), 1337 ANA_CL_VLAN_CTRL_PORT_PCP | 1338 ANA_CL_VLAN_CTRL_PORT_DEI, 1339 sparx5, ANA_CL_VLAN_CTRL(port->portno)); 1340 1341 return 0; 1342 } 1343