162306a36Sopenharmony_ci/* 262306a36Sopenharmony_ci * B53 register access through MII registers 362306a36Sopenharmony_ci * 462306a36Sopenharmony_ci * Copyright (C) 2011-2013 Jonas Gorski <jogo@openwrt.org> 562306a36Sopenharmony_ci * 662306a36Sopenharmony_ci * Permission to use, copy, modify, and/or distribute this software for any 762306a36Sopenharmony_ci * purpose with or without fee is hereby granted, provided that the above 862306a36Sopenharmony_ci * copyright notice and this permission notice appear in all copies. 962306a36Sopenharmony_ci * 1062306a36Sopenharmony_ci * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES 1162306a36Sopenharmony_ci * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF 1262306a36Sopenharmony_ci * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR 1362306a36Sopenharmony_ci * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES 1462306a36Sopenharmony_ci * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN 1562306a36Sopenharmony_ci * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF 1662306a36Sopenharmony_ci * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. 1762306a36Sopenharmony_ci */ 1862306a36Sopenharmony_ci 1962306a36Sopenharmony_ci#include <linux/kernel.h> 2062306a36Sopenharmony_ci#include <linux/phy.h> 2162306a36Sopenharmony_ci#include <linux/module.h> 2262306a36Sopenharmony_ci#include <linux/of.h> 2362306a36Sopenharmony_ci#include <linux/delay.h> 2462306a36Sopenharmony_ci#include <linux/brcmphy.h> 2562306a36Sopenharmony_ci#include <linux/rtnetlink.h> 2662306a36Sopenharmony_ci#include <net/dsa.h> 2762306a36Sopenharmony_ci 2862306a36Sopenharmony_ci#include "b53_priv.h" 2962306a36Sopenharmony_ci 3062306a36Sopenharmony_ci/* MII registers */ 3162306a36Sopenharmony_ci#define REG_MII_PAGE 0x10 /* MII Page register */ 3262306a36Sopenharmony_ci#define REG_MII_ADDR 0x11 /* MII Address register */ 3362306a36Sopenharmony_ci#define REG_MII_DATA0 0x18 /* MII Data register 0 */ 3462306a36Sopenharmony_ci#define REG_MII_DATA1 0x19 /* MII Data register 1 */ 3562306a36Sopenharmony_ci#define REG_MII_DATA2 0x1a /* MII Data register 2 */ 3662306a36Sopenharmony_ci#define REG_MII_DATA3 0x1b /* MII Data register 3 */ 3762306a36Sopenharmony_ci 3862306a36Sopenharmony_ci#define REG_MII_PAGE_ENABLE BIT(0) 3962306a36Sopenharmony_ci#define REG_MII_ADDR_WRITE BIT(0) 4062306a36Sopenharmony_ci#define REG_MII_ADDR_READ BIT(1) 4162306a36Sopenharmony_ci 4262306a36Sopenharmony_cistatic int b53_mdio_op(struct b53_device *dev, u8 page, u8 reg, u16 op) 4362306a36Sopenharmony_ci{ 4462306a36Sopenharmony_ci int i; 4562306a36Sopenharmony_ci u16 v; 4662306a36Sopenharmony_ci int ret; 4762306a36Sopenharmony_ci struct mii_bus *bus = dev->priv; 4862306a36Sopenharmony_ci 4962306a36Sopenharmony_ci if (dev->current_page != page) { 5062306a36Sopenharmony_ci /* set page number */ 5162306a36Sopenharmony_ci v = (page << 8) | REG_MII_PAGE_ENABLE; 5262306a36Sopenharmony_ci ret = mdiobus_write_nested(bus, BRCM_PSEUDO_PHY_ADDR, 5362306a36Sopenharmony_ci REG_MII_PAGE, v); 5462306a36Sopenharmony_ci if (ret) 5562306a36Sopenharmony_ci return ret; 5662306a36Sopenharmony_ci dev->current_page = page; 5762306a36Sopenharmony_ci } 5862306a36Sopenharmony_ci 5962306a36Sopenharmony_ci /* set register address */ 6062306a36Sopenharmony_ci v = (reg << 8) | op; 6162306a36Sopenharmony_ci ret = mdiobus_write_nested(bus, BRCM_PSEUDO_PHY_ADDR, REG_MII_ADDR, v); 6262306a36Sopenharmony_ci if (ret) 6362306a36Sopenharmony_ci return ret; 6462306a36Sopenharmony_ci 6562306a36Sopenharmony_ci /* check if operation completed */ 6662306a36Sopenharmony_ci for (i = 0; i < 5; ++i) { 6762306a36Sopenharmony_ci v = mdiobus_read_nested(bus, BRCM_PSEUDO_PHY_ADDR, 6862306a36Sopenharmony_ci REG_MII_ADDR); 6962306a36Sopenharmony_ci if (!(v & (REG_MII_ADDR_WRITE | REG_MII_ADDR_READ))) 7062306a36Sopenharmony_ci break; 7162306a36Sopenharmony_ci usleep_range(10, 100); 7262306a36Sopenharmony_ci } 7362306a36Sopenharmony_ci 7462306a36Sopenharmony_ci if (WARN_ON(i == 5)) 7562306a36Sopenharmony_ci return -EIO; 7662306a36Sopenharmony_ci 7762306a36Sopenharmony_ci return 0; 7862306a36Sopenharmony_ci} 7962306a36Sopenharmony_ci 8062306a36Sopenharmony_cistatic int b53_mdio_read8(struct b53_device *dev, u8 page, u8 reg, u8 *val) 8162306a36Sopenharmony_ci{ 8262306a36Sopenharmony_ci struct mii_bus *bus = dev->priv; 8362306a36Sopenharmony_ci int ret; 8462306a36Sopenharmony_ci 8562306a36Sopenharmony_ci ret = b53_mdio_op(dev, page, reg, REG_MII_ADDR_READ); 8662306a36Sopenharmony_ci if (ret) 8762306a36Sopenharmony_ci return ret; 8862306a36Sopenharmony_ci 8962306a36Sopenharmony_ci *val = mdiobus_read_nested(bus, BRCM_PSEUDO_PHY_ADDR, 9062306a36Sopenharmony_ci REG_MII_DATA0) & 0xff; 9162306a36Sopenharmony_ci 9262306a36Sopenharmony_ci return 0; 9362306a36Sopenharmony_ci} 9462306a36Sopenharmony_ci 9562306a36Sopenharmony_cistatic int b53_mdio_read16(struct b53_device *dev, u8 page, u8 reg, u16 *val) 9662306a36Sopenharmony_ci{ 9762306a36Sopenharmony_ci struct mii_bus *bus = dev->priv; 9862306a36Sopenharmony_ci int ret; 9962306a36Sopenharmony_ci 10062306a36Sopenharmony_ci ret = b53_mdio_op(dev, page, reg, REG_MII_ADDR_READ); 10162306a36Sopenharmony_ci if (ret) 10262306a36Sopenharmony_ci return ret; 10362306a36Sopenharmony_ci 10462306a36Sopenharmony_ci *val = mdiobus_read_nested(bus, BRCM_PSEUDO_PHY_ADDR, REG_MII_DATA0); 10562306a36Sopenharmony_ci 10662306a36Sopenharmony_ci return 0; 10762306a36Sopenharmony_ci} 10862306a36Sopenharmony_ci 10962306a36Sopenharmony_cistatic int b53_mdio_read32(struct b53_device *dev, u8 page, u8 reg, u32 *val) 11062306a36Sopenharmony_ci{ 11162306a36Sopenharmony_ci struct mii_bus *bus = dev->priv; 11262306a36Sopenharmony_ci int ret; 11362306a36Sopenharmony_ci 11462306a36Sopenharmony_ci ret = b53_mdio_op(dev, page, reg, REG_MII_ADDR_READ); 11562306a36Sopenharmony_ci if (ret) 11662306a36Sopenharmony_ci return ret; 11762306a36Sopenharmony_ci 11862306a36Sopenharmony_ci *val = mdiobus_read_nested(bus, BRCM_PSEUDO_PHY_ADDR, REG_MII_DATA0); 11962306a36Sopenharmony_ci *val |= mdiobus_read_nested(bus, BRCM_PSEUDO_PHY_ADDR, 12062306a36Sopenharmony_ci REG_MII_DATA1) << 16; 12162306a36Sopenharmony_ci 12262306a36Sopenharmony_ci return 0; 12362306a36Sopenharmony_ci} 12462306a36Sopenharmony_ci 12562306a36Sopenharmony_cistatic int b53_mdio_read48(struct b53_device *dev, u8 page, u8 reg, u64 *val) 12662306a36Sopenharmony_ci{ 12762306a36Sopenharmony_ci struct mii_bus *bus = dev->priv; 12862306a36Sopenharmony_ci u64 temp = 0; 12962306a36Sopenharmony_ci int i; 13062306a36Sopenharmony_ci int ret; 13162306a36Sopenharmony_ci 13262306a36Sopenharmony_ci ret = b53_mdio_op(dev, page, reg, REG_MII_ADDR_READ); 13362306a36Sopenharmony_ci if (ret) 13462306a36Sopenharmony_ci return ret; 13562306a36Sopenharmony_ci 13662306a36Sopenharmony_ci for (i = 2; i >= 0; i--) { 13762306a36Sopenharmony_ci temp <<= 16; 13862306a36Sopenharmony_ci temp |= mdiobus_read_nested(bus, BRCM_PSEUDO_PHY_ADDR, 13962306a36Sopenharmony_ci REG_MII_DATA0 + i); 14062306a36Sopenharmony_ci } 14162306a36Sopenharmony_ci 14262306a36Sopenharmony_ci *val = temp; 14362306a36Sopenharmony_ci 14462306a36Sopenharmony_ci return 0; 14562306a36Sopenharmony_ci} 14662306a36Sopenharmony_ci 14762306a36Sopenharmony_cistatic int b53_mdio_read64(struct b53_device *dev, u8 page, u8 reg, u64 *val) 14862306a36Sopenharmony_ci{ 14962306a36Sopenharmony_ci struct mii_bus *bus = dev->priv; 15062306a36Sopenharmony_ci u64 temp = 0; 15162306a36Sopenharmony_ci int i; 15262306a36Sopenharmony_ci int ret; 15362306a36Sopenharmony_ci 15462306a36Sopenharmony_ci ret = b53_mdio_op(dev, page, reg, REG_MII_ADDR_READ); 15562306a36Sopenharmony_ci if (ret) 15662306a36Sopenharmony_ci return ret; 15762306a36Sopenharmony_ci 15862306a36Sopenharmony_ci for (i = 3; i >= 0; i--) { 15962306a36Sopenharmony_ci temp <<= 16; 16062306a36Sopenharmony_ci temp |= mdiobus_read_nested(bus, BRCM_PSEUDO_PHY_ADDR, 16162306a36Sopenharmony_ci REG_MII_DATA0 + i); 16262306a36Sopenharmony_ci } 16362306a36Sopenharmony_ci 16462306a36Sopenharmony_ci *val = temp; 16562306a36Sopenharmony_ci 16662306a36Sopenharmony_ci return 0; 16762306a36Sopenharmony_ci} 16862306a36Sopenharmony_ci 16962306a36Sopenharmony_cistatic int b53_mdio_write8(struct b53_device *dev, u8 page, u8 reg, u8 value) 17062306a36Sopenharmony_ci{ 17162306a36Sopenharmony_ci struct mii_bus *bus = dev->priv; 17262306a36Sopenharmony_ci int ret; 17362306a36Sopenharmony_ci 17462306a36Sopenharmony_ci ret = mdiobus_write_nested(bus, BRCM_PSEUDO_PHY_ADDR, 17562306a36Sopenharmony_ci REG_MII_DATA0, value); 17662306a36Sopenharmony_ci if (ret) 17762306a36Sopenharmony_ci return ret; 17862306a36Sopenharmony_ci 17962306a36Sopenharmony_ci return b53_mdio_op(dev, page, reg, REG_MII_ADDR_WRITE); 18062306a36Sopenharmony_ci} 18162306a36Sopenharmony_ci 18262306a36Sopenharmony_cistatic int b53_mdio_write16(struct b53_device *dev, u8 page, u8 reg, 18362306a36Sopenharmony_ci u16 value) 18462306a36Sopenharmony_ci{ 18562306a36Sopenharmony_ci struct mii_bus *bus = dev->priv; 18662306a36Sopenharmony_ci int ret; 18762306a36Sopenharmony_ci 18862306a36Sopenharmony_ci ret = mdiobus_write_nested(bus, BRCM_PSEUDO_PHY_ADDR, 18962306a36Sopenharmony_ci REG_MII_DATA0, value); 19062306a36Sopenharmony_ci if (ret) 19162306a36Sopenharmony_ci return ret; 19262306a36Sopenharmony_ci 19362306a36Sopenharmony_ci return b53_mdio_op(dev, page, reg, REG_MII_ADDR_WRITE); 19462306a36Sopenharmony_ci} 19562306a36Sopenharmony_ci 19662306a36Sopenharmony_cistatic int b53_mdio_write32(struct b53_device *dev, u8 page, u8 reg, 19762306a36Sopenharmony_ci u32 value) 19862306a36Sopenharmony_ci{ 19962306a36Sopenharmony_ci struct mii_bus *bus = dev->priv; 20062306a36Sopenharmony_ci unsigned int i; 20162306a36Sopenharmony_ci u32 temp = value; 20262306a36Sopenharmony_ci 20362306a36Sopenharmony_ci for (i = 0; i < 2; i++) { 20462306a36Sopenharmony_ci int ret = mdiobus_write_nested(bus, BRCM_PSEUDO_PHY_ADDR, 20562306a36Sopenharmony_ci REG_MII_DATA0 + i, 20662306a36Sopenharmony_ci temp & 0xffff); 20762306a36Sopenharmony_ci if (ret) 20862306a36Sopenharmony_ci return ret; 20962306a36Sopenharmony_ci temp >>= 16; 21062306a36Sopenharmony_ci } 21162306a36Sopenharmony_ci 21262306a36Sopenharmony_ci return b53_mdio_op(dev, page, reg, REG_MII_ADDR_WRITE); 21362306a36Sopenharmony_ci} 21462306a36Sopenharmony_ci 21562306a36Sopenharmony_cistatic int b53_mdio_write48(struct b53_device *dev, u8 page, u8 reg, 21662306a36Sopenharmony_ci u64 value) 21762306a36Sopenharmony_ci{ 21862306a36Sopenharmony_ci struct mii_bus *bus = dev->priv; 21962306a36Sopenharmony_ci unsigned int i; 22062306a36Sopenharmony_ci u64 temp = value; 22162306a36Sopenharmony_ci 22262306a36Sopenharmony_ci for (i = 0; i < 3; i++) { 22362306a36Sopenharmony_ci int ret = mdiobus_write_nested(bus, BRCM_PSEUDO_PHY_ADDR, 22462306a36Sopenharmony_ci REG_MII_DATA0 + i, 22562306a36Sopenharmony_ci temp & 0xffff); 22662306a36Sopenharmony_ci if (ret) 22762306a36Sopenharmony_ci return ret; 22862306a36Sopenharmony_ci temp >>= 16; 22962306a36Sopenharmony_ci } 23062306a36Sopenharmony_ci 23162306a36Sopenharmony_ci return b53_mdio_op(dev, page, reg, REG_MII_ADDR_WRITE); 23262306a36Sopenharmony_ci} 23362306a36Sopenharmony_ci 23462306a36Sopenharmony_cistatic int b53_mdio_write64(struct b53_device *dev, u8 page, u8 reg, 23562306a36Sopenharmony_ci u64 value) 23662306a36Sopenharmony_ci{ 23762306a36Sopenharmony_ci struct mii_bus *bus = dev->priv; 23862306a36Sopenharmony_ci unsigned int i; 23962306a36Sopenharmony_ci u64 temp = value; 24062306a36Sopenharmony_ci 24162306a36Sopenharmony_ci for (i = 0; i < 4; i++) { 24262306a36Sopenharmony_ci int ret = mdiobus_write_nested(bus, BRCM_PSEUDO_PHY_ADDR, 24362306a36Sopenharmony_ci REG_MII_DATA0 + i, 24462306a36Sopenharmony_ci temp & 0xffff); 24562306a36Sopenharmony_ci if (ret) 24662306a36Sopenharmony_ci return ret; 24762306a36Sopenharmony_ci temp >>= 16; 24862306a36Sopenharmony_ci } 24962306a36Sopenharmony_ci 25062306a36Sopenharmony_ci return b53_mdio_op(dev, page, reg, REG_MII_ADDR_WRITE); 25162306a36Sopenharmony_ci} 25262306a36Sopenharmony_ci 25362306a36Sopenharmony_cistatic int b53_mdio_phy_read16(struct b53_device *dev, int addr, int reg, 25462306a36Sopenharmony_ci u16 *value) 25562306a36Sopenharmony_ci{ 25662306a36Sopenharmony_ci struct mii_bus *bus = dev->priv; 25762306a36Sopenharmony_ci 25862306a36Sopenharmony_ci *value = mdiobus_read_nested(bus, addr, reg); 25962306a36Sopenharmony_ci 26062306a36Sopenharmony_ci return 0; 26162306a36Sopenharmony_ci} 26262306a36Sopenharmony_ci 26362306a36Sopenharmony_cistatic int b53_mdio_phy_write16(struct b53_device *dev, int addr, int reg, 26462306a36Sopenharmony_ci u16 value) 26562306a36Sopenharmony_ci{ 26662306a36Sopenharmony_ci struct mii_bus *bus = dev->bus; 26762306a36Sopenharmony_ci 26862306a36Sopenharmony_ci return mdiobus_write_nested(bus, addr, reg, value); 26962306a36Sopenharmony_ci} 27062306a36Sopenharmony_ci 27162306a36Sopenharmony_cistatic const struct b53_io_ops b53_mdio_ops = { 27262306a36Sopenharmony_ci .read8 = b53_mdio_read8, 27362306a36Sopenharmony_ci .read16 = b53_mdio_read16, 27462306a36Sopenharmony_ci .read32 = b53_mdio_read32, 27562306a36Sopenharmony_ci .read48 = b53_mdio_read48, 27662306a36Sopenharmony_ci .read64 = b53_mdio_read64, 27762306a36Sopenharmony_ci .write8 = b53_mdio_write8, 27862306a36Sopenharmony_ci .write16 = b53_mdio_write16, 27962306a36Sopenharmony_ci .write32 = b53_mdio_write32, 28062306a36Sopenharmony_ci .write48 = b53_mdio_write48, 28162306a36Sopenharmony_ci .write64 = b53_mdio_write64, 28262306a36Sopenharmony_ci .phy_read16 = b53_mdio_phy_read16, 28362306a36Sopenharmony_ci .phy_write16 = b53_mdio_phy_write16, 28462306a36Sopenharmony_ci}; 28562306a36Sopenharmony_ci 28662306a36Sopenharmony_ci#define B53_BRCM_OUI_1 0x0143bc00 28762306a36Sopenharmony_ci#define B53_BRCM_OUI_2 0x03625c00 28862306a36Sopenharmony_ci#define B53_BRCM_OUI_3 0x00406000 28962306a36Sopenharmony_ci#define B53_BRCM_OUI_4 0x01410c00 29062306a36Sopenharmony_ci#define B53_BRCM_OUI_5 0xae025000 29162306a36Sopenharmony_ci 29262306a36Sopenharmony_cistatic int b53_mdio_probe(struct mdio_device *mdiodev) 29362306a36Sopenharmony_ci{ 29462306a36Sopenharmony_ci struct b53_device *dev; 29562306a36Sopenharmony_ci u32 phy_id; 29662306a36Sopenharmony_ci int ret; 29762306a36Sopenharmony_ci 29862306a36Sopenharmony_ci /* allow the generic PHY driver to take over the non-management MDIO 29962306a36Sopenharmony_ci * addresses 30062306a36Sopenharmony_ci */ 30162306a36Sopenharmony_ci if (mdiodev->addr != BRCM_PSEUDO_PHY_ADDR && mdiodev->addr != 0) { 30262306a36Sopenharmony_ci dev_err(&mdiodev->dev, "leaving address %d to PHY\n", 30362306a36Sopenharmony_ci mdiodev->addr); 30462306a36Sopenharmony_ci return -ENODEV; 30562306a36Sopenharmony_ci } 30662306a36Sopenharmony_ci 30762306a36Sopenharmony_ci /* read the first port's id */ 30862306a36Sopenharmony_ci phy_id = mdiobus_read(mdiodev->bus, 0, 2) << 16; 30962306a36Sopenharmony_ci phy_id |= mdiobus_read(mdiodev->bus, 0, 3); 31062306a36Sopenharmony_ci 31162306a36Sopenharmony_ci /* BCM5325, BCM539x (OUI_1) 31262306a36Sopenharmony_ci * BCM53125, BCM53128 (OUI_2) 31362306a36Sopenharmony_ci * BCM5365 (OUI_3) 31462306a36Sopenharmony_ci */ 31562306a36Sopenharmony_ci if ((phy_id & 0xfffffc00) != B53_BRCM_OUI_1 && 31662306a36Sopenharmony_ci (phy_id & 0xfffffc00) != B53_BRCM_OUI_2 && 31762306a36Sopenharmony_ci (phy_id & 0xfffffc00) != B53_BRCM_OUI_3 && 31862306a36Sopenharmony_ci (phy_id & 0xfffffc00) != B53_BRCM_OUI_4 && 31962306a36Sopenharmony_ci (phy_id & 0xfffffc00) != B53_BRCM_OUI_5) { 32062306a36Sopenharmony_ci dev_err(&mdiodev->dev, "Unsupported device: 0x%08x\n", phy_id); 32162306a36Sopenharmony_ci return -ENODEV; 32262306a36Sopenharmony_ci } 32362306a36Sopenharmony_ci 32462306a36Sopenharmony_ci /* First probe will come from SWITCH_MDIO controller on the 7445D0 32562306a36Sopenharmony_ci * switch, which will conflict with the 7445 integrated switch 32662306a36Sopenharmony_ci * pseudo-phy (we end-up programming both). In that case, we return 32762306a36Sopenharmony_ci * -EPROBE_DEFER for the first time we get here, and wait until we come 32862306a36Sopenharmony_ci * back with the slave MDIO bus which has the correct indirection 32962306a36Sopenharmony_ci * layer setup 33062306a36Sopenharmony_ci */ 33162306a36Sopenharmony_ci if (of_machine_is_compatible("brcm,bcm7445d0") && 33262306a36Sopenharmony_ci strcmp(mdiodev->bus->name, "sf2 slave mii")) 33362306a36Sopenharmony_ci return -EPROBE_DEFER; 33462306a36Sopenharmony_ci 33562306a36Sopenharmony_ci dev = b53_switch_alloc(&mdiodev->dev, &b53_mdio_ops, mdiodev->bus); 33662306a36Sopenharmony_ci if (!dev) 33762306a36Sopenharmony_ci return -ENOMEM; 33862306a36Sopenharmony_ci 33962306a36Sopenharmony_ci /* we don't use page 0xff, so force a page set */ 34062306a36Sopenharmony_ci dev->current_page = 0xff; 34162306a36Sopenharmony_ci dev->bus = mdiodev->bus; 34262306a36Sopenharmony_ci 34362306a36Sopenharmony_ci dev_set_drvdata(&mdiodev->dev, dev); 34462306a36Sopenharmony_ci 34562306a36Sopenharmony_ci ret = b53_switch_register(dev); 34662306a36Sopenharmony_ci if (ret) { 34762306a36Sopenharmony_ci dev_err(&mdiodev->dev, "failed to register switch: %i\n", ret); 34862306a36Sopenharmony_ci return ret; 34962306a36Sopenharmony_ci } 35062306a36Sopenharmony_ci 35162306a36Sopenharmony_ci return ret; 35262306a36Sopenharmony_ci} 35362306a36Sopenharmony_ci 35462306a36Sopenharmony_cistatic void b53_mdio_remove(struct mdio_device *mdiodev) 35562306a36Sopenharmony_ci{ 35662306a36Sopenharmony_ci struct b53_device *dev = dev_get_drvdata(&mdiodev->dev); 35762306a36Sopenharmony_ci 35862306a36Sopenharmony_ci if (!dev) 35962306a36Sopenharmony_ci return; 36062306a36Sopenharmony_ci 36162306a36Sopenharmony_ci b53_switch_remove(dev); 36262306a36Sopenharmony_ci} 36362306a36Sopenharmony_ci 36462306a36Sopenharmony_cistatic void b53_mdio_shutdown(struct mdio_device *mdiodev) 36562306a36Sopenharmony_ci{ 36662306a36Sopenharmony_ci struct b53_device *dev = dev_get_drvdata(&mdiodev->dev); 36762306a36Sopenharmony_ci 36862306a36Sopenharmony_ci if (!dev) 36962306a36Sopenharmony_ci return; 37062306a36Sopenharmony_ci 37162306a36Sopenharmony_ci b53_switch_shutdown(dev); 37262306a36Sopenharmony_ci 37362306a36Sopenharmony_ci dev_set_drvdata(&mdiodev->dev, NULL); 37462306a36Sopenharmony_ci} 37562306a36Sopenharmony_ci 37662306a36Sopenharmony_cistatic const struct of_device_id b53_of_match[] = { 37762306a36Sopenharmony_ci { .compatible = "brcm,bcm5325" }, 37862306a36Sopenharmony_ci { .compatible = "brcm,bcm53115" }, 37962306a36Sopenharmony_ci { .compatible = "brcm,bcm53125" }, 38062306a36Sopenharmony_ci { .compatible = "brcm,bcm53128" }, 38162306a36Sopenharmony_ci { .compatible = "brcm,bcm53134" }, 38262306a36Sopenharmony_ci { .compatible = "brcm,bcm5365" }, 38362306a36Sopenharmony_ci { .compatible = "brcm,bcm5389" }, 38462306a36Sopenharmony_ci { .compatible = "brcm,bcm5395" }, 38562306a36Sopenharmony_ci { .compatible = "brcm,bcm5397" }, 38662306a36Sopenharmony_ci { .compatible = "brcm,bcm5398" }, 38762306a36Sopenharmony_ci { /* sentinel */ }, 38862306a36Sopenharmony_ci}; 38962306a36Sopenharmony_ciMODULE_DEVICE_TABLE(of, b53_of_match); 39062306a36Sopenharmony_ci 39162306a36Sopenharmony_cistatic struct mdio_driver b53_mdio_driver = { 39262306a36Sopenharmony_ci .probe = b53_mdio_probe, 39362306a36Sopenharmony_ci .remove = b53_mdio_remove, 39462306a36Sopenharmony_ci .shutdown = b53_mdio_shutdown, 39562306a36Sopenharmony_ci .mdiodrv.driver = { 39662306a36Sopenharmony_ci .name = "bcm53xx", 39762306a36Sopenharmony_ci .of_match_table = b53_of_match, 39862306a36Sopenharmony_ci }, 39962306a36Sopenharmony_ci}; 40062306a36Sopenharmony_cimdio_module_driver(b53_mdio_driver); 40162306a36Sopenharmony_ci 40262306a36Sopenharmony_ciMODULE_DESCRIPTION("B53 MDIO access driver"); 40362306a36Sopenharmony_ciMODULE_LICENSE("Dual BSD/GPL"); 404