xref: /openbmc/linux/drivers/net/ethernet/fungible/funeth/funeth_ethtool.c (revision 22a41e9a5044bf3519f05b4a00e99af34bfeb40c)
1 // SPDX-License-Identifier: (GPL-2.0-only OR BSD-3-Clause)
2 
3 #include <linux/ethtool.h>
4 #include <linux/linkmode.h>
5 #include <linux/netdevice.h>
6 #include <linux/nvme.h>
7 #include <linux/io.h>
8 #include <linux/io-64-nonatomic-lo-hi.h>
9 #include <linux/pci.h>
10 #include <linux/rtnetlink.h>
11 #include "funeth.h"
12 #include "fun_port.h"
13 #include "funeth_txrx.h"
14 
15 /* Min queue depth. The smallest power-of-2 supporting jumbo frames with 4K
16  * pages is 8. Require it for all types of queues though some could work with
17  * fewer entries.
18  */
19 #define FUNETH_MIN_QDEPTH 8
20 
21 static const char mac_tx_stat_names[][ETH_GSTRING_LEN] = {
22 	"mac_tx_octets_total",
23 	"mac_tx_frames_total",
24 	"mac_tx_vlan_frames_ok",
25 	"mac_tx_unicast_frames",
26 	"mac_tx_multicast_frames",
27 	"mac_tx_broadcast_frames",
28 	"mac_tx_errors",
29 	"mac_tx_CBFCPAUSE0",
30 	"mac_tx_CBFCPAUSE1",
31 	"mac_tx_CBFCPAUSE2",
32 	"mac_tx_CBFCPAUSE3",
33 	"mac_tx_CBFCPAUSE4",
34 	"mac_tx_CBFCPAUSE5",
35 	"mac_tx_CBFCPAUSE6",
36 	"mac_tx_CBFCPAUSE7",
37 	"mac_tx_CBFCPAUSE8",
38 	"mac_tx_CBFCPAUSE9",
39 	"mac_tx_CBFCPAUSE10",
40 	"mac_tx_CBFCPAUSE11",
41 	"mac_tx_CBFCPAUSE12",
42 	"mac_tx_CBFCPAUSE13",
43 	"mac_tx_CBFCPAUSE14",
44 	"mac_tx_CBFCPAUSE15",
45 };
46 
47 static const char mac_rx_stat_names[][ETH_GSTRING_LEN] = {
48 	"mac_rx_octets_total",
49 	"mac_rx_frames_total",
50 	"mac_rx_VLAN_frames_ok",
51 	"mac_rx_unicast_frames",
52 	"mac_rx_multicast_frames",
53 	"mac_rx_broadcast_frames",
54 	"mac_rx_drop_events",
55 	"mac_rx_errors",
56 	"mac_rx_alignment_errors",
57 	"mac_rx_CBFCPAUSE0",
58 	"mac_rx_CBFCPAUSE1",
59 	"mac_rx_CBFCPAUSE2",
60 	"mac_rx_CBFCPAUSE3",
61 	"mac_rx_CBFCPAUSE4",
62 	"mac_rx_CBFCPAUSE5",
63 	"mac_rx_CBFCPAUSE6",
64 	"mac_rx_CBFCPAUSE7",
65 	"mac_rx_CBFCPAUSE8",
66 	"mac_rx_CBFCPAUSE9",
67 	"mac_rx_CBFCPAUSE10",
68 	"mac_rx_CBFCPAUSE11",
69 	"mac_rx_CBFCPAUSE12",
70 	"mac_rx_CBFCPAUSE13",
71 	"mac_rx_CBFCPAUSE14",
72 	"mac_rx_CBFCPAUSE15",
73 };
74 
75 static const char * const txq_stat_names[] = {
76 	"tx_pkts",
77 	"tx_bytes",
78 	"tx_cso",
79 	"tx_tso",
80 	"tx_encapsulated_tso",
81 	"tx_more",
82 	"tx_queue_stops",
83 	"tx_queue_restarts",
84 	"tx_mapping_errors",
85 	"tx_tls_encrypted_packets",
86 	"tx_tls_encrypted_bytes",
87 	"tx_tls_ooo",
88 	"tx_tls_drop_no_sync_data",
89 };
90 
91 static const char * const xdpq_stat_names[] = {
92 	"tx_xdp_pkts",
93 	"tx_xdp_bytes",
94 	"tx_xdp_full",
95 	"tx_xdp_mapping_errors",
96 };
97 
98 static const char * const rxq_stat_names[] = {
99 	"rx_pkts",
100 	"rx_bytes",
101 	"rx_cso",
102 	"gro_pkts",
103 	"gro_merged",
104 	"rx_xdp_tx",
105 	"rx_xdp_redir",
106 	"rx_xdp_drops",
107 	"rx_buffers",
108 	"rx_page_allocs",
109 	"rx_drops",
110 	"rx_budget_exhausted",
111 	"rx_mapping_errors",
112 };
113 
114 static const char * const tls_stat_names[] = {
115 	"tx_tls_ctx",
116 	"tx_tls_del",
117 	"tx_tls_resync",
118 };
119 
120 static void fun_link_modes_to_ethtool(u64 modes,
121 				      unsigned long *ethtool_modes_map)
122 {
123 #define ADD_LINK_MODE(mode) \
124 	__set_bit(ETHTOOL_LINK_MODE_ ## mode ## _BIT, ethtool_modes_map)
125 
126 	if (modes & FUN_PORT_CAP_AUTONEG)
127 		ADD_LINK_MODE(Autoneg);
128 	if (modes & FUN_PORT_CAP_1000_X)
129 		ADD_LINK_MODE(1000baseX_Full);
130 	if (modes & FUN_PORT_CAP_10G_R) {
131 		ADD_LINK_MODE(10000baseCR_Full);
132 		ADD_LINK_MODE(10000baseSR_Full);
133 		ADD_LINK_MODE(10000baseLR_Full);
134 		ADD_LINK_MODE(10000baseER_Full);
135 	}
136 	if (modes & FUN_PORT_CAP_25G_R) {
137 		ADD_LINK_MODE(25000baseCR_Full);
138 		ADD_LINK_MODE(25000baseSR_Full);
139 	}
140 	if (modes & FUN_PORT_CAP_40G_R4) {
141 		ADD_LINK_MODE(40000baseCR4_Full);
142 		ADD_LINK_MODE(40000baseSR4_Full);
143 		ADD_LINK_MODE(40000baseLR4_Full);
144 	}
145 	if (modes & FUN_PORT_CAP_50G_R2) {
146 		ADD_LINK_MODE(50000baseCR2_Full);
147 		ADD_LINK_MODE(50000baseSR2_Full);
148 	}
149 	if (modes & FUN_PORT_CAP_50G_R) {
150 		ADD_LINK_MODE(50000baseCR_Full);
151 		ADD_LINK_MODE(50000baseSR_Full);
152 		ADD_LINK_MODE(50000baseLR_ER_FR_Full);
153 	}
154 	if (modes & FUN_PORT_CAP_100G_R4) {
155 		ADD_LINK_MODE(100000baseCR4_Full);
156 		ADD_LINK_MODE(100000baseSR4_Full);
157 		ADD_LINK_MODE(100000baseLR4_ER4_Full);
158 	}
159 	if (modes & FUN_PORT_CAP_100G_R2) {
160 		ADD_LINK_MODE(100000baseCR2_Full);
161 		ADD_LINK_MODE(100000baseSR2_Full);
162 		ADD_LINK_MODE(100000baseLR2_ER2_FR2_Full);
163 	}
164 	if (modes & FUN_PORT_CAP_FEC_NONE)
165 		ADD_LINK_MODE(FEC_NONE);
166 	if (modes & FUN_PORT_CAP_FEC_FC)
167 		ADD_LINK_MODE(FEC_BASER);
168 	if (modes & FUN_PORT_CAP_FEC_RS)
169 		ADD_LINK_MODE(FEC_RS);
170 	if (modes & FUN_PORT_CAP_RX_PAUSE)
171 		ADD_LINK_MODE(Pause);
172 
173 #undef ADD_LINK_MODE
174 }
175 
176 static void set_asym_pause(u64 advertising, struct ethtool_link_ksettings *ks)
177 {
178 	bool rx_pause, tx_pause;
179 
180 	rx_pause = advertising & FUN_PORT_CAP_RX_PAUSE;
181 	tx_pause = advertising & FUN_PORT_CAP_TX_PAUSE;
182 	if (tx_pause ^ rx_pause)
183 		ethtool_link_ksettings_add_link_mode(ks, advertising,
184 						     Asym_Pause);
185 }
186 
187 static unsigned int fun_port_type(unsigned int xcvr)
188 {
189 	if (!xcvr)
190 		return PORT_NONE;
191 
192 	switch (xcvr & 7) {
193 	case FUN_XCVR_BASET:
194 		return PORT_TP;
195 	case FUN_XCVR_CU:
196 		return PORT_DA;
197 	default:
198 		return PORT_FIBRE;
199 	}
200 }
201 
202 static int fun_get_link_ksettings(struct net_device *netdev,
203 				  struct ethtool_link_ksettings *ks)
204 {
205 	const struct funeth_priv *fp = netdev_priv(netdev);
206 	unsigned int seq, speed, xcvr;
207 	u64 lp_advertising;
208 	bool link_up;
209 
210 	ethtool_link_ksettings_zero_link_mode(ks, supported);
211 	ethtool_link_ksettings_zero_link_mode(ks, advertising);
212 	ethtool_link_ksettings_zero_link_mode(ks, lp_advertising);
213 
214 	/* Link settings change asynchronously, take a consistent snapshot */
215 	do {
216 		seq = read_seqcount_begin(&fp->link_seq);
217 		link_up = netif_carrier_ok(netdev);
218 		speed = fp->link_speed;
219 		xcvr = fp->xcvr_type;
220 		lp_advertising = fp->lp_advertising;
221 	} while (read_seqcount_retry(&fp->link_seq, seq));
222 
223 	if (link_up) {
224 		ks->base.speed = speed;
225 		ks->base.duplex = DUPLEX_FULL;
226 		fun_link_modes_to_ethtool(lp_advertising,
227 					  ks->link_modes.lp_advertising);
228 	} else {
229 		ks->base.speed = SPEED_UNKNOWN;
230 		ks->base.duplex = DUPLEX_UNKNOWN;
231 	}
232 
233 	ks->base.autoneg = (fp->advertising & FUN_PORT_CAP_AUTONEG) ?
234 			   AUTONEG_ENABLE : AUTONEG_DISABLE;
235 	ks->base.port = fun_port_type(xcvr);
236 
237 	fun_link_modes_to_ethtool(fp->port_caps, ks->link_modes.supported);
238 	if (fp->port_caps & (FUN_PORT_CAP_RX_PAUSE | FUN_PORT_CAP_TX_PAUSE))
239 		ethtool_link_ksettings_add_link_mode(ks, supported, Asym_Pause);
240 
241 	fun_link_modes_to_ethtool(fp->advertising, ks->link_modes.advertising);
242 	set_asym_pause(fp->advertising, ks);
243 	return 0;
244 }
245 
246 static u64 fun_advert_modes(const struct ethtool_link_ksettings *ks)
247 {
248 	u64 modes = 0;
249 
250 #define HAS_MODE(mode) \
251 	ethtool_link_ksettings_test_link_mode(ks, advertising, mode)
252 
253 	if (HAS_MODE(1000baseX_Full))
254 		modes |= FUN_PORT_CAP_1000_X;
255 	if (HAS_MODE(10000baseCR_Full) || HAS_MODE(10000baseSR_Full) ||
256 	    HAS_MODE(10000baseLR_Full) || HAS_MODE(10000baseER_Full))
257 		modes |= FUN_PORT_CAP_10G_R;
258 	if (HAS_MODE(25000baseCR_Full) || HAS_MODE(25000baseSR_Full))
259 		modes |= FUN_PORT_CAP_25G_R;
260 	if (HAS_MODE(40000baseCR4_Full) || HAS_MODE(40000baseSR4_Full) ||
261 	    HAS_MODE(40000baseLR4_Full))
262 		modes |= FUN_PORT_CAP_40G_R4;
263 	if (HAS_MODE(50000baseCR2_Full) || HAS_MODE(50000baseSR2_Full))
264 		modes |= FUN_PORT_CAP_50G_R2;
265 	if (HAS_MODE(50000baseCR_Full) || HAS_MODE(50000baseSR_Full) ||
266 	    HAS_MODE(50000baseLR_ER_FR_Full))
267 		modes |= FUN_PORT_CAP_50G_R;
268 	if (HAS_MODE(100000baseCR4_Full) || HAS_MODE(100000baseSR4_Full) ||
269 	    HAS_MODE(100000baseLR4_ER4_Full))
270 		modes |= FUN_PORT_CAP_100G_R4;
271 	if (HAS_MODE(100000baseCR2_Full) || HAS_MODE(100000baseSR2_Full) ||
272 	    HAS_MODE(100000baseLR2_ER2_FR2_Full))
273 		modes |= FUN_PORT_CAP_100G_R2;
274 
275 	return modes;
276 #undef HAS_MODE
277 }
278 
279 static u64 fun_speed_to_link_mode(unsigned int speed)
280 {
281 	switch (speed) {
282 	case SPEED_100000:
283 		return FUN_PORT_CAP_100G_R4 | FUN_PORT_CAP_100G_R2;
284 	case SPEED_50000:
285 		return FUN_PORT_CAP_50G_R | FUN_PORT_CAP_50G_R2;
286 	case SPEED_40000:
287 		return FUN_PORT_CAP_40G_R4;
288 	case SPEED_25000:
289 		return FUN_PORT_CAP_25G_R;
290 	case SPEED_10000:
291 		return FUN_PORT_CAP_10G_R;
292 	case SPEED_1000:
293 		return FUN_PORT_CAP_1000_X;
294 	default:
295 		return 0;
296 	}
297 }
298 
299 static int fun_change_advert(struct funeth_priv *fp, u64 new_advert)
300 {
301 	int err;
302 
303 	if (new_advert == fp->advertising)
304 		return 0;
305 
306 	err = fun_port_write_cmd(fp, FUN_ADMIN_PORT_KEY_ADVERT, new_advert);
307 	if (!err)
308 		fp->advertising = new_advert;
309 	return err;
310 }
311 
312 #define FUN_PORT_CAP_FEC_MASK \
313 	(FUN_PORT_CAP_FEC_NONE | FUN_PORT_CAP_FEC_FC | FUN_PORT_CAP_FEC_RS)
314 
315 static int fun_set_link_ksettings(struct net_device *netdev,
316 				  const struct ethtool_link_ksettings *ks)
317 {
318 	__ETHTOOL_DECLARE_LINK_MODE_MASK(supported) = {};
319 	struct funeth_priv *fp = netdev_priv(netdev);
320 	u64 new_advert;
321 
322 	/* eswitch ports don't support mode changes */
323 	if (fp->port_caps & FUN_PORT_CAP_VPORT)
324 		return -EOPNOTSUPP;
325 
326 	if (ks->base.duplex == DUPLEX_HALF)
327 		return -EINVAL;
328 	if (ks->base.autoneg == AUTONEG_ENABLE &&
329 	    !(fp->port_caps & FUN_PORT_CAP_AUTONEG))
330 		return -EINVAL;
331 
332 	if (ks->base.autoneg == AUTONEG_ENABLE) {
333 		if (linkmode_empty(ks->link_modes.advertising))
334 			return -EINVAL;
335 
336 		fun_link_modes_to_ethtool(fp->port_caps, supported);
337 		if (!linkmode_subset(ks->link_modes.advertising, supported))
338 			return -EINVAL;
339 
340 		new_advert = fun_advert_modes(ks) | FUN_PORT_CAP_AUTONEG;
341 	} else {
342 		new_advert = fun_speed_to_link_mode(ks->base.speed);
343 		new_advert &= fp->port_caps;
344 		if (!new_advert)
345 			return -EINVAL;
346 	}
347 	new_advert |= fp->advertising &
348 		      (FUN_PORT_CAP_PAUSE_MASK | FUN_PORT_CAP_FEC_MASK);
349 
350 	return fun_change_advert(fp, new_advert);
351 }
352 
353 static void fun_get_pauseparam(struct net_device *netdev,
354 			       struct ethtool_pauseparam *pause)
355 {
356 	const struct funeth_priv *fp = netdev_priv(netdev);
357 	u8 active_pause = fp->active_fc;
358 
359 	pause->rx_pause = !!(active_pause & FUN_PORT_CAP_RX_PAUSE);
360 	pause->tx_pause = !!(active_pause & FUN_PORT_CAP_TX_PAUSE);
361 	pause->autoneg = !!(fp->advertising & FUN_PORT_CAP_AUTONEG);
362 }
363 
364 static int fun_set_pauseparam(struct net_device *netdev,
365 			      struct ethtool_pauseparam *pause)
366 {
367 	struct funeth_priv *fp = netdev_priv(netdev);
368 	u64 new_advert;
369 
370 	if (fp->port_caps & FUN_PORT_CAP_VPORT)
371 		return -EOPNOTSUPP;
372 	/* Forcing PAUSE settings with AN enabled is unsupported. */
373 	if (!pause->autoneg && (fp->advertising & FUN_PORT_CAP_AUTONEG))
374 		return -EOPNOTSUPP;
375 	if (pause->autoneg && !(fp->advertising & FUN_PORT_CAP_AUTONEG))
376 		return -EINVAL;
377 	if (pause->tx_pause && !(fp->port_caps & FUN_PORT_CAP_TX_PAUSE))
378 		return -EINVAL;
379 	if (pause->rx_pause && !(fp->port_caps & FUN_PORT_CAP_RX_PAUSE))
380 		return -EINVAL;
381 
382 	new_advert = fp->advertising & ~FUN_PORT_CAP_PAUSE_MASK;
383 	if (pause->tx_pause)
384 		new_advert |= FUN_PORT_CAP_TX_PAUSE;
385 	if (pause->rx_pause)
386 		new_advert |= FUN_PORT_CAP_RX_PAUSE;
387 
388 	return fun_change_advert(fp, new_advert);
389 }
390 
391 static int fun_restart_an(struct net_device *netdev)
392 {
393 	struct funeth_priv *fp = netdev_priv(netdev);
394 
395 	if (!(fp->advertising & FUN_PORT_CAP_AUTONEG))
396 		return -EOPNOTSUPP;
397 
398 	return fun_port_write_cmd(fp, FUN_ADMIN_PORT_KEY_ADVERT,
399 				  FUN_PORT_CAP_AUTONEG);
400 }
401 
402 static int fun_set_phys_id(struct net_device *netdev,
403 			   enum ethtool_phys_id_state state)
404 {
405 	struct funeth_priv *fp = netdev_priv(netdev);
406 	unsigned int beacon;
407 
408 	if (fp->port_caps & FUN_PORT_CAP_VPORT)
409 		return -EOPNOTSUPP;
410 	if (state != ETHTOOL_ID_ACTIVE && state != ETHTOOL_ID_INACTIVE)
411 		return -EOPNOTSUPP;
412 
413 	beacon = state == ETHTOOL_ID_ACTIVE ? FUN_PORT_LED_BEACON_ON :
414 					      FUN_PORT_LED_BEACON_OFF;
415 	return fun_port_write_cmd(fp, FUN_ADMIN_PORT_KEY_LED, beacon);
416 }
417 
418 static void fun_get_drvinfo(struct net_device *netdev,
419 			    struct ethtool_drvinfo *info)
420 {
421 	const struct funeth_priv *fp = netdev_priv(netdev);
422 
423 	strscpy(info->driver, KBUILD_MODNAME, sizeof(info->driver));
424 	strscpy(info->bus_info, pci_name(fp->pdev), sizeof(info->bus_info));
425 }
426 
427 static u32 fun_get_msglevel(struct net_device *netdev)
428 {
429 	const struct funeth_priv *fp = netdev_priv(netdev);
430 
431 	return fp->msg_enable;
432 }
433 
434 static void fun_set_msglevel(struct net_device *netdev, u32 value)
435 {
436 	struct funeth_priv *fp = netdev_priv(netdev);
437 
438 	fp->msg_enable = value;
439 }
440 
441 static int fun_get_regs_len(struct net_device *dev)
442 {
443 	return NVME_REG_ACQ + sizeof(u64);
444 }
445 
446 static void fun_get_regs(struct net_device *dev, struct ethtool_regs *regs,
447 			 void *buf)
448 {
449 	const struct funeth_priv *fp = netdev_priv(dev);
450 	void __iomem *bar = fp->fdev->bar;
451 
452 	regs->version = 0;
453 	*(u64 *)(buf + NVME_REG_CAP)   = readq(bar + NVME_REG_CAP);
454 	*(u32 *)(buf + NVME_REG_VS)    = readl(bar + NVME_REG_VS);
455 	*(u32 *)(buf + NVME_REG_INTMS) = readl(bar + NVME_REG_INTMS);
456 	*(u32 *)(buf + NVME_REG_INTMC) = readl(bar + NVME_REG_INTMC);
457 	*(u32 *)(buf + NVME_REG_CC)    = readl(bar + NVME_REG_CC);
458 	*(u32 *)(buf + NVME_REG_CSTS)  = readl(bar + NVME_REG_CSTS);
459 	*(u32 *)(buf + NVME_REG_AQA)   = readl(bar + NVME_REG_AQA);
460 	*(u64 *)(buf + NVME_REG_ASQ)   = readq(bar + NVME_REG_ASQ);
461 	*(u64 *)(buf + NVME_REG_ACQ)   = readq(bar + NVME_REG_ACQ);
462 }
463 
464 static int fun_get_coalesce(struct net_device *netdev,
465 			    struct ethtool_coalesce *coal,
466 			    struct kernel_ethtool_coalesce *kcoal,
467 			    struct netlink_ext_ack *ext_ack)
468 {
469 	const struct funeth_priv *fp = netdev_priv(netdev);
470 
471 	coal->rx_coalesce_usecs        = fp->rx_coal_usec;
472 	coal->rx_max_coalesced_frames  = fp->rx_coal_count;
473 	coal->use_adaptive_rx_coalesce = !fp->cq_irq_db;
474 	coal->tx_coalesce_usecs        = fp->tx_coal_usec;
475 	coal->tx_max_coalesced_frames  = fp->tx_coal_count;
476 	return 0;
477 }
478 
479 static int fun_set_coalesce(struct net_device *netdev,
480 			    struct ethtool_coalesce *coal,
481 			    struct kernel_ethtool_coalesce *kcoal,
482 			    struct netlink_ext_ack *ext_ack)
483 {
484 	struct funeth_priv *fp = netdev_priv(netdev);
485 	struct funeth_rxq **rxqs;
486 	unsigned int i, db_val;
487 
488 	if (coal->rx_coalesce_usecs > FUN_DB_INTCOAL_USEC_M ||
489 	    coal->rx_max_coalesced_frames > FUN_DB_INTCOAL_ENTRIES_M ||
490 	    (coal->rx_coalesce_usecs | coal->rx_max_coalesced_frames) == 0 ||
491 	    coal->tx_coalesce_usecs > FUN_DB_INTCOAL_USEC_M ||
492 	    coal->tx_max_coalesced_frames > FUN_DB_INTCOAL_ENTRIES_M ||
493 	    (coal->tx_coalesce_usecs | coal->tx_max_coalesced_frames) == 0)
494 		return -EINVAL;
495 
496 	/* a timer is required if there's any coalescing */
497 	if ((coal->rx_max_coalesced_frames > 1 && !coal->rx_coalesce_usecs) ||
498 	    (coal->tx_max_coalesced_frames > 1 && !coal->tx_coalesce_usecs))
499 		return -EINVAL;
500 
501 	fp->rx_coal_usec  = coal->rx_coalesce_usecs;
502 	fp->rx_coal_count = coal->rx_max_coalesced_frames;
503 	fp->tx_coal_usec  = coal->tx_coalesce_usecs;
504 	fp->tx_coal_count = coal->tx_max_coalesced_frames;
505 
506 	db_val = FUN_IRQ_CQ_DB(fp->rx_coal_usec, fp->rx_coal_count);
507 	WRITE_ONCE(fp->cq_irq_db, db_val);
508 
509 	rxqs = rtnl_dereference(fp->rxqs);
510 	if (!rxqs)
511 		return 0;
512 
513 	for (i = 0; i < netdev->real_num_rx_queues; i++)
514 		WRITE_ONCE(rxqs[i]->irq_db_val, db_val);
515 
516 	db_val = FUN_IRQ_SQ_DB(fp->tx_coal_usec, fp->tx_coal_count);
517 	for (i = 0; i < netdev->real_num_tx_queues; i++)
518 		WRITE_ONCE(fp->txqs[i]->irq_db_val, db_val);
519 
520 	return 0;
521 }
522 
523 static void fun_get_channels(struct net_device *netdev,
524 			     struct ethtool_channels *chan)
525 {
526 	chan->max_rx   = netdev->num_rx_queues;
527 	chan->rx_count = netdev->real_num_rx_queues;
528 
529 	chan->max_tx   = netdev->num_tx_queues;
530 	chan->tx_count = netdev->real_num_tx_queues;
531 }
532 
533 static int fun_set_channels(struct net_device *netdev,
534 			    struct ethtool_channels *chan)
535 {
536 	if (!chan->tx_count || !chan->rx_count)
537 		return -EINVAL;
538 
539 	if (chan->tx_count == netdev->real_num_tx_queues &&
540 	    chan->rx_count == netdev->real_num_rx_queues)
541 		return 0;
542 
543 	if (netif_running(netdev))
544 		return fun_change_num_queues(netdev, chan->tx_count,
545 					     chan->rx_count);
546 
547 	fun_set_ring_count(netdev, chan->tx_count, chan->rx_count);
548 	return 0;
549 }
550 
551 static void fun_get_ringparam(struct net_device *netdev,
552 			      struct ethtool_ringparam *ring,
553 			      struct kernel_ethtool_ringparam *kring,
554 			      struct netlink_ext_ack *extack)
555 {
556 	const struct funeth_priv *fp = netdev_priv(netdev);
557 	unsigned int max_depth = fp->fdev->q_depth;
558 
559 	/* We size CQs to be twice the RQ depth so max RQ depth is half the
560 	 * max queue depth.
561 	 */
562 	ring->rx_max_pending = max_depth / 2;
563 	ring->tx_max_pending = max_depth;
564 
565 	ring->rx_pending = fp->rq_depth;
566 	ring->tx_pending = fp->sq_depth;
567 
568 	kring->rx_buf_len = PAGE_SIZE;
569 	kring->cqe_size = FUNETH_CQE_SIZE;
570 }
571 
572 static int fun_set_ringparam(struct net_device *netdev,
573 			     struct ethtool_ringparam *ring,
574 			     struct kernel_ethtool_ringparam *kring,
575 			     struct netlink_ext_ack *extack)
576 {
577 	struct funeth_priv *fp = netdev_priv(netdev);
578 	int rc;
579 
580 	if (ring->rx_mini_pending || ring->rx_jumbo_pending)
581 		return -EINVAL;
582 
583 	/* queue depths must be powers-of-2 */
584 	if (!is_power_of_2(ring->rx_pending) ||
585 	    !is_power_of_2(ring->tx_pending))
586 		return -EINVAL;
587 
588 	if (ring->rx_pending < FUNETH_MIN_QDEPTH ||
589 	    ring->tx_pending < FUNETH_MIN_QDEPTH)
590 		return -EINVAL;
591 
592 	if (fp->sq_depth == ring->tx_pending &&
593 	    fp->rq_depth == ring->rx_pending)
594 		return 0;
595 
596 	if (netif_running(netdev)) {
597 		struct fun_qset req = {
598 			.cq_depth = 2 * ring->rx_pending,
599 			.rq_depth = ring->rx_pending,
600 			.sq_depth = ring->tx_pending
601 		};
602 
603 		rc = fun_replace_queues(netdev, &req, extack);
604 		if (rc)
605 			return rc;
606 	}
607 
608 	fp->sq_depth = ring->tx_pending;
609 	fp->rq_depth = ring->rx_pending;
610 	fp->cq_depth = 2 * fp->rq_depth;
611 	return 0;
612 }
613 
614 static int fun_get_sset_count(struct net_device *dev, int sset)
615 {
616 	const struct funeth_priv *fp = netdev_priv(dev);
617 	int n;
618 
619 	switch (sset) {
620 	case ETH_SS_STATS:
621 		n = (dev->real_num_tx_queues + 1) * ARRAY_SIZE(txq_stat_names) +
622 		    (dev->real_num_rx_queues + 1) * ARRAY_SIZE(rxq_stat_names) +
623 		    (fp->num_xdpqs + 1) * ARRAY_SIZE(xdpq_stat_names) +
624 		    ARRAY_SIZE(tls_stat_names);
625 		if (fp->port_caps & FUN_PORT_CAP_STATS) {
626 			n += ARRAY_SIZE(mac_tx_stat_names) +
627 			     ARRAY_SIZE(mac_rx_stat_names);
628 		}
629 		return n;
630 	default:
631 		break;
632 	}
633 	return 0;
634 }
635 
636 static void fun_get_strings(struct net_device *netdev, u32 sset, u8 *data)
637 {
638 	const struct funeth_priv *fp = netdev_priv(netdev);
639 	unsigned int i, j;
640 	u8 *p = data;
641 
642 	switch (sset) {
643 	case ETH_SS_STATS:
644 		if (fp->port_caps & FUN_PORT_CAP_STATS) {
645 			memcpy(p, mac_tx_stat_names, sizeof(mac_tx_stat_names));
646 			p += sizeof(mac_tx_stat_names);
647 			memcpy(p, mac_rx_stat_names, sizeof(mac_rx_stat_names));
648 			p += sizeof(mac_rx_stat_names);
649 		}
650 
651 		for (i = 0; i < netdev->real_num_tx_queues; i++) {
652 			for (j = 0; j < ARRAY_SIZE(txq_stat_names); j++)
653 				ethtool_sprintf(&p, "%s[%u]", txq_stat_names[j],
654 						i);
655 		}
656 		for (j = 0; j < ARRAY_SIZE(txq_stat_names); j++)
657 			ethtool_sprintf(&p, txq_stat_names[j]);
658 
659 		for (i = 0; i < fp->num_xdpqs; i++) {
660 			for (j = 0; j < ARRAY_SIZE(xdpq_stat_names); j++)
661 				ethtool_sprintf(&p, "%s[%u]",
662 						xdpq_stat_names[j], i);
663 		}
664 		for (j = 0; j < ARRAY_SIZE(xdpq_stat_names); j++)
665 			ethtool_sprintf(&p, xdpq_stat_names[j]);
666 
667 		for (i = 0; i < netdev->real_num_rx_queues; i++) {
668 			for (j = 0; j < ARRAY_SIZE(rxq_stat_names); j++)
669 				ethtool_sprintf(&p, "%s[%u]", rxq_stat_names[j],
670 						i);
671 		}
672 		for (j = 0; j < ARRAY_SIZE(rxq_stat_names); j++)
673 			ethtool_sprintf(&p, rxq_stat_names[j]);
674 
675 		for (j = 0; j < ARRAY_SIZE(tls_stat_names); j++)
676 			ethtool_sprintf(&p, tls_stat_names[j]);
677 		break;
678 	default:
679 		break;
680 	}
681 }
682 
683 static u64 *get_mac_stats(const struct funeth_priv *fp, u64 *data)
684 {
685 #define TX_STAT(s) \
686 	*data++ = be64_to_cpu(fp->stats[PORT_MAC_RX_STATS_MAX + PORT_MAC_TX_##s])
687 
688 	TX_STAT(etherStatsOctets);
689 	TX_STAT(etherStatsPkts);
690 	TX_STAT(VLANTransmittedOK);
691 	TX_STAT(ifOutUcastPkts);
692 	TX_STAT(ifOutMulticastPkts);
693 	TX_STAT(ifOutBroadcastPkts);
694 	TX_STAT(ifOutErrors);
695 	TX_STAT(CBFCPAUSEFramesTransmitted_0);
696 	TX_STAT(CBFCPAUSEFramesTransmitted_1);
697 	TX_STAT(CBFCPAUSEFramesTransmitted_2);
698 	TX_STAT(CBFCPAUSEFramesTransmitted_3);
699 	TX_STAT(CBFCPAUSEFramesTransmitted_4);
700 	TX_STAT(CBFCPAUSEFramesTransmitted_5);
701 	TX_STAT(CBFCPAUSEFramesTransmitted_6);
702 	TX_STAT(CBFCPAUSEFramesTransmitted_7);
703 	TX_STAT(CBFCPAUSEFramesTransmitted_8);
704 	TX_STAT(CBFCPAUSEFramesTransmitted_9);
705 	TX_STAT(CBFCPAUSEFramesTransmitted_10);
706 	TX_STAT(CBFCPAUSEFramesTransmitted_11);
707 	TX_STAT(CBFCPAUSEFramesTransmitted_12);
708 	TX_STAT(CBFCPAUSEFramesTransmitted_13);
709 	TX_STAT(CBFCPAUSEFramesTransmitted_14);
710 	TX_STAT(CBFCPAUSEFramesTransmitted_15);
711 
712 #define RX_STAT(s) *data++ = be64_to_cpu(fp->stats[PORT_MAC_RX_##s])
713 
714 	RX_STAT(etherStatsOctets);
715 	RX_STAT(etherStatsPkts);
716 	RX_STAT(VLANReceivedOK);
717 	RX_STAT(ifInUcastPkts);
718 	RX_STAT(ifInMulticastPkts);
719 	RX_STAT(ifInBroadcastPkts);
720 	RX_STAT(etherStatsDropEvents);
721 	RX_STAT(ifInErrors);
722 	RX_STAT(aAlignmentErrors);
723 	RX_STAT(CBFCPAUSEFramesReceived_0);
724 	RX_STAT(CBFCPAUSEFramesReceived_1);
725 	RX_STAT(CBFCPAUSEFramesReceived_2);
726 	RX_STAT(CBFCPAUSEFramesReceived_3);
727 	RX_STAT(CBFCPAUSEFramesReceived_4);
728 	RX_STAT(CBFCPAUSEFramesReceived_5);
729 	RX_STAT(CBFCPAUSEFramesReceived_6);
730 	RX_STAT(CBFCPAUSEFramesReceived_7);
731 	RX_STAT(CBFCPAUSEFramesReceived_8);
732 	RX_STAT(CBFCPAUSEFramesReceived_9);
733 	RX_STAT(CBFCPAUSEFramesReceived_10);
734 	RX_STAT(CBFCPAUSEFramesReceived_11);
735 	RX_STAT(CBFCPAUSEFramesReceived_12);
736 	RX_STAT(CBFCPAUSEFramesReceived_13);
737 	RX_STAT(CBFCPAUSEFramesReceived_14);
738 	RX_STAT(CBFCPAUSEFramesReceived_15);
739 
740 	return data;
741 
742 #undef TX_STAT
743 #undef RX_STAT
744 }
745 
746 static void fun_get_ethtool_stats(struct net_device *netdev,
747 				  struct ethtool_stats *stats, u64 *data)
748 {
749 	const struct funeth_priv *fp = netdev_priv(netdev);
750 	struct funeth_txq_stats txs;
751 	struct funeth_rxq_stats rxs;
752 	struct funeth_txq **xdpqs;
753 	struct funeth_rxq **rxqs;
754 	unsigned int i, start;
755 	u64 *totals, *tot;
756 
757 	if (fp->port_caps & FUN_PORT_CAP_STATS)
758 		data = get_mac_stats(fp, data);
759 
760 	rxqs = rtnl_dereference(fp->rxqs);
761 	if (!rxqs)
762 		return;
763 
764 #define ADD_STAT(cnt) do { \
765 	*data = (cnt); *tot++ += *data++; \
766 } while (0)
767 
768 	/* Tx queues */
769 	totals = data + netdev->real_num_tx_queues * ARRAY_SIZE(txq_stat_names);
770 
771 	for (i = 0; i < netdev->real_num_tx_queues; i++) {
772 		tot = totals;
773 
774 		FUN_QSTAT_READ(fp->txqs[i], start, txs);
775 
776 		ADD_STAT(txs.tx_pkts);
777 		ADD_STAT(txs.tx_bytes);
778 		ADD_STAT(txs.tx_cso);
779 		ADD_STAT(txs.tx_tso);
780 		ADD_STAT(txs.tx_encap_tso);
781 		ADD_STAT(txs.tx_more);
782 		ADD_STAT(txs.tx_nstops);
783 		ADD_STAT(txs.tx_nrestarts);
784 		ADD_STAT(txs.tx_map_err);
785 		ADD_STAT(txs.tx_tls_pkts);
786 		ADD_STAT(txs.tx_tls_bytes);
787 		ADD_STAT(txs.tx_tls_fallback);
788 		ADD_STAT(txs.tx_tls_drops);
789 	}
790 	data += ARRAY_SIZE(txq_stat_names);
791 
792 	/* XDP Tx queues */
793 	xdpqs = rtnl_dereference(fp->xdpqs);
794 	totals = data + fp->num_xdpqs * ARRAY_SIZE(xdpq_stat_names);
795 
796 	for (i = 0; i < fp->num_xdpqs; i++) {
797 		tot = totals;
798 
799 		FUN_QSTAT_READ(xdpqs[i], start, txs);
800 
801 		ADD_STAT(txs.tx_pkts);
802 		ADD_STAT(txs.tx_bytes);
803 		ADD_STAT(txs.tx_xdp_full);
804 		ADD_STAT(txs.tx_map_err);
805 	}
806 	data += ARRAY_SIZE(xdpq_stat_names);
807 
808 	/* Rx queues */
809 	totals = data + netdev->real_num_rx_queues * ARRAY_SIZE(rxq_stat_names);
810 
811 	for (i = 0; i < netdev->real_num_rx_queues; i++) {
812 		tot = totals;
813 
814 		FUN_QSTAT_READ(rxqs[i], start, rxs);
815 
816 		ADD_STAT(rxs.rx_pkts);
817 		ADD_STAT(rxs.rx_bytes);
818 		ADD_STAT(rxs.rx_cso);
819 		ADD_STAT(rxs.gro_pkts);
820 		ADD_STAT(rxs.gro_merged);
821 		ADD_STAT(rxs.xdp_tx);
822 		ADD_STAT(rxs.xdp_redir);
823 		ADD_STAT(rxs.xdp_drops);
824 		ADD_STAT(rxs.rx_bufs);
825 		ADD_STAT(rxs.rx_page_alloc);
826 		ADD_STAT(rxs.rx_mem_drops + rxs.xdp_err);
827 		ADD_STAT(rxs.rx_budget);
828 		ADD_STAT(rxs.rx_map_err);
829 	}
830 	data += ARRAY_SIZE(rxq_stat_names);
831 #undef ADD_STAT
832 
833 	*data++ = atomic64_read(&fp->tx_tls_add);
834 	*data++ = atomic64_read(&fp->tx_tls_del);
835 	*data++ = atomic64_read(&fp->tx_tls_resync);
836 }
837 
838 #define RX_STAT(fp, s) be64_to_cpu((fp)->stats[PORT_MAC_RX_##s])
839 #define TX_STAT(fp, s) \
840 	be64_to_cpu((fp)->stats[PORT_MAC_RX_STATS_MAX + PORT_MAC_TX_##s])
841 #define FEC_STAT(fp, s) \
842 	be64_to_cpu((fp)->stats[PORT_MAC_RX_STATS_MAX + \
843 				PORT_MAC_TX_STATS_MAX + PORT_MAC_FEC_##s])
844 
845 static void fun_get_pause_stats(struct net_device *netdev,
846 				struct ethtool_pause_stats *stats)
847 {
848 	const struct funeth_priv *fp = netdev_priv(netdev);
849 
850 	if (!(fp->port_caps & FUN_PORT_CAP_STATS))
851 		return;
852 
853 	stats->tx_pause_frames = TX_STAT(fp, aPAUSEMACCtrlFramesTransmitted);
854 	stats->rx_pause_frames = RX_STAT(fp, aPAUSEMACCtrlFramesReceived);
855 }
856 
857 static void fun_get_802_3_stats(struct net_device *netdev,
858 				struct ethtool_eth_mac_stats *stats)
859 {
860 	const struct funeth_priv *fp = netdev_priv(netdev);
861 
862 	if (!(fp->port_caps & FUN_PORT_CAP_STATS))
863 		return;
864 
865 	stats->FramesTransmittedOK = TX_STAT(fp, aFramesTransmittedOK);
866 	stats->FramesReceivedOK = RX_STAT(fp, aFramesReceivedOK);
867 	stats->FrameCheckSequenceErrors = RX_STAT(fp, aFrameCheckSequenceErrors);
868 	stats->OctetsTransmittedOK = TX_STAT(fp, OctetsTransmittedOK);
869 	stats->OctetsReceivedOK = RX_STAT(fp, OctetsReceivedOK);
870 	stats->InRangeLengthErrors = RX_STAT(fp, aInRangeLengthErrors);
871 	stats->FrameTooLongErrors = RX_STAT(fp, aFrameTooLongErrors);
872 }
873 
874 static void fun_get_802_3_ctrl_stats(struct net_device *netdev,
875 				     struct ethtool_eth_ctrl_stats *stats)
876 {
877 	const struct funeth_priv *fp = netdev_priv(netdev);
878 
879 	if (!(fp->port_caps & FUN_PORT_CAP_STATS))
880 		return;
881 
882 	stats->MACControlFramesTransmitted = TX_STAT(fp, MACControlFramesTransmitted);
883 	stats->MACControlFramesReceived = RX_STAT(fp, MACControlFramesReceived);
884 }
885 
886 static void fun_get_rmon_stats(struct net_device *netdev,
887 			       struct ethtool_rmon_stats *stats,
888 			       const struct ethtool_rmon_hist_range **ranges)
889 {
890 	static const struct ethtool_rmon_hist_range rmon_ranges[] = {
891 		{   64,    64 },
892 		{   65,   127 },
893 		{  128,   255 },
894 		{  256,   511 },
895 		{  512,  1023 },
896 		{ 1024,  1518 },
897 		{ 1519, 32767 },
898 		{}
899 	};
900 
901 	const struct funeth_priv *fp = netdev_priv(netdev);
902 
903 	if (!(fp->port_caps & FUN_PORT_CAP_STATS))
904 		return;
905 
906 	stats->undersize_pkts = RX_STAT(fp, etherStatsUndersizePkts);
907 	stats->oversize_pkts = RX_STAT(fp, etherStatsOversizePkts);
908 	stats->fragments = RX_STAT(fp, etherStatsFragments);
909 	stats->jabbers = RX_STAT(fp, etherStatsJabbers);
910 
911 	stats->hist[0] = RX_STAT(fp, etherStatsPkts64Octets);
912 	stats->hist[1] = RX_STAT(fp, etherStatsPkts65to127Octets);
913 	stats->hist[2] = RX_STAT(fp, etherStatsPkts128to255Octets);
914 	stats->hist[3] = RX_STAT(fp, etherStatsPkts256to511Octets);
915 	stats->hist[4] = RX_STAT(fp, etherStatsPkts512to1023Octets);
916 	stats->hist[5] = RX_STAT(fp, etherStatsPkts1024to1518Octets);
917 	stats->hist[6] = RX_STAT(fp, etherStatsPkts1519toMaxOctets);
918 
919 	stats->hist_tx[0] = TX_STAT(fp, etherStatsPkts64Octets);
920 	stats->hist_tx[1] = TX_STAT(fp, etherStatsPkts65to127Octets);
921 	stats->hist_tx[2] = TX_STAT(fp, etherStatsPkts128to255Octets);
922 	stats->hist_tx[3] = TX_STAT(fp, etherStatsPkts256to511Octets);
923 	stats->hist_tx[4] = TX_STAT(fp, etherStatsPkts512to1023Octets);
924 	stats->hist_tx[5] = TX_STAT(fp, etherStatsPkts1024to1518Octets);
925 	stats->hist_tx[6] = TX_STAT(fp, etherStatsPkts1519toMaxOctets);
926 
927 	*ranges = rmon_ranges;
928 }
929 
930 static void fun_get_fec_stats(struct net_device *netdev,
931 			      struct ethtool_fec_stats *stats)
932 {
933 	const struct funeth_priv *fp = netdev_priv(netdev);
934 
935 	if (!(fp->port_caps & FUN_PORT_CAP_STATS))
936 		return;
937 
938 	stats->corrected_blocks.total = FEC_STAT(fp, Correctable);
939 	stats->uncorrectable_blocks.total = FEC_STAT(fp, Uncorrectable);
940 }
941 
942 #undef RX_STAT
943 #undef TX_STAT
944 #undef FEC_STAT
945 
946 static int fun_get_rxnfc(struct net_device *netdev, struct ethtool_rxnfc *cmd,
947 			 u32 *rule_locs)
948 {
949 	switch (cmd->cmd) {
950 	case ETHTOOL_GRXRINGS:
951 		cmd->data = netdev->real_num_rx_queues;
952 		return 0;
953 	default:
954 		break;
955 	}
956 	return -EOPNOTSUPP;
957 }
958 
959 static int fun_set_rxnfc(struct net_device *netdev, struct ethtool_rxnfc *info)
960 {
961 	return 0;
962 }
963 
964 static u32 fun_get_rxfh_indir_size(struct net_device *netdev)
965 {
966 	const struct funeth_priv *fp = netdev_priv(netdev);
967 
968 	return fp->indir_table_nentries;
969 }
970 
971 static u32 fun_get_rxfh_key_size(struct net_device *netdev)
972 {
973 	const struct funeth_priv *fp = netdev_priv(netdev);
974 
975 	return sizeof(fp->rss_key);
976 }
977 
978 static int fun_get_rxfh(struct net_device *netdev, u32 *indir, u8 *key,
979 			u8 *hfunc)
980 {
981 	const struct funeth_priv *fp = netdev_priv(netdev);
982 
983 	if (!fp->rss_cfg)
984 		return -EOPNOTSUPP;
985 
986 	if (indir)
987 		memcpy(indir, fp->indir_table,
988 		       sizeof(u32) * fp->indir_table_nentries);
989 
990 	if (key)
991 		memcpy(key, fp->rss_key, sizeof(fp->rss_key));
992 
993 	if (hfunc)
994 		*hfunc = fp->hash_algo == FUN_ETH_RSS_ALG_TOEPLITZ ?
995 				ETH_RSS_HASH_TOP : ETH_RSS_HASH_CRC32;
996 
997 	return 0;
998 }
999 
1000 static int fun_set_rxfh(struct net_device *netdev, const u32 *indir,
1001 			const u8 *key, const u8 hfunc)
1002 {
1003 	struct funeth_priv *fp = netdev_priv(netdev);
1004 	const u32 *rss_indir = indir ? indir : fp->indir_table;
1005 	const u8 *rss_key = key ? key : fp->rss_key;
1006 	enum fun_eth_hash_alg algo;
1007 
1008 	if (!fp->rss_cfg)
1009 		return -EOPNOTSUPP;
1010 
1011 	if (hfunc == ETH_RSS_HASH_NO_CHANGE)
1012 		algo = fp->hash_algo;
1013 	else if (hfunc == ETH_RSS_HASH_CRC32)
1014 		algo = FUN_ETH_RSS_ALG_CRC32;
1015 	else if (hfunc == ETH_RSS_HASH_TOP)
1016 		algo = FUN_ETH_RSS_ALG_TOEPLITZ;
1017 	else
1018 		return -EINVAL;
1019 
1020 	/* If the port is enabled try to reconfigure RSS and keep the new
1021 	 * settings if successful. If it is down we update the RSS settings
1022 	 * and apply them at the next UP time.
1023 	 */
1024 	if (netif_running(netdev)) {
1025 		int rc = fun_config_rss(netdev, algo, rss_key, rss_indir,
1026 					FUN_ADMIN_SUBOP_MODIFY);
1027 		if (rc)
1028 			return rc;
1029 	}
1030 
1031 	fp->hash_algo = algo;
1032 	if (key)
1033 		memcpy(fp->rss_key, key, sizeof(fp->rss_key));
1034 	if (indir)
1035 		memcpy(fp->indir_table, indir,
1036 		       sizeof(u32) * fp->indir_table_nentries);
1037 	return 0;
1038 }
1039 
1040 static int fun_get_ts_info(struct net_device *netdev,
1041 			   struct ethtool_ts_info *info)
1042 {
1043 	info->so_timestamping = SOF_TIMESTAMPING_RX_SOFTWARE |
1044 				SOF_TIMESTAMPING_RX_HARDWARE |
1045 				SOF_TIMESTAMPING_TX_SOFTWARE |
1046 				SOF_TIMESTAMPING_SOFTWARE |
1047 				SOF_TIMESTAMPING_RAW_HARDWARE;
1048 	info->phc_index = -1;
1049 	info->tx_types = BIT(HWTSTAMP_TX_OFF);
1050 	info->rx_filters = BIT(HWTSTAMP_FILTER_NONE) | BIT(HWTSTAMP_FILTER_ALL);
1051 	return 0;
1052 }
1053 
1054 static unsigned int to_ethtool_fec(unsigned int fun_fec)
1055 {
1056 	unsigned int fec = 0;
1057 
1058 	if (fun_fec == FUN_PORT_FEC_NA)
1059 		fec |= ETHTOOL_FEC_NONE;
1060 	if (fun_fec & FUN_PORT_FEC_OFF)
1061 		fec |= ETHTOOL_FEC_OFF;
1062 	if (fun_fec & FUN_PORT_FEC_RS)
1063 		fec |= ETHTOOL_FEC_RS;
1064 	if (fun_fec & FUN_PORT_FEC_FC)
1065 		fec |= ETHTOOL_FEC_BASER;
1066 	if (fun_fec & FUN_PORT_FEC_AUTO)
1067 		fec |= ETHTOOL_FEC_AUTO;
1068 	return fec;
1069 }
1070 
1071 static int fun_get_fecparam(struct net_device *netdev,
1072 			    struct ethtool_fecparam *fec)
1073 {
1074 	struct funeth_priv *fp = netdev_priv(netdev);
1075 	u64 fec_data;
1076 	int rc;
1077 
1078 	rc = fun_port_read_cmd(fp, FUN_ADMIN_PORT_KEY_FEC, &fec_data);
1079 	if (rc)
1080 		return rc;
1081 
1082 	fec->active_fec = to_ethtool_fec(fec_data & 0xff);
1083 	fec->fec = to_ethtool_fec(fec_data >> 8);
1084 	return 0;
1085 }
1086 
1087 static int fun_set_fecparam(struct net_device *netdev,
1088 			    struct ethtool_fecparam *fec)
1089 {
1090 	struct funeth_priv *fp = netdev_priv(netdev);
1091 	u64 fec_mode;
1092 
1093 	switch (fec->fec) {
1094 	case ETHTOOL_FEC_AUTO:
1095 		fec_mode = FUN_PORT_FEC_AUTO;
1096 		break;
1097 	case ETHTOOL_FEC_OFF:
1098 		if (!(fp->port_caps & FUN_PORT_CAP_FEC_NONE))
1099 			return -EINVAL;
1100 		fec_mode = FUN_PORT_FEC_OFF;
1101 		break;
1102 	case ETHTOOL_FEC_BASER:
1103 		if (!(fp->port_caps & FUN_PORT_CAP_FEC_FC))
1104 			return -EINVAL;
1105 		fec_mode = FUN_PORT_FEC_FC;
1106 		break;
1107 	case ETHTOOL_FEC_RS:
1108 		if (!(fp->port_caps & FUN_PORT_CAP_FEC_RS))
1109 			return -EINVAL;
1110 		fec_mode = FUN_PORT_FEC_RS;
1111 		break;
1112 	default:
1113 		return -EINVAL;
1114 	}
1115 
1116 	return fun_port_write_cmd(fp, FUN_ADMIN_PORT_KEY_FEC, fec_mode);
1117 }
1118 
1119 static const struct ethtool_ops fun_ethtool_ops = {
1120 	.supported_coalesce_params = ETHTOOL_COALESCE_USECS |
1121 				     ETHTOOL_COALESCE_MAX_FRAMES,
1122 	.get_link_ksettings  = fun_get_link_ksettings,
1123 	.set_link_ksettings  = fun_set_link_ksettings,
1124 	.set_phys_id         = fun_set_phys_id,
1125 	.get_drvinfo         = fun_get_drvinfo,
1126 	.get_msglevel        = fun_get_msglevel,
1127 	.set_msglevel        = fun_set_msglevel,
1128 	.get_regs_len        = fun_get_regs_len,
1129 	.get_regs            = fun_get_regs,
1130 	.get_link	     = ethtool_op_get_link,
1131 	.get_coalesce        = fun_get_coalesce,
1132 	.set_coalesce        = fun_set_coalesce,
1133 	.get_ts_info         = fun_get_ts_info,
1134 	.get_ringparam       = fun_get_ringparam,
1135 	.set_ringparam       = fun_set_ringparam,
1136 	.get_sset_count      = fun_get_sset_count,
1137 	.get_strings         = fun_get_strings,
1138 	.get_ethtool_stats   = fun_get_ethtool_stats,
1139 	.get_rxnfc	     = fun_get_rxnfc,
1140 	.set_rxnfc           = fun_set_rxnfc,
1141 	.get_rxfh_indir_size = fun_get_rxfh_indir_size,
1142 	.get_rxfh_key_size   = fun_get_rxfh_key_size,
1143 	.get_rxfh            = fun_get_rxfh,
1144 	.set_rxfh            = fun_set_rxfh,
1145 	.get_channels        = fun_get_channels,
1146 	.set_channels        = fun_set_channels,
1147 	.get_fecparam	     = fun_get_fecparam,
1148 	.set_fecparam	     = fun_set_fecparam,
1149 	.get_pauseparam      = fun_get_pauseparam,
1150 	.set_pauseparam      = fun_set_pauseparam,
1151 	.nway_reset          = fun_restart_an,
1152 	.get_pause_stats     = fun_get_pause_stats,
1153 	.get_fec_stats       = fun_get_fec_stats,
1154 	.get_eth_mac_stats   = fun_get_802_3_stats,
1155 	.get_eth_ctrl_stats  = fun_get_802_3_ctrl_stats,
1156 	.get_rmon_stats      = fun_get_rmon_stats,
1157 };
1158 
1159 void fun_set_ethtool_ops(struct net_device *netdev)
1160 {
1161 	netdev->ethtool_ops = &fun_ethtool_ops;
1162 }
1163