123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074 |
- /*
- * (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 11
- #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_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_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)
- #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
- /*
- * 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_6096 0x0980
- #define PORT_SWITCH_ID_6097 0x0990
- #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,
- 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)
- {
- 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_fixed_port_setup(struct phy_device *phydev, u8 port)
- {
- 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);
- val |= PORT_REG_PHYS_CTRL_PCS_AN_EN |
- PORT_REG_PHYS_CTRL_PCS_AN_RST |
- PORT_REG_PHYS_CTRL_FC_FORCE |
- PORT_REG_PHYS_CTRL_DUPLEX_VALUE |
- PORT_REG_PHYS_CTRL_DUPLEX_FORCE |
- PORT_REG_PHYS_CTRL_SPD1000;
- return mv88e61xx_port_write(phydev, port, PORT_REG_PHYS_CTRL,
- val);
- }
- 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;
- } 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)
- {
- 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,
- };
- 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,
- };
- int phy_mv88e61xx_init(void)
- {
- phy_register(&mv88e61xx_driver);
- phy_register(&mv88e609x_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;
- }
|