133 lines
		
	
	
		
			3.7 KiB
		
	
	
	
		
			Diff
		
	
	
	
	
	
			
		
		
	
	
			133 lines
		
	
	
		
			3.7 KiB
		
	
	
	
		
			Diff
		
	
	
	
	
	
| From 4a4aca08b11501cb1b2c509113bbb65eb66a1f45 Mon Sep 17 00:00:00 2001
 | |
| From: Russell King <rmk+kernel@armlinux.org.uk>
 | |
| Date: Fri, 14 Apr 2017 14:21:25 +0100
 | |
| Subject: sfp: hack: allow marvell 10G phy support to use SFP
 | |
| 
 | |
| Allow the Marvell 10G PHY to register with the SFP bus, so that SFP+
 | |
| cages can work.  This bypasses phylink, meaning that socket status
 | |
| is not taken into account for the link state.  Also, the tx-disable
 | |
| signal must be commented out in DT for this to work...
 | |
| 
 | |
| Signed-off-by: Russell King <rmk+kernel@armlinux.org.uk>
 | |
| ---
 | |
|  drivers/net/phy/marvell10g.c | 54 +++++++++++++++++++++++++++++++++++++++++++-
 | |
|  1 file changed, 53 insertions(+), 1 deletion(-)
 | |
| 
 | |
| --- a/drivers/net/phy/marvell10g.c
 | |
| +++ b/drivers/net/phy/marvell10g.c
 | |
| @@ -15,8 +15,10 @@
 | |
|   * If both the fiber and copper ports are connected, the first to gain
 | |
|   * link takes priority and the other port is completely locked out.
 | |
|   */
 | |
| +#include <linux/of.h>
 | |
|  #include <linux/phy.h>
 | |
|  #include <linux/marvell_phy.h>
 | |
| +#include <linux/sfp.h>
 | |
|  
 | |
|  enum {
 | |
|  	MV_PMA_BOOT		= 0xc050,
 | |
| @@ -41,6 +43,11 @@ enum {
 | |
|  	MV_AN_RESULT_SPD_10000	= BIT(15),
 | |
|  };
 | |
|  
 | |
| +struct mv3310_priv {
 | |
| +	struct device_node *sfp_node;
 | |
| +	struct sfp_bus *sfp_bus;
 | |
| +};
 | |
| +
 | |
|  static int mv3310_modify(struct phy_device *phydev, int devad, u16 reg,
 | |
|  			 u16 mask, u16 bits)
 | |
|  {
 | |
| @@ -59,8 +66,25 @@ static int mv3310_modify(struct phy_devi
 | |
|  	return ret < 0 ? ret : 1;
 | |
|  }
 | |
|  
 | |
| +static int mv3310_sfp_insert(void *upstream, const struct sfp_eeprom_id *id)
 | |
| +{
 | |
| +	struct phy_device *phydev = upstream;
 | |
| +	struct mv3310_priv *priv = dev_get_drvdata(&phydev->mdio.dev);
 | |
| +
 | |
| +	if (sfp_parse_interface(priv->sfp_bus, id) != PHY_INTERFACE_MODE_10GKR) {
 | |
| +		dev_err(&phydev->mdio.dev, "incompatible SFP module inserted\n");
 | |
| +		return -EINVAL;
 | |
| +	}
 | |
| +	return 0;
 | |
| +}
 | |
| +
 | |
| +static const struct sfp_upstream_ops mv3310_sfp_ops = {
 | |
| +	.module_insert = mv3310_sfp_insert,
 | |
| +};
 | |
| +
 | |
|  static int mv3310_probe(struct phy_device *phydev)
 | |
|  {
 | |
| +	struct mv3310_priv *priv;
 | |
|  	u32 mmd_mask = MDIO_DEVS_PMAPMD | MDIO_DEVS_AN;
 | |
|  	int ret;
 | |
|  
 | |
| @@ -78,9 +102,27 @@ static int mv3310_probe(struct phy_devic
 | |
|  		return -ENODEV;
 | |
|  	}
 | |
|  
 | |
| +	priv = devm_kzalloc(&phydev->mdio.dev, sizeof(*priv), GFP_KERNEL);
 | |
| +	if (!priv)
 | |
| +	return -ENOMEM;
 | |
| +
 | |
| +	dev_set_drvdata(&phydev->mdio.dev, priv);
 | |
| +
 | |
| +	if (phydev->mdio.dev.of_node)
 | |
| +		priv->sfp_node = of_parse_phandle(phydev->mdio.dev.of_node,
 | |
| +						  "sfp", 0);
 | |
| +
 | |
|  	return 0;
 | |
|  }
 | |
|  
 | |
| +static void mv3310_remove(struct phy_device *phydev)
 | |
| +{
 | |
| +	struct mv3310_priv *priv = dev_get_drvdata(&phydev->mdio.dev);
 | |
| +
 | |
| +	if (priv->sfp_bus)
 | |
| +		sfp_unregister_upstream(priv->sfp_bus);
 | |
| +}
 | |
| +
 | |
|  /*
 | |
|   * Resetting the MV88X3310 causes it to become non-responsive.  Avoid
 | |
|   * setting the reset bit(s).
 | |
| @@ -92,6 +134,7 @@ static int mv3310_soft_reset(struct phy_
 | |
|  
 | |
|  static int mv3310_config_init(struct phy_device *phydev)
 | |
|  {
 | |
| +	struct mv3310_priv *priv = dev_get_drvdata(&phydev->mdio.dev);
 | |
|  	__ETHTOOL_DECLARE_LINK_MODE_MASK(supported) = { 0, };
 | |
|  	u32 mask;
 | |
|  	int val;
 | |
| @@ -180,6 +223,14 @@ static int mv3310_config_init(struct phy
 | |
|  	phydev->supported &= mask;
 | |
|  	phydev->advertising &= phydev->supported;
 | |
|  
 | |
| +	/* Would be nice to do this in the probe function, but unfortunately,
 | |
| +	 * phylib doesn't have phydev->attached_dev set there.
 | |
| +	 */
 | |
| +	if (priv->sfp_node && !priv->sfp_bus)
 | |
| +		priv->sfp_bus = sfp_register_upstream(priv->sfp_node,
 | |
| +						      phydev->attached_dev,
 | |
| +						      phydev, &mv3310_sfp_ops);
 | |
| +
 | |
|  	return 0;
 | |
|  }
 | |
|  
 | |
| @@ -363,12 +414,13 @@ static struct phy_driver mv3310_drivers[
 | |
|  				  SUPPORTED_FIBRE |
 | |
|  				  SUPPORTED_10000baseT_Full |
 | |
|  				  SUPPORTED_Backplane,
 | |
| -		.probe		= mv3310_probe,
 | |
|  		.soft_reset	= mv3310_soft_reset,
 | |
|  		.config_init	= mv3310_config_init,
 | |
| +		.probe		= mv3310_probe,
 | |
|  		.config_aneg	= mv3310_config_aneg,
 | |
|  		.aneg_done	= mv3310_aneg_done,
 | |
|  		.read_status	= mv3310_read_status,
 | |
| +		.remove		= mv3310_remove,
 | |
|  	},
 | |
|  };
 | |
|  
 | 
