123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254 |
- /* 10G controller driver for Samsung SoCs
- *
- * Copyright (C) 2013 Samsung Electronics Co., Ltd.
- * http://www.samsung.com
- *
- * Author: Siva Reddy Kallam <siva.kallam@samsung.com>
- *
- * This program is free software; you can redistribute it and/or modify
- * it under the terms of the GNU General Public License version 2 as
- * published by the Free Software Foundation.
- */
- #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
- #include <linux/io.h>
- #include <linux/mii.h>
- #include <linux/netdevice.h>
- #include <linux/platform_device.h>
- #include <linux/phy.h>
- #include <linux/slab.h>
- #include <linux/sxgbe_platform.h>
- #include "sxgbe_common.h"
- #include "sxgbe_reg.h"
- #define SXGBE_SMA_WRITE_CMD 0x01 /* write command */
- #define SXGBE_SMA_PREAD_CMD 0x02 /* post read increament address */
- #define SXGBE_SMA_READ_CMD 0x03 /* read command */
- #define SXGBE_SMA_SKIP_ADDRFRM 0x00040000 /* skip the address frame */
- #define SXGBE_MII_BUSY 0x00400000 /* mii busy */
- static int sxgbe_mdio_busy_wait(void __iomem *ioaddr, unsigned int mii_data)
- {
- unsigned long fin_time = jiffies + 3 * HZ; /* 3 seconds */
- while (!time_after(jiffies, fin_time)) {
- if (!(readl(ioaddr + mii_data) & SXGBE_MII_BUSY))
- return 0;
- cpu_relax();
- }
- return -EBUSY;
- }
- static void sxgbe_mdio_ctrl_data(struct sxgbe_priv_data *sp, u32 cmd,
- u16 phydata)
- {
- u32 reg = phydata;
- reg |= (cmd << 16) | SXGBE_SMA_SKIP_ADDRFRM |
- ((sp->clk_csr & 0x7) << 19) | SXGBE_MII_BUSY;
- writel(reg, sp->ioaddr + sp->hw->mii.data);
- }
- static void sxgbe_mdio_c45(struct sxgbe_priv_data *sp, u32 cmd, int phyaddr,
- int phyreg, u16 phydata)
- {
- u32 reg;
- /* set mdio address register */
- reg = ((phyreg >> 16) & 0x1f) << 21;
- reg |= (phyaddr << 16) | (phyreg & 0xffff);
- writel(reg, sp->ioaddr + sp->hw->mii.addr);
- sxgbe_mdio_ctrl_data(sp, cmd, phydata);
- }
- static void sxgbe_mdio_c22(struct sxgbe_priv_data *sp, u32 cmd, int phyaddr,
- int phyreg, u16 phydata)
- {
- u32 reg;
- writel(1 << phyaddr, sp->ioaddr + SXGBE_MDIO_CLAUSE22_PORT_REG);
- /* set mdio address register */
- reg = (phyaddr << 16) | (phyreg & 0x1f);
- writel(reg, sp->ioaddr + sp->hw->mii.addr);
- sxgbe_mdio_ctrl_data(sp, cmd, phydata);
- }
- static int sxgbe_mdio_access(struct sxgbe_priv_data *sp, u32 cmd, int phyaddr,
- int phyreg, u16 phydata)
- {
- const struct mii_regs *mii = &sp->hw->mii;
- int rc;
- rc = sxgbe_mdio_busy_wait(sp->ioaddr, mii->data);
- if (rc < 0)
- return rc;
- if (phyreg & MII_ADDR_C45) {
- sxgbe_mdio_c45(sp, cmd, phyaddr, phyreg, phydata);
- } else {
- /* Ports 0-3 only support C22. */
- if (phyaddr >= 4)
- return -ENODEV;
- sxgbe_mdio_c22(sp, cmd, phyaddr, phyreg, phydata);
- }
- return sxgbe_mdio_busy_wait(sp->ioaddr, mii->data);
- }
- /**
- * sxgbe_mdio_read
- * @bus: points to the mii_bus structure
- * @phyaddr: address of phy port
- * @phyreg: address of register with in phy register
- * Description: this function used for C45 and C22 MDIO Read
- */
- static int sxgbe_mdio_read(struct mii_bus *bus, int phyaddr, int phyreg)
- {
- struct net_device *ndev = bus->priv;
- struct sxgbe_priv_data *priv = netdev_priv(ndev);
- int rc;
- rc = sxgbe_mdio_access(priv, SXGBE_SMA_READ_CMD, phyaddr, phyreg, 0);
- if (rc < 0)
- return rc;
- return readl(priv->ioaddr + priv->hw->mii.data) & 0xffff;
- }
- /**
- * sxgbe_mdio_write
- * @bus: points to the mii_bus structure
- * @phyaddr: address of phy port
- * @phyreg: address of phy registers
- * @phydata: data to be written into phy register
- * Description: this function is used for C45 and C22 MDIO write
- */
- static int sxgbe_mdio_write(struct mii_bus *bus, int phyaddr, int phyreg,
- u16 phydata)
- {
- struct net_device *ndev = bus->priv;
- struct sxgbe_priv_data *priv = netdev_priv(ndev);
- return sxgbe_mdio_access(priv, SXGBE_SMA_WRITE_CMD, phyaddr, phyreg,
- phydata);
- }
- int sxgbe_mdio_register(struct net_device *ndev)
- {
- struct mii_bus *mdio_bus;
- struct sxgbe_priv_data *priv = netdev_priv(ndev);
- struct sxgbe_mdio_bus_data *mdio_data = priv->plat->mdio_bus_data;
- int err, phy_addr;
- int *irqlist;
- bool phy_found = false;
- bool act;
- /* allocate the new mdio bus */
- mdio_bus = mdiobus_alloc();
- if (!mdio_bus) {
- netdev_err(ndev, "%s: mii bus allocation failed\n", __func__);
- return -ENOMEM;
- }
- if (mdio_data->irqs)
- irqlist = mdio_data->irqs;
- else
- irqlist = priv->mii_irq;
- /* assign mii bus fields */
- mdio_bus->name = "sxgbe";
- mdio_bus->read = &sxgbe_mdio_read;
- mdio_bus->write = &sxgbe_mdio_write;
- snprintf(mdio_bus->id, MII_BUS_ID_SIZE, "%s-%x",
- mdio_bus->name, priv->plat->bus_id);
- mdio_bus->priv = ndev;
- mdio_bus->phy_mask = mdio_data->phy_mask;
- mdio_bus->parent = priv->device;
- /* register with kernel subsystem */
- err = mdiobus_register(mdio_bus);
- if (err != 0) {
- netdev_err(ndev, "mdiobus register failed\n");
- goto mdiobus_err;
- }
- for (phy_addr = 0; phy_addr < PHY_MAX_ADDR; phy_addr++) {
- struct phy_device *phy = mdio_bus->phy_map[phy_addr];
- if (phy) {
- char irq_num[4];
- char *irq_str;
- /* If an IRQ was provided to be assigned after
- * the bus probe, do it here.
- */
- if ((mdio_data->irqs == NULL) &&
- (mdio_data->probed_phy_irq > 0)) {
- irqlist[phy_addr] = mdio_data->probed_phy_irq;
- phy->irq = mdio_data->probed_phy_irq;
- }
- /* If we're going to bind the MAC to this PHY bus,
- * and no PHY number was provided to the MAC,
- * use the one probed here.
- */
- if (priv->plat->phy_addr == -1)
- priv->plat->phy_addr = phy_addr;
- act = (priv->plat->phy_addr == phy_addr);
- switch (phy->irq) {
- case PHY_POLL:
- irq_str = "POLL";
- break;
- case PHY_IGNORE_INTERRUPT:
- irq_str = "IGNORE";
- break;
- default:
- sprintf(irq_num, "%d", phy->irq);
- irq_str = irq_num;
- break;
- }
- netdev_info(ndev, "PHY ID %08x at %d IRQ %s (%s)%s\n",
- phy->phy_id, phy_addr, irq_str,
- dev_name(&phy->dev), act ? " active" : "");
- phy_found = true;
- }
- }
- if (!phy_found) {
- netdev_err(ndev, "PHY not found\n");
- goto phyfound_err;
- }
- priv->mii = mdio_bus;
- return 0;
- phyfound_err:
- err = -ENODEV;
- mdiobus_unregister(mdio_bus);
- mdiobus_err:
- mdiobus_free(mdio_bus);
- return err;
- }
- int sxgbe_mdio_unregister(struct net_device *ndev)
- {
- struct sxgbe_priv_data *priv = netdev_priv(ndev);
- if (!priv->mii)
- return 0;
- mdiobus_unregister(priv->mii);
- priv->mii->priv = NULL;
- mdiobus_free(priv->mii);
- priv->mii = NULL;
- return 0;
- }
|