| diff --git a/drivers/net/phy/Kconfig b/drivers/net/phy/Kconfig |
| index e0f724a..1f74ff2 100644 |
| --- a/drivers/net/phy/Kconfig |
| +++ b/drivers/net/phy/Kconfig |
| @@ -511,6 +511,12 @@ config MARVELL_10G_PHY |
| ---help--- |
| Support for the Marvell Alaska MV88X3310 and compatible PHYs. |
| |
| +config MAXLINEAR_GPHY |
| + tristate "Maxlinear Ethernet PHYs" |
| + help |
| + Support for the Maxlinear GPY115, GPY211, GPY212, GPY215, |
| + GPY241, GPY245 PHYs. |
| + |
| config MESON_GXL_PHY |
| tristate "Amlogic Meson GXL Internal PHY" |
| depends on ARCH_MESON || COMPILE_TEST |
| diff --git a/drivers/net/phy/Makefile b/drivers/net/phy/Makefile |
| index e3c411f..7b44a98 100644 |
| --- a/drivers/net/phy/Makefile |
| +++ b/drivers/net/phy/Makefile |
| @@ -94,6 +94,7 @@ obj-$(CONFIG_LSI_ET1011C_PHY) += et1011c.o |
| obj-$(CONFIG_LXT_PHY) += lxt.o |
| obj-$(CONFIG_MARVELL_PHY) += marvell.o |
| obj-$(CONFIG_MARVELL_10G_PHY) += marvell10g.o |
| +obj-$(CONFIG_MAXLINEAR_GPHY) += mxl-gpy.o |
| obj-$(CONFIG_MEDIATEK_GE_PHY) += mediatek-ge.o |
| obj-$(CONFIG_MESON_GXL_PHY) += meson-gxl.o |
| obj-$(CONFIG_MICREL_KS8995MA) += spi_ks8995.o |
| diff --git a/drivers/net/phy/mxl-gpy.c b/drivers/net/phy/mxl-gpy.c |
| new file mode 100644 |
| index 0000000..7304278 |
| --- /dev/null |
| +++ b/drivers/net/phy/mxl-gpy.c |
| @@ -0,0 +1,738 @@ |
| +// SPDX-License-Identifier: GPL-2.0+ |
| +/* Copyright (C) 2021 Maxlinear Corporation |
| + * Copyright (C) 2020 Intel Corporation |
| + * |
| + * Drivers for Maxlinear Ethernet GPY |
| + * |
| + */ |
| + |
| +#include <linux/module.h> |
| +#include <linux/bitfield.h> |
| +#include <linux/phy.h> |
| +#include <linux/netdevice.h> |
| + |
| +/* PHY ID */ |
| +#define PHY_ID_GPYx15B_MASK 0xFFFFFFFC |
| +#define PHY_ID_GPY21xB_MASK 0xFFFFFFF9 |
| +#define PHY_ID_GPY2xx 0x67C9DC00 |
| +#define PHY_ID_GPY115B 0x67C9DF00 |
| +#define PHY_ID_GPY115C 0x67C9DF10 |
| +#define PHY_ID_GPY211B 0x67C9DE08 |
| +#define PHY_ID_GPY211C 0x67C9DE10 |
| +#define PHY_ID_GPY212B 0x67C9DE09 |
| +#define PHY_ID_GPY212C 0x67C9DE20 |
| +#define PHY_ID_GPY215B 0x67C9DF04 |
| +#define PHY_ID_GPY215C 0x67C9DF20 |
| +#define PHY_ID_GPY241B 0x67C9DE40 |
| +#define PHY_ID_GPY241BM 0x67C9DE80 |
| +#define PHY_ID_GPY245B 0x67C9DEC0 |
| + |
| +#define PHY_MIISTAT 0x18 /* MII state */ |
| +#define PHY_IMASK 0x19 /* interrupt mask */ |
| +#define PHY_ISTAT 0x1A /* interrupt status */ |
| +#define PHY_FWV 0x1E /* firmware version */ |
| + |
| +#define PHY_MIISTAT_SPD_MASK GENMASK(2, 0) |
| +#define PHY_MIISTAT_DPX BIT(3) |
| +#define PHY_MIISTAT_LS BIT(10) |
| + |
| +#define PHY_MIISTAT_SPD_10 0 |
| +#define PHY_MIISTAT_SPD_100 1 |
| +#define PHY_MIISTAT_SPD_1000 2 |
| +#define PHY_MIISTAT_SPD_2500 4 |
| + |
| +#define PHY_IMASK_WOL BIT(15) /* Wake-on-LAN */ |
| +#define PHY_IMASK_ANC BIT(10) /* Auto-Neg complete */ |
| +#define PHY_IMASK_ADSC BIT(5) /* Link auto-downspeed detect */ |
| +#define PHY_IMASK_DXMC BIT(2) /* Duplex mode change */ |
| +#define PHY_IMASK_LSPC BIT(1) /* Link speed change */ |
| +#define PHY_IMASK_LSTC BIT(0) /* Link state change */ |
| +#define PHY_IMASK_MASK (PHY_IMASK_LSTC | \ |
| + PHY_IMASK_LSPC | \ |
| + PHY_IMASK_DXMC | \ |
| + PHY_IMASK_ADSC | \ |
| + PHY_IMASK_ANC) |
| + |
| +#define PHY_FWV_REL_MASK BIT(15) |
| +#define PHY_FWV_TYPE_MASK GENMASK(11, 8) |
| +#define PHY_FWV_MINOR_MASK GENMASK(7, 0) |
| + |
| +/* SGMII */ |
| +#define VSPEC1_SGMII_CTRL 0x08 |
| +#define VSPEC1_SGMII_CTRL_ANEN BIT(12) /* Aneg enable */ |
| +#define VSPEC1_SGMII_CTRL_ANRS BIT(9) /* Restart Aneg */ |
| +#define VSPEC1_SGMII_ANEN_ANRS (VSPEC1_SGMII_CTRL_ANEN | \ |
| + VSPEC1_SGMII_CTRL_ANRS) |
| + |
| +/* WoL */ |
| +#define VPSPEC2_WOL_CTL 0x0E06 |
| +#define VPSPEC2_WOL_AD01 0x0E08 |
| +#define VPSPEC2_WOL_AD23 0x0E09 |
| +#define VPSPEC2_WOL_AD45 0x0E0A |
| +#define WOL_EN BIT(0) |
| + |
| +static const struct { |
| + int type; |
| + int minor; |
| +} ver_need_sgmii_reaneg[] = { |
| + {7, 0x6D}, |
| + {8, 0x6D}, |
| + {9, 0x73}, |
| +}; |
| + |
| +static int gpy_config_init(struct phy_device *phydev) |
| +{ |
| + int ret; |
| + |
| + /* Mask all interrupts */ |
| + ret = phy_write(phydev, PHY_IMASK, 0); |
| + if (ret) |
| + return ret; |
| + |
| + /* Clear all pending interrupts */ |
| + ret = phy_read(phydev, PHY_ISTAT); |
| + return ret < 0 ? ret : 0; |
| +} |
| + |
| +static int gpy_probe(struct phy_device *phydev) |
| +{ |
| + int ret; |
| + |
| + /* Show GPY PHY FW version in dmesg */ |
| + ret = phy_read(phydev, PHY_FWV); |
| + if (ret < 0) |
| + return ret; |
| + |
| + phydev_info(phydev, "Firmware Version: 0x%04X (%s)\n", ret, |
| + (ret & PHY_FWV_REL_MASK) ? "release" : "test"); |
| + |
| + return 0; |
| +} |
| + |
| +static bool gpy_sgmii_need_reaneg(struct phy_device *phydev) |
| +{ |
| + int fw_ver, fw_type, fw_minor; |
| + size_t i; |
| + |
| + fw_ver = phy_read(phydev, PHY_FWV); |
| + if (fw_ver < 0) |
| + return true; |
| + |
| + fw_type = FIELD_GET(PHY_FWV_TYPE_MASK, fw_ver); |
| + fw_minor = FIELD_GET(PHY_FWV_MINOR_MASK, fw_ver); |
| + |
| + for (i = 0; i < ARRAY_SIZE(ver_need_sgmii_reaneg); i++) { |
| + if (fw_type != ver_need_sgmii_reaneg[i].type) |
| + continue; |
| + if (fw_minor < ver_need_sgmii_reaneg[i].minor) |
| + return true; |
| + break; |
| + } |
| + |
| + return false; |
| +} |
| + |
| +static bool gpy_2500basex_chk(struct phy_device *phydev) |
| +{ |
| + int ret; |
| + |
| + ret = phy_read(phydev, PHY_MIISTAT); |
| + if (ret < 0) { |
| + phydev_err(phydev, "Error: MDIO register access failed: %d\n", |
| + ret); |
| + return false; |
| + } |
| + |
| + if (!(ret & PHY_MIISTAT_LS) || |
| + FIELD_GET(PHY_MIISTAT_SPD_MASK, ret) != PHY_MIISTAT_SPD_2500) |
| + return false; |
| + |
| + phydev->speed = SPEED_2500; |
| + phydev->interface = PHY_INTERFACE_MODE_2500BASEX; |
| + phy_modify_mmd(phydev, MDIO_MMD_VEND1, VSPEC1_SGMII_CTRL, |
| + VSPEC1_SGMII_CTRL_ANEN, 0); |
| + return true; |
| +} |
| + |
| +static bool gpy_sgmii_aneg_en(struct phy_device *phydev) |
| +{ |
| + int ret; |
| + |
| + ret = phy_read_mmd(phydev, MDIO_MMD_VEND1, VSPEC1_SGMII_CTRL); |
| + if (ret < 0) { |
| + phydev_err(phydev, "Error: MMD register access failed: %d\n", |
| + ret); |
| + return true; |
| + } |
| + |
| + return (ret & VSPEC1_SGMII_CTRL_ANEN) ? true : false; |
| +} |
| + |
| +static int gpy_config_aneg(struct phy_device *phydev) |
| +{ |
| + bool changed = false; |
| + u32 adv; |
| + int ret; |
| + |
| + if (phydev->autoneg == AUTONEG_DISABLE) { |
| + /* Configure half duplex with genphy_setup_forced, |
| + * because genphy_c45_pma_setup_forced does not support. |
| + */ |
| + return phydev->duplex != DUPLEX_FULL |
| + ? genphy_setup_forced(phydev) |
| + : genphy_c45_pma_setup_forced(phydev); |
| + } |
| + |
| + ret = genphy_c45_an_config_aneg(phydev); |
| + if (ret < 0) |
| + return ret; |
| + if (ret > 0) |
| + changed = true; |
| + |
| + adv = linkmode_adv_to_mii_ctrl1000_t(phydev->advertising); |
| + ret = phy_modify_changed(phydev, MII_CTRL1000, |
| + ADVERTISE_1000FULL | ADVERTISE_1000HALF, |
| + adv); |
| + if (ret < 0) |
| + return ret; |
| + if (ret > 0) |
| + changed = true; |
| + |
| + ret = genphy_c45_check_and_restart_aneg(phydev, changed); |
| + if (ret < 0) |
| + return ret; |
| + |
| + if (phydev->interface == PHY_INTERFACE_MODE_USXGMII || |
| + phydev->interface == PHY_INTERFACE_MODE_INTERNAL) |
| + return 0; |
| + |
| + /* No need to trigger re-ANEG if link speed is 2.5G or SGMII ANEG is |
| + * disabled. |
| + */ |
| + if (!gpy_sgmii_need_reaneg(phydev) || gpy_2500basex_chk(phydev) || |
| + !gpy_sgmii_aneg_en(phydev)) |
| + return 0; |
| + |
| + /* There is a design constraint in GPY2xx device where SGMII AN is |
| + * only triggered when there is change of speed. If, PHY link |
| + * partner`s speed is still same even after PHY TPI is down and up |
| + * again, SGMII AN is not triggered and hence no new in-band message |
| + * from GPY to MAC side SGMII. |
| + * This could cause an issue during power up, when PHY is up prior to |
| + * MAC. At this condition, once MAC side SGMII is up, MAC side SGMII |
| + * wouldn`t receive new in-band message from GPY with correct link |
| + * status, speed and duplex info. |
| + * |
| + * 1) If PHY is already up and TPI link status is still down (such as |
| + * hard reboot), TPI link status is polled for 4 seconds before |
| + * retriggerring SGMII AN. |
| + * 2) If PHY is already up and TPI link status is also up (such as soft |
| + * reboot), polling of TPI link status is not needed and SGMII AN is |
| + * immediately retriggered. |
| + * 3) Other conditions such as PHY is down, speed change etc, skip |
| + * retriggering SGMII AN. Note: in case of speed change, GPY FW will |
| + * initiate SGMII AN. |
| + */ |
| + |
| + if (phydev->state != PHY_UP) |
| + return 0; |
| + |
| + ret = phy_read_poll_timeout(phydev, MII_BMSR, ret, ret & BMSR_LSTATUS, |
| + 20000, 4000000, false); |
| + if (ret == -ETIMEDOUT) |
| + return 0; |
| + else if (ret < 0) |
| + return ret; |
| + |
| + /* Trigger SGMII AN. */ |
| + return phy_modify_mmd(phydev, MDIO_MMD_VEND1, VSPEC1_SGMII_CTRL, |
| + VSPEC1_SGMII_CTRL_ANRS, VSPEC1_SGMII_CTRL_ANRS); |
| +} |
| + |
| +static void gpy_update_interface(struct phy_device *phydev) |
| +{ |
| + int ret; |
| + |
| + /* Interface mode is fixed for USXGMII and integrated PHY */ |
| + if (phydev->interface == PHY_INTERFACE_MODE_USXGMII || |
| + phydev->interface == PHY_INTERFACE_MODE_INTERNAL) |
| + return; |
| + |
| + /* Automatically switch SERDES interface between SGMII and 2500-BaseX |
| + * according to speed. Disable ANEG in 2500-BaseX mode. |
| + */ |
| + switch (phydev->speed) { |
| + case SPEED_2500: |
| + phydev->interface = PHY_INTERFACE_MODE_2500BASEX; |
| + ret = phy_modify_mmd(phydev, MDIO_MMD_VEND1, VSPEC1_SGMII_CTRL, |
| + VSPEC1_SGMII_CTRL_ANEN, 0); |
| + if (ret < 0) |
| + phydev_err(phydev, |
| + "Error: Disable of SGMII ANEG failed: %d\n", |
| + ret); |
| + break; |
| + case SPEED_1000: |
| + case SPEED_100: |
| + case SPEED_10: |
| + phydev->interface = PHY_INTERFACE_MODE_SGMII; |
| + if (gpy_sgmii_aneg_en(phydev)) |
| + break; |
| + /* Enable and restart SGMII ANEG for 10/100/1000Mbps link speed |
| + * if ANEG is disabled (in 2500-BaseX mode). |
| + */ |
| + ret = phy_modify_mmd(phydev, MDIO_MMD_VEND1, VSPEC1_SGMII_CTRL, |
| + VSPEC1_SGMII_ANEN_ANRS, |
| + VSPEC1_SGMII_ANEN_ANRS); |
| + if (ret < 0) |
| + phydev_err(phydev, |
| + "Error: Enable of SGMII ANEG failed: %d\n", |
| + ret); |
| + break; |
| + } |
| +} |
| + |
| +static int gpy_read_status(struct phy_device *phydev) |
| +{ |
| + int ret; |
| + |
| + ret = genphy_update_link(phydev); |
| + if (ret) |
| + return ret; |
| + |
| + phydev->speed = SPEED_UNKNOWN; |
| + phydev->duplex = DUPLEX_UNKNOWN; |
| + phydev->pause = 0; |
| + phydev->asym_pause = 0; |
| + |
| + if (phydev->autoneg == AUTONEG_ENABLE && phydev->autoneg_complete) { |
| + ret = genphy_c45_read_lpa(phydev); |
| + if (ret < 0) |
| + return ret; |
| + |
| + /* Read the link partner's 1G advertisement */ |
| + ret = phy_read(phydev, MII_STAT1000); |
| + if (ret < 0) |
| + return ret; |
| + mii_stat1000_mod_linkmode_lpa_t(phydev->lp_advertising, ret); |
| + } else if (phydev->autoneg == AUTONEG_DISABLE) { |
| + linkmode_zero(phydev->lp_advertising); |
| + } |
| + |
| + ret = phy_read(phydev, PHY_MIISTAT); |
| + if (ret < 0) |
| + return ret; |
| + |
| + phydev->link = (ret & PHY_MIISTAT_LS) ? 1 : 0; |
| + phydev->duplex = (ret & PHY_MIISTAT_DPX) ? DUPLEX_FULL : DUPLEX_HALF; |
| + switch (FIELD_GET(PHY_MIISTAT_SPD_MASK, ret)) { |
| + case PHY_MIISTAT_SPD_10: |
| + phydev->speed = SPEED_10; |
| + break; |
| + case PHY_MIISTAT_SPD_100: |
| + phydev->speed = SPEED_100; |
| + break; |
| + case PHY_MIISTAT_SPD_1000: |
| + phydev->speed = SPEED_1000; |
| + break; |
| + case PHY_MIISTAT_SPD_2500: |
| + phydev->speed = SPEED_2500; |
| + break; |
| + } |
| + |
| + if (phydev->link) |
| + gpy_update_interface(phydev); |
| + |
| + return 0; |
| +} |
| + |
| +static int gpy_config_intr(struct phy_device *phydev) |
| +{ |
| + u16 mask = 0; |
| + |
| + if (phydev->interrupts == PHY_INTERRUPT_ENABLED) |
| + mask = PHY_IMASK_MASK; |
| + |
| + return phy_write(phydev, PHY_IMASK, mask); |
| +} |
| + |
| +static int gpy_handle_interrupt(struct phy_device *phydev) |
| +{ |
| + int reg; |
| + |
| + reg = phy_read(phydev, PHY_ISTAT); |
| + if (reg < 0) |
| + return -1; |
| + |
| + if (!(reg & PHY_IMASK_MASK)) |
| + return -1; |
| + |
| + phy_queue_state_machine(phydev, 0); |
| + |
| + return 0; |
| +} |
| + |
| +static int gpy_set_wol(struct phy_device *phydev, |
| + struct ethtool_wolinfo *wol) |
| +{ |
| + struct net_device *attach_dev = phydev->attached_dev; |
| + int ret; |
| + |
| + if (wol->wolopts & WAKE_MAGIC) { |
| + /* MAC address - Byte0:Byte1:Byte2:Byte3:Byte4:Byte5 |
| + * VPSPEC2_WOL_AD45 = Byte0:Byte1 |
| + * VPSPEC2_WOL_AD23 = Byte2:Byte3 |
| + * VPSPEC2_WOL_AD01 = Byte4:Byte5 |
| + */ |
| + ret = phy_set_bits_mmd(phydev, MDIO_MMD_VEND2, |
| + VPSPEC2_WOL_AD45, |
| + ((attach_dev->dev_addr[0] << 8) | |
| + attach_dev->dev_addr[1])); |
| + if (ret < 0) |
| + return ret; |
| + |
| + ret = phy_set_bits_mmd(phydev, MDIO_MMD_VEND2, |
| + VPSPEC2_WOL_AD23, |
| + ((attach_dev->dev_addr[2] << 8) | |
| + attach_dev->dev_addr[3])); |
| + if (ret < 0) |
| + return ret; |
| + |
| + ret = phy_set_bits_mmd(phydev, MDIO_MMD_VEND2, |
| + VPSPEC2_WOL_AD01, |
| + ((attach_dev->dev_addr[4] << 8) | |
| + attach_dev->dev_addr[5])); |
| + if (ret < 0) |
| + return ret; |
| + |
| + /* Enable the WOL interrupt */ |
| + ret = phy_write(phydev, PHY_IMASK, PHY_IMASK_WOL); |
| + if (ret < 0) |
| + return ret; |
| + |
| + /* Enable magic packet matching */ |
| + ret = phy_set_bits_mmd(phydev, MDIO_MMD_VEND2, |
| + VPSPEC2_WOL_CTL, |
| + WOL_EN); |
| + if (ret < 0) |
| + return ret; |
| + |
| + /* Clear the interrupt status register. |
| + * Only WoL is enabled so clear all. |
| + */ |
| + ret = phy_read(phydev, PHY_ISTAT); |
| + if (ret < 0) |
| + return ret; |
| + } else { |
| + /* Disable magic packet matching */ |
| + ret = phy_clear_bits_mmd(phydev, MDIO_MMD_VEND2, |
| + VPSPEC2_WOL_CTL, |
| + WOL_EN); |
| + if (ret < 0) |
| + return ret; |
| + } |
| + |
| + if (wol->wolopts & WAKE_PHY) { |
| + /* Enable the link state change interrupt */ |
| + ret = phy_set_bits(phydev, PHY_IMASK, PHY_IMASK_LSTC); |
| + if (ret < 0) |
| + return ret; |
| + |
| + /* Clear the interrupt status register */ |
| + ret = phy_read(phydev, PHY_ISTAT); |
| + if (ret < 0) |
| + return ret; |
| + |
| + if (ret & (PHY_IMASK_MASK & ~PHY_IMASK_LSTC)) |
| + phy_queue_state_machine(phydev, 0); |
| + |
| + return 0; |
| + } |
| + |
| + /* Disable the link state change interrupt */ |
| + return phy_clear_bits(phydev, PHY_IMASK, PHY_IMASK_LSTC); |
| +} |
| + |
| +static void gpy_get_wol(struct phy_device *phydev, |
| + struct ethtool_wolinfo *wol) |
| +{ |
| + int ret; |
| + |
| + wol->supported = WAKE_MAGIC | WAKE_PHY; |
| + wol->wolopts = 0; |
| + |
| + ret = phy_read_mmd(phydev, MDIO_MMD_VEND2, VPSPEC2_WOL_CTL); |
| + if (ret & WOL_EN) |
| + wol->wolopts |= WAKE_MAGIC; |
| + |
| + ret = phy_read(phydev, PHY_IMASK); |
| + if (ret & PHY_IMASK_LSTC) |
| + wol->wolopts |= WAKE_PHY; |
| +} |
| + |
| +static int gpy_loopback(struct phy_device *phydev, bool enable) |
| +{ |
| + int ret; |
| + |
| + ret = phy_modify(phydev, MII_BMCR, BMCR_LOOPBACK, |
| + enable ? BMCR_LOOPBACK : 0); |
| + if (!ret) { |
| + /* It takes some time for PHY device to switch |
| + * into/out-of loopback mode. |
| + */ |
| + msleep(100); |
| + } |
| + |
| + return ret; |
| +} |
| + |
| +static int gpy115_loopback(struct phy_device *phydev, bool enable) |
| +{ |
| + int ret; |
| + int fw_minor; |
| + |
| + if (enable) |
| + return gpy_loopback(phydev, enable); |
| + |
| + ret = phy_read(phydev, PHY_FWV); |
| + if (ret < 0) |
| + return ret; |
| + |
| + fw_minor = FIELD_GET(PHY_FWV_MINOR_MASK, ret); |
| + if (fw_minor > 0x0076) |
| + return gpy_loopback(phydev, 0); |
| + |
| + return genphy_soft_reset(phydev); |
| +} |
| + |
| +static struct phy_driver gpy_drivers[] = { |
| + { |
| + PHY_ID_MATCH_MODEL(PHY_ID_GPY2xx), |
| + .name = "Maxlinear Ethernet GPY2xx", |
| + .get_features = genphy_c45_pma_read_abilities, |
| + .config_init = gpy_config_init, |
| + .probe = gpy_probe, |
| + .suspend = genphy_suspend, |
| + .resume = genphy_resume, |
| + .config_aneg = gpy_config_aneg, |
| + .aneg_done = genphy_c45_aneg_done, |
| + .read_status = gpy_read_status, |
| + .config_intr = gpy_config_intr, |
| + .handle_interrupt = gpy_handle_interrupt, |
| + .set_wol = gpy_set_wol, |
| + .get_wol = gpy_get_wol, |
| + .set_loopback = gpy_loopback, |
| + }, |
| + { |
| + .phy_id = PHY_ID_GPY115B, |
| + .phy_id_mask = PHY_ID_GPYx15B_MASK, |
| + .name = "Maxlinear Ethernet GPY115B", |
| + .get_features = genphy_c45_pma_read_abilities, |
| + .config_init = gpy_config_init, |
| + .probe = gpy_probe, |
| + .suspend = genphy_suspend, |
| + .resume = genphy_resume, |
| + .config_aneg = gpy_config_aneg, |
| + .aneg_done = genphy_c45_aneg_done, |
| + .read_status = gpy_read_status, |
| + .config_intr = gpy_config_intr, |
| + .handle_interrupt = gpy_handle_interrupt, |
| + .set_wol = gpy_set_wol, |
| + .get_wol = gpy_get_wol, |
| + .set_loopback = gpy115_loopback, |
| + }, |
| + { |
| + PHY_ID_MATCH_MODEL(PHY_ID_GPY115C), |
| + .name = "Maxlinear Ethernet GPY115C", |
| + .get_features = genphy_c45_pma_read_abilities, |
| + .config_init = gpy_config_init, |
| + .probe = gpy_probe, |
| + .suspend = genphy_suspend, |
| + .resume = genphy_resume, |
| + .config_aneg = gpy_config_aneg, |
| + .aneg_done = genphy_c45_aneg_done, |
| + .read_status = gpy_read_status, |
| + .config_intr = gpy_config_intr, |
| + .handle_interrupt = gpy_handle_interrupt, |
| + .set_wol = gpy_set_wol, |
| + .get_wol = gpy_get_wol, |
| + .set_loopback = gpy115_loopback, |
| + }, |
| + { |
| + .phy_id = PHY_ID_GPY211B, |
| + .phy_id_mask = PHY_ID_GPY21xB_MASK, |
| + .name = "Maxlinear Ethernet GPY211B", |
| + .get_features = genphy_c45_pma_read_abilities, |
| + .config_init = gpy_config_init, |
| + .probe = gpy_probe, |
| + .suspend = genphy_suspend, |
| + .resume = genphy_resume, |
| + .config_aneg = gpy_config_aneg, |
| + .aneg_done = genphy_c45_aneg_done, |
| + .read_status = gpy_read_status, |
| + .config_intr = gpy_config_intr, |
| + .handle_interrupt = gpy_handle_interrupt, |
| + .set_wol = gpy_set_wol, |
| + .get_wol = gpy_get_wol, |
| + .set_loopback = gpy_loopback, |
| + }, |
| + { |
| + PHY_ID_MATCH_MODEL(PHY_ID_GPY211C), |
| + .name = "Maxlinear Ethernet GPY211C", |
| + .get_features = genphy_c45_pma_read_abilities, |
| + .config_init = gpy_config_init, |
| + .probe = gpy_probe, |
| + .suspend = genphy_suspend, |
| + .resume = genphy_resume, |
| + .config_aneg = gpy_config_aneg, |
| + .aneg_done = genphy_c45_aneg_done, |
| + .read_status = gpy_read_status, |
| + .config_intr = gpy_config_intr, |
| + .handle_interrupt = gpy_handle_interrupt, |
| + .set_wol = gpy_set_wol, |
| + .get_wol = gpy_get_wol, |
| + .set_loopback = gpy_loopback, |
| + }, |
| + { |
| + .phy_id = PHY_ID_GPY212B, |
| + .phy_id_mask = PHY_ID_GPY21xB_MASK, |
| + .name = "Maxlinear Ethernet GPY212B", |
| + .get_features = genphy_c45_pma_read_abilities, |
| + .config_init = gpy_config_init, |
| + .probe = gpy_probe, |
| + .suspend = genphy_suspend, |
| + .resume = genphy_resume, |
| + .config_aneg = gpy_config_aneg, |
| + .aneg_done = genphy_c45_aneg_done, |
| + .read_status = gpy_read_status, |
| + .config_intr = gpy_config_intr, |
| + .handle_interrupt = gpy_handle_interrupt, |
| + .set_wol = gpy_set_wol, |
| + .get_wol = gpy_get_wol, |
| + .set_loopback = gpy_loopback, |
| + }, |
| + { |
| + PHY_ID_MATCH_MODEL(PHY_ID_GPY212C), |
| + .name = "Maxlinear Ethernet GPY212C", |
| + .get_features = genphy_c45_pma_read_abilities, |
| + .config_init = gpy_config_init, |
| + .probe = gpy_probe, |
| + .suspend = genphy_suspend, |
| + .resume = genphy_resume, |
| + .config_aneg = gpy_config_aneg, |
| + .aneg_done = genphy_c45_aneg_done, |
| + .read_status = gpy_read_status, |
| + .config_intr = gpy_config_intr, |
| + .handle_interrupt = gpy_handle_interrupt, |
| + .set_wol = gpy_set_wol, |
| + .get_wol = gpy_get_wol, |
| + .set_loopback = gpy_loopback, |
| + }, |
| + { |
| + .phy_id = PHY_ID_GPY215B, |
| + .phy_id_mask = PHY_ID_GPYx15B_MASK, |
| + .name = "Maxlinear Ethernet GPY215B", |
| + .get_features = genphy_c45_pma_read_abilities, |
| + .config_init = gpy_config_init, |
| + .probe = gpy_probe, |
| + .suspend = genphy_suspend, |
| + .resume = genphy_resume, |
| + .config_aneg = gpy_config_aneg, |
| + .aneg_done = genphy_c45_aneg_done, |
| + .read_status = gpy_read_status, |
| + .config_intr = gpy_config_intr, |
| + .handle_interrupt = gpy_handle_interrupt, |
| + .set_wol = gpy_set_wol, |
| + .get_wol = gpy_get_wol, |
| + .set_loopback = gpy_loopback, |
| + }, |
| + { |
| + PHY_ID_MATCH_MODEL(PHY_ID_GPY215C), |
| + .name = "Maxlinear Ethernet GPY215C", |
| + .get_features = genphy_c45_pma_read_abilities, |
| + .config_init = gpy_config_init, |
| + .probe = gpy_probe, |
| + .suspend = genphy_suspend, |
| + .resume = genphy_resume, |
| + .config_aneg = gpy_config_aneg, |
| + .aneg_done = genphy_c45_aneg_done, |
| + .read_status = gpy_read_status, |
| + .config_intr = gpy_config_intr, |
| + .handle_interrupt = gpy_handle_interrupt, |
| + .set_wol = gpy_set_wol, |
| + .get_wol = gpy_get_wol, |
| + .set_loopback = gpy_loopback, |
| + }, |
| + { |
| + PHY_ID_MATCH_MODEL(PHY_ID_GPY241B), |
| + .name = "Maxlinear Ethernet GPY241B", |
| + .get_features = genphy_c45_pma_read_abilities, |
| + .config_init = gpy_config_init, |
| + .probe = gpy_probe, |
| + .suspend = genphy_suspend, |
| + .resume = genphy_resume, |
| + .config_aneg = gpy_config_aneg, |
| + .aneg_done = genphy_c45_aneg_done, |
| + .read_status = gpy_read_status, |
| + .config_intr = gpy_config_intr, |
| + .handle_interrupt = gpy_handle_interrupt, |
| + .set_wol = gpy_set_wol, |
| + .get_wol = gpy_get_wol, |
| + .set_loopback = gpy_loopback, |
| + }, |
| + { |
| + PHY_ID_MATCH_MODEL(PHY_ID_GPY241BM), |
| + .name = "Maxlinear Ethernet GPY241BM", |
| + .get_features = genphy_c45_pma_read_abilities, |
| + .config_init = gpy_config_init, |
| + .probe = gpy_probe, |
| + .suspend = genphy_suspend, |
| + .resume = genphy_resume, |
| + .config_aneg = gpy_config_aneg, |
| + .aneg_done = genphy_c45_aneg_done, |
| + .read_status = gpy_read_status, |
| + .config_intr = gpy_config_intr, |
| + .handle_interrupt = gpy_handle_interrupt, |
| + .set_wol = gpy_set_wol, |
| + .get_wol = gpy_get_wol, |
| + .set_loopback = gpy_loopback, |
| + }, |
| + { |
| + PHY_ID_MATCH_MODEL(PHY_ID_GPY245B), |
| + .name = "Maxlinear Ethernet GPY245B", |
| + .get_features = genphy_c45_pma_read_abilities, |
| + .config_init = gpy_config_init, |
| + .probe = gpy_probe, |
| + .suspend = genphy_suspend, |
| + .resume = genphy_resume, |
| + .config_aneg = gpy_config_aneg, |
| + .aneg_done = genphy_c45_aneg_done, |
| + .read_status = gpy_read_status, |
| + .config_intr = gpy_config_intr, |
| + .handle_interrupt = gpy_handle_interrupt, |
| + .set_wol = gpy_set_wol, |
| + .get_wol = gpy_get_wol, |
| + .set_loopback = gpy_loopback, |
| + }, |
| +}; |
| +module_phy_driver(gpy_drivers); |
| + |
| +static struct mdio_device_id __maybe_unused gpy_tbl[] = { |
| + {PHY_ID_MATCH_MODEL(PHY_ID_GPY2xx)}, |
| + {PHY_ID_GPY115B, PHY_ID_GPYx15B_MASK}, |
| + {PHY_ID_MATCH_MODEL(PHY_ID_GPY115C)}, |
| + {PHY_ID_GPY211B, PHY_ID_GPY21xB_MASK}, |
| + {PHY_ID_MATCH_MODEL(PHY_ID_GPY211C)}, |
| + {PHY_ID_GPY212B, PHY_ID_GPY21xB_MASK}, |
| + {PHY_ID_MATCH_MODEL(PHY_ID_GPY212C)}, |
| + {PHY_ID_GPY215B, PHY_ID_GPYx15B_MASK}, |
| + {PHY_ID_MATCH_MODEL(PHY_ID_GPY215C)}, |
| + {PHY_ID_MATCH_MODEL(PHY_ID_GPY241B)}, |
| + {PHY_ID_MATCH_MODEL(PHY_ID_GPY241BM)}, |
| + {PHY_ID_MATCH_MODEL(PHY_ID_GPY245B)}, |
| + { } |
| +}; |
| +MODULE_DEVICE_TABLE(mdio, gpy_tbl); |
| + |
| +MODULE_DESCRIPTION("Maxlinear Ethernet GPY Driver"); |
| +MODULE_AUTHOR("Xu Liang"); |
| +MODULE_LICENSE("GPL"); |
| diff --git a/include/linux/iopoll.h b/include/linux/iopoll.h |
| index 35e15df..2c8860e 100644 |
| --- a/include/linux/iopoll.h |
| +++ b/include/linux/iopoll.h |
| @@ -14,36 +14,41 @@ |
| #include <linux/io.h> |
| |
| /** |
| - * readx_poll_timeout - Periodically poll an address until a condition is met or a timeout occurs |
| - * @op: accessor function (takes @addr as its only argument) |
| - * @addr: Address to poll |
| + * read_poll_timeout - Periodically poll an address until a condition is |
| + * met or a timeout occurs |
| + * @op: accessor function (takes @args as its arguments) |
| * @val: Variable to read the value into |
| * @cond: Break condition (usually involving @val) |
| * @sleep_us: Maximum time to sleep between reads in us (0 |
| * tight-loops). Should be less than ~20ms since usleep_range |
| * is used (see Documentation/timers/timers-howto.rst). |
| * @timeout_us: Timeout in us, 0 means never timeout |
| + * @sleep_before_read: if it is true, sleep @sleep_us before read. |
| + * @args: arguments for @op poll |
| * |
| * Returns 0 on success and -ETIMEDOUT upon a timeout. In either |
| - * case, the last read value at @addr is stored in @val. Must not |
| + * case, the last read value at @args is stored in @val. Must not |
| * be called from atomic context if sleep_us or timeout_us are used. |
| * |
| * When available, you'll probably want to use one of the specialized |
| * macros defined below rather than this macro directly. |
| */ |
| -#define readx_poll_timeout(op, addr, val, cond, sleep_us, timeout_us) \ |
| +#define read_poll_timeout(op, val, cond, sleep_us, timeout_us, \ |
| + sleep_before_read, args...) \ |
| ({ \ |
| u64 __timeout_us = (timeout_us); \ |
| unsigned long __sleep_us = (sleep_us); \ |
| ktime_t __timeout = ktime_add_us(ktime_get(), __timeout_us); \ |
| might_sleep_if((__sleep_us) != 0); \ |
| + if (sleep_before_read && __sleep_us) \ |
| + usleep_range((__sleep_us >> 2) + 1, __sleep_us); \ |
| for (;;) { \ |
| - (val) = op(addr); \ |
| + (val) = op(args); \ |
| if (cond) \ |
| break; \ |
| if (__timeout_us && \ |
| ktime_compare(ktime_get(), __timeout) > 0) { \ |
| - (val) = op(addr); \ |
| + (val) = op(args); \ |
| break; \ |
| } \ |
| if (__sleep_us) \ |
| @@ -53,42 +58,87 @@ |
| }) |
| |
| /** |
| - * readx_poll_timeout_atomic - Periodically poll an address until a condition is met or a timeout occurs |
| - * @op: accessor function (takes @addr as its only argument) |
| - * @addr: Address to poll |
| + * read_poll_timeout_atomic - Periodically poll an address until a condition is |
| + * met or a timeout occurs |
| + * @op: accessor function (takes @args as its arguments) |
| * @val: Variable to read the value into |
| * @cond: Break condition (usually involving @val) |
| * @delay_us: Time to udelay between reads in us (0 tight-loops). Should |
| * be less than ~10us since udelay is used (see |
| * Documentation/timers/timers-howto.rst). |
| * @timeout_us: Timeout in us, 0 means never timeout |
| + * @delay_before_read: if it is true, delay @delay_us before read. |
| + * @args: arguments for @op poll |
| * |
| * Returns 0 on success and -ETIMEDOUT upon a timeout. In either |
| - * case, the last read value at @addr is stored in @val. |
| + * case, the last read value at @args is stored in @val. |
| * |
| * When available, you'll probably want to use one of the specialized |
| * macros defined below rather than this macro directly. |
| */ |
| -#define readx_poll_timeout_atomic(op, addr, val, cond, delay_us, timeout_us) \ |
| +#define read_poll_timeout_atomic(op, val, cond, delay_us, timeout_us, \ |
| + delay_before_read, args...) \ |
| ({ \ |
| u64 __timeout_us = (timeout_us); \ |
| unsigned long __delay_us = (delay_us); \ |
| ktime_t __timeout = ktime_add_us(ktime_get(), __timeout_us); \ |
| + if (delay_before_read && __delay_us) \ |
| + udelay(__delay_us); \ |
| for (;;) { \ |
| - (val) = op(addr); \ |
| + (val) = op(args); \ |
| if (cond) \ |
| break; \ |
| if (__timeout_us && \ |
| ktime_compare(ktime_get(), __timeout) > 0) { \ |
| - (val) = op(addr); \ |
| + (val) = op(args); \ |
| break; \ |
| } \ |
| if (__delay_us) \ |
| - udelay(__delay_us); \ |
| + udelay(__delay_us); \ |
| } \ |
| (cond) ? 0 : -ETIMEDOUT; \ |
| }) |
| |
| +/** |
| + * readx_poll_timeout - Periodically poll an address until a condition is met or a timeout occurs |
| + * @op: accessor function (takes @addr as its only argument) |
| + * @addr: Address to poll |
| + * @val: Variable to read the value into |
| + * @cond: Break condition (usually involving @val) |
| + * @sleep_us: Maximum time to sleep between reads in us (0 |
| + * tight-loops). Should be less than ~20ms since usleep_range |
| + * is used (see Documentation/timers/timers-howto.rst). |
| + * @timeout_us: Timeout in us, 0 means never timeout |
| + * |
| + * Returns 0 on success and -ETIMEDOUT upon a timeout. In either |
| + * case, the last read value at @addr is stored in @val. Must not |
| + * be called from atomic context if sleep_us or timeout_us are used. |
| + * |
| + * When available, you'll probably want to use one of the specialized |
| + * macros defined below rather than this macro directly. |
| + */ |
| +#define readx_poll_timeout(op, addr, val, cond, sleep_us, timeout_us) \ |
| + read_poll_timeout(op, val, cond, sleep_us, timeout_us, false, addr) |
| + |
| +/** |
| + * readx_poll_timeout_atomic - Periodically poll an address until a condition is met or a timeout occurs |
| + * @op: accessor function (takes @addr as its only argument) |
| + * @addr: Address to poll |
| + * @val: Variable to read the value into |
| + * @cond: Break condition (usually involving @val) |
| + * @delay_us: Time to udelay between reads in us (0 tight-loops). Should |
| + * be less than ~10us since udelay is used (see |
| + * Documentation/timers/timers-howto.rst). |
| + * @timeout_us: Timeout in us, 0 means never timeout |
| + * |
| + * Returns 0 on success and -ETIMEDOUT upon a timeout. In either |
| + * case, the last read value at @addr is stored in @val. |
| + * |
| + * When available, you'll probably want to use one of the specialized |
| + * macros defined below rather than this macro directly. |
| + */ |
| +#define readx_poll_timeout_atomic(op, addr, val, cond, delay_us, timeout_us) \ |
| + read_poll_timeout_atomic(op, val, cond, delay_us, timeout_us, false, addr) |
| |
| #define readb_poll_timeout(addr, val, cond, delay_us, timeout_us) \ |
| readx_poll_timeout(readb, addr, val, cond, delay_us, timeout_us) |
| diff --git a/include/linux/phy.h b/include/linux/phy.h |
| index 19444cd..34bdd16 100644 |
| --- a/include/linux/phy.h |
| +++ b/include/linux/phy.h |
| @@ -21,6 +21,7 @@ |
| #include <linux/timer.h> |
| #include <linux/workqueue.h> |
| #include <linux/mod_devicetable.h> |
| +#include <linux/iopoll.h> |
| |
| #include <linux/atomic.h> |
| |
| @@ -711,6 +712,18 @@ static inline int phy_read(struct phy_device *phydev, u32 regnum) |
| return mdiobus_read(phydev->mdio.bus, phydev->mdio.addr, regnum); |
| } |
| |
| +#define phy_read_poll_timeout(phydev, regnum, val, cond, sleep_us, \ |
| + timeout_us, sleep_before_read) \ |
| +({ \ |
| + int __ret = read_poll_timeout(phy_read, val, (cond) || val < 0, \ |
| + sleep_us, timeout_us, sleep_before_read, phydev, regnum); \ |
| + if (val < 0) \ |
| + __ret = val; \ |
| + if (__ret) \ |
| + phydev_err(phydev, "%s failed: %d\n", __func__, __ret); \ |
| + __ret; \ |
| +}) |
| + |
| /** |
| * __phy_read - convenience function for reading a given PHY register |
| * @phydev: the phy_device struct |