1224 lines
		
	
	
		
			30 KiB
		
	
	
	
		
			C
		
	
	
	
			
		
		
	
	
			1224 lines
		
	
	
		
			30 KiB
		
	
	
	
		
			C
		
	
	
	
// SPDX-License-Identifier: GPL-2.0+
 | 
						|
/*
 | 
						|
 * (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>
 | 
						|
 */
 | 
						|
 | 
						|
/*
 | 
						|
 * 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 <log.h>
 | 
						|
#include <linux/bitops.h>
 | 
						|
#include <linux/delay.h>
 | 
						|
 | 
						|
#include <bitfield.h>
 | 
						|
#include <errno.h>
 | 
						|
#include <malloc.h>
 | 
						|
#include <miiphy.h>
 | 
						|
#include <netdev.h>
 | 
						|
 | 
						|
#define PHY_AUTONEGOTIATE_TIMEOUT	5000
 | 
						|
 | 
						|
#define PORT_MASK(port_count)		((1 << (port_count)) - 1)
 | 
						|
 | 
						|
/* Device addresses */
 | 
						|
#define DEVADDR_PHY(p)			(p)
 | 
						|
#define DEVADDR_SERDES			0x0F
 | 
						|
 | 
						|
/* 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_SPEED_SHIFT	8
 | 
						|
#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_PCS_AN_EN	BIT(10)
 | 
						|
#define PORT_REG_PHYS_CTRL_PCS_AN_RST	BIT(9)
 | 
						|
#define PORT_REG_PHYS_CTRL_FC_VALUE	BIT(7)
 | 
						|
#define PORT_REG_PHYS_CTRL_FC_FORCE	BIT(6)
 | 
						|
#define PORT_REG_PHYS_CTRL_LINK_VALUE	BIT(5)
 | 
						|
#define PORT_REG_PHYS_CTRL_LINK_FORCE	BIT(4)
 | 
						|
#define PORT_REG_PHYS_CTRL_DUPLEX_VALUE	BIT(3)
 | 
						|
#define PORT_REG_PHYS_CTRL_DUPLEX_FORCE	BIT(2)
 | 
						|
#define PORT_REG_PHYS_CTRL_SPD1000	BIT(1)
 | 
						|
#define PORT_REG_PHYS_CTRL_SPD100	BIT(0)
 | 
						|
#define PORT_REG_PHYS_CTRL_SPD_MASK	(BIT(1) | BIT(0))
 | 
						|
 | 
						|
#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)
 | 
						|
 | 
						|
/* 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_PULSE	1
 | 
						|
#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
 | 
						|
 | 
						|
/*
 | 
						|
 *  These are ports without PHYs that may be wired directly
 | 
						|
 * to other serdes interfaces
 | 
						|
 */
 | 
						|
#ifndef CONFIG_MV88E61XX_FIXED_PORTS
 | 
						|
#define CONFIG_MV88E61XX_FIXED_PORTS 0
 | 
						|
#endif
 | 
						|
 | 
						|
/* ID register values for different switch models */
 | 
						|
#define PORT_SWITCH_ID_6020		0x0200
 | 
						|
#define PORT_SWITCH_ID_6070		0x0700
 | 
						|
#define PORT_SWITCH_ID_6071		0x0710
 | 
						|
#define PORT_SWITCH_ID_6096		0x0980
 | 
						|
#define PORT_SWITCH_ID_6097		0x0990
 | 
						|
#define PORT_SWITCH_ID_6172		0x1720
 | 
						|
#define PORT_SWITCH_ID_6176		0x1760
 | 
						|
#define PORT_SWITCH_ID_6220		0x2200
 | 
						|
#define PORT_SWITCH_ID_6240		0x2400
 | 
						|
#define PORT_SWITCH_ID_6250		0x2500
 | 
						|
#define PORT_SWITCH_ID_6352		0x3520
 | 
						|
 | 
						|
struct mv88e61xx_phy_priv {
 | 
						|
	struct mii_dev *mdio_bus;
 | 
						|
	int smi_addr;
 | 
						|
	int id;
 | 
						|
	int port_count;		/* Number of switch ports */
 | 
						|
	int port_reg_base;	/* Base of the switch port registers */
 | 
						|
	u16 port_stat_link_mask;/* Bitmask for port link status bits */
 | 
						|
	u16 port_stat_dup_mask; /* Bitmask for port duplex status bits */
 | 
						|
	u8 port_stat_speed_width;/* Width of speed status bitfield */
 | 
						|
	u8 global1;	/* Offset of Switch Global 1 registers */
 | 
						|
	u8 global2;	/* Offset of Switch Global 2 registers */
 | 
						|
	u8 phy_ctrl1_en_det_shift; /* 'EDet' bit field offset */
 | 
						|
	u8 phy_ctrl1_en_det_width; /* Width of 'EDet' bit field */
 | 
						|
	u8 phy_ctrl1_en_det_ctrl;  /* 'EDet' control value */
 | 
						|
};
 | 
						|
 | 
						|
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)
 | 
						|
{
 | 
						|
	struct mv88e61xx_phy_priv *priv = phydev->priv;
 | 
						|
	int val;
 | 
						|
	u32 timeout = 100;
 | 
						|
 | 
						|
	do {
 | 
						|
		val = mv88e61xx_reg_read(phydev, priv->global2,
 | 
						|
					 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 mv88e61xx_phy_priv *priv;
 | 
						|
	struct phy_device *phydev;
 | 
						|
	int res;
 | 
						|
 | 
						|
	phydev = (struct phy_device *)smi_wrapper->priv;
 | 
						|
	priv = phydev->priv;
 | 
						|
 | 
						|
	/* Issue command to read */
 | 
						|
	res = mv88e61xx_reg_write(phydev, priv->global2,
 | 
						|
				  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, priv->global2,
 | 
						|
				  GLOBAL2_REG_PHY_DATA);
 | 
						|
}
 | 
						|
 | 
						|
static int mv88e61xx_phy_write_indirect(struct mii_dev *smi_wrapper, int dev,
 | 
						|
		int devad, int reg, u16 data)
 | 
						|
{
 | 
						|
	struct mv88e61xx_phy_priv *priv;
 | 
						|
	struct phy_device *phydev;
 | 
						|
	int res;
 | 
						|
 | 
						|
	phydev = (struct phy_device *)smi_wrapper->priv;
 | 
						|
	priv = phydev->priv;
 | 
						|
 | 
						|
	/* Set the data to write */
 | 
						|
	res = mv88e61xx_reg_write(phydev, priv->global2,
 | 
						|
				  GLOBAL2_REG_PHY_DATA, data);
 | 
						|
	if (res < 0)
 | 
						|
		return res;
 | 
						|
	/* Issue the write command */
 | 
						|
	res = mv88e61xx_reg_write(phydev, priv->global2,
 | 
						|
				  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)
 | 
						|
{
 | 
						|
	struct mv88e61xx_phy_priv *priv = phydev->priv;
 | 
						|
 | 
						|
	return mv88e61xx_reg_read(phydev, priv->port_reg_base + port, reg);
 | 
						|
}
 | 
						|
 | 
						|
static int mv88e61xx_port_write(struct phy_device *phydev, u8 port, u8 reg,
 | 
						|
								u16 val)
 | 
						|
{
 | 
						|
	struct mv88e61xx_phy_priv *priv = phydev->priv;
 | 
						|
 | 
						|
	return mv88e61xx_reg_write(phydev, priv->port_reg_base + 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)
 | 
						|
{
 | 
						|
	struct mv88e61xx_phy_priv *priv = phydev->priv;
 | 
						|
	int time;
 | 
						|
	int val;
 | 
						|
	u8 port;
 | 
						|
 | 
						|
	/* Disable all ports */
 | 
						|
	for (port = 0; port < priv->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, priv->global1, GLOBAL1_CTRL);
 | 
						|
	if (val < 0)
 | 
						|
		return val;
 | 
						|
	val |= GLOBAL1_CTRL_SWRESET;
 | 
						|
	val = mv88e61xx_reg_write(phydev, priv->global1,
 | 
						|
				  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, priv->global1,
 | 
						|
					 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,
 | 
						|
							u16 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)
 | 
						|
{
 | 
						|
	struct mv88e61xx_phy_priv *priv = phydev->priv;
 | 
						|
	int res;
 | 
						|
	int val;
 | 
						|
	bool forced = false;
 | 
						|
 | 
						|
	val = mv88e61xx_port_read(phydev, port, PORT_REG_STATUS);
 | 
						|
	if (val < 0)
 | 
						|
		return val;
 | 
						|
	if (!(val & priv->port_stat_link_mask)) {
 | 
						|
		/* 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) {
 | 
						|
				res = -EIO;
 | 
						|
				goto unforce;
 | 
						|
			}
 | 
						|
			if (val & priv->port_stat_link_mask)
 | 
						|
				break;
 | 
						|
		} while (--timeout);
 | 
						|
 | 
						|
		if (timeout == 0) {
 | 
						|
			res = -ETIMEDOUT;
 | 
						|
			goto unforce;
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	if (val & priv->port_stat_dup_mask)
 | 
						|
		phydev->duplex = DUPLEX_FULL;
 | 
						|
	else
 | 
						|
		phydev->duplex = DUPLEX_HALF;
 | 
						|
 | 
						|
	val = bitfield_extract(val, PORT_REG_STATUS_SPEED_SHIFT,
 | 
						|
			       priv->port_stat_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_fixed_port_setup(struct phy_device *phydev, u8 port)
 | 
						|
{
 | 
						|
	struct mv88e61xx_phy_priv *priv = phydev->priv;
 | 
						|
	int val;
 | 
						|
 | 
						|
	val = mv88e61xx_port_read(phydev, port, PORT_REG_PHYS_CTRL);
 | 
						|
	if (val < 0)
 | 
						|
		return val;
 | 
						|
 | 
						|
	val &= ~(PORT_REG_PHYS_CTRL_SPD_MASK |
 | 
						|
		 PORT_REG_PHYS_CTRL_FC_VALUE |
 | 
						|
		 PORT_REG_PHYS_CTRL_FC_FORCE);
 | 
						|
	val |= PORT_REG_PHYS_CTRL_FC_FORCE |
 | 
						|
	       PORT_REG_PHYS_CTRL_DUPLEX_VALUE |
 | 
						|
	       PORT_REG_PHYS_CTRL_DUPLEX_FORCE;
 | 
						|
 | 
						|
	if (priv->id == PORT_SWITCH_ID_6071) {
 | 
						|
		val |= PORT_REG_PHYS_CTRL_SPD100;
 | 
						|
	} else {
 | 
						|
		val |= PORT_REG_PHYS_CTRL_PCS_AN_EN |
 | 
						|
		       PORT_REG_PHYS_CTRL_PCS_AN_RST |
 | 
						|
		       PORT_REG_PHYS_CTRL_SPD1000;
 | 
						|
	}
 | 
						|
 | 
						|
	if (port == CONFIG_MV88E61XX_CPU_PORT)
 | 
						|
		val |= PORT_REG_PHYS_CTRL_LINK_VALUE |
 | 
						|
		       PORT_REG_PHYS_CTRL_LINK_FORCE;
 | 
						|
 | 
						|
	return mv88e61xx_port_write(phydev, port, PORT_REG_PHYS_CTRL,
 | 
						|
				   val);
 | 
						|
}
 | 
						|
 | 
						|
static int mv88e61xx_set_cpu_port(struct phy_device *phydev)
 | 
						|
{
 | 
						|
	struct mv88e61xx_phy_priv *priv = phydev->priv;
 | 
						|
	int val;
 | 
						|
 | 
						|
	/* Set CPUDest */
 | 
						|
	val = mv88e61xx_reg_read(phydev, priv->global1, 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, priv->global1,
 | 
						|
				  GLOBAL1_MON_CTRL, val);
 | 
						|
	if (val < 0)
 | 
						|
		return val;
 | 
						|
 | 
						|
	/* Allow CPU to route to any port */
 | 
						|
	val = PORT_MASK(priv->port_count) & ~(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;
 | 
						|
		}
 | 
						|
	} else {
 | 
						|
		val = mv88e61xx_fixed_port_setup(phydev,
 | 
						|
						 CONFIG_MV88E61XX_CPU_PORT);
 | 
						|
		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)
 | 
						|
{
 | 
						|
	struct mv88e61xx_phy_priv *priv = phydev->priv;
 | 
						|
	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, priv->phy_ctrl1_en_det_shift,
 | 
						|
			       priv->phy_ctrl1_en_det_width,
 | 
						|
			       priv->phy_ctrl1_en_det_ctrl);
 | 
						|
	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;
 | 
						|
}
 | 
						|
 | 
						|
/*
 | 
						|
 * This function is used to pre-configure the required register
 | 
						|
 * offsets, so that the indirect register access to the PHY registers
 | 
						|
 * is possible. This is necessary to be able to read the PHY ID
 | 
						|
 * while driver probing or in get_phy_id(). The globalN register
 | 
						|
 * offsets must be initialized correctly for a detected switch,
 | 
						|
 * otherwise detection of the PHY ID won't work!
 | 
						|
 */
 | 
						|
static int mv88e61xx_priv_reg_offs_pre_init(struct phy_device *phydev)
 | 
						|
{
 | 
						|
	struct mv88e61xx_phy_priv *priv = phydev->priv;
 | 
						|
 | 
						|
	/*
 | 
						|
	 * Initial 'port_reg_base' value must be an offset of existing
 | 
						|
	 * port register, then reading the ID should succeed. First, try
 | 
						|
	 * to read via port registers with device address 0x10 (88E6096
 | 
						|
	 * and compatible switches).
 | 
						|
	 */
 | 
						|
	priv->port_reg_base = 0x10;
 | 
						|
	priv->id = mv88e61xx_get_switch_id(phydev);
 | 
						|
	if (priv->id != 0xfff0) {
 | 
						|
		priv->global1 = 0x1B;
 | 
						|
		priv->global2 = 0x1C;
 | 
						|
		return 0;
 | 
						|
	}
 | 
						|
 | 
						|
	/*
 | 
						|
	 * Now try via port registers with device address 0x08
 | 
						|
	 * (88E6020 and compatible switches).
 | 
						|
	 */
 | 
						|
	priv->port_reg_base = 0x08;
 | 
						|
	priv->id = mv88e61xx_get_switch_id(phydev);
 | 
						|
	if (priv->id != 0xfff0) {
 | 
						|
		priv->global1 = 0x0F;
 | 
						|
		priv->global2 = 0x07;
 | 
						|
		return 0;
 | 
						|
	}
 | 
						|
 | 
						|
	debug("%s Unknown ID 0x%x\n", __func__, priv->id);
 | 
						|
	return -ENODEV;
 | 
						|
}
 | 
						|
 | 
						|
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;
 | 
						|
 | 
						|
	res = mv88e61xx_priv_reg_offs_pre_init(phydev);
 | 
						|
	if (res < 0)
 | 
						|
		return res;
 | 
						|
 | 
						|
	debug("%s ID 0x%x\n", __func__, priv->id);
 | 
						|
 | 
						|
	switch (priv->id) {
 | 
						|
	case PORT_SWITCH_ID_6096:
 | 
						|
	case PORT_SWITCH_ID_6097:
 | 
						|
	case PORT_SWITCH_ID_6172:
 | 
						|
	case PORT_SWITCH_ID_6176:
 | 
						|
	case PORT_SWITCH_ID_6240:
 | 
						|
	case PORT_SWITCH_ID_6352:
 | 
						|
		priv->port_count = 11;
 | 
						|
		priv->port_stat_link_mask = BIT(11);
 | 
						|
		priv->port_stat_dup_mask = BIT(10);
 | 
						|
		priv->port_stat_speed_width = 2;
 | 
						|
		priv->phy_ctrl1_en_det_shift = 8;
 | 
						|
		priv->phy_ctrl1_en_det_width = 2;
 | 
						|
		priv->phy_ctrl1_en_det_ctrl =
 | 
						|
			PHY_REG_CTRL1_ENERGY_DET_SENSE_XMIT;
 | 
						|
		break;
 | 
						|
	case PORT_SWITCH_ID_6020:
 | 
						|
	case PORT_SWITCH_ID_6070:
 | 
						|
	case PORT_SWITCH_ID_6071:
 | 
						|
	case PORT_SWITCH_ID_6220:
 | 
						|
	case PORT_SWITCH_ID_6250:
 | 
						|
		priv->port_count = 7;
 | 
						|
		priv->port_stat_link_mask = BIT(12);
 | 
						|
		priv->port_stat_dup_mask = BIT(9);
 | 
						|
		priv->port_stat_speed_width = 1;
 | 
						|
		priv->phy_ctrl1_en_det_shift = 14;
 | 
						|
		priv->phy_ctrl1_en_det_width = 1;
 | 
						|
		priv->phy_ctrl1_en_det_ctrl =
 | 
						|
			PHY_REG_CTRL1_ENERGY_DET_SENSE_PULSE;
 | 
						|
		break;
 | 
						|
	default:
 | 
						|
		free(priv);
 | 
						|
		return -ENODEV;
 | 
						|
	}
 | 
						|
 | 
						|
	res = mdio_register(smi_wrapper);
 | 
						|
	if (res)
 | 
						|
		printf("Failed to register SMI bus\n");
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static int mv88e61xx_phy_config(struct phy_device *phydev)
 | 
						|
{
 | 
						|
	struct mv88e61xx_phy_priv *priv = phydev->priv;
 | 
						|
	int res;
 | 
						|
	int i;
 | 
						|
	int ret = -1;
 | 
						|
 | 
						|
	res = mv88e61xx_switch_init(phydev);
 | 
						|
	if (res < 0)
 | 
						|
		return res;
 | 
						|
 | 
						|
	for (i = 0; i < priv->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 = phy_reset(phydev);
 | 
						|
			if (res < 0) {
 | 
						|
				printf("Error resetting PHY %i\n", i);
 | 
						|
				continue;
 | 
						|
			}
 | 
						|
			res = genphy_config_aneg(phydev);
 | 
						|
			if (res < 0) {
 | 
						|
				printf("Error setting PHY %i autoneg\n", i);
 | 
						|
				continue;
 | 
						|
			}
 | 
						|
 | 
						|
			/* Return success if any PHY succeeds */
 | 
						|
			ret = 0;
 | 
						|
		} else if ((1 << i) & CONFIG_MV88E61XX_FIXED_PORTS) {
 | 
						|
			res = mv88e61xx_fixed_port_setup(phydev, i);
 | 
						|
			if (res < 0) {
 | 
						|
				printf("Error configuring port %i\n", i);
 | 
						|
				continue;
 | 
						|
			}
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	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)
 | 
						|
{
 | 
						|
	struct mv88e61xx_phy_priv *priv = phydev->priv;
 | 
						|
	int i;
 | 
						|
	int link = 0;
 | 
						|
	int res;
 | 
						|
	int speed = phydev->speed;
 | 
						|
	int duplex = phydev->duplex;
 | 
						|
 | 
						|
	for (i = 0; i < priv->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,
 | 
						|
};
 | 
						|
 | 
						|
static struct phy_driver mv88e609x_driver = {
 | 
						|
	.name = "Marvell MV88E609x",
 | 
						|
	.uid = 0x1410c89,
 | 
						|
	.mask = 0xfffffff0,
 | 
						|
	.features = PHY_GBIT_FEATURES,
 | 
						|
	.probe = mv88e61xx_probe,
 | 
						|
	.config = mv88e61xx_phy_config,
 | 
						|
	.startup = mv88e61xx_phy_startup,
 | 
						|
	.shutdown = &genphy_shutdown,
 | 
						|
};
 | 
						|
 | 
						|
static struct phy_driver mv88e6071_driver = {
 | 
						|
	.name = "Marvell MV88E6071",
 | 
						|
	.uid = 0x1410db0,
 | 
						|
	.mask = 0xfffffff0,
 | 
						|
	.features = PHY_BASIC_FEATURES | SUPPORTED_MII,
 | 
						|
	.probe = mv88e61xx_probe,
 | 
						|
	.config = mv88e61xx_phy_config,
 | 
						|
	.startup = mv88e61xx_phy_startup,
 | 
						|
	.shutdown = &genphy_shutdown,
 | 
						|
};
 | 
						|
 | 
						|
int phy_mv88e61xx_init(void)
 | 
						|
{
 | 
						|
	phy_register(&mv88e61xx_driver);
 | 
						|
	phy_register(&mv88e609x_driver);
 | 
						|
	phy_register(&mv88e6071_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;
 | 
						|
 | 
						|
	/*
 | 
						|
	 * get_phy_id() can be called by framework before mv88e61xx driver
 | 
						|
	 * probing, in this case the global register offsets are not
 | 
						|
	 * initialized yet. Do this initialization here before indirect
 | 
						|
	 * PHY register access.
 | 
						|
	 */
 | 
						|
	val = mv88e61xx_priv_reg_offs_pre_init(&temp_phy);
 | 
						|
	if (val < 0)
 | 
						|
		return val;
 | 
						|
 | 
						|
	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;
 | 
						|
}
 |