ixgb_ethtool.c 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739
  1. /*******************************************************************************
  2. Intel PRO/10GbE Linux driver
  3. Copyright(c) 1999 - 2006 Intel Corporation.
  4. This program is free software; you can redistribute it and/or modify it
  5. under the terms and conditions of the GNU General Public License,
  6. version 2, as published by the Free Software Foundation.
  7. This program is distributed in the hope it will be useful, but WITHOUT
  8. ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
  9. FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
  10. more details.
  11. You should have received a copy of the GNU General Public License along with
  12. this program; if not, write to the Free Software Foundation, Inc.,
  13. 51 Franklin St - Fifth Floor, Boston, MA 02110-1301 USA.
  14. The full GNU General Public License is included in this distribution in
  15. the file called "COPYING".
  16. Contact Information:
  17. Linux NICS <linux.nics@intel.com>
  18. e1000-devel Mailing List <e1000-devel@lists.sourceforge.net>
  19. Intel Corporation, 5200 N.E. Elam Young Parkway, Hillsboro, OR 97124-6497
  20. *******************************************************************************/
  21. /* ethtool support for ixgb */
  22. #include "ixgb.h"
  23. #include <asm/uaccess.h>
  24. extern char ixgb_driver_name[];
  25. extern char ixgb_driver_version[];
  26. extern int ixgb_up(struct ixgb_adapter *adapter);
  27. extern void ixgb_down(struct ixgb_adapter *adapter, boolean_t kill_watchdog);
  28. extern void ixgb_reset(struct ixgb_adapter *adapter);
  29. extern int ixgb_setup_rx_resources(struct ixgb_adapter *adapter);
  30. extern int ixgb_setup_tx_resources(struct ixgb_adapter *adapter);
  31. extern void ixgb_free_rx_resources(struct ixgb_adapter *adapter);
  32. extern void ixgb_free_tx_resources(struct ixgb_adapter *adapter);
  33. extern void ixgb_update_stats(struct ixgb_adapter *adapter);
  34. #define IXGB_ALL_RAR_ENTRIES 16
  35. struct ixgb_stats {
  36. char stat_string[ETH_GSTRING_LEN];
  37. int sizeof_stat;
  38. int stat_offset;
  39. };
  40. #define IXGB_STAT(m) sizeof(((struct ixgb_adapter *)0)->m), \
  41. offsetof(struct ixgb_adapter, m)
  42. static struct ixgb_stats ixgb_gstrings_stats[] = {
  43. {"rx_packets", IXGB_STAT(net_stats.rx_packets)},
  44. {"tx_packets", IXGB_STAT(net_stats.tx_packets)},
  45. {"rx_bytes", IXGB_STAT(net_stats.rx_bytes)},
  46. {"tx_bytes", IXGB_STAT(net_stats.tx_bytes)},
  47. {"rx_errors", IXGB_STAT(net_stats.rx_errors)},
  48. {"tx_errors", IXGB_STAT(net_stats.tx_errors)},
  49. {"rx_dropped", IXGB_STAT(net_stats.rx_dropped)},
  50. {"tx_dropped", IXGB_STAT(net_stats.tx_dropped)},
  51. {"multicast", IXGB_STAT(net_stats.multicast)},
  52. {"collisions", IXGB_STAT(net_stats.collisions)},
  53. /* { "rx_length_errors", IXGB_STAT(net_stats.rx_length_errors) }, */
  54. {"rx_over_errors", IXGB_STAT(net_stats.rx_over_errors)},
  55. {"rx_crc_errors", IXGB_STAT(net_stats.rx_crc_errors)},
  56. {"rx_frame_errors", IXGB_STAT(net_stats.rx_frame_errors)},
  57. {"rx_fifo_errors", IXGB_STAT(net_stats.rx_fifo_errors)},
  58. {"rx_missed_errors", IXGB_STAT(net_stats.rx_missed_errors)},
  59. {"tx_aborted_errors", IXGB_STAT(net_stats.tx_aborted_errors)},
  60. {"tx_carrier_errors", IXGB_STAT(net_stats.tx_carrier_errors)},
  61. {"tx_fifo_errors", IXGB_STAT(net_stats.tx_fifo_errors)},
  62. {"tx_heartbeat_errors", IXGB_STAT(net_stats.tx_heartbeat_errors)},
  63. {"tx_window_errors", IXGB_STAT(net_stats.tx_window_errors)},
  64. {"tx_deferred_ok", IXGB_STAT(stats.dc)},
  65. {"tx_timeout_count", IXGB_STAT(tx_timeout_count) },
  66. {"tx_restart_queue", IXGB_STAT(restart_queue) },
  67. {"rx_long_length_errors", IXGB_STAT(stats.roc)},
  68. {"rx_short_length_errors", IXGB_STAT(stats.ruc)},
  69. #ifdef NETIF_F_TSO
  70. {"tx_tcp_seg_good", IXGB_STAT(stats.tsctc)},
  71. {"tx_tcp_seg_failed", IXGB_STAT(stats.tsctfc)},
  72. #endif
  73. {"rx_flow_control_xon", IXGB_STAT(stats.xonrxc)},
  74. {"rx_flow_control_xoff", IXGB_STAT(stats.xoffrxc)},
  75. {"tx_flow_control_xon", IXGB_STAT(stats.xontxc)},
  76. {"tx_flow_control_xoff", IXGB_STAT(stats.xofftxc)},
  77. {"rx_csum_offload_good", IXGB_STAT(hw_csum_rx_good)},
  78. {"rx_csum_offload_errors", IXGB_STAT(hw_csum_rx_error)},
  79. {"tx_csum_offload_good", IXGB_STAT(hw_csum_tx_good)},
  80. {"tx_csum_offload_errors", IXGB_STAT(hw_csum_tx_error)}
  81. };
  82. #define IXGB_STATS_LEN \
  83. sizeof(ixgb_gstrings_stats) / sizeof(struct ixgb_stats)
  84. static int
  85. ixgb_get_settings(struct net_device *netdev, struct ethtool_cmd *ecmd)
  86. {
  87. struct ixgb_adapter *adapter = netdev_priv(netdev);
  88. ecmd->supported = (SUPPORTED_10000baseT_Full | SUPPORTED_FIBRE);
  89. ecmd->advertising = (ADVERTISED_10000baseT_Full | ADVERTISED_FIBRE);
  90. ecmd->port = PORT_FIBRE;
  91. ecmd->transceiver = XCVR_EXTERNAL;
  92. if(netif_carrier_ok(adapter->netdev)) {
  93. ecmd->speed = SPEED_10000;
  94. ecmd->duplex = DUPLEX_FULL;
  95. } else {
  96. ecmd->speed = -1;
  97. ecmd->duplex = -1;
  98. }
  99. ecmd->autoneg = AUTONEG_DISABLE;
  100. return 0;
  101. }
  102. static void ixgb_set_speed_duplex(struct net_device *netdev)
  103. {
  104. struct ixgb_adapter *adapter = netdev_priv(netdev);
  105. /* be optimistic about our link, since we were up before */
  106. adapter->link_speed = 10000;
  107. adapter->link_duplex = FULL_DUPLEX;
  108. netif_carrier_on(netdev);
  109. netif_wake_queue(netdev);
  110. }
  111. static int
  112. ixgb_set_settings(struct net_device *netdev, struct ethtool_cmd *ecmd)
  113. {
  114. struct ixgb_adapter *adapter = netdev_priv(netdev);
  115. if(ecmd->autoneg == AUTONEG_ENABLE ||
  116. ecmd->speed + ecmd->duplex != SPEED_10000 + DUPLEX_FULL)
  117. return -EINVAL;
  118. if(netif_running(adapter->netdev)) {
  119. ixgb_down(adapter, TRUE);
  120. ixgb_reset(adapter);
  121. ixgb_up(adapter);
  122. ixgb_set_speed_duplex(netdev);
  123. } else
  124. ixgb_reset(adapter);
  125. return 0;
  126. }
  127. static void
  128. ixgb_get_pauseparam(struct net_device *netdev,
  129. struct ethtool_pauseparam *pause)
  130. {
  131. struct ixgb_adapter *adapter = netdev_priv(netdev);
  132. struct ixgb_hw *hw = &adapter->hw;
  133. pause->autoneg = AUTONEG_DISABLE;
  134. if(hw->fc.type == ixgb_fc_rx_pause)
  135. pause->rx_pause = 1;
  136. else if(hw->fc.type == ixgb_fc_tx_pause)
  137. pause->tx_pause = 1;
  138. else if(hw->fc.type == ixgb_fc_full) {
  139. pause->rx_pause = 1;
  140. pause->tx_pause = 1;
  141. }
  142. }
  143. static int
  144. ixgb_set_pauseparam(struct net_device *netdev,
  145. struct ethtool_pauseparam *pause)
  146. {
  147. struct ixgb_adapter *adapter = netdev_priv(netdev);
  148. struct ixgb_hw *hw = &adapter->hw;
  149. if(pause->autoneg == AUTONEG_ENABLE)
  150. return -EINVAL;
  151. if(pause->rx_pause && pause->tx_pause)
  152. hw->fc.type = ixgb_fc_full;
  153. else if(pause->rx_pause && !pause->tx_pause)
  154. hw->fc.type = ixgb_fc_rx_pause;
  155. else if(!pause->rx_pause && pause->tx_pause)
  156. hw->fc.type = ixgb_fc_tx_pause;
  157. else if(!pause->rx_pause && !pause->tx_pause)
  158. hw->fc.type = ixgb_fc_none;
  159. if(netif_running(adapter->netdev)) {
  160. ixgb_down(adapter, TRUE);
  161. ixgb_up(adapter);
  162. ixgb_set_speed_duplex(netdev);
  163. } else
  164. ixgb_reset(adapter);
  165. return 0;
  166. }
  167. static uint32_t
  168. ixgb_get_rx_csum(struct net_device *netdev)
  169. {
  170. struct ixgb_adapter *adapter = netdev_priv(netdev);
  171. return adapter->rx_csum;
  172. }
  173. static int
  174. ixgb_set_rx_csum(struct net_device *netdev, uint32_t data)
  175. {
  176. struct ixgb_adapter *adapter = netdev_priv(netdev);
  177. adapter->rx_csum = data;
  178. if(netif_running(netdev)) {
  179. ixgb_down(adapter,TRUE);
  180. ixgb_up(adapter);
  181. ixgb_set_speed_duplex(netdev);
  182. } else
  183. ixgb_reset(adapter);
  184. return 0;
  185. }
  186. static uint32_t
  187. ixgb_get_tx_csum(struct net_device *netdev)
  188. {
  189. return (netdev->features & NETIF_F_HW_CSUM) != 0;
  190. }
  191. static int
  192. ixgb_set_tx_csum(struct net_device *netdev, uint32_t data)
  193. {
  194. if (data)
  195. netdev->features |= NETIF_F_HW_CSUM;
  196. else
  197. netdev->features &= ~NETIF_F_HW_CSUM;
  198. return 0;
  199. }
  200. #ifdef NETIF_F_TSO
  201. static int
  202. ixgb_set_tso(struct net_device *netdev, uint32_t data)
  203. {
  204. if(data)
  205. netdev->features |= NETIF_F_TSO;
  206. else
  207. netdev->features &= ~NETIF_F_TSO;
  208. return 0;
  209. }
  210. #endif /* NETIF_F_TSO */
  211. static uint32_t
  212. ixgb_get_msglevel(struct net_device *netdev)
  213. {
  214. struct ixgb_adapter *adapter = netdev_priv(netdev);
  215. return adapter->msg_enable;
  216. }
  217. static void
  218. ixgb_set_msglevel(struct net_device *netdev, uint32_t data)
  219. {
  220. struct ixgb_adapter *adapter = netdev_priv(netdev);
  221. adapter->msg_enable = data;
  222. }
  223. #define IXGB_GET_STAT(_A_, _R_) _A_->stats._R_
  224. static int
  225. ixgb_get_regs_len(struct net_device *netdev)
  226. {
  227. #define IXGB_REG_DUMP_LEN 136*sizeof(uint32_t)
  228. return IXGB_REG_DUMP_LEN;
  229. }
  230. static void
  231. ixgb_get_regs(struct net_device *netdev,
  232. struct ethtool_regs *regs, void *p)
  233. {
  234. struct ixgb_adapter *adapter = netdev_priv(netdev);
  235. struct ixgb_hw *hw = &adapter->hw;
  236. uint32_t *reg = p;
  237. uint32_t *reg_start = reg;
  238. uint8_t i;
  239. /* the 1 (one) below indicates an attempt at versioning, if the
  240. * interface in ethtool or the driver changes, this 1 should be
  241. * incremented */
  242. regs->version = (1<<24) | hw->revision_id << 16 | hw->device_id;
  243. /* General Registers */
  244. *reg++ = IXGB_READ_REG(hw, CTRL0); /* 0 */
  245. *reg++ = IXGB_READ_REG(hw, CTRL1); /* 1 */
  246. *reg++ = IXGB_READ_REG(hw, STATUS); /* 2 */
  247. *reg++ = IXGB_READ_REG(hw, EECD); /* 3 */
  248. *reg++ = IXGB_READ_REG(hw, MFS); /* 4 */
  249. /* Interrupt */
  250. *reg++ = IXGB_READ_REG(hw, ICR); /* 5 */
  251. *reg++ = IXGB_READ_REG(hw, ICS); /* 6 */
  252. *reg++ = IXGB_READ_REG(hw, IMS); /* 7 */
  253. *reg++ = IXGB_READ_REG(hw, IMC); /* 8 */
  254. /* Receive */
  255. *reg++ = IXGB_READ_REG(hw, RCTL); /* 9 */
  256. *reg++ = IXGB_READ_REG(hw, FCRTL); /* 10 */
  257. *reg++ = IXGB_READ_REG(hw, FCRTH); /* 11 */
  258. *reg++ = IXGB_READ_REG(hw, RDBAL); /* 12 */
  259. *reg++ = IXGB_READ_REG(hw, RDBAH); /* 13 */
  260. *reg++ = IXGB_READ_REG(hw, RDLEN); /* 14 */
  261. *reg++ = IXGB_READ_REG(hw, RDH); /* 15 */
  262. *reg++ = IXGB_READ_REG(hw, RDT); /* 16 */
  263. *reg++ = IXGB_READ_REG(hw, RDTR); /* 17 */
  264. *reg++ = IXGB_READ_REG(hw, RXDCTL); /* 18 */
  265. *reg++ = IXGB_READ_REG(hw, RAIDC); /* 19 */
  266. *reg++ = IXGB_READ_REG(hw, RXCSUM); /* 20 */
  267. /* there are 16 RAR entries in hardware, we only use 3 */
  268. for(i = 0; i < IXGB_ALL_RAR_ENTRIES; i++) {
  269. *reg++ = IXGB_READ_REG_ARRAY(hw, RAL, (i << 1)); /*21,...,51 */
  270. *reg++ = IXGB_READ_REG_ARRAY(hw, RAH, (i << 1)); /*22,...,52 */
  271. }
  272. /* Transmit */
  273. *reg++ = IXGB_READ_REG(hw, TCTL); /* 53 */
  274. *reg++ = IXGB_READ_REG(hw, TDBAL); /* 54 */
  275. *reg++ = IXGB_READ_REG(hw, TDBAH); /* 55 */
  276. *reg++ = IXGB_READ_REG(hw, TDLEN); /* 56 */
  277. *reg++ = IXGB_READ_REG(hw, TDH); /* 57 */
  278. *reg++ = IXGB_READ_REG(hw, TDT); /* 58 */
  279. *reg++ = IXGB_READ_REG(hw, TIDV); /* 59 */
  280. *reg++ = IXGB_READ_REG(hw, TXDCTL); /* 60 */
  281. *reg++ = IXGB_READ_REG(hw, TSPMT); /* 61 */
  282. *reg++ = IXGB_READ_REG(hw, PAP); /* 62 */
  283. /* Physical */
  284. *reg++ = IXGB_READ_REG(hw, PCSC1); /* 63 */
  285. *reg++ = IXGB_READ_REG(hw, PCSC2); /* 64 */
  286. *reg++ = IXGB_READ_REG(hw, PCSS1); /* 65 */
  287. *reg++ = IXGB_READ_REG(hw, PCSS2); /* 66 */
  288. *reg++ = IXGB_READ_REG(hw, XPCSS); /* 67 */
  289. *reg++ = IXGB_READ_REG(hw, UCCR); /* 68 */
  290. *reg++ = IXGB_READ_REG(hw, XPCSTC); /* 69 */
  291. *reg++ = IXGB_READ_REG(hw, MACA); /* 70 */
  292. *reg++ = IXGB_READ_REG(hw, APAE); /* 71 */
  293. *reg++ = IXGB_READ_REG(hw, ARD); /* 72 */
  294. *reg++ = IXGB_READ_REG(hw, AIS); /* 73 */
  295. *reg++ = IXGB_READ_REG(hw, MSCA); /* 74 */
  296. *reg++ = IXGB_READ_REG(hw, MSRWD); /* 75 */
  297. /* Statistics */
  298. *reg++ = IXGB_GET_STAT(adapter, tprl); /* 76 */
  299. *reg++ = IXGB_GET_STAT(adapter, tprh); /* 77 */
  300. *reg++ = IXGB_GET_STAT(adapter, gprcl); /* 78 */
  301. *reg++ = IXGB_GET_STAT(adapter, gprch); /* 79 */
  302. *reg++ = IXGB_GET_STAT(adapter, bprcl); /* 80 */
  303. *reg++ = IXGB_GET_STAT(adapter, bprch); /* 81 */
  304. *reg++ = IXGB_GET_STAT(adapter, mprcl); /* 82 */
  305. *reg++ = IXGB_GET_STAT(adapter, mprch); /* 83 */
  306. *reg++ = IXGB_GET_STAT(adapter, uprcl); /* 84 */
  307. *reg++ = IXGB_GET_STAT(adapter, uprch); /* 85 */
  308. *reg++ = IXGB_GET_STAT(adapter, vprcl); /* 86 */
  309. *reg++ = IXGB_GET_STAT(adapter, vprch); /* 87 */
  310. *reg++ = IXGB_GET_STAT(adapter, jprcl); /* 88 */
  311. *reg++ = IXGB_GET_STAT(adapter, jprch); /* 89 */
  312. *reg++ = IXGB_GET_STAT(adapter, gorcl); /* 90 */
  313. *reg++ = IXGB_GET_STAT(adapter, gorch); /* 91 */
  314. *reg++ = IXGB_GET_STAT(adapter, torl); /* 92 */
  315. *reg++ = IXGB_GET_STAT(adapter, torh); /* 93 */
  316. *reg++ = IXGB_GET_STAT(adapter, rnbc); /* 94 */
  317. *reg++ = IXGB_GET_STAT(adapter, ruc); /* 95 */
  318. *reg++ = IXGB_GET_STAT(adapter, roc); /* 96 */
  319. *reg++ = IXGB_GET_STAT(adapter, rlec); /* 97 */
  320. *reg++ = IXGB_GET_STAT(adapter, crcerrs); /* 98 */
  321. *reg++ = IXGB_GET_STAT(adapter, icbc); /* 99 */
  322. *reg++ = IXGB_GET_STAT(adapter, ecbc); /* 100 */
  323. *reg++ = IXGB_GET_STAT(adapter, mpc); /* 101 */
  324. *reg++ = IXGB_GET_STAT(adapter, tptl); /* 102 */
  325. *reg++ = IXGB_GET_STAT(adapter, tpth); /* 103 */
  326. *reg++ = IXGB_GET_STAT(adapter, gptcl); /* 104 */
  327. *reg++ = IXGB_GET_STAT(adapter, gptch); /* 105 */
  328. *reg++ = IXGB_GET_STAT(adapter, bptcl); /* 106 */
  329. *reg++ = IXGB_GET_STAT(adapter, bptch); /* 107 */
  330. *reg++ = IXGB_GET_STAT(adapter, mptcl); /* 108 */
  331. *reg++ = IXGB_GET_STAT(adapter, mptch); /* 109 */
  332. *reg++ = IXGB_GET_STAT(adapter, uptcl); /* 110 */
  333. *reg++ = IXGB_GET_STAT(adapter, uptch); /* 111 */
  334. *reg++ = IXGB_GET_STAT(adapter, vptcl); /* 112 */
  335. *reg++ = IXGB_GET_STAT(adapter, vptch); /* 113 */
  336. *reg++ = IXGB_GET_STAT(adapter, jptcl); /* 114 */
  337. *reg++ = IXGB_GET_STAT(adapter, jptch); /* 115 */
  338. *reg++ = IXGB_GET_STAT(adapter, gotcl); /* 116 */
  339. *reg++ = IXGB_GET_STAT(adapter, gotch); /* 117 */
  340. *reg++ = IXGB_GET_STAT(adapter, totl); /* 118 */
  341. *reg++ = IXGB_GET_STAT(adapter, toth); /* 119 */
  342. *reg++ = IXGB_GET_STAT(adapter, dc); /* 120 */
  343. *reg++ = IXGB_GET_STAT(adapter, plt64c); /* 121 */
  344. *reg++ = IXGB_GET_STAT(adapter, tsctc); /* 122 */
  345. *reg++ = IXGB_GET_STAT(adapter, tsctfc); /* 123 */
  346. *reg++ = IXGB_GET_STAT(adapter, ibic); /* 124 */
  347. *reg++ = IXGB_GET_STAT(adapter, rfc); /* 125 */
  348. *reg++ = IXGB_GET_STAT(adapter, lfc); /* 126 */
  349. *reg++ = IXGB_GET_STAT(adapter, pfrc); /* 127 */
  350. *reg++ = IXGB_GET_STAT(adapter, pftc); /* 128 */
  351. *reg++ = IXGB_GET_STAT(adapter, mcfrc); /* 129 */
  352. *reg++ = IXGB_GET_STAT(adapter, mcftc); /* 130 */
  353. *reg++ = IXGB_GET_STAT(adapter, xonrxc); /* 131 */
  354. *reg++ = IXGB_GET_STAT(adapter, xontxc); /* 132 */
  355. *reg++ = IXGB_GET_STAT(adapter, xoffrxc); /* 133 */
  356. *reg++ = IXGB_GET_STAT(adapter, xofftxc); /* 134 */
  357. *reg++ = IXGB_GET_STAT(adapter, rjc); /* 135 */
  358. regs->len = (reg - reg_start) * sizeof(uint32_t);
  359. }
  360. static int
  361. ixgb_get_eeprom_len(struct net_device *netdev)
  362. {
  363. /* return size in bytes */
  364. return (IXGB_EEPROM_SIZE << 1);
  365. }
  366. static int
  367. ixgb_get_eeprom(struct net_device *netdev,
  368. struct ethtool_eeprom *eeprom, uint8_t *bytes)
  369. {
  370. struct ixgb_adapter *adapter = netdev_priv(netdev);
  371. struct ixgb_hw *hw = &adapter->hw;
  372. uint16_t *eeprom_buff;
  373. int i, max_len, first_word, last_word;
  374. int ret_val = 0;
  375. if(eeprom->len == 0) {
  376. ret_val = -EINVAL;
  377. goto geeprom_error;
  378. }
  379. eeprom->magic = hw->vendor_id | (hw->device_id << 16);
  380. max_len = ixgb_get_eeprom_len(netdev);
  381. if(eeprom->offset > eeprom->offset + eeprom->len) {
  382. ret_val = -EINVAL;
  383. goto geeprom_error;
  384. }
  385. if((eeprom->offset + eeprom->len) > max_len)
  386. eeprom->len = (max_len - eeprom->offset);
  387. first_word = eeprom->offset >> 1;
  388. last_word = (eeprom->offset + eeprom->len - 1) >> 1;
  389. eeprom_buff = kmalloc(sizeof(uint16_t) *
  390. (last_word - first_word + 1), GFP_KERNEL);
  391. if(!eeprom_buff)
  392. return -ENOMEM;
  393. /* note the eeprom was good because the driver loaded */
  394. for(i = 0; i <= (last_word - first_word); i++) {
  395. eeprom_buff[i] = ixgb_get_eeprom_word(hw, (first_word + i));
  396. }
  397. memcpy(bytes, (uint8_t *)eeprom_buff + (eeprom->offset & 1),
  398. eeprom->len);
  399. kfree(eeprom_buff);
  400. geeprom_error:
  401. return ret_val;
  402. }
  403. static int
  404. ixgb_set_eeprom(struct net_device *netdev,
  405. struct ethtool_eeprom *eeprom, uint8_t *bytes)
  406. {
  407. struct ixgb_adapter *adapter = netdev_priv(netdev);
  408. struct ixgb_hw *hw = &adapter->hw;
  409. uint16_t *eeprom_buff;
  410. void *ptr;
  411. int max_len, first_word, last_word;
  412. uint16_t i;
  413. if(eeprom->len == 0)
  414. return -EINVAL;
  415. if(eeprom->magic != (hw->vendor_id | (hw->device_id << 16)))
  416. return -EFAULT;
  417. max_len = ixgb_get_eeprom_len(netdev);
  418. if(eeprom->offset > eeprom->offset + eeprom->len)
  419. return -EINVAL;
  420. if((eeprom->offset + eeprom->len) > max_len)
  421. eeprom->len = (max_len - eeprom->offset);
  422. first_word = eeprom->offset >> 1;
  423. last_word = (eeprom->offset + eeprom->len - 1) >> 1;
  424. eeprom_buff = kmalloc(max_len, GFP_KERNEL);
  425. if(!eeprom_buff)
  426. return -ENOMEM;
  427. ptr = (void *)eeprom_buff;
  428. if(eeprom->offset & 1) {
  429. /* need read/modify/write of first changed EEPROM word */
  430. /* only the second byte of the word is being modified */
  431. eeprom_buff[0] = ixgb_read_eeprom(hw, first_word);
  432. ptr++;
  433. }
  434. if((eeprom->offset + eeprom->len) & 1) {
  435. /* need read/modify/write of last changed EEPROM word */
  436. /* only the first byte of the word is being modified */
  437. eeprom_buff[last_word - first_word]
  438. = ixgb_read_eeprom(hw, last_word);
  439. }
  440. memcpy(ptr, bytes, eeprom->len);
  441. for(i = 0; i <= (last_word - first_word); i++)
  442. ixgb_write_eeprom(hw, first_word + i, eeprom_buff[i]);
  443. /* Update the checksum over the first part of the EEPROM if needed */
  444. if(first_word <= EEPROM_CHECKSUM_REG)
  445. ixgb_update_eeprom_checksum(hw);
  446. kfree(eeprom_buff);
  447. return 0;
  448. }
  449. static void
  450. ixgb_get_drvinfo(struct net_device *netdev,
  451. struct ethtool_drvinfo *drvinfo)
  452. {
  453. struct ixgb_adapter *adapter = netdev_priv(netdev);
  454. strncpy(drvinfo->driver, ixgb_driver_name, 32);
  455. strncpy(drvinfo->version, ixgb_driver_version, 32);
  456. strncpy(drvinfo->fw_version, "N/A", 32);
  457. strncpy(drvinfo->bus_info, pci_name(adapter->pdev), 32);
  458. drvinfo->n_stats = IXGB_STATS_LEN;
  459. drvinfo->regdump_len = ixgb_get_regs_len(netdev);
  460. drvinfo->eedump_len = ixgb_get_eeprom_len(netdev);
  461. }
  462. static void
  463. ixgb_get_ringparam(struct net_device *netdev,
  464. struct ethtool_ringparam *ring)
  465. {
  466. struct ixgb_adapter *adapter = netdev_priv(netdev);
  467. struct ixgb_desc_ring *txdr = &adapter->tx_ring;
  468. struct ixgb_desc_ring *rxdr = &adapter->rx_ring;
  469. ring->rx_max_pending = MAX_RXD;
  470. ring->tx_max_pending = MAX_TXD;
  471. ring->rx_mini_max_pending = 0;
  472. ring->rx_jumbo_max_pending = 0;
  473. ring->rx_pending = rxdr->count;
  474. ring->tx_pending = txdr->count;
  475. ring->rx_mini_pending = 0;
  476. ring->rx_jumbo_pending = 0;
  477. }
  478. static int
  479. ixgb_set_ringparam(struct net_device *netdev,
  480. struct ethtool_ringparam *ring)
  481. {
  482. struct ixgb_adapter *adapter = netdev_priv(netdev);
  483. struct ixgb_desc_ring *txdr = &adapter->tx_ring;
  484. struct ixgb_desc_ring *rxdr = &adapter->rx_ring;
  485. struct ixgb_desc_ring tx_old, tx_new, rx_old, rx_new;
  486. int err;
  487. tx_old = adapter->tx_ring;
  488. rx_old = adapter->rx_ring;
  489. if((ring->rx_mini_pending) || (ring->rx_jumbo_pending))
  490. return -EINVAL;
  491. if(netif_running(adapter->netdev))
  492. ixgb_down(adapter,TRUE);
  493. rxdr->count = max(ring->rx_pending,(uint32_t)MIN_RXD);
  494. rxdr->count = min(rxdr->count,(uint32_t)MAX_RXD);
  495. IXGB_ROUNDUP(rxdr->count, IXGB_REQ_RX_DESCRIPTOR_MULTIPLE);
  496. txdr->count = max(ring->tx_pending,(uint32_t)MIN_TXD);
  497. txdr->count = min(txdr->count,(uint32_t)MAX_TXD);
  498. IXGB_ROUNDUP(txdr->count, IXGB_REQ_TX_DESCRIPTOR_MULTIPLE);
  499. if(netif_running(adapter->netdev)) {
  500. /* Try to get new resources before deleting old */
  501. if((err = ixgb_setup_rx_resources(adapter)))
  502. goto err_setup_rx;
  503. if((err = ixgb_setup_tx_resources(adapter)))
  504. goto err_setup_tx;
  505. /* save the new, restore the old in order to free it,
  506. * then restore the new back again */
  507. rx_new = adapter->rx_ring;
  508. tx_new = adapter->tx_ring;
  509. adapter->rx_ring = rx_old;
  510. adapter->tx_ring = tx_old;
  511. ixgb_free_rx_resources(adapter);
  512. ixgb_free_tx_resources(adapter);
  513. adapter->rx_ring = rx_new;
  514. adapter->tx_ring = tx_new;
  515. if((err = ixgb_up(adapter)))
  516. return err;
  517. ixgb_set_speed_duplex(netdev);
  518. }
  519. return 0;
  520. err_setup_tx:
  521. ixgb_free_rx_resources(adapter);
  522. err_setup_rx:
  523. adapter->rx_ring = rx_old;
  524. adapter->tx_ring = tx_old;
  525. ixgb_up(adapter);
  526. return err;
  527. }
  528. /* toggle LED 4 times per second = 2 "blinks" per second */
  529. #define IXGB_ID_INTERVAL (HZ/4)
  530. /* bit defines for adapter->led_status */
  531. #define IXGB_LED_ON 0
  532. static void
  533. ixgb_led_blink_callback(unsigned long data)
  534. {
  535. struct ixgb_adapter *adapter = (struct ixgb_adapter *)data;
  536. if(test_and_change_bit(IXGB_LED_ON, &adapter->led_status))
  537. ixgb_led_off(&adapter->hw);
  538. else
  539. ixgb_led_on(&adapter->hw);
  540. mod_timer(&adapter->blink_timer, jiffies + IXGB_ID_INTERVAL);
  541. }
  542. static int
  543. ixgb_phys_id(struct net_device *netdev, uint32_t data)
  544. {
  545. struct ixgb_adapter *adapter = netdev_priv(netdev);
  546. if(!data || data > (uint32_t)(MAX_SCHEDULE_TIMEOUT / HZ))
  547. data = (uint32_t)(MAX_SCHEDULE_TIMEOUT / HZ);
  548. if(!adapter->blink_timer.function) {
  549. init_timer(&adapter->blink_timer);
  550. adapter->blink_timer.function = ixgb_led_blink_callback;
  551. adapter->blink_timer.data = (unsigned long)adapter;
  552. }
  553. mod_timer(&adapter->blink_timer, jiffies);
  554. msleep_interruptible(data * 1000);
  555. del_timer_sync(&adapter->blink_timer);
  556. ixgb_led_off(&adapter->hw);
  557. clear_bit(IXGB_LED_ON, &adapter->led_status);
  558. return 0;
  559. }
  560. static int
  561. ixgb_get_stats_count(struct net_device *netdev)
  562. {
  563. return IXGB_STATS_LEN;
  564. }
  565. static void
  566. ixgb_get_ethtool_stats(struct net_device *netdev,
  567. struct ethtool_stats *stats, uint64_t *data)
  568. {
  569. struct ixgb_adapter *adapter = netdev_priv(netdev);
  570. int i;
  571. ixgb_update_stats(adapter);
  572. for(i = 0; i < IXGB_STATS_LEN; i++) {
  573. char *p = (char *)adapter+ixgb_gstrings_stats[i].stat_offset;
  574. data[i] = (ixgb_gstrings_stats[i].sizeof_stat ==
  575. sizeof(uint64_t)) ? *(uint64_t *)p : *(uint32_t *)p;
  576. }
  577. }
  578. static void
  579. ixgb_get_strings(struct net_device *netdev, uint32_t stringset, uint8_t *data)
  580. {
  581. int i;
  582. switch(stringset) {
  583. case ETH_SS_STATS:
  584. for(i=0; i < IXGB_STATS_LEN; i++) {
  585. memcpy(data + i * ETH_GSTRING_LEN,
  586. ixgb_gstrings_stats[i].stat_string,
  587. ETH_GSTRING_LEN);
  588. }
  589. break;
  590. }
  591. }
  592. static const struct ethtool_ops ixgb_ethtool_ops = {
  593. .get_settings = ixgb_get_settings,
  594. .set_settings = ixgb_set_settings,
  595. .get_drvinfo = ixgb_get_drvinfo,
  596. .get_regs_len = ixgb_get_regs_len,
  597. .get_regs = ixgb_get_regs,
  598. .get_link = ethtool_op_get_link,
  599. .get_eeprom_len = ixgb_get_eeprom_len,
  600. .get_eeprom = ixgb_get_eeprom,
  601. .set_eeprom = ixgb_set_eeprom,
  602. .get_ringparam = ixgb_get_ringparam,
  603. .set_ringparam = ixgb_set_ringparam,
  604. .get_pauseparam = ixgb_get_pauseparam,
  605. .set_pauseparam = ixgb_set_pauseparam,
  606. .get_rx_csum = ixgb_get_rx_csum,
  607. .set_rx_csum = ixgb_set_rx_csum,
  608. .get_tx_csum = ixgb_get_tx_csum,
  609. .set_tx_csum = ixgb_set_tx_csum,
  610. .get_sg = ethtool_op_get_sg,
  611. .set_sg = ethtool_op_set_sg,
  612. .get_msglevel = ixgb_get_msglevel,
  613. .set_msglevel = ixgb_set_msglevel,
  614. #ifdef NETIF_F_TSO
  615. .get_tso = ethtool_op_get_tso,
  616. .set_tso = ixgb_set_tso,
  617. #endif
  618. .get_strings = ixgb_get_strings,
  619. .phys_id = ixgb_phys_id,
  620. .get_stats_count = ixgb_get_stats_count,
  621. .get_ethtool_stats = ixgb_get_ethtool_stats,
  622. .get_perm_addr = ethtool_op_get_perm_addr,
  623. };
  624. void ixgb_set_ethtool_ops(struct net_device *netdev)
  625. {
  626. SET_ETHTOOL_OPS(netdev, &ixgb_ethtool_ops);
  627. }