| // 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; |
| } |