• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C) 2015-2017 Netronome Systems, Inc.
3  *
4  * This software is dual licensed under the GNU General License Version 2,
5  * June 1991 as shown in the file COPYING in the top-level directory of this
6  * source tree or the BSD 2-Clause License provided below.  You have the
7  * option to license this software under the complete terms of either license.
8  *
9  * The BSD 2-Clause License:
10  *
11  *     Redistribution and use in source and binary forms, with or
12  *     without modification, are permitted provided that the following
13  *     conditions are met:
14  *
15  *      1. Redistributions of source code must retain the above
16  *         copyright notice, this list of conditions and the following
17  *         disclaimer.
18  *
19  *      2. Redistributions in binary form must reproduce the above
20  *         copyright notice, this list of conditions and the following
21  *         disclaimer in the documentation and/or other materials
22  *         provided with the distribution.
23  *
24  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
25  * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
26  * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
27  * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
28  * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
29  * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
30  * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
31  * SOFTWARE.
32  */
33 
34 /* Authors: David Brunecz <david.brunecz@netronome.com>
35  *          Jakub Kicinski <jakub.kicinski@netronome.com>
36  *          Jason Mcmullan <jason.mcmullan@netronome.com>
37  */
38 
39 #include <linux/bitfield.h>
40 #include <linux/ethtool.h>
41 #include <linux/if_ether.h>
42 #include <linux/kernel.h>
43 #include <linux/module.h>
44 
45 #include "nfp.h"
46 #include "nfp_nsp.h"
47 #include "nfp6000/nfp6000.h"
48 
49 #define NSP_ETH_NBI_PORT_COUNT		24
50 #define NSP_ETH_MAX_COUNT		(2 * NSP_ETH_NBI_PORT_COUNT)
51 #define NSP_ETH_TABLE_SIZE		(NSP_ETH_MAX_COUNT *		\
52 					 sizeof(union eth_table_entry))
53 
54 #define NSP_ETH_PORT_LANES		GENMASK_ULL(3, 0)
55 #define NSP_ETH_PORT_INDEX		GENMASK_ULL(15, 8)
56 #define NSP_ETH_PORT_LABEL		GENMASK_ULL(53, 48)
57 #define NSP_ETH_PORT_PHYLABEL		GENMASK_ULL(59, 54)
58 
59 #define NSP_ETH_PORT_LANES_MASK		cpu_to_le64(NSP_ETH_PORT_LANES)
60 
61 #define NSP_ETH_STATE_CONFIGURED	BIT_ULL(0)
62 #define NSP_ETH_STATE_ENABLED		BIT_ULL(1)
63 #define NSP_ETH_STATE_TX_ENABLED	BIT_ULL(2)
64 #define NSP_ETH_STATE_RX_ENABLED	BIT_ULL(3)
65 #define NSP_ETH_STATE_RATE		GENMASK_ULL(11, 8)
66 #define NSP_ETH_STATE_INTERFACE		GENMASK_ULL(19, 12)
67 #define NSP_ETH_STATE_MEDIA		GENMASK_ULL(21, 20)
68 #define NSP_ETH_STATE_OVRD_CHNG		BIT_ULL(22)
69 #define NSP_ETH_STATE_ANEG		GENMASK_ULL(25, 23)
70 
71 #define NSP_ETH_CTRL_CONFIGURED		BIT_ULL(0)
72 #define NSP_ETH_CTRL_ENABLED		BIT_ULL(1)
73 #define NSP_ETH_CTRL_TX_ENABLED		BIT_ULL(2)
74 #define NSP_ETH_CTRL_RX_ENABLED		BIT_ULL(3)
75 #define NSP_ETH_CTRL_SET_RATE		BIT_ULL(4)
76 #define NSP_ETH_CTRL_SET_LANES		BIT_ULL(5)
77 #define NSP_ETH_CTRL_SET_ANEG		BIT_ULL(6)
78 
79 enum nfp_eth_raw {
80 	NSP_ETH_RAW_PORT = 0,
81 	NSP_ETH_RAW_STATE,
82 	NSP_ETH_RAW_MAC,
83 	NSP_ETH_RAW_CONTROL,
84 
85 	NSP_ETH_NUM_RAW
86 };
87 
88 enum nfp_eth_rate {
89 	RATE_INVALID = 0,
90 	RATE_10M,
91 	RATE_100M,
92 	RATE_1G,
93 	RATE_10G,
94 	RATE_25G,
95 };
96 
97 union eth_table_entry {
98 	struct {
99 		__le64 port;
100 		__le64 state;
101 		u8 mac_addr[6];
102 		u8 resv[2];
103 		__le64 control;
104 	};
105 	__le64 raw[NSP_ETH_NUM_RAW];
106 };
107 
108 static const struct {
109 	enum nfp_eth_rate rate;
110 	unsigned int speed;
111 } nsp_eth_rate_tbl[] = {
112 	{ RATE_INVALID,	0, },
113 	{ RATE_10M,	SPEED_10, },
114 	{ RATE_100M,	SPEED_100, },
115 	{ RATE_1G,	SPEED_1000, },
116 	{ RATE_10G,	SPEED_10000, },
117 	{ RATE_25G,	SPEED_25000, },
118 };
119 
nfp_eth_rate2speed(enum nfp_eth_rate rate)120 static unsigned int nfp_eth_rate2speed(enum nfp_eth_rate rate)
121 {
122 	int i;
123 
124 	for (i = 0; i < ARRAY_SIZE(nsp_eth_rate_tbl); i++)
125 		if (nsp_eth_rate_tbl[i].rate == rate)
126 			return nsp_eth_rate_tbl[i].speed;
127 
128 	return 0;
129 }
130 
nfp_eth_speed2rate(unsigned int speed)131 static unsigned int nfp_eth_speed2rate(unsigned int speed)
132 {
133 	int i;
134 
135 	for (i = 0; i < ARRAY_SIZE(nsp_eth_rate_tbl); i++)
136 		if (nsp_eth_rate_tbl[i].speed == speed)
137 			return nsp_eth_rate_tbl[i].rate;
138 
139 	return RATE_INVALID;
140 }
141 
nfp_eth_copy_mac_reverse(u8 * dst,const u8 * src)142 static void nfp_eth_copy_mac_reverse(u8 *dst, const u8 *src)
143 {
144 	int i;
145 
146 	for (i = 0; i < ETH_ALEN; i++)
147 		dst[ETH_ALEN - i - 1] = src[i];
148 }
149 
150 static void
nfp_eth_port_translate(struct nfp_nsp * nsp,const union eth_table_entry * src,unsigned int index,struct nfp_eth_table_port * dst)151 nfp_eth_port_translate(struct nfp_nsp *nsp, const union eth_table_entry *src,
152 		       unsigned int index, struct nfp_eth_table_port *dst)
153 {
154 	unsigned int rate;
155 	u64 port, state;
156 
157 	port = le64_to_cpu(src->port);
158 	state = le64_to_cpu(src->state);
159 
160 	dst->eth_index = FIELD_GET(NSP_ETH_PORT_INDEX, port);
161 	dst->index = index;
162 	dst->nbi = index / NSP_ETH_NBI_PORT_COUNT;
163 	dst->base = index % NSP_ETH_NBI_PORT_COUNT;
164 	dst->lanes = FIELD_GET(NSP_ETH_PORT_LANES, port);
165 
166 	dst->enabled = FIELD_GET(NSP_ETH_STATE_ENABLED, state);
167 	dst->tx_enabled = FIELD_GET(NSP_ETH_STATE_TX_ENABLED, state);
168 	dst->rx_enabled = FIELD_GET(NSP_ETH_STATE_RX_ENABLED, state);
169 
170 	rate = nfp_eth_rate2speed(FIELD_GET(NSP_ETH_STATE_RATE, state));
171 	dst->speed = dst->lanes * rate;
172 
173 	dst->interface = FIELD_GET(NSP_ETH_STATE_INTERFACE, state);
174 	dst->media = FIELD_GET(NSP_ETH_STATE_MEDIA, state);
175 
176 	nfp_eth_copy_mac_reverse(dst->mac_addr, src->mac_addr);
177 
178 	dst->label_port = FIELD_GET(NSP_ETH_PORT_PHYLABEL, port);
179 	dst->label_subport = FIELD_GET(NSP_ETH_PORT_LABEL, port);
180 
181 	if (nfp_nsp_get_abi_ver_minor(nsp) < 17)
182 		return;
183 
184 	dst->override_changed = FIELD_GET(NSP_ETH_STATE_OVRD_CHNG, state);
185 	dst->aneg = FIELD_GET(NSP_ETH_STATE_ANEG, state);
186 }
187 
188 static void
nfp_eth_calc_port_geometry(struct nfp_cpp * cpp,struct nfp_eth_table * table)189 nfp_eth_calc_port_geometry(struct nfp_cpp *cpp, struct nfp_eth_table *table)
190 {
191 	unsigned int i, j;
192 
193 	for (i = 0; i < table->count; i++) {
194 		table->max_index = max(table->max_index, table->ports[i].index);
195 
196 		for (j = 0; j < table->count; j++) {
197 			if (table->ports[i].label_port !=
198 			    table->ports[j].label_port)
199 				continue;
200 			table->ports[i].port_lanes += table->ports[j].lanes;
201 
202 			if (i == j)
203 				continue;
204 			if (table->ports[i].label_subport ==
205 			    table->ports[j].label_subport)
206 				nfp_warn(cpp,
207 					 "Port %d subport %d is a duplicate\n",
208 					 table->ports[i].label_port,
209 					 table->ports[i].label_subport);
210 
211 			table->ports[i].is_split = true;
212 		}
213 	}
214 }
215 
216 static void
nfp_eth_calc_port_type(struct nfp_cpp * cpp,struct nfp_eth_table_port * entry)217 nfp_eth_calc_port_type(struct nfp_cpp *cpp, struct nfp_eth_table_port *entry)
218 {
219 	if (entry->interface == NFP_INTERFACE_NONE) {
220 		entry->port_type = PORT_NONE;
221 		return;
222 	}
223 
224 	if (entry->media == NFP_MEDIA_FIBRE)
225 		entry->port_type = PORT_FIBRE;
226 	else
227 		entry->port_type = PORT_DA;
228 }
229 
230 /**
231  * nfp_eth_read_ports() - retrieve port information
232  * @cpp:	NFP CPP handle
233  *
234  * Read the port information from the device.  Returned structure should
235  * be freed with kfree() once no longer needed.
236  *
237  * Return: populated ETH table or NULL on error.
238  */
nfp_eth_read_ports(struct nfp_cpp * cpp)239 struct nfp_eth_table *nfp_eth_read_ports(struct nfp_cpp *cpp)
240 {
241 	struct nfp_eth_table *ret;
242 	struct nfp_nsp *nsp;
243 
244 	nsp = nfp_nsp_open(cpp);
245 	if (IS_ERR(nsp))
246 		return NULL;
247 
248 	ret = __nfp_eth_read_ports(cpp, nsp);
249 	nfp_nsp_close(nsp);
250 
251 	return ret;
252 }
253 
254 struct nfp_eth_table *
__nfp_eth_read_ports(struct nfp_cpp * cpp,struct nfp_nsp * nsp)255 __nfp_eth_read_ports(struct nfp_cpp *cpp, struct nfp_nsp *nsp)
256 {
257 	union eth_table_entry *entries;
258 	struct nfp_eth_table *table;
259 	int i, j, ret, cnt = 0;
260 
261 	entries = kzalloc(NSP_ETH_TABLE_SIZE, GFP_KERNEL);
262 	if (!entries)
263 		return NULL;
264 
265 	ret = nfp_nsp_read_eth_table(nsp, entries, NSP_ETH_TABLE_SIZE);
266 	if (ret < 0) {
267 		nfp_err(cpp, "reading port table failed %d\n", ret);
268 		goto err;
269 	}
270 
271 	for (i = 0; i < NSP_ETH_MAX_COUNT; i++)
272 		if (entries[i].port & NSP_ETH_PORT_LANES_MASK)
273 			cnt++;
274 
275 	/* Some versions of flash will give us 0 instead of port count.
276 	 * For those that give a port count, verify it against the value
277 	 * calculated above.
278 	 */
279 	if (ret && ret != cnt) {
280 		nfp_err(cpp, "table entry count reported (%d) does not match entries present (%d)\n",
281 			ret, cnt);
282 		goto err;
283 	}
284 
285 	table = kzalloc(sizeof(*table) +
286 			sizeof(struct nfp_eth_table_port) * cnt, GFP_KERNEL);
287 	if (!table)
288 		goto err;
289 
290 	table->count = cnt;
291 	for (i = 0, j = 0; i < NSP_ETH_MAX_COUNT; i++)
292 		if (entries[i].port & NSP_ETH_PORT_LANES_MASK)
293 			nfp_eth_port_translate(nsp, &entries[i], i,
294 					       &table->ports[j++]);
295 
296 	nfp_eth_calc_port_geometry(cpp, table);
297 	for (i = 0; i < table->count; i++)
298 		nfp_eth_calc_port_type(cpp, &table->ports[i]);
299 
300 	kfree(entries);
301 
302 	return table;
303 
304 err:
305 	kfree(entries);
306 	return NULL;
307 }
308 
nfp_eth_config_start(struct nfp_cpp * cpp,unsigned int idx)309 struct nfp_nsp *nfp_eth_config_start(struct nfp_cpp *cpp, unsigned int idx)
310 {
311 	union eth_table_entry *entries;
312 	struct nfp_nsp *nsp;
313 	int ret;
314 
315 	entries = kzalloc(NSP_ETH_TABLE_SIZE, GFP_KERNEL);
316 	if (!entries)
317 		return ERR_PTR(-ENOMEM);
318 
319 	nsp = nfp_nsp_open(cpp);
320 	if (IS_ERR(nsp)) {
321 		kfree(entries);
322 		return nsp;
323 	}
324 
325 	ret = nfp_nsp_read_eth_table(nsp, entries, NSP_ETH_TABLE_SIZE);
326 	if (ret < 0) {
327 		nfp_err(cpp, "reading port table failed %d\n", ret);
328 		goto err;
329 	}
330 
331 	if (!(entries[idx].port & NSP_ETH_PORT_LANES_MASK)) {
332 		nfp_warn(cpp, "trying to set port state on disabled port %d\n",
333 			 idx);
334 		goto err;
335 	}
336 
337 	nfp_nsp_config_set_state(nsp, entries, idx);
338 	return nsp;
339 
340 err:
341 	nfp_nsp_close(nsp);
342 	kfree(entries);
343 	return ERR_PTR(-EIO);
344 }
345 
nfp_eth_config_cleanup_end(struct nfp_nsp * nsp)346 void nfp_eth_config_cleanup_end(struct nfp_nsp *nsp)
347 {
348 	union eth_table_entry *entries = nfp_nsp_config_entries(nsp);
349 
350 	nfp_nsp_config_set_modified(nsp, false);
351 	nfp_nsp_config_clear_state(nsp);
352 	nfp_nsp_close(nsp);
353 	kfree(entries);
354 }
355 
356 /**
357  * nfp_eth_config_commit_end() - perform recorded configuration changes
358  * @nsp:	NFP NSP handle returned from nfp_eth_config_start()
359  *
360  * Perform the configuration which was requested with __nfp_eth_set_*()
361  * helpers and recorded in @nsp state.  If device was already configured
362  * as requested or no __nfp_eth_set_*() operations were made no NSP command
363  * will be performed.
364  *
365  * Return:
366  * 0 - configuration successful;
367  * 1 - no changes were needed;
368  * -ERRNO - configuration failed.
369  */
nfp_eth_config_commit_end(struct nfp_nsp * nsp)370 int nfp_eth_config_commit_end(struct nfp_nsp *nsp)
371 {
372 	union eth_table_entry *entries = nfp_nsp_config_entries(nsp);
373 	int ret = 1;
374 
375 	if (nfp_nsp_config_modified(nsp)) {
376 		ret = nfp_nsp_write_eth_table(nsp, entries, NSP_ETH_TABLE_SIZE);
377 		ret = ret < 0 ? ret : 0;
378 	}
379 
380 	nfp_eth_config_cleanup_end(nsp);
381 
382 	return ret;
383 }
384 
385 /**
386  * nfp_eth_set_mod_enable() - set PHY module enable control bit
387  * @cpp:	NFP CPP handle
388  * @idx:	NFP chip-wide port index
389  * @enable:	Desired state
390  *
391  * Enable or disable PHY module (this usually means setting the TX lanes
392  * disable bits).
393  *
394  * Return:
395  * 0 - configuration successful;
396  * 1 - no changes were needed;
397  * -ERRNO - configuration failed.
398  */
nfp_eth_set_mod_enable(struct nfp_cpp * cpp,unsigned int idx,bool enable)399 int nfp_eth_set_mod_enable(struct nfp_cpp *cpp, unsigned int idx, bool enable)
400 {
401 	union eth_table_entry *entries;
402 	struct nfp_nsp *nsp;
403 	u64 reg;
404 
405 	nsp = nfp_eth_config_start(cpp, idx);
406 	if (IS_ERR(nsp))
407 		return PTR_ERR(nsp);
408 
409 	entries = nfp_nsp_config_entries(nsp);
410 
411 	/* Check if we are already in requested state */
412 	reg = le64_to_cpu(entries[idx].state);
413 	if (enable != FIELD_GET(NSP_ETH_CTRL_ENABLED, reg)) {
414 		reg = le64_to_cpu(entries[idx].control);
415 		reg &= ~NSP_ETH_CTRL_ENABLED;
416 		reg |= FIELD_PREP(NSP_ETH_CTRL_ENABLED, enable);
417 		entries[idx].control = cpu_to_le64(reg);
418 
419 		nfp_nsp_config_set_modified(nsp, true);
420 	}
421 
422 	return nfp_eth_config_commit_end(nsp);
423 }
424 
425 /**
426  * nfp_eth_set_configured() - set PHY module configured control bit
427  * @cpp:	NFP CPP handle
428  * @idx:	NFP chip-wide port index
429  * @configed:	Desired state
430  *
431  * Set the ifup/ifdown state on the PHY.
432  *
433  * Return:
434  * 0 - configuration successful;
435  * 1 - no changes were needed;
436  * -ERRNO - configuration failed.
437  */
nfp_eth_set_configured(struct nfp_cpp * cpp,unsigned int idx,bool configed)438 int nfp_eth_set_configured(struct nfp_cpp *cpp, unsigned int idx, bool configed)
439 {
440 	union eth_table_entry *entries;
441 	struct nfp_nsp *nsp;
442 	u64 reg;
443 
444 	nsp = nfp_eth_config_start(cpp, idx);
445 	if (IS_ERR(nsp))
446 		return PTR_ERR(nsp);
447 
448 	/* Older ABI versions did support this feature, however this has only
449 	 * been reliable since ABI 20.
450 	 */
451 	if (nfp_nsp_get_abi_ver_minor(nsp) < 20) {
452 		nfp_eth_config_cleanup_end(nsp);
453 		return -EOPNOTSUPP;
454 	}
455 
456 	entries = nfp_nsp_config_entries(nsp);
457 
458 	/* Check if we are already in requested state */
459 	reg = le64_to_cpu(entries[idx].state);
460 	if (configed != FIELD_GET(NSP_ETH_STATE_CONFIGURED, reg)) {
461 		reg = le64_to_cpu(entries[idx].control);
462 		reg &= ~NSP_ETH_CTRL_CONFIGURED;
463 		reg |= FIELD_PREP(NSP_ETH_CTRL_CONFIGURED, configed);
464 		entries[idx].control = cpu_to_le64(reg);
465 
466 		nfp_nsp_config_set_modified(nsp, true);
467 	}
468 
469 	return nfp_eth_config_commit_end(nsp);
470 }
471 
472 /* Force inline, FIELD_* macroes require masks to be compilation-time known */
473 static __always_inline int
nfp_eth_set_bit_config(struct nfp_nsp * nsp,unsigned int raw_idx,const u64 mask,unsigned int val,const u64 ctrl_bit)474 nfp_eth_set_bit_config(struct nfp_nsp *nsp, unsigned int raw_idx,
475 		       const u64 mask, unsigned int val, const u64 ctrl_bit)
476 {
477 	union eth_table_entry *entries = nfp_nsp_config_entries(nsp);
478 	unsigned int idx = nfp_nsp_config_idx(nsp);
479 	u64 reg;
480 
481 	/* Note: set features were added in ABI 0.14 but the error
482 	 *	 codes were initially not populated correctly.
483 	 */
484 	if (nfp_nsp_get_abi_ver_minor(nsp) < 17) {
485 		nfp_err(nfp_nsp_cpp(nsp),
486 			"set operations not supported, please update flash\n");
487 		return -EOPNOTSUPP;
488 	}
489 
490 	/* Check if we are already in requested state */
491 	reg = le64_to_cpu(entries[idx].raw[raw_idx]);
492 	if (val == FIELD_GET(mask, reg))
493 		return 0;
494 
495 	reg &= ~mask;
496 	reg |= FIELD_PREP(mask, val);
497 	entries[idx].raw[raw_idx] = cpu_to_le64(reg);
498 
499 	entries[idx].control |= cpu_to_le64(ctrl_bit);
500 
501 	nfp_nsp_config_set_modified(nsp, true);
502 
503 	return 0;
504 }
505 
506 /**
507  * __nfp_eth_set_aneg() - set PHY autonegotiation control bit
508  * @nsp:	NFP NSP handle returned from nfp_eth_config_start()
509  * @mode:	Desired autonegotiation mode
510  *
511  * Allow/disallow PHY module to advertise/perform autonegotiation.
512  * Will write to hwinfo overrides in the flash (persistent config).
513  *
514  * Return: 0 or -ERRNO.
515  */
__nfp_eth_set_aneg(struct nfp_nsp * nsp,enum nfp_eth_aneg mode)516 int __nfp_eth_set_aneg(struct nfp_nsp *nsp, enum nfp_eth_aneg mode)
517 {
518 	return nfp_eth_set_bit_config(nsp, NSP_ETH_RAW_STATE,
519 				      NSP_ETH_STATE_ANEG, mode,
520 				      NSP_ETH_CTRL_SET_ANEG);
521 }
522 
523 /**
524  * __nfp_eth_set_speed() - set interface speed/rate
525  * @nsp:	NFP NSP handle returned from nfp_eth_config_start()
526  * @speed:	Desired speed (per lane)
527  *
528  * Set lane speed.  Provided @speed value should be subport speed divided
529  * by number of lanes this subport is spanning (i.e. 10000 for 40G, 25000 for
530  * 50G, etc.)
531  * Will write to hwinfo overrides in the flash (persistent config).
532  *
533  * Return: 0 or -ERRNO.
534  */
__nfp_eth_set_speed(struct nfp_nsp * nsp,unsigned int speed)535 int __nfp_eth_set_speed(struct nfp_nsp *nsp, unsigned int speed)
536 {
537 	enum nfp_eth_rate rate;
538 
539 	rate = nfp_eth_speed2rate(speed);
540 	if (rate == RATE_INVALID) {
541 		nfp_warn(nfp_nsp_cpp(nsp),
542 			 "could not find matching lane rate for speed %u\n",
543 			 speed);
544 		return -EINVAL;
545 	}
546 
547 	return nfp_eth_set_bit_config(nsp, NSP_ETH_RAW_STATE,
548 				      NSP_ETH_STATE_RATE, rate,
549 				      NSP_ETH_CTRL_SET_RATE);
550 }
551 
552 /**
553  * __nfp_eth_set_split() - set interface lane split
554  * @nsp:	NFP NSP handle returned from nfp_eth_config_start()
555  * @lanes:	Desired lanes per port
556  *
557  * Set number of lanes in the port.
558  * Will write to hwinfo overrides in the flash (persistent config).
559  *
560  * Return: 0 or -ERRNO.
561  */
__nfp_eth_set_split(struct nfp_nsp * nsp,unsigned int lanes)562 int __nfp_eth_set_split(struct nfp_nsp *nsp, unsigned int lanes)
563 {
564 	return nfp_eth_set_bit_config(nsp, NSP_ETH_RAW_PORT, NSP_ETH_PORT_LANES,
565 				      lanes, NSP_ETH_CTRL_SET_LANES);
566 }
567