mirror of
				git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
				synced 2025-09-04 20:19:47 +08:00 
			
		
		
		
	net: stmmac: Convert to phylink and remove phylib logic
Convert everything to phylink. Signed-off-by: Jose Abreu <joabreu@synopsys.com> Cc: Joao Pinto <jpinto@synopsys.com> Cc: David S. Miller <davem@davemloft.net> Cc: Giuseppe Cavallaro <peppe.cavallaro@st.com> Cc: Alexandre Torgue <alexandre.torgue@st.com> Cc: Russell King <linux@armlinux.org.uk> Cc: Andrew Lunn <andrew@lunn.ch> Cc: Florian Fainelli <f.fainelli@gmail.com> Cc: Heiner Kallweit <hkallweit1@gmail.com> Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
		
							parent
							
								
									eeef2f6b9f
								
							
						
					
					
						commit
						74371272f9
					
				| @ -3,7 +3,6 @@ config STMMAC_ETH | |||||||
| 	tristate "STMicroelectronics 10/100/1000/EQOS Ethernet driver" | 	tristate "STMicroelectronics 10/100/1000/EQOS Ethernet driver" | ||||||
| 	depends on HAS_IOMEM && HAS_DMA | 	depends on HAS_IOMEM && HAS_DMA | ||||||
| 	select MII | 	select MII | ||||||
| 	select PHYLIB |  | ||||||
| 	select PHYLINK | 	select PHYLINK | ||||||
| 	select CRC32 | 	select CRC32 | ||||||
| 	imply PTP_1588_CLOCK | 	imply PTP_1588_CLOCK | ||||||
| @ -42,7 +41,6 @@ if STMMAC_PLATFORM | |||||||
| 
 | 
 | ||||||
| config DWMAC_DWC_QOS_ETH | config DWMAC_DWC_QOS_ETH | ||||||
| 	tristate "Support for snps,dwc-qos-ethernet.txt DT binding." | 	tristate "Support for snps,dwc-qos-ethernet.txt DT binding." | ||||||
| 	select PHYLIB |  | ||||||
| 	select CRC32 | 	select CRC32 | ||||||
| 	select MII | 	select MII | ||||||
| 	depends on OF && HAS_DMA | 	depends on OF && HAS_DMA | ||||||
|  | |||||||
| @ -24,7 +24,6 @@ | |||||||
| 
 | 
 | ||||||
| #include <linux/clk.h> | #include <linux/clk.h> | ||||||
| #include <linux/stmmac.h> | #include <linux/stmmac.h> | ||||||
| #include <linux/phy.h> |  | ||||||
| #include <linux/phylink.h> | #include <linux/phylink.h> | ||||||
| #include <linux/pci.h> | #include <linux/pci.h> | ||||||
| #include "common.h" | #include "common.h" | ||||||
| @ -148,9 +147,7 @@ struct stmmac_priv { | |||||||
| 	/* Generic channel for NAPI */ | 	/* Generic channel for NAPI */ | ||||||
| 	struct stmmac_channel channel[STMMAC_CH_MAX]; | 	struct stmmac_channel channel[STMMAC_CH_MAX]; | ||||||
| 
 | 
 | ||||||
| 	bool oldlink; |  | ||||||
| 	int speed; | 	int speed; | ||||||
| 	int oldduplex; |  | ||||||
| 	unsigned int flow_ctrl; | 	unsigned int flow_ctrl; | ||||||
| 	unsigned int pause; | 	unsigned int pause; | ||||||
| 	struct mii_bus *mii; | 	struct mii_bus *mii; | ||||||
|  | |||||||
| @ -22,7 +22,7 @@ | |||||||
| #include <linux/ethtool.h> | #include <linux/ethtool.h> | ||||||
| #include <linux/interrupt.h> | #include <linux/interrupt.h> | ||||||
| #include <linux/mii.h> | #include <linux/mii.h> | ||||||
| #include <linux/phy.h> | #include <linux/phylink.h> | ||||||
| #include <linux/net_tstamp.h> | #include <linux/net_tstamp.h> | ||||||
| #include <asm/io.h> | #include <asm/io.h> | ||||||
| 
 | 
 | ||||||
| @ -274,7 +274,6 @@ static int stmmac_ethtool_get_link_ksettings(struct net_device *dev, | |||||||
| 					     struct ethtool_link_ksettings *cmd) | 					     struct ethtool_link_ksettings *cmd) | ||||||
| { | { | ||||||
| 	struct stmmac_priv *priv = netdev_priv(dev); | 	struct stmmac_priv *priv = netdev_priv(dev); | ||||||
| 	struct phy_device *phy = dev->phydev; |  | ||||||
| 
 | 
 | ||||||
| 	if (priv->hw->pcs & STMMAC_PCS_RGMII || | 	if (priv->hw->pcs & STMMAC_PCS_RGMII || | ||||||
| 	    priv->hw->pcs & STMMAC_PCS_SGMII) { | 	    priv->hw->pcs & STMMAC_PCS_SGMII) { | ||||||
| @ -353,18 +352,7 @@ static int stmmac_ethtool_get_link_ksettings(struct net_device *dev, | |||||||
| 		return 0; | 		return 0; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	if (phy == NULL) { | 	return phylink_ethtool_ksettings_get(priv->phylink, cmd); | ||||||
| 		pr_err("%s: %s: PHY is not registered\n", |  | ||||||
| 		       __func__, dev->name); |  | ||||||
| 		return -ENODEV; |  | ||||||
| 	} |  | ||||||
| 	if (!netif_running(dev)) { |  | ||||||
| 		pr_err("%s: interface is disabled: we cannot track " |  | ||||||
| 		"link speed / duplex setting\n", dev->name); |  | ||||||
| 		return -EBUSY; |  | ||||||
| 	} |  | ||||||
| 	phy_ethtool_ksettings_get(phy, cmd); |  | ||||||
| 	return 0; |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static int | static int | ||||||
| @ -372,8 +360,6 @@ stmmac_ethtool_set_link_ksettings(struct net_device *dev, | |||||||
| 				  const struct ethtool_link_ksettings *cmd) | 				  const struct ethtool_link_ksettings *cmd) | ||||||
| { | { | ||||||
| 	struct stmmac_priv *priv = netdev_priv(dev); | 	struct stmmac_priv *priv = netdev_priv(dev); | ||||||
| 	struct phy_device *phy = dev->phydev; |  | ||||||
| 	int rc; |  | ||||||
| 
 | 
 | ||||||
| 	if (priv->hw->pcs & STMMAC_PCS_RGMII || | 	if (priv->hw->pcs & STMMAC_PCS_RGMII || | ||||||
| 	    priv->hw->pcs & STMMAC_PCS_SGMII) { | 	    priv->hw->pcs & STMMAC_PCS_SGMII) { | ||||||
| @ -397,9 +383,7 @@ stmmac_ethtool_set_link_ksettings(struct net_device *dev, | |||||||
| 		return 0; | 		return 0; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	rc = phy_ethtool_ksettings_set(phy, cmd); | 	return phylink_ethtool_ksettings_set(priv->phylink, cmd); | ||||||
| 
 |  | ||||||
| 	return rc; |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static u32 stmmac_ethtool_getmsglevel(struct net_device *dev) | static u32 stmmac_ethtool_getmsglevel(struct net_device *dev) | ||||||
| @ -443,6 +427,13 @@ static void stmmac_ethtool_gregs(struct net_device *dev, | |||||||
| 	       NUM_DWMAC1000_DMA_REGS * 4); | 	       NUM_DWMAC1000_DMA_REGS * 4); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | static int stmmac_nway_reset(struct net_device *dev) | ||||||
|  | { | ||||||
|  | 	struct stmmac_priv *priv = netdev_priv(dev); | ||||||
|  | 
 | ||||||
|  | 	return phylink_ethtool_nway_reset(priv->phylink); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| static void | static void | ||||||
| stmmac_get_pauseparam(struct net_device *netdev, | stmmac_get_pauseparam(struct net_device *netdev, | ||||||
| 		      struct ethtool_pauseparam *pause) | 		      struct ethtool_pauseparam *pause) | ||||||
| @ -450,28 +441,13 @@ stmmac_get_pauseparam(struct net_device *netdev, | |||||||
| 	struct stmmac_priv *priv = netdev_priv(netdev); | 	struct stmmac_priv *priv = netdev_priv(netdev); | ||||||
| 	struct rgmii_adv adv_lp; | 	struct rgmii_adv adv_lp; | ||||||
| 
 | 
 | ||||||
| 	pause->rx_pause = 0; |  | ||||||
| 	pause->tx_pause = 0; |  | ||||||
| 
 |  | ||||||
| 	if (priv->hw->pcs && !stmmac_pcs_get_adv_lp(priv, priv->ioaddr, &adv_lp)) { | 	if (priv->hw->pcs && !stmmac_pcs_get_adv_lp(priv, priv->ioaddr, &adv_lp)) { | ||||||
| 		pause->autoneg = 1; | 		pause->autoneg = 1; | ||||||
| 		if (!adv_lp.pause) | 		if (!adv_lp.pause) | ||||||
| 			return; | 			return; | ||||||
| 	} else { | 	} else { | ||||||
| 		if (!linkmode_test_bit(ETHTOOL_LINK_MODE_Pause_BIT, | 		phylink_ethtool_get_pauseparam(priv->phylink, pause); | ||||||
| 				       netdev->phydev->supported) || |  | ||||||
| 		    !linkmode_test_bit(ETHTOOL_LINK_MODE_Asym_Pause_BIT, |  | ||||||
| 				      netdev->phydev->supported)) |  | ||||||
| 			return; |  | ||||||
| 	} | 	} | ||||||
| 
 |  | ||||||
| 	pause->autoneg = netdev->phydev->autoneg; |  | ||||||
| 
 |  | ||||||
| 	if (priv->flow_ctrl & FLOW_RX) |  | ||||||
| 		pause->rx_pause = 1; |  | ||||||
| 	if (priv->flow_ctrl & FLOW_TX) |  | ||||||
| 		pause->tx_pause = 1; |  | ||||||
| 
 |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static int | static int | ||||||
| @ -479,39 +455,16 @@ stmmac_set_pauseparam(struct net_device *netdev, | |||||||
| 		      struct ethtool_pauseparam *pause) | 		      struct ethtool_pauseparam *pause) | ||||||
| { | { | ||||||
| 	struct stmmac_priv *priv = netdev_priv(netdev); | 	struct stmmac_priv *priv = netdev_priv(netdev); | ||||||
| 	u32 tx_cnt = priv->plat->tx_queues_to_use; |  | ||||||
| 	struct phy_device *phy = netdev->phydev; |  | ||||||
| 	int new_pause = FLOW_OFF; |  | ||||||
| 	struct rgmii_adv adv_lp; | 	struct rgmii_adv adv_lp; | ||||||
| 
 | 
 | ||||||
| 	if (priv->hw->pcs && !stmmac_pcs_get_adv_lp(priv, priv->ioaddr, &adv_lp)) { | 	if (priv->hw->pcs && !stmmac_pcs_get_adv_lp(priv, priv->ioaddr, &adv_lp)) { | ||||||
| 		pause->autoneg = 1; | 		pause->autoneg = 1; | ||||||
| 		if (!adv_lp.pause) | 		if (!adv_lp.pause) | ||||||
| 			return -EOPNOTSUPP; | 			return -EOPNOTSUPP; | ||||||
| 	} else { |  | ||||||
| 		if (!linkmode_test_bit(ETHTOOL_LINK_MODE_Pause_BIT, |  | ||||||
| 				       phy->supported) || |  | ||||||
| 		    !linkmode_test_bit(ETHTOOL_LINK_MODE_Asym_Pause_BIT, |  | ||||||
| 				      phy->supported)) |  | ||||||
| 			return -EOPNOTSUPP; |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	if (pause->rx_pause) |  | ||||||
| 		new_pause |= FLOW_RX; |  | ||||||
| 	if (pause->tx_pause) |  | ||||||
| 		new_pause |= FLOW_TX; |  | ||||||
| 
 |  | ||||||
| 	priv->flow_ctrl = new_pause; |  | ||||||
| 	phy->autoneg = pause->autoneg; |  | ||||||
| 
 |  | ||||||
| 	if (phy->autoneg) { |  | ||||||
| 		if (netif_running(netdev)) |  | ||||||
| 			return phy_start_aneg(phy); |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	stmmac_flow_ctrl(priv, priv->hw, phy->duplex, priv->flow_ctrl, |  | ||||||
| 			priv->pause, tx_cnt); |  | ||||||
| 		return 0; | 		return 0; | ||||||
|  | 	} else { | ||||||
|  | 		return phylink_ethtool_set_pauseparam(priv->phylink, pause); | ||||||
|  | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static void stmmac_get_ethtool_stats(struct net_device *dev, | static void stmmac_get_ethtool_stats(struct net_device *dev, | ||||||
| @ -549,7 +502,7 @@ static void stmmac_get_ethtool_stats(struct net_device *dev, | |||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
| 		if (priv->eee_enabled) { | 		if (priv->eee_enabled) { | ||||||
| 			int val = phy_get_eee_err(dev->phydev); | 			int val = phylink_get_eee_err(priv->phylink); | ||||||
| 			if (val) | 			if (val) | ||||||
| 				priv->xstats.phy_eee_wakeup_error_n = val; | 				priv->xstats.phy_eee_wakeup_error_n = val; | ||||||
| 		} | 		} | ||||||
| @ -694,7 +647,7 @@ static int stmmac_ethtool_op_get_eee(struct net_device *dev, | |||||||
| 	edata->eee_active = priv->eee_active; | 	edata->eee_active = priv->eee_active; | ||||||
| 	edata->tx_lpi_timer = priv->tx_lpi_timer; | 	edata->tx_lpi_timer = priv->tx_lpi_timer; | ||||||
| 
 | 
 | ||||||
| 	return phy_ethtool_get_eee(dev->phydev, edata); | 	return phylink_ethtool_get_eee(priv->phylink, edata); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static int stmmac_ethtool_op_set_eee(struct net_device *dev, | static int stmmac_ethtool_op_set_eee(struct net_device *dev, | ||||||
| @ -715,7 +668,7 @@ static int stmmac_ethtool_op_set_eee(struct net_device *dev, | |||||||
| 			return -EOPNOTSUPP; | 			return -EOPNOTSUPP; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	ret = phy_ethtool_set_eee(dev->phydev, edata); | 	ret = phylink_ethtool_set_eee(priv->phylink, edata); | ||||||
| 	if (ret) | 	if (ret) | ||||||
| 		return ret; | 		return ret; | ||||||
| 
 | 
 | ||||||
| @ -892,7 +845,7 @@ static const struct ethtool_ops stmmac_ethtool_ops = { | |||||||
| 	.get_regs = stmmac_ethtool_gregs, | 	.get_regs = stmmac_ethtool_gregs, | ||||||
| 	.get_regs_len = stmmac_ethtool_get_regs_len, | 	.get_regs_len = stmmac_ethtool_get_regs_len, | ||||||
| 	.get_link = ethtool_op_get_link, | 	.get_link = ethtool_op_get_link, | ||||||
| 	.nway_reset = phy_ethtool_nway_reset, | 	.nway_reset = stmmac_nway_reset, | ||||||
| 	.get_pauseparam = stmmac_get_pauseparam, | 	.get_pauseparam = stmmac_get_pauseparam, | ||||||
| 	.set_pauseparam = stmmac_set_pauseparam, | 	.set_pauseparam = stmmac_set_pauseparam, | ||||||
| 	.self_test = stmmac_selftest_run, | 	.self_test = stmmac_selftest_run, | ||||||
|  | |||||||
| @ -328,21 +328,6 @@ static inline u32 stmmac_rx_dirty(struct stmmac_priv *priv, u32 queue) | |||||||
| 	return dirty; | 	return dirty; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /**
 |  | ||||||
|  * stmmac_hw_fix_mac_speed - callback for speed selection |  | ||||||
|  * @priv: driver private structure |  | ||||||
|  * Description: on some platforms (e.g. ST), some HW system configuration |  | ||||||
|  * registers have to be set according to the link speed negotiated. |  | ||||||
|  */ |  | ||||||
| static inline void stmmac_hw_fix_mac_speed(struct stmmac_priv *priv) |  | ||||||
| { |  | ||||||
| 	struct net_device *ndev = priv->dev; |  | ||||||
| 	struct phy_device *phydev = ndev->phydev; |  | ||||||
| 
 |  | ||||||
| 	if (likely(priv->plat->fix_mac_speed)) |  | ||||||
| 		priv->plat->fix_mac_speed(priv->plat->bsp_priv, phydev->speed); |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| /**
 | /**
 | ||||||
|  * stmmac_enable_eee_mode - check and enter in LPI mode |  * stmmac_enable_eee_mode - check and enter in LPI mode | ||||||
|  * @priv: driver private structure |  * @priv: driver private structure | ||||||
| @ -406,14 +391,7 @@ static void stmmac_eee_ctrl_timer(struct timer_list *t) | |||||||
|  */ |  */ | ||||||
| bool stmmac_eee_init(struct stmmac_priv *priv) | bool stmmac_eee_init(struct stmmac_priv *priv) | ||||||
| { | { | ||||||
| 	struct net_device *ndev = priv->dev; | 	int tx_lpi_timer = priv->tx_lpi_timer; | ||||||
| 	int interface = priv->plat->interface; |  | ||||||
| 	bool ret = false; |  | ||||||
| 
 |  | ||||||
| 	if ((interface != PHY_INTERFACE_MODE_MII) && |  | ||||||
| 	    (interface != PHY_INTERFACE_MODE_GMII) && |  | ||||||
| 	    !phy_interface_mode_is_rgmii(interface)) |  | ||||||
| 		goto out; |  | ||||||
| 
 | 
 | ||||||
| 	/* Using PCS we cannot dial with the phy registers at this stage
 | 	/* Using PCS we cannot dial with the phy registers at this stage
 | ||||||
| 	 * so we do not support extra feature like EEE. | 	 * so we do not support extra feature like EEE. | ||||||
| @ -421,52 +399,32 @@ bool stmmac_eee_init(struct stmmac_priv *priv) | |||||||
| 	if ((priv->hw->pcs == STMMAC_PCS_RGMII) || | 	if ((priv->hw->pcs == STMMAC_PCS_RGMII) || | ||||||
| 	    (priv->hw->pcs == STMMAC_PCS_TBI) || | 	    (priv->hw->pcs == STMMAC_PCS_TBI) || | ||||||
| 	    (priv->hw->pcs == STMMAC_PCS_RTBI)) | 	    (priv->hw->pcs == STMMAC_PCS_RTBI)) | ||||||
| 		goto out; | 		return false; | ||||||
| 
 | 
 | ||||||
| 	/* MAC core supports the EEE feature. */ | 	/* Check if MAC core supports the EEE feature. */ | ||||||
| 	if (priv->dma_cap.eee) { | 	if (!priv->dma_cap.eee) | ||||||
| 		int tx_lpi_timer = priv->tx_lpi_timer; | 		return false; | ||||||
| 
 | 
 | ||||||
| 		/* Check if the PHY supports EEE */ |  | ||||||
| 		if (phy_init_eee(ndev->phydev, 1)) { |  | ||||||
| 			/* To manage at run-time if the EEE cannot be supported
 |  | ||||||
| 			 * anymore (for example because the lp caps have been |  | ||||||
| 			 * changed). |  | ||||||
| 			 * In that case the driver disable own timers. |  | ||||||
| 			 */ |  | ||||||
| 	mutex_lock(&priv->lock); | 	mutex_lock(&priv->lock); | ||||||
| 			if (priv->eee_active) { | 
 | ||||||
|  | 	/* Check if it needs to be deactivated */ | ||||||
|  | 	if (!priv->eee_active && priv->eee_enabled) { | ||||||
| 		netdev_dbg(priv->dev, "disable EEE\n"); | 		netdev_dbg(priv->dev, "disable EEE\n"); | ||||||
| 		del_timer_sync(&priv->eee_ctrl_timer); | 		del_timer_sync(&priv->eee_ctrl_timer); | ||||||
| 				stmmac_set_eee_timer(priv, priv->hw, 0, | 		stmmac_set_eee_timer(priv, priv->hw, 0, tx_lpi_timer); | ||||||
|  | 		return false; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if (priv->eee_active && !priv->eee_enabled) { | ||||||
|  | 		timer_setup(&priv->eee_ctrl_timer, stmmac_eee_ctrl_timer, 0); | ||||||
|  | 		mod_timer(&priv->eee_ctrl_timer, STMMAC_LPI_T(eee_timer)); | ||||||
|  | 		stmmac_set_eee_timer(priv, priv->hw, STMMAC_DEFAULT_LIT_LS, | ||||||
| 				     tx_lpi_timer); | 				     tx_lpi_timer); | ||||||
| 	} | 	} | ||||||
| 			priv->eee_active = 0; | 
 | ||||||
| 	mutex_unlock(&priv->lock); | 	mutex_unlock(&priv->lock); | ||||||
| 			goto out; |  | ||||||
| 		} |  | ||||||
| 		/* Activate the EEE and start timers */ |  | ||||||
| 		mutex_lock(&priv->lock); |  | ||||||
| 		if (!priv->eee_active) { |  | ||||||
| 			priv->eee_active = 1; |  | ||||||
| 			timer_setup(&priv->eee_ctrl_timer, |  | ||||||
| 				    stmmac_eee_ctrl_timer, 0); |  | ||||||
| 			mod_timer(&priv->eee_ctrl_timer, |  | ||||||
| 				  STMMAC_LPI_T(eee_timer)); |  | ||||||
| 
 |  | ||||||
| 			stmmac_set_eee_timer(priv, priv->hw, |  | ||||||
| 					STMMAC_DEFAULT_LIT_LS, tx_lpi_timer); |  | ||||||
| 		} |  | ||||||
| 		/* Set HW EEE according to the speed */ |  | ||||||
| 		stmmac_set_eee_pls(priv, priv->hw, ndev->phydev->link); |  | ||||||
| 
 |  | ||||||
| 		ret = true; |  | ||||||
| 		mutex_unlock(&priv->lock); |  | ||||||
| 
 |  | ||||||
| 	netdev_dbg(priv->dev, "Energy-Efficient Ethernet initialized\n"); | 	netdev_dbg(priv->dev, "Energy-Efficient Ethernet initialized\n"); | ||||||
| 	} | 	return true; | ||||||
| out: |  | ||||||
| 	return ret; |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /* stmmac_get_tx_hwtstamp - get HW TX timestamps
 | /* stmmac_get_tx_hwtstamp - get HW TX timestamps
 | ||||||
| @ -882,18 +840,16 @@ static int stmmac_mac_link_state(struct phylink_config *config, | |||||||
| 	return -EOPNOTSUPP; | 	return -EOPNOTSUPP; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static void stmmac_mac_config(struct net_device *dev) | static void stmmac_mac_config(struct phylink_config *config, unsigned int mode, | ||||||
|  | 			      const struct phylink_link_state *state) | ||||||
| { | { | ||||||
| 	struct stmmac_priv *priv = netdev_priv(dev); | 	struct stmmac_priv *priv = netdev_priv(to_net_dev(config->dev)); | ||||||
| 	struct phy_device *phydev = dev->phydev; |  | ||||||
| 	u32 ctrl; | 	u32 ctrl; | ||||||
| 
 | 
 | ||||||
| 	ctrl = readl(priv->ioaddr + MAC_CTRL_REG); | 	ctrl = readl(priv->ioaddr + MAC_CTRL_REG); | ||||||
| 
 |  | ||||||
| 	if (phydev->speed != priv->speed) { |  | ||||||
| 	ctrl &= ~priv->hw->link.speed_mask; | 	ctrl &= ~priv->hw->link.speed_mask; | ||||||
| 
 | 
 | ||||||
| 		switch (phydev->speed) { | 	switch (state->speed) { | ||||||
| 	case SPEED_1000: | 	case SPEED_1000: | ||||||
| 		ctrl |= priv->hw->link.speed1000; | 		ctrl |= priv->hw->link.speed1000; | ||||||
| 		break; | 		break; | ||||||
| @ -904,32 +860,22 @@ static void stmmac_mac_config(struct net_device *dev) | |||||||
| 		ctrl |= priv->hw->link.speed10; | 		ctrl |= priv->hw->link.speed10; | ||||||
| 		break; | 		break; | ||||||
| 	default: | 	default: | ||||||
| 			netif_warn(priv, link, priv->dev, | 		return; | ||||||
| 				   "broken speed: %d\n", phydev->speed); |  | ||||||
| 			phydev->speed = SPEED_UNKNOWN; |  | ||||||
| 			break; |  | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 		if (phydev->speed != SPEED_UNKNOWN) | 	priv->speed = state->speed; | ||||||
| 			stmmac_hw_fix_mac_speed(priv); |  | ||||||
| 
 | 
 | ||||||
| 		priv->speed = phydev->speed; | 	if (priv->plat->fix_mac_speed) | ||||||
| 	} | 		priv->plat->fix_mac_speed(priv->plat->bsp_priv, state->speed); | ||||||
| 
 | 
 | ||||||
| 	/* Now we make sure that we can be in full duplex mode.
 | 	if (!state->duplex) | ||||||
| 	 * If not, we operate in half-duplex mode. */ |  | ||||||
| 	if (phydev->duplex != priv->oldduplex) { |  | ||||||
| 		if (!phydev->duplex) |  | ||||||
| 		ctrl &= ~priv->hw->link.duplex; | 		ctrl &= ~priv->hw->link.duplex; | ||||||
| 	else | 	else | ||||||
| 		ctrl |= priv->hw->link.duplex; | 		ctrl |= priv->hw->link.duplex; | ||||||
| 
 | 
 | ||||||
| 		priv->oldduplex = phydev->duplex; |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	/* Flow Control operation */ | 	/* Flow Control operation */ | ||||||
| 	if (phydev->pause) | 	if (state->pause) | ||||||
| 		stmmac_mac_flow_ctrl(priv, phydev->duplex); | 		stmmac_mac_flow_ctrl(priv, state->duplex); | ||||||
| 
 | 
 | ||||||
| 	writel(ctrl, priv->ioaddr + MAC_CTRL_REG); | 	writel(ctrl, priv->ioaddr + MAC_CTRL_REG); | ||||||
| } | } | ||||||
| @ -939,85 +885,40 @@ static void stmmac_mac_an_restart(struct phylink_config *config) | |||||||
| 	/* Not Supported */ | 	/* Not Supported */ | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static void stmmac_mac_link_down(struct net_device *dev, bool autoneg) | static void stmmac_mac_link_down(struct phylink_config *config, | ||||||
|  | 				 unsigned int mode, phy_interface_t interface) | ||||||
| { | { | ||||||
| 	struct stmmac_priv *priv = netdev_priv(dev); | 	struct stmmac_priv *priv = netdev_priv(to_net_dev(config->dev)); | ||||||
| 
 | 
 | ||||||
| 	stmmac_mac_set(priv, priv->ioaddr, false); | 	stmmac_mac_set(priv, priv->ioaddr, false); | ||||||
|  | 	priv->eee_active = false; | ||||||
|  | 	stmmac_eee_init(priv); | ||||||
|  | 	stmmac_set_eee_pls(priv, priv->hw, false); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static void stmmac_mac_link_up(struct net_device *dev, bool autoneg) | static void stmmac_mac_link_up(struct phylink_config *config, | ||||||
|  | 			       unsigned int mode, phy_interface_t interface, | ||||||
|  | 			       struct phy_device *phy) | ||||||
| { | { | ||||||
| 	struct stmmac_priv *priv = netdev_priv(dev); | 	struct stmmac_priv *priv = netdev_priv(to_net_dev(config->dev)); | ||||||
| 
 | 
 | ||||||
| 	stmmac_mac_set(priv, priv->ioaddr, true); | 	stmmac_mac_set(priv, priv->ioaddr, true); | ||||||
|  | 	if (phy) { | ||||||
|  | 		priv->eee_active = phy_init_eee(phy, 1) >= 0; | ||||||
|  | 		priv->eee_enabled = stmmac_eee_init(priv); | ||||||
|  | 		stmmac_set_eee_pls(priv, priv->hw, true); | ||||||
|  | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static const struct phylink_mac_ops __maybe_unused stmmac_phylink_mac_ops = { | static const struct phylink_mac_ops stmmac_phylink_mac_ops = { | ||||||
| 	.validate = stmmac_validate, | 	.validate = stmmac_validate, | ||||||
| 	.mac_link_state = stmmac_mac_link_state, | 	.mac_link_state = stmmac_mac_link_state, | ||||||
| 	.mac_config = NULL, /* TO BE FILLED */ | 	.mac_config = stmmac_mac_config, | ||||||
| 	.mac_an_restart = stmmac_mac_an_restart, | 	.mac_an_restart = stmmac_mac_an_restart, | ||||||
| 	.mac_link_down = NULL, /* TO BE FILLED */ | 	.mac_link_down = stmmac_mac_link_down, | ||||||
| 	.mac_link_up = NULL, /* TO BE FILLED */ | 	.mac_link_up = stmmac_mac_link_up, | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| /**
 |  | ||||||
|  * stmmac_adjust_link - adjusts the link parameters |  | ||||||
|  * @dev: net device structure |  | ||||||
|  * Description: this is the helper called by the physical abstraction layer |  | ||||||
|  * drivers to communicate the phy link status. According the speed and duplex |  | ||||||
|  * this driver can invoke registered glue-logic as well. |  | ||||||
|  * It also invoke the eee initialization because it could happen when switch |  | ||||||
|  * on different networks (that are eee capable). |  | ||||||
|  */ |  | ||||||
| static void stmmac_adjust_link(struct net_device *dev) |  | ||||||
| { |  | ||||||
| 	struct stmmac_priv *priv = netdev_priv(dev); |  | ||||||
| 	struct phy_device *phydev = dev->phydev; |  | ||||||
| 	bool new_state = false; |  | ||||||
| 
 |  | ||||||
| 	if (!phydev) |  | ||||||
| 		return; |  | ||||||
| 
 |  | ||||||
| 	mutex_lock(&priv->lock); |  | ||||||
| 
 |  | ||||||
| 	if (phydev->link) { |  | ||||||
| 		stmmac_mac_config(dev); |  | ||||||
| 
 |  | ||||||
| 		if (!priv->oldlink) { |  | ||||||
| 			new_state = true; |  | ||||||
| 			priv->oldlink = true; |  | ||||||
| 		} |  | ||||||
| 	} else if (priv->oldlink) { |  | ||||||
| 		new_state = true; |  | ||||||
| 		priv->oldlink = false; |  | ||||||
| 		priv->speed = SPEED_UNKNOWN; |  | ||||||
| 		priv->oldduplex = DUPLEX_UNKNOWN; |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	if (phydev->link) |  | ||||||
| 		stmmac_mac_link_up(dev, false); |  | ||||||
| 	else |  | ||||||
| 		stmmac_mac_link_down(dev, false); |  | ||||||
| 
 |  | ||||||
| 	if (new_state && netif_msg_link(priv)) |  | ||||||
| 		phy_print_status(phydev); |  | ||||||
| 
 |  | ||||||
| 	mutex_unlock(&priv->lock); |  | ||||||
| 
 |  | ||||||
| 	if (phydev->is_pseudo_fixed_link) |  | ||||||
| 		/* Stop PHY layer to call the hook to adjust the link in case
 |  | ||||||
| 		 * of a switch is attached to the stmmac driver. |  | ||||||
| 		 */ |  | ||||||
| 		phydev->irq = PHY_IGNORE_INTERRUPT; |  | ||||||
| 	else |  | ||||||
| 		/* At this stage, init the EEE if supported.
 |  | ||||||
| 		 * Never called in case of fixed_link. |  | ||||||
| 		 */ |  | ||||||
| 		priv->eee_enabled = stmmac_eee_init(priv); |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| /**
 | /**
 | ||||||
|  * stmmac_check_pcs_mode - verify if RGMII/SGMII is supported |  * stmmac_check_pcs_mode - verify if RGMII/SGMII is supported | ||||||
|  * @priv: driver private structure |  * @priv: driver private structure | ||||||
| @ -1054,79 +955,44 @@ static void stmmac_check_pcs_mode(struct stmmac_priv *priv) | |||||||
| static int stmmac_init_phy(struct net_device *dev) | static int stmmac_init_phy(struct net_device *dev) | ||||||
| { | { | ||||||
| 	struct stmmac_priv *priv = netdev_priv(dev); | 	struct stmmac_priv *priv = netdev_priv(dev); | ||||||
| 	u32 tx_cnt = priv->plat->tx_queues_to_use; | 	struct device_node *node; | ||||||
| 	struct phy_device *phydev; | 	int ret; | ||||||
| 	char phy_id_fmt[MII_BUS_ID_SIZE + 3]; |  | ||||||
| 	char bus_id[MII_BUS_ID_SIZE]; |  | ||||||
| 	int interface = priv->plat->interface; |  | ||||||
| 	int max_speed = priv->plat->max_speed; |  | ||||||
| 	priv->oldlink = false; |  | ||||||
| 	priv->speed = SPEED_UNKNOWN; |  | ||||||
| 	priv->oldduplex = DUPLEX_UNKNOWN; |  | ||||||
| 
 | 
 | ||||||
| 	if (priv->plat->phy_node) { | 	node = priv->plat->phy_node; | ||||||
| 		phydev = of_phy_connect(dev, priv->plat->phy_node, | 
 | ||||||
| 					&stmmac_adjust_link, 0, interface); | 	if (node) { | ||||||
|  | 		ret = phylink_of_phy_connect(priv->phylink, node, 0); | ||||||
| 	} else { | 	} else { | ||||||
| 		snprintf(bus_id, MII_BUS_ID_SIZE, "stmmac-%x", | 		int addr = priv->plat->phy_addr; | ||||||
| 			 priv->plat->bus_id); | 		struct phy_device *phydev; | ||||||
| 
 | 
 | ||||||
| 		snprintf(phy_id_fmt, MII_BUS_ID_SIZE + 3, PHY_ID_FMT, bus_id, | 		phydev = mdiobus_get_phy(priv->mii, addr); | ||||||
| 			 priv->plat->phy_addr); | 		if (!phydev) { | ||||||
| 		netdev_dbg(priv->dev, "%s: trying to attach to %s\n", __func__, | 			netdev_err(priv->dev, "no phy at addr %d\n", addr); | ||||||
| 			   phy_id_fmt); |  | ||||||
| 
 |  | ||||||
| 		phydev = phy_connect(dev, phy_id_fmt, &stmmac_adjust_link, |  | ||||||
| 				     interface); |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	if (IS_ERR_OR_NULL(phydev)) { |  | ||||||
| 		netdev_err(priv->dev, "Could not attach to PHY\n"); |  | ||||||
| 		if (!phydev) |  | ||||||
| 			return -ENODEV; |  | ||||||
| 
 |  | ||||||
| 		return PTR_ERR(phydev); |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	/* Stop Advertising 1000BASE Capability if interface is not GMII */ |  | ||||||
| 	if ((interface == PHY_INTERFACE_MODE_MII) || |  | ||||||
| 	    (interface == PHY_INTERFACE_MODE_RMII) || |  | ||||||
| 		(max_speed < 1000 && max_speed > 0)) |  | ||||||
| 		phy_set_max_speed(phydev, SPEED_100); |  | ||||||
| 
 |  | ||||||
| 	/*
 |  | ||||||
| 	 * Half-duplex mode not supported with multiqueue |  | ||||||
| 	 * half-duplex can only works with single queue |  | ||||||
| 	 */ |  | ||||||
| 	if (tx_cnt > 1) { |  | ||||||
| 		phy_remove_link_mode(phydev, |  | ||||||
| 				     ETHTOOL_LINK_MODE_10baseT_Half_BIT); |  | ||||||
| 		phy_remove_link_mode(phydev, |  | ||||||
| 				     ETHTOOL_LINK_MODE_100baseT_Half_BIT); |  | ||||||
| 		phy_remove_link_mode(phydev, |  | ||||||
| 				     ETHTOOL_LINK_MODE_1000baseT_Half_BIT); |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	/*
 |  | ||||||
| 	 * Broken HW is sometimes missing the pull-up resistor on the |  | ||||||
| 	 * MDIO line, which results in reads to non-existent devices returning |  | ||||||
| 	 * 0 rather than 0xffff. Catch this here and treat 0 as a non-existent |  | ||||||
| 	 * device as well. |  | ||||||
| 	 * Note: phydev->phy_id is the result of reading the UID PHY registers. |  | ||||||
| 	 */ |  | ||||||
| 	if (!priv->plat->phy_node && phydev->phy_id == 0) { |  | ||||||
| 		phy_disconnect(phydev); |  | ||||||
| 			return -ENODEV; | 			return -ENODEV; | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 	/* stmmac_adjust_link will change this to PHY_IGNORE_INTERRUPT to avoid
 | 		ret = phylink_connect_phy(priv->phylink, phydev); | ||||||
| 	 * subsequent PHY polling, make sure we force a link transition if | 	} | ||||||
| 	 * we have a UP/DOWN/UP transition |  | ||||||
| 	 */ |  | ||||||
| 	if (phydev->is_pseudo_fixed_link) |  | ||||||
| 		phydev->irq = PHY_POLL; |  | ||||||
| 
 | 
 | ||||||
| 	phy_attached_info(phydev); | 	return ret; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int stmmac_phy_setup(struct stmmac_priv *priv) | ||||||
|  | { | ||||||
|  | 	struct device_node *node = priv->plat->phy_node; | ||||||
|  | 	int mode = priv->plat->interface; | ||||||
|  | 	struct phylink *phylink; | ||||||
|  | 
 | ||||||
|  | 	priv->phylink_config.dev = &priv->dev->dev; | ||||||
|  | 	priv->phylink_config.type = PHYLINK_NETDEV; | ||||||
|  | 
 | ||||||
|  | 	phylink = phylink_create(&priv->phylink_config, of_fwnode_handle(node), | ||||||
|  | 				 mode, &stmmac_phylink_mac_ops); | ||||||
|  | 	if (IS_ERR(phylink)) | ||||||
|  | 		return PTR_ERR(phylink); | ||||||
|  | 
 | ||||||
|  | 	priv->phylink = phylink; | ||||||
| 	return 0; | 	return 0; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -2739,8 +2605,7 @@ static int stmmac_open(struct net_device *dev) | |||||||
| 
 | 
 | ||||||
| 	stmmac_init_tx_coalesce(priv); | 	stmmac_init_tx_coalesce(priv); | ||||||
| 
 | 
 | ||||||
| 	if (dev->phydev) | 	phylink_start(priv->phylink); | ||||||
| 		phy_start(dev->phydev); |  | ||||||
| 
 | 
 | ||||||
| 	/* Request the IRQ lines */ | 	/* Request the IRQ lines */ | ||||||
| 	ret = request_irq(dev->irq, stmmac_interrupt, | 	ret = request_irq(dev->irq, stmmac_interrupt, | ||||||
| @ -2787,8 +2652,7 @@ lpiirq_error: | |||||||
| wolirq_error: | wolirq_error: | ||||||
| 	free_irq(dev->irq, dev); | 	free_irq(dev->irq, dev); | ||||||
| irq_error: | irq_error: | ||||||
| 	if (dev->phydev) | 	phylink_stop(priv->phylink); | ||||||
| 		phy_stop(dev->phydev); |  | ||||||
| 
 | 
 | ||||||
| 	for (chan = 0; chan < priv->plat->tx_queues_to_use; chan++) | 	for (chan = 0; chan < priv->plat->tx_queues_to_use; chan++) | ||||||
| 		del_timer_sync(&priv->tx_queue[chan].txtimer); | 		del_timer_sync(&priv->tx_queue[chan].txtimer); | ||||||
| @ -2797,9 +2661,7 @@ irq_error: | |||||||
| init_error: | init_error: | ||||||
| 	free_dma_desc_resources(priv); | 	free_dma_desc_resources(priv); | ||||||
| dma_desc_error: | dma_desc_error: | ||||||
| 	if (dev->phydev) | 	phylink_disconnect_phy(priv->phylink); | ||||||
| 		phy_disconnect(dev->phydev); |  | ||||||
| 
 |  | ||||||
| 	return ret; | 	return ret; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -2818,10 +2680,8 @@ static int stmmac_release(struct net_device *dev) | |||||||
| 		del_timer_sync(&priv->eee_ctrl_timer); | 		del_timer_sync(&priv->eee_ctrl_timer); | ||||||
| 
 | 
 | ||||||
| 	/* Stop and disconnect the PHY */ | 	/* Stop and disconnect the PHY */ | ||||||
| 	if (dev->phydev) { | 	phylink_stop(priv->phylink); | ||||||
| 		phy_stop(dev->phydev); | 	phylink_disconnect_phy(priv->phylink); | ||||||
| 		phy_disconnect(dev->phydev); |  | ||||||
| 	} |  | ||||||
| 
 | 
 | ||||||
| 	stmmac_stop_all_queues(priv); | 	stmmac_stop_all_queues(priv); | ||||||
| 
 | 
 | ||||||
| @ -3878,6 +3738,7 @@ static void stmmac_poll_controller(struct net_device *dev) | |||||||
|  */ |  */ | ||||||
| static int stmmac_ioctl(struct net_device *dev, struct ifreq *rq, int cmd) | static int stmmac_ioctl(struct net_device *dev, struct ifreq *rq, int cmd) | ||||||
| { | { | ||||||
|  | 	struct stmmac_priv *priv = netdev_priv (dev); | ||||||
| 	int ret = -EOPNOTSUPP; | 	int ret = -EOPNOTSUPP; | ||||||
| 
 | 
 | ||||||
| 	if (!netif_running(dev)) | 	if (!netif_running(dev)) | ||||||
| @ -3887,9 +3748,7 @@ static int stmmac_ioctl(struct net_device *dev, struct ifreq *rq, int cmd) | |||||||
| 	case SIOCGMIIPHY: | 	case SIOCGMIIPHY: | ||||||
| 	case SIOCGMIIREG: | 	case SIOCGMIIREG: | ||||||
| 	case SIOCSMIIREG: | 	case SIOCSMIIREG: | ||||||
| 		if (!dev->phydev) | 		ret = phylink_mii_ioctl(priv->phylink, rq, cmd); | ||||||
| 			return -EINVAL; |  | ||||||
| 		ret = phy_mii_ioctl(dev->phydev, rq, cmd); |  | ||||||
| 		break; | 		break; | ||||||
| 	case SIOCSHWTSTAMP: | 	case SIOCSHWTSTAMP: | ||||||
| 		ret = stmmac_hwtstamp_set(dev, rq); | 		ret = stmmac_hwtstamp_set(dev, rq); | ||||||
| @ -4480,6 +4339,12 @@ int stmmac_dvr_probe(struct device *device, | |||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | 	ret = stmmac_phy_setup(priv); | ||||||
|  | 	if (ret) { | ||||||
|  | 		netdev_err(ndev, "failed to setup phy (%d)\n", ret); | ||||||
|  | 		goto error_phy_setup; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
| 	ret = register_netdev(ndev); | 	ret = register_netdev(ndev); | ||||||
| 	if (ret) { | 	if (ret) { | ||||||
| 		dev_err(priv->device, "%s: ERROR %i registering the device\n", | 		dev_err(priv->device, "%s: ERROR %i registering the device\n", | ||||||
| @ -4497,6 +4362,8 @@ int stmmac_dvr_probe(struct device *device, | |||||||
| 	return ret; | 	return ret; | ||||||
| 
 | 
 | ||||||
| error_netdev_register: | error_netdev_register: | ||||||
|  | 	phylink_destroy(priv->phylink); | ||||||
|  | error_phy_setup: | ||||||
| 	if (priv->hw->pcs != STMMAC_PCS_RGMII && | 	if (priv->hw->pcs != STMMAC_PCS_RGMII && | ||||||
| 	    priv->hw->pcs != STMMAC_PCS_TBI && | 	    priv->hw->pcs != STMMAC_PCS_TBI && | ||||||
| 	    priv->hw->pcs != STMMAC_PCS_RTBI) | 	    priv->hw->pcs != STMMAC_PCS_RTBI) | ||||||
| @ -4538,6 +4405,7 @@ int stmmac_dvr_remove(struct device *dev) | |||||||
| 	stmmac_mac_set(priv, priv->ioaddr, false); | 	stmmac_mac_set(priv, priv->ioaddr, false); | ||||||
| 	netif_carrier_off(ndev); | 	netif_carrier_off(ndev); | ||||||
| 	unregister_netdev(ndev); | 	unregister_netdev(ndev); | ||||||
|  | 	phylink_destroy(priv->phylink); | ||||||
| 	if (priv->plat->stmmac_rst) | 	if (priv->plat->stmmac_rst) | ||||||
| 		reset_control_assert(priv->plat->stmmac_rst); | 		reset_control_assert(priv->plat->stmmac_rst); | ||||||
| 	clk_disable_unprepare(priv->plat->pclk); | 	clk_disable_unprepare(priv->plat->pclk); | ||||||
| @ -4568,8 +4436,7 @@ int stmmac_suspend(struct device *dev) | |||||||
| 	if (!ndev || !netif_running(ndev)) | 	if (!ndev || !netif_running(ndev)) | ||||||
| 		return 0; | 		return 0; | ||||||
| 
 | 
 | ||||||
| 	if (ndev->phydev) | 	phylink_stop(priv->phylink); | ||||||
| 		phy_stop(ndev->phydev); |  | ||||||
| 
 | 
 | ||||||
| 	mutex_lock(&priv->lock); | 	mutex_lock(&priv->lock); | ||||||
| 
 | 
 | ||||||
| @ -4594,9 +4461,7 @@ int stmmac_suspend(struct device *dev) | |||||||
| 	} | 	} | ||||||
| 	mutex_unlock(&priv->lock); | 	mutex_unlock(&priv->lock); | ||||||
| 
 | 
 | ||||||
| 	priv->oldlink = false; |  | ||||||
| 	priv->speed = SPEED_UNKNOWN; | 	priv->speed = SPEED_UNKNOWN; | ||||||
| 	priv->oldduplex = DUPLEX_UNKNOWN; |  | ||||||
| 	return 0; | 	return 0; | ||||||
| } | } | ||||||
| EXPORT_SYMBOL_GPL(stmmac_suspend); | EXPORT_SYMBOL_GPL(stmmac_suspend); | ||||||
| @ -4680,8 +4545,7 @@ int stmmac_resume(struct device *dev) | |||||||
| 
 | 
 | ||||||
| 	mutex_unlock(&priv->lock); | 	mutex_unlock(&priv->lock); | ||||||
| 
 | 
 | ||||||
| 	if (ndev->phydev) | 	phylink_start(priv->phylink); | ||||||
| 		phy_start(ndev->phydev); |  | ||||||
| 
 | 
 | ||||||
| 	return 0; | 	return 0; | ||||||
| } | } | ||||||
|  | |||||||
| @ -333,21 +333,6 @@ static int stmmac_dt_phy(struct plat_stmmacenet_data *plat, | |||||||
| 		{}, | 		{}, | ||||||
| 	}; | 	}; | ||||||
| 
 | 
 | ||||||
| 	/* If phy-handle property is passed from DT, use it as the PHY */ |  | ||||||
| 	plat->phy_node = of_parse_phandle(np, "phy-handle", 0); |  | ||||||
| 	if (plat->phy_node) |  | ||||||
| 		dev_dbg(dev, "Found phy-handle subnode\n"); |  | ||||||
| 
 |  | ||||||
| 	/* If phy-handle is not specified, check if we have a fixed-phy */ |  | ||||||
| 	if (!plat->phy_node && of_phy_is_fixed_link(np)) { |  | ||||||
| 		if ((of_phy_register_fixed_link(np) < 0)) |  | ||||||
| 			return -ENODEV; |  | ||||||
| 
 |  | ||||||
| 		dev_dbg(dev, "Found fixed-link subnode\n"); |  | ||||||
| 		plat->phy_node = of_node_get(np); |  | ||||||
| 		mdio = false; |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	if (of_match_node(need_mdio_ids, np)) { | 	if (of_match_node(need_mdio_ids, np)) { | ||||||
| 		plat->mdio_node = of_get_child_by_name(np, "mdio"); | 		plat->mdio_node = of_get_child_by_name(np, "mdio"); | ||||||
| 	} else { | 	} else { | ||||||
| @ -396,6 +381,7 @@ stmmac_probe_config_dt(struct platform_device *pdev, const char **mac) | |||||||
| 
 | 
 | ||||||
| 	*mac = of_get_mac_address(np); | 	*mac = of_get_mac_address(np); | ||||||
| 	plat->interface = of_get_phy_mode(np); | 	plat->interface = of_get_phy_mode(np); | ||||||
|  | 	plat->phy_node = np; | ||||||
| 
 | 
 | ||||||
| 	/* Get max speed of operation from device tree */ | 	/* Get max speed of operation from device tree */ | ||||||
| 	if (of_property_read_u32(np, "max-speed", &plat->max_speed)) | 	if (of_property_read_u32(np, "max-speed", &plat->max_speed)) | ||||||
| @ -591,11 +577,6 @@ error_pclk_get: | |||||||
| void stmmac_remove_config_dt(struct platform_device *pdev, | void stmmac_remove_config_dt(struct platform_device *pdev, | ||||||
| 			     struct plat_stmmacenet_data *plat) | 			     struct plat_stmmacenet_data *plat) | ||||||
| { | { | ||||||
| 	struct device_node *np = pdev->dev.of_node; |  | ||||||
| 
 |  | ||||||
| 	if (of_phy_is_fixed_link(np)) |  | ||||||
| 		of_phy_deregister_fixed_link(np); |  | ||||||
| 	of_node_put(plat->phy_node); |  | ||||||
| 	of_node_put(plat->mdio_node); | 	of_node_put(plat->mdio_node); | ||||||
| } | } | ||||||
| #else | #else | ||||||
|  | |||||||
		Loading…
	
		Reference in New Issue
	
	Block a user
	 Jose Abreu
						Jose Abreu