drivers: net: xgene: ethtool: Add get/set_pauseparam
authorIyappan Subramanian <isubramanian@apm.com>
Fri, 2 Dec 2016 00:41:44 +0000 (16:41 -0800)
committerDavid S. Miller <davem@davemloft.net>
Sat, 3 Dec 2016 20:46:50 +0000 (15:46 -0500)
This patch adds get_pauseparam and set_pauseparam functions.

Signed-off-by: Iyappan Subramanian <isubramanian@apm.com>
Signed-off-by: Quan Nguyen <qnguyen@apm.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
drivers/net/ethernet/apm/xgene/xgene_enet_ethtool.c

index d372d4235c810494276c757ac798b024111a31c8..28fdedc30b746dae3667f6f274a2c24068b21749 100644 (file)
@@ -163,6 +163,74 @@ static void xgene_get_ethtool_stats(struct net_device *ndev,
                *data++ = *(u64 *)(pdata + gstrings_stats[i].offset);
 }
 
+static void xgene_get_pauseparam(struct net_device *ndev,
+                                struct ethtool_pauseparam *pp)
+{
+       struct xgene_enet_pdata *pdata = netdev_priv(ndev);
+
+       pp->autoneg = pdata->pause_autoneg;
+       pp->tx_pause = pdata->tx_pause;
+       pp->rx_pause = pdata->rx_pause;
+}
+
+static int xgene_set_pauseparam(struct net_device *ndev,
+                               struct ethtool_pauseparam *pp)
+{
+       struct xgene_enet_pdata *pdata = netdev_priv(ndev);
+       struct phy_device *phydev = ndev->phydev;
+       u32 oldadv, newadv;
+
+       if (pdata->phy_mode == PHY_INTERFACE_MODE_RGMII ||
+           pdata->phy_mode == PHY_INTERFACE_MODE_SGMII) {
+               if (!phydev)
+                       return -EINVAL;
+
+               if (!(phydev->supported & SUPPORTED_Pause) ||
+                   (!(phydev->supported & SUPPORTED_Asym_Pause) &&
+                    pp->rx_pause != pp->tx_pause))
+                       return -EINVAL;
+
+               pdata->pause_autoneg = pp->autoneg;
+               pdata->tx_pause = pp->tx_pause;
+               pdata->rx_pause = pp->rx_pause;
+
+               oldadv = phydev->advertising;
+               newadv = oldadv & ~(ADVERTISED_Pause | ADVERTISED_Asym_Pause);
+
+               if (pp->rx_pause)
+                       newadv |= ADVERTISED_Pause | ADVERTISED_Asym_Pause;
+
+               if (pp->tx_pause)
+                       newadv ^= ADVERTISED_Asym_Pause;
+
+               if (oldadv ^ newadv) {
+                       phydev->advertising = newadv;
+
+                       if (phydev->autoneg)
+                               return phy_start_aneg(phydev);
+
+                       if (!pp->autoneg) {
+                               pdata->mac_ops->flowctl_tx(pdata,
+                                                          pdata->tx_pause);
+                               pdata->mac_ops->flowctl_rx(pdata,
+                                                          pdata->rx_pause);
+                       }
+               }
+
+       } else {
+               if (pp->autoneg)
+                       return -EINVAL;
+
+               pdata->tx_pause = pp->tx_pause;
+               pdata->rx_pause = pp->rx_pause;
+
+               pdata->mac_ops->flowctl_tx(pdata, pdata->tx_pause);
+               pdata->mac_ops->flowctl_rx(pdata, pdata->rx_pause);
+       }
+
+       return 0;
+}
+
 static const struct ethtool_ops xgene_ethtool_ops = {
        .get_drvinfo = xgene_get_drvinfo,
        .get_link = ethtool_op_get_link,
@@ -171,6 +239,8 @@ static const struct ethtool_ops xgene_ethtool_ops = {
        .get_ethtool_stats = xgene_get_ethtool_stats,
        .get_link_ksettings = xgene_get_link_ksettings,
        .set_link_ksettings = xgene_set_link_ksettings,
+       .get_pauseparam = xgene_get_pauseparam,
+       .set_pauseparam = xgene_set_pauseparam
 };
 
 void xgene_enet_set_ethtool_ops(struct net_device *ndev)