net: dsa: bcm_sf2: Register our slave MDIO bus
authorFlorian Fainelli <f.fainelli@gmail.com>
Tue, 7 Jun 2016 23:32:43 +0000 (16:32 -0700)
committerDavid S. Miller <davem@davemloft.net>
Wed, 8 Jun 2016 18:23:42 +0000 (11:23 -0700)
Register a slave MDIO bus which allows us to divert problematic
read/writes towards conflicting pseudo-PHY address (30). Do no longer
rely on DSA's slave_mii_bus, but instead provide our own implementation
which offers more flexibility as to what to do, and when to register it.

We need to register it by the time we are able to get access to our
memory mapped registers, which is not until drv->setup() time. In order
to avoid forward declarations, we need to re-order the function bodies a
bit.

Reviewed-by: Andrew Lunn <andrew@lunn.ch>
Reviewed-by: Vivien Didelot <vivien.didelot@savoirfairelinux.com>
Signed-off-by: Florian Fainelli <f.fainelli@gmail.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
drivers/net/dsa/bcm_sf2.c
drivers/net/dsa/bcm_sf2.h

index 73df91bb04666b35365cb82458c33808fd72d596..d6625783703f2a75d63db5e1e7daca631705c532 100644 (file)
@@ -22,6 +22,7 @@
 #include <linux/of_irq.h>
 #include <linux/of_address.h>
 #include <linux/of_net.h>
+#include <linux/of_mdio.h>
 #include <net/dsa.h>
 #include <linux/ethtool.h>
 #include <linux/if_bridge.h>
@@ -836,6 +837,66 @@ static int bcm_sf2_sw_fdb_dump(struct dsa_switch *ds, int port,
        return 0;
 }
 
+static int bcm_sf2_sw_indir_rw(struct bcm_sf2_priv *priv, int op, int addr,
+                              int regnum, u16 val)
+{
+       int ret = 0;
+       u32 reg;
+
+       reg = reg_readl(priv, REG_SWITCH_CNTRL);
+       reg |= MDIO_MASTER_SEL;
+       reg_writel(priv, reg, REG_SWITCH_CNTRL);
+
+       /* Page << 8 | offset */
+       reg = 0x70;
+       reg <<= 2;
+       core_writel(priv, addr, reg);
+
+       /* Page << 8 | offset */
+       reg = 0x80 << 8 | regnum << 1;
+       reg <<= 2;
+
+       if (op)
+               ret = core_readl(priv, reg);
+       else
+               core_writel(priv, val, reg);
+
+       reg = reg_readl(priv, REG_SWITCH_CNTRL);
+       reg &= ~MDIO_MASTER_SEL;
+       reg_writel(priv, reg, REG_SWITCH_CNTRL);
+
+       return ret & 0xffff;
+}
+
+static int bcm_sf2_sw_mdio_read(struct mii_bus *bus, int addr, int regnum)
+{
+       struct bcm_sf2_priv *priv = bus->priv;
+
+       /* Intercept reads from Broadcom pseudo-PHY address, else, send
+        * them to our master MDIO bus controller
+        */
+       if (addr == BRCM_PSEUDO_PHY_ADDR && priv->indir_phy_mask & BIT(addr))
+               return bcm_sf2_sw_indir_rw(priv, 1, addr, regnum, 0);
+       else
+               return mdiobus_read(priv->master_mii_bus, addr, regnum);
+}
+
+static int bcm_sf2_sw_mdio_write(struct mii_bus *bus, int addr, int regnum,
+                                u16 val)
+{
+       struct bcm_sf2_priv *priv = bus->priv;
+
+       /* Intercept writes to the Broadcom pseudo-PHY address, else,
+        * send them to our master MDIO bus controller
+        */
+       if (addr == BRCM_PSEUDO_PHY_ADDR && priv->indir_phy_mask & BIT(addr))
+               bcm_sf2_sw_indir_rw(priv, 0, addr, regnum, val);
+       else
+               mdiobus_write(priv->master_mii_bus, addr, regnum, val);
+
+       return 0;
+}
+
 static irqreturn_t bcm_sf2_switch_0_isr(int irq, void *dev_id)
 {
        struct bcm_sf2_priv *priv = dev_id;
@@ -932,6 +993,72 @@ static void bcm_sf2_identify_ports(struct bcm_sf2_priv *priv,
        }
 }
 
+static int bcm_sf2_mdio_register(struct dsa_switch *ds)
+{
+       struct bcm_sf2_priv *priv = ds_to_priv(ds);
+       struct device_node *dn;
+       static int index;
+       int err;
+
+       /* Find our integrated MDIO bus node */
+       dn = of_find_compatible_node(NULL, NULL, "brcm,unimac-mdio");
+       priv->master_mii_bus = of_mdio_find_bus(dn);
+       if (!priv->master_mii_bus)
+               return -EPROBE_DEFER;
+
+       get_device(&priv->master_mii_bus->dev);
+       priv->master_mii_dn = dn;
+
+       priv->slave_mii_bus = devm_mdiobus_alloc(ds->dev);
+       if (!priv->slave_mii_bus)
+               return -ENOMEM;
+
+       priv->slave_mii_bus->priv = priv;
+       priv->slave_mii_bus->name = "sf2 slave mii";
+       priv->slave_mii_bus->read = bcm_sf2_sw_mdio_read;
+       priv->slave_mii_bus->write = bcm_sf2_sw_mdio_write;
+       snprintf(priv->slave_mii_bus->id, MII_BUS_ID_SIZE, "sf2-%d",
+                index++);
+       priv->slave_mii_bus->dev.of_node = dn;
+
+       /* Include the pseudo-PHY address to divert reads towards our
+        * workaround. This is only required for 7445D0, since 7445E0
+        * disconnects the internal switch pseudo-PHY such that we can use the
+        * regular SWITCH_MDIO master controller instead.
+        *
+        * Here we flag the pseudo PHY as needing special treatment and would
+        * otherwise make all other PHY read/writes go to the master MDIO bus
+        * controller that comes with this switch backed by the "mdio-unimac"
+        * driver.
+        */
+       if (of_machine_is_compatible("brcm,bcm7445d0"))
+               priv->indir_phy_mask |= (1 << BRCM_PSEUDO_PHY_ADDR);
+       else
+               priv->indir_phy_mask = 0;
+
+       ds->phys_mii_mask = priv->indir_phy_mask;
+       ds->slave_mii_bus = priv->slave_mii_bus;
+       priv->slave_mii_bus->parent = ds->dev->parent;
+       priv->slave_mii_bus->phy_mask = ~priv->indir_phy_mask;
+
+       if (dn)
+               err = of_mdiobus_register(priv->slave_mii_bus, dn);
+       else
+               err = mdiobus_register(priv->slave_mii_bus);
+
+       if (err)
+               of_node_put(dn);
+
+       return err;
+}
+
+static void bcm_sf2_mdio_unregister(struct bcm_sf2_priv *priv)
+{
+       mdiobus_unregister(priv->slave_mii_bus);
+       if (priv->master_mii_dn)
+               of_node_put(priv->master_mii_dn);
+}
+
 static int bcm_sf2_sw_setup(struct dsa_switch *ds)
 {
        const char *reg_names[BCM_SF2_REGS_NUM] = BCM_SF2_REGS_NAME;
@@ -972,6 +1099,12 @@ static int bcm_sf2_sw_setup(struct dsa_switch *ds)
                goto out_unmap;
        }
 
+       ret = bcm_sf2_mdio_register(ds);
+       if (ret) {
+               pr_err("failed to register MDIO bus\n");
+               goto out_unmap;
+       }
+
        /* Disable all interrupts and request them */
        bcm_sf2_intr_disable(priv);
 
@@ -1017,23 +1150,6 @@ static int bcm_sf2_sw_setup(struct dsa_switch *ds)
                        bcm_sf2_port_disable(ds, port, NULL);
        }
 
-       /* Include the pseudo-PHY address and the broadcast PHY address to
-        * divert reads towards our workaround. This is only required for
-        * 7445D0, since 7445E0 disconnects the internal switch pseudo-PHY such
-        * that we can use the regular SWITCH_MDIO master controller instead.
-        *
-        * By default, DSA initializes ds->phys_mii_mask to
-        * ds->enabled_port_mask to have a 1:1 mapping between Port address
-        * and PHY address in order to utilize the slave_mii_bus instance to
-        * read from Port PHYs. This is not what we want here, so we
-        * initialize phys_mii_mask 0 to always utilize the "master" MDIO
-        * bus backed by the "mdio-unimac" driver.
-        */
-       if (of_machine_is_compatible("brcm,bcm7445d0"))
-               ds->phys_mii_mask |= ((1 << BRCM_PSEUDO_PHY_ADDR) | (1 << 0));
-       else
-               ds->phys_mii_mask = 0;
-
        rev = reg_readl(priv, REG_SWITCH_REVISION);
        priv->hw_params.top_rev = (rev >> SWITCH_TOP_REV_SHIFT) &
                                        SWITCH_TOP_REV_MASK;
@@ -1058,6 +1174,7 @@ out_unmap:
                        iounmap(*base);
                base++;
        }
+       bcm_sf2_mdio_unregister(priv);
        return ret;
 }
 
@@ -1078,68 +1195,6 @@ static u32 bcm_sf2_sw_get_phy_flags(struct dsa_switch *ds, int port)
        return priv->hw_params.gphy_rev;
 }
 
-static int bcm_sf2_sw_indir_rw(struct dsa_switch *ds, int op, int addr,
-                              int regnum, u16 val)
-{
-       struct bcm_sf2_priv *priv = ds_to_priv(ds);
-       int ret = 0;
-       u32 reg;
-
-       reg = reg_readl(priv, REG_SWITCH_CNTRL);
-       reg |= MDIO_MASTER_SEL;
-       reg_writel(priv, reg, REG_SWITCH_CNTRL);
-
-       /* Page << 8 | offset */
-       reg = 0x70;
-       reg <<= 2;
-       core_writel(priv, addr, reg);
-
-       /* Page << 8 | offset */
-       reg = 0x80 << 8 | regnum << 1;
-       reg <<= 2;
-
-       if (op)
-               ret = core_readl(priv, reg);
-       else
-               core_writel(priv, val, reg);
-
-       reg = reg_readl(priv, REG_SWITCH_CNTRL);
-       reg &= ~MDIO_MASTER_SEL;
-       reg_writel(priv, reg, REG_SWITCH_CNTRL);
-
-       return ret & 0xffff;
-}
-
-static int bcm_sf2_sw_phy_read(struct dsa_switch *ds, int addr, int regnum)
-{
-       /* Intercept reads from the MDIO broadcast address or Broadcom
-        * pseudo-PHY address
-        */
-       switch (addr) {
-       case 0:
-       case BRCM_PSEUDO_PHY_ADDR:
-               return bcm_sf2_sw_indir_rw(ds, 1, addr, regnum, 0);
-       default:
-               return 0xffff;
-       }
-}
-
-static int bcm_sf2_sw_phy_write(struct dsa_switch *ds, int addr, int regnum,
-                               u16 val)
-{
-       /* Intercept writes to the MDIO broadcast address or Broadcom
-        * pseudo-PHY address
-        */
-       switch (addr) {
-       case 0:
-       case BRCM_PSEUDO_PHY_ADDR:
-               bcm_sf2_sw_indir_rw(ds, 0, addr, regnum, val);
-               break;
-       }
-
-       return 0;
-}
-
 static void bcm_sf2_sw_adjust_link(struct dsa_switch *ds, int port,
                                   struct phy_device *phydev)
 {
@@ -1376,8 +1431,6 @@ static struct dsa_switch_driver bcm_sf2_switch_driver = {
        .setup                  = bcm_sf2_sw_setup,
        .set_addr               = bcm_sf2_sw_set_addr,
        .get_phy_flags          = bcm_sf2_sw_get_phy_flags,
-       .phy_read               = bcm_sf2_sw_phy_read,
-       .phy_write              = bcm_sf2_sw_phy_write,
        .get_strings            = bcm_sf2_sw_get_strings,
        .get_ethtool_stats      = bcm_sf2_sw_get_ethtool_stats,
        .get_sset_count         = bcm_sf2_sw_get_sset_count,
index 200b1f5fdb5679ad45b7c1c9fa30a9d381e62b72..bde11ebb2742bc8b49ed56e4b7f9e85658f06730 100644 (file)
@@ -142,6 +142,12 @@ struct bcm_sf2_priv {
 
        /* Bitmask of ports having an integrated PHY */
        unsigned int                    int_phy_mask;
+
+       /* Master and slave MDIO bus controller */
+       unsigned int                    indir_phy_mask;
+       struct device_node              *master_mii_dn;
+       struct mii_bus                  *slave_mii_bus;
+       struct mii_bus                  *master_mii_bus;
 };
 
 struct bcm_sf2_hw_stats {