Linux 6.7-rc7
[linux-modified.git] / net / devlink / rate.c
1 // SPDX-License-Identifier: GPL-2.0-or-later
2 /*
3  * Copyright (c) 2016 Mellanox Technologies. All rights reserved.
4  * Copyright (c) 2016 Jiri Pirko <jiri@mellanox.com>
5  */
6
7 #include "devl_internal.h"
8
9 static inline bool
10 devlink_rate_is_leaf(struct devlink_rate *devlink_rate)
11 {
12         return devlink_rate->type == DEVLINK_RATE_TYPE_LEAF;
13 }
14
15 static inline bool
16 devlink_rate_is_node(struct devlink_rate *devlink_rate)
17 {
18         return devlink_rate->type == DEVLINK_RATE_TYPE_NODE;
19 }
20
21 static struct devlink_rate *
22 devlink_rate_leaf_get_from_info(struct devlink *devlink, struct genl_info *info)
23 {
24         struct devlink_rate *devlink_rate;
25         struct devlink_port *devlink_port;
26
27         devlink_port = devlink_port_get_from_attrs(devlink, info->attrs);
28         if (IS_ERR(devlink_port))
29                 return ERR_CAST(devlink_port);
30         devlink_rate = devlink_port->devlink_rate;
31         return devlink_rate ?: ERR_PTR(-ENODEV);
32 }
33
34 static struct devlink_rate *
35 devlink_rate_node_get_by_name(struct devlink *devlink, const char *node_name)
36 {
37         static struct devlink_rate *devlink_rate;
38
39         list_for_each_entry(devlink_rate, &devlink->rate_list, list) {
40                 if (devlink_rate_is_node(devlink_rate) &&
41                     !strcmp(node_name, devlink_rate->name))
42                         return devlink_rate;
43         }
44         return ERR_PTR(-ENODEV);
45 }
46
47 static struct devlink_rate *
48 devlink_rate_node_get_from_attrs(struct devlink *devlink, struct nlattr **attrs)
49 {
50         const char *rate_node_name;
51         size_t len;
52
53         if (!attrs[DEVLINK_ATTR_RATE_NODE_NAME])
54                 return ERR_PTR(-EINVAL);
55         rate_node_name = nla_data(attrs[DEVLINK_ATTR_RATE_NODE_NAME]);
56         len = strlen(rate_node_name);
57         /* Name cannot be empty or decimal number */
58         if (!len || strspn(rate_node_name, "0123456789") == len)
59                 return ERR_PTR(-EINVAL);
60
61         return devlink_rate_node_get_by_name(devlink, rate_node_name);
62 }
63
64 static struct devlink_rate *
65 devlink_rate_node_get_from_info(struct devlink *devlink, struct genl_info *info)
66 {
67         return devlink_rate_node_get_from_attrs(devlink, info->attrs);
68 }
69
70 static struct devlink_rate *
71 devlink_rate_get_from_info(struct devlink *devlink, struct genl_info *info)
72 {
73         struct nlattr **attrs = info->attrs;
74
75         if (attrs[DEVLINK_ATTR_PORT_INDEX])
76                 return devlink_rate_leaf_get_from_info(devlink, info);
77         else if (attrs[DEVLINK_ATTR_RATE_NODE_NAME])
78                 return devlink_rate_node_get_from_info(devlink, info);
79         else
80                 return ERR_PTR(-EINVAL);
81 }
82
83 static int devlink_nl_rate_fill(struct sk_buff *msg,
84                                 struct devlink_rate *devlink_rate,
85                                 enum devlink_command cmd, u32 portid, u32 seq,
86                                 int flags, struct netlink_ext_ack *extack)
87 {
88         struct devlink *devlink = devlink_rate->devlink;
89         void *hdr;
90
91         hdr = genlmsg_put(msg, portid, seq, &devlink_nl_family, flags, cmd);
92         if (!hdr)
93                 return -EMSGSIZE;
94
95         if (devlink_nl_put_handle(msg, devlink))
96                 goto nla_put_failure;
97
98         if (nla_put_u16(msg, DEVLINK_ATTR_RATE_TYPE, devlink_rate->type))
99                 goto nla_put_failure;
100
101         if (devlink_rate_is_leaf(devlink_rate)) {
102                 if (nla_put_u32(msg, DEVLINK_ATTR_PORT_INDEX,
103                                 devlink_rate->devlink_port->index))
104                         goto nla_put_failure;
105         } else if (devlink_rate_is_node(devlink_rate)) {
106                 if (nla_put_string(msg, DEVLINK_ATTR_RATE_NODE_NAME,
107                                    devlink_rate->name))
108                         goto nla_put_failure;
109         }
110
111         if (nla_put_u64_64bit(msg, DEVLINK_ATTR_RATE_TX_SHARE,
112                               devlink_rate->tx_share, DEVLINK_ATTR_PAD))
113                 goto nla_put_failure;
114
115         if (nla_put_u64_64bit(msg, DEVLINK_ATTR_RATE_TX_MAX,
116                               devlink_rate->tx_max, DEVLINK_ATTR_PAD))
117                 goto nla_put_failure;
118
119         if (nla_put_u32(msg, DEVLINK_ATTR_RATE_TX_PRIORITY,
120                         devlink_rate->tx_priority))
121                 goto nla_put_failure;
122
123         if (nla_put_u32(msg, DEVLINK_ATTR_RATE_TX_WEIGHT,
124                         devlink_rate->tx_weight))
125                 goto nla_put_failure;
126
127         if (devlink_rate->parent)
128                 if (nla_put_string(msg, DEVLINK_ATTR_RATE_PARENT_NODE_NAME,
129                                    devlink_rate->parent->name))
130                         goto nla_put_failure;
131
132         genlmsg_end(msg, hdr);
133         return 0;
134
135 nla_put_failure:
136         genlmsg_cancel(msg, hdr);
137         return -EMSGSIZE;
138 }
139
140 static void devlink_rate_notify(struct devlink_rate *devlink_rate,
141                                 enum devlink_command cmd)
142 {
143         struct devlink *devlink = devlink_rate->devlink;
144         struct sk_buff *msg;
145         int err;
146
147         WARN_ON(cmd != DEVLINK_CMD_RATE_NEW && cmd != DEVLINK_CMD_RATE_DEL);
148
149         if (!xa_get_mark(&devlinks, devlink->index, DEVLINK_REGISTERED))
150                 return;
151
152         msg = nlmsg_new(NLMSG_DEFAULT_SIZE, GFP_KERNEL);
153         if (!msg)
154                 return;
155
156         err = devlink_nl_rate_fill(msg, devlink_rate, cmd, 0, 0, 0, NULL);
157         if (err) {
158                 nlmsg_free(msg);
159                 return;
160         }
161
162         genlmsg_multicast_netns(&devlink_nl_family, devlink_net(devlink), msg,
163                                 0, DEVLINK_MCGRP_CONFIG, GFP_KERNEL);
164 }
165
166 void devlink_rates_notify_register(struct devlink *devlink)
167 {
168         struct devlink_rate *rate_node;
169
170         list_for_each_entry(rate_node, &devlink->rate_list, list)
171                 devlink_rate_notify(rate_node, DEVLINK_CMD_RATE_NEW);
172 }
173
174 void devlink_rates_notify_unregister(struct devlink *devlink)
175 {
176         struct devlink_rate *rate_node;
177
178         list_for_each_entry_reverse(rate_node, &devlink->rate_list, list)
179                 devlink_rate_notify(rate_node, DEVLINK_CMD_RATE_DEL);
180 }
181
182 static int
183 devlink_nl_rate_get_dump_one(struct sk_buff *msg, struct devlink *devlink,
184                              struct netlink_callback *cb, int flags)
185 {
186         struct devlink_nl_dump_state *state = devlink_dump_state(cb);
187         struct devlink_rate *devlink_rate;
188         int idx = 0;
189         int err = 0;
190
191         list_for_each_entry(devlink_rate, &devlink->rate_list, list) {
192                 enum devlink_command cmd = DEVLINK_CMD_RATE_NEW;
193                 u32 id = NETLINK_CB(cb->skb).portid;
194
195                 if (idx < state->idx) {
196                         idx++;
197                         continue;
198                 }
199                 err = devlink_nl_rate_fill(msg, devlink_rate, cmd, id,
200                                            cb->nlh->nlmsg_seq, flags, NULL);
201                 if (err) {
202                         state->idx = idx;
203                         break;
204                 }
205                 idx++;
206         }
207
208         return err;
209 }
210
211 int devlink_nl_rate_get_dumpit(struct sk_buff *skb, struct netlink_callback *cb)
212 {
213         return devlink_nl_dumpit(skb, cb, devlink_nl_rate_get_dump_one);
214 }
215
216 int devlink_nl_rate_get_doit(struct sk_buff *skb, struct genl_info *info)
217 {
218         struct devlink *devlink = info->user_ptr[0];
219         struct devlink_rate *devlink_rate;
220         struct sk_buff *msg;
221         int err;
222
223         devlink_rate = devlink_rate_get_from_info(devlink, info);
224         if (IS_ERR(devlink_rate))
225                 return PTR_ERR(devlink_rate);
226
227         msg = nlmsg_new(NLMSG_DEFAULT_SIZE, GFP_KERNEL);
228         if (!msg)
229                 return -ENOMEM;
230
231         err = devlink_nl_rate_fill(msg, devlink_rate, DEVLINK_CMD_RATE_NEW,
232                                    info->snd_portid, info->snd_seq, 0,
233                                    info->extack);
234         if (err) {
235                 nlmsg_free(msg);
236                 return err;
237         }
238
239         return genlmsg_reply(msg, info);
240 }
241
242 static bool
243 devlink_rate_is_parent_node(struct devlink_rate *devlink_rate,
244                             struct devlink_rate *parent)
245 {
246         while (parent) {
247                 if (parent == devlink_rate)
248                         return true;
249                 parent = parent->parent;
250         }
251         return false;
252 }
253
254 static int
255 devlink_nl_rate_parent_node_set(struct devlink_rate *devlink_rate,
256                                 struct genl_info *info,
257                                 struct nlattr *nla_parent)
258 {
259         struct devlink *devlink = devlink_rate->devlink;
260         const char *parent_name = nla_data(nla_parent);
261         const struct devlink_ops *ops = devlink->ops;
262         size_t len = strlen(parent_name);
263         struct devlink_rate *parent;
264         int err = -EOPNOTSUPP;
265
266         parent = devlink_rate->parent;
267
268         if (parent && !len) {
269                 if (devlink_rate_is_leaf(devlink_rate))
270                         err = ops->rate_leaf_parent_set(devlink_rate, NULL,
271                                                         devlink_rate->priv, NULL,
272                                                         info->extack);
273                 else if (devlink_rate_is_node(devlink_rate))
274                         err = ops->rate_node_parent_set(devlink_rate, NULL,
275                                                         devlink_rate->priv, NULL,
276                                                         info->extack);
277                 if (err)
278                         return err;
279
280                 refcount_dec(&parent->refcnt);
281                 devlink_rate->parent = NULL;
282         } else if (len) {
283                 parent = devlink_rate_node_get_by_name(devlink, parent_name);
284                 if (IS_ERR(parent))
285                         return -ENODEV;
286
287                 if (parent == devlink_rate) {
288                         NL_SET_ERR_MSG(info->extack, "Parent to self is not allowed");
289                         return -EINVAL;
290                 }
291
292                 if (devlink_rate_is_node(devlink_rate) &&
293                     devlink_rate_is_parent_node(devlink_rate, parent->parent)) {
294                         NL_SET_ERR_MSG(info->extack, "Node is already a parent of parent node.");
295                         return -EEXIST;
296                 }
297
298                 if (devlink_rate_is_leaf(devlink_rate))
299                         err = ops->rate_leaf_parent_set(devlink_rate, parent,
300                                                         devlink_rate->priv, parent->priv,
301                                                         info->extack);
302                 else if (devlink_rate_is_node(devlink_rate))
303                         err = ops->rate_node_parent_set(devlink_rate, parent,
304                                                         devlink_rate->priv, parent->priv,
305                                                         info->extack);
306                 if (err)
307                         return err;
308
309                 if (devlink_rate->parent)
310                         /* we're reassigning to other parent in this case */
311                         refcount_dec(&devlink_rate->parent->refcnt);
312
313                 refcount_inc(&parent->refcnt);
314                 devlink_rate->parent = parent;
315         }
316
317         return 0;
318 }
319
320 static int devlink_nl_rate_set(struct devlink_rate *devlink_rate,
321                                const struct devlink_ops *ops,
322                                struct genl_info *info)
323 {
324         struct nlattr *nla_parent, **attrs = info->attrs;
325         int err = -EOPNOTSUPP;
326         u32 priority;
327         u32 weight;
328         u64 rate;
329
330         if (attrs[DEVLINK_ATTR_RATE_TX_SHARE]) {
331                 rate = nla_get_u64(attrs[DEVLINK_ATTR_RATE_TX_SHARE]);
332                 if (devlink_rate_is_leaf(devlink_rate))
333                         err = ops->rate_leaf_tx_share_set(devlink_rate, devlink_rate->priv,
334                                                           rate, info->extack);
335                 else if (devlink_rate_is_node(devlink_rate))
336                         err = ops->rate_node_tx_share_set(devlink_rate, devlink_rate->priv,
337                                                           rate, info->extack);
338                 if (err)
339                         return err;
340                 devlink_rate->tx_share = rate;
341         }
342
343         if (attrs[DEVLINK_ATTR_RATE_TX_MAX]) {
344                 rate = nla_get_u64(attrs[DEVLINK_ATTR_RATE_TX_MAX]);
345                 if (devlink_rate_is_leaf(devlink_rate))
346                         err = ops->rate_leaf_tx_max_set(devlink_rate, devlink_rate->priv,
347                                                         rate, info->extack);
348                 else if (devlink_rate_is_node(devlink_rate))
349                         err = ops->rate_node_tx_max_set(devlink_rate, devlink_rate->priv,
350                                                         rate, info->extack);
351                 if (err)
352                         return err;
353                 devlink_rate->tx_max = rate;
354         }
355
356         if (attrs[DEVLINK_ATTR_RATE_TX_PRIORITY]) {
357                 priority = nla_get_u32(attrs[DEVLINK_ATTR_RATE_TX_PRIORITY]);
358                 if (devlink_rate_is_leaf(devlink_rate))
359                         err = ops->rate_leaf_tx_priority_set(devlink_rate, devlink_rate->priv,
360                                                              priority, info->extack);
361                 else if (devlink_rate_is_node(devlink_rate))
362                         err = ops->rate_node_tx_priority_set(devlink_rate, devlink_rate->priv,
363                                                              priority, info->extack);
364
365                 if (err)
366                         return err;
367                 devlink_rate->tx_priority = priority;
368         }
369
370         if (attrs[DEVLINK_ATTR_RATE_TX_WEIGHT]) {
371                 weight = nla_get_u32(attrs[DEVLINK_ATTR_RATE_TX_WEIGHT]);
372                 if (devlink_rate_is_leaf(devlink_rate))
373                         err = ops->rate_leaf_tx_weight_set(devlink_rate, devlink_rate->priv,
374                                                            weight, info->extack);
375                 else if (devlink_rate_is_node(devlink_rate))
376                         err = ops->rate_node_tx_weight_set(devlink_rate, devlink_rate->priv,
377                                                            weight, info->extack);
378
379                 if (err)
380                         return err;
381                 devlink_rate->tx_weight = weight;
382         }
383
384         nla_parent = attrs[DEVLINK_ATTR_RATE_PARENT_NODE_NAME];
385         if (nla_parent) {
386                 err = devlink_nl_rate_parent_node_set(devlink_rate, info,
387                                                       nla_parent);
388                 if (err)
389                         return err;
390         }
391
392         return 0;
393 }
394
395 static bool devlink_rate_set_ops_supported(const struct devlink_ops *ops,
396                                            struct genl_info *info,
397                                            enum devlink_rate_type type)
398 {
399         struct nlattr **attrs = info->attrs;
400
401         if (type == DEVLINK_RATE_TYPE_LEAF) {
402                 if (attrs[DEVLINK_ATTR_RATE_TX_SHARE] && !ops->rate_leaf_tx_share_set) {
403                         NL_SET_ERR_MSG(info->extack, "TX share set isn't supported for the leafs");
404                         return false;
405                 }
406                 if (attrs[DEVLINK_ATTR_RATE_TX_MAX] && !ops->rate_leaf_tx_max_set) {
407                         NL_SET_ERR_MSG(info->extack, "TX max set isn't supported for the leafs");
408                         return false;
409                 }
410                 if (attrs[DEVLINK_ATTR_RATE_PARENT_NODE_NAME] &&
411                     !ops->rate_leaf_parent_set) {
412                         NL_SET_ERR_MSG(info->extack, "Parent set isn't supported for the leafs");
413                         return false;
414                 }
415                 if (attrs[DEVLINK_ATTR_RATE_TX_PRIORITY] && !ops->rate_leaf_tx_priority_set) {
416                         NL_SET_ERR_MSG_ATTR(info->extack,
417                                             attrs[DEVLINK_ATTR_RATE_TX_PRIORITY],
418                                             "TX priority set isn't supported for the leafs");
419                         return false;
420                 }
421                 if (attrs[DEVLINK_ATTR_RATE_TX_WEIGHT] && !ops->rate_leaf_tx_weight_set) {
422                         NL_SET_ERR_MSG_ATTR(info->extack,
423                                             attrs[DEVLINK_ATTR_RATE_TX_WEIGHT],
424                                             "TX weight set isn't supported for the leafs");
425                         return false;
426                 }
427         } else if (type == DEVLINK_RATE_TYPE_NODE) {
428                 if (attrs[DEVLINK_ATTR_RATE_TX_SHARE] && !ops->rate_node_tx_share_set) {
429                         NL_SET_ERR_MSG(info->extack, "TX share set isn't supported for the nodes");
430                         return false;
431                 }
432                 if (attrs[DEVLINK_ATTR_RATE_TX_MAX] && !ops->rate_node_tx_max_set) {
433                         NL_SET_ERR_MSG(info->extack, "TX max set isn't supported for the nodes");
434                         return false;
435                 }
436                 if (attrs[DEVLINK_ATTR_RATE_PARENT_NODE_NAME] &&
437                     !ops->rate_node_parent_set) {
438                         NL_SET_ERR_MSG(info->extack, "Parent set isn't supported for the nodes");
439                         return false;
440                 }
441                 if (attrs[DEVLINK_ATTR_RATE_TX_PRIORITY] && !ops->rate_node_tx_priority_set) {
442                         NL_SET_ERR_MSG_ATTR(info->extack,
443                                             attrs[DEVLINK_ATTR_RATE_TX_PRIORITY],
444                                             "TX priority set isn't supported for the nodes");
445                         return false;
446                 }
447                 if (attrs[DEVLINK_ATTR_RATE_TX_WEIGHT] && !ops->rate_node_tx_weight_set) {
448                         NL_SET_ERR_MSG_ATTR(info->extack,
449                                             attrs[DEVLINK_ATTR_RATE_TX_WEIGHT],
450                                             "TX weight set isn't supported for the nodes");
451                         return false;
452                 }
453         } else {
454                 WARN(1, "Unknown type of rate object");
455                 return false;
456         }
457
458         return true;
459 }
460
461 int devlink_nl_rate_set_doit(struct sk_buff *skb, struct genl_info *info)
462 {
463         struct devlink *devlink = info->user_ptr[0];
464         struct devlink_rate *devlink_rate;
465         const struct devlink_ops *ops;
466         int err;
467
468         devlink_rate = devlink_rate_get_from_info(devlink, info);
469         if (IS_ERR(devlink_rate))
470                 return PTR_ERR(devlink_rate);
471
472         ops = devlink->ops;
473         if (!ops || !devlink_rate_set_ops_supported(ops, info, devlink_rate->type))
474                 return -EOPNOTSUPP;
475
476         err = devlink_nl_rate_set(devlink_rate, ops, info);
477
478         if (!err)
479                 devlink_rate_notify(devlink_rate, DEVLINK_CMD_RATE_NEW);
480         return err;
481 }
482
483 int devlink_nl_rate_new_doit(struct sk_buff *skb, struct genl_info *info)
484 {
485         struct devlink *devlink = info->user_ptr[0];
486         struct devlink_rate *rate_node;
487         const struct devlink_ops *ops;
488         int err;
489
490         ops = devlink->ops;
491         if (!ops || !ops->rate_node_new || !ops->rate_node_del) {
492                 NL_SET_ERR_MSG(info->extack, "Rate nodes aren't supported");
493                 return -EOPNOTSUPP;
494         }
495
496         if (!devlink_rate_set_ops_supported(ops, info, DEVLINK_RATE_TYPE_NODE))
497                 return -EOPNOTSUPP;
498
499         rate_node = devlink_rate_node_get_from_attrs(devlink, info->attrs);
500         if (!IS_ERR(rate_node))
501                 return -EEXIST;
502         else if (rate_node == ERR_PTR(-EINVAL))
503                 return -EINVAL;
504
505         rate_node = kzalloc(sizeof(*rate_node), GFP_KERNEL);
506         if (!rate_node)
507                 return -ENOMEM;
508
509         rate_node->devlink = devlink;
510         rate_node->type = DEVLINK_RATE_TYPE_NODE;
511         rate_node->name = nla_strdup(info->attrs[DEVLINK_ATTR_RATE_NODE_NAME], GFP_KERNEL);
512         if (!rate_node->name) {
513                 err = -ENOMEM;
514                 goto err_strdup;
515         }
516
517         err = ops->rate_node_new(rate_node, &rate_node->priv, info->extack);
518         if (err)
519                 goto err_node_new;
520
521         err = devlink_nl_rate_set(rate_node, ops, info);
522         if (err)
523                 goto err_rate_set;
524
525         refcount_set(&rate_node->refcnt, 1);
526         list_add(&rate_node->list, &devlink->rate_list);
527         devlink_rate_notify(rate_node, DEVLINK_CMD_RATE_NEW);
528         return 0;
529
530 err_rate_set:
531         ops->rate_node_del(rate_node, rate_node->priv, info->extack);
532 err_node_new:
533         kfree(rate_node->name);
534 err_strdup:
535         kfree(rate_node);
536         return err;
537 }
538
539 int devlink_nl_rate_del_doit(struct sk_buff *skb, struct genl_info *info)
540 {
541         struct devlink *devlink = info->user_ptr[0];
542         struct devlink_rate *rate_node;
543         int err;
544
545         rate_node = devlink_rate_node_get_from_info(devlink, info);
546         if (IS_ERR(rate_node))
547                 return PTR_ERR(rate_node);
548
549         if (refcount_read(&rate_node->refcnt) > 1) {
550                 NL_SET_ERR_MSG(info->extack, "Node has children. Cannot delete node.");
551                 return -EBUSY;
552         }
553
554         devlink_rate_notify(rate_node, DEVLINK_CMD_RATE_DEL);
555         err = devlink->ops->rate_node_del(rate_node, rate_node->priv,
556                                           info->extack);
557         if (rate_node->parent)
558                 refcount_dec(&rate_node->parent->refcnt);
559         list_del(&rate_node->list);
560         kfree(rate_node->name);
561         kfree(rate_node);
562         return err;
563 }
564
565 int devlink_rate_nodes_check(struct devlink *devlink, u16 mode,
566                              struct netlink_ext_ack *extack)
567 {
568         struct devlink_rate *devlink_rate;
569
570         list_for_each_entry(devlink_rate, &devlink->rate_list, list)
571                 if (devlink_rate_is_node(devlink_rate)) {
572                         NL_SET_ERR_MSG(extack, "Rate node(s) exists.");
573                         return -EBUSY;
574                 }
575         return 0;
576 }
577
578 /**
579  * devl_rate_node_create - create devlink rate node
580  * @devlink: devlink instance
581  * @priv: driver private data
582  * @node_name: name of the resulting node
583  * @parent: parent devlink_rate struct
584  *
585  * Create devlink rate object of type node
586  */
587 struct devlink_rate *
588 devl_rate_node_create(struct devlink *devlink, void *priv, char *node_name,
589                       struct devlink_rate *parent)
590 {
591         struct devlink_rate *rate_node;
592
593         rate_node = devlink_rate_node_get_by_name(devlink, node_name);
594         if (!IS_ERR(rate_node))
595                 return ERR_PTR(-EEXIST);
596
597         rate_node = kzalloc(sizeof(*rate_node), GFP_KERNEL);
598         if (!rate_node)
599                 return ERR_PTR(-ENOMEM);
600
601         if (parent) {
602                 rate_node->parent = parent;
603                 refcount_inc(&rate_node->parent->refcnt);
604         }
605
606         rate_node->type = DEVLINK_RATE_TYPE_NODE;
607         rate_node->devlink = devlink;
608         rate_node->priv = priv;
609
610         rate_node->name = kstrdup(node_name, GFP_KERNEL);
611         if (!rate_node->name) {
612                 kfree(rate_node);
613                 return ERR_PTR(-ENOMEM);
614         }
615
616         refcount_set(&rate_node->refcnt, 1);
617         list_add(&rate_node->list, &devlink->rate_list);
618         devlink_rate_notify(rate_node, DEVLINK_CMD_RATE_NEW);
619         return rate_node;
620 }
621 EXPORT_SYMBOL_GPL(devl_rate_node_create);
622
623 /**
624  * devl_rate_leaf_create - create devlink rate leaf
625  * @devlink_port: devlink port object to create rate object on
626  * @priv: driver private data
627  * @parent: parent devlink_rate struct
628  *
629  * Create devlink rate object of type leaf on provided @devlink_port.
630  */
631 int devl_rate_leaf_create(struct devlink_port *devlink_port, void *priv,
632                           struct devlink_rate *parent)
633 {
634         struct devlink *devlink = devlink_port->devlink;
635         struct devlink_rate *devlink_rate;
636
637         devl_assert_locked(devlink_port->devlink);
638
639         if (WARN_ON(devlink_port->devlink_rate))
640                 return -EBUSY;
641
642         devlink_rate = kzalloc(sizeof(*devlink_rate), GFP_KERNEL);
643         if (!devlink_rate)
644                 return -ENOMEM;
645
646         if (parent) {
647                 devlink_rate->parent = parent;
648                 refcount_inc(&devlink_rate->parent->refcnt);
649         }
650
651         devlink_rate->type = DEVLINK_RATE_TYPE_LEAF;
652         devlink_rate->devlink = devlink;
653         devlink_rate->devlink_port = devlink_port;
654         devlink_rate->priv = priv;
655         list_add_tail(&devlink_rate->list, &devlink->rate_list);
656         devlink_port->devlink_rate = devlink_rate;
657         devlink_rate_notify(devlink_rate, DEVLINK_CMD_RATE_NEW);
658
659         return 0;
660 }
661 EXPORT_SYMBOL_GPL(devl_rate_leaf_create);
662
663 /**
664  * devl_rate_leaf_destroy - destroy devlink rate leaf
665  *
666  * @devlink_port: devlink port linked to the rate object
667  *
668  * Destroy the devlink rate object of type leaf on provided @devlink_port.
669  */
670 void devl_rate_leaf_destroy(struct devlink_port *devlink_port)
671 {
672         struct devlink_rate *devlink_rate = devlink_port->devlink_rate;
673
674         devl_assert_locked(devlink_port->devlink);
675         if (!devlink_rate)
676                 return;
677
678         devlink_rate_notify(devlink_rate, DEVLINK_CMD_RATE_DEL);
679         if (devlink_rate->parent)
680                 refcount_dec(&devlink_rate->parent->refcnt);
681         list_del(&devlink_rate->list);
682         devlink_port->devlink_rate = NULL;
683         kfree(devlink_rate);
684 }
685 EXPORT_SYMBOL_GPL(devl_rate_leaf_destroy);
686
687 /**
688  * devl_rate_nodes_destroy - destroy all devlink rate nodes on device
689  * @devlink: devlink instance
690  *
691  * Unset parent for all rate objects and destroy all rate nodes
692  * on specified device.
693  */
694 void devl_rate_nodes_destroy(struct devlink *devlink)
695 {
696         static struct devlink_rate *devlink_rate, *tmp;
697         const struct devlink_ops *ops = devlink->ops;
698
699         devl_assert_locked(devlink);
700
701         list_for_each_entry(devlink_rate, &devlink->rate_list, list) {
702                 if (!devlink_rate->parent)
703                         continue;
704
705                 refcount_dec(&devlink_rate->parent->refcnt);
706                 if (devlink_rate_is_leaf(devlink_rate))
707                         ops->rate_leaf_parent_set(devlink_rate, NULL, devlink_rate->priv,
708                                                   NULL, NULL);
709                 else if (devlink_rate_is_node(devlink_rate))
710                         ops->rate_node_parent_set(devlink_rate, NULL, devlink_rate->priv,
711                                                   NULL, NULL);
712         }
713         list_for_each_entry_safe(devlink_rate, tmp, &devlink->rate_list, list) {
714                 if (devlink_rate_is_node(devlink_rate)) {
715                         ops->rate_node_del(devlink_rate, devlink_rate->priv, NULL);
716                         list_del(&devlink_rate->list);
717                         kfree(devlink_rate->name);
718                         kfree(devlink_rate);
719                 }
720         }
721 }
722 EXPORT_SYMBOL_GPL(devl_rate_nodes_destroy);