ethtool.c 5.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233
  1. /* This program is free software; you can redistribute it and/or modify
  2. * it under the terms of the GNU General Public License as published by
  3. * the Free Software Foundation; version 2 of the License
  4. *
  5. * This program is distributed in the hope that it will be useful,
  6. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  7. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  8. * GNU General Public License for more details.
  9. *
  10. * Copyright (C) 2009-2015 John Crispin <[email protected]>
  11. * Copyright (C) 2009-2015 Felix Fietkau <[email protected]>
  12. * Copyright (C) 2013-2015 Michael Lee <[email protected]>
  13. */
  14. #include "mtk_eth_soc.h"
  15. static const char fe_gdma_str[][ETH_GSTRING_LEN] = {
  16. #define _FE(x...) # x,
  17. FE_STAT_REG_DECLARE
  18. #undef _FE
  19. };
  20. static int fe_get_settings(struct net_device *dev,
  21. struct ethtool_cmd *cmd)
  22. {
  23. struct fe_priv *priv = netdev_priv(dev);
  24. int err;
  25. if (!priv->phy_dev)
  26. goto out_gset;
  27. if (priv->phy_flags == FE_PHY_FLAG_ATTACH) {
  28. err = phy_read_status(priv->phy_dev);
  29. if (err)
  30. goto out_gset;
  31. }
  32. return phy_ethtool_gset(priv->phy_dev, cmd);
  33. out_gset:
  34. return -ENODEV;
  35. }
  36. static int fe_set_settings(struct net_device *dev,
  37. struct ethtool_cmd *cmd)
  38. {
  39. struct fe_priv *priv = netdev_priv(dev);
  40. if (!priv->phy_dev)
  41. goto out_sset;
  42. if (cmd->phy_address != priv->phy_dev->mdio.addr) {
  43. if (priv->phy->phy_node[cmd->phy_address]) {
  44. priv->phy_dev = priv->phy->phy[cmd->phy_address];
  45. priv->phy_flags = FE_PHY_FLAG_PORT;
  46. } else if (priv->mii_bus && mdiobus_get_phy(priv->mii_bus, cmd->phy_address)) {
  47. priv->phy_dev = mdiobus_get_phy(priv->mii_bus, cmd->phy_address);
  48. priv->phy_flags = FE_PHY_FLAG_ATTACH;
  49. } else {
  50. goto out_sset;
  51. }
  52. }
  53. return phy_ethtool_sset(priv->phy_dev, cmd);
  54. out_sset:
  55. return -ENODEV;
  56. }
  57. static void fe_get_drvinfo(struct net_device *dev,
  58. struct ethtool_drvinfo *info)
  59. {
  60. struct fe_priv *priv = netdev_priv(dev);
  61. struct fe_soc_data *soc = priv->soc;
  62. strlcpy(info->driver, priv->device->driver->name, sizeof(info->driver));
  63. strlcpy(info->version, MTK_FE_DRV_VERSION, sizeof(info->version));
  64. strlcpy(info->bus_info, dev_name(priv->device), sizeof(info->bus_info));
  65. if (soc->reg_table[FE_REG_FE_COUNTER_BASE])
  66. info->n_stats = ARRAY_SIZE(fe_gdma_str);
  67. }
  68. static u32 fe_get_msglevel(struct net_device *dev)
  69. {
  70. struct fe_priv *priv = netdev_priv(dev);
  71. return priv->msg_enable;
  72. }
  73. static void fe_set_msglevel(struct net_device *dev, u32 value)
  74. {
  75. struct fe_priv *priv = netdev_priv(dev);
  76. priv->msg_enable = value;
  77. }
  78. static int fe_nway_reset(struct net_device *dev)
  79. {
  80. struct fe_priv *priv = netdev_priv(dev);
  81. if (!priv->phy_dev)
  82. goto out_nway_reset;
  83. return genphy_restart_aneg(priv->phy_dev);
  84. out_nway_reset:
  85. return -EOPNOTSUPP;
  86. }
  87. static u32 fe_get_link(struct net_device *dev)
  88. {
  89. struct fe_priv *priv = netdev_priv(dev);
  90. int err;
  91. if (!priv->phy_dev)
  92. goto out_get_link;
  93. if (priv->phy_flags == FE_PHY_FLAG_ATTACH) {
  94. err = genphy_update_link(priv->phy_dev);
  95. if (err)
  96. goto out_get_link;
  97. }
  98. return priv->phy_dev->link;
  99. out_get_link:
  100. return ethtool_op_get_link(dev);
  101. }
  102. static int fe_set_ringparam(struct net_device *dev,
  103. struct ethtool_ringparam *ring)
  104. {
  105. struct fe_priv *priv = netdev_priv(dev);
  106. if ((ring->tx_pending < 2) ||
  107. (ring->rx_pending < 2) ||
  108. (ring->rx_pending > MAX_DMA_DESC) ||
  109. (ring->tx_pending > MAX_DMA_DESC))
  110. return -EINVAL;
  111. dev->netdev_ops->ndo_stop(dev);
  112. priv->tx_ring.tx_ring_size = BIT(fls(ring->tx_pending) - 1);
  113. priv->rx_ring.rx_ring_size = BIT(fls(ring->rx_pending) - 1);
  114. dev->netdev_ops->ndo_open(dev);
  115. return 0;
  116. }
  117. static void fe_get_ringparam(struct net_device *dev,
  118. struct ethtool_ringparam *ring)
  119. {
  120. struct fe_priv *priv = netdev_priv(dev);
  121. ring->rx_max_pending = MAX_DMA_DESC;
  122. ring->tx_max_pending = MAX_DMA_DESC;
  123. ring->rx_pending = priv->rx_ring.rx_ring_size;
  124. ring->tx_pending = priv->tx_ring.tx_ring_size;
  125. }
  126. static void fe_get_strings(struct net_device *dev, u32 stringset, u8 *data)
  127. {
  128. switch (stringset) {
  129. case ETH_SS_STATS:
  130. memcpy(data, *fe_gdma_str, sizeof(fe_gdma_str));
  131. break;
  132. }
  133. }
  134. static int fe_get_sset_count(struct net_device *dev, int sset)
  135. {
  136. switch (sset) {
  137. case ETH_SS_STATS:
  138. return ARRAY_SIZE(fe_gdma_str);
  139. default:
  140. return -EOPNOTSUPP;
  141. }
  142. }
  143. static void fe_get_ethtool_stats(struct net_device *dev,
  144. struct ethtool_stats *stats, u64 *data)
  145. {
  146. struct fe_priv *priv = netdev_priv(dev);
  147. struct fe_hw_stats *hwstats = priv->hw_stats;
  148. u64 *data_src, *data_dst;
  149. unsigned int start;
  150. int i;
  151. if (netif_running(dev) && netif_device_present(dev)) {
  152. if (spin_trylock(&hwstats->stats_lock)) {
  153. fe_stats_update(priv);
  154. spin_unlock(&hwstats->stats_lock);
  155. }
  156. }
  157. do {
  158. data_src = &hwstats->tx_bytes;
  159. data_dst = data;
  160. start = u64_stats_fetch_begin_irq(&hwstats->syncp);
  161. for (i = 0; i < ARRAY_SIZE(fe_gdma_str); i++)
  162. *data_dst++ = *data_src++;
  163. } while (u64_stats_fetch_retry_irq(&hwstats->syncp, start));
  164. }
  165. static struct ethtool_ops fe_ethtool_ops = {
  166. .get_settings = fe_get_settings,
  167. .set_settings = fe_set_settings,
  168. .get_drvinfo = fe_get_drvinfo,
  169. .get_msglevel = fe_get_msglevel,
  170. .set_msglevel = fe_set_msglevel,
  171. .nway_reset = fe_nway_reset,
  172. .get_link = fe_get_link,
  173. .set_ringparam = fe_set_ringparam,
  174. .get_ringparam = fe_get_ringparam,
  175. };
  176. void fe_set_ethtool_ops(struct net_device *netdev)
  177. {
  178. struct fe_priv *priv = netdev_priv(netdev);
  179. struct fe_soc_data *soc = priv->soc;
  180. if (soc->reg_table[FE_REG_FE_COUNTER_BASE]) {
  181. fe_ethtool_ops.get_strings = fe_get_strings;
  182. fe_ethtool_ops.get_sset_count = fe_get_sset_count;
  183. fe_ethtool_ops.get_ethtool_stats = fe_get_ethtool_stats;
  184. }
  185. netdev->ethtool_ops = &fe_ethtool_ops;
  186. }