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
devlink_rate_is_leaf(struct devlink_rate * devlink_rate)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
devlink_rate_is_node(struct devlink_rate * devlink_rate)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 *
devlink_rate_leaf_get_from_info(struct devlink * devlink,struct genl_info * info)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 *
devlink_rate_node_get_by_name(struct devlink * devlink,const char * node_name)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 *
devlink_rate_node_get_from_attrs(struct devlink * devlink,struct nlattr ** attrs)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 *
devlink_rate_node_get_from_info(struct devlink * devlink,struct genl_info * info)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 *
devlink_rate_get_from_info(struct devlink * devlink,struct genl_info * info)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  
devlink_nl_rate_fill(struct sk_buff * msg,struct devlink_rate * devlink_rate,enum devlink_command cmd,u32 portid,u32 seq,int flags,struct netlink_ext_ack * extack)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  
devlink_rate_notify(struct devlink_rate * devlink_rate,enum devlink_command cmd)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 (!devl_is_registered(devlink) || !devlink_nl_notify_need(devlink))
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  	devlink_nl_notify_send(devlink, msg);
163  }
164  
devlink_rates_notify_register(struct devlink * devlink)165  void devlink_rates_notify_register(struct devlink *devlink)
166  {
167  	struct devlink_rate *rate_node;
168  
169  	list_for_each_entry(rate_node, &devlink->rate_list, list)
170  		devlink_rate_notify(rate_node, DEVLINK_CMD_RATE_NEW);
171  }
172  
devlink_rates_notify_unregister(struct devlink * devlink)173  void devlink_rates_notify_unregister(struct devlink *devlink)
174  {
175  	struct devlink_rate *rate_node;
176  
177  	list_for_each_entry_reverse(rate_node, &devlink->rate_list, list)
178  		devlink_rate_notify(rate_node, DEVLINK_CMD_RATE_DEL);
179  }
180  
181  static int
devlink_nl_rate_get_dump_one(struct sk_buff * msg,struct devlink * devlink,struct netlink_callback * cb,int flags)182  devlink_nl_rate_get_dump_one(struct sk_buff *msg, struct devlink *devlink,
183  			     struct netlink_callback *cb, int flags)
184  {
185  	struct devlink_nl_dump_state *state = devlink_dump_state(cb);
186  	struct devlink_rate *devlink_rate;
187  	int idx = 0;
188  	int err = 0;
189  
190  	list_for_each_entry(devlink_rate, &devlink->rate_list, list) {
191  		enum devlink_command cmd = DEVLINK_CMD_RATE_NEW;
192  		u32 id = NETLINK_CB(cb->skb).portid;
193  
194  		if (idx < state->idx) {
195  			idx++;
196  			continue;
197  		}
198  		err = devlink_nl_rate_fill(msg, devlink_rate, cmd, id,
199  					   cb->nlh->nlmsg_seq, flags, NULL);
200  		if (err) {
201  			state->idx = idx;
202  			break;
203  		}
204  		idx++;
205  	}
206  
207  	return err;
208  }
209  
devlink_nl_rate_get_dumpit(struct sk_buff * skb,struct netlink_callback * cb)210  int devlink_nl_rate_get_dumpit(struct sk_buff *skb, struct netlink_callback *cb)
211  {
212  	return devlink_nl_dumpit(skb, cb, devlink_nl_rate_get_dump_one);
213  }
214  
devlink_nl_rate_get_doit(struct sk_buff * skb,struct genl_info * info)215  int devlink_nl_rate_get_doit(struct sk_buff *skb, struct genl_info *info)
216  {
217  	struct devlink *devlink = info->user_ptr[0];
218  	struct devlink_rate *devlink_rate;
219  	struct sk_buff *msg;
220  	int err;
221  
222  	devlink_rate = devlink_rate_get_from_info(devlink, info);
223  	if (IS_ERR(devlink_rate))
224  		return PTR_ERR(devlink_rate);
225  
226  	msg = nlmsg_new(NLMSG_DEFAULT_SIZE, GFP_KERNEL);
227  	if (!msg)
228  		return -ENOMEM;
229  
230  	err = devlink_nl_rate_fill(msg, devlink_rate, DEVLINK_CMD_RATE_NEW,
231  				   info->snd_portid, info->snd_seq, 0,
232  				   info->extack);
233  	if (err) {
234  		nlmsg_free(msg);
235  		return err;
236  	}
237  
238  	return genlmsg_reply(msg, info);
239  }
240  
241  static bool
devlink_rate_is_parent_node(struct devlink_rate * devlink_rate,struct devlink_rate * parent)242  devlink_rate_is_parent_node(struct devlink_rate *devlink_rate,
243  			    struct devlink_rate *parent)
244  {
245  	while (parent) {
246  		if (parent == devlink_rate)
247  			return true;
248  		parent = parent->parent;
249  	}
250  	return false;
251  }
252  
253  static int
devlink_nl_rate_parent_node_set(struct devlink_rate * devlink_rate,struct genl_info * info,struct nlattr * nla_parent)254  devlink_nl_rate_parent_node_set(struct devlink_rate *devlink_rate,
255  				struct genl_info *info,
256  				struct nlattr *nla_parent)
257  {
258  	struct devlink *devlink = devlink_rate->devlink;
259  	const char *parent_name = nla_data(nla_parent);
260  	const struct devlink_ops *ops = devlink->ops;
261  	size_t len = strlen(parent_name);
262  	struct devlink_rate *parent;
263  	int err = -EOPNOTSUPP;
264  
265  	parent = devlink_rate->parent;
266  
267  	if (parent && !len) {
268  		if (devlink_rate_is_leaf(devlink_rate))
269  			err = ops->rate_leaf_parent_set(devlink_rate, NULL,
270  							devlink_rate->priv, NULL,
271  							info->extack);
272  		else if (devlink_rate_is_node(devlink_rate))
273  			err = ops->rate_node_parent_set(devlink_rate, NULL,
274  							devlink_rate->priv, NULL,
275  							info->extack);
276  		if (err)
277  			return err;
278  
279  		refcount_dec(&parent->refcnt);
280  		devlink_rate->parent = NULL;
281  	} else if (len) {
282  		parent = devlink_rate_node_get_by_name(devlink, parent_name);
283  		if (IS_ERR(parent))
284  			return -ENODEV;
285  
286  		if (parent == devlink_rate) {
287  			NL_SET_ERR_MSG(info->extack, "Parent to self is not allowed");
288  			return -EINVAL;
289  		}
290  
291  		if (devlink_rate_is_node(devlink_rate) &&
292  		    devlink_rate_is_parent_node(devlink_rate, parent->parent)) {
293  			NL_SET_ERR_MSG(info->extack, "Node is already a parent of parent node.");
294  			return -EEXIST;
295  		}
296  
297  		if (devlink_rate_is_leaf(devlink_rate))
298  			err = ops->rate_leaf_parent_set(devlink_rate, parent,
299  							devlink_rate->priv, parent->priv,
300  							info->extack);
301  		else if (devlink_rate_is_node(devlink_rate))
302  			err = ops->rate_node_parent_set(devlink_rate, parent,
303  							devlink_rate->priv, parent->priv,
304  							info->extack);
305  		if (err)
306  			return err;
307  
308  		if (devlink_rate->parent)
309  			/* we're reassigning to other parent in this case */
310  			refcount_dec(&devlink_rate->parent->refcnt);
311  
312  		refcount_inc(&parent->refcnt);
313  		devlink_rate->parent = parent;
314  	}
315  
316  	return 0;
317  }
318  
devlink_nl_rate_set(struct devlink_rate * devlink_rate,const struct devlink_ops * ops,struct genl_info * info)319  static int devlink_nl_rate_set(struct devlink_rate *devlink_rate,
320  			       const struct devlink_ops *ops,
321  			       struct genl_info *info)
322  {
323  	struct nlattr *nla_parent, **attrs = info->attrs;
324  	int err = -EOPNOTSUPP;
325  	u32 priority;
326  	u32 weight;
327  	u64 rate;
328  
329  	if (attrs[DEVLINK_ATTR_RATE_TX_SHARE]) {
330  		rate = nla_get_u64(attrs[DEVLINK_ATTR_RATE_TX_SHARE]);
331  		if (devlink_rate_is_leaf(devlink_rate))
332  			err = ops->rate_leaf_tx_share_set(devlink_rate, devlink_rate->priv,
333  							  rate, info->extack);
334  		else if (devlink_rate_is_node(devlink_rate))
335  			err = ops->rate_node_tx_share_set(devlink_rate, devlink_rate->priv,
336  							  rate, info->extack);
337  		if (err)
338  			return err;
339  		devlink_rate->tx_share = rate;
340  	}
341  
342  	if (attrs[DEVLINK_ATTR_RATE_TX_MAX]) {
343  		rate = nla_get_u64(attrs[DEVLINK_ATTR_RATE_TX_MAX]);
344  		if (devlink_rate_is_leaf(devlink_rate))
345  			err = ops->rate_leaf_tx_max_set(devlink_rate, devlink_rate->priv,
346  							rate, info->extack);
347  		else if (devlink_rate_is_node(devlink_rate))
348  			err = ops->rate_node_tx_max_set(devlink_rate, devlink_rate->priv,
349  							rate, info->extack);
350  		if (err)
351  			return err;
352  		devlink_rate->tx_max = rate;
353  	}
354  
355  	if (attrs[DEVLINK_ATTR_RATE_TX_PRIORITY]) {
356  		priority = nla_get_u32(attrs[DEVLINK_ATTR_RATE_TX_PRIORITY]);
357  		if (devlink_rate_is_leaf(devlink_rate))
358  			err = ops->rate_leaf_tx_priority_set(devlink_rate, devlink_rate->priv,
359  							     priority, info->extack);
360  		else if (devlink_rate_is_node(devlink_rate))
361  			err = ops->rate_node_tx_priority_set(devlink_rate, devlink_rate->priv,
362  							     priority, info->extack);
363  
364  		if (err)
365  			return err;
366  		devlink_rate->tx_priority = priority;
367  	}
368  
369  	if (attrs[DEVLINK_ATTR_RATE_TX_WEIGHT]) {
370  		weight = nla_get_u32(attrs[DEVLINK_ATTR_RATE_TX_WEIGHT]);
371  		if (devlink_rate_is_leaf(devlink_rate))
372  			err = ops->rate_leaf_tx_weight_set(devlink_rate, devlink_rate->priv,
373  							   weight, info->extack);
374  		else if (devlink_rate_is_node(devlink_rate))
375  			err = ops->rate_node_tx_weight_set(devlink_rate, devlink_rate->priv,
376  							   weight, info->extack);
377  
378  		if (err)
379  			return err;
380  		devlink_rate->tx_weight = weight;
381  	}
382  
383  	nla_parent = attrs[DEVLINK_ATTR_RATE_PARENT_NODE_NAME];
384  	if (nla_parent) {
385  		err = devlink_nl_rate_parent_node_set(devlink_rate, info,
386  						      nla_parent);
387  		if (err)
388  			return err;
389  	}
390  
391  	return 0;
392  }
393  
devlink_rate_set_ops_supported(const struct devlink_ops * ops,struct genl_info * info,enum devlink_rate_type type)394  static bool devlink_rate_set_ops_supported(const struct devlink_ops *ops,
395  					   struct genl_info *info,
396  					   enum devlink_rate_type type)
397  {
398  	struct nlattr **attrs = info->attrs;
399  
400  	if (type == DEVLINK_RATE_TYPE_LEAF) {
401  		if (attrs[DEVLINK_ATTR_RATE_TX_SHARE] && !ops->rate_leaf_tx_share_set) {
402  			NL_SET_ERR_MSG(info->extack, "TX share set isn't supported for the leafs");
403  			return false;
404  		}
405  		if (attrs[DEVLINK_ATTR_RATE_TX_MAX] && !ops->rate_leaf_tx_max_set) {
406  			NL_SET_ERR_MSG(info->extack, "TX max set isn't supported for the leafs");
407  			return false;
408  		}
409  		if (attrs[DEVLINK_ATTR_RATE_PARENT_NODE_NAME] &&
410  		    !ops->rate_leaf_parent_set) {
411  			NL_SET_ERR_MSG(info->extack, "Parent set isn't supported for the leafs");
412  			return false;
413  		}
414  		if (attrs[DEVLINK_ATTR_RATE_TX_PRIORITY] && !ops->rate_leaf_tx_priority_set) {
415  			NL_SET_ERR_MSG_ATTR(info->extack,
416  					    attrs[DEVLINK_ATTR_RATE_TX_PRIORITY],
417  					    "TX priority set isn't supported for the leafs");
418  			return false;
419  		}
420  		if (attrs[DEVLINK_ATTR_RATE_TX_WEIGHT] && !ops->rate_leaf_tx_weight_set) {
421  			NL_SET_ERR_MSG_ATTR(info->extack,
422  					    attrs[DEVLINK_ATTR_RATE_TX_WEIGHT],
423  					    "TX weight set isn't supported for the leafs");
424  			return false;
425  		}
426  	} else if (type == DEVLINK_RATE_TYPE_NODE) {
427  		if (attrs[DEVLINK_ATTR_RATE_TX_SHARE] && !ops->rate_node_tx_share_set) {
428  			NL_SET_ERR_MSG(info->extack, "TX share set isn't supported for the nodes");
429  			return false;
430  		}
431  		if (attrs[DEVLINK_ATTR_RATE_TX_MAX] && !ops->rate_node_tx_max_set) {
432  			NL_SET_ERR_MSG(info->extack, "TX max set isn't supported for the nodes");
433  			return false;
434  		}
435  		if (attrs[DEVLINK_ATTR_RATE_PARENT_NODE_NAME] &&
436  		    !ops->rate_node_parent_set) {
437  			NL_SET_ERR_MSG(info->extack, "Parent set isn't supported for the nodes");
438  			return false;
439  		}
440  		if (attrs[DEVLINK_ATTR_RATE_TX_PRIORITY] && !ops->rate_node_tx_priority_set) {
441  			NL_SET_ERR_MSG_ATTR(info->extack,
442  					    attrs[DEVLINK_ATTR_RATE_TX_PRIORITY],
443  					    "TX priority set isn't supported for the nodes");
444  			return false;
445  		}
446  		if (attrs[DEVLINK_ATTR_RATE_TX_WEIGHT] && !ops->rate_node_tx_weight_set) {
447  			NL_SET_ERR_MSG_ATTR(info->extack,
448  					    attrs[DEVLINK_ATTR_RATE_TX_WEIGHT],
449  					    "TX weight set isn't supported for the nodes");
450  			return false;
451  		}
452  	} else {
453  		WARN(1, "Unknown type of rate object");
454  		return false;
455  	}
456  
457  	return true;
458  }
459  
devlink_nl_rate_set_doit(struct sk_buff * skb,struct genl_info * info)460  int devlink_nl_rate_set_doit(struct sk_buff *skb, struct genl_info *info)
461  {
462  	struct devlink *devlink = info->user_ptr[0];
463  	struct devlink_rate *devlink_rate;
464  	const struct devlink_ops *ops;
465  	int err;
466  
467  	devlink_rate = devlink_rate_get_from_info(devlink, info);
468  	if (IS_ERR(devlink_rate))
469  		return PTR_ERR(devlink_rate);
470  
471  	ops = devlink->ops;
472  	if (!ops || !devlink_rate_set_ops_supported(ops, info, devlink_rate->type))
473  		return -EOPNOTSUPP;
474  
475  	err = devlink_nl_rate_set(devlink_rate, ops, info);
476  
477  	if (!err)
478  		devlink_rate_notify(devlink_rate, DEVLINK_CMD_RATE_NEW);
479  	return err;
480  }
481  
devlink_nl_rate_new_doit(struct sk_buff * skb,struct genl_info * info)482  int devlink_nl_rate_new_doit(struct sk_buff *skb, struct genl_info *info)
483  {
484  	struct devlink *devlink = info->user_ptr[0];
485  	struct devlink_rate *rate_node;
486  	const struct devlink_ops *ops;
487  	int err;
488  
489  	ops = devlink->ops;
490  	if (!ops || !ops->rate_node_new || !ops->rate_node_del) {
491  		NL_SET_ERR_MSG(info->extack, "Rate nodes aren't supported");
492  		return -EOPNOTSUPP;
493  	}
494  
495  	if (!devlink_rate_set_ops_supported(ops, info, DEVLINK_RATE_TYPE_NODE))
496  		return -EOPNOTSUPP;
497  
498  	rate_node = devlink_rate_node_get_from_attrs(devlink, info->attrs);
499  	if (!IS_ERR(rate_node))
500  		return -EEXIST;
501  	else if (rate_node == ERR_PTR(-EINVAL))
502  		return -EINVAL;
503  
504  	rate_node = kzalloc(sizeof(*rate_node), GFP_KERNEL);
505  	if (!rate_node)
506  		return -ENOMEM;
507  
508  	rate_node->devlink = devlink;
509  	rate_node->type = DEVLINK_RATE_TYPE_NODE;
510  	rate_node->name = nla_strdup(info->attrs[DEVLINK_ATTR_RATE_NODE_NAME], GFP_KERNEL);
511  	if (!rate_node->name) {
512  		err = -ENOMEM;
513  		goto err_strdup;
514  	}
515  
516  	err = ops->rate_node_new(rate_node, &rate_node->priv, info->extack);
517  	if (err)
518  		goto err_node_new;
519  
520  	err = devlink_nl_rate_set(rate_node, ops, info);
521  	if (err)
522  		goto err_rate_set;
523  
524  	refcount_set(&rate_node->refcnt, 1);
525  	list_add(&rate_node->list, &devlink->rate_list);
526  	devlink_rate_notify(rate_node, DEVLINK_CMD_RATE_NEW);
527  	return 0;
528  
529  err_rate_set:
530  	ops->rate_node_del(rate_node, rate_node->priv, info->extack);
531  err_node_new:
532  	kfree(rate_node->name);
533  err_strdup:
534  	kfree(rate_node);
535  	return err;
536  }
537  
devlink_nl_rate_del_doit(struct sk_buff * skb,struct genl_info * info)538  int devlink_nl_rate_del_doit(struct sk_buff *skb, struct genl_info *info)
539  {
540  	struct devlink *devlink = info->user_ptr[0];
541  	struct devlink_rate *rate_node;
542  	int err;
543  
544  	rate_node = devlink_rate_node_get_from_info(devlink, info);
545  	if (IS_ERR(rate_node))
546  		return PTR_ERR(rate_node);
547  
548  	if (refcount_read(&rate_node->refcnt) > 1) {
549  		NL_SET_ERR_MSG(info->extack, "Node has children. Cannot delete node.");
550  		return -EBUSY;
551  	}
552  
553  	devlink_rate_notify(rate_node, DEVLINK_CMD_RATE_DEL);
554  	err = devlink->ops->rate_node_del(rate_node, rate_node->priv,
555  					  info->extack);
556  	if (rate_node->parent)
557  		refcount_dec(&rate_node->parent->refcnt);
558  	list_del(&rate_node->list);
559  	kfree(rate_node->name);
560  	kfree(rate_node);
561  	return err;
562  }
563  
devlink_rate_nodes_check(struct devlink * devlink,u16 mode,struct netlink_ext_ack * extack)564  int devlink_rate_nodes_check(struct devlink *devlink, u16 mode,
565  			     struct netlink_ext_ack *extack)
566  {
567  	struct devlink_rate *devlink_rate;
568  
569  	list_for_each_entry(devlink_rate, &devlink->rate_list, list)
570  		if (devlink_rate_is_node(devlink_rate)) {
571  			NL_SET_ERR_MSG(extack, "Rate node(s) exists.");
572  			return -EBUSY;
573  		}
574  	return 0;
575  }
576  
577  /**
578   * devl_rate_node_create - create devlink rate node
579   * @devlink: devlink instance
580   * @priv: driver private data
581   * @node_name: name of the resulting node
582   * @parent: parent devlink_rate struct
583   *
584   * Create devlink rate object of type node
585   */
586  struct devlink_rate *
devl_rate_node_create(struct devlink * devlink,void * priv,char * node_name,struct devlink_rate * parent)587  devl_rate_node_create(struct devlink *devlink, void *priv, char *node_name,
588  		      struct devlink_rate *parent)
589  {
590  	struct devlink_rate *rate_node;
591  
592  	rate_node = devlink_rate_node_get_by_name(devlink, node_name);
593  	if (!IS_ERR(rate_node))
594  		return ERR_PTR(-EEXIST);
595  
596  	rate_node = kzalloc(sizeof(*rate_node), GFP_KERNEL);
597  	if (!rate_node)
598  		return ERR_PTR(-ENOMEM);
599  
600  	if (parent) {
601  		rate_node->parent = parent;
602  		refcount_inc(&rate_node->parent->refcnt);
603  	}
604  
605  	rate_node->type = DEVLINK_RATE_TYPE_NODE;
606  	rate_node->devlink = devlink;
607  	rate_node->priv = priv;
608  
609  	rate_node->name = kstrdup(node_name, GFP_KERNEL);
610  	if (!rate_node->name) {
611  		kfree(rate_node);
612  		return ERR_PTR(-ENOMEM);
613  	}
614  
615  	refcount_set(&rate_node->refcnt, 1);
616  	list_add(&rate_node->list, &devlink->rate_list);
617  	devlink_rate_notify(rate_node, DEVLINK_CMD_RATE_NEW);
618  	return rate_node;
619  }
620  EXPORT_SYMBOL_GPL(devl_rate_node_create);
621  
622  /**
623   * devl_rate_leaf_create - create devlink rate leaf
624   * @devlink_port: devlink port object to create rate object on
625   * @priv: driver private data
626   * @parent: parent devlink_rate struct
627   *
628   * Create devlink rate object of type leaf on provided @devlink_port.
629   */
devl_rate_leaf_create(struct devlink_port * devlink_port,void * priv,struct devlink_rate * parent)630  int devl_rate_leaf_create(struct devlink_port *devlink_port, void *priv,
631  			  struct devlink_rate *parent)
632  {
633  	struct devlink *devlink = devlink_port->devlink;
634  	struct devlink_rate *devlink_rate;
635  
636  	devl_assert_locked(devlink_port->devlink);
637  
638  	if (WARN_ON(devlink_port->devlink_rate))
639  		return -EBUSY;
640  
641  	devlink_rate = kzalloc(sizeof(*devlink_rate), GFP_KERNEL);
642  	if (!devlink_rate)
643  		return -ENOMEM;
644  
645  	if (parent) {
646  		devlink_rate->parent = parent;
647  		refcount_inc(&devlink_rate->parent->refcnt);
648  	}
649  
650  	devlink_rate->type = DEVLINK_RATE_TYPE_LEAF;
651  	devlink_rate->devlink = devlink;
652  	devlink_rate->devlink_port = devlink_port;
653  	devlink_rate->priv = priv;
654  	list_add_tail(&devlink_rate->list, &devlink->rate_list);
655  	devlink_port->devlink_rate = devlink_rate;
656  	devlink_rate_notify(devlink_rate, DEVLINK_CMD_RATE_NEW);
657  
658  	return 0;
659  }
660  EXPORT_SYMBOL_GPL(devl_rate_leaf_create);
661  
662  /**
663   * devl_rate_leaf_destroy - destroy devlink rate leaf
664   *
665   * @devlink_port: devlink port linked to the rate object
666   *
667   * Destroy the devlink rate object of type leaf on provided @devlink_port.
668   */
devl_rate_leaf_destroy(struct devlink_port * devlink_port)669  void devl_rate_leaf_destroy(struct devlink_port *devlink_port)
670  {
671  	struct devlink_rate *devlink_rate = devlink_port->devlink_rate;
672  
673  	devl_assert_locked(devlink_port->devlink);
674  	if (!devlink_rate)
675  		return;
676  
677  	devlink_rate_notify(devlink_rate, DEVLINK_CMD_RATE_DEL);
678  	if (devlink_rate->parent)
679  		refcount_dec(&devlink_rate->parent->refcnt);
680  	list_del(&devlink_rate->list);
681  	devlink_port->devlink_rate = NULL;
682  	kfree(devlink_rate);
683  }
684  EXPORT_SYMBOL_GPL(devl_rate_leaf_destroy);
685  
686  /**
687   * devl_rate_nodes_destroy - destroy all devlink rate nodes on device
688   * @devlink: devlink instance
689   *
690   * Unset parent for all rate objects and destroy all rate nodes
691   * on specified device.
692   */
devl_rate_nodes_destroy(struct devlink * devlink)693  void devl_rate_nodes_destroy(struct devlink *devlink)
694  {
695  	static struct devlink_rate *devlink_rate, *tmp;
696  	const struct devlink_ops *ops = devlink->ops;
697  
698  	devl_assert_locked(devlink);
699  
700  	list_for_each_entry(devlink_rate, &devlink->rate_list, list) {
701  		if (!devlink_rate->parent)
702  			continue;
703  
704  		refcount_dec(&devlink_rate->parent->refcnt);
705  		if (devlink_rate_is_leaf(devlink_rate))
706  			ops->rate_leaf_parent_set(devlink_rate, NULL, devlink_rate->priv,
707  						  NULL, NULL);
708  		else if (devlink_rate_is_node(devlink_rate))
709  			ops->rate_node_parent_set(devlink_rate, NULL, devlink_rate->priv,
710  						  NULL, NULL);
711  	}
712  	list_for_each_entry_safe(devlink_rate, tmp, &devlink->rate_list, list) {
713  		if (devlink_rate_is_node(devlink_rate)) {
714  			ops->rate_node_del(devlink_rate, devlink_rate->priv, NULL);
715  			list_del(&devlink_rate->list);
716  			kfree(devlink_rate->name);
717  			kfree(devlink_rate);
718  		}
719  	}
720  }
721  EXPORT_SYMBOL_GPL(devl_rate_nodes_destroy);
722