diff options
author | trem <tremyfr@yahoo.fr> | 2012-12-04 03:52:10 -0500 |
---|---|---|
committer | David S. Miller <davem@davemloft.net> | 2012-12-07 12:48:00 -0500 |
commit | 6ded7cd605502eff7341bbd912ebc90c3674c764 (patch) | |
tree | 70a120c195ffe3e61900de404b54b8a53756030e | |
parent | 4bd9b0fffb193d2e288f67f81821af32df8d4349 (diff) |
net: phy: smsc: force all capable mode if the phy is started in powerdown mode
A SMSC PHY in power down mode can't be used.
If a SMSC PHY is in this mode in the config_init
stage, the mode "all capable" is set. So the PHY
could then be used.
Signed-off-by: Philippe Reynes <tremyfr@yahoo.fr>
Signed-off-by: David S. Miller <davem@davemloft.net>
-rw-r--r-- | drivers/net/phy/smsc.c | 26 | ||||
-rw-r--r-- | include/linux/smscphy.h | 5 |
2 files changed, 30 insertions, 1 deletions
diff --git a/drivers/net/phy/smsc.c b/drivers/net/phy/smsc.c index 16dceed29d8c..ef883e3048c0 100644 --- a/drivers/net/phy/smsc.c +++ b/drivers/net/phy/smsc.c | |||
@@ -43,7 +43,31 @@ static int smsc_phy_ack_interrupt(struct phy_device *phydev) | |||
43 | 43 | ||
44 | static int smsc_phy_config_init(struct phy_device *phydev) | 44 | static int smsc_phy_config_init(struct phy_device *phydev) |
45 | { | 45 | { |
46 | int rc = phy_read(phydev, MII_LAN83C185_CTRL_STATUS); | 46 | int rc = phy_read(phydev, MII_LAN83C185_SPECIAL_MODES); |
47 | if (rc < 0) | ||
48 | return rc; | ||
49 | |||
50 | /* If the SMSC PHY is in power down mode, then set it | ||
51 | * in all capable mode before using it. | ||
52 | */ | ||
53 | if ((rc & MII_LAN83C185_MODE_MASK) == MII_LAN83C185_MODE_POWERDOWN) { | ||
54 | int timeout = 50000; | ||
55 | |||
56 | /* set "all capable" mode and reset the phy */ | ||
57 | rc |= MII_LAN83C185_MODE_ALL; | ||
58 | phy_write(phydev, MII_LAN83C185_SPECIAL_MODES, rc); | ||
59 | phy_write(phydev, MII_BMCR, BMCR_RESET); | ||
60 | |||
61 | /* wait end of reset (max 500 ms) */ | ||
62 | do { | ||
63 | udelay(10); | ||
64 | if (timeout-- == 0) | ||
65 | return -1; | ||
66 | rc = phy_read(phydev, MII_BMCR); | ||
67 | } while (rc & BMCR_RESET); | ||
68 | } | ||
69 | |||
70 | rc = phy_read(phydev, MII_LAN83C185_CTRL_STATUS); | ||
47 | if (rc < 0) | 71 | if (rc < 0) |
48 | return rc; | 72 | return rc; |
49 | 73 | ||
diff --git a/include/linux/smscphy.h b/include/linux/smscphy.h index ce718cbce435..f4bf16e16e16 100644 --- a/include/linux/smscphy.h +++ b/include/linux/smscphy.h | |||
@@ -4,6 +4,7 @@ | |||
4 | #define MII_LAN83C185_ISF 29 /* Interrupt Source Flags */ | 4 | #define MII_LAN83C185_ISF 29 /* Interrupt Source Flags */ |
5 | #define MII_LAN83C185_IM 30 /* Interrupt Mask */ | 5 | #define MII_LAN83C185_IM 30 /* Interrupt Mask */ |
6 | #define MII_LAN83C185_CTRL_STATUS 17 /* Mode/Status Register */ | 6 | #define MII_LAN83C185_CTRL_STATUS 17 /* Mode/Status Register */ |
7 | #define MII_LAN83C185_SPECIAL_MODES 18 /* Special Modes Register */ | ||
7 | 8 | ||
8 | #define MII_LAN83C185_ISF_INT1 (1<<1) /* Auto-Negotiation Page Received */ | 9 | #define MII_LAN83C185_ISF_INT1 (1<<1) /* Auto-Negotiation Page Received */ |
9 | #define MII_LAN83C185_ISF_INT2 (1<<2) /* Parallel Detection Fault */ | 10 | #define MII_LAN83C185_ISF_INT2 (1<<2) /* Parallel Detection Fault */ |
@@ -22,4 +23,8 @@ | |||
22 | #define MII_LAN83C185_EDPWRDOWN (1 << 13) /* EDPWRDOWN */ | 23 | #define MII_LAN83C185_EDPWRDOWN (1 << 13) /* EDPWRDOWN */ |
23 | #define MII_LAN83C185_ENERGYON (1 << 1) /* ENERGYON */ | 24 | #define MII_LAN83C185_ENERGYON (1 << 1) /* ENERGYON */ |
24 | 25 | ||
26 | #define MII_LAN83C185_MODE_MASK 0xE0 | ||
27 | #define MII_LAN83C185_MODE_POWERDOWN 0xC0 /* Power Down mode */ | ||
28 | #define MII_LAN83C185_MODE_ALL 0xE0 /* All capable mode */ | ||
29 | |||
25 | #endif /* __LINUX_SMSCPHY_H__ */ | 30 | #endif /* __LINUX_SMSCPHY_H__ */ |