1 // SPDX-License-Identifier: GPL-2.0-or-later
3 * Copyright (c) 2016 Mellanox Technologies. All rights reserved.
4 * Copyright (c) 2016 Jiri Pirko <jiri@mellanox.com>
7 #include "devl_internal.h"
10 struct list_head list;
13 u16 ingress_pools_count;
14 u16 egress_pools_count;
19 static u16 devlink_sb_pool_count(struct devlink_sb *devlink_sb)
21 return devlink_sb->ingress_pools_count + devlink_sb->egress_pools_count;
24 static struct devlink_sb *devlink_sb_get_by_index(struct devlink *devlink,
25 unsigned int sb_index)
27 struct devlink_sb *devlink_sb;
29 list_for_each_entry(devlink_sb, &devlink->sb_list, list) {
30 if (devlink_sb->index == sb_index)
36 static bool devlink_sb_index_exists(struct devlink *devlink,
37 unsigned int sb_index)
39 return devlink_sb_get_by_index(devlink, sb_index);
42 static struct devlink_sb *devlink_sb_get_from_attrs(struct devlink *devlink,
43 struct nlattr **attrs)
45 if (attrs[DEVLINK_ATTR_SB_INDEX]) {
46 u32 sb_index = nla_get_u32(attrs[DEVLINK_ATTR_SB_INDEX]);
47 struct devlink_sb *devlink_sb;
49 devlink_sb = devlink_sb_get_by_index(devlink, sb_index);
51 return ERR_PTR(-ENODEV);
54 return ERR_PTR(-EINVAL);
57 static struct devlink_sb *devlink_sb_get_from_info(struct devlink *devlink,
58 struct genl_info *info)
60 return devlink_sb_get_from_attrs(devlink, info->attrs);
63 static int devlink_sb_pool_index_get_from_attrs(struct devlink_sb *devlink_sb,
64 struct nlattr **attrs,
69 if (!attrs[DEVLINK_ATTR_SB_POOL_INDEX])
72 val = nla_get_u16(attrs[DEVLINK_ATTR_SB_POOL_INDEX]);
73 if (val >= devlink_sb_pool_count(devlink_sb))
79 static int devlink_sb_pool_index_get_from_info(struct devlink_sb *devlink_sb,
80 struct genl_info *info,
83 return devlink_sb_pool_index_get_from_attrs(devlink_sb, info->attrs,
88 devlink_sb_pool_type_get_from_attrs(struct nlattr **attrs,
89 enum devlink_sb_pool_type *p_pool_type)
93 if (!attrs[DEVLINK_ATTR_SB_POOL_TYPE])
96 val = nla_get_u8(attrs[DEVLINK_ATTR_SB_POOL_TYPE]);
97 if (val != DEVLINK_SB_POOL_TYPE_INGRESS &&
98 val != DEVLINK_SB_POOL_TYPE_EGRESS)
105 devlink_sb_pool_type_get_from_info(struct genl_info *info,
106 enum devlink_sb_pool_type *p_pool_type)
108 return devlink_sb_pool_type_get_from_attrs(info->attrs, p_pool_type);
112 devlink_sb_th_type_get_from_attrs(struct nlattr **attrs,
113 enum devlink_sb_threshold_type *p_th_type)
117 if (!attrs[DEVLINK_ATTR_SB_POOL_THRESHOLD_TYPE])
120 val = nla_get_u8(attrs[DEVLINK_ATTR_SB_POOL_THRESHOLD_TYPE]);
121 if (val != DEVLINK_SB_THRESHOLD_TYPE_STATIC &&
122 val != DEVLINK_SB_THRESHOLD_TYPE_DYNAMIC)
129 devlink_sb_th_type_get_from_info(struct genl_info *info,
130 enum devlink_sb_threshold_type *p_th_type)
132 return devlink_sb_th_type_get_from_attrs(info->attrs, p_th_type);
136 devlink_sb_tc_index_get_from_attrs(struct devlink_sb *devlink_sb,
137 struct nlattr **attrs,
138 enum devlink_sb_pool_type pool_type,
143 if (!attrs[DEVLINK_ATTR_SB_TC_INDEX])
146 val = nla_get_u16(attrs[DEVLINK_ATTR_SB_TC_INDEX]);
147 if (pool_type == DEVLINK_SB_POOL_TYPE_INGRESS &&
148 val >= devlink_sb->ingress_tc_count)
150 if (pool_type == DEVLINK_SB_POOL_TYPE_EGRESS &&
151 val >= devlink_sb->egress_tc_count)
158 devlink_sb_tc_index_get_from_info(struct devlink_sb *devlink_sb,
159 struct genl_info *info,
160 enum devlink_sb_pool_type pool_type,
163 return devlink_sb_tc_index_get_from_attrs(devlink_sb, info->attrs,
164 pool_type, p_tc_index);
167 static int devlink_nl_sb_fill(struct sk_buff *msg, struct devlink *devlink,
168 struct devlink_sb *devlink_sb,
169 enum devlink_command cmd, u32 portid,
174 hdr = genlmsg_put(msg, portid, seq, &devlink_nl_family, flags, cmd);
178 if (devlink_nl_put_handle(msg, devlink))
179 goto nla_put_failure;
180 if (nla_put_u32(msg, DEVLINK_ATTR_SB_INDEX, devlink_sb->index))
181 goto nla_put_failure;
182 if (nla_put_u32(msg, DEVLINK_ATTR_SB_SIZE, devlink_sb->size))
183 goto nla_put_failure;
184 if (nla_put_u16(msg, DEVLINK_ATTR_SB_INGRESS_POOL_COUNT,
185 devlink_sb->ingress_pools_count))
186 goto nla_put_failure;
187 if (nla_put_u16(msg, DEVLINK_ATTR_SB_EGRESS_POOL_COUNT,
188 devlink_sb->egress_pools_count))
189 goto nla_put_failure;
190 if (nla_put_u16(msg, DEVLINK_ATTR_SB_INGRESS_TC_COUNT,
191 devlink_sb->ingress_tc_count))
192 goto nla_put_failure;
193 if (nla_put_u16(msg, DEVLINK_ATTR_SB_EGRESS_TC_COUNT,
194 devlink_sb->egress_tc_count))
195 goto nla_put_failure;
197 genlmsg_end(msg, hdr);
201 genlmsg_cancel(msg, hdr);
205 int devlink_nl_sb_get_doit(struct sk_buff *skb, struct genl_info *info)
207 struct devlink *devlink = info->user_ptr[0];
208 struct devlink_sb *devlink_sb;
212 devlink_sb = devlink_sb_get_from_info(devlink, info);
213 if (IS_ERR(devlink_sb))
214 return PTR_ERR(devlink_sb);
216 msg = nlmsg_new(NLMSG_DEFAULT_SIZE, GFP_KERNEL);
220 err = devlink_nl_sb_fill(msg, devlink, devlink_sb,
222 info->snd_portid, info->snd_seq, 0);
228 return genlmsg_reply(msg, info);
232 devlink_nl_sb_get_dump_one(struct sk_buff *msg, struct devlink *devlink,
233 struct netlink_callback *cb, int flags)
235 struct devlink_nl_dump_state *state = devlink_dump_state(cb);
236 struct devlink_sb *devlink_sb;
240 list_for_each_entry(devlink_sb, &devlink->sb_list, list) {
241 if (idx < state->idx) {
245 err = devlink_nl_sb_fill(msg, devlink, devlink_sb,
247 NETLINK_CB(cb->skb).portid,
248 cb->nlh->nlmsg_seq, flags);
259 int devlink_nl_sb_get_dumpit(struct sk_buff *skb, struct netlink_callback *cb)
261 return devlink_nl_dumpit(skb, cb, devlink_nl_sb_get_dump_one);
264 static int devlink_nl_sb_pool_fill(struct sk_buff *msg, struct devlink *devlink,
265 struct devlink_sb *devlink_sb,
266 u16 pool_index, enum devlink_command cmd,
267 u32 portid, u32 seq, int flags)
269 struct devlink_sb_pool_info pool_info;
273 err = devlink->ops->sb_pool_get(devlink, devlink_sb->index,
274 pool_index, &pool_info);
278 hdr = genlmsg_put(msg, portid, seq, &devlink_nl_family, flags, cmd);
282 if (devlink_nl_put_handle(msg, devlink))
283 goto nla_put_failure;
284 if (nla_put_u32(msg, DEVLINK_ATTR_SB_INDEX, devlink_sb->index))
285 goto nla_put_failure;
286 if (nla_put_u16(msg, DEVLINK_ATTR_SB_POOL_INDEX, pool_index))
287 goto nla_put_failure;
288 if (nla_put_u8(msg, DEVLINK_ATTR_SB_POOL_TYPE, pool_info.pool_type))
289 goto nla_put_failure;
290 if (nla_put_u32(msg, DEVLINK_ATTR_SB_POOL_SIZE, pool_info.size))
291 goto nla_put_failure;
292 if (nla_put_u8(msg, DEVLINK_ATTR_SB_POOL_THRESHOLD_TYPE,
293 pool_info.threshold_type))
294 goto nla_put_failure;
295 if (nla_put_u32(msg, DEVLINK_ATTR_SB_POOL_CELL_SIZE,
296 pool_info.cell_size))
297 goto nla_put_failure;
299 genlmsg_end(msg, hdr);
303 genlmsg_cancel(msg, hdr);
307 int devlink_nl_sb_pool_get_doit(struct sk_buff *skb, struct genl_info *info)
309 struct devlink *devlink = info->user_ptr[0];
310 struct devlink_sb *devlink_sb;
315 devlink_sb = devlink_sb_get_from_info(devlink, info);
316 if (IS_ERR(devlink_sb))
317 return PTR_ERR(devlink_sb);
319 err = devlink_sb_pool_index_get_from_info(devlink_sb, info,
324 if (!devlink->ops->sb_pool_get)
327 msg = nlmsg_new(NLMSG_DEFAULT_SIZE, GFP_KERNEL);
331 err = devlink_nl_sb_pool_fill(msg, devlink, devlink_sb, pool_index,
332 DEVLINK_CMD_SB_POOL_NEW,
333 info->snd_portid, info->snd_seq, 0);
339 return genlmsg_reply(msg, info);
342 static int __sb_pool_get_dumpit(struct sk_buff *msg, int start, int *p_idx,
343 struct devlink *devlink,
344 struct devlink_sb *devlink_sb,
345 u32 portid, u32 seq, int flags)
347 u16 pool_count = devlink_sb_pool_count(devlink_sb);
351 for (pool_index = 0; pool_index < pool_count; pool_index++) {
352 if (*p_idx < start) {
356 err = devlink_nl_sb_pool_fill(msg, devlink,
359 DEVLINK_CMD_SB_POOL_NEW,
369 devlink_nl_sb_pool_get_dump_one(struct sk_buff *msg, struct devlink *devlink,
370 struct netlink_callback *cb, int flags)
372 struct devlink_nl_dump_state *state = devlink_dump_state(cb);
373 struct devlink_sb *devlink_sb;
377 if (!devlink->ops->sb_pool_get)
380 list_for_each_entry(devlink_sb, &devlink->sb_list, list) {
381 err = __sb_pool_get_dumpit(msg, state->idx, &idx,
383 NETLINK_CB(cb->skb).portid,
384 cb->nlh->nlmsg_seq, flags);
385 if (err == -EOPNOTSUPP) {
396 int devlink_nl_sb_pool_get_dumpit(struct sk_buff *skb,
397 struct netlink_callback *cb)
399 return devlink_nl_dumpit(skb, cb, devlink_nl_sb_pool_get_dump_one);
402 static int devlink_sb_pool_set(struct devlink *devlink, unsigned int sb_index,
403 u16 pool_index, u32 size,
404 enum devlink_sb_threshold_type threshold_type,
405 struct netlink_ext_ack *extack)
408 const struct devlink_ops *ops = devlink->ops;
410 if (ops->sb_pool_set)
411 return ops->sb_pool_set(devlink, sb_index, pool_index,
412 size, threshold_type, extack);
416 int devlink_nl_cmd_sb_pool_set_doit(struct sk_buff *skb, struct genl_info *info)
418 struct devlink *devlink = info->user_ptr[0];
419 enum devlink_sb_threshold_type threshold_type;
420 struct devlink_sb *devlink_sb;
425 devlink_sb = devlink_sb_get_from_info(devlink, info);
426 if (IS_ERR(devlink_sb))
427 return PTR_ERR(devlink_sb);
429 err = devlink_sb_pool_index_get_from_info(devlink_sb, info,
434 err = devlink_sb_th_type_get_from_info(info, &threshold_type);
438 if (GENL_REQ_ATTR_CHECK(info, DEVLINK_ATTR_SB_POOL_SIZE))
441 size = nla_get_u32(info->attrs[DEVLINK_ATTR_SB_POOL_SIZE]);
442 return devlink_sb_pool_set(devlink, devlink_sb->index,
443 pool_index, size, threshold_type,
447 static int devlink_nl_sb_port_pool_fill(struct sk_buff *msg,
448 struct devlink *devlink,
449 struct devlink_port *devlink_port,
450 struct devlink_sb *devlink_sb,
452 enum devlink_command cmd,
453 u32 portid, u32 seq, int flags)
455 const struct devlink_ops *ops = devlink->ops;
460 err = ops->sb_port_pool_get(devlink_port, devlink_sb->index,
461 pool_index, &threshold);
465 hdr = genlmsg_put(msg, portid, seq, &devlink_nl_family, flags, cmd);
469 if (devlink_nl_put_handle(msg, devlink))
470 goto nla_put_failure;
471 if (nla_put_u32(msg, DEVLINK_ATTR_PORT_INDEX, devlink_port->index))
472 goto nla_put_failure;
473 if (nla_put_u32(msg, DEVLINK_ATTR_SB_INDEX, devlink_sb->index))
474 goto nla_put_failure;
475 if (nla_put_u16(msg, DEVLINK_ATTR_SB_POOL_INDEX, pool_index))
476 goto nla_put_failure;
477 if (nla_put_u32(msg, DEVLINK_ATTR_SB_THRESHOLD, threshold))
478 goto nla_put_failure;
480 if (ops->sb_occ_port_pool_get) {
484 err = ops->sb_occ_port_pool_get(devlink_port, devlink_sb->index,
485 pool_index, &cur, &max);
486 if (err && err != -EOPNOTSUPP)
487 goto sb_occ_get_failure;
489 if (nla_put_u32(msg, DEVLINK_ATTR_SB_OCC_CUR, cur))
490 goto nla_put_failure;
491 if (nla_put_u32(msg, DEVLINK_ATTR_SB_OCC_MAX, max))
492 goto nla_put_failure;
496 genlmsg_end(msg, hdr);
502 genlmsg_cancel(msg, hdr);
506 int devlink_nl_sb_port_pool_get_doit(struct sk_buff *skb,
507 struct genl_info *info)
509 struct devlink_port *devlink_port = info->user_ptr[1];
510 struct devlink *devlink = devlink_port->devlink;
511 struct devlink_sb *devlink_sb;
516 devlink_sb = devlink_sb_get_from_info(devlink, info);
517 if (IS_ERR(devlink_sb))
518 return PTR_ERR(devlink_sb);
520 err = devlink_sb_pool_index_get_from_info(devlink_sb, info,
525 if (!devlink->ops->sb_port_pool_get)
528 msg = nlmsg_new(NLMSG_DEFAULT_SIZE, GFP_KERNEL);
532 err = devlink_nl_sb_port_pool_fill(msg, devlink, devlink_port,
533 devlink_sb, pool_index,
534 DEVLINK_CMD_SB_PORT_POOL_NEW,
535 info->snd_portid, info->snd_seq, 0);
541 return genlmsg_reply(msg, info);
544 static int __sb_port_pool_get_dumpit(struct sk_buff *msg, int start, int *p_idx,
545 struct devlink *devlink,
546 struct devlink_sb *devlink_sb,
547 u32 portid, u32 seq, int flags)
549 struct devlink_port *devlink_port;
550 u16 pool_count = devlink_sb_pool_count(devlink_sb);
551 unsigned long port_index;
555 xa_for_each(&devlink->ports, port_index, devlink_port) {
556 for (pool_index = 0; pool_index < pool_count; pool_index++) {
557 if (*p_idx < start) {
561 err = devlink_nl_sb_port_pool_fill(msg, devlink,
565 DEVLINK_CMD_SB_PORT_POOL_NEW,
576 devlink_nl_sb_port_pool_get_dump_one(struct sk_buff *msg,
577 struct devlink *devlink,
578 struct netlink_callback *cb, int flags)
580 struct devlink_nl_dump_state *state = devlink_dump_state(cb);
581 struct devlink_sb *devlink_sb;
585 if (!devlink->ops->sb_port_pool_get)
588 list_for_each_entry(devlink_sb, &devlink->sb_list, list) {
589 err = __sb_port_pool_get_dumpit(msg, state->idx, &idx,
591 NETLINK_CB(cb->skb).portid,
592 cb->nlh->nlmsg_seq, flags);
593 if (err == -EOPNOTSUPP) {
604 int devlink_nl_sb_port_pool_get_dumpit(struct sk_buff *skb,
605 struct netlink_callback *cb)
607 return devlink_nl_dumpit(skb, cb, devlink_nl_sb_port_pool_get_dump_one);
610 static int devlink_sb_port_pool_set(struct devlink_port *devlink_port,
611 unsigned int sb_index, u16 pool_index,
613 struct netlink_ext_ack *extack)
616 const struct devlink_ops *ops = devlink_port->devlink->ops;
618 if (ops->sb_port_pool_set)
619 return ops->sb_port_pool_set(devlink_port, sb_index,
620 pool_index, threshold, extack);
624 int devlink_nl_cmd_sb_port_pool_set_doit(struct sk_buff *skb,
625 struct genl_info *info)
627 struct devlink_port *devlink_port = info->user_ptr[1];
628 struct devlink *devlink = info->user_ptr[0];
629 struct devlink_sb *devlink_sb;
634 devlink_sb = devlink_sb_get_from_info(devlink, info);
635 if (IS_ERR(devlink_sb))
636 return PTR_ERR(devlink_sb);
638 err = devlink_sb_pool_index_get_from_info(devlink_sb, info,
643 if (GENL_REQ_ATTR_CHECK(info, DEVLINK_ATTR_SB_THRESHOLD))
646 threshold = nla_get_u32(info->attrs[DEVLINK_ATTR_SB_THRESHOLD]);
647 return devlink_sb_port_pool_set(devlink_port, devlink_sb->index,
648 pool_index, threshold, info->extack);
652 devlink_nl_sb_tc_pool_bind_fill(struct sk_buff *msg, struct devlink *devlink,
653 struct devlink_port *devlink_port,
654 struct devlink_sb *devlink_sb, u16 tc_index,
655 enum devlink_sb_pool_type pool_type,
656 enum devlink_command cmd,
657 u32 portid, u32 seq, int flags)
659 const struct devlink_ops *ops = devlink->ops;
665 err = ops->sb_tc_pool_bind_get(devlink_port, devlink_sb->index,
667 &pool_index, &threshold);
671 hdr = genlmsg_put(msg, portid, seq, &devlink_nl_family, flags, cmd);
675 if (devlink_nl_put_handle(msg, devlink))
676 goto nla_put_failure;
677 if (nla_put_u32(msg, DEVLINK_ATTR_PORT_INDEX, devlink_port->index))
678 goto nla_put_failure;
679 if (nla_put_u32(msg, DEVLINK_ATTR_SB_INDEX, devlink_sb->index))
680 goto nla_put_failure;
681 if (nla_put_u16(msg, DEVLINK_ATTR_SB_TC_INDEX, tc_index))
682 goto nla_put_failure;
683 if (nla_put_u8(msg, DEVLINK_ATTR_SB_POOL_TYPE, pool_type))
684 goto nla_put_failure;
685 if (nla_put_u16(msg, DEVLINK_ATTR_SB_POOL_INDEX, pool_index))
686 goto nla_put_failure;
687 if (nla_put_u32(msg, DEVLINK_ATTR_SB_THRESHOLD, threshold))
688 goto nla_put_failure;
690 if (ops->sb_occ_tc_port_bind_get) {
694 err = ops->sb_occ_tc_port_bind_get(devlink_port,
698 if (err && err != -EOPNOTSUPP)
701 if (nla_put_u32(msg, DEVLINK_ATTR_SB_OCC_CUR, cur))
702 goto nla_put_failure;
703 if (nla_put_u32(msg, DEVLINK_ATTR_SB_OCC_MAX, max))
704 goto nla_put_failure;
708 genlmsg_end(msg, hdr);
712 genlmsg_cancel(msg, hdr);
716 int devlink_nl_sb_tc_pool_bind_get_doit(struct sk_buff *skb,
717 struct genl_info *info)
719 struct devlink_port *devlink_port = info->user_ptr[1];
720 struct devlink *devlink = devlink_port->devlink;
721 struct devlink_sb *devlink_sb;
723 enum devlink_sb_pool_type pool_type;
727 devlink_sb = devlink_sb_get_from_info(devlink, info);
728 if (IS_ERR(devlink_sb))
729 return PTR_ERR(devlink_sb);
731 err = devlink_sb_pool_type_get_from_info(info, &pool_type);
735 err = devlink_sb_tc_index_get_from_info(devlink_sb, info,
736 pool_type, &tc_index);
740 if (!devlink->ops->sb_tc_pool_bind_get)
743 msg = nlmsg_new(NLMSG_DEFAULT_SIZE, GFP_KERNEL);
747 err = devlink_nl_sb_tc_pool_bind_fill(msg, devlink, devlink_port,
748 devlink_sb, tc_index, pool_type,
749 DEVLINK_CMD_SB_TC_POOL_BIND_NEW,
757 return genlmsg_reply(msg, info);
760 static int __sb_tc_pool_bind_get_dumpit(struct sk_buff *msg,
761 int start, int *p_idx,
762 struct devlink *devlink,
763 struct devlink_sb *devlink_sb,
764 u32 portid, u32 seq, int flags)
766 struct devlink_port *devlink_port;
767 unsigned long port_index;
771 xa_for_each(&devlink->ports, port_index, devlink_port) {
773 tc_index < devlink_sb->ingress_tc_count; tc_index++) {
774 if (*p_idx < start) {
778 err = devlink_nl_sb_tc_pool_bind_fill(msg, devlink,
782 DEVLINK_SB_POOL_TYPE_INGRESS,
783 DEVLINK_CMD_SB_TC_POOL_BIND_NEW,
791 tc_index < devlink_sb->egress_tc_count; tc_index++) {
792 if (*p_idx < start) {
796 err = devlink_nl_sb_tc_pool_bind_fill(msg, devlink,
800 DEVLINK_SB_POOL_TYPE_EGRESS,
801 DEVLINK_CMD_SB_TC_POOL_BIND_NEW,
812 static int devlink_nl_sb_tc_pool_bind_get_dump_one(struct sk_buff *msg,
813 struct devlink *devlink,
814 struct netlink_callback *cb,
817 struct devlink_nl_dump_state *state = devlink_dump_state(cb);
818 struct devlink_sb *devlink_sb;
822 if (!devlink->ops->sb_tc_pool_bind_get)
825 list_for_each_entry(devlink_sb, &devlink->sb_list, list) {
826 err = __sb_tc_pool_bind_get_dumpit(msg, state->idx, &idx,
828 NETLINK_CB(cb->skb).portid,
829 cb->nlh->nlmsg_seq, flags);
830 if (err == -EOPNOTSUPP) {
841 int devlink_nl_sb_tc_pool_bind_get_dumpit(struct sk_buff *skb,
842 struct netlink_callback *cb)
844 return devlink_nl_dumpit(skb, cb,
845 devlink_nl_sb_tc_pool_bind_get_dump_one);
848 static int devlink_sb_tc_pool_bind_set(struct devlink_port *devlink_port,
849 unsigned int sb_index, u16 tc_index,
850 enum devlink_sb_pool_type pool_type,
851 u16 pool_index, u32 threshold,
852 struct netlink_ext_ack *extack)
855 const struct devlink_ops *ops = devlink_port->devlink->ops;
857 if (ops->sb_tc_pool_bind_set)
858 return ops->sb_tc_pool_bind_set(devlink_port, sb_index,
860 pool_index, threshold, extack);
864 int devlink_nl_cmd_sb_tc_pool_bind_set_doit(struct sk_buff *skb,
865 struct genl_info *info)
867 struct devlink_port *devlink_port = info->user_ptr[1];
868 struct devlink *devlink = info->user_ptr[0];
869 enum devlink_sb_pool_type pool_type;
870 struct devlink_sb *devlink_sb;
876 devlink_sb = devlink_sb_get_from_info(devlink, info);
877 if (IS_ERR(devlink_sb))
878 return PTR_ERR(devlink_sb);
880 err = devlink_sb_pool_type_get_from_info(info, &pool_type);
884 err = devlink_sb_tc_index_get_from_info(devlink_sb, info,
885 pool_type, &tc_index);
889 err = devlink_sb_pool_index_get_from_info(devlink_sb, info,
894 if (GENL_REQ_ATTR_CHECK(info, DEVLINK_ATTR_SB_THRESHOLD))
897 threshold = nla_get_u32(info->attrs[DEVLINK_ATTR_SB_THRESHOLD]);
898 return devlink_sb_tc_pool_bind_set(devlink_port, devlink_sb->index,
900 pool_index, threshold, info->extack);
903 int devlink_nl_cmd_sb_occ_snapshot_doit(struct sk_buff *skb,
904 struct genl_info *info)
906 struct devlink *devlink = info->user_ptr[0];
907 const struct devlink_ops *ops = devlink->ops;
908 struct devlink_sb *devlink_sb;
910 devlink_sb = devlink_sb_get_from_info(devlink, info);
911 if (IS_ERR(devlink_sb))
912 return PTR_ERR(devlink_sb);
914 if (ops->sb_occ_snapshot)
915 return ops->sb_occ_snapshot(devlink, devlink_sb->index);
919 int devlink_nl_cmd_sb_occ_max_clear_doit(struct sk_buff *skb,
920 struct genl_info *info)
922 struct devlink *devlink = info->user_ptr[0];
923 const struct devlink_ops *ops = devlink->ops;
924 struct devlink_sb *devlink_sb;
926 devlink_sb = devlink_sb_get_from_info(devlink, info);
927 if (IS_ERR(devlink_sb))
928 return PTR_ERR(devlink_sb);
930 if (ops->sb_occ_max_clear)
931 return ops->sb_occ_max_clear(devlink, devlink_sb->index);
935 int devl_sb_register(struct devlink *devlink, unsigned int sb_index,
936 u32 size, u16 ingress_pools_count,
937 u16 egress_pools_count, u16 ingress_tc_count,
940 struct devlink_sb *devlink_sb;
942 lockdep_assert_held(&devlink->lock);
944 if (devlink_sb_index_exists(devlink, sb_index))
947 devlink_sb = kzalloc(sizeof(*devlink_sb), GFP_KERNEL);
950 devlink_sb->index = sb_index;
951 devlink_sb->size = size;
952 devlink_sb->ingress_pools_count = ingress_pools_count;
953 devlink_sb->egress_pools_count = egress_pools_count;
954 devlink_sb->ingress_tc_count = ingress_tc_count;
955 devlink_sb->egress_tc_count = egress_tc_count;
956 list_add_tail(&devlink_sb->list, &devlink->sb_list);
959 EXPORT_SYMBOL_GPL(devl_sb_register);
961 int devlink_sb_register(struct devlink *devlink, unsigned int sb_index,
962 u32 size, u16 ingress_pools_count,
963 u16 egress_pools_count, u16 ingress_tc_count,
969 err = devl_sb_register(devlink, sb_index, size, ingress_pools_count,
970 egress_pools_count, ingress_tc_count,
972 devl_unlock(devlink);
975 EXPORT_SYMBOL_GPL(devlink_sb_register);
977 void devl_sb_unregister(struct devlink *devlink, unsigned int sb_index)
979 struct devlink_sb *devlink_sb;
981 lockdep_assert_held(&devlink->lock);
983 devlink_sb = devlink_sb_get_by_index(devlink, sb_index);
984 WARN_ON(!devlink_sb);
985 list_del(&devlink_sb->list);
988 EXPORT_SYMBOL_GPL(devl_sb_unregister);
990 void devlink_sb_unregister(struct devlink *devlink, unsigned int sb_index)
993 devl_sb_unregister(devlink, sb_index);
994 devl_unlock(devlink);
996 EXPORT_SYMBOL_GPL(devlink_sb_unregister);