Phy/Marvell: Rewrite the MV88E1111 phy config function based on kernel code
The original m88e1111s_config() does not do the SGMII mode initialization and is buggy. Rewrite the function according to 3.0.6 kernel function m88e1111_config_init() in drivers/net/phy/marvell.c Signed-off-by: Roy Zang <tie-fei.zang@freescale.com> Acked-by: Andy Fleming <afleming@freescale.com> Cc: Kumar Gala <galak@kernel.crashing.org>
This commit is contained in:
parent
163b29469d
commit
fa12a08ec0
@ -43,6 +43,24 @@
|
||||
#define MIIM_88E1111_PHY_LED_DIRECT 0x4100
|
||||
#define MIIM_88E1111_PHY_LED_COMBINE 0x411C
|
||||
|
||||
/* 88E1111 Extended PHY Specific Control Register */
|
||||
#define MIIM_88E1111_PHY_EXT_CR 0x14
|
||||
#define MIIM_88E1111_RX_DELAY 0x80
|
||||
#define MIIM_88E1111_TX_DELAY 0x2
|
||||
|
||||
/* 88E1111 Extended PHY Specific Status Register */
|
||||
#define MIIM_88E1111_PHY_EXT_SR 0x1b
|
||||
#define MIIM_88E1111_HWCFG_MODE_MASK 0xf
|
||||
#define MIIM_88E1111_HWCFG_MODE_COPPER_RGMII 0xb
|
||||
#define MIIM_88E1111_HWCFG_MODE_FIBER_RGMII 0x3
|
||||
#define MIIM_88E1111_HWCFG_MODE_SGMII_NO_CLK 0x4
|
||||
#define MIIM_88E1111_HWCFG_MODE_COPPER_RTBI 0x9
|
||||
#define MIIM_88E1111_HWCFG_FIBER_COPPER_AUTO 0x8000
|
||||
#define MIIM_88E1111_HWCFG_FIBER_COPPER_RES 0x2000
|
||||
|
||||
#define MIIM_88E1111_COPPER 0
|
||||
#define MIIM_88E1111_FIBER 1
|
||||
|
||||
/* 88E1118 PHY defines */
|
||||
#define MIIM_88E1118_PHY_PAGE 22
|
||||
#define MIIM_88E1118_PHY_LED_PAGE 3
|
||||
@ -162,19 +180,102 @@ static int m88e1011s_startup(struct phy_device *phydev)
|
||||
static int m88e1111s_config(struct phy_device *phydev)
|
||||
{
|
||||
int reg;
|
||||
int timeout;
|
||||
|
||||
if ((phydev->interface == PHY_INTERFACE_MODE_RGMII) ||
|
||||
(phydev->interface == PHY_INTERFACE_MODE_RGMII_ID) ||
|
||||
(phydev->interface == PHY_INTERFACE_MODE_RGMII_RXID) ||
|
||||
(phydev->interface == PHY_INTERFACE_MODE_RGMII_TXID)) {
|
||||
reg = phy_read(phydev, MDIO_DEVAD_NONE, 0x1b);
|
||||
reg = (reg & 0xfff0) | 0xb;
|
||||
phy_write(phydev, MDIO_DEVAD_NONE, 0x1b, reg);
|
||||
} else {
|
||||
phy_write(phydev, MDIO_DEVAD_NONE, 0x1b, 0x1f);
|
||||
reg = phy_read(phydev,
|
||||
MDIO_DEVAD_NONE, MIIM_88E1111_PHY_EXT_CR);
|
||||
if ((phydev->interface == PHY_INTERFACE_MODE_RGMII) ||
|
||||
(phydev->interface == PHY_INTERFACE_MODE_RGMII_ID)) {
|
||||
reg |= (MIIM_88E1111_RX_DELAY | MIIM_88E1111_TX_DELAY);
|
||||
} else if (phydev->interface == PHY_INTERFACE_MODE_RGMII_RXID) {
|
||||
reg &= ~MIIM_88E1111_TX_DELAY;
|
||||
reg |= MIIM_88E1111_RX_DELAY;
|
||||
} else if (phydev->interface == PHY_INTERFACE_MODE_RGMII_TXID) {
|
||||
reg &= ~MIIM_88E1111_RX_DELAY;
|
||||
reg |= MIIM_88E1111_TX_DELAY;
|
||||
}
|
||||
|
||||
phy_write(phydev,
|
||||
MDIO_DEVAD_NONE, MIIM_88E1111_PHY_EXT_CR, reg);
|
||||
|
||||
reg = phy_read(phydev,
|
||||
MDIO_DEVAD_NONE, MIIM_88E1111_PHY_EXT_SR);
|
||||
|
||||
reg &= ~(MIIM_88E1111_HWCFG_MODE_MASK);
|
||||
|
||||
if (reg & MIIM_88E1111_HWCFG_FIBER_COPPER_RES)
|
||||
reg |= MIIM_88E1111_HWCFG_MODE_FIBER_RGMII;
|
||||
else
|
||||
reg |= MIIM_88E1111_HWCFG_MODE_COPPER_RGMII;
|
||||
|
||||
phy_write(phydev,
|
||||
MDIO_DEVAD_NONE, MIIM_88E1111_PHY_EXT_SR, reg);
|
||||
}
|
||||
|
||||
phy_write(phydev, MDIO_DEVAD_NONE, 0x14, 0x0cd2);
|
||||
if (phydev->interface == PHY_INTERFACE_MODE_SGMII) {
|
||||
reg = phy_read(phydev,
|
||||
MDIO_DEVAD_NONE, MIIM_88E1111_PHY_EXT_SR);
|
||||
|
||||
reg &= ~(MIIM_88E1111_HWCFG_MODE_MASK);
|
||||
reg |= MIIM_88E1111_HWCFG_MODE_SGMII_NO_CLK;
|
||||
reg |= MIIM_88E1111_HWCFG_FIBER_COPPER_AUTO;
|
||||
|
||||
phy_write(phydev, MDIO_DEVAD_NONE,
|
||||
MIIM_88E1111_PHY_EXT_SR, reg);
|
||||
}
|
||||
|
||||
if (phydev->interface == PHY_INTERFACE_MODE_RTBI) {
|
||||
reg = phy_read(phydev,
|
||||
MDIO_DEVAD_NONE, MIIM_88E1111_PHY_EXT_CR);
|
||||
reg |= (MIIM_88E1111_RX_DELAY | MIIM_88E1111_TX_DELAY);
|
||||
phy_write(phydev,
|
||||
MDIO_DEVAD_NONE, MIIM_88E1111_PHY_EXT_CR, reg);
|
||||
|
||||
reg = phy_read(phydev, MDIO_DEVAD_NONE,
|
||||
MIIM_88E1111_PHY_EXT_SR);
|
||||
reg &= ~(MIIM_88E1111_HWCFG_MODE_MASK |
|
||||
MIIM_88E1111_HWCFG_FIBER_COPPER_RES);
|
||||
reg |= 0x7 | MIIM_88E1111_HWCFG_FIBER_COPPER_AUTO;
|
||||
phy_write(phydev, MDIO_DEVAD_NONE,
|
||||
MIIM_88E1111_PHY_EXT_SR, reg);
|
||||
|
||||
/* soft reset */
|
||||
timeout = 1000;
|
||||
phy_write(phydev, MDIO_DEVAD_NONE, MII_BMCR, BMCR_RESET);
|
||||
udelay(1000);
|
||||
reg = phy_read(phydev, MDIO_DEVAD_NONE, MII_BMCR);
|
||||
while ((reg & BMCR_RESET) && --timeout) {
|
||||
udelay(1000);
|
||||
reg = phy_read(phydev, MDIO_DEVAD_NONE, MII_BMCR);
|
||||
}
|
||||
if (!timeout)
|
||||
printf("%s: phy soft reset timeout\n", __func__);
|
||||
|
||||
reg = phy_read(phydev, MDIO_DEVAD_NONE,
|
||||
MIIM_88E1111_PHY_EXT_SR);
|
||||
reg &= ~(MIIM_88E1111_HWCFG_MODE_MASK |
|
||||
MIIM_88E1111_HWCFG_FIBER_COPPER_RES);
|
||||
reg |= MIIM_88E1111_HWCFG_MODE_COPPER_RTBI |
|
||||
MIIM_88E1111_HWCFG_FIBER_COPPER_AUTO;
|
||||
phy_write(phydev, MDIO_DEVAD_NONE,
|
||||
MIIM_88E1111_PHY_EXT_SR, reg);
|
||||
}
|
||||
|
||||
/* soft reset */
|
||||
timeout = 1000;
|
||||
phy_write(phydev, MDIO_DEVAD_NONE, MII_BMCR, BMCR_RESET);
|
||||
udelay(1000);
|
||||
reg = phy_read(phydev, MDIO_DEVAD_NONE, MII_BMCR);
|
||||
while ((reg & BMCR_RESET) && --timeout) {
|
||||
udelay(1000);
|
||||
reg = phy_read(phydev, MDIO_DEVAD_NONE, MII_BMCR);
|
||||
}
|
||||
if (!timeout)
|
||||
printf("%s: phy soft reset timeout\n", __func__);
|
||||
|
||||
genphy_config_aneg(phydev);
|
||||
|
||||
|
Loading…
Reference in New Issue
Block a user