net: pcs: xpcs: add _modify() accessors
authorRussell King (Oracle) <rmk+kernel@armlinux.org.uk>
Fri, 4 Oct 2024 10:21:22 +0000 (11:21 +0100)
committerDavid S. Miller <davem@davemloft.net>
Wed, 9 Oct 2024 11:13:11 +0000 (12:13 +0100)
The xpcs driver does a lot of read-modify-write operations on
registers, which leads to long-winded code to read the register, check
whether the read was successful, modify the value in some way, and then
write it back.

We have a mdiodev _modify() accessor that encapsulates this, and does
the register modification under the MDIO bus lock ensuring that the
modification is atomic with respect to other bus operations. Convert
the xpcs driver to use this accessor.

Signed-off-by: Russell King (Oracle) <rmk+kernel@armlinux.org.uk>
Signed-off-by: David S. Miller <davem@davemloft.net>
drivers/net/pcs/pcs-xpcs-nxp.c
drivers/net/pcs/pcs-xpcs-wx.c
drivers/net/pcs/pcs-xpcs.c
drivers/net/pcs/pcs-xpcs.h

index d16fc58cd48d3d8d877dc02ad2096c9a91244221..e8efe94cf4ec7d0eadacbab1707c5697c06fa6f7 100644 (file)
@@ -152,26 +152,18 @@ static int nxp_sja1110_pma_config(struct dw_xpcs *xpcs,
        /* Enable TX and RX PLLs and circuits.
         * Release reset of PMA to enable data flow to/from PCS.
         */
-       ret = xpcs_read(xpcs, MDIO_MMD_VEND2, SJA1110_POWERDOWN_ENABLE);
-       if (ret < 0)
-               return ret;
-
-       val = ret & ~(SJA1110_TXPLL_PD | SJA1110_TXPD | SJA1110_RXCH_PD |
-                     SJA1110_RXBIAS_PD | SJA1110_RESET_SER_EN |
-                     SJA1110_RESET_SER | SJA1110_RESET_DES);
-       val |= SJA1110_RXPKDETEN | SJA1110_RCVEN;
-
-       ret = xpcs_write(xpcs, MDIO_MMD_VEND2, SJA1110_POWERDOWN_ENABLE, val);
+       ret = xpcs_modify(xpcs, MDIO_MMD_VEND2, SJA1110_POWERDOWN_ENABLE,
+                         SJA1110_TXPLL_PD | SJA1110_TXPD | SJA1110_RXCH_PD |
+                         SJA1110_RXBIAS_PD | SJA1110_RESET_SER_EN |
+                         SJA1110_RESET_SER | SJA1110_RESET_DES |
+                         SJA1110_RXPKDETEN | SJA1110_RCVEN,
+                         SJA1110_RXPKDETEN | SJA1110_RCVEN);
        if (ret < 0)
                return ret;
 
        /* Program continuous-time linear equalizer (CTLE) settings. */
-       ret = xpcs_write(xpcs, MDIO_MMD_VEND2, SJA1110_RX_CDR_CTLE,
-                        rx_cdr_ctle);
-       if (ret < 0)
-               return ret;
-
-       return 0;
+       return xpcs_write(xpcs, MDIO_MMD_VEND2, SJA1110_RX_CDR_CTLE,
+                         rx_cdr_ctle);
 }
 
 int nxp_sja1110_sgmii_pma_config(struct dw_xpcs *xpcs)
index 5f5cd3596cb846309514a1b133e773eae783c84e..fc52f7aa5f5965d91c6e70b3a9f299892b161b9c 100644 (file)
 #define TXGBE_VCO_CAL_LD0              0x72
 #define TXGBE_VCO_CAL_REF0             0x76
 
-static int txgbe_read_pma(struct dw_xpcs *xpcs, int reg)
+static int txgbe_write_pma(struct dw_xpcs *xpcs, int reg, u16 val)
 {
-       return xpcs_read(xpcs, MDIO_MMD_PMAPMD, TXGBE_PMA_MMD + reg);
+       return xpcs_write(xpcs, MDIO_MMD_PMAPMD, TXGBE_PMA_MMD + reg, val);
 }
 
-static int txgbe_write_pma(struct dw_xpcs *xpcs, int reg, u16 val)
+static int txgbe_modify_pma(struct dw_xpcs *xpcs, int reg, u16 mask, u16 set)
 {
-       return xpcs_write(xpcs, MDIO_MMD_PMAPMD, TXGBE_PMA_MMD + reg, val);
+       return xpcs_modify(xpcs, MDIO_MMD_PMAPMD, TXGBE_PMA_MMD + reg, mask,
+                          set);
 }
 
 static void txgbe_pma_config_10gbaser(struct dw_xpcs *xpcs)
 {
-       int val;
-
        txgbe_write_pma(xpcs, TXGBE_MPLLA_CTL0, 0x21);
        txgbe_write_pma(xpcs, TXGBE_MPLLA_CTL3, 0);
-       val = txgbe_read_pma(xpcs, TXGBE_TX_GENCTL1);
-       val = u16_replace_bits(val, 0x5, TXGBE_TX_GENCTL1_VBOOST_LVL);
-       txgbe_write_pma(xpcs, TXGBE_TX_GENCTL1, val);
+       txgbe_modify_pma(xpcs, TXGBE_TX_GENCTL1, TXGBE_TX_GENCTL1_VBOOST_LVL,
+                        FIELD_PREP(TXGBE_TX_GENCTL1_VBOOST_LVL, 0x5));
        txgbe_write_pma(xpcs, TXGBE_MISC_CTL0, TXGBE_MISC_CTL0_PLL |
                        TXGBE_MISC_CTL0_CR_PARA_SEL | TXGBE_MISC_CTL0_RX_VREF(0xF));
        txgbe_write_pma(xpcs, TXGBE_VCO_CAL_LD0, 0x549);
@@ -78,38 +76,29 @@ static void txgbe_pma_config_10gbaser(struct dw_xpcs *xpcs)
 
        txgbe_write_pma(xpcs, TXGBE_RX_EQ_CTL0, TXGBE_RX_EQ_CTL0_CTLE_POLE(2) |
                        TXGBE_RX_EQ_CTL0_CTLE_BOOST(5));
-       val = txgbe_read_pma(xpcs, TXGBE_RX_EQ_ATTN_CTL);
-       val &= ~TXGBE_RX_EQ_ATTN_LVL0;
-       txgbe_write_pma(xpcs, TXGBE_RX_EQ_ATTN_CTL, val);
+       txgbe_modify_pma(xpcs, TXGBE_RX_EQ_ATTN_CTL, TXGBE_RX_EQ_ATTN_LVL0, 0);
        txgbe_write_pma(xpcs, TXGBE_DFE_TAP_CTL0, 0xBE);
-       val = txgbe_read_pma(xpcs, TXGBE_AFE_DFE_ENABLE);
-       val &= ~(TXGBE_DFE_EN_0 | TXGBE_AFE_EN_0);
-       txgbe_write_pma(xpcs, TXGBE_AFE_DFE_ENABLE, val);
-       val = txgbe_read_pma(xpcs, TXGBE_RX_EQ_CTL4);
-       val &= ~TXGBE_RX_EQ_CTL4_CONT_ADAPT0;
-       txgbe_write_pma(xpcs, TXGBE_RX_EQ_CTL4, val);
+       txgbe_modify_pma(xpcs, TXGBE_AFE_DFE_ENABLE,
+                        TXGBE_DFE_EN_0 | TXGBE_AFE_EN_0, 0);
+       txgbe_modify_pma(xpcs, TXGBE_RX_EQ_CTL4, TXGBE_RX_EQ_CTL4_CONT_ADAPT0,
+                        0);
 }
 
 static void txgbe_pma_config_1g(struct dw_xpcs *xpcs)
 {
-       int val;
-
-       val = txgbe_read_pma(xpcs, TXGBE_TX_GENCTL1);
-       val = u16_replace_bits(val, 0x5, TXGBE_TX_GENCTL1_VBOOST_LVL);
-       val &= ~TXGBE_TX_GENCTL1_VBOOST_EN0;
-       txgbe_write_pma(xpcs, TXGBE_TX_GENCTL1, val);
+       txgbe_modify_pma(xpcs, TXGBE_TX_GENCTL1,
+                        TXGBE_TX_GENCTL1_VBOOST_LVL |
+                        TXGBE_TX_GENCTL1_VBOOST_EN0,
+                        FIELD_PREP(TXGBE_TX_GENCTL1_VBOOST_LVL, 0x5));
        txgbe_write_pma(xpcs, TXGBE_MISC_CTL0, TXGBE_MISC_CTL0_PLL |
                        TXGBE_MISC_CTL0_CR_PARA_SEL | TXGBE_MISC_CTL0_RX_VREF(0xF));
 
        txgbe_write_pma(xpcs, TXGBE_RX_EQ_CTL0, TXGBE_RX_EQ_CTL0_VGA1_GAIN(7) |
                        TXGBE_RX_EQ_CTL0_VGA2_GAIN(7) | TXGBE_RX_EQ_CTL0_CTLE_BOOST(6));
-       val = txgbe_read_pma(xpcs, TXGBE_RX_EQ_ATTN_CTL);
-       val &= ~TXGBE_RX_EQ_ATTN_LVL0;
-       txgbe_write_pma(xpcs, TXGBE_RX_EQ_ATTN_CTL, val);
+       txgbe_modify_pma(xpcs, TXGBE_RX_EQ_ATTN_CTL, TXGBE_RX_EQ_ATTN_LVL0, 0);
        txgbe_write_pma(xpcs, TXGBE_DFE_TAP_CTL0, 0);
-       val = txgbe_read_pma(xpcs, TXGBE_RX_GEN_CTL3);
-       val = u16_replace_bits(val, 0x4, TXGBE_RX_GEN_CTL3_LOS_TRSHLD0);
-       txgbe_write_pma(xpcs, TXGBE_RX_GEN_CTL3, val);
+       txgbe_modify_pma(xpcs, TXGBE_RX_GEN_CTL3, TXGBE_RX_GEN_CTL3_LOS_TRSHLD0,
+                        FIELD_PREP(TXGBE_RX_GEN_CTL3_LOS_TRSHLD0, 0x4));
 
        txgbe_write_pma(xpcs, TXGBE_MPLLA_CTL0, 0x20);
        txgbe_write_pma(xpcs, TXGBE_MPLLA_CTL3, 0x46);
@@ -172,7 +161,7 @@ static bool txgbe_xpcs_mode_quirk(struct dw_xpcs *xpcs)
 
 int txgbe_xpcs_switch_mode(struct dw_xpcs *xpcs, phy_interface_t interface)
 {
-       int val, ret;
+       int ret;
 
        switch (interface) {
        case PHY_INTERFACE_MODE_10GBASER:
@@ -194,9 +183,8 @@ int txgbe_xpcs_switch_mode(struct dw_xpcs *xpcs, phy_interface_t interface)
 
        if (interface == PHY_INTERFACE_MODE_10GBASER) {
                xpcs_write(xpcs, MDIO_MMD_PCS, MDIO_CTRL2, MDIO_PCS_CTRL2_10GBR);
-               val = xpcs_read(xpcs, MDIO_MMD_PMAPMD, MDIO_CTRL1);
-               val |= MDIO_CTRL1_SPEED10G;
-               xpcs_write(xpcs, MDIO_MMD_PMAPMD, MDIO_CTRL1, val);
+               xpcs_modify(xpcs, MDIO_MMD_PMAPMD, MDIO_CTRL1,
+                           MDIO_CTRL1_SPEED10G, MDIO_CTRL1_SPEED10G);
                txgbe_pma_config_10gbaser(xpcs);
        } else {
                xpcs_write(xpcs, MDIO_MMD_PCS, MDIO_CTRL2, MDIO_PCS_CTRL2_10GBX);
index f55bc180c6249b46b6cb3763a53925e2b395ba39..5ac8262ac264e957cf95646b2775b1ff5121a3ec 100644 (file)
@@ -175,6 +175,11 @@ int xpcs_write(struct dw_xpcs *xpcs, int dev, u32 reg, u16 val)
        return mdiodev_c45_write(xpcs->mdiodev, dev, reg, val);
 }
 
+int xpcs_modify(struct dw_xpcs *xpcs, int dev, u32 reg, u16 mask, u16 set)
+{
+       return mdiodev_c45_modify(xpcs->mdiodev, dev, reg, mask, set);
+}
+
 static int xpcs_modify_changed(struct dw_xpcs *xpcs, int dev, u32 reg,
                               u16 mask, u16 set)
 {
@@ -192,6 +197,12 @@ static int xpcs_write_vendor(struct dw_xpcs *xpcs, int dev, int reg,
        return xpcs_write(xpcs, dev, DW_VENDOR | reg, val);
 }
 
+static int xpcs_modify_vendor(struct dw_xpcs *xpcs, int dev, int reg, u16 mask,
+                             u16 set)
+{
+       return xpcs_modify(xpcs, dev, DW_VENDOR | reg, mask, set);
+}
+
 int xpcs_read_vpcs(struct dw_xpcs *xpcs, int reg)
 {
        return xpcs_read_vendor(xpcs, MDIO_MMD_PCS, reg);
@@ -202,6 +213,11 @@ int xpcs_write_vpcs(struct dw_xpcs *xpcs, int reg, u16 val)
        return xpcs_write_vendor(xpcs, MDIO_MMD_PCS, reg, val);
 }
 
+static int xpcs_modify_vpcs(struct dw_xpcs *xpcs, int reg, u16 mask, u16 val)
+{
+       return xpcs_modify_vendor(xpcs, MDIO_MMD_PCS, reg, mask, val);
+}
+
 static int xpcs_poll_reset(struct dw_xpcs *xpcs, int dev)
 {
        /* Poll until the reset bit clears (50ms per retry == 0.6 sec) */
@@ -326,30 +342,17 @@ static void xpcs_config_usxgmii(struct dw_xpcs *xpcs, int speed)
                return;
        }
 
-       ret = xpcs_read_vpcs(xpcs, MDIO_CTRL1);
+       ret = xpcs_modify_vpcs(xpcs, MDIO_CTRL1, DW_USXGMII_EN, DW_USXGMII_EN);
        if (ret < 0)
                goto out;
 
-       ret = xpcs_write_vpcs(xpcs, MDIO_CTRL1, ret | DW_USXGMII_EN);
+       ret = xpcs_modify(xpcs, MDIO_MMD_VEND2, MDIO_CTRL1, DW_USXGMII_SS_MASK,
+                         speed_sel | DW_USXGMII_FULL);
        if (ret < 0)
                goto out;
 
-       ret = xpcs_read(xpcs, MDIO_MMD_VEND2, MDIO_CTRL1);
-       if (ret < 0)
-               goto out;
-
-       ret &= ~DW_USXGMII_SS_MASK;
-       ret |= speed_sel | DW_USXGMII_FULL;
-
-       ret = xpcs_write(xpcs, MDIO_MMD_VEND2, MDIO_CTRL1, ret);
-       if (ret < 0)
-               goto out;
-
-       ret = xpcs_read_vpcs(xpcs, MDIO_CTRL1);
-       if (ret < 0)
-               goto out;
-
-       ret = xpcs_write_vpcs(xpcs, MDIO_CTRL1, ret | DW_USXGMII_RST);
+       ret = xpcs_modify_vpcs(xpcs, MDIO_CTRL1, DW_USXGMII_RST,
+                              DW_USXGMII_RST);
        if (ret < 0)
                goto out;
 
@@ -413,13 +416,9 @@ static int xpcs_config_aneg_c73(struct dw_xpcs *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);
+       return xpcs_modify(xpcs, MDIO_MMD_AN, MDIO_CTRL1,
+                          MDIO_AN_CTRL1_ENABLE | MDIO_AN_CTRL1_RESTART,
+                          MDIO_AN_CTRL1_ENABLE | MDIO_AN_CTRL1_RESTART);
 }
 
 static int xpcs_aneg_done_c73(struct dw_xpcs *xpcs,
@@ -581,40 +580,31 @@ EXPORT_SYMBOL_GPL(xpcs_get_interfaces);
 
 int xpcs_config_eee(struct dw_xpcs *xpcs, int mult_fact_100ns, int enable)
 {
+       u16 mask, val;
        int ret;
 
-       ret = xpcs_read(xpcs, MDIO_MMD_VEND2, DW_VR_MII_EEE_MCTRL0);
-       if (ret < 0)
-               return ret;
+       mask = DW_VR_MII_EEE_LTX_EN | DW_VR_MII_EEE_LRX_EN |
+              DW_VR_MII_EEE_TX_QUIET_EN | DW_VR_MII_EEE_RX_QUIET_EN |
+              DW_VR_MII_EEE_TX_EN_CTRL | DW_VR_MII_EEE_RX_EN_CTRL |
+              DW_VR_MII_EEE_MULT_FACT_100NS;
 
-       if (enable) {
-       /* Enable EEE */
-               ret = DW_VR_MII_EEE_LTX_EN | DW_VR_MII_EEE_LRX_EN |
+       if (enable)
+               val = DW_VR_MII_EEE_LTX_EN | DW_VR_MII_EEE_LRX_EN |
                      DW_VR_MII_EEE_TX_QUIET_EN | DW_VR_MII_EEE_RX_QUIET_EN |
                      DW_VR_MII_EEE_TX_EN_CTRL | DW_VR_MII_EEE_RX_EN_CTRL |
                      FIELD_PREP(DW_VR_MII_EEE_MULT_FACT_100NS,
                                 mult_fact_100ns);
-       } else {
-               ret &= ~(DW_VR_MII_EEE_LTX_EN | DW_VR_MII_EEE_LRX_EN |
-                      DW_VR_MII_EEE_TX_QUIET_EN | DW_VR_MII_EEE_RX_QUIET_EN |
-                      DW_VR_MII_EEE_TX_EN_CTRL | DW_VR_MII_EEE_RX_EN_CTRL |
-                      DW_VR_MII_EEE_MULT_FACT_100NS);
-       }
-
-       ret = xpcs_write(xpcs, MDIO_MMD_VEND2, DW_VR_MII_EEE_MCTRL0, ret);
-       if (ret < 0)
-               return ret;
+       else
+               val = 0;
 
-       ret = xpcs_read(xpcs, MDIO_MMD_VEND2, DW_VR_MII_EEE_MCTRL1);
+       ret = xpcs_modify(xpcs, MDIO_MMD_VEND2, DW_VR_MII_EEE_MCTRL0, mask,
+                         val);
        if (ret < 0)
                return ret;
 
-       if (enable)
-               ret |= DW_VR_MII_EEE_TRN_LPI;
-       else
-               ret &= ~DW_VR_MII_EEE_TRN_LPI;
-
-       return xpcs_write(xpcs, MDIO_MMD_VEND2, DW_VR_MII_EEE_MCTRL1, ret);
+       return xpcs_modify(xpcs, MDIO_MMD_VEND2, DW_VR_MII_EEE_MCTRL1,
+                          DW_VR_MII_EEE_TRN_LPI,
+                          enable ? DW_VR_MII_EEE_TRN_LPI : 0);
 }
 EXPORT_SYMBOL_GPL(xpcs_config_eee);
 
@@ -646,6 +636,7 @@ static int xpcs_config_aneg_c37_sgmii(struct dw_xpcs *xpcs,
                                      unsigned int neg_mode)
 {
        int ret, mdio_ctrl, tx_conf;
+       u16 mask, val;
 
        if (xpcs->info.pma == WX_TXGBE_XPCS_PMA_10G_ID)
                xpcs_write_vpcs(xpcs, DW_VR_XS_PCS_DIG_CTRL1, DW_CL37_BP | DW_EN_VSMMD1);
@@ -677,38 +668,35 @@ static int xpcs_config_aneg_c37_sgmii(struct dw_xpcs *xpcs,
                        return ret;
        }
 
-       ret = xpcs_read(xpcs, MDIO_MMD_VEND2, DW_VR_MII_AN_CTRL);
-       if (ret < 0)
-               return ret;
+       mask = DW_VR_MII_PCS_MODE_MASK | DW_VR_MII_TX_CONFIG_MASK;
+       val = FIELD_PREP(DW_VR_MII_PCS_MODE_MASK,
+                        DW_VR_MII_PCS_MODE_C37_SGMII);
 
-       ret &= ~(DW_VR_MII_PCS_MODE_MASK | DW_VR_MII_TX_CONFIG_MASK);
-       ret |= FIELD_PREP(DW_VR_MII_PCS_MODE_MASK,
-                         DW_VR_MII_PCS_MODE_C37_SGMII);
        if (xpcs->info.pma == WX_TXGBE_XPCS_PMA_10G_ID) {
-               ret |= DW_VR_MII_AN_CTRL_8BIT;
+               mask |= DW_VR_MII_AN_CTRL_8BIT;
+               val |= DW_VR_MII_AN_CTRL_8BIT;
                /* Hardware requires it to be PHY side SGMII */
                tx_conf = DW_VR_MII_TX_CONFIG_PHY_SIDE_SGMII;
        } else {
                tx_conf = DW_VR_MII_TX_CONFIG_MAC_SIDE_SGMII;
        }
-       ret |= FIELD_PREP(DW_VR_MII_TX_CONFIG_MASK, tx_conf);
-       ret = xpcs_write(xpcs, MDIO_MMD_VEND2, DW_VR_MII_AN_CTRL, ret);
-       if (ret < 0)
-               return ret;
 
-       ret = xpcs_read(xpcs, MDIO_MMD_VEND2, DW_VR_MII_DIG_CTRL1);
+       val |= FIELD_PREP(DW_VR_MII_TX_CONFIG_MASK, tx_conf);
+
+       ret = xpcs_modify(xpcs, MDIO_MMD_VEND2, DW_VR_MII_AN_CTRL, mask, val);
        if (ret < 0)
                return ret;
 
+       mask = DW_VR_MII_DIG_CTRL1_MAC_AUTO_SW;
        if (neg_mode == PHYLINK_PCS_NEG_INBAND_ENABLED)
-               ret |= DW_VR_MII_DIG_CTRL1_MAC_AUTO_SW;
-       else
-               ret &= ~DW_VR_MII_DIG_CTRL1_MAC_AUTO_SW;
+               val = DW_VR_MII_DIG_CTRL1_MAC_AUTO_SW;
 
-       if (xpcs->info.pma == WX_TXGBE_XPCS_PMA_10G_ID)
-               ret |= DW_VR_MII_DIG_CTRL1_PHY_MODE_CTRL;
+       if (xpcs->info.pma == WX_TXGBE_XPCS_PMA_10G_ID) {
+               mask |= DW_VR_MII_DIG_CTRL1_PHY_MODE_CTRL;
+               val |= DW_VR_MII_DIG_CTRL1_PHY_MODE_CTRL;
+       }
 
-       ret = xpcs_write(xpcs, MDIO_MMD_VEND2, DW_VR_MII_DIG_CTRL1, ret);
+       ret = xpcs_modify(xpcs, MDIO_MMD_VEND2, DW_VR_MII_DIG_CTRL1, mask, val);
        if (ret < 0)
                return ret;
 
@@ -726,6 +714,7 @@ static int xpcs_config_aneg_c37_1000basex(struct dw_xpcs *xpcs,
        phy_interface_t interface = PHY_INTERFACE_MODE_1000BASEX;
        int ret, mdio_ctrl, adv;
        bool changed = 0;
+       u16 mask, val;
 
        if (xpcs->info.pma == WX_TXGBE_XPCS_PMA_10G_ID)
                xpcs_write_vpcs(xpcs, DW_VR_XS_PCS_DIG_CTRL1, DW_CL37_BP | DW_EN_VSMMD1);
@@ -746,14 +735,16 @@ static int xpcs_config_aneg_c37_1000basex(struct dw_xpcs *xpcs,
                        return ret;
        }
 
-       ret = xpcs_read(xpcs, MDIO_MMD_VEND2, DW_VR_MII_AN_CTRL);
-       if (ret < 0)
-               return ret;
+       mask = DW_VR_MII_PCS_MODE_MASK;
+       val = FIELD_PREP(DW_VR_MII_PCS_MODE_MASK,
+                        DW_VR_MII_PCS_MODE_C37_1000BASEX);
+
+       if (!xpcs->pcs.poll) {
+               mask |= DW_VR_MII_AN_INTR_EN;
+               val |= DW_VR_MII_AN_INTR_EN;
+       }
 
-       ret &= ~DW_VR_MII_PCS_MODE_MASK;
-       if (!xpcs->pcs.poll)
-               ret |= DW_VR_MII_AN_INTR_EN;
-       ret = xpcs_write(xpcs, MDIO_MMD_VEND2, DW_VR_MII_AN_CTRL, ret);
+       ret = xpcs_modify(xpcs, MDIO_MMD_VEND2, DW_VR_MII_AN_CTRL, mask, val);
        if (ret < 0)
                return ret;
 
@@ -790,22 +781,16 @@ static int xpcs_config_2500basex(struct dw_xpcs *xpcs)
 {
        int ret;
 
-       ret = xpcs_read(xpcs, MDIO_MMD_VEND2, DW_VR_MII_DIG_CTRL1);
-       if (ret < 0)
-               return ret;
-       ret |= DW_VR_MII_DIG_CTRL1_2G5_EN;
-       ret &= ~DW_VR_MII_DIG_CTRL1_MAC_AUTO_SW;
-       ret = xpcs_write(xpcs, MDIO_MMD_VEND2, DW_VR_MII_DIG_CTRL1, ret);
+       ret = xpcs_modify(xpcs, MDIO_MMD_VEND2, DW_VR_MII_DIG_CTRL1,
+                         DW_VR_MII_DIG_CTRL1_2G5_EN |
+                         DW_VR_MII_DIG_CTRL1_MAC_AUTO_SW,
+                         DW_VR_MII_DIG_CTRL1_2G5_EN);
        if (ret < 0)
                return ret;
 
-       ret = xpcs_read(xpcs, MDIO_MMD_VEND2, DW_VR_MII_MMD_CTRL);
-       if (ret < 0)
-               return ret;
-       ret &= ~AN_CL37_EN;
-       ret |= SGMII_SPEED_SS6;
-       ret &= ~SGMII_SPEED_SS13;
-       return xpcs_write(xpcs, MDIO_MMD_VEND2, DW_VR_MII_MMD_CTRL, ret);
+       return xpcs_modify(xpcs, MDIO_MMD_VEND2, DW_VR_MII_MMD_CTRL,
+                          AN_CL37_EN | SGMII_SPEED_SS6 | SGMII_SPEED_SS13,
+                          SGMII_SPEED_SS6);
 }
 
 static int xpcs_do_config(struct dw_xpcs *xpcs, phy_interface_t interface,
@@ -1179,13 +1164,9 @@ static void xpcs_link_up(struct phylink_pcs *pcs, unsigned int neg_mode,
 static void xpcs_an_restart(struct phylink_pcs *pcs)
 {
        struct dw_xpcs *xpcs = phylink_pcs_to_xpcs(pcs);
-       int ret;
 
-       ret = xpcs_read(xpcs, MDIO_MMD_VEND2, MDIO_CTRL1);
-       if (ret >= 0) {
-               ret |= BMCR_ANRESTART;
-               xpcs_write(xpcs, MDIO_MMD_VEND2, MDIO_CTRL1, ret);
-       }
+       xpcs_modify(xpcs, MDIO_MMD_VEND2, MDIO_CTRL1, BMCR_ANRESTART,
+                   BMCR_ANRESTART);
 }
 
 static int xpcs_read_ids(struct dw_xpcs *xpcs)
index 8902485730a23dc1cc2ef3c0c4829581c765a3e8..b80b956ec286512a72eb895b9572ee9b99f7b60e 100644 (file)
@@ -139,6 +139,7 @@ struct dw_xpcs {
 
 int xpcs_read(struct dw_xpcs *xpcs, int dev, u32 reg);
 int xpcs_write(struct dw_xpcs *xpcs, int dev, u32 reg, u16 val);
+int xpcs_modify(struct dw_xpcs *xpcs, int dev, u32 reg, u16 mask, u16 set);
 int xpcs_read_vpcs(struct dw_xpcs *xpcs, int reg);
 int xpcs_write_vpcs(struct dw_xpcs *xpcs, int reg, u16 val);
 int nxp_sja1105_sgmii_pma_config(struct dw_xpcs *xpcs);