Linux kernel mirror (for testing) git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
kernel os linux

net: dsa: qca8k: set internal delay also for sgmii

QCA original code report port instability and sa that SGMII also require
to set internal delay. Generalize the rgmii delay function and apply the
advised value if they are not defined in DT.

Signed-off-by: Ansuel Smith <ansuelsmth@gmail.com>
Signed-off-by: David S. Miller <davem@davemloft.net>

authored by

Ansuel Smith and committed by
David S. Miller
cef08115 f477d1c8

+62 -28
+60 -28
drivers/net/dsa/qca8k.c
··· 1004 1004 case PHY_INTERFACE_MODE_RGMII_ID: 1005 1005 case PHY_INTERFACE_MODE_RGMII_TXID: 1006 1006 case PHY_INTERFACE_MODE_RGMII_RXID: 1007 + case PHY_INTERFACE_MODE_SGMII: 1007 1008 delay = 0; 1008 1009 1009 1010 if (!of_property_read_u32(port_dn, "tx-internal-delay-ps", &delay)) ··· 1037 1036 1038 1037 priv->rgmii_rx_delay[cpu_port_index] = delay; 1039 1038 1040 - break; 1041 - case PHY_INTERFACE_MODE_SGMII: 1039 + /* Skip sgmii parsing for rgmii* mode */ 1040 + if (mode == PHY_INTERFACE_MODE_RGMII || 1041 + mode == PHY_INTERFACE_MODE_RGMII_ID || 1042 + mode == PHY_INTERFACE_MODE_RGMII_TXID || 1043 + mode == PHY_INTERFACE_MODE_RGMII_RXID) 1044 + break; 1045 + 1042 1046 if (of_property_read_bool(port_dn, "qca,sgmii-txclk-falling-edge")) 1043 1047 priv->sgmii_tx_clk_falling_edge = true; 1044 1048 ··· 1267 1261 } 1268 1262 1269 1263 static void 1264 + qca8k_mac_config_setup_internal_delay(struct qca8k_priv *priv, int cpu_port_index, 1265 + u32 reg) 1266 + { 1267 + u32 delay, val = 0; 1268 + int ret; 1269 + 1270 + /* Delay can be declared in 3 different way. 1271 + * Mode to rgmii and internal-delay standard binding defined 1272 + * rgmii-id or rgmii-tx/rx phy mode set. 1273 + * The parse logic set a delay different than 0 only when one 1274 + * of the 3 different way is used. In all other case delay is 1275 + * not enabled. With ID or TX/RXID delay is enabled and set 1276 + * to the default and recommended value. 1277 + */ 1278 + if (priv->rgmii_tx_delay[cpu_port_index]) { 1279 + delay = priv->rgmii_tx_delay[cpu_port_index]; 1280 + 1281 + val |= QCA8K_PORT_PAD_RGMII_TX_DELAY(delay) | 1282 + QCA8K_PORT_PAD_RGMII_TX_DELAY_EN; 1283 + } 1284 + 1285 + if (priv->rgmii_rx_delay[cpu_port_index]) { 1286 + delay = priv->rgmii_rx_delay[cpu_port_index]; 1287 + 1288 + val |= QCA8K_PORT_PAD_RGMII_RX_DELAY(delay) | 1289 + QCA8K_PORT_PAD_RGMII_RX_DELAY_EN; 1290 + } 1291 + 1292 + /* Set RGMII delay based on the selected values */ 1293 + ret = qca8k_rmw(priv, reg, 1294 + QCA8K_PORT_PAD_RGMII_TX_DELAY_MASK | 1295 + QCA8K_PORT_PAD_RGMII_RX_DELAY_MASK | 1296 + QCA8K_PORT_PAD_RGMII_TX_DELAY_EN | 1297 + QCA8K_PORT_PAD_RGMII_RX_DELAY_EN, 1298 + val); 1299 + if (ret) 1300 + dev_err(priv->dev, "Failed to set internal delay for CPU port%d", 1301 + cpu_port_index == QCA8K_CPU_PORT0 ? 0 : 6); 1302 + } 1303 + 1304 + static void 1270 1305 qca8k_phylink_mac_config(struct dsa_switch *ds, int port, unsigned int mode, 1271 1306 const struct phylink_link_state *state) 1272 1307 { 1273 1308 struct qca8k_priv *priv = ds->priv; 1274 1309 int cpu_port_index, ret; 1275 - u32 reg, val, delay; 1310 + u32 reg, val; 1276 1311 1277 1312 switch (port) { 1278 1313 case 0: /* 1st CPU port */ ··· 1362 1315 case PHY_INTERFACE_MODE_RGMII_ID: 1363 1316 case PHY_INTERFACE_MODE_RGMII_TXID: 1364 1317 case PHY_INTERFACE_MODE_RGMII_RXID: 1365 - val = QCA8K_PORT_PAD_RGMII_EN; 1318 + qca8k_write(priv, reg, QCA8K_PORT_PAD_RGMII_EN); 1366 1319 1367 - /* Delay can be declared in 3 different way. 1368 - * Mode to rgmii and internal-delay standard binding defined 1369 - * rgmii-id or rgmii-tx/rx phy mode set. 1370 - * The parse logic set a delay different than 0 only when one 1371 - * of the 3 different way is used. In all other case delay is 1372 - * not enabled. With ID or TX/RXID delay is enabled and set 1373 - * to the default and recommended value. 1374 - */ 1375 - if (priv->rgmii_tx_delay[cpu_port_index]) { 1376 - delay = priv->rgmii_tx_delay[cpu_port_index]; 1377 - 1378 - val |= QCA8K_PORT_PAD_RGMII_TX_DELAY(delay) | 1379 - QCA8K_PORT_PAD_RGMII_TX_DELAY_EN; 1380 - } 1381 - 1382 - if (priv->rgmii_rx_delay[cpu_port_index]) { 1383 - delay = priv->rgmii_rx_delay[cpu_port_index]; 1384 - 1385 - val |= QCA8K_PORT_PAD_RGMII_RX_DELAY(delay) | 1386 - QCA8K_PORT_PAD_RGMII_RX_DELAY_EN; 1387 - } 1388 - 1389 - /* Set RGMII delay based on the selected values */ 1390 - qca8k_write(priv, reg, val); 1320 + /* Configure rgmii delay */ 1321 + qca8k_mac_config_setup_internal_delay(priv, cpu_port_index, reg); 1391 1322 1392 1323 /* QCA8337 requires to set rgmii rx delay for all ports. 1393 1324 * This is enabled through PORT5_PAD_CTRL for all ports, ··· 1436 1411 QCA8K_PORT0_PAD_SGMII_RXCLK_FALLING_EDGE | 1437 1412 QCA8K_PORT0_PAD_SGMII_TXCLK_FALLING_EDGE, 1438 1413 val); 1414 + 1415 + /* From original code is reported port instability as SGMII also 1416 + * require delay set. Apply advised values here or take them from DT. 1417 + */ 1418 + if (state->interface == PHY_INTERFACE_MODE_SGMII) 1419 + qca8k_mac_config_setup_internal_delay(priv, cpu_port_index, reg); 1420 + 1439 1421 break; 1440 1422 default: 1441 1423 dev_err(ds->dev, "xMII mode %s not supported for port %d\n",
+2
drivers/net/dsa/qca8k.h
··· 39 39 #define QCA8K_REG_PORT5_PAD_CTRL 0x008 40 40 #define QCA8K_REG_PORT6_PAD_CTRL 0x00c 41 41 #define QCA8K_PORT_PAD_RGMII_EN BIT(26) 42 + #define QCA8K_PORT_PAD_RGMII_TX_DELAY_MASK GENMASK(23, 22) 42 43 #define QCA8K_PORT_PAD_RGMII_TX_DELAY(x) ((x) << 22) 44 + #define QCA8K_PORT_PAD_RGMII_RX_DELAY_MASK GENMASK(21, 20) 43 45 #define QCA8K_PORT_PAD_RGMII_RX_DELAY(x) ((x) << 20) 44 46 #define QCA8K_PORT_PAD_RGMII_TX_DELAY_EN BIT(25) 45 47 #define QCA8K_PORT_PAD_RGMII_RX_DELAY_EN BIT(24)