1018 lines
		
	
	
		
			24 KiB
		
	
	
	
		
			C
		
	
	
	
			
		
		
	
	
			1018 lines
		
	
	
		
			24 KiB
		
	
	
	
		
			C
		
	
	
	
| /*
 | |
|  * (C) Copyright 2015
 | |
|  * Elecsys Corporation <www.elecsyscorp.com>
 | |
|  * Kevin Smith <kevin.smith@elecsyscorp.com>
 | |
|  *
 | |
|  * Original driver:
 | |
|  * (C) Copyright 2009
 | |
|  * Marvell Semiconductor <www.marvell.com>
 | |
|  * Prafulla Wadaskar <prafulla@marvell.com>
 | |
|  *
 | |
|  * SPDX-License-Identifier:	GPL-2.0+
 | |
|  */
 | |
| 
 | |
| /*
 | |
|  * PHY driver for mv88e61xx ethernet switches.
 | |
|  *
 | |
|  * This driver configures the mv88e61xx for basic use as a PHY.  The switch
 | |
|  * supports a VLAN configuration that determines how traffic will be routed
 | |
|  * between the ports.  This driver uses a simple configuration that routes
 | |
|  * traffic from each PHY port only to the CPU port, and from the CPU port to
 | |
|  * any PHY port.
 | |
|  *
 | |
|  * The configuration determines which PHY ports to activate using the
 | |
|  * CONFIG_MV88E61XX_PHY_PORTS bitmask.  Setting bit 0 will activate port 0, bit
 | |
|  * 1 activates port 1, etc.  Do not set the bit for the port the CPU is
 | |
|  * connected to unless it is connected over a PHY interface (not MII).
 | |
|  *
 | |
|  * This driver was written for and tested on the mv88e6176 with an SGMII
 | |
|  * connection.  Other configurations should be supported, but some additions or
 | |
|  * changes may be required.
 | |
|  */
 | |
| 
 | |
| #include <common.h>
 | |
| 
 | |
| #include <bitfield.h>
 | |
| #include <errno.h>
 | |
| #include <malloc.h>
 | |
| #include <miiphy.h>
 | |
| #include <netdev.h>
 | |
| 
 | |
| #define PHY_AUTONEGOTIATE_TIMEOUT	5000
 | |
| 
 | |
| #define PORT_COUNT			7
 | |
| #define PORT_MASK			((1 << PORT_COUNT) - 1)
 | |
| 
 | |
| /* Device addresses */
 | |
| #define DEVADDR_PHY(p)			(p)
 | |
| #define DEVADDR_PORT(p)			(0x10 + (p))
 | |
| #define DEVADDR_SERDES			0x0F
 | |
| #define DEVADDR_GLOBAL_1		0x1B
 | |
| #define DEVADDR_GLOBAL_2		0x1C
 | |
| 
 | |
| /* SMI indirection registers for multichip addressing mode */
 | |
| #define SMI_CMD_REG			0x00
 | |
| #define SMI_DATA_REG			0x01
 | |
| 
 | |
| /* Global registers */
 | |
| #define GLOBAL1_STATUS			0x00
 | |
| #define GLOBAL1_CTRL			0x04
 | |
| #define GLOBAL1_MON_CTRL		0x1A
 | |
| 
 | |
| /* Global 2 registers */
 | |
| #define GLOBAL2_REG_PHY_CMD		0x18
 | |
| #define GLOBAL2_REG_PHY_DATA		0x19
 | |
| 
 | |
| /* Port registers */
 | |
| #define PORT_REG_STATUS			0x00
 | |
| #define PORT_REG_PHYS_CTRL		0x01
 | |
| #define PORT_REG_SWITCH_ID		0x03
 | |
| #define PORT_REG_CTRL			0x04
 | |
| #define PORT_REG_VLAN_MAP		0x06
 | |
| #define PORT_REG_VLAN_ID		0x07
 | |
| 
 | |
| /* Phy registers */
 | |
| #define PHY_REG_CTRL1			0x10
 | |
| #define PHY_REG_STATUS1			0x11
 | |
| #define PHY_REG_PAGE			0x16
 | |
| 
 | |
| /* Serdes registers */
 | |
| #define SERDES_REG_CTRL_1		0x10
 | |
| 
 | |
| /* Phy page numbers */
 | |
| #define PHY_PAGE_COPPER			0
 | |
| #define PHY_PAGE_SERDES			1
 | |
| 
 | |
| /* Register fields */
 | |
| #define GLOBAL1_CTRL_SWRESET		BIT(15)
 | |
| 
 | |
| #define GLOBAL1_MON_CTRL_CPUDEST_SHIFT	4
 | |
| #define GLOBAL1_MON_CTRL_CPUDEST_WIDTH	4
 | |
| 
 | |
| #define PORT_REG_STATUS_LINK		BIT(11)
 | |
| #define PORT_REG_STATUS_DUPLEX		BIT(10)
 | |
| 
 | |
| #define PORT_REG_STATUS_SPEED_SHIFT	8
 | |
| #define PORT_REG_STATUS_SPEED_WIDTH	2
 | |
| #define PORT_REG_STATUS_SPEED_10	0
 | |
| #define PORT_REG_STATUS_SPEED_100	1
 | |
| #define PORT_REG_STATUS_SPEED_1000	2
 | |
| 
 | |
| #define PORT_REG_STATUS_CMODE_MASK		0xF
 | |
| #define PORT_REG_STATUS_CMODE_100BASE_X		0x8
 | |
| #define PORT_REG_STATUS_CMODE_1000BASE_X	0x9
 | |
| #define PORT_REG_STATUS_CMODE_SGMII		0xa
 | |
| 
 | |
| #define PORT_REG_PHYS_CTRL_LINK_VALUE	BIT(5)
 | |
| #define PORT_REG_PHYS_CTRL_LINK_FORCE	BIT(4)
 | |
| 
 | |
| #define PORT_REG_CTRL_PSTATE_SHIFT	0
 | |
| #define PORT_REG_CTRL_PSTATE_WIDTH	2
 | |
| 
 | |
| #define PORT_REG_VLAN_ID_DEF_VID_SHIFT	0
 | |
| #define PORT_REG_VLAN_ID_DEF_VID_WIDTH	12
 | |
| 
 | |
| #define PORT_REG_VLAN_MAP_TABLE_SHIFT	0
 | |
| #define PORT_REG_VLAN_MAP_TABLE_WIDTH	11
 | |
| 
 | |
| #define SERDES_REG_CTRL_1_FORCE_LINK	BIT(10)
 | |
| 
 | |
| #define PHY_REG_CTRL1_ENERGY_DET_SHIFT	8
 | |
| #define PHY_REG_CTRL1_ENERGY_DET_WIDTH	2
 | |
| 
 | |
| /* Field values */
 | |
| #define PORT_REG_CTRL_PSTATE_DISABLED	0
 | |
| #define PORT_REG_CTRL_PSTATE_FORWARD	3
 | |
| 
 | |
| #define PHY_REG_CTRL1_ENERGY_DET_OFF	0
 | |
| #define PHY_REG_CTRL1_ENERGY_DET_SENSE_ONLY	2
 | |
| #define PHY_REG_CTRL1_ENERGY_DET_SENSE_XMIT	3
 | |
| 
 | |
| /* PHY Status Register */
 | |
| #define PHY_REG_STATUS1_SPEED		0xc000
 | |
| #define PHY_REG_STATUS1_GBIT		0x8000
 | |
| #define PHY_REG_STATUS1_100		0x4000
 | |
| #define PHY_REG_STATUS1_DUPLEX		0x2000
 | |
| #define PHY_REG_STATUS1_SPDDONE		0x0800
 | |
| #define PHY_REG_STATUS1_LINK		0x0400
 | |
| #define PHY_REG_STATUS1_ENERGY		0x0010
 | |
| 
 | |
| /*
 | |
|  * Macros for building commands for indirect addressing modes.  These are valid
 | |
|  * for both the indirect multichip addressing mode and the PHY indirection
 | |
|  * required for the writes to any PHY register.
 | |
|  */
 | |
| #define SMI_BUSY			BIT(15)
 | |
| #define SMI_CMD_CLAUSE_22		BIT(12)
 | |
| #define SMI_CMD_CLAUSE_22_OP_READ	(2 << 10)
 | |
| #define SMI_CMD_CLAUSE_22_OP_WRITE	(1 << 10)
 | |
| 
 | |
| #define SMI_CMD_READ			(SMI_BUSY | SMI_CMD_CLAUSE_22 | \
 | |
| 					 SMI_CMD_CLAUSE_22_OP_READ)
 | |
| #define SMI_CMD_WRITE			(SMI_BUSY | SMI_CMD_CLAUSE_22 | \
 | |
| 					 SMI_CMD_CLAUSE_22_OP_WRITE)
 | |
| 
 | |
| #define SMI_CMD_ADDR_SHIFT		5
 | |
| #define SMI_CMD_ADDR_WIDTH		5
 | |
| #define SMI_CMD_REG_SHIFT		0
 | |
| #define SMI_CMD_REG_WIDTH		5
 | |
| 
 | |
| /* Check for required macros */
 | |
| #ifndef CONFIG_MV88E61XX_PHY_PORTS
 | |
| #error Define CONFIG_MV88E61XX_PHY_PORTS to indicate which physical ports \
 | |
| 	to activate
 | |
| #endif
 | |
| #ifndef CONFIG_MV88E61XX_CPU_PORT
 | |
| #error Define CONFIG_MV88E61XX_CPU_PORT to the port the CPU is attached to
 | |
| #endif
 | |
| 
 | |
| /* ID register values for different switch models */
 | |
| #define PORT_SWITCH_ID_6172		0x1720
 | |
| #define PORT_SWITCH_ID_6176		0x1760
 | |
| #define PORT_SWITCH_ID_6240		0x2400
 | |
| #define PORT_SWITCH_ID_6352		0x3520
 | |
| 
 | |
| struct mv88e61xx_phy_priv {
 | |
| 	struct mii_dev *mdio_bus;
 | |
| 	int smi_addr;
 | |
| 	int id;
 | |
| };
 | |
| 
 | |
| static inline int smi_cmd(int cmd, int addr, int reg)
 | |
| {
 | |
| 	cmd = bitfield_replace(cmd, SMI_CMD_ADDR_SHIFT, SMI_CMD_ADDR_WIDTH,
 | |
| 			       addr);
 | |
| 	cmd = bitfield_replace(cmd, SMI_CMD_REG_SHIFT, SMI_CMD_REG_WIDTH, reg);
 | |
| 	return cmd;
 | |
| }
 | |
| 
 | |
| static inline int smi_cmd_read(int addr, int reg)
 | |
| {
 | |
| 	return smi_cmd(SMI_CMD_READ, addr, reg);
 | |
| }
 | |
| 
 | |
| static inline int smi_cmd_write(int addr, int reg)
 | |
| {
 | |
| 	return smi_cmd(SMI_CMD_WRITE, addr, reg);
 | |
| }
 | |
| 
 | |
| __weak int mv88e61xx_hw_reset(struct phy_device *phydev)
 | |
| {
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| /* Wait for the current SMI indirect command to complete */
 | |
| static int mv88e61xx_smi_wait(struct mii_dev *bus, int smi_addr)
 | |
| {
 | |
| 	int val;
 | |
| 	u32 timeout = 100;
 | |
| 
 | |
| 	do {
 | |
| 		val = bus->read(bus, smi_addr, MDIO_DEVAD_NONE, SMI_CMD_REG);
 | |
| 		if (val >= 0 && (val & SMI_BUSY) == 0)
 | |
| 			return 0;
 | |
| 
 | |
| 		mdelay(1);
 | |
| 	} while (--timeout);
 | |
| 
 | |
| 	puts("SMI busy timeout\n");
 | |
| 	return -ETIMEDOUT;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * The mv88e61xx has three types of addresses: the smi bus address, the device
 | |
|  * address, and the register address.  The smi bus address distinguishes it on
 | |
|  * the smi bus from other PHYs or switches.  The device address determines
 | |
|  * which on-chip register set you are reading/writing (the various PHYs, their
 | |
|  * associated ports, or global configuration registers).  The register address
 | |
|  * is the offset of the register you are reading/writing.
 | |
|  *
 | |
|  * When the mv88e61xx is hardware configured to have address zero, it behaves in
 | |
|  * single-chip addressing mode, where it responds to all SMI addresses, using
 | |
|  * the smi address as its device address.  This obviously only works when this
 | |
|  * is the only chip on the SMI bus.  This allows the driver to access device
 | |
|  * registers without using indirection.  When the chip is configured to a
 | |
|  * non-zero address, it only responds to that SMI address and requires indirect
 | |
|  * writes to access the different device addresses.
 | |
|  */
 | |
| static int mv88e61xx_reg_read(struct phy_device *phydev, int dev, int reg)
 | |
| {
 | |
| 	struct mv88e61xx_phy_priv *priv = phydev->priv;
 | |
| 	struct mii_dev *mdio_bus = priv->mdio_bus;
 | |
| 	int smi_addr = priv->smi_addr;
 | |
| 	int res;
 | |
| 
 | |
| 	/* In single-chip mode, the device can be addressed directly */
 | |
| 	if (smi_addr == 0)
 | |
| 		return mdio_bus->read(mdio_bus, dev, MDIO_DEVAD_NONE, reg);
 | |
| 
 | |
| 	/* Wait for the bus to become free */
 | |
| 	res = mv88e61xx_smi_wait(mdio_bus, smi_addr);
 | |
| 	if (res < 0)
 | |
| 		return res;
 | |
| 
 | |
| 	/* Issue the read command */
 | |
| 	res = mdio_bus->write(mdio_bus, smi_addr, MDIO_DEVAD_NONE, SMI_CMD_REG,
 | |
| 			 smi_cmd_read(dev, reg));
 | |
| 	if (res < 0)
 | |
| 		return res;
 | |
| 
 | |
| 	/* Wait for the read command to complete */
 | |
| 	res = mv88e61xx_smi_wait(mdio_bus, smi_addr);
 | |
| 	if (res < 0)
 | |
| 		return res;
 | |
| 
 | |
| 	/* Read the data */
 | |
| 	res = mdio_bus->read(mdio_bus, smi_addr, MDIO_DEVAD_NONE, SMI_DATA_REG);
 | |
| 	if (res < 0)
 | |
| 		return res;
 | |
| 
 | |
| 	return bitfield_extract(res, 0, 16);
 | |
| }
 | |
| 
 | |
| /* See the comment above mv88e61xx_reg_read */
 | |
| static int mv88e61xx_reg_write(struct phy_device *phydev, int dev, int reg,
 | |
| 			       u16 val)
 | |
| {
 | |
| 	struct mv88e61xx_phy_priv *priv = phydev->priv;
 | |
| 	struct mii_dev *mdio_bus = priv->mdio_bus;
 | |
| 	int smi_addr = priv->smi_addr;
 | |
| 	int res;
 | |
| 
 | |
| 	/* In single-chip mode, the device can be addressed directly */
 | |
| 	if (smi_addr == 0) {
 | |
| 		return mdio_bus->write(mdio_bus, dev, MDIO_DEVAD_NONE, reg,
 | |
| 				val);
 | |
| 	}
 | |
| 
 | |
| 	/* Wait for the bus to become free */
 | |
| 	res = mv88e61xx_smi_wait(mdio_bus, smi_addr);
 | |
| 	if (res < 0)
 | |
| 		return res;
 | |
| 
 | |
| 	/* Set the data to write */
 | |
| 	res = mdio_bus->write(mdio_bus, smi_addr, MDIO_DEVAD_NONE,
 | |
| 				SMI_DATA_REG, val);
 | |
| 	if (res < 0)
 | |
| 		return res;
 | |
| 
 | |
| 	/* Issue the write command */
 | |
| 	res = mdio_bus->write(mdio_bus, smi_addr, MDIO_DEVAD_NONE, SMI_CMD_REG,
 | |
| 				smi_cmd_write(dev, reg));
 | |
| 	if (res < 0)
 | |
| 		return res;
 | |
| 
 | |
| 	/* Wait for the write command to complete */
 | |
| 	res = mv88e61xx_smi_wait(mdio_bus, smi_addr);
 | |
| 	if (res < 0)
 | |
| 		return res;
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int mv88e61xx_phy_wait(struct phy_device *phydev)
 | |
| {
 | |
| 	int val;
 | |
| 	u32 timeout = 100;
 | |
| 
 | |
| 	do {
 | |
| 		val = mv88e61xx_reg_read(phydev, DEVADDR_GLOBAL_2,
 | |
| 					 GLOBAL2_REG_PHY_CMD);
 | |
| 		if (val >= 0 && (val & SMI_BUSY) == 0)
 | |
| 			return 0;
 | |
| 
 | |
| 		mdelay(1);
 | |
| 	} while (--timeout);
 | |
| 
 | |
| 	return -ETIMEDOUT;
 | |
| }
 | |
| 
 | |
| static int mv88e61xx_phy_read_indirect(struct mii_dev *smi_wrapper, int dev,
 | |
| 		int devad, int reg)
 | |
| {
 | |
| 	struct phy_device *phydev;
 | |
| 	int res;
 | |
| 
 | |
| 	phydev = (struct phy_device *)smi_wrapper->priv;
 | |
| 
 | |
| 	/* Issue command to read */
 | |
| 	res = mv88e61xx_reg_write(phydev, DEVADDR_GLOBAL_2,
 | |
| 				  GLOBAL2_REG_PHY_CMD,
 | |
| 				  smi_cmd_read(dev, reg));
 | |
| 
 | |
| 	/* Wait for data to be read */
 | |
| 	res = mv88e61xx_phy_wait(phydev);
 | |
| 	if (res < 0)
 | |
| 		return res;
 | |
| 
 | |
| 	/* Read retrieved data */
 | |
| 	return mv88e61xx_reg_read(phydev, DEVADDR_GLOBAL_2,
 | |
| 				  GLOBAL2_REG_PHY_DATA);
 | |
| }
 | |
| 
 | |
| static int mv88e61xx_phy_write_indirect(struct mii_dev *smi_wrapper, int dev,
 | |
| 		int devad, int reg, u16 data)
 | |
| {
 | |
| 	struct phy_device *phydev;
 | |
| 	int res;
 | |
| 
 | |
| 	phydev = (struct phy_device *)smi_wrapper->priv;
 | |
| 
 | |
| 	/* Set the data to write */
 | |
| 	res = mv88e61xx_reg_write(phydev, DEVADDR_GLOBAL_2,
 | |
| 				  GLOBAL2_REG_PHY_DATA, data);
 | |
| 	if (res < 0)
 | |
| 		return res;
 | |
| 	/* Issue the write command */
 | |
| 	res = mv88e61xx_reg_write(phydev, DEVADDR_GLOBAL_2,
 | |
| 				  GLOBAL2_REG_PHY_CMD,
 | |
| 				  smi_cmd_write(dev, reg));
 | |
| 	if (res < 0)
 | |
| 		return res;
 | |
| 
 | |
| 	/* Wait for command to complete */
 | |
| 	return mv88e61xx_phy_wait(phydev);
 | |
| }
 | |
| 
 | |
| /* Wrapper function to make calls to phy_read_indirect simpler */
 | |
| static int mv88e61xx_phy_read(struct phy_device *phydev, int phy, int reg)
 | |
| {
 | |
| 	return mv88e61xx_phy_read_indirect(phydev->bus, DEVADDR_PHY(phy),
 | |
| 					   MDIO_DEVAD_NONE, reg);
 | |
| }
 | |
| 
 | |
| /* Wrapper function to make calls to phy_read_indirect simpler */
 | |
| static int mv88e61xx_phy_write(struct phy_device *phydev, int phy,
 | |
| 		int reg, u16 val)
 | |
| {
 | |
| 	return mv88e61xx_phy_write_indirect(phydev->bus, DEVADDR_PHY(phy),
 | |
| 					    MDIO_DEVAD_NONE, reg, val);
 | |
| }
 | |
| 
 | |
| static int mv88e61xx_port_read(struct phy_device *phydev, u8 port, u8 reg)
 | |
| {
 | |
| 	return mv88e61xx_reg_read(phydev, DEVADDR_PORT(port), reg);
 | |
| }
 | |
| 
 | |
| static int mv88e61xx_port_write(struct phy_device *phydev, u8 port, u8 reg,
 | |
| 								u16 val)
 | |
| {
 | |
| 	return mv88e61xx_reg_write(phydev, DEVADDR_PORT(port), reg, val);
 | |
| }
 | |
| 
 | |
| static int mv88e61xx_set_page(struct phy_device *phydev, u8 phy, u8 page)
 | |
| {
 | |
| 	return mv88e61xx_phy_write(phydev, phy, PHY_REG_PAGE, page);
 | |
| }
 | |
| 
 | |
| static int mv88e61xx_get_switch_id(struct phy_device *phydev)
 | |
| {
 | |
| 	int res;
 | |
| 
 | |
| 	res = mv88e61xx_port_read(phydev, 0, PORT_REG_SWITCH_ID);
 | |
| 	if (res < 0)
 | |
| 		return res;
 | |
| 	return res & 0xfff0;
 | |
| }
 | |
| 
 | |
| static bool mv88e61xx_6352_family(struct phy_device *phydev)
 | |
| {
 | |
| 	struct mv88e61xx_phy_priv *priv = phydev->priv;
 | |
| 
 | |
| 	switch (priv->id) {
 | |
| 	case PORT_SWITCH_ID_6172:
 | |
| 	case PORT_SWITCH_ID_6176:
 | |
| 	case PORT_SWITCH_ID_6240:
 | |
| 	case PORT_SWITCH_ID_6352:
 | |
| 		return true;
 | |
| 	}
 | |
| 	return false;
 | |
| }
 | |
| 
 | |
| static int mv88e61xx_get_cmode(struct phy_device *phydev, u8 port)
 | |
| {
 | |
| 	int res;
 | |
| 
 | |
| 	res = mv88e61xx_port_read(phydev, port, PORT_REG_STATUS);
 | |
| 	if (res < 0)
 | |
| 		return res;
 | |
| 	return res & PORT_REG_STATUS_CMODE_MASK;
 | |
| }
 | |
| 
 | |
| static int mv88e61xx_parse_status(struct phy_device *phydev)
 | |
| {
 | |
| 	unsigned int speed;
 | |
| 	unsigned int mii_reg;
 | |
| 
 | |
| 	mii_reg = phy_read(phydev, MDIO_DEVAD_NONE, PHY_REG_STATUS1);
 | |
| 
 | |
| 	if ((mii_reg & PHY_REG_STATUS1_LINK) &&
 | |
| 	    !(mii_reg & PHY_REG_STATUS1_SPDDONE)) {
 | |
| 		int i = 0;
 | |
| 
 | |
| 		puts("Waiting for PHY realtime link");
 | |
| 		while (!(mii_reg & PHY_REG_STATUS1_SPDDONE)) {
 | |
| 			/* Timeout reached ? */
 | |
| 			if (i > PHY_AUTONEGOTIATE_TIMEOUT) {
 | |
| 				puts(" TIMEOUT !\n");
 | |
| 				phydev->link = 0;
 | |
| 				break;
 | |
| 			}
 | |
| 
 | |
| 			if ((i++ % 1000) == 0)
 | |
| 				putc('.');
 | |
| 			udelay(1000);
 | |
| 			mii_reg = phy_read(phydev, MDIO_DEVAD_NONE,
 | |
| 					PHY_REG_STATUS1);
 | |
| 		}
 | |
| 		puts(" done\n");
 | |
| 		udelay(500000);	/* another 500 ms (results in faster booting) */
 | |
| 	} else {
 | |
| 		if (mii_reg & PHY_REG_STATUS1_LINK)
 | |
| 			phydev->link = 1;
 | |
| 		else
 | |
| 			phydev->link = 0;
 | |
| 	}
 | |
| 
 | |
| 	if (mii_reg & PHY_REG_STATUS1_DUPLEX)
 | |
| 		phydev->duplex = DUPLEX_FULL;
 | |
| 	else
 | |
| 		phydev->duplex = DUPLEX_HALF;
 | |
| 
 | |
| 	speed = mii_reg & PHY_REG_STATUS1_SPEED;
 | |
| 
 | |
| 	switch (speed) {
 | |
| 	case PHY_REG_STATUS1_GBIT:
 | |
| 		phydev->speed = SPEED_1000;
 | |
| 		break;
 | |
| 	case PHY_REG_STATUS1_100:
 | |
| 		phydev->speed = SPEED_100;
 | |
| 		break;
 | |
| 	default:
 | |
| 		phydev->speed = SPEED_10;
 | |
| 		break;
 | |
| 	}
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int mv88e61xx_switch_reset(struct phy_device *phydev)
 | |
| {
 | |
| 	int time;
 | |
| 	int val;
 | |
| 	u8 port;
 | |
| 
 | |
| 	/* Disable all ports */
 | |
| 	for (port = 0; port < PORT_COUNT; port++) {
 | |
| 		val = mv88e61xx_port_read(phydev, port, PORT_REG_CTRL);
 | |
| 		if (val < 0)
 | |
| 			return val;
 | |
| 		val = bitfield_replace(val, PORT_REG_CTRL_PSTATE_SHIFT,
 | |
| 				       PORT_REG_CTRL_PSTATE_WIDTH,
 | |
| 				       PORT_REG_CTRL_PSTATE_DISABLED);
 | |
| 		val = mv88e61xx_port_write(phydev, port, PORT_REG_CTRL, val);
 | |
| 		if (val < 0)
 | |
| 			return val;
 | |
| 	}
 | |
| 
 | |
| 	/* Wait 2 ms for queues to drain */
 | |
| 	udelay(2000);
 | |
| 
 | |
| 	/* Reset switch */
 | |
| 	val = mv88e61xx_reg_read(phydev, DEVADDR_GLOBAL_1, GLOBAL1_CTRL);
 | |
| 	if (val < 0)
 | |
| 		return val;
 | |
| 	val |= GLOBAL1_CTRL_SWRESET;
 | |
| 	val = mv88e61xx_reg_write(phydev, DEVADDR_GLOBAL_1,
 | |
| 				     GLOBAL1_CTRL, val);
 | |
| 	if (val < 0)
 | |
| 		return val;
 | |
| 
 | |
| 	/* Wait up to 1 second for switch reset complete */
 | |
| 	for (time = 1000; time; time--) {
 | |
| 		val = mv88e61xx_reg_read(phydev, DEVADDR_GLOBAL_1,
 | |
| 					    GLOBAL1_CTRL);
 | |
| 		if (val >= 0 && ((val & GLOBAL1_CTRL_SWRESET) == 0))
 | |
| 			break;
 | |
| 		udelay(1000);
 | |
| 	}
 | |
| 	if (!time)
 | |
| 		return -ETIMEDOUT;
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int mv88e61xx_serdes_init(struct phy_device *phydev)
 | |
| {
 | |
| 	int val;
 | |
| 
 | |
| 	val = mv88e61xx_set_page(phydev, DEVADDR_SERDES, PHY_PAGE_SERDES);
 | |
| 	if (val < 0)
 | |
| 		return val;
 | |
| 
 | |
| 	/* Power up serdes module */
 | |
| 	val = mv88e61xx_phy_read(phydev, DEVADDR_SERDES, MII_BMCR);
 | |
| 	if (val < 0)
 | |
| 		return val;
 | |
| 	val &= ~(BMCR_PDOWN);
 | |
| 	val = mv88e61xx_phy_write(phydev, DEVADDR_SERDES, MII_BMCR, val);
 | |
| 	if (val < 0)
 | |
| 		return val;
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int mv88e61xx_port_enable(struct phy_device *phydev, u8 port)
 | |
| {
 | |
| 	int val;
 | |
| 
 | |
| 	val = mv88e61xx_port_read(phydev, port, PORT_REG_CTRL);
 | |
| 	if (val < 0)
 | |
| 		return val;
 | |
| 	val = bitfield_replace(val, PORT_REG_CTRL_PSTATE_SHIFT,
 | |
| 			       PORT_REG_CTRL_PSTATE_WIDTH,
 | |
| 			       PORT_REG_CTRL_PSTATE_FORWARD);
 | |
| 	val = mv88e61xx_port_write(phydev, port, PORT_REG_CTRL, val);
 | |
| 	if (val < 0)
 | |
| 		return val;
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int mv88e61xx_port_set_vlan(struct phy_device *phydev, u8 port,
 | |
| 							u8 mask)
 | |
| {
 | |
| 	int val;
 | |
| 
 | |
| 	/* Set VID to port number plus one */
 | |
| 	val = mv88e61xx_port_read(phydev, port, PORT_REG_VLAN_ID);
 | |
| 	if (val < 0)
 | |
| 		return val;
 | |
| 	val = bitfield_replace(val, PORT_REG_VLAN_ID_DEF_VID_SHIFT,
 | |
| 			       PORT_REG_VLAN_ID_DEF_VID_WIDTH,
 | |
| 			       port + 1);
 | |
| 	val = mv88e61xx_port_write(phydev, port, PORT_REG_VLAN_ID, val);
 | |
| 	if (val < 0)
 | |
| 		return val;
 | |
| 
 | |
| 	/* Set VID mask */
 | |
| 	val = mv88e61xx_port_read(phydev, port, PORT_REG_VLAN_MAP);
 | |
| 	if (val < 0)
 | |
| 		return val;
 | |
| 	val = bitfield_replace(val, PORT_REG_VLAN_MAP_TABLE_SHIFT,
 | |
| 			       PORT_REG_VLAN_MAP_TABLE_WIDTH,
 | |
| 			       mask);
 | |
| 	val = mv88e61xx_port_write(phydev, port, PORT_REG_VLAN_MAP, val);
 | |
| 	if (val < 0)
 | |
| 		return val;
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int mv88e61xx_read_port_config(struct phy_device *phydev, u8 port)
 | |
| {
 | |
| 	int res;
 | |
| 	int val;
 | |
| 	bool forced = false;
 | |
| 
 | |
| 	val = mv88e61xx_port_read(phydev, port, PORT_REG_STATUS);
 | |
| 	if (val < 0)
 | |
| 		return val;
 | |
| 	if (!(val & PORT_REG_STATUS_LINK)) {
 | |
| 		/* Temporarily force link to read port configuration */
 | |
| 		u32 timeout = 100;
 | |
| 		forced = true;
 | |
| 
 | |
| 		val = mv88e61xx_port_read(phydev, port, PORT_REG_PHYS_CTRL);
 | |
| 		if (val < 0)
 | |
| 			return val;
 | |
| 		val |= (PORT_REG_PHYS_CTRL_LINK_FORCE |
 | |
| 				PORT_REG_PHYS_CTRL_LINK_VALUE);
 | |
| 		val = mv88e61xx_port_write(phydev, port, PORT_REG_PHYS_CTRL,
 | |
| 					   val);
 | |
| 		if (val < 0)
 | |
| 			return val;
 | |
| 
 | |
| 		/* Wait for status register to reflect forced link */
 | |
| 		do {
 | |
| 			val = mv88e61xx_port_read(phydev, port,
 | |
| 						  PORT_REG_STATUS);
 | |
| 			if (val < 0)
 | |
| 				goto unforce;
 | |
| 			if (val & PORT_REG_STATUS_LINK)
 | |
| 				break;
 | |
| 		} while (--timeout);
 | |
| 
 | |
| 		if (timeout == 0) {
 | |
| 			res = -ETIMEDOUT;
 | |
| 			goto unforce;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	if (val & PORT_REG_STATUS_DUPLEX)
 | |
| 		phydev->duplex = DUPLEX_FULL;
 | |
| 	else
 | |
| 		phydev->duplex = DUPLEX_HALF;
 | |
| 
 | |
| 	val = bitfield_extract(val, PORT_REG_STATUS_SPEED_SHIFT,
 | |
| 			       PORT_REG_STATUS_SPEED_WIDTH);
 | |
| 	switch (val) {
 | |
| 	case PORT_REG_STATUS_SPEED_1000:
 | |
| 		phydev->speed = SPEED_1000;
 | |
| 		break;
 | |
| 	case PORT_REG_STATUS_SPEED_100:
 | |
| 		phydev->speed = SPEED_100;
 | |
| 		break;
 | |
| 	default:
 | |
| 		phydev->speed = SPEED_10;
 | |
| 		break;
 | |
| 	}
 | |
| 
 | |
| 	res = 0;
 | |
| 
 | |
| unforce:
 | |
| 	if (forced) {
 | |
| 		val = mv88e61xx_port_read(phydev, port, PORT_REG_PHYS_CTRL);
 | |
| 		if (val < 0)
 | |
| 			return val;
 | |
| 		val &= ~(PORT_REG_PHYS_CTRL_LINK_FORCE |
 | |
| 				PORT_REG_PHYS_CTRL_LINK_VALUE);
 | |
| 		val = mv88e61xx_port_write(phydev, port, PORT_REG_PHYS_CTRL,
 | |
| 					   val);
 | |
| 		if (val < 0)
 | |
| 			return val;
 | |
| 	}
 | |
| 
 | |
| 	return res;
 | |
| }
 | |
| 
 | |
| static int mv88e61xx_set_cpu_port(struct phy_device *phydev)
 | |
| {
 | |
| 	int val;
 | |
| 
 | |
| 	/* Set CPUDest */
 | |
| 	val = mv88e61xx_reg_read(phydev, DEVADDR_GLOBAL_1, GLOBAL1_MON_CTRL);
 | |
| 	if (val < 0)
 | |
| 		return val;
 | |
| 	val = bitfield_replace(val, GLOBAL1_MON_CTRL_CPUDEST_SHIFT,
 | |
| 			       GLOBAL1_MON_CTRL_CPUDEST_WIDTH,
 | |
| 			       CONFIG_MV88E61XX_CPU_PORT);
 | |
| 	val = mv88e61xx_reg_write(phydev, DEVADDR_GLOBAL_1,
 | |
| 				     GLOBAL1_MON_CTRL, val);
 | |
| 	if (val < 0)
 | |
| 		return val;
 | |
| 
 | |
| 	/* Allow CPU to route to any port */
 | |
| 	val = PORT_MASK & ~(1 << CONFIG_MV88E61XX_CPU_PORT);
 | |
| 	val = mv88e61xx_port_set_vlan(phydev, CONFIG_MV88E61XX_CPU_PORT, val);
 | |
| 	if (val < 0)
 | |
| 		return val;
 | |
| 
 | |
| 	/* Enable CPU port */
 | |
| 	val = mv88e61xx_port_enable(phydev, CONFIG_MV88E61XX_CPU_PORT);
 | |
| 	if (val < 0)
 | |
| 		return val;
 | |
| 
 | |
| 	val = mv88e61xx_read_port_config(phydev, CONFIG_MV88E61XX_CPU_PORT);
 | |
| 	if (val < 0)
 | |
| 		return val;
 | |
| 
 | |
| 	/* If CPU is connected to serdes, initialize serdes */
 | |
| 	if (mv88e61xx_6352_family(phydev)) {
 | |
| 		val = mv88e61xx_get_cmode(phydev, CONFIG_MV88E61XX_CPU_PORT);
 | |
| 		if (val < 0)
 | |
| 			return val;
 | |
| 		if (val == PORT_REG_STATUS_CMODE_100BASE_X ||
 | |
| 		    val == PORT_REG_STATUS_CMODE_1000BASE_X ||
 | |
| 		    val == PORT_REG_STATUS_CMODE_SGMII) {
 | |
| 			val = mv88e61xx_serdes_init(phydev);
 | |
| 			if (val < 0)
 | |
| 				return val;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int mv88e61xx_switch_init(struct phy_device *phydev)
 | |
| {
 | |
| 	static int init;
 | |
| 	int res;
 | |
| 
 | |
| 	if (init)
 | |
| 		return 0;
 | |
| 
 | |
| 	res = mv88e61xx_switch_reset(phydev);
 | |
| 	if (res < 0)
 | |
| 		return res;
 | |
| 
 | |
| 	res = mv88e61xx_set_cpu_port(phydev);
 | |
| 	if (res < 0)
 | |
| 		return res;
 | |
| 
 | |
| 	init = 1;
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int mv88e61xx_phy_enable(struct phy_device *phydev, u8 phy)
 | |
| {
 | |
| 	int val;
 | |
| 
 | |
| 	val = mv88e61xx_phy_read(phydev, phy, MII_BMCR);
 | |
| 	if (val < 0)
 | |
| 		return val;
 | |
| 	val &= ~(BMCR_PDOWN);
 | |
| 	val = mv88e61xx_phy_write(phydev, phy, MII_BMCR, val);
 | |
| 	if (val < 0)
 | |
| 		return val;
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int mv88e61xx_phy_setup(struct phy_device *phydev, u8 phy)
 | |
| {
 | |
| 	int val;
 | |
| 
 | |
| 	/*
 | |
| 	 * Enable energy-detect sensing on PHY, used to determine when a PHY
 | |
| 	 * port is physically connected
 | |
| 	 */
 | |
| 	val = mv88e61xx_phy_read(phydev, phy, PHY_REG_CTRL1);
 | |
| 	if (val < 0)
 | |
| 		return val;
 | |
| 	val = bitfield_replace(val, PHY_REG_CTRL1_ENERGY_DET_SHIFT,
 | |
| 			       PHY_REG_CTRL1_ENERGY_DET_WIDTH,
 | |
| 			       PHY_REG_CTRL1_ENERGY_DET_SENSE_XMIT);
 | |
| 	val = mv88e61xx_phy_write(phydev, phy, PHY_REG_CTRL1, val);
 | |
| 	if (val < 0)
 | |
| 		return val;
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int mv88e61xx_phy_config_port(struct phy_device *phydev, u8 phy)
 | |
| {
 | |
| 	int val;
 | |
| 
 | |
| 	val = mv88e61xx_port_enable(phydev, phy);
 | |
| 	if (val < 0)
 | |
| 		return val;
 | |
| 
 | |
| 	val = mv88e61xx_port_set_vlan(phydev, phy,
 | |
| 			1 << CONFIG_MV88E61XX_CPU_PORT);
 | |
| 	if (val < 0)
 | |
| 		return val;
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int mv88e61xx_probe(struct phy_device *phydev)
 | |
| {
 | |
| 	struct mii_dev *smi_wrapper;
 | |
| 	struct mv88e61xx_phy_priv *priv;
 | |
| 	int res;
 | |
| 
 | |
| 	res = mv88e61xx_hw_reset(phydev);
 | |
| 	if (res < 0)
 | |
| 		return res;
 | |
| 
 | |
| 	priv = malloc(sizeof(*priv));
 | |
| 	if (!priv)
 | |
| 		return -ENOMEM;
 | |
| 
 | |
| 	memset(priv, 0, sizeof(*priv));
 | |
| 
 | |
| 	/*
 | |
| 	 * This device requires indirect reads/writes to the PHY registers
 | |
| 	 * which the generic PHY code can't handle.  Make a wrapper MII device
 | |
| 	 * to handle reads/writes
 | |
| 	 */
 | |
| 	smi_wrapper = mdio_alloc();
 | |
| 	if (!smi_wrapper) {
 | |
| 		free(priv);
 | |
| 		return -ENOMEM;
 | |
| 	}
 | |
| 
 | |
| 	/*
 | |
| 	 * Store the mdio bus in the private data, as we are going to replace
 | |
| 	 * the bus with the wrapper bus
 | |
| 	 */
 | |
| 	priv->mdio_bus = phydev->bus;
 | |
| 
 | |
| 	/*
 | |
| 	 * Store the smi bus address in private data.  This lets us use the
 | |
| 	 * phydev addr field for device address instead, as the genphy code
 | |
| 	 * expects.
 | |
| 	 */
 | |
| 	priv->smi_addr = phydev->addr;
 | |
| 
 | |
| 	/*
 | |
| 	 * Store the phy_device in the wrapper mii device. This lets us get it
 | |
| 	 * back when genphy functions call phy_read/phy_write.
 | |
| 	 */
 | |
| 	smi_wrapper->priv = phydev;
 | |
| 	strncpy(smi_wrapper->name, "indirect mii", sizeof(smi_wrapper->name));
 | |
| 	smi_wrapper->read = mv88e61xx_phy_read_indirect;
 | |
| 	smi_wrapper->write = mv88e61xx_phy_write_indirect;
 | |
| 
 | |
| 	/* Replace the bus with the wrapper device */
 | |
| 	phydev->bus = smi_wrapper;
 | |
| 
 | |
| 	phydev->priv = priv;
 | |
| 
 | |
| 	priv->id = mv88e61xx_get_switch_id(phydev);
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int mv88e61xx_phy_config(struct phy_device *phydev)
 | |
| {
 | |
| 	int res;
 | |
| 	int i;
 | |
| 	int ret = -1;
 | |
| 
 | |
| 	res = mv88e61xx_switch_init(phydev);
 | |
| 	if (res < 0)
 | |
| 		return res;
 | |
| 
 | |
| 	for (i = 0; i < PORT_COUNT; i++) {
 | |
| 		if ((1 << i) & CONFIG_MV88E61XX_PHY_PORTS) {
 | |
| 			phydev->addr = i;
 | |
| 
 | |
| 			res = mv88e61xx_phy_enable(phydev, i);
 | |
| 			if (res < 0) {
 | |
| 				printf("Error enabling PHY %i\n", i);
 | |
| 				continue;
 | |
| 			}
 | |
| 			res = mv88e61xx_phy_setup(phydev, i);
 | |
| 			if (res < 0) {
 | |
| 				printf("Error setting up PHY %i\n", i);
 | |
| 				continue;
 | |
| 			}
 | |
| 			res = mv88e61xx_phy_config_port(phydev, i);
 | |
| 			if (res < 0) {
 | |
| 				printf("Error configuring PHY %i\n", i);
 | |
| 				continue;
 | |
| 			}
 | |
| 
 | |
| 			res = genphy_config_aneg(phydev);
 | |
| 			if (res < 0) {
 | |
| 				printf("Error setting PHY %i autoneg\n", i);
 | |
| 				continue;
 | |
| 			}
 | |
| 			res = phy_reset(phydev);
 | |
| 			if (res < 0) {
 | |
| 				printf("Error resetting PHY %i\n", i);
 | |
| 				continue;
 | |
| 			}
 | |
| 
 | |
| 			/* Return success if any PHY succeeds */
 | |
| 			ret = 0;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return ret;
 | |
| }
 | |
| 
 | |
| static int mv88e61xx_phy_is_connected(struct phy_device *phydev)
 | |
| {
 | |
| 	int val;
 | |
| 
 | |
| 	val = mv88e61xx_phy_read(phydev, phydev->addr, PHY_REG_STATUS1);
 | |
| 	if (val < 0)
 | |
| 		return 0;
 | |
| 
 | |
| 	/*
 | |
| 	 * After reset, the energy detect signal remains high for a few seconds
 | |
| 	 * regardless of whether a cable is connected.  This function will
 | |
| 	 * return false positives during this time.
 | |
| 	 */
 | |
| 	return (val & PHY_REG_STATUS1_ENERGY) == 0;
 | |
| }
 | |
| 
 | |
| static int mv88e61xx_phy_startup(struct phy_device *phydev)
 | |
| {
 | |
| 	int i;
 | |
| 	int link = 0;
 | |
| 	int res;
 | |
| 	int speed = phydev->speed;
 | |
| 	int duplex = phydev->duplex;
 | |
| 
 | |
| 	for (i = 0; i < PORT_COUNT; i++) {
 | |
| 		if ((1 << i) & CONFIG_MV88E61XX_PHY_PORTS) {
 | |
| 			phydev->addr = i;
 | |
| 			if (!mv88e61xx_phy_is_connected(phydev))
 | |
| 				continue;
 | |
| 			res = genphy_update_link(phydev);
 | |
| 			if (res < 0)
 | |
| 				continue;
 | |
| 			res = mv88e61xx_parse_status(phydev);
 | |
| 			if (res < 0)
 | |
| 				continue;
 | |
| 			link = (link || phydev->link);
 | |
| 		}
 | |
| 	}
 | |
| 	phydev->link = link;
 | |
| 
 | |
| 	/* Restore CPU interface speed and duplex after it was changed for
 | |
| 	 * other ports */
 | |
| 	phydev->speed = speed;
 | |
| 	phydev->duplex = duplex;
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static struct phy_driver mv88e61xx_driver = {
 | |
| 	.name = "Marvell MV88E61xx",
 | |
| 	.uid = 0x01410eb1,
 | |
| 	.mask = 0xfffffff0,
 | |
| 	.features = PHY_GBIT_FEATURES,
 | |
| 	.probe = mv88e61xx_probe,
 | |
| 	.config = mv88e61xx_phy_config,
 | |
| 	.startup = mv88e61xx_phy_startup,
 | |
| 	.shutdown = &genphy_shutdown,
 | |
| };
 | |
| 
 | |
| int phy_mv88e61xx_init(void)
 | |
| {
 | |
| 	phy_register(&mv88e61xx_driver);
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Overload weak get_phy_id definition since we need non-standard functions
 | |
|  * to read PHY registers
 | |
|  */
 | |
| int get_phy_id(struct mii_dev *bus, int smi_addr, int devad, u32 *phy_id)
 | |
| {
 | |
| 	struct phy_device temp_phy;
 | |
| 	struct mv88e61xx_phy_priv temp_priv;
 | |
| 	struct mii_dev temp_mii;
 | |
| 	int val;
 | |
| 
 | |
| 	/*
 | |
| 	 * Buid temporary data structures that the chip reading code needs to
 | |
| 	 * read the ID
 | |
| 	 */
 | |
| 	temp_priv.mdio_bus = bus;
 | |
| 	temp_priv.smi_addr = smi_addr;
 | |
| 	temp_phy.priv = &temp_priv;
 | |
| 	temp_mii.priv = &temp_phy;
 | |
| 
 | |
| 	val = mv88e61xx_phy_read_indirect(&temp_mii, 0, devad, MII_PHYSID1);
 | |
| 	if (val < 0)
 | |
| 		return -EIO;
 | |
| 
 | |
| 	*phy_id = val << 16;
 | |
| 
 | |
| 	val = mv88e61xx_phy_read_indirect(&temp_mii, 0, devad, MII_PHYSID2);
 | |
| 	if (val < 0)
 | |
| 		return -EIO;
 | |
| 
 | |
| 	*phy_id |= (val & 0xffff);
 | |
| 
 | |
| 	return 0;
 | |
| }
 |