|  | // SPDX-License-Identifier: GPL-2.0 | 
|  | /* | 
|  | * Copyright (c) 2020 Synopsys, Inc. and/or its affiliates. | 
|  | * Synopsys DesignWare XPCS helpers | 
|  | * | 
|  | * Author: Jose Abreu <Jose.Abreu@synopsys.com> | 
|  | */ | 
|  |  | 
|  | #include <linux/delay.h> | 
|  | #include <linux/mdio.h> | 
|  | #include <linux/mdio-xpcs.h> | 
|  | #include <linux/phylink.h> | 
|  | #include <linux/workqueue.h> | 
|  |  | 
|  | #define SYNOPSYS_XPCS_USXGMII_ID	0x7996ced0 | 
|  | #define SYNOPSYS_XPCS_10GKR_ID		0x7996ced0 | 
|  | #define SYNOPSYS_XPCS_XLGMII_ID		0x7996ced0 | 
|  | #define SYNOPSYS_XPCS_MASK		0xffffffff | 
|  |  | 
|  | /* Vendor regs access */ | 
|  | #define DW_VENDOR			BIT(15) | 
|  |  | 
|  | /* VR_XS_PCS */ | 
|  | #define DW_USXGMII_RST			BIT(10) | 
|  | #define DW_USXGMII_EN			BIT(9) | 
|  | #define DW_VR_XS_PCS_DIG_STS		0x0010 | 
|  | #define DW_RXFIFO_ERR			GENMASK(6, 5) | 
|  |  | 
|  | /* SR_MII */ | 
|  | #define DW_USXGMII_FULL			BIT(8) | 
|  | #define DW_USXGMII_SS_MASK		(BIT(13) | BIT(6) | BIT(5)) | 
|  | #define DW_USXGMII_10000		(BIT(13) | BIT(6)) | 
|  | #define DW_USXGMII_5000			(BIT(13) | BIT(5)) | 
|  | #define DW_USXGMII_2500			(BIT(5)) | 
|  | #define DW_USXGMII_1000			(BIT(6)) | 
|  | #define DW_USXGMII_100			(BIT(13)) | 
|  | #define DW_USXGMII_10			(0) | 
|  |  | 
|  | /* SR_AN */ | 
|  | #define DW_SR_AN_ADV1			0x10 | 
|  | #define DW_SR_AN_ADV2			0x11 | 
|  | #define DW_SR_AN_ADV3			0x12 | 
|  | #define DW_SR_AN_LP_ABL1		0x13 | 
|  | #define DW_SR_AN_LP_ABL2		0x14 | 
|  | #define DW_SR_AN_LP_ABL3		0x15 | 
|  |  | 
|  | /* Clause 73 Defines */ | 
|  | /* AN_LP_ABL1 */ | 
|  | #define DW_C73_PAUSE			BIT(10) | 
|  | #define DW_C73_ASYM_PAUSE		BIT(11) | 
|  | #define DW_C73_AN_ADV_SF		0x1 | 
|  | /* AN_LP_ABL2 */ | 
|  | #define DW_C73_1000KX			BIT(5) | 
|  | #define DW_C73_10000KX4			BIT(6) | 
|  | #define DW_C73_10000KR			BIT(7) | 
|  | /* AN_LP_ABL3 */ | 
|  | #define DW_C73_2500KX			BIT(0) | 
|  | #define DW_C73_5000KR			BIT(1) | 
|  |  | 
|  | static const int xpcs_usxgmii_features[] = { | 
|  | ETHTOOL_LINK_MODE_Pause_BIT, | 
|  | ETHTOOL_LINK_MODE_Asym_Pause_BIT, | 
|  | ETHTOOL_LINK_MODE_Autoneg_BIT, | 
|  | ETHTOOL_LINK_MODE_1000baseKX_Full_BIT, | 
|  | ETHTOOL_LINK_MODE_10000baseKX4_Full_BIT, | 
|  | ETHTOOL_LINK_MODE_10000baseKR_Full_BIT, | 
|  | ETHTOOL_LINK_MODE_2500baseX_Full_BIT, | 
|  | __ETHTOOL_LINK_MODE_MASK_NBITS, | 
|  | }; | 
|  |  | 
|  | static const int xpcs_10gkr_features[] = { | 
|  | ETHTOOL_LINK_MODE_Pause_BIT, | 
|  | ETHTOOL_LINK_MODE_Asym_Pause_BIT, | 
|  | ETHTOOL_LINK_MODE_10000baseKR_Full_BIT, | 
|  | __ETHTOOL_LINK_MODE_MASK_NBITS, | 
|  | }; | 
|  |  | 
|  | static const int xpcs_xlgmii_features[] = { | 
|  | ETHTOOL_LINK_MODE_Pause_BIT, | 
|  | ETHTOOL_LINK_MODE_Asym_Pause_BIT, | 
|  | ETHTOOL_LINK_MODE_25000baseCR_Full_BIT, | 
|  | ETHTOOL_LINK_MODE_25000baseKR_Full_BIT, | 
|  | ETHTOOL_LINK_MODE_25000baseSR_Full_BIT, | 
|  | ETHTOOL_LINK_MODE_40000baseKR4_Full_BIT, | 
|  | ETHTOOL_LINK_MODE_40000baseCR4_Full_BIT, | 
|  | ETHTOOL_LINK_MODE_40000baseSR4_Full_BIT, | 
|  | ETHTOOL_LINK_MODE_40000baseLR4_Full_BIT, | 
|  | ETHTOOL_LINK_MODE_50000baseCR2_Full_BIT, | 
|  | ETHTOOL_LINK_MODE_50000baseKR2_Full_BIT, | 
|  | ETHTOOL_LINK_MODE_50000baseSR2_Full_BIT, | 
|  | ETHTOOL_LINK_MODE_50000baseKR_Full_BIT, | 
|  | ETHTOOL_LINK_MODE_50000baseSR_Full_BIT, | 
|  | ETHTOOL_LINK_MODE_50000baseCR_Full_BIT, | 
|  | ETHTOOL_LINK_MODE_50000baseLR_ER_FR_Full_BIT, | 
|  | ETHTOOL_LINK_MODE_50000baseDR_Full_BIT, | 
|  | ETHTOOL_LINK_MODE_100000baseKR4_Full_BIT, | 
|  | ETHTOOL_LINK_MODE_100000baseSR4_Full_BIT, | 
|  | ETHTOOL_LINK_MODE_100000baseCR4_Full_BIT, | 
|  | ETHTOOL_LINK_MODE_100000baseLR4_ER4_Full_BIT, | 
|  | ETHTOOL_LINK_MODE_100000baseKR2_Full_BIT, | 
|  | ETHTOOL_LINK_MODE_100000baseSR2_Full_BIT, | 
|  | ETHTOOL_LINK_MODE_100000baseCR2_Full_BIT, | 
|  | ETHTOOL_LINK_MODE_100000baseLR2_ER2_FR2_Full_BIT, | 
|  | ETHTOOL_LINK_MODE_100000baseDR2_Full_BIT, | 
|  | __ETHTOOL_LINK_MODE_MASK_NBITS, | 
|  | }; | 
|  |  | 
|  | static const phy_interface_t xpcs_usxgmii_interfaces[] = { | 
|  | PHY_INTERFACE_MODE_USXGMII, | 
|  | PHY_INTERFACE_MODE_MAX, | 
|  | }; | 
|  |  | 
|  | static const phy_interface_t xpcs_10gkr_interfaces[] = { | 
|  | PHY_INTERFACE_MODE_10GKR, | 
|  | PHY_INTERFACE_MODE_MAX, | 
|  | }; | 
|  |  | 
|  | static const phy_interface_t xpcs_xlgmii_interfaces[] = { | 
|  | PHY_INTERFACE_MODE_XLGMII, | 
|  | PHY_INTERFACE_MODE_MAX, | 
|  | }; | 
|  |  | 
|  | static struct xpcs_id { | 
|  | u32 id; | 
|  | u32 mask; | 
|  | const int *supported; | 
|  | const phy_interface_t *interface; | 
|  | } xpcs_id_list[] = { | 
|  | { | 
|  | .id = SYNOPSYS_XPCS_USXGMII_ID, | 
|  | .mask = SYNOPSYS_XPCS_MASK, | 
|  | .supported = xpcs_usxgmii_features, | 
|  | .interface = xpcs_usxgmii_interfaces, | 
|  | }, { | 
|  | .id = SYNOPSYS_XPCS_10GKR_ID, | 
|  | .mask = SYNOPSYS_XPCS_MASK, | 
|  | .supported = xpcs_10gkr_features, | 
|  | .interface = xpcs_10gkr_interfaces, | 
|  | }, { | 
|  | .id = SYNOPSYS_XPCS_XLGMII_ID, | 
|  | .mask = SYNOPSYS_XPCS_MASK, | 
|  | .supported = xpcs_xlgmii_features, | 
|  | .interface = xpcs_xlgmii_interfaces, | 
|  | }, | 
|  | }; | 
|  |  | 
|  | static int xpcs_read(struct mdio_xpcs_args *xpcs, int dev, u32 reg) | 
|  | { | 
|  | u32 reg_addr = MII_ADDR_C45 | dev << 16 | reg; | 
|  |  | 
|  | return mdiobus_read(xpcs->bus, xpcs->addr, reg_addr); | 
|  | } | 
|  |  | 
|  | static int xpcs_write(struct mdio_xpcs_args *xpcs, int dev, u32 reg, u16 val) | 
|  | { | 
|  | u32 reg_addr = MII_ADDR_C45 | dev << 16 | reg; | 
|  |  | 
|  | return mdiobus_write(xpcs->bus, xpcs->addr, reg_addr, val); | 
|  | } | 
|  |  | 
|  | static int xpcs_read_vendor(struct mdio_xpcs_args *xpcs, int dev, u32 reg) | 
|  | { | 
|  | return xpcs_read(xpcs, dev, DW_VENDOR | reg); | 
|  | } | 
|  |  | 
|  | static int xpcs_write_vendor(struct mdio_xpcs_args *xpcs, int dev, int reg, | 
|  | u16 val) | 
|  | { | 
|  | return xpcs_write(xpcs, dev, DW_VENDOR | reg, val); | 
|  | } | 
|  |  | 
|  | static int xpcs_read_vpcs(struct mdio_xpcs_args *xpcs, int reg) | 
|  | { | 
|  | return xpcs_read_vendor(xpcs, MDIO_MMD_PCS, reg); | 
|  | } | 
|  |  | 
|  | static int xpcs_write_vpcs(struct mdio_xpcs_args *xpcs, int reg, u16 val) | 
|  | { | 
|  | return xpcs_write_vendor(xpcs, MDIO_MMD_PCS, reg, val); | 
|  | } | 
|  |  | 
|  | static int xpcs_poll_reset(struct mdio_xpcs_args *xpcs, int dev) | 
|  | { | 
|  | /* Poll until the reset bit clears (50ms per retry == 0.6 sec) */ | 
|  | unsigned int retries = 12; | 
|  | int ret; | 
|  |  | 
|  | do { | 
|  | msleep(50); | 
|  | ret = xpcs_read(xpcs, dev, MDIO_CTRL1); | 
|  | if (ret < 0) | 
|  | return ret; | 
|  | } while (ret & MDIO_CTRL1_RESET && --retries); | 
|  |  | 
|  | return (ret & MDIO_CTRL1_RESET) ? -ETIMEDOUT : 0; | 
|  | } | 
|  |  | 
|  | static int xpcs_soft_reset(struct mdio_xpcs_args *xpcs, int dev) | 
|  | { | 
|  | int ret; | 
|  |  | 
|  | ret = xpcs_write(xpcs, dev, MDIO_CTRL1, MDIO_CTRL1_RESET); | 
|  | if (ret < 0) | 
|  | return ret; | 
|  |  | 
|  | return xpcs_poll_reset(xpcs, dev); | 
|  | } | 
|  |  | 
|  | #define xpcs_warn(__xpcs, __state, __args...) \ | 
|  | ({ \ | 
|  | if ((__state)->link) \ | 
|  | dev_warn(&(__xpcs)->bus->dev, ##__args); \ | 
|  | }) | 
|  |  | 
|  | static int xpcs_read_fault(struct mdio_xpcs_args *xpcs, | 
|  | struct phylink_link_state *state) | 
|  | { | 
|  | int ret; | 
|  |  | 
|  | ret = xpcs_read(xpcs, MDIO_MMD_PCS, MDIO_STAT1); | 
|  | if (ret < 0) | 
|  | return ret; | 
|  |  | 
|  | if (ret & MDIO_STAT1_FAULT) { | 
|  | xpcs_warn(xpcs, state, "Link fault condition detected!\n"); | 
|  | return -EFAULT; | 
|  | } | 
|  |  | 
|  | ret = xpcs_read(xpcs, MDIO_MMD_PCS, MDIO_STAT2); | 
|  | if (ret < 0) | 
|  | return ret; | 
|  |  | 
|  | if (ret & MDIO_STAT2_RXFAULT) | 
|  | xpcs_warn(xpcs, state, "Receiver fault detected!\n"); | 
|  | if (ret & MDIO_STAT2_TXFAULT) | 
|  | xpcs_warn(xpcs, state, "Transmitter fault detected!\n"); | 
|  |  | 
|  | ret = xpcs_read_vendor(xpcs, MDIO_MMD_PCS, DW_VR_XS_PCS_DIG_STS); | 
|  | if (ret < 0) | 
|  | return ret; | 
|  |  | 
|  | if (ret & DW_RXFIFO_ERR) { | 
|  | xpcs_warn(xpcs, state, "FIFO fault condition detected!\n"); | 
|  | return -EFAULT; | 
|  | } | 
|  |  | 
|  | ret = xpcs_read(xpcs, MDIO_MMD_PCS, MDIO_PCS_10GBRT_STAT1); | 
|  | if (ret < 0) | 
|  | return ret; | 
|  |  | 
|  | if (!(ret & MDIO_PCS_10GBRT_STAT1_BLKLK)) | 
|  | xpcs_warn(xpcs, state, "Link is not locked!\n"); | 
|  |  | 
|  | ret = xpcs_read(xpcs, MDIO_MMD_PCS, MDIO_PCS_10GBRT_STAT2); | 
|  | if (ret < 0) | 
|  | return ret; | 
|  |  | 
|  | if (ret & MDIO_PCS_10GBRT_STAT2_ERR) { | 
|  | xpcs_warn(xpcs, state, "Link has errors!\n"); | 
|  | return -EFAULT; | 
|  | } | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int xpcs_read_link(struct mdio_xpcs_args *xpcs, bool an) | 
|  | { | 
|  | bool link = true; | 
|  | int ret; | 
|  |  | 
|  | ret = xpcs_read(xpcs, MDIO_MMD_PCS, MDIO_STAT1); | 
|  | if (ret < 0) | 
|  | return ret; | 
|  |  | 
|  | if (!(ret & MDIO_STAT1_LSTATUS)) | 
|  | link = false; | 
|  |  | 
|  | if (an) { | 
|  | ret = xpcs_read(xpcs, MDIO_MMD_AN, MDIO_STAT1); | 
|  | if (ret < 0) | 
|  | return ret; | 
|  |  | 
|  | if (!(ret & MDIO_STAT1_LSTATUS)) | 
|  | link = false; | 
|  | } | 
|  |  | 
|  | return link; | 
|  | } | 
|  |  | 
|  | static int xpcs_get_max_usxgmii_speed(const unsigned long *supported) | 
|  | { | 
|  | int max = SPEED_UNKNOWN; | 
|  |  | 
|  | if (phylink_test(supported, 1000baseKX_Full)) | 
|  | max = SPEED_1000; | 
|  | if (phylink_test(supported, 2500baseX_Full)) | 
|  | max = SPEED_2500; | 
|  | if (phylink_test(supported, 10000baseKX4_Full)) | 
|  | max = SPEED_10000; | 
|  | if (phylink_test(supported, 10000baseKR_Full)) | 
|  | max = SPEED_10000; | 
|  |  | 
|  | return max; | 
|  | } | 
|  |  | 
|  | static int xpcs_config_usxgmii(struct mdio_xpcs_args *xpcs, int speed) | 
|  | { | 
|  | int ret, speed_sel; | 
|  |  | 
|  | switch (speed) { | 
|  | case SPEED_10: | 
|  | speed_sel = DW_USXGMII_10; | 
|  | break; | 
|  | case SPEED_100: | 
|  | speed_sel = DW_USXGMII_100; | 
|  | break; | 
|  | case SPEED_1000: | 
|  | speed_sel = DW_USXGMII_1000; | 
|  | break; | 
|  | case SPEED_2500: | 
|  | speed_sel = DW_USXGMII_2500; | 
|  | break; | 
|  | case SPEED_5000: | 
|  | speed_sel = DW_USXGMII_5000; | 
|  | break; | 
|  | case SPEED_10000: | 
|  | speed_sel = DW_USXGMII_10000; | 
|  | break; | 
|  | default: | 
|  | /* Nothing to do here */ | 
|  | return -EINVAL; | 
|  | } | 
|  |  | 
|  | ret = xpcs_read_vpcs(xpcs, MDIO_CTRL1); | 
|  | if (ret < 0) | 
|  | return ret; | 
|  |  | 
|  | ret = xpcs_write_vpcs(xpcs, MDIO_CTRL1, ret | DW_USXGMII_EN); | 
|  | if (ret < 0) | 
|  | return ret; | 
|  |  | 
|  | ret = xpcs_read(xpcs, MDIO_MMD_VEND2, MDIO_CTRL1); | 
|  | if (ret < 0) | 
|  | return ret; | 
|  |  | 
|  | ret &= ~DW_USXGMII_SS_MASK; | 
|  | ret |= speed_sel | DW_USXGMII_FULL; | 
|  |  | 
|  | ret = xpcs_write(xpcs, MDIO_MMD_VEND2, MDIO_CTRL1, ret); | 
|  | if (ret < 0) | 
|  | return ret; | 
|  |  | 
|  | ret = xpcs_read_vpcs(xpcs, MDIO_CTRL1); | 
|  | if (ret < 0) | 
|  | return ret; | 
|  |  | 
|  | return xpcs_write_vpcs(xpcs, MDIO_CTRL1, ret | DW_USXGMII_RST); | 
|  | } | 
|  |  | 
|  | static int xpcs_config_aneg_c73(struct mdio_xpcs_args *xpcs) | 
|  | { | 
|  | int ret, adv; | 
|  |  | 
|  | /* By default, in USXGMII mode XPCS operates at 10G baud and | 
|  | * replicates data to achieve lower speeds. Hereby, in this | 
|  | * default configuration we need to advertise all supported | 
|  | * modes and not only the ones we want to use. | 
|  | */ | 
|  |  | 
|  | /* SR_AN_ADV3 */ | 
|  | adv = 0; | 
|  | if (phylink_test(xpcs->supported, 2500baseX_Full)) | 
|  | adv |= DW_C73_2500KX; | 
|  |  | 
|  | /* TODO: 5000baseKR */ | 
|  |  | 
|  | ret = xpcs_write(xpcs, MDIO_MMD_AN, DW_SR_AN_ADV3, adv); | 
|  | if (ret < 0) | 
|  | return ret; | 
|  |  | 
|  | /* SR_AN_ADV2 */ | 
|  | adv = 0; | 
|  | if (phylink_test(xpcs->supported, 1000baseKX_Full)) | 
|  | adv |= DW_C73_1000KX; | 
|  | if (phylink_test(xpcs->supported, 10000baseKX4_Full)) | 
|  | adv |= DW_C73_10000KX4; | 
|  | if (phylink_test(xpcs->supported, 10000baseKR_Full)) | 
|  | adv |= DW_C73_10000KR; | 
|  |  | 
|  | ret = xpcs_write(xpcs, MDIO_MMD_AN, DW_SR_AN_ADV2, adv); | 
|  | if (ret < 0) | 
|  | return ret; | 
|  |  | 
|  | /* SR_AN_ADV1 */ | 
|  | adv = DW_C73_AN_ADV_SF; | 
|  | if (phylink_test(xpcs->supported, Pause)) | 
|  | adv |= DW_C73_PAUSE; | 
|  | if (phylink_test(xpcs->supported, Asym_Pause)) | 
|  | adv |= DW_C73_ASYM_PAUSE; | 
|  |  | 
|  | return xpcs_write(xpcs, MDIO_MMD_AN, DW_SR_AN_ADV1, adv); | 
|  | } | 
|  |  | 
|  | static int xpcs_config_aneg(struct mdio_xpcs_args *xpcs) | 
|  | { | 
|  | int ret; | 
|  |  | 
|  | ret = xpcs_config_aneg_c73(xpcs); | 
|  | if (ret < 0) | 
|  | return ret; | 
|  |  | 
|  | ret = xpcs_read(xpcs, MDIO_MMD_AN, MDIO_CTRL1); | 
|  | if (ret < 0) | 
|  | return ret; | 
|  |  | 
|  | ret |= MDIO_AN_CTRL1_ENABLE | MDIO_AN_CTRL1_RESTART; | 
|  |  | 
|  | return xpcs_write(xpcs, MDIO_MMD_AN, MDIO_CTRL1, ret); | 
|  | } | 
|  |  | 
|  | static int xpcs_aneg_done(struct mdio_xpcs_args *xpcs, | 
|  | struct phylink_link_state *state) | 
|  | { | 
|  | int ret; | 
|  |  | 
|  | ret = xpcs_read(xpcs, MDIO_MMD_AN, MDIO_STAT1); | 
|  | if (ret < 0) | 
|  | return ret; | 
|  |  | 
|  | if (ret & MDIO_AN_STAT1_COMPLETE) { | 
|  | ret = xpcs_read(xpcs, MDIO_MMD_AN, DW_SR_AN_LP_ABL1); | 
|  | if (ret < 0) | 
|  | return ret; | 
|  |  | 
|  | /* Check if Aneg outcome is valid */ | 
|  | if (!(ret & DW_C73_AN_ADV_SF)) { | 
|  | xpcs_config_aneg(xpcs); | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | return 1; | 
|  | } | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int xpcs_read_lpa(struct mdio_xpcs_args *xpcs, | 
|  | struct phylink_link_state *state) | 
|  | { | 
|  | int ret; | 
|  |  | 
|  | ret = xpcs_read(xpcs, MDIO_MMD_AN, MDIO_STAT1); | 
|  | if (ret < 0) | 
|  | return ret; | 
|  |  | 
|  | if (!(ret & MDIO_AN_STAT1_LPABLE)) { | 
|  | phylink_clear(state->lp_advertising, Autoneg); | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | phylink_set(state->lp_advertising, Autoneg); | 
|  |  | 
|  | /* Clause 73 outcome */ | 
|  | ret = xpcs_read(xpcs, MDIO_MMD_AN, DW_SR_AN_LP_ABL3); | 
|  | if (ret < 0) | 
|  | return ret; | 
|  |  | 
|  | if (ret & DW_C73_2500KX) | 
|  | phylink_set(state->lp_advertising, 2500baseX_Full); | 
|  |  | 
|  | ret = xpcs_read(xpcs, MDIO_MMD_AN, DW_SR_AN_LP_ABL2); | 
|  | if (ret < 0) | 
|  | return ret; | 
|  |  | 
|  | if (ret & DW_C73_1000KX) | 
|  | phylink_set(state->lp_advertising, 1000baseKX_Full); | 
|  | if (ret & DW_C73_10000KX4) | 
|  | phylink_set(state->lp_advertising, 10000baseKX4_Full); | 
|  | if (ret & DW_C73_10000KR) | 
|  | phylink_set(state->lp_advertising, 10000baseKR_Full); | 
|  |  | 
|  | ret = xpcs_read(xpcs, MDIO_MMD_AN, DW_SR_AN_LP_ABL1); | 
|  | if (ret < 0) | 
|  | return ret; | 
|  |  | 
|  | if (ret & DW_C73_PAUSE) | 
|  | phylink_set(state->lp_advertising, Pause); | 
|  | if (ret & DW_C73_ASYM_PAUSE) | 
|  | phylink_set(state->lp_advertising, Asym_Pause); | 
|  |  | 
|  | linkmode_and(state->lp_advertising, state->lp_advertising, | 
|  | state->advertising); | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static void xpcs_resolve_lpa(struct mdio_xpcs_args *xpcs, | 
|  | struct phylink_link_state *state) | 
|  | { | 
|  | int max_speed = xpcs_get_max_usxgmii_speed(state->lp_advertising); | 
|  |  | 
|  | state->pause = MLO_PAUSE_TX | MLO_PAUSE_RX; | 
|  | state->speed = max_speed; | 
|  | state->duplex = DUPLEX_FULL; | 
|  | } | 
|  |  | 
|  | static int xpcs_get_max_xlgmii_speed(struct mdio_xpcs_args *xpcs, | 
|  | struct phylink_link_state *state) | 
|  | { | 
|  | unsigned long *adv = state->advertising; | 
|  | int speed = SPEED_UNKNOWN; | 
|  | int bit; | 
|  |  | 
|  | for_each_set_bit(bit, adv, __ETHTOOL_LINK_MODE_MASK_NBITS) { | 
|  | int new_speed = SPEED_UNKNOWN; | 
|  |  | 
|  | switch (bit) { | 
|  | case ETHTOOL_LINK_MODE_25000baseCR_Full_BIT: | 
|  | case ETHTOOL_LINK_MODE_25000baseKR_Full_BIT: | 
|  | case ETHTOOL_LINK_MODE_25000baseSR_Full_BIT: | 
|  | new_speed = SPEED_25000; | 
|  | break; | 
|  | case ETHTOOL_LINK_MODE_40000baseKR4_Full_BIT: | 
|  | case ETHTOOL_LINK_MODE_40000baseCR4_Full_BIT: | 
|  | case ETHTOOL_LINK_MODE_40000baseSR4_Full_BIT: | 
|  | case ETHTOOL_LINK_MODE_40000baseLR4_Full_BIT: | 
|  | new_speed = SPEED_40000; | 
|  | break; | 
|  | case ETHTOOL_LINK_MODE_50000baseCR2_Full_BIT: | 
|  | case ETHTOOL_LINK_MODE_50000baseKR2_Full_BIT: | 
|  | case ETHTOOL_LINK_MODE_50000baseSR2_Full_BIT: | 
|  | case ETHTOOL_LINK_MODE_50000baseKR_Full_BIT: | 
|  | case ETHTOOL_LINK_MODE_50000baseSR_Full_BIT: | 
|  | case ETHTOOL_LINK_MODE_50000baseCR_Full_BIT: | 
|  | case ETHTOOL_LINK_MODE_50000baseLR_ER_FR_Full_BIT: | 
|  | case ETHTOOL_LINK_MODE_50000baseDR_Full_BIT: | 
|  | new_speed = SPEED_50000; | 
|  | break; | 
|  | case ETHTOOL_LINK_MODE_100000baseKR4_Full_BIT: | 
|  | case ETHTOOL_LINK_MODE_100000baseSR4_Full_BIT: | 
|  | case ETHTOOL_LINK_MODE_100000baseCR4_Full_BIT: | 
|  | case ETHTOOL_LINK_MODE_100000baseLR4_ER4_Full_BIT: | 
|  | case ETHTOOL_LINK_MODE_100000baseKR2_Full_BIT: | 
|  | case ETHTOOL_LINK_MODE_100000baseSR2_Full_BIT: | 
|  | case ETHTOOL_LINK_MODE_100000baseCR2_Full_BIT: | 
|  | case ETHTOOL_LINK_MODE_100000baseLR2_ER2_FR2_Full_BIT: | 
|  | case ETHTOOL_LINK_MODE_100000baseDR2_Full_BIT: | 
|  | new_speed = SPEED_100000; | 
|  | break; | 
|  | default: | 
|  | continue; | 
|  | } | 
|  |  | 
|  | if (new_speed > speed) | 
|  | speed = new_speed; | 
|  | } | 
|  |  | 
|  | return speed; | 
|  | } | 
|  |  | 
|  | static void xpcs_resolve_pma(struct mdio_xpcs_args *xpcs, | 
|  | struct phylink_link_state *state) | 
|  | { | 
|  | state->pause = MLO_PAUSE_TX | MLO_PAUSE_RX; | 
|  | state->duplex = DUPLEX_FULL; | 
|  |  | 
|  | switch (state->interface) { | 
|  | case PHY_INTERFACE_MODE_10GKR: | 
|  | state->speed = SPEED_10000; | 
|  | break; | 
|  | case PHY_INTERFACE_MODE_XLGMII: | 
|  | state->speed = xpcs_get_max_xlgmii_speed(xpcs, state); | 
|  | break; | 
|  | default: | 
|  | state->speed = SPEED_UNKNOWN; | 
|  | break; | 
|  | } | 
|  | } | 
|  |  | 
|  | static int xpcs_validate(struct mdio_xpcs_args *xpcs, | 
|  | unsigned long *supported, | 
|  | struct phylink_link_state *state) | 
|  | { | 
|  | linkmode_and(supported, supported, xpcs->supported); | 
|  | linkmode_and(state->advertising, state->advertising, xpcs->supported); | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int xpcs_config(struct mdio_xpcs_args *xpcs, | 
|  | const struct phylink_link_state *state) | 
|  | { | 
|  | int ret; | 
|  |  | 
|  | if (state->an_enabled) { | 
|  | ret = xpcs_config_aneg(xpcs); | 
|  | if (ret) | 
|  | return ret; | 
|  | } | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int xpcs_get_state(struct mdio_xpcs_args *xpcs, | 
|  | struct phylink_link_state *state) | 
|  | { | 
|  | int ret; | 
|  |  | 
|  | /* Link needs to be read first ... */ | 
|  | state->link = xpcs_read_link(xpcs, state->an_enabled) > 0 ? 1 : 0; | 
|  |  | 
|  | /* ... and then we check the faults. */ | 
|  | ret = xpcs_read_fault(xpcs, state); | 
|  | if (ret) { | 
|  | ret = xpcs_soft_reset(xpcs, MDIO_MMD_PCS); | 
|  | if (ret) | 
|  | return ret; | 
|  |  | 
|  | state->link = 0; | 
|  |  | 
|  | return xpcs_config(xpcs, state); | 
|  | } | 
|  |  | 
|  | if (state->an_enabled && xpcs_aneg_done(xpcs, state)) { | 
|  | state->an_complete = true; | 
|  | xpcs_read_lpa(xpcs, state); | 
|  | xpcs_resolve_lpa(xpcs, state); | 
|  | } else if (state->an_enabled) { | 
|  | state->link = 0; | 
|  | } else if (state->link) { | 
|  | xpcs_resolve_pma(xpcs, state); | 
|  | } | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int xpcs_link_up(struct mdio_xpcs_args *xpcs, int speed, | 
|  | phy_interface_t interface) | 
|  | { | 
|  | if (interface == PHY_INTERFACE_MODE_USXGMII) | 
|  | return xpcs_config_usxgmii(xpcs, speed); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static u32 xpcs_get_id(struct mdio_xpcs_args *xpcs) | 
|  | { | 
|  | int ret; | 
|  | u32 id; | 
|  |  | 
|  | ret = xpcs_read(xpcs, MDIO_MMD_PCS, MII_PHYSID1); | 
|  | if (ret < 0) | 
|  | return 0xffffffff; | 
|  |  | 
|  | id = ret << 16; | 
|  |  | 
|  | ret = xpcs_read(xpcs, MDIO_MMD_PCS, MII_PHYSID2); | 
|  | if (ret < 0) | 
|  | return 0xffffffff; | 
|  |  | 
|  | return id | ret; | 
|  | } | 
|  |  | 
|  | static bool xpcs_check_features(struct mdio_xpcs_args *xpcs, | 
|  | struct xpcs_id *match, | 
|  | phy_interface_t interface) | 
|  | { | 
|  | int i; | 
|  |  | 
|  | for (i = 0; match->interface[i] != PHY_INTERFACE_MODE_MAX; i++) { | 
|  | if (match->interface[i] == interface) | 
|  | break; | 
|  | } | 
|  |  | 
|  | if (match->interface[i] == PHY_INTERFACE_MODE_MAX) | 
|  | return false; | 
|  |  | 
|  | for (i = 0; match->supported[i] != __ETHTOOL_LINK_MODE_MASK_NBITS; i++) | 
|  | set_bit(match->supported[i], xpcs->supported); | 
|  |  | 
|  | return true; | 
|  | } | 
|  |  | 
|  | static int xpcs_probe(struct mdio_xpcs_args *xpcs, phy_interface_t interface) | 
|  | { | 
|  | u32 xpcs_id = xpcs_get_id(xpcs); | 
|  | struct xpcs_id *match = NULL; | 
|  | int i; | 
|  |  | 
|  | for (i = 0; i < ARRAY_SIZE(xpcs_id_list); i++) { | 
|  | struct xpcs_id *entry = &xpcs_id_list[i]; | 
|  |  | 
|  | if ((xpcs_id & entry->mask) == entry->id) { | 
|  | match = entry; | 
|  |  | 
|  | if (xpcs_check_features(xpcs, match, interface)) | 
|  | return xpcs_soft_reset(xpcs, MDIO_MMD_PCS); | 
|  | } | 
|  | } | 
|  |  | 
|  | return -ENODEV; | 
|  | } | 
|  |  | 
|  | static struct mdio_xpcs_ops xpcs_ops = { | 
|  | .validate = xpcs_validate, | 
|  | .config = xpcs_config, | 
|  | .get_state = xpcs_get_state, | 
|  | .link_up = xpcs_link_up, | 
|  | .probe = xpcs_probe, | 
|  | }; | 
|  |  | 
|  | struct mdio_xpcs_ops *mdio_xpcs_get_ops(void) | 
|  | { | 
|  | return &xpcs_ops; | 
|  | } | 
|  | EXPORT_SYMBOL_GPL(mdio_xpcs_get_ops); | 
|  |  | 
|  | MODULE_LICENSE("GPL v2"); |