blob: 9c6b58adf5320a99f9759d4e08b7e92b08f002e5 [file] [log] [blame]
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -07001/**********************************************************************
2* Author: Cavium, Inc.
3*
4* Contact: support@cavium.com
5* Please include "LiquidIO" in the subject.
6*
7* Copyright (c) 2003-2015 Cavium, Inc.
8*
9* This file is free software; you can redistribute it and/or modify
10* it under the terms of the GNU General Public License, Version 2, as
11* published by the Free Software Foundation.
12*
13* This file is distributed in the hope that it will be useful, but
14* AS-IS and WITHOUT ANY WARRANTY; without even the implied warranty
15* of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE, TITLE, or
16* NONINFRINGEMENT. See the GNU General Public License for more
17* details.
18*
19* This file may also be available under a different license from Cavium.
20* Contact Cavium, Inc. for more information
21**********************************************************************/
22#include <linux/version.h>
23#include <linux/netdevice.h>
24#include <linux/net_tstamp.h>
25#include <linux/ethtool.h>
26#include <linux/dma-mapping.h>
27#include <linux/pci.h>
28#include "octeon_config.h"
29#include "liquidio_common.h"
30#include "octeon_droq.h"
31#include "octeon_iq.h"
32#include "response_manager.h"
33#include "octeon_device.h"
34#include "octeon_nic.h"
35#include "octeon_main.h"
36#include "octeon_network.h"
37#include "cn66xx_regs.h"
38#include "cn66xx_device.h"
39#include "cn68xx_regs.h"
40#include "cn68xx_device.h"
41#include "liquidio_image.h"
42
43struct oct_mdio_cmd_context {
44 int octeon_id;
45 wait_queue_head_t wc;
46 int cond;
47};
48
49struct oct_mdio_cmd_resp {
50 u64 rh;
51 struct oct_mdio_cmd resp;
52 u64 status;
53};
54
55#define OCT_MDIO45_RESP_SIZE (sizeof(struct oct_mdio_cmd_resp))
56
57/* Octeon's interface mode of operation */
58enum {
59 INTERFACE_MODE_DISABLED,
60 INTERFACE_MODE_RGMII,
61 INTERFACE_MODE_GMII,
62 INTERFACE_MODE_SPI,
63 INTERFACE_MODE_PCIE,
64 INTERFACE_MODE_XAUI,
65 INTERFACE_MODE_SGMII,
66 INTERFACE_MODE_PICMG,
67 INTERFACE_MODE_NPI,
68 INTERFACE_MODE_LOOP,
69 INTERFACE_MODE_SRIO,
70 INTERFACE_MODE_ILK,
71 INTERFACE_MODE_RXAUI,
72 INTERFACE_MODE_QSGMII,
73 INTERFACE_MODE_AGL,
74};
75
76#define ARRAY_LENGTH(a) (sizeof(a) / sizeof((a)[0]))
77#define OCT_ETHTOOL_REGDUMP_LEN 4096
78#define OCT_ETHTOOL_REGSVER 1
79
80static const char oct_iq_stats_strings[][ETH_GSTRING_LEN] = {
81 "Instr posted",
82 "Instr processed",
83 "Instr dropped",
84 "Bytes Sent",
85 "Sgentry_sent",
86 "Inst cntreg",
87 "Tx done",
88 "Tx Iq busy",
89 "Tx dropped",
90 "Tx bytes",
91};
92
93static const char oct_droq_stats_strings[][ETH_GSTRING_LEN] = {
94 "OQ Pkts Received",
95 "OQ Bytes Received",
96 "Dropped no dispatch",
97 "Dropped nomem",
98 "Dropped toomany",
99 "Stack RX cnt",
100 "Stack RX Bytes",
101 "RX dropped",
102};
103
104#define OCTNIC_NCMD_AUTONEG_ON 0x1
105#define OCTNIC_NCMD_PHY_ON 0x2
106
107static int lio_get_settings(struct net_device *netdev, struct ethtool_cmd *ecmd)
108{
109 struct lio *lio = GET_LIO(netdev);
110 struct octeon_device *oct = lio->oct_dev;
111 struct oct_link_info *linfo;
112
113 linfo = &lio->linfo;
114
115 if (linfo->link.s.interface == INTERFACE_MODE_XAUI ||
116 linfo->link.s.interface == INTERFACE_MODE_RXAUI) {
117 ecmd->port = PORT_FIBRE;
118 ecmd->supported =
119 (SUPPORTED_10000baseT_Full | SUPPORTED_FIBRE |
120 SUPPORTED_Pause);
121 ecmd->advertising =
122 (ADVERTISED_10000baseT_Full | ADVERTISED_Pause);
123 ecmd->transceiver = XCVR_EXTERNAL;
124 ecmd->autoneg = AUTONEG_DISABLE;
125
126 } else {
127 dev_err(&oct->pci_dev->dev, "Unknown link interface reported\n");
128 }
129
Raghu Vatsavayi0cece6c2016-06-14 16:54:50 -0700130 if (linfo->link.s.link_up) {
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -0700131 ethtool_cmd_speed_set(ecmd, linfo->link.s.speed);
132 ecmd->duplex = linfo->link.s.duplex;
133 } else {
134 ethtool_cmd_speed_set(ecmd, SPEED_UNKNOWN);
135 ecmd->duplex = DUPLEX_UNKNOWN;
136 }
137
138 return 0;
139}
140
141static void
142lio_get_drvinfo(struct net_device *netdev, struct ethtool_drvinfo *drvinfo)
143{
144 struct lio *lio;
145 struct octeon_device *oct;
146
147 lio = GET_LIO(netdev);
148 oct = lio->oct_dev;
149
150 memset(drvinfo, 0, sizeof(struct ethtool_drvinfo));
151 strcpy(drvinfo->driver, "liquidio");
152 strcpy(drvinfo->version, LIQUIDIO_VERSION);
153 strncpy(drvinfo->fw_version, oct->fw_info.liquidio_firmware_version,
154 ETHTOOL_FWVERS_LEN);
155 strncpy(drvinfo->bus_info, pci_name(oct->pci_dev), 32);
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -0700156}
157
158static void
159lio_ethtool_get_channels(struct net_device *dev,
160 struct ethtool_channels *channel)
161{
162 struct lio *lio = GET_LIO(dev);
163 struct octeon_device *oct = lio->oct_dev;
164 u32 max_rx = 0, max_tx = 0, tx_count = 0, rx_count = 0;
165
166 if (OCTEON_CN6XXX(oct)) {
167 struct octeon_config *conf6x = CHIP_FIELD(oct, cn6xxx, conf);
168
169 max_rx = CFG_GET_OQ_MAX_Q(conf6x);
170 max_tx = CFG_GET_IQ_MAX_Q(conf6x);
171 rx_count = CFG_GET_NUM_RXQS_NIC_IF(conf6x, lio->ifidx);
172 tx_count = CFG_GET_NUM_TXQS_NIC_IF(conf6x, lio->ifidx);
173 }
174
175 channel->max_rx = max_rx;
176 channel->max_tx = max_tx;
177 channel->rx_count = rx_count;
178 channel->tx_count = tx_count;
179}
180
181static int lio_get_eeprom_len(struct net_device *netdev)
182{
183 u8 buf[128];
184 struct lio *lio = GET_LIO(netdev);
185 struct octeon_device *oct_dev = lio->oct_dev;
186 struct octeon_board_info *board_info;
187 int len;
188
189 board_info = (struct octeon_board_info *)(&oct_dev->boardinfo);
190 len = sprintf(buf, "boardname:%s serialnum:%s maj:%lld min:%lld\n",
191 board_info->name, board_info->serial_number,
192 board_info->major, board_info->minor);
193
194 return len;
195}
196
197static int
198lio_get_eeprom(struct net_device *netdev, struct ethtool_eeprom *eeprom,
199 u8 *bytes)
200{
201 struct lio *lio = GET_LIO(netdev);
202 struct octeon_device *oct_dev = lio->oct_dev;
203 struct octeon_board_info *board_info;
204 int len;
205
206 if (eeprom->offset != 0)
207 return -EINVAL;
208
209 eeprom->magic = oct_dev->pci_dev->vendor;
210 board_info = (struct octeon_board_info *)(&oct_dev->boardinfo);
211 len =
212 sprintf((char *)bytes,
213 "boardname:%s serialnum:%s maj:%lld min:%lld\n",
214 board_info->name, board_info->serial_number,
215 board_info->major, board_info->minor);
216
217 return 0;
218}
219
220static int octnet_gpio_access(struct net_device *netdev, int addr, int val)
221{
222 struct lio *lio = GET_LIO(netdev);
223 struct octeon_device *oct = lio->oct_dev;
224 struct octnic_ctrl_pkt nctrl;
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -0700225 int ret = 0;
226
227 memset(&nctrl, 0, sizeof(struct octnic_ctrl_pkt));
228
229 nctrl.ncmd.u64 = 0;
230 nctrl.ncmd.s.cmd = OCTNET_CMD_GPIO_ACCESS;
Raghu Vatsavayi0cece6c2016-06-14 16:54:50 -0700231 nctrl.ncmd.s.param1 = addr;
232 nctrl.ncmd.s.param2 = val;
233 nctrl.iq_no = lio->linfo.txpciq[0].s.q_no;
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -0700234 nctrl.wait_time = 100;
235 nctrl.netpndev = (u64)netdev;
236 nctrl.cb_fn = liquidio_link_ctrl_cmd_completion;
237
Raghu Vatsavayi0cece6c2016-06-14 16:54:50 -0700238 ret = octnet_send_nic_ctrl_pkt(lio->oct_dev, &nctrl);
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -0700239 if (ret < 0) {
240 dev_err(&oct->pci_dev->dev, "Failed to configure gpio value\n");
241 return -EINVAL;
242 }
243
244 return 0;
245}
246
247/* Callback for when mdio command response arrives
248 */
249static void octnet_mdio_resp_callback(struct octeon_device *oct,
250 u32 status,
251 void *buf)
252{
253 struct oct_mdio_cmd_resp *mdio_cmd_rsp;
254 struct oct_mdio_cmd_context *mdio_cmd_ctx;
255 struct octeon_soft_command *sc = (struct octeon_soft_command *)buf;
256
257 mdio_cmd_rsp = (struct oct_mdio_cmd_resp *)sc->virtrptr;
258 mdio_cmd_ctx = (struct oct_mdio_cmd_context *)sc->ctxptr;
259
260 oct = lio_get_device(mdio_cmd_ctx->octeon_id);
261 if (status) {
262 dev_err(&oct->pci_dev->dev, "MIDO instruction failed. Status: %llx\n",
263 CVM_CAST64(status));
264 ACCESS_ONCE(mdio_cmd_ctx->cond) = -1;
265 } else {
266 ACCESS_ONCE(mdio_cmd_ctx->cond) = 1;
267 }
268 wake_up_interruptible(&mdio_cmd_ctx->wc);
269}
270
271/* This routine provides PHY access routines for
272 * mdio clause45 .
273 */
274static int
275octnet_mdio45_access(struct lio *lio, int op, int loc, int *value)
276{
277 struct octeon_device *oct_dev = lio->oct_dev;
278 struct octeon_soft_command *sc;
279 struct oct_mdio_cmd_resp *mdio_cmd_rsp;
280 struct oct_mdio_cmd_context *mdio_cmd_ctx;
281 struct oct_mdio_cmd *mdio_cmd;
282 int retval = 0;
283
284 sc = (struct octeon_soft_command *)
285 octeon_alloc_soft_command(oct_dev,
286 sizeof(struct oct_mdio_cmd),
287 sizeof(struct oct_mdio_cmd_resp),
288 sizeof(struct oct_mdio_cmd_context));
289
290 if (!sc)
291 return -ENOMEM;
292
293 mdio_cmd_ctx = (struct oct_mdio_cmd_context *)sc->ctxptr;
294 mdio_cmd_rsp = (struct oct_mdio_cmd_resp *)sc->virtrptr;
295 mdio_cmd = (struct oct_mdio_cmd *)sc->virtdptr;
296
297 ACCESS_ONCE(mdio_cmd_ctx->cond) = 0;
298 mdio_cmd_ctx->octeon_id = lio_get_device_id(oct_dev);
299 mdio_cmd->op = op;
300 mdio_cmd->mdio_addr = loc;
301 if (op)
302 mdio_cmd->value1 = *value;
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -0700303 octeon_swap_8B_data((u64 *)mdio_cmd, sizeof(struct oct_mdio_cmd) / 8);
304
Raghu Vatsavayi0cece6c2016-06-14 16:54:50 -0700305 sc->iq_no = lio->linfo.txpciq[0].s.q_no;
306
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -0700307 octeon_prepare_soft_command(oct_dev, sc, OPCODE_NIC, OPCODE_NIC_MDIO45,
308 0, 0, 0);
309
310 sc->wait_time = 1000;
311 sc->callback = octnet_mdio_resp_callback;
312 sc->callback_arg = sc;
313
314 init_waitqueue_head(&mdio_cmd_ctx->wc);
315
316 retval = octeon_send_soft_command(oct_dev, sc);
317
Raghu Vatsavayiddc173a2016-06-14 16:54:43 -0700318 if (retval == IQ_SEND_FAILED) {
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -0700319 dev_err(&oct_dev->pci_dev->dev,
320 "octnet_mdio45_access instruction failed status: %x\n",
321 retval);
322 retval = -EBUSY;
323 } else {
324 /* Sleep on a wait queue till the cond flag indicates that the
325 * response arrived
326 */
327 sleep_cond(&mdio_cmd_ctx->wc, &mdio_cmd_ctx->cond);
328 retval = mdio_cmd_rsp->status;
329 if (retval) {
330 dev_err(&oct_dev->pci_dev->dev, "octnet mdio45 access failed\n");
331 retval = -EBUSY;
332 } else {
333 octeon_swap_8B_data((u64 *)(&mdio_cmd_rsp->resp),
334 sizeof(struct oct_mdio_cmd) / 8);
335
336 if (ACCESS_ONCE(mdio_cmd_ctx->cond) == 1) {
337 if (!op)
338 *value = mdio_cmd_rsp->resp.value1;
339 } else {
340 retval = -EINVAL;
341 }
342 }
343 }
344
345 octeon_free_soft_command(oct_dev, sc);
346
347 return retval;
348}
349
350static int lio_set_phys_id(struct net_device *netdev,
351 enum ethtool_phys_id_state state)
352{
353 struct lio *lio = GET_LIO(netdev);
354 struct octeon_device *oct = lio->oct_dev;
355 int value, ret;
356
357 switch (state) {
358 case ETHTOOL_ID_ACTIVE:
359 if (oct->chip_id == OCTEON_CN66XX) {
360 octnet_gpio_access(netdev, VITESSE_PHY_GPIO_CFG,
361 VITESSE_PHY_GPIO_DRIVEON);
362 return 2;
363
364 } else if (oct->chip_id == OCTEON_CN68XX) {
365 /* Save the current LED settings */
366 ret = octnet_mdio45_access(lio, 0,
367 LIO68XX_LED_BEACON_ADDR,
368 &lio->phy_beacon_val);
369 if (ret)
370 return ret;
371
372 ret = octnet_mdio45_access(lio, 0,
373 LIO68XX_LED_CTRL_ADDR,
374 &lio->led_ctrl_val);
375 if (ret)
376 return ret;
377
378 /* Configure Beacon values */
379 value = LIO68XX_LED_BEACON_CFGON;
380 ret =
381 octnet_mdio45_access(lio, 1,
382 LIO68XX_LED_BEACON_ADDR,
383 &value);
384 if (ret)
385 return ret;
386
387 value = LIO68XX_LED_CTRL_CFGON;
388 ret =
389 octnet_mdio45_access(lio, 1,
390 LIO68XX_LED_CTRL_ADDR,
391 &value);
392 if (ret)
393 return ret;
394 } else {
395 return -EINVAL;
396 }
397 break;
398
399 case ETHTOOL_ID_ON:
400 if (oct->chip_id == OCTEON_CN66XX) {
401 octnet_gpio_access(netdev, VITESSE_PHY_GPIO_CFG,
402 VITESSE_PHY_GPIO_HIGH);
403
404 } else if (oct->chip_id == OCTEON_CN68XX) {
405 return -EINVAL;
406 } else {
407 return -EINVAL;
408 }
409 break;
410
411 case ETHTOOL_ID_OFF:
412 if (oct->chip_id == OCTEON_CN66XX)
413 octnet_gpio_access(netdev, VITESSE_PHY_GPIO_CFG,
414 VITESSE_PHY_GPIO_LOW);
415 else if (oct->chip_id == OCTEON_CN68XX)
416 return -EINVAL;
417 else
418 return -EINVAL;
419
420 break;
421
422 case ETHTOOL_ID_INACTIVE:
423 if (oct->chip_id == OCTEON_CN66XX) {
424 octnet_gpio_access(netdev, VITESSE_PHY_GPIO_CFG,
425 VITESSE_PHY_GPIO_DRIVEOFF);
426 } else if (oct->chip_id == OCTEON_CN68XX) {
427 /* Restore LED settings */
428 ret = octnet_mdio45_access(lio, 1,
429 LIO68XX_LED_CTRL_ADDR,
430 &lio->led_ctrl_val);
431 if (ret)
432 return ret;
433
Dan Carpentercbdb9772015-06-24 17:47:02 +0300434 ret = octnet_mdio45_access(lio, 1,
435 LIO68XX_LED_BEACON_ADDR,
436 &lio->phy_beacon_val);
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -0700437 if (ret)
438 return ret;
439
440 } else {
441 return -EINVAL;
442 }
443 break;
444
445 default:
446 return -EINVAL;
447 }
448
449 return 0;
450}
451
452static void
453lio_ethtool_get_ringparam(struct net_device *netdev,
454 struct ethtool_ringparam *ering)
455{
456 struct lio *lio = GET_LIO(netdev);
457 struct octeon_device *oct = lio->oct_dev;
458 u32 tx_max_pending = 0, rx_max_pending = 0, tx_pending = 0,
459 rx_pending = 0;
460
461 if (OCTEON_CN6XXX(oct)) {
462 struct octeon_config *conf6x = CHIP_FIELD(oct, cn6xxx, conf);
463
464 tx_max_pending = CN6XXX_MAX_IQ_DESCRIPTORS;
465 rx_max_pending = CN6XXX_MAX_OQ_DESCRIPTORS;
466 rx_pending = CFG_GET_NUM_RX_DESCS_NIC_IF(conf6x, lio->ifidx);
467 tx_pending = CFG_GET_NUM_TX_DESCS_NIC_IF(conf6x, lio->ifidx);
468 }
469
470 if (lio->mtu > OCTNET_DEFAULT_FRM_SIZE) {
471 ering->rx_pending = 0;
472 ering->rx_max_pending = 0;
473 ering->rx_mini_pending = 0;
474 ering->rx_jumbo_pending = rx_pending;
475 ering->rx_mini_max_pending = 0;
476 ering->rx_jumbo_max_pending = rx_max_pending;
477 } else {
478 ering->rx_pending = rx_pending;
479 ering->rx_max_pending = rx_max_pending;
480 ering->rx_mini_pending = 0;
481 ering->rx_jumbo_pending = 0;
482 ering->rx_mini_max_pending = 0;
483 ering->rx_jumbo_max_pending = 0;
484 }
485
486 ering->tx_pending = tx_pending;
487 ering->tx_max_pending = tx_max_pending;
488}
489
490static u32 lio_get_msglevel(struct net_device *netdev)
491{
492 struct lio *lio = GET_LIO(netdev);
493
494 return lio->msg_enable;
495}
496
497static void lio_set_msglevel(struct net_device *netdev, u32 msglvl)
498{
499 struct lio *lio = GET_LIO(netdev);
500
501 if ((msglvl ^ lio->msg_enable) & NETIF_MSG_HW) {
502 if (msglvl & NETIF_MSG_HW)
503 liquidio_set_feature(netdev,
Raghu Vatsavayi0cece6c2016-06-14 16:54:50 -0700504 OCTNET_CMD_VERBOSE_ENABLE, 0);
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -0700505 else
506 liquidio_set_feature(netdev,
Raghu Vatsavayi0cece6c2016-06-14 16:54:50 -0700507 OCTNET_CMD_VERBOSE_DISABLE, 0);
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -0700508 }
509
510 lio->msg_enable = msglvl;
511}
512
513static void
514lio_get_pauseparam(struct net_device *netdev, struct ethtool_pauseparam *pause)
515{
516 /* Notes: Not supporting any auto negotiation in these
517 * drivers. Just report pause frame support.
518 */
519 pause->tx_pause = 1;
520 pause->rx_pause = 1; /* TODO: Need to support RX pause frame!!. */
521}
522
523static void
524lio_get_ethtool_stats(struct net_device *netdev,
525 struct ethtool_stats *stats, u64 *data)
526{
527 struct lio *lio = GET_LIO(netdev);
528 struct octeon_device *oct_dev = lio->oct_dev;
529 int i = 0, j;
530
Raghu Vatsavayi63da8402016-06-21 22:53:03 -0700531 for (j = 0; j < MAX_OCTEON_INSTR_QUEUES(oct); j++) {
532 if (!(oct_dev->io_qmask.iq & (1ULL << j)))
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -0700533 continue;
534 data[i++] =
535 CVM_CAST64(oct_dev->instr_queue[j]->stats.instr_posted);
536 data[i++] =
537 CVM_CAST64(
538 oct_dev->instr_queue[j]->stats.instr_processed);
539 data[i++] =
540 CVM_CAST64(
541 oct_dev->instr_queue[j]->stats.instr_dropped);
542 data[i++] =
543 CVM_CAST64(oct_dev->instr_queue[j]->stats.bytes_sent);
544 data[i++] =
545 CVM_CAST64(oct_dev->instr_queue[j]->stats.sgentry_sent);
546 data[i++] =
547 readl(oct_dev->instr_queue[j]->inst_cnt_reg);
548 data[i++] =
549 CVM_CAST64(oct_dev->instr_queue[j]->stats.tx_done);
550 data[i++] =
551 CVM_CAST64(oct_dev->instr_queue[j]->stats.tx_iq_busy);
552 data[i++] =
553 CVM_CAST64(oct_dev->instr_queue[j]->stats.tx_dropped);
554 data[i++] =
555 CVM_CAST64(oct_dev->instr_queue[j]->stats.tx_tot_bytes);
556 }
557
558 /* for (j = 0; j < oct_dev->num_oqs; j++){ */
Raghu Vatsavayi63da8402016-06-21 22:53:03 -0700559 for (j = 0; j < MAX_OCTEON_OUTPUT_QUEUES(oct); j++) {
560 if (!(oct_dev->io_qmask.oq & (1ULL << j)))
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -0700561 continue;
562 data[i++] = CVM_CAST64(oct_dev->droq[j]->stats.pkts_received);
563 data[i++] = CVM_CAST64(oct_dev->droq[j]->stats.bytes_received);
564 data[i++] =
565 CVM_CAST64(oct_dev->droq[j]->stats.dropped_nodispatch);
566 data[i++] = CVM_CAST64(oct_dev->droq[j]->stats.dropped_nomem);
567 data[i++] = CVM_CAST64(oct_dev->droq[j]->stats.dropped_toomany);
568 data[i++] =
569 CVM_CAST64(oct_dev->droq[j]->stats.rx_pkts_received);
570 data[i++] =
571 CVM_CAST64(oct_dev->droq[j]->stats.rx_bytes_received);
572 data[i++] =
573 CVM_CAST64(oct_dev->droq[j]->stats.rx_dropped);
574 }
575}
576
577static void lio_get_strings(struct net_device *netdev, u32 stringset, u8 *data)
578{
579 struct lio *lio = GET_LIO(netdev);
580 struct octeon_device *oct_dev = lio->oct_dev;
581 int num_iq_stats, num_oq_stats, i, j;
582
583 num_iq_stats = ARRAY_SIZE(oct_iq_stats_strings);
Raghu Vatsavayi63da8402016-06-21 22:53:03 -0700584 for (i = 0; i < MAX_OCTEON_INSTR_QUEUES(oct); i++) {
585 if (!(oct_dev->io_qmask.iq & (1ULL << i)))
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -0700586 continue;
587 for (j = 0; j < num_iq_stats; j++) {
588 sprintf(data, "IQ%d %s", i, oct_iq_stats_strings[j]);
589 data += ETH_GSTRING_LEN;
590 }
591 }
592
593 num_oq_stats = ARRAY_SIZE(oct_droq_stats_strings);
594 /* for (i = 0; i < oct_dev->num_oqs; i++) { */
Raghu Vatsavayi63da8402016-06-21 22:53:03 -0700595 for (i = 0; i < MAX_OCTEON_OUTPUT_QUEUES(oct); i++) {
596 if (!(oct_dev->io_qmask.oq & (1ULL << i)))
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -0700597 continue;
598 for (j = 0; j < num_oq_stats; j++) {
599 sprintf(data, "OQ%d %s", i, oct_droq_stats_strings[j]);
600 data += ETH_GSTRING_LEN;
601 }
602 }
603}
604
605static int lio_get_sset_count(struct net_device *netdev, int sset)
606{
607 struct lio *lio = GET_LIO(netdev);
608 struct octeon_device *oct_dev = lio->oct_dev;
609
610 return (ARRAY_SIZE(oct_iq_stats_strings) * oct_dev->num_iqs) +
611 (ARRAY_SIZE(oct_droq_stats_strings) * oct_dev->num_oqs);
612}
613
614static int lio_get_intr_coalesce(struct net_device *netdev,
615 struct ethtool_coalesce *intr_coal)
616{
617 struct lio *lio = GET_LIO(netdev);
618 struct octeon_device *oct = lio->oct_dev;
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -0700619 struct octeon_instr_queue *iq;
620 struct oct_intrmod_cfg *intrmod_cfg;
621
622 intrmod_cfg = &oct->intrmod;
623
624 switch (oct->chip_id) {
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -0700625 case OCTEON_CN68XX:
Raghu Vatsavayi78e6a9b2016-06-21 22:53:10 -0700626 case OCTEON_CN66XX: {
627 struct octeon_cn6xxx *cn6xxx =
628 (struct octeon_cn6xxx *)oct->chip;
629
630 if (!intrmod_cfg->rx_enable) {
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -0700631 intr_coal->rx_coalesce_usecs =
632 CFG_GET_OQ_INTR_TIME(cn6xxx->conf);
633 intr_coal->rx_max_coalesced_frames =
634 CFG_GET_OQ_INTR_PKT(cn6xxx->conf);
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -0700635 }
636
Raghu Vatsavayi26236fa2016-06-14 16:54:44 -0700637 iq = oct->instr_queue[lio->linfo.txpciq[0].s.q_no];
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -0700638 intr_coal->tx_max_coalesced_frames = iq->fill_threshold;
639 break;
Raghu Vatsavayi78e6a9b2016-06-21 22:53:10 -0700640 }
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -0700641 default:
642 netif_info(lio, drv, lio->netdev, "Unknown Chip !!\n");
643 return -EINVAL;
644 }
Raghu Vatsavayi78e6a9b2016-06-21 22:53:10 -0700645 if (intrmod_cfg->rx_enable) {
646 intr_coal->use_adaptive_rx_coalesce =
647 intrmod_cfg->rx_enable;
648 intr_coal->rate_sample_interval =
649 intrmod_cfg->check_intrvl;
650 intr_coal->pkt_rate_high =
651 intrmod_cfg->maxpkt_ratethr;
652 intr_coal->pkt_rate_low =
653 intrmod_cfg->minpkt_ratethr;
654 intr_coal->rx_max_coalesced_frames_high =
655 intrmod_cfg->rx_maxcnt_trigger;
656 intr_coal->rx_coalesce_usecs_high =
657 intrmod_cfg->rx_maxtmr_trigger;
658 intr_coal->rx_coalesce_usecs_low =
659 intrmod_cfg->rx_mintmr_trigger;
660 intr_coal->rx_max_coalesced_frames_low =
661 intrmod_cfg->rx_mincnt_trigger;
662 }
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -0700663 return 0;
664}
665
666/* Callback function for intrmod */
667static void octnet_intrmod_callback(struct octeon_device *oct_dev,
668 u32 status,
669 void *ptr)
670{
671 struct oct_intrmod_cmd *cmd = ptr;
672 struct octeon_soft_command *sc = cmd->sc;
673
674 oct_dev = cmd->oct_dev;
675
676 if (status)
677 dev_err(&oct_dev->pci_dev->dev, "intrmod config failed. Status: %llx\n",
678 CVM_CAST64(status));
679 else
680 dev_info(&oct_dev->pci_dev->dev,
681 "Rx-Adaptive Interrupt moderation enabled:%llx\n",
Raghu Vatsavayi78e6a9b2016-06-21 22:53:10 -0700682 oct_dev->intrmod.rx_enable);
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -0700683
684 octeon_free_soft_command(oct_dev, sc);
685}
686
687/* Configure interrupt moderation parameters */
Raghu Vatsavayi78e6a9b2016-06-21 22:53:10 -0700688static int octnet_set_intrmod_cfg(struct lio *lio,
689 struct oct_intrmod_cfg *intr_cfg)
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -0700690{
691 struct octeon_soft_command *sc;
692 struct oct_intrmod_cmd *cmd;
693 struct oct_intrmod_cfg *cfg;
694 int retval;
Raghu Vatsavayi78e6a9b2016-06-21 22:53:10 -0700695 struct octeon_device *oct_dev = lio->oct_dev;
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -0700696
697 /* Alloc soft command */
698 sc = (struct octeon_soft_command *)
699 octeon_alloc_soft_command(oct_dev,
700 sizeof(struct oct_intrmod_cfg),
701 0,
702 sizeof(struct oct_intrmod_cmd));
703
704 if (!sc)
705 return -ENOMEM;
706
707 cmd = (struct oct_intrmod_cmd *)sc->ctxptr;
708 cfg = (struct oct_intrmod_cfg *)sc->virtdptr;
709
710 memcpy(cfg, intr_cfg, sizeof(struct oct_intrmod_cfg));
711 octeon_swap_8B_data((u64 *)cfg, (sizeof(struct oct_intrmod_cfg)) / 8);
712 cmd->sc = sc;
713 cmd->cfg = cfg;
714 cmd->oct_dev = oct_dev;
715
Raghu Vatsavayi78e6a9b2016-06-21 22:53:10 -0700716 sc->iq_no = lio->linfo.txpciq[0].s.q_no;
717
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -0700718 octeon_prepare_soft_command(oct_dev, sc, OPCODE_NIC,
719 OPCODE_NIC_INTRMOD_CFG, 0, 0, 0);
720
721 sc->callback = octnet_intrmod_callback;
722 sc->callback_arg = cmd;
723 sc->wait_time = 1000;
724
725 retval = octeon_send_soft_command(oct_dev, sc);
Raghu Vatsavayiddc173a2016-06-14 16:54:43 -0700726 if (retval == IQ_SEND_FAILED) {
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -0700727 octeon_free_soft_command(oct_dev, sc);
728 return -EINVAL;
729 }
730
731 return 0;
732}
733
734/* Enable/Disable auto interrupt Moderation */
735static int oct_cfg_adaptive_intr(struct lio *lio, struct ethtool_coalesce
Raghu Vatsavayi78e6a9b2016-06-21 22:53:10 -0700736 *intr_coal)
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -0700737{
738 int ret = 0;
739 struct octeon_device *oct = lio->oct_dev;
740 struct oct_intrmod_cfg *intrmod_cfg;
741
742 intrmod_cfg = &oct->intrmod;
743
Raghu Vatsavayi78e6a9b2016-06-21 22:53:10 -0700744 if (oct->intrmod.rx_enable || oct->intrmod.tx_enable) {
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -0700745 if (intr_coal->rate_sample_interval)
Raghu Vatsavayi78e6a9b2016-06-21 22:53:10 -0700746 intrmod_cfg->check_intrvl =
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -0700747 intr_coal->rate_sample_interval;
748 else
Raghu Vatsavayi78e6a9b2016-06-21 22:53:10 -0700749 intrmod_cfg->check_intrvl =
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -0700750 LIO_INTRMOD_CHECK_INTERVAL;
751
752 if (intr_coal->pkt_rate_high)
Raghu Vatsavayi78e6a9b2016-06-21 22:53:10 -0700753 intrmod_cfg->maxpkt_ratethr =
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -0700754 intr_coal->pkt_rate_high;
755 else
Raghu Vatsavayi78e6a9b2016-06-21 22:53:10 -0700756 intrmod_cfg->maxpkt_ratethr =
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -0700757 LIO_INTRMOD_MAXPKT_RATETHR;
758
759 if (intr_coal->pkt_rate_low)
Raghu Vatsavayi78e6a9b2016-06-21 22:53:10 -0700760 intrmod_cfg->minpkt_ratethr =
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -0700761 intr_coal->pkt_rate_low;
762 else
Raghu Vatsavayi78e6a9b2016-06-21 22:53:10 -0700763 intrmod_cfg->minpkt_ratethr =
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -0700764 LIO_INTRMOD_MINPKT_RATETHR;
Raghu Vatsavayi78e6a9b2016-06-21 22:53:10 -0700765 }
766 if (oct->intrmod.rx_enable) {
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -0700767 if (intr_coal->rx_max_coalesced_frames_high)
Raghu Vatsavayi78e6a9b2016-06-21 22:53:10 -0700768 intrmod_cfg->rx_maxcnt_trigger =
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -0700769 intr_coal->rx_max_coalesced_frames_high;
770 else
Raghu Vatsavayi78e6a9b2016-06-21 22:53:10 -0700771 intrmod_cfg->rx_maxcnt_trigger =
772 LIO_INTRMOD_RXMAXCNT_TRIGGER;
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -0700773
774 if (intr_coal->rx_coalesce_usecs_high)
Raghu Vatsavayi78e6a9b2016-06-21 22:53:10 -0700775 intrmod_cfg->rx_maxtmr_trigger =
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -0700776 intr_coal->rx_coalesce_usecs_high;
777 else
Raghu Vatsavayi78e6a9b2016-06-21 22:53:10 -0700778 intrmod_cfg->rx_maxtmr_trigger =
779 LIO_INTRMOD_RXMAXTMR_TRIGGER;
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -0700780
781 if (intr_coal->rx_coalesce_usecs_low)
Raghu Vatsavayi78e6a9b2016-06-21 22:53:10 -0700782 intrmod_cfg->rx_mintmr_trigger =
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -0700783 intr_coal->rx_coalesce_usecs_low;
784 else
Raghu Vatsavayi78e6a9b2016-06-21 22:53:10 -0700785 intrmod_cfg->rx_mintmr_trigger =
786 LIO_INTRMOD_RXMINTMR_TRIGGER;
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -0700787
788 if (intr_coal->rx_max_coalesced_frames_low)
Raghu Vatsavayi78e6a9b2016-06-21 22:53:10 -0700789 intrmod_cfg->rx_mincnt_trigger =
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -0700790 intr_coal->rx_max_coalesced_frames_low;
791 else
Raghu Vatsavayi78e6a9b2016-06-21 22:53:10 -0700792 intrmod_cfg->rx_mincnt_trigger =
793 LIO_INTRMOD_RXMINCNT_TRIGGER;
794 }
795 if (oct->intrmod.tx_enable) {
796 if (intr_coal->tx_max_coalesced_frames_high)
797 intrmod_cfg->tx_maxcnt_trigger =
798 intr_coal->tx_max_coalesced_frames_high;
799 else
800 intrmod_cfg->tx_maxcnt_trigger =
801 LIO_INTRMOD_TXMAXCNT_TRIGGER;
802 if (intr_coal->tx_max_coalesced_frames_low)
803 intrmod_cfg->tx_mincnt_trigger =
804 intr_coal->tx_max_coalesced_frames_low;
805 else
806 intrmod_cfg->tx_mincnt_trigger =
807 LIO_INTRMOD_TXMINCNT_TRIGGER;
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -0700808 }
809
Raghu Vatsavayi78e6a9b2016-06-21 22:53:10 -0700810 ret = octnet_set_intrmod_cfg(lio, intrmod_cfg);
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -0700811
812 return ret;
813}
814
815static int
816oct_cfg_rx_intrcnt(struct lio *lio, struct ethtool_coalesce *intr_coal)
817{
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -0700818 struct octeon_device *oct = lio->oct_dev;
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -0700819 u32 rx_max_coalesced_frames;
820
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -0700821 /* Config Cnt based interrupt values */
Raghu Vatsavayi78e6a9b2016-06-21 22:53:10 -0700822 switch (oct->chip_id) {
823 case OCTEON_CN68XX:
824 case OCTEON_CN66XX: {
825 struct octeon_cn6xxx *cn6xxx =
826 (struct octeon_cn6xxx *)oct->chip;
827
828 if (!intr_coal->rx_max_coalesced_frames)
829 rx_max_coalesced_frames = CN6XXX_OQ_INTR_PKT;
830 else
831 rx_max_coalesced_frames =
832 intr_coal->rx_max_coalesced_frames;
833 octeon_write_csr(oct, CN6XXX_SLI_OQ_INT_LEVEL_PKTS,
834 rx_max_coalesced_frames);
835 CFG_SET_OQ_INTR_PKT(cn6xxx->conf, rx_max_coalesced_frames);
836 break;
837 }
838 default:
839 return -EINVAL;
840 }
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -0700841 return 0;
842}
843
844static int oct_cfg_rx_intrtime(struct lio *lio, struct ethtool_coalesce
845 *intr_coal)
846{
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -0700847 struct octeon_device *oct = lio->oct_dev;
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -0700848 u32 time_threshold, rx_coalesce_usecs;
849
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -0700850 /* Config Time based interrupt values */
Raghu Vatsavayi78e6a9b2016-06-21 22:53:10 -0700851 switch (oct->chip_id) {
852 case OCTEON_CN68XX:
853 case OCTEON_CN66XX: {
854 struct octeon_cn6xxx *cn6xxx =
855 (struct octeon_cn6xxx *)oct->chip;
856 if (!intr_coal->rx_coalesce_usecs)
857 rx_coalesce_usecs = CN6XXX_OQ_INTR_TIME;
858 else
859 rx_coalesce_usecs = intr_coal->rx_coalesce_usecs;
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -0700860
Raghu Vatsavayi78e6a9b2016-06-21 22:53:10 -0700861 time_threshold = lio_cn6xxx_get_oq_ticks(oct,
862 rx_coalesce_usecs);
863 octeon_write_csr(oct,
864 CN6XXX_SLI_OQ_INT_LEVEL_TIME,
865 time_threshold);
866
867 CFG_SET_OQ_INTR_TIME(cn6xxx->conf, rx_coalesce_usecs);
868 break;
869 }
870 default:
871 return -EINVAL;
872 }
873
874 return 0;
875}
876
877static int
878oct_cfg_tx_intrcnt(struct lio *lio, struct ethtool_coalesce *intr_coal
879 __attribute__((unused)))
880{
881 struct octeon_device *oct = lio->oct_dev;
882
883 /* Config Cnt based interrupt values */
884 switch (oct->chip_id) {
885 case OCTEON_CN68XX:
886 case OCTEON_CN66XX:
887 break;
888 default:
889 return -EINVAL;
890 }
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -0700891 return 0;
892}
893
894static int lio_set_intr_coalesce(struct net_device *netdev,
895 struct ethtool_coalesce *intr_coal)
896{
897 struct lio *lio = GET_LIO(netdev);
898 int ret;
899 struct octeon_device *oct = lio->oct_dev;
900 u32 j, q_no;
Raghu Vatsavayi78e6a9b2016-06-21 22:53:10 -0700901 int db_max, db_min;
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -0700902
Raghu Vatsavayi78e6a9b2016-06-21 22:53:10 -0700903 switch (oct->chip_id) {
904 case OCTEON_CN68XX:
905 case OCTEON_CN66XX:
906 db_min = CN6XXX_DB_MIN;
907 db_max = CN6XXX_DB_MAX;
908 if ((intr_coal->tx_max_coalesced_frames >= db_min) &&
909 (intr_coal->tx_max_coalesced_frames <= db_max)) {
910 for (j = 0; j < lio->linfo.num_txpciq; j++) {
911 q_no = lio->linfo.txpciq[j].s.q_no;
912 oct->instr_queue[q_no]->fill_threshold =
913 intr_coal->tx_max_coalesced_frames;
914 }
915 } else {
916 dev_err(&oct->pci_dev->dev,
917 "LIQUIDIO: Invalid tx-frames:%d. Range is min:%d max:%d\n",
918 intr_coal->tx_max_coalesced_frames, db_min,
919 db_max);
920 return -EINVAL;
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -0700921 }
Raghu Vatsavayi78e6a9b2016-06-21 22:53:10 -0700922 break;
923 default:
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -0700924 return -EINVAL;
925 }
926
Raghu Vatsavayi78e6a9b2016-06-21 22:53:10 -0700927 oct->intrmod.rx_enable = intr_coal->use_adaptive_rx_coalesce ? 1 : 0;
928 oct->intrmod.tx_enable = intr_coal->use_adaptive_tx_coalesce ? 1 : 0;
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -0700929
Raghu Vatsavayi78e6a9b2016-06-21 22:53:10 -0700930 ret = oct_cfg_adaptive_intr(lio, intr_coal);
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -0700931
Raghu Vatsavayi78e6a9b2016-06-21 22:53:10 -0700932 if (!intr_coal->use_adaptive_rx_coalesce) {
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -0700933 ret = oct_cfg_rx_intrtime(lio, intr_coal);
934 if (ret)
935 goto ret_intrmod;
936
937 ret = oct_cfg_rx_intrcnt(lio, intr_coal);
938 if (ret)
939 goto ret_intrmod;
940 }
Raghu Vatsavayi78e6a9b2016-06-21 22:53:10 -0700941 if (!intr_coal->use_adaptive_tx_coalesce) {
942 ret = oct_cfg_tx_intrcnt(lio, intr_coal);
943 if (ret)
944 goto ret_intrmod;
945 }
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -0700946
947 return 0;
948ret_intrmod:
949 return ret;
950}
951
952static int lio_get_ts_info(struct net_device *netdev,
953 struct ethtool_ts_info *info)
954{
955 struct lio *lio = GET_LIO(netdev);
956
957 info->so_timestamping =
958 SOF_TIMESTAMPING_TX_HARDWARE |
959 SOF_TIMESTAMPING_TX_SOFTWARE |
960 SOF_TIMESTAMPING_RX_HARDWARE |
961 SOF_TIMESTAMPING_RX_SOFTWARE |
962 SOF_TIMESTAMPING_SOFTWARE | SOF_TIMESTAMPING_RAW_HARDWARE;
963
964 if (lio->ptp_clock)
965 info->phc_index = ptp_clock_index(lio->ptp_clock);
966 else
967 info->phc_index = -1;
968
969 info->tx_types = (1 << HWTSTAMP_TX_OFF) | (1 << HWTSTAMP_TX_ON);
970
971 info->rx_filters = (1 << HWTSTAMP_FILTER_NONE) |
972 (1 << HWTSTAMP_FILTER_PTP_V1_L4_EVENT) |
973 (1 << HWTSTAMP_FILTER_PTP_V2_L2_EVENT) |
974 (1 << HWTSTAMP_FILTER_PTP_V2_L4_EVENT);
975
976 return 0;
977}
978
979static int lio_set_settings(struct net_device *netdev, struct ethtool_cmd *ecmd)
980{
981 struct lio *lio = GET_LIO(netdev);
982 struct octeon_device *oct = lio->oct_dev;
983 struct oct_link_info *linfo;
984 struct octnic_ctrl_pkt nctrl;
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -0700985 int ret = 0;
986
987 /* get the link info */
988 linfo = &lio->linfo;
989
990 if (ecmd->autoneg != AUTONEG_ENABLE && ecmd->autoneg != AUTONEG_DISABLE)
991 return -EINVAL;
992
993 if (ecmd->autoneg == AUTONEG_DISABLE && ((ecmd->speed != SPEED_100 &&
994 ecmd->speed != SPEED_10) ||
995 (ecmd->duplex != DUPLEX_HALF &&
996 ecmd->duplex != DUPLEX_FULL)))
997 return -EINVAL;
998
999 /* Ethtool Support is not provided for XAUI and RXAUI Interfaces
1000 * as they operate at fixed Speed and Duplex settings
1001 */
1002 if (linfo->link.s.interface == INTERFACE_MODE_XAUI ||
1003 linfo->link.s.interface == INTERFACE_MODE_RXAUI) {
1004 dev_info(&oct->pci_dev->dev, "XAUI IFs settings cannot be modified.\n");
1005 return -EINVAL;
1006 }
1007
1008 memset(&nctrl, 0, sizeof(struct octnic_ctrl_pkt));
1009
1010 nctrl.ncmd.u64 = 0;
1011 nctrl.ncmd.s.cmd = OCTNET_CMD_SET_SETTINGS;
Raghu Vatsavayi0cece6c2016-06-14 16:54:50 -07001012 nctrl.iq_no = lio->linfo.txpciq[0].s.q_no;
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -07001013 nctrl.wait_time = 1000;
1014 nctrl.netpndev = (u64)netdev;
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -07001015 nctrl.cb_fn = liquidio_link_ctrl_cmd_completion;
1016
1017 /* Passing the parameters sent by ethtool like Speed, Autoneg & Duplex
1018 * to SE core application using ncmd.s.more & ncmd.s.param
1019 */
1020 if (ecmd->autoneg == AUTONEG_ENABLE) {
1021 /* Autoneg ON */
1022 nctrl.ncmd.s.more = OCTNIC_NCMD_PHY_ON |
1023 OCTNIC_NCMD_AUTONEG_ON;
Raghu Vatsavayi0cece6c2016-06-14 16:54:50 -07001024 nctrl.ncmd.s.param1 = ecmd->advertising;
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -07001025 } else {
1026 /* Autoneg OFF */
1027 nctrl.ncmd.s.more = OCTNIC_NCMD_PHY_ON;
1028
Raghu Vatsavayi0cece6c2016-06-14 16:54:50 -07001029 nctrl.ncmd.s.param2 = ecmd->duplex;
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -07001030
Raghu Vatsavayi0cece6c2016-06-14 16:54:50 -07001031 nctrl.ncmd.s.param1 = ecmd->speed;
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -07001032 }
1033
Raghu Vatsavayi0cece6c2016-06-14 16:54:50 -07001034 ret = octnet_send_nic_ctrl_pkt(lio->oct_dev, &nctrl);
Raghu Vatsavayif21fb3e2015-06-09 18:15:23 -07001035 if (ret < 0) {
1036 dev_err(&oct->pci_dev->dev, "Failed to set settings\n");
1037 return -1;
1038 }
1039
1040 return 0;
1041}
1042
1043static int lio_nway_reset(struct net_device *netdev)
1044{
1045 if (netif_running(netdev)) {
1046 struct ethtool_cmd ecmd;
1047
1048 memset(&ecmd, 0, sizeof(struct ethtool_cmd));
1049 ecmd.autoneg = 0;
1050 ecmd.speed = 0;
1051 ecmd.duplex = 0;
1052 lio_set_settings(netdev, &ecmd);
1053 }
1054 return 0;
1055}
1056
1057/* Return register dump len. */
1058static int lio_get_regs_len(struct net_device *dev)
1059{
1060 return OCT_ETHTOOL_REGDUMP_LEN;
1061}
1062
1063static int cn6xxx_read_csr_reg(char *s, struct octeon_device *oct)
1064{
1065 u32 reg;
1066 int i, len = 0;
1067
1068 /* PCI Window Registers */
1069
1070 len += sprintf(s + len, "\n\t Octeon CSR Registers\n\n");
1071 reg = CN6XXX_WIN_WR_ADDR_LO;
1072 len += sprintf(s + len, "\n[%02x] (WIN_WR_ADDR_LO): %08x\n",
1073 CN6XXX_WIN_WR_ADDR_LO, octeon_read_csr(oct, reg));
1074 reg = CN6XXX_WIN_WR_ADDR_HI;
1075 len += sprintf(s + len, "[%02x] (WIN_WR_ADDR_HI): %08x\n",
1076 CN6XXX_WIN_WR_ADDR_HI, octeon_read_csr(oct, reg));
1077 reg = CN6XXX_WIN_RD_ADDR_LO;
1078 len += sprintf(s + len, "[%02x] (WIN_RD_ADDR_LO): %08x\n",
1079 CN6XXX_WIN_RD_ADDR_LO, octeon_read_csr(oct, reg));
1080 reg = CN6XXX_WIN_RD_ADDR_HI;
1081 len += sprintf(s + len, "[%02x] (WIN_RD_ADDR_HI): %08x\n",
1082 CN6XXX_WIN_RD_ADDR_HI, octeon_read_csr(oct, reg));
1083 reg = CN6XXX_WIN_WR_DATA_LO;
1084 len += sprintf(s + len, "[%02x] (WIN_WR_DATA_LO): %08x\n",
1085 CN6XXX_WIN_WR_DATA_LO, octeon_read_csr(oct, reg));
1086 reg = CN6XXX_WIN_WR_DATA_HI;
1087 len += sprintf(s + len, "[%02x] (WIN_WR_DATA_HI): %08x\n",
1088 CN6XXX_WIN_WR_DATA_HI, octeon_read_csr(oct, reg));
1089 len += sprintf(s + len, "[%02x] (WIN_WR_MASK_REG): %08x\n",
1090 CN6XXX_WIN_WR_MASK_REG,
1091 octeon_read_csr(oct, CN6XXX_WIN_WR_MASK_REG));
1092
1093 /* PCI Interrupt Register */
1094 len += sprintf(s + len, "\n[%x] (INT_ENABLE PORT 0): %08x\n",
1095 CN6XXX_SLI_INT_ENB64_PORT0, octeon_read_csr(oct,
1096 CN6XXX_SLI_INT_ENB64_PORT0));
1097 len += sprintf(s + len, "\n[%x] (INT_ENABLE PORT 1): %08x\n",
1098 CN6XXX_SLI_INT_ENB64_PORT1,
1099 octeon_read_csr(oct, CN6XXX_SLI_INT_ENB64_PORT1));
1100 len += sprintf(s + len, "[%x] (INT_SUM): %08x\n", CN6XXX_SLI_INT_SUM64,
1101 octeon_read_csr(oct, CN6XXX_SLI_INT_SUM64));
1102
1103 /* PCI Output queue registers */
1104 for (i = 0; i < oct->num_oqs; i++) {
1105 reg = CN6XXX_SLI_OQ_PKTS_SENT(i);
1106 len += sprintf(s + len, "\n[%x] (PKTS_SENT_%d): %08x\n",
1107 reg, i, octeon_read_csr(oct, reg));
1108 reg = CN6XXX_SLI_OQ_PKTS_CREDIT(i);
1109 len += sprintf(s + len, "[%x] (PKT_CREDITS_%d): %08x\n",
1110 reg, i, octeon_read_csr(oct, reg));
1111 }
1112 reg = CN6XXX_SLI_OQ_INT_LEVEL_PKTS;
1113 len += sprintf(s + len, "\n[%x] (PKTS_SENT_INT_LEVEL): %08x\n",
1114 reg, octeon_read_csr(oct, reg));
1115 reg = CN6XXX_SLI_OQ_INT_LEVEL_TIME;
1116 len += sprintf(s + len, "[%x] (PKTS_SENT_TIME): %08x\n",
1117 reg, octeon_read_csr(oct, reg));
1118
1119 /* PCI Input queue registers */
1120 for (i = 0; i <= 3; i++) {
1121 u32 reg;
1122
1123 reg = CN6XXX_SLI_IQ_DOORBELL(i);
1124 len += sprintf(s + len, "\n[%x] (INSTR_DOORBELL_%d): %08x\n",
1125 reg, i, octeon_read_csr(oct, reg));
1126 reg = CN6XXX_SLI_IQ_INSTR_COUNT(i);
1127 len += sprintf(s + len, "[%x] (INSTR_COUNT_%d): %08x\n",
1128 reg, i, octeon_read_csr(oct, reg));
1129 }
1130
1131 /* PCI DMA registers */
1132
1133 len += sprintf(s + len, "\n[%x] (DMA_CNT_0): %08x\n",
1134 CN6XXX_DMA_CNT(0),
1135 octeon_read_csr(oct, CN6XXX_DMA_CNT(0)));
1136 reg = CN6XXX_DMA_PKT_INT_LEVEL(0);
1137 len += sprintf(s + len, "[%x] (DMA_INT_LEV_0): %08x\n",
1138 CN6XXX_DMA_PKT_INT_LEVEL(0), octeon_read_csr(oct, reg));
1139 reg = CN6XXX_DMA_TIME_INT_LEVEL(0);
1140 len += sprintf(s + len, "[%x] (DMA_TIME_0): %08x\n",
1141 CN6XXX_DMA_TIME_INT_LEVEL(0),
1142 octeon_read_csr(oct, reg));
1143
1144 len += sprintf(s + len, "\n[%x] (DMA_CNT_1): %08x\n",
1145 CN6XXX_DMA_CNT(1),
1146 octeon_read_csr(oct, CN6XXX_DMA_CNT(1)));
1147 reg = CN6XXX_DMA_PKT_INT_LEVEL(1);
1148 len += sprintf(s + len, "[%x] (DMA_INT_LEV_1): %08x\n",
1149 CN6XXX_DMA_PKT_INT_LEVEL(1),
1150 octeon_read_csr(oct, reg));
1151 reg = CN6XXX_DMA_PKT_INT_LEVEL(1);
1152 len += sprintf(s + len, "[%x] (DMA_TIME_1): %08x\n",
1153 CN6XXX_DMA_TIME_INT_LEVEL(1),
1154 octeon_read_csr(oct, reg));
1155
1156 /* PCI Index registers */
1157
1158 len += sprintf(s + len, "\n");
1159
1160 for (i = 0; i < 16; i++) {
1161 reg = lio_pci_readq(oct, CN6XXX_BAR1_REG(i, oct->pcie_port));
1162 len += sprintf(s + len, "[%llx] (BAR1_INDEX_%02d): %08x\n",
1163 CN6XXX_BAR1_REG(i, oct->pcie_port), i, reg);
1164 }
1165
1166 return len;
1167}
1168
1169static int cn6xxx_read_config_reg(char *s, struct octeon_device *oct)
1170{
1171 u32 val;
1172 int i, len = 0;
1173
1174 /* PCI CONFIG Registers */
1175
1176 len += sprintf(s + len,
1177 "\n\t Octeon Config space Registers\n\n");
1178
1179 for (i = 0; i <= 13; i++) {
1180 pci_read_config_dword(oct->pci_dev, (i * 4), &val);
1181 len += sprintf(s + len, "[0x%x] (Config[%d]): 0x%08x\n",
1182 (i * 4), i, val);
1183 }
1184
1185 for (i = 30; i <= 34; i++) {
1186 pci_read_config_dword(oct->pci_dev, (i * 4), &val);
1187 len += sprintf(s + len, "[0x%x] (Config[%d]): 0x%08x\n",
1188 (i * 4), i, val);
1189 }
1190
1191 return len;
1192}
1193
1194/* Return register dump user app. */
1195static void lio_get_regs(struct net_device *dev,
1196 struct ethtool_regs *regs, void *regbuf)
1197{
1198 struct lio *lio = GET_LIO(dev);
1199 int len = 0;
1200 struct octeon_device *oct = lio->oct_dev;
1201
1202 memset(regbuf, 0, OCT_ETHTOOL_REGDUMP_LEN);
1203 regs->version = OCT_ETHTOOL_REGSVER;
1204
1205 switch (oct->chip_id) {
1206 /* case OCTEON_CN73XX: Todo */
1207 case OCTEON_CN68XX:
1208 case OCTEON_CN66XX:
1209 len += cn6xxx_read_csr_reg(regbuf + len, oct);
1210 len += cn6xxx_read_config_reg(regbuf + len, oct);
1211 break;
1212 default:
1213 dev_err(&oct->pci_dev->dev, "%s Unknown chipid: %d\n",
1214 __func__, oct->chip_id);
1215 }
1216}
1217
1218static const struct ethtool_ops lio_ethtool_ops = {
1219 .get_settings = lio_get_settings,
1220 .get_link = ethtool_op_get_link,
1221 .get_drvinfo = lio_get_drvinfo,
1222 .get_ringparam = lio_ethtool_get_ringparam,
1223 .get_channels = lio_ethtool_get_channels,
1224 .set_phys_id = lio_set_phys_id,
1225 .get_eeprom_len = lio_get_eeprom_len,
1226 .get_eeprom = lio_get_eeprom,
1227 .get_strings = lio_get_strings,
1228 .get_ethtool_stats = lio_get_ethtool_stats,
1229 .get_pauseparam = lio_get_pauseparam,
1230 .get_regs_len = lio_get_regs_len,
1231 .get_regs = lio_get_regs,
1232 .get_msglevel = lio_get_msglevel,
1233 .set_msglevel = lio_set_msglevel,
1234 .get_sset_count = lio_get_sset_count,
1235 .nway_reset = lio_nway_reset,
1236 .set_settings = lio_set_settings,
1237 .get_coalesce = lio_get_intr_coalesce,
1238 .set_coalesce = lio_set_intr_coalesce,
1239 .get_ts_info = lio_get_ts_info,
1240};
1241
1242void liquidio_set_ethtool_ops(struct net_device *netdev)
1243{
1244 netdev->ethtool_ops = &lio_ethtool_ops;
1245}