1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33#include <linux/netdevice.h>
34#include <linux/mlx5/driver.h>
35#include <linux/mlx5/vport.h>
36#include "mlx5_core.h"
37#include "eswitch.h"
38#include "lag.h"
39#include "lag_mp.h"
40
41
42
43
44
45static DEFINE_MUTEX(lag_mutex);
46
47static int mlx5_cmd_create_lag(struct mlx5_core_dev *dev, u8 remap_port1,
48 u8 remap_port2)
49{
50 u32 in[MLX5_ST_SZ_DW(create_lag_in)] = {0};
51 u32 out[MLX5_ST_SZ_DW(create_lag_out)] = {0};
52 void *lag_ctx = MLX5_ADDR_OF(create_lag_in, in, ctx);
53
54 MLX5_SET(create_lag_in, in, opcode, MLX5_CMD_OP_CREATE_LAG);
55
56 MLX5_SET(lagc, lag_ctx, tx_remap_affinity_1, remap_port1);
57 MLX5_SET(lagc, lag_ctx, tx_remap_affinity_2, remap_port2);
58
59 return mlx5_cmd_exec(dev, in, sizeof(in), out, sizeof(out));
60}
61
62static int mlx5_cmd_modify_lag(struct mlx5_core_dev *dev, u8 remap_port1,
63 u8 remap_port2)
64{
65 u32 in[MLX5_ST_SZ_DW(modify_lag_in)] = {0};
66 u32 out[MLX5_ST_SZ_DW(modify_lag_out)] = {0};
67 void *lag_ctx = MLX5_ADDR_OF(modify_lag_in, in, ctx);
68
69 MLX5_SET(modify_lag_in, in, opcode, MLX5_CMD_OP_MODIFY_LAG);
70 MLX5_SET(modify_lag_in, in, field_select, 0x1);
71
72 MLX5_SET(lagc, lag_ctx, tx_remap_affinity_1, remap_port1);
73 MLX5_SET(lagc, lag_ctx, tx_remap_affinity_2, remap_port2);
74
75 return mlx5_cmd_exec(dev, in, sizeof(in), out, sizeof(out));
76}
77
78static int mlx5_cmd_destroy_lag(struct mlx5_core_dev *dev)
79{
80 u32 in[MLX5_ST_SZ_DW(destroy_lag_in)] = {0};
81 u32 out[MLX5_ST_SZ_DW(destroy_lag_out)] = {0};
82
83 MLX5_SET(destroy_lag_in, in, opcode, MLX5_CMD_OP_DESTROY_LAG);
84
85 return mlx5_cmd_exec(dev, in, sizeof(in), out, sizeof(out));
86}
87
88int mlx5_cmd_create_vport_lag(struct mlx5_core_dev *dev)
89{
90 u32 in[MLX5_ST_SZ_DW(create_vport_lag_in)] = {0};
91 u32 out[MLX5_ST_SZ_DW(create_vport_lag_out)] = {0};
92
93 MLX5_SET(create_vport_lag_in, in, opcode, MLX5_CMD_OP_CREATE_VPORT_LAG);
94
95 return mlx5_cmd_exec(dev, in, sizeof(in), out, sizeof(out));
96}
97EXPORT_SYMBOL(mlx5_cmd_create_vport_lag);
98
99int mlx5_cmd_destroy_vport_lag(struct mlx5_core_dev *dev)
100{
101 u32 in[MLX5_ST_SZ_DW(destroy_vport_lag_in)] = {0};
102 u32 out[MLX5_ST_SZ_DW(destroy_vport_lag_out)] = {0};
103
104 MLX5_SET(destroy_vport_lag_in, in, opcode, MLX5_CMD_OP_DESTROY_VPORT_LAG);
105
106 return mlx5_cmd_exec(dev, in, sizeof(in), out, sizeof(out));
107}
108EXPORT_SYMBOL(mlx5_cmd_destroy_vport_lag);
109
110static int mlx5_cmd_query_cong_counter(struct mlx5_core_dev *dev,
111 bool reset, void *out, int out_size)
112{
113 u32 in[MLX5_ST_SZ_DW(query_cong_statistics_in)] = { };
114
115 MLX5_SET(query_cong_statistics_in, in, opcode,
116 MLX5_CMD_OP_QUERY_CONG_STATISTICS);
117 MLX5_SET(query_cong_statistics_in, in, clear, reset);
118 return mlx5_cmd_exec(dev, in, sizeof(in), out, out_size);
119}
120
121int mlx5_lag_dev_get_netdev_idx(struct mlx5_lag *ldev,
122 struct net_device *ndev)
123{
124 int i;
125
126 for (i = 0; i < MLX5_MAX_PORTS; i++)
127 if (ldev->pf[i].netdev == ndev)
128 return i;
129
130 return -1;
131}
132
133static bool __mlx5_lag_is_roce(struct mlx5_lag *ldev)
134{
135 return !!(ldev->flags & MLX5_LAG_FLAG_ROCE);
136}
137
138static bool __mlx5_lag_is_sriov(struct mlx5_lag *ldev)
139{
140 return !!(ldev->flags & MLX5_LAG_FLAG_SRIOV);
141}
142
143static void mlx5_infer_tx_affinity_mapping(struct lag_tracker *tracker,
144 u8 *port1, u8 *port2)
145{
146 *port1 = 1;
147 *port2 = 2;
148 if (!tracker->netdev_state[0].tx_enabled ||
149 !tracker->netdev_state[0].link_up) {
150 *port1 = 2;
151 return;
152 }
153
154 if (!tracker->netdev_state[1].tx_enabled ||
155 !tracker->netdev_state[1].link_up)
156 *port2 = 1;
157}
158
159void mlx5_modify_lag(struct mlx5_lag *ldev,
160 struct lag_tracker *tracker)
161{
162 struct mlx5_core_dev *dev0 = ldev->pf[0].dev;
163 u8 v2p_port1, v2p_port2;
164 int err;
165
166 mlx5_infer_tx_affinity_mapping(tracker, &v2p_port1,
167 &v2p_port2);
168
169 if (v2p_port1 != ldev->v2p_map[0] ||
170 v2p_port2 != ldev->v2p_map[1]) {
171 ldev->v2p_map[0] = v2p_port1;
172 ldev->v2p_map[1] = v2p_port2;
173
174 mlx5_core_info(dev0, "modify lag map port 1:%d port 2:%d",
175 ldev->v2p_map[0], ldev->v2p_map[1]);
176
177 err = mlx5_cmd_modify_lag(dev0, v2p_port1, v2p_port2);
178 if (err)
179 mlx5_core_err(dev0,
180 "Failed to modify LAG (%d)\n",
181 err);
182 }
183}
184
185static int mlx5_create_lag(struct mlx5_lag *ldev,
186 struct lag_tracker *tracker)
187{
188 struct mlx5_core_dev *dev0 = ldev->pf[0].dev;
189 int err;
190
191 mlx5_infer_tx_affinity_mapping(tracker, &ldev->v2p_map[0],
192 &ldev->v2p_map[1]);
193
194 mlx5_core_info(dev0, "lag map port 1:%d port 2:%d",
195 ldev->v2p_map[0], ldev->v2p_map[1]);
196
197 err = mlx5_cmd_create_lag(dev0, ldev->v2p_map[0], ldev->v2p_map[1]);
198 if (err)
199 mlx5_core_err(dev0,
200 "Failed to create LAG (%d)\n",
201 err);
202 return err;
203}
204
205int mlx5_activate_lag(struct mlx5_lag *ldev,
206 struct lag_tracker *tracker,
207 u8 flags)
208{
209 bool roce_lag = !!(flags & MLX5_LAG_FLAG_ROCE);
210 struct mlx5_core_dev *dev0 = ldev->pf[0].dev;
211 int err;
212
213 err = mlx5_create_lag(ldev, tracker);
214 if (err) {
215 if (roce_lag) {
216 mlx5_core_err(dev0,
217 "Failed to activate RoCE LAG\n");
218 } else {
219 mlx5_core_err(dev0,
220 "Failed to activate VF LAG\n"
221 "Make sure all VFs are unbound prior to VF LAG activation or deactivation\n");
222 }
223 return err;
224 }
225
226 ldev->flags |= flags;
227 return 0;
228}
229
230static int mlx5_deactivate_lag(struct mlx5_lag *ldev)
231{
232 struct mlx5_core_dev *dev0 = ldev->pf[0].dev;
233 bool roce_lag = __mlx5_lag_is_roce(ldev);
234 int err;
235
236 ldev->flags &= ~MLX5_LAG_MODE_FLAGS;
237
238 err = mlx5_cmd_destroy_lag(dev0);
239 if (err) {
240 if (roce_lag) {
241 mlx5_core_err(dev0,
242 "Failed to deactivate RoCE LAG; driver restart required\n");
243 } else {
244 mlx5_core_err(dev0,
245 "Failed to deactivate VF LAG; driver restart required\n"
246 "Make sure all VFs are unbound prior to VF LAG activation or deactivation\n");
247 }
248 }
249
250 return err;
251}
252
253static bool mlx5_lag_check_prereq(struct mlx5_lag *ldev)
254{
255 if (!ldev->pf[0].dev || !ldev->pf[1].dev)
256 return false;
257
258#ifdef CONFIG_MLX5_ESWITCH
259 return mlx5_esw_lag_prereq(ldev->pf[0].dev, ldev->pf[1].dev);
260#else
261 return (!mlx5_sriov_is_enabled(ldev->pf[0].dev) &&
262 !mlx5_sriov_is_enabled(ldev->pf[1].dev));
263#endif
264}
265
266static void mlx5_lag_add_ib_devices(struct mlx5_lag *ldev)
267{
268 int i;
269
270 for (i = 0; i < MLX5_MAX_PORTS; i++)
271 if (ldev->pf[i].dev)
272 mlx5_add_dev_by_protocol(ldev->pf[i].dev,
273 MLX5_INTERFACE_PROTOCOL_IB);
274}
275
276static void mlx5_lag_remove_ib_devices(struct mlx5_lag *ldev)
277{
278 int i;
279
280 for (i = 0; i < MLX5_MAX_PORTS; i++)
281 if (ldev->pf[i].dev)
282 mlx5_remove_dev_by_protocol(ldev->pf[i].dev,
283 MLX5_INTERFACE_PROTOCOL_IB);
284}
285
286static void mlx5_do_bond(struct mlx5_lag *ldev)
287{
288 struct mlx5_core_dev *dev0 = ldev->pf[0].dev;
289 struct mlx5_core_dev *dev1 = ldev->pf[1].dev;
290 struct lag_tracker tracker;
291 bool do_bond, roce_lag;
292 int err;
293
294 if (!dev0 || !dev1)
295 return;
296
297 mutex_lock(&lag_mutex);
298 tracker = ldev->tracker;
299 mutex_unlock(&lag_mutex);
300
301 do_bond = tracker.is_bonded && mlx5_lag_check_prereq(ldev);
302
303 if (do_bond && !__mlx5_lag_is_active(ldev)) {
304 roce_lag = !mlx5_sriov_is_enabled(dev0) &&
305 !mlx5_sriov_is_enabled(dev1);
306
307#ifdef CONFIG_MLX5_ESWITCH
308 roce_lag &= dev0->priv.eswitch->mode == SRIOV_NONE &&
309 dev1->priv.eswitch->mode == SRIOV_NONE;
310#endif
311
312 if (roce_lag)
313 mlx5_lag_remove_ib_devices(ldev);
314
315 err = mlx5_activate_lag(ldev, &tracker,
316 roce_lag ? MLX5_LAG_FLAG_ROCE :
317 MLX5_LAG_FLAG_SRIOV);
318 if (err) {
319 if (roce_lag)
320 mlx5_lag_add_ib_devices(ldev);
321
322 return;
323 }
324
325 if (roce_lag) {
326 mlx5_add_dev_by_protocol(dev0, MLX5_INTERFACE_PROTOCOL_IB);
327 mlx5_nic_vport_enable_roce(dev1);
328 }
329 } else if (do_bond && __mlx5_lag_is_active(ldev)) {
330 mlx5_modify_lag(ldev, &tracker);
331 } else if (!do_bond && __mlx5_lag_is_active(ldev)) {
332 roce_lag = __mlx5_lag_is_roce(ldev);
333
334 if (roce_lag) {
335 mlx5_remove_dev_by_protocol(dev0, MLX5_INTERFACE_PROTOCOL_IB);
336 mlx5_nic_vport_disable_roce(dev1);
337 }
338
339 err = mlx5_deactivate_lag(ldev);
340 if (err)
341 return;
342
343 if (roce_lag)
344 mlx5_lag_add_ib_devices(ldev);
345 }
346}
347
348static void mlx5_queue_bond_work(struct mlx5_lag *ldev, unsigned long delay)
349{
350 queue_delayed_work(ldev->wq, &ldev->bond_work, delay);
351}
352
353static void mlx5_do_bond_work(struct work_struct *work)
354{
355 struct delayed_work *delayed_work = to_delayed_work(work);
356 struct mlx5_lag *ldev = container_of(delayed_work, struct mlx5_lag,
357 bond_work);
358 int status;
359
360 status = mlx5_dev_list_trylock();
361 if (!status) {
362
363 mlx5_queue_bond_work(ldev, HZ);
364 return;
365 }
366
367 mlx5_do_bond(ldev);
368 mlx5_dev_list_unlock();
369}
370
371static int mlx5_handle_changeupper_event(struct mlx5_lag *ldev,
372 struct lag_tracker *tracker,
373 struct net_device *ndev,
374 struct netdev_notifier_changeupper_info *info)
375{
376 struct net_device *upper = info->upper_dev, *ndev_tmp;
377 struct netdev_lag_upper_info *lag_upper_info = NULL;
378 bool is_bonded;
379 int bond_status = 0;
380 int num_slaves = 0;
381 int idx;
382
383 if (!netif_is_lag_master(upper))
384 return 0;
385
386 if (info->linking)
387 lag_upper_info = info->upper_info;
388
389
390
391
392
393
394 rcu_read_lock();
395 for_each_netdev_in_bond_rcu(upper, ndev_tmp) {
396 idx = mlx5_lag_dev_get_netdev_idx(ldev, ndev_tmp);
397 if (idx > -1)
398 bond_status |= (1 << idx);
399
400 num_slaves++;
401 }
402 rcu_read_unlock();
403
404
405 if (!(bond_status & 0x3))
406 return 0;
407
408 if (lag_upper_info)
409 tracker->tx_type = lag_upper_info->tx_type;
410
411
412
413
414
415
416 is_bonded = (num_slaves == MLX5_MAX_PORTS) &&
417 (bond_status == 0x3) &&
418 ((tracker->tx_type == NETDEV_LAG_TX_TYPE_ACTIVEBACKUP) ||
419 (tracker->tx_type == NETDEV_LAG_TX_TYPE_HASH));
420
421 if (tracker->is_bonded != is_bonded) {
422 tracker->is_bonded = is_bonded;
423 return 1;
424 }
425
426 return 0;
427}
428
429static int mlx5_handle_changelowerstate_event(struct mlx5_lag *ldev,
430 struct lag_tracker *tracker,
431 struct net_device *ndev,
432 struct netdev_notifier_changelowerstate_info *info)
433{
434 struct netdev_lag_lower_state_info *lag_lower_info;
435 int idx;
436
437 if (!netif_is_lag_port(ndev))
438 return 0;
439
440 idx = mlx5_lag_dev_get_netdev_idx(ldev, ndev);
441 if (idx == -1)
442 return 0;
443
444
445
446
447 lag_lower_info = info->lower_state_info;
448 if (!lag_lower_info)
449 return 0;
450
451 tracker->netdev_state[idx] = *lag_lower_info;
452
453 return 1;
454}
455
456static int mlx5_lag_netdev_event(struct notifier_block *this,
457 unsigned long event, void *ptr)
458{
459 struct net_device *ndev = netdev_notifier_info_to_dev(ptr);
460 struct lag_tracker tracker;
461 struct mlx5_lag *ldev;
462 int changed = 0;
463
464 if (!net_eq(dev_net(ndev), &init_net))
465 return NOTIFY_DONE;
466
467 if ((event != NETDEV_CHANGEUPPER) && (event != NETDEV_CHANGELOWERSTATE))
468 return NOTIFY_DONE;
469
470 ldev = container_of(this, struct mlx5_lag, nb);
471 tracker = ldev->tracker;
472
473 switch (event) {
474 case NETDEV_CHANGEUPPER:
475 changed = mlx5_handle_changeupper_event(ldev, &tracker, ndev,
476 ptr);
477 break;
478 case NETDEV_CHANGELOWERSTATE:
479 changed = mlx5_handle_changelowerstate_event(ldev, &tracker,
480 ndev, ptr);
481 break;
482 }
483
484 mutex_lock(&lag_mutex);
485 ldev->tracker = tracker;
486 mutex_unlock(&lag_mutex);
487
488 if (changed)
489 mlx5_queue_bond_work(ldev, 0);
490
491 return NOTIFY_DONE;
492}
493
494static struct mlx5_lag *mlx5_lag_dev_alloc(void)
495{
496 struct mlx5_lag *ldev;
497
498 ldev = kzalloc(sizeof(*ldev), GFP_KERNEL);
499 if (!ldev)
500 return NULL;
501
502 ldev->wq = create_singlethread_workqueue("mlx5_lag");
503 if (!ldev->wq) {
504 kfree(ldev);
505 return NULL;
506 }
507
508 INIT_DELAYED_WORK(&ldev->bond_work, mlx5_do_bond_work);
509
510 return ldev;
511}
512
513static void mlx5_lag_dev_free(struct mlx5_lag *ldev)
514{
515 destroy_workqueue(ldev->wq);
516 kfree(ldev);
517}
518
519static void mlx5_lag_dev_add_pf(struct mlx5_lag *ldev,
520 struct mlx5_core_dev *dev,
521 struct net_device *netdev)
522{
523 unsigned int fn = PCI_FUNC(dev->pdev->devfn);
524
525 if (fn >= MLX5_MAX_PORTS)
526 return;
527
528 mutex_lock(&lag_mutex);
529 ldev->pf[fn].dev = dev;
530 ldev->pf[fn].netdev = netdev;
531 ldev->tracker.netdev_state[fn].link_up = 0;
532 ldev->tracker.netdev_state[fn].tx_enabled = 0;
533
534 dev->priv.lag = ldev;
535
536 mutex_unlock(&lag_mutex);
537}
538
539static void mlx5_lag_dev_remove_pf(struct mlx5_lag *ldev,
540 struct mlx5_core_dev *dev)
541{
542 int i;
543
544 for (i = 0; i < MLX5_MAX_PORTS; i++)
545 if (ldev->pf[i].dev == dev)
546 break;
547
548 if (i == MLX5_MAX_PORTS)
549 return;
550
551 mutex_lock(&lag_mutex);
552 memset(&ldev->pf[i], 0, sizeof(*ldev->pf));
553
554 dev->priv.lag = NULL;
555 mutex_unlock(&lag_mutex);
556}
557
558
559void mlx5_lag_add(struct mlx5_core_dev *dev, struct net_device *netdev)
560{
561 struct mlx5_lag *ldev = NULL;
562 struct mlx5_core_dev *tmp_dev;
563 int err;
564
565 if (!MLX5_CAP_GEN(dev, vport_group_manager) ||
566 !MLX5_CAP_GEN(dev, lag_master) ||
567 (MLX5_CAP_GEN(dev, num_lag_ports) != MLX5_MAX_PORTS))
568 return;
569
570 tmp_dev = mlx5_get_next_phys_dev(dev);
571 if (tmp_dev)
572 ldev = tmp_dev->priv.lag;
573
574 if (!ldev) {
575 ldev = mlx5_lag_dev_alloc();
576 if (!ldev) {
577 mlx5_core_err(dev, "Failed to alloc lag dev\n");
578 return;
579 }
580 }
581
582 mlx5_lag_dev_add_pf(ldev, dev, netdev);
583
584 if (!ldev->nb.notifier_call) {
585 ldev->nb.notifier_call = mlx5_lag_netdev_event;
586 if (register_netdevice_notifier(&ldev->nb)) {
587 ldev->nb.notifier_call = NULL;
588 mlx5_core_err(dev, "Failed to register LAG netdev notifier\n");
589 }
590 }
591
592 err = mlx5_lag_mp_init(ldev);
593 if (err)
594 mlx5_core_err(dev, "Failed to init multipath lag err=%d\n",
595 err);
596}
597
598
599void mlx5_lag_remove(struct mlx5_core_dev *dev)
600{
601 struct mlx5_lag *ldev;
602 int i;
603
604 ldev = mlx5_lag_dev_get(dev);
605 if (!ldev)
606 return;
607
608 if (__mlx5_lag_is_active(ldev))
609 mlx5_deactivate_lag(ldev);
610
611 mlx5_lag_dev_remove_pf(ldev, dev);
612
613 for (i = 0; i < MLX5_MAX_PORTS; i++)
614 if (ldev->pf[i].dev)
615 break;
616
617 if (i == MLX5_MAX_PORTS) {
618 if (ldev->nb.notifier_call)
619 unregister_netdevice_notifier(&ldev->nb);
620 mlx5_lag_mp_cleanup(ldev);
621 cancel_delayed_work_sync(&ldev->bond_work);
622 mlx5_lag_dev_free(ldev);
623 }
624}
625
626bool mlx5_lag_is_roce(struct mlx5_core_dev *dev)
627{
628 struct mlx5_lag *ldev;
629 bool res;
630
631 mutex_lock(&lag_mutex);
632 ldev = mlx5_lag_dev_get(dev);
633 res = ldev && __mlx5_lag_is_roce(ldev);
634 mutex_unlock(&lag_mutex);
635
636 return res;
637}
638EXPORT_SYMBOL(mlx5_lag_is_roce);
639
640bool mlx5_lag_is_active(struct mlx5_core_dev *dev)
641{
642 struct mlx5_lag *ldev;
643 bool res;
644
645 mutex_lock(&lag_mutex);
646 ldev = mlx5_lag_dev_get(dev);
647 res = ldev && __mlx5_lag_is_active(ldev);
648 mutex_unlock(&lag_mutex);
649
650 return res;
651}
652EXPORT_SYMBOL(mlx5_lag_is_active);
653
654bool mlx5_lag_is_sriov(struct mlx5_core_dev *dev)
655{
656 struct mlx5_lag *ldev;
657 bool res;
658
659 mutex_lock(&lag_mutex);
660 ldev = mlx5_lag_dev_get(dev);
661 res = ldev && __mlx5_lag_is_sriov(ldev);
662 mutex_unlock(&lag_mutex);
663
664 return res;
665}
666EXPORT_SYMBOL(mlx5_lag_is_sriov);
667
668void mlx5_lag_update(struct mlx5_core_dev *dev)
669{
670 struct mlx5_lag *ldev;
671
672 mlx5_dev_list_lock();
673 ldev = mlx5_lag_dev_get(dev);
674 if (!ldev)
675 goto unlock;
676
677 mlx5_do_bond(ldev);
678
679unlock:
680 mlx5_dev_list_unlock();
681}
682
683struct net_device *mlx5_lag_get_roce_netdev(struct mlx5_core_dev *dev)
684{
685 struct net_device *ndev = NULL;
686 struct mlx5_lag *ldev;
687
688 mutex_lock(&lag_mutex);
689 ldev = mlx5_lag_dev_get(dev);
690
691 if (!(ldev && __mlx5_lag_is_roce(ldev)))
692 goto unlock;
693
694 if (ldev->tracker.tx_type == NETDEV_LAG_TX_TYPE_ACTIVEBACKUP) {
695 ndev = ldev->tracker.netdev_state[0].tx_enabled ?
696 ldev->pf[0].netdev : ldev->pf[1].netdev;
697 } else {
698 ndev = ldev->pf[0].netdev;
699 }
700 if (ndev)
701 dev_hold(ndev);
702
703unlock:
704 mutex_unlock(&lag_mutex);
705
706 return ndev;
707}
708EXPORT_SYMBOL(mlx5_lag_get_roce_netdev);
709
710bool mlx5_lag_intf_add(struct mlx5_interface *intf, struct mlx5_priv *priv)
711{
712 struct mlx5_core_dev *dev = container_of(priv, struct mlx5_core_dev,
713 priv);
714 struct mlx5_lag *ldev;
715
716 if (intf->protocol != MLX5_INTERFACE_PROTOCOL_IB)
717 return true;
718
719 ldev = mlx5_lag_dev_get(dev);
720 if (!ldev || !__mlx5_lag_is_roce(ldev) || ldev->pf[0].dev == dev)
721 return true;
722
723
724 return false;
725}
726
727int mlx5_lag_query_cong_counters(struct mlx5_core_dev *dev,
728 u64 *values,
729 int num_counters,
730 size_t *offsets)
731{
732 int outlen = MLX5_ST_SZ_BYTES(query_cong_statistics_out);
733 struct mlx5_core_dev *mdev[MLX5_MAX_PORTS];
734 struct mlx5_lag *ldev;
735 int num_ports;
736 int ret, i, j;
737 void *out;
738
739 out = kvzalloc(outlen, GFP_KERNEL);
740 if (!out)
741 return -ENOMEM;
742
743 memset(values, 0, sizeof(*values) * num_counters);
744
745 mutex_lock(&lag_mutex);
746 ldev = mlx5_lag_dev_get(dev);
747 if (ldev && __mlx5_lag_is_roce(ldev)) {
748 num_ports = MLX5_MAX_PORTS;
749 mdev[0] = ldev->pf[0].dev;
750 mdev[1] = ldev->pf[1].dev;
751 } else {
752 num_ports = 1;
753 mdev[0] = dev;
754 }
755
756 for (i = 0; i < num_ports; ++i) {
757 ret = mlx5_cmd_query_cong_counter(mdev[i], false, out, outlen);
758 if (ret)
759 goto unlock;
760
761 for (j = 0; j < num_counters; ++j)
762 values[j] += be64_to_cpup((__be64 *)(out + offsets[j]));
763 }
764
765unlock:
766 mutex_unlock(&lag_mutex);
767 kvfree(out);
768 return ret;
769}
770EXPORT_SYMBOL(mlx5_lag_query_cong_counters);
771