// SPDX-License-Identifier: GPL-2.0 /* Driver for the Texas Instruments DP83TG720 PHY * Copyright (c) 2023 Pengutronix, Oleksij Rempel */ #include #include #include #include #include #include "open_alliance_helpers.h" #define DP83TG720S_PHY_ID 0x2000a284 /* MDIO_MMD_VEND2 registers */ #define DP83TG720S_MII_REG_10 0x10 #define DP83TG720S_STS_MII_INT BIT(7) #define DP83TG720S_LINK_STATUS BIT(0) /* TDR Configuration Register (0x1E) */ #define DP83TG720S_TDR_CFG 0x1e /* 1b = TDR start, 0b = No TDR */ #define DP83TG720S_TDR_START BIT(15) /* 1b = TDR auto on link down, 0b = Manual TDR start */ #define DP83TG720S_CFG_TDR_AUTO_RUN BIT(14) /* 1b = TDR done, 0b = TDR in progress */ #define DP83TG720S_TDR_DONE BIT(1) /* 1b = TDR fail, 0b = TDR success */ #define DP83TG720S_TDR_FAIL BIT(0) #define DP83TG720S_PHY_RESET 0x1f #define DP83TG720S_HW_RESET BIT(15) #define DP83TG720S_LPS_CFG3 0x18c /* Power modes are documented as bit fields but used as values */ /* Power Mode 0 is Normal mode */ #define DP83TG720S_LPS_CFG3_PWR_MODE_0 BIT(0) /* Open Aliance 1000BaseT1 compatible HDD.TDR Fault Status Register */ #define DP83TG720S_TDR_FAULT_STATUS 0x30f /* Register 0x0301: TDR Configuration 2 */ #define DP83TG720S_TDR_CFG2 0x301 /* Register 0x0303: TDR Configuration 3 */ #define DP83TG720S_TDR_CFG3 0x303 /* Register 0x0304: TDR Configuration 4 */ #define DP83TG720S_TDR_CFG4 0x304 /* Register 0x0405: Unknown Register */ #define DP83TG720S_UNKNOWN_0405 0x405 #define DP83TG720S_LINK_QUAL_3 0x547 #define DP83TG720S_LINK_LOSS_CNT_MASK GENMASK(15, 10) /* Register 0x0576: TDR Master Link Down Control */ #define DP83TG720S_TDR_MASTER_LINK_DOWN 0x576 #define DP83TG720S_RGMII_DELAY_CTRL 0x602 /* In RGMII mode, Enable or disable the internal delay for RXD */ #define DP83TG720S_RGMII_RX_CLK_SEL BIT(1) /* In RGMII mode, Enable or disable the internal delay for TXD */ #define DP83TG720S_RGMII_TX_CLK_SEL BIT(0) /* * DP83TG720S_PKT_STAT_x registers correspond to similarly named registers * in the datasheet (PKT_STAT_1 through PKT_STAT_6). These registers store * 32-bit or 16-bit counters for TX and RX statistics and must be read in * sequence to ensure the counters are cleared correctly. * * - DP83TG720S_PKT_STAT_1: Contains TX packet count bits [15:0]. * - DP83TG720S_PKT_STAT_2: Contains TX packet count bits [31:16]. * - DP83TG720S_PKT_STAT_3: Contains TX error packet count. * - DP83TG720S_PKT_STAT_4: Contains RX packet count bits [15:0]. * - DP83TG720S_PKT_STAT_5: Contains RX packet count bits [31:16]. * - DP83TG720S_PKT_STAT_6: Contains RX error packet count. * * Keeping the register names as defined in the datasheet helps maintain * clarity and alignment with the documentation. */ #define DP83TG720S_PKT_STAT_1 0x639 #define DP83TG720S_PKT_STAT_2 0x63a #define DP83TG720S_PKT_STAT_3 0x63b #define DP83TG720S_PKT_STAT_4 0x63c #define DP83TG720S_PKT_STAT_5 0x63d #define DP83TG720S_PKT_STAT_6 0x63e /* Register 0x083F: Unknown Register */ #define DP83TG720S_UNKNOWN_083F 0x83f #define DP83TG720S_SQI_REG_1 0x871 #define DP83TG720S_SQI_OUT_WORST GENMASK(7, 5) #define DP83TG720S_SQI_OUT GENMASK(3, 1) #define DP83TG720_SQI_MAX 7 struct dp83tg720_stats { u64 link_loss_cnt; u64 tx_pkt_cnt; u64 tx_err_pkt_cnt; u64 rx_pkt_cnt; u64 rx_err_pkt_cnt; }; struct dp83tg720_priv { struct dp83tg720_stats stats; }; /** * dp83tg720_update_stats - Update the PHY statistics for the DP83TD510 PHY. * @phydev: Pointer to the phy_device structure. * * The function reads the PHY statistics registers and updates the statistics * structure. * * Returns: 0 on success or a negative error code on failure. */ static int dp83tg720_update_stats(struct phy_device *phydev) { struct dp83tg720_priv *priv = phydev->priv; u32 count; int ret; /* Read the link loss count */ ret = phy_read_mmd(phydev, MDIO_MMD_VEND2, DP83TG720S_LINK_QUAL_3); if (ret < 0) return ret; /* link_loss_cnt */ count = FIELD_GET(DP83TG720S_LINK_LOSS_CNT_MASK, ret); priv->stats.link_loss_cnt += count; /* The DP83TG720S_PKT_STAT registers are divided into two groups: * - Group 1 (TX stats): DP83TG720S_PKT_STAT_1 to DP83TG720S_PKT_STAT_3 * - Group 2 (RX stats): DP83TG720S_PKT_STAT_4 to DP83TG720S_PKT_STAT_6 * * Registers in each group are cleared only after reading them in a * plain sequence (e.g., 1, 2, 3 for Group 1 or 4, 5, 6 for Group 2). * Any deviation from the sequence, such as reading 1, 2, 1, 2, 3, will * prevent the group from being cleared. Additionally, the counters * for a group are frozen as soon as the first register in that group * is accessed. */ ret = phy_read_mmd(phydev, MDIO_MMD_VEND2, DP83TG720S_PKT_STAT_1); if (ret < 0) return ret; /* tx_pkt_cnt_15_0 */ count = ret; ret = phy_read_mmd(phydev, MDIO_MMD_VEND2, DP83TG720S_PKT_STAT_2); if (ret < 0) return ret; /* tx_pkt_cnt_31_16 */ count |= ret << 16; priv->stats.tx_pkt_cnt += count; ret = phy_read_mmd(phydev, MDIO_MMD_VEND2, DP83TG720S_PKT_STAT_3); if (ret < 0) return ret; /* tx_err_pkt_cnt */ priv->stats.tx_err_pkt_cnt += ret; ret = phy_read_mmd(phydev, MDIO_MMD_VEND2, DP83TG720S_PKT_STAT_4); if (ret < 0) return ret; /* rx_pkt_cnt_15_0 */ count = ret; ret = phy_read_mmd(phydev, MDIO_MMD_VEND2, DP83TG720S_PKT_STAT_5); if (ret < 0) return ret; /* rx_pkt_cnt_31_16 */ count |= ret << 16; priv->stats.rx_pkt_cnt += count; ret = phy_read_mmd(phydev, MDIO_MMD_VEND2, DP83TG720S_PKT_STAT_6); if (ret < 0) return ret; /* rx_err_pkt_cnt */ priv->stats.rx_err_pkt_cnt += ret; return 0; } static void dp83tg720_get_link_stats(struct phy_device *phydev, struct ethtool_link_ext_stats *link_stats) { struct dp83tg720_priv *priv = phydev->priv; link_stats->link_down_events = priv->stats.link_loss_cnt; } static void dp83tg720_get_phy_stats(struct phy_device *phydev, struct ethtool_eth_phy_stats *eth_stats, struct ethtool_phy_stats *stats) { struct dp83tg720_priv *priv = phydev->priv; stats->tx_packets = priv->stats.tx_pkt_cnt; stats->tx_errors = priv->stats.tx_err_pkt_cnt; stats->rx_packets = priv->stats.rx_pkt_cnt; stats->rx_errors = priv->stats.rx_err_pkt_cnt; } /** * dp83tg720_cable_test_start - Start the cable test for the DP83TG720 PHY. * @phydev: Pointer to the phy_device structure. * * This sequence is based on the documented procedure for the DP83TG720 PHY. * * Returns: 0 on success, a negative error code on failure. */ static int dp83tg720_cable_test_start(struct phy_device *phydev) { int ret; /* Initialize the PHY to run the TDR test as described in the * "DP83TG720S-Q1: Configuring for Open Alliance Specification * Compliance (Rev. B)" application note. * Most of the registers are not documented. Some of register names * are guessed by comparing the register offsets with the DP83TD510E. */ /* Force master link down */ ret = phy_set_bits_mmd(phydev, MDIO_MMD_VEND2, DP83TG720S_TDR_MASTER_LINK_DOWN, 0x0400); if (ret) return ret; ret = phy_write_mmd(phydev, MDIO_MMD_VEND2, DP83TG720S_TDR_CFG2, 0xa008); if (ret) return ret; ret = phy_write_mmd(phydev, MDIO_MMD_VEND2, DP83TG720S_TDR_CFG3, 0x0928); if (ret) return ret; ret = phy_write_mmd(phydev, MDIO_MMD_VEND2, DP83TG720S_TDR_CFG4, 0x0004); if (ret) return ret; ret = phy_write_mmd(phydev, MDIO_MMD_VEND2, DP83TG720S_UNKNOWN_0405, 0x6400); if (ret) return ret; ret = phy_write_mmd(phydev, MDIO_MMD_VEND2, DP83TG720S_UNKNOWN_083F, 0x3003); if (ret) return ret; /* Start the TDR */ ret = phy_set_bits_mmd(phydev, MDIO_MMD_VEND2, DP83TG720S_TDR_CFG, DP83TG720S_TDR_START); if (ret) return ret; return 0; } /** * dp83tg720_cable_test_get_status - Get the status of the cable test for the * DP83TG720 PHY. * @phydev: Pointer to the phy_device structure. * @finished: Pointer to a boolean that indicates whether the test is finished. * * The function sets the @finished flag to true if the test is complete. * * Returns: 0 on success or a negative error code on failure. */ static int dp83tg720_cable_test_get_status(struct phy_device *phydev, bool *finished) { int ret, stat; *finished = false; /* Read the TDR status */ ret = phy_read_mmd(phydev, MDIO_MMD_VEND2, DP83TG720S_TDR_CFG); if (ret < 0) return ret; /* Check if the TDR test is done */ if (!(ret & DP83TG720S_TDR_DONE)) return 0; /* Check for TDR test failure */ if (!(ret & DP83TG720S_TDR_FAIL)) { int location; /* Read fault status */ ret = phy_read_mmd(phydev, MDIO_MMD_VEND2, DP83TG720S_TDR_FAULT_STATUS); if (ret < 0) return ret; /* Get fault type */ stat = oa_1000bt1_get_ethtool_cable_result_code(ret); /* Determine fault location */ location = oa_1000bt1_get_tdr_distance(ret); if (location > 0) ethnl_cable_test_fault_length(phydev, ETHTOOL_A_CABLE_PAIR_A, location); } else { /* Active link partner or other issues */ stat = ETHTOOL_A_CABLE_RESULT_CODE_UNSPEC; } *finished = true; ethnl_cable_test_result(phydev, ETHTOOL_A_CABLE_PAIR_A, stat); /* save the current stats before resetting the PHY */ ret = dp83tg720_update_stats(phydev); if (ret) return ret; return phy_init_hw(phydev); } static int dp83tg720_config_aneg(struct phy_device *phydev) { int ret; /* Autoneg is not supported and this PHY supports only one speed. * We need to care only about master/slave configuration if it was * changed by user. */ ret = genphy_c45_pma_baset1_setup_master_slave(phydev); if (ret) return ret; /* Re-read role configuration to make changes visible even if * the link is in administrative down state. */ return genphy_c45_pma_baset1_read_master_slave(phydev); } static int dp83tg720_read_status(struct phy_device *phydev) { u16 phy_sts; int ret; phydev->pause = 0; phydev->asym_pause = 0; /* Most of Clause 45 registers are not present, so we can't use * genphy_c45_read_status() here. */ phy_sts = phy_read(phydev, DP83TG720S_MII_REG_10); phydev->link = !!(phy_sts & DP83TG720S_LINK_STATUS); if (!phydev->link) { /* save the current stats before resetting the PHY */ ret = dp83tg720_update_stats(phydev); if (ret) return ret; /* According to the "DP83TC81x, DP83TG72x Software * Implementation Guide", the PHY needs to be reset after a * link loss or if no link is created after at least 100ms. * * Currently we are polling with the PHY_STATE_TIME (1000ms) * interval, which is still enough for not automotive use cases. */ ret = phy_init_hw(phydev); if (ret) return ret; /* After HW reset we need to restore master/slave configuration. * genphy_c45_pma_baset1_read_master_slave() call will be done * by the dp83tg720_config_aneg() function. */ ret = dp83tg720_config_aneg(phydev); if (ret) return ret; phydev->speed = SPEED_UNKNOWN; phydev->duplex = DUPLEX_UNKNOWN; } else { /* PMA/PMD control 1 register (Register 1.0) is present, but it * doesn't contain the link speed information. * So genphy_c45_read_pma() can't be used here. */ ret = genphy_c45_pma_baset1_read_master_slave(phydev); if (ret) return ret; phydev->duplex = DUPLEX_FULL; phydev->speed = SPEED_1000; } return 0; } static int dp83tg720_get_sqi(struct phy_device *phydev) { int ret; if (!phydev->link) return 0; ret = phy_read_mmd(phydev, MDIO_MMD_VEND2, DP83TG720S_SQI_REG_1); if (ret < 0) return ret; return FIELD_GET(DP83TG720S_SQI_OUT, ret); } static int dp83tg720_get_sqi_max(struct phy_device *phydev) { return DP83TG720_SQI_MAX; } static int dp83tg720_config_rgmii_delay(struct phy_device *phydev) { u16 rgmii_delay_mask; u16 rgmii_delay = 0; switch (phydev->interface) { case PHY_INTERFACE_MODE_RGMII: rgmii_delay = 0; break; case PHY_INTERFACE_MODE_RGMII_ID: rgmii_delay = DP83TG720S_RGMII_RX_CLK_SEL | DP83TG720S_RGMII_TX_CLK_SEL; break; case PHY_INTERFACE_MODE_RGMII_RXID: rgmii_delay = DP83TG720S_RGMII_RX_CLK_SEL; break; case PHY_INTERFACE_MODE_RGMII_TXID: rgmii_delay = DP83TG720S_RGMII_TX_CLK_SEL; break; default: return 0; } rgmii_delay_mask = DP83TG720S_RGMII_RX_CLK_SEL | DP83TG720S_RGMII_TX_CLK_SEL; return phy_modify_mmd(phydev, MDIO_MMD_VEND2, DP83TG720S_RGMII_DELAY_CTRL, rgmii_delay_mask, rgmii_delay); } static int dp83tg720_config_init(struct phy_device *phydev) { int ret; /* Software Restart is not enough to recover from a link failure. * Using Hardware Reset instead. */ ret = phy_write(phydev, DP83TG720S_PHY_RESET, DP83TG720S_HW_RESET); if (ret) return ret; /* Wait until MDC can be used again. * The wait value of one 1ms is documented in "DP83TG720S-Q1 1000BASE-T1 * Automotive Ethernet PHY with SGMII and RGMII" datasheet. */ usleep_range(1000, 2000); if (phy_interface_is_rgmii(phydev)) { ret = dp83tg720_config_rgmii_delay(phydev); if (ret) return ret; } /* In case the PHY is bootstrapped in managed mode, we need to * wake it. */ ret = phy_write_mmd(phydev, MDIO_MMD_VEND2, DP83TG720S_LPS_CFG3, DP83TG720S_LPS_CFG3_PWR_MODE_0); if (ret) return ret; /* Make role configuration visible for ethtool on init and after * rest. */ return genphy_c45_pma_baset1_read_master_slave(phydev); } static int dp83tg720_probe(struct phy_device *phydev) { struct device *dev = &phydev->mdio.dev; struct dp83tg720_priv *priv; priv = devm_kzalloc(dev, sizeof(*priv), GFP_KERNEL); if (!priv) return -ENOMEM; phydev->priv = priv; return 0; } static struct phy_driver dp83tg720_driver[] = { { PHY_ID_MATCH_MODEL(DP83TG720S_PHY_ID), .name = "TI DP83TG720S", .flags = PHY_POLL_CABLE_TEST, .probe = dp83tg720_probe, .config_aneg = dp83tg720_config_aneg, .read_status = dp83tg720_read_status, .get_features = genphy_c45_pma_read_ext_abilities, .config_init = dp83tg720_config_init, .get_sqi = dp83tg720_get_sqi, .get_sqi_max = dp83tg720_get_sqi_max, .cable_test_start = dp83tg720_cable_test_start, .cable_test_get_status = dp83tg720_cable_test_get_status, .get_link_stats = dp83tg720_get_link_stats, .get_phy_stats = dp83tg720_get_phy_stats, .update_stats = dp83tg720_update_stats, .suspend = genphy_suspend, .resume = genphy_resume, } }; module_phy_driver(dp83tg720_driver); static const struct mdio_device_id __maybe_unused dp83tg720_tbl[] = { { PHY_ID_MATCH_MODEL(DP83TG720S_PHY_ID) }, { } }; MODULE_DEVICE_TABLE(mdio, dp83tg720_tbl); MODULE_DESCRIPTION("Texas Instruments DP83TG720S PHY driver"); MODULE_AUTHOR("Oleksij Rempel "); MODULE_LICENSE("GPL");