Commit cd5e4123 authored by Lars Persson's avatar Lars Persson Committed by David S. Miller

dwc_eth_qos: do phy_start before resetting hardware

This reverts the changed init order from commit 3647bc35
("dwc_eth_qos: Reset hardware before PHY start") and makes another fix
for the race.

It turned out that the reset state machine of the dwceqos hardware
requires PHY clocks to be present in order to complete the reset
cycle.

To plug the race with the phy state machine we defer link speed
setting until the hardware init has finished.
Signed-off-by: default avatarLars Persson <larper@axis.com>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parent 016a91c6
...@@ -650,6 +650,11 @@ struct net_local { ...@@ -650,6 +650,11 @@ struct net_local {
u32 mmc_tx_counters_mask; u32 mmc_tx_counters_mask;
struct dwceqos_flowcontrol flowcontrol; struct dwceqos_flowcontrol flowcontrol;
/* Tracks the intermediate state of phy started but hardware
* init not finished yet.
*/
bool phy_defer;
}; };
static void dwceqos_read_mmc_counters(struct net_local *lp, u32 rx_mask, static void dwceqos_read_mmc_counters(struct net_local *lp, u32 rx_mask,
...@@ -901,6 +906,9 @@ static void dwceqos_adjust_link(struct net_device *ndev) ...@@ -901,6 +906,9 @@ static void dwceqos_adjust_link(struct net_device *ndev)
struct phy_device *phydev = lp->phy_dev; struct phy_device *phydev = lp->phy_dev;
int status_change = 0; int status_change = 0;
if (lp->phy_defer)
return;
if (phydev->link) { if (phydev->link) {
if ((lp->speed != phydev->speed) || if ((lp->speed != phydev->speed) ||
(lp->duplex != phydev->duplex)) { (lp->duplex != phydev->duplex)) {
...@@ -1635,6 +1643,12 @@ static void dwceqos_init_hw(struct net_local *lp) ...@@ -1635,6 +1643,12 @@ static void dwceqos_init_hw(struct net_local *lp)
regval = dwceqos_read(lp, REG_DWCEQOS_MAC_CFG); regval = dwceqos_read(lp, REG_DWCEQOS_MAC_CFG);
dwceqos_write(lp, REG_DWCEQOS_MAC_CFG, dwceqos_write(lp, REG_DWCEQOS_MAC_CFG,
regval | DWCEQOS_MAC_CFG_TE | DWCEQOS_MAC_CFG_RE); regval | DWCEQOS_MAC_CFG_TE | DWCEQOS_MAC_CFG_RE);
lp->phy_defer = false;
mutex_lock(&lp->phy_dev->lock);
phy_read_status(lp->phy_dev);
dwceqos_adjust_link(lp->ndev);
mutex_unlock(&lp->phy_dev->lock);
} }
static void dwceqos_tx_reclaim(unsigned long data) static void dwceqos_tx_reclaim(unsigned long data)
...@@ -1880,9 +1894,13 @@ static int dwceqos_open(struct net_device *ndev) ...@@ -1880,9 +1894,13 @@ static int dwceqos_open(struct net_device *ndev)
} }
netdev_reset_queue(ndev); netdev_reset_queue(ndev);
/* The dwceqos reset state machine requires all phy clocks to complete,
* hence the unusual init order with phy_start first.
*/
lp->phy_defer = true;
phy_start(lp->phy_dev);
dwceqos_init_hw(lp); dwceqos_init_hw(lp);
napi_enable(&lp->napi); napi_enable(&lp->napi);
phy_start(lp->phy_dev);
netif_start_queue(ndev); netif_start_queue(ndev);
tasklet_enable(&lp->tx_bdreclaim_tasklet); tasklet_enable(&lp->tx_bdreclaim_tasklet);
...@@ -1915,8 +1933,6 @@ static int dwceqos_stop(struct net_device *ndev) ...@@ -1915,8 +1933,6 @@ static int dwceqos_stop(struct net_device *ndev)
{ {
struct net_local *lp = netdev_priv(ndev); struct net_local *lp = netdev_priv(ndev);
phy_stop(lp->phy_dev);
tasklet_disable(&lp->tx_bdreclaim_tasklet); tasklet_disable(&lp->tx_bdreclaim_tasklet);
napi_disable(&lp->napi); napi_disable(&lp->napi);
...@@ -1927,6 +1943,7 @@ static int dwceqos_stop(struct net_device *ndev) ...@@ -1927,6 +1943,7 @@ static int dwceqos_stop(struct net_device *ndev)
dwceqos_drain_dma(lp); dwceqos_drain_dma(lp);
dwceqos_reset_hw(lp); dwceqos_reset_hw(lp);
phy_stop(lp->phy_dev);
dwceqos_descriptor_free(lp); dwceqos_descriptor_free(lp);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment