root/drivers/net/phy/mediatek/mtk-ge.c
// SPDX-License-Identifier: GPL-2.0+
#include <linux/bitfield.h>
#include <linux/module.h>
#include <linux/phy.h>

#include "mtk.h"

#define MTK_GPHY_ID_MT7530              0x03a29412
#define MTK_GPHY_ID_MT7531              0x03a29441

#define MTK_PHY_PAGE_EXTENDED_2                 0x0002
#define MTK_PHY_PAGE_EXTENDED_3                 0x0003
#define MTK_PHY_RG_LPI_PCS_DSP_CTRL_REG11       0x11

#define MTK_PHY_PAGE_EXTENDED_2A30              0x2a30

/* Registers on Token Ring debug nodes */
/* ch_addr = 0x1, node_addr = 0xf, data_addr = 0x17 */
#define SLAVE_DSP_READY_TIME_MASK               GENMASK(22, 15)

/* Registers on MDIO_MMD_VEND1 */
#define MTK_PHY_GBE_MODE_TX_DELAY_SEL           0x13
#define MTK_PHY_TEST_MODE_TX_DELAY_SEL          0x14
#define   MTK_TX_DELAY_PAIR_B_MASK              GENMASK(10, 8)
#define   MTK_TX_DELAY_PAIR_D_MASK              GENMASK(2, 0)

#define MTK_PHY_MCC_CTRL_AND_TX_POWER_CTRL      0xa6
#define   MTK_MCC_NEARECHO_OFFSET_MASK          GENMASK(15, 8)

#define MTK_PHY_RXADC_CTRL_RG7                  0xc6
#define   MTK_PHY_DA_AD_BUF_BIAS_LP_MASK        GENMASK(9, 8)

#define MTK_PHY_RG_LPI_PCS_DSP_CTRL_REG123      0x123
#define   MTK_PHY_LPI_NORM_MSE_LO_THRESH100_MASK        GENMASK(15, 8)
#define   MTK_PHY_LPI_NORM_MSE_HI_THRESH100_MASK        GENMASK(7, 0)

static void mtk_gephy_config_init(struct phy_device *phydev)
{
        /* Enable HW auto downshift */
        phy_modify_paged(phydev, MTK_PHY_PAGE_EXTENDED_1,
                         MTK_PHY_AUX_CTRL_AND_STATUS,
                         0, MTK_PHY_ENABLE_DOWNSHIFT);

        /* Increase SlvDPSready time */
        mtk_tr_modify(phydev, 0x1, 0xf, 0x17, SLAVE_DSP_READY_TIME_MASK,
                      FIELD_PREP(SLAVE_DSP_READY_TIME_MASK, 0x5e));

        /* Adjust 100_mse_threshold */
        phy_modify_mmd(phydev, MDIO_MMD_VEND1,
                       MTK_PHY_RG_LPI_PCS_DSP_CTRL_REG123,
                       MTK_PHY_LPI_NORM_MSE_LO_THRESH100_MASK |
                       MTK_PHY_LPI_NORM_MSE_HI_THRESH100_MASK,
                       FIELD_PREP(MTK_PHY_LPI_NORM_MSE_LO_THRESH100_MASK,
                                  0xff) |
                       FIELD_PREP(MTK_PHY_LPI_NORM_MSE_HI_THRESH100_MASK,
                                  0xff));

        /* If echo time is narrower than 0x3, it will be regarded as noise */
        phy_modify_mmd(phydev, MDIO_MMD_VEND1,
                       MTK_PHY_MCC_CTRL_AND_TX_POWER_CTRL,
                       MTK_MCC_NEARECHO_OFFSET_MASK,
                       FIELD_PREP(MTK_MCC_NEARECHO_OFFSET_MASK, 0x3));
}

static int mt7530_phy_config_init(struct phy_device *phydev)
{
        mtk_gephy_config_init(phydev);

        /* Increase post_update_timer */
        phy_write_paged(phydev, MTK_PHY_PAGE_EXTENDED_3,
                        MTK_PHY_RG_LPI_PCS_DSP_CTRL_REG11, 0x4b);

        return 0;
}

static int mt7531_phy_config_init(struct phy_device *phydev)
{
        mtk_gephy_config_init(phydev);

        /* PHY link down power saving enable */
        phy_set_bits(phydev, 0x17, BIT(4));
        phy_modify_mmd(phydev, MDIO_MMD_VEND1, MTK_PHY_RXADC_CTRL_RG7,
                       MTK_PHY_DA_AD_BUF_BIAS_LP_MASK,
                       FIELD_PREP(MTK_PHY_DA_AD_BUF_BIAS_LP_MASK, 0x3));

        /* Set TX Pair delay selection */
        phy_modify_mmd(phydev, MDIO_MMD_VEND1, MTK_PHY_GBE_MODE_TX_DELAY_SEL,
                       MTK_TX_DELAY_PAIR_B_MASK | MTK_TX_DELAY_PAIR_D_MASK,
                       FIELD_PREP(MTK_TX_DELAY_PAIR_B_MASK, 0x4) |
                       FIELD_PREP(MTK_TX_DELAY_PAIR_D_MASK, 0x4));
        phy_modify_mmd(phydev, MDIO_MMD_VEND1, MTK_PHY_TEST_MODE_TX_DELAY_SEL,
                       MTK_TX_DELAY_PAIR_B_MASK | MTK_TX_DELAY_PAIR_D_MASK,
                       FIELD_PREP(MTK_TX_DELAY_PAIR_B_MASK, 0x4) |
                       FIELD_PREP(MTK_TX_DELAY_PAIR_D_MASK, 0x4));

        return 0;
}

static struct phy_driver mtk_gephy_driver[] = {
        {
                PHY_ID_MATCH_EXACT(MTK_GPHY_ID_MT7530),
                .name           = "MediaTek MT7530 PHY",
                .config_init    = mt7530_phy_config_init,
                /* Interrupts are handled by the switch, not the PHY
                 * itself.
                 */
                .config_intr    = genphy_no_config_intr,
                .handle_interrupt = genphy_handle_interrupt_no_ack,
                .suspend        = genphy_suspend,
                .resume         = genphy_resume,
                .read_page      = mtk_phy_read_page,
                .write_page     = mtk_phy_write_page,
        },
        {
                PHY_ID_MATCH_EXACT(MTK_GPHY_ID_MT7531),
                .name           = "MediaTek MT7531 PHY",
                .config_init    = mt7531_phy_config_init,
                /* Interrupts are handled by the switch, not the PHY
                 * itself.
                 */
                .config_intr    = genphy_no_config_intr,
                .handle_interrupt = genphy_handle_interrupt_no_ack,
                .suspend        = genphy_suspend,
                .resume         = genphy_resume,
                .read_page      = mtk_phy_read_page,
                .write_page     = mtk_phy_write_page,
        },
};

module_phy_driver(mtk_gephy_driver);

static const struct mdio_device_id __maybe_unused mtk_gephy_tbl[] = {
        { PHY_ID_MATCH_EXACT(MTK_GPHY_ID_MT7530) },
        { PHY_ID_MATCH_EXACT(MTK_GPHY_ID_MT7531) },
        { }
};

MODULE_DESCRIPTION("MediaTek Gigabit Ethernet PHY driver");
MODULE_AUTHOR("DENG, Qingfang <dqfext@gmail.com>");
MODULE_LICENSE("GPL");

MODULE_DEVICE_TABLE(mdio, mtk_gephy_tbl);