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

net: phy: dp83867: Add speed optimization feature

Set the speed optimization bit on the DP83867 PHY.
This feature can also be strapped on the 64 pin PHY devices
but the 48 pin devices do not have the strap pin available to enable
this feature in the hardware. PHY team suggests to have this bit set.

With this bit set the PHY will auto negotiate and report the link
parameters in the PHYSTS register. This register provides a single
location within the register set for quick access to commonly accessed
information.

In this case when auto negotiation is on the PHY core reads the bits
that have been configured or if auto negotiation is off the PHY core
reads the BMCR register and sets the phydev parameters accordingly.

This Giga bit PHY can throttle the speed to 100Mbps or 10Mbps to accomodate a
4-wire cable. If this should occur the PHYSTS register contains the
current negotiated speed and duplex mode.

In overriding the genphy_read_status the dp83867_read_status will do a
genphy_read_status to setup the LP and pause bits. And then the PHYSTS
register is read and the phydev speed and duplex mode settings are
updated.

Signed-off-by: Dan Murphy <dmurphy@ti.com>
Signed-off-by: David S. Miller <davem@davemloft.net>

authored by

Dan Murphy and committed by
David S. Miller
cd26d72d feb5d98e

+150
+150
drivers/net/phy/dp83867.c
··· 14 14 #include <linux/delay.h> 15 15 #include <linux/netdevice.h> 16 16 #include <linux/etherdevice.h> 17 + #include <linux/bitfield.h> 17 18 18 19 #include <dt-bindings/net/ti-dp83867.h> 19 20 ··· 22 21 #define DP83867_DEVADDR 0x1f 23 22 24 23 #define MII_DP83867_PHYCTRL 0x10 24 + #define MII_DP83867_PHYSTS 0x11 25 25 #define MII_DP83867_MICR 0x12 26 26 #define MII_DP83867_ISR 0x13 27 27 #define DP83867_CFG2 0x14 ··· 119 117 #define DP83867_IO_MUX_CFG_CLK_O_DISABLE BIT(6) 120 118 #define DP83867_IO_MUX_CFG_CLK_O_SEL_MASK (0x1f << 8) 121 119 #define DP83867_IO_MUX_CFG_CLK_O_SEL_SHIFT 8 120 + 121 + /* PHY STS bits */ 122 + #define DP83867_PHYSTS_1000 BIT(15) 123 + #define DP83867_PHYSTS_100 BIT(14) 124 + #define DP83867_PHYSTS_DUPLEX BIT(13) 125 + #define DP83867_PHYSTS_LINK BIT(10) 126 + 127 + /* CFG2 bits */ 128 + #define DP83867_DOWNSHIFT_EN (BIT(8) | BIT(9)) 129 + #define DP83867_DOWNSHIFT_ATTEMPT_MASK (BIT(10) | BIT(11)) 130 + #define DP83867_DOWNSHIFT_1_COUNT_VAL 0 131 + #define DP83867_DOWNSHIFT_2_COUNT_VAL 1 132 + #define DP83867_DOWNSHIFT_4_COUNT_VAL 2 133 + #define DP83867_DOWNSHIFT_8_COUNT_VAL 3 134 + #define DP83867_DOWNSHIFT_1_COUNT 1 135 + #define DP83867_DOWNSHIFT_2_COUNT 2 136 + #define DP83867_DOWNSHIFT_4_COUNT 4 137 + #define DP83867_DOWNSHIFT_8_COUNT 8 122 138 123 139 /* CFG3 bits */ 124 140 #define DP83867_CFG3_INT_OE BIT(7) ··· 305 285 306 286 micr_status = 0x0; 307 287 return phy_write(phydev, MII_DP83867_MICR, micr_status); 288 + } 289 + 290 + static int dp83867_read_status(struct phy_device *phydev) 291 + { 292 + int status = phy_read(phydev, MII_DP83867_PHYSTS); 293 + int ret; 294 + 295 + ret = genphy_read_status(phydev); 296 + if (ret) 297 + return ret; 298 + 299 + if (status < 0) 300 + return status; 301 + 302 + if (status & DP83867_PHYSTS_DUPLEX) 303 + phydev->duplex = DUPLEX_FULL; 304 + else 305 + phydev->duplex = DUPLEX_HALF; 306 + 307 + if (status & DP83867_PHYSTS_1000) 308 + phydev->speed = SPEED_1000; 309 + else if (status & DP83867_PHYSTS_100) 310 + phydev->speed = SPEED_100; 311 + else 312 + phydev->speed = SPEED_10; 313 + 314 + return 0; 315 + } 316 + 317 + static int dp83867_get_downshift(struct phy_device *phydev, u8 *data) 318 + { 319 + int val, cnt, enable, count; 320 + 321 + val = phy_read(phydev, DP83867_CFG2); 322 + if (val < 0) 323 + return val; 324 + 325 + enable = FIELD_GET(DP83867_DOWNSHIFT_EN, val); 326 + cnt = FIELD_GET(DP83867_DOWNSHIFT_ATTEMPT_MASK, val); 327 + 328 + switch (cnt) { 329 + case DP83867_DOWNSHIFT_1_COUNT_VAL: 330 + count = DP83867_DOWNSHIFT_1_COUNT; 331 + break; 332 + case DP83867_DOWNSHIFT_2_COUNT_VAL: 333 + count = DP83867_DOWNSHIFT_2_COUNT; 334 + break; 335 + case DP83867_DOWNSHIFT_4_COUNT_VAL: 336 + count = DP83867_DOWNSHIFT_4_COUNT; 337 + break; 338 + case DP83867_DOWNSHIFT_8_COUNT_VAL: 339 + count = DP83867_DOWNSHIFT_8_COUNT; 340 + break; 341 + default: 342 + return -EINVAL; 343 + }; 344 + 345 + *data = enable ? count : DOWNSHIFT_DEV_DISABLE; 346 + 347 + return 0; 348 + } 349 + 350 + static int dp83867_set_downshift(struct phy_device *phydev, u8 cnt) 351 + { 352 + int val, count; 353 + 354 + if (cnt > DP83867_DOWNSHIFT_8_COUNT) 355 + return -E2BIG; 356 + 357 + if (!cnt) 358 + return phy_clear_bits(phydev, DP83867_CFG2, 359 + DP83867_DOWNSHIFT_EN); 360 + 361 + switch (cnt) { 362 + case DP83867_DOWNSHIFT_1_COUNT: 363 + count = DP83867_DOWNSHIFT_1_COUNT_VAL; 364 + break; 365 + case DP83867_DOWNSHIFT_2_COUNT: 366 + count = DP83867_DOWNSHIFT_2_COUNT_VAL; 367 + break; 368 + case DP83867_DOWNSHIFT_4_COUNT: 369 + count = DP83867_DOWNSHIFT_4_COUNT_VAL; 370 + break; 371 + case DP83867_DOWNSHIFT_8_COUNT: 372 + count = DP83867_DOWNSHIFT_8_COUNT_VAL; 373 + break; 374 + default: 375 + phydev_err(phydev, 376 + "Downshift count must be 1, 2, 4 or 8\n"); 377 + return -EINVAL; 378 + }; 379 + 380 + val = DP83867_DOWNSHIFT_EN; 381 + val |= FIELD_PREP(DP83867_DOWNSHIFT_ATTEMPT_MASK, count); 382 + 383 + return phy_modify(phydev, DP83867_CFG2, 384 + DP83867_DOWNSHIFT_EN | DP83867_DOWNSHIFT_ATTEMPT_MASK, 385 + val); 386 + } 387 + 388 + static int dp83867_get_tunable(struct phy_device *phydev, 389 + struct ethtool_tunable *tuna, void *data) 390 + { 391 + switch (tuna->id) { 392 + case ETHTOOL_PHY_DOWNSHIFT: 393 + return dp83867_get_downshift(phydev, data); 394 + default: 395 + return -EOPNOTSUPP; 396 + } 397 + } 398 + 399 + static int dp83867_set_tunable(struct phy_device *phydev, 400 + struct ethtool_tunable *tuna, const void *data) 401 + { 402 + switch (tuna->id) { 403 + case ETHTOOL_PHY_DOWNSHIFT: 404 + return dp83867_set_downshift(phydev, *(const u8 *)data); 405 + default: 406 + return -EOPNOTSUPP; 407 + } 308 408 } 309 409 310 410 static int dp83867_config_port_mirroring(struct phy_device *phydev) ··· 606 466 struct dp83867_private *dp83867 = phydev->priv; 607 467 int ret, val, bs; 608 468 u16 delay; 469 + 470 + /* Force speed optimization for the PHY even if it strapped */ 471 + ret = phy_modify(phydev, DP83867_CFG2, DP83867_DOWNSHIFT_EN, 472 + DP83867_DOWNSHIFT_EN); 473 + if (ret) 474 + return ret; 609 475 610 476 ret = dp83867_verify_rgmii_cfg(phydev); 611 477 if (ret) ··· 800 654 .probe = dp83867_probe, 801 655 .config_init = dp83867_config_init, 802 656 .soft_reset = dp83867_phy_reset, 657 + 658 + .read_status = dp83867_read_status, 659 + .get_tunable = dp83867_get_tunable, 660 + .set_tunable = dp83867_set_tunable, 803 661 804 662 .get_wol = dp83867_get_wol, 805 663 .set_wol = dp83867_set_wol,