17cc7194eSJiri Pirko // SPDX-License-Identifier: GPL-2.0-or-later
27cc7194eSJiri Pirko /*
37cc7194eSJiri Pirko * Copyright (c) 2016 Mellanox Technologies. All rights reserved.
47cc7194eSJiri Pirko * Copyright (c) 2016 Jiri Pirko <jiri@mellanox.com>
57cc7194eSJiri Pirko */
67cc7194eSJiri Pirko
77cc7194eSJiri Pirko #include "devl_internal.h"
87cc7194eSJiri Pirko
97cc7194eSJiri Pirko static inline bool
devlink_rate_is_leaf(struct devlink_rate * devlink_rate)107cc7194eSJiri Pirko devlink_rate_is_leaf(struct devlink_rate *devlink_rate)
117cc7194eSJiri Pirko {
127cc7194eSJiri Pirko return devlink_rate->type == DEVLINK_RATE_TYPE_LEAF;
137cc7194eSJiri Pirko }
147cc7194eSJiri Pirko
157cc7194eSJiri Pirko static inline bool
devlink_rate_is_node(struct devlink_rate * devlink_rate)167cc7194eSJiri Pirko devlink_rate_is_node(struct devlink_rate *devlink_rate)
177cc7194eSJiri Pirko {
187cc7194eSJiri Pirko return devlink_rate->type == DEVLINK_RATE_TYPE_NODE;
197cc7194eSJiri Pirko }
207cc7194eSJiri Pirko
217cc7194eSJiri Pirko static struct devlink_rate *
devlink_rate_leaf_get_from_info(struct devlink * devlink,struct genl_info * info)227cc7194eSJiri Pirko devlink_rate_leaf_get_from_info(struct devlink *devlink, struct genl_info *info)
237cc7194eSJiri Pirko {
247cc7194eSJiri Pirko struct devlink_rate *devlink_rate;
257cc7194eSJiri Pirko struct devlink_port *devlink_port;
267cc7194eSJiri Pirko
277cc7194eSJiri Pirko devlink_port = devlink_port_get_from_attrs(devlink, info->attrs);
287cc7194eSJiri Pirko if (IS_ERR(devlink_port))
297cc7194eSJiri Pirko return ERR_CAST(devlink_port);
307cc7194eSJiri Pirko devlink_rate = devlink_port->devlink_rate;
317cc7194eSJiri Pirko return devlink_rate ?: ERR_PTR(-ENODEV);
327cc7194eSJiri Pirko }
337cc7194eSJiri Pirko
347cc7194eSJiri Pirko static struct devlink_rate *
devlink_rate_node_get_by_name(struct devlink * devlink,const char * node_name)357cc7194eSJiri Pirko devlink_rate_node_get_by_name(struct devlink *devlink, const char *node_name)
367cc7194eSJiri Pirko {
377cc7194eSJiri Pirko static struct devlink_rate *devlink_rate;
387cc7194eSJiri Pirko
397cc7194eSJiri Pirko list_for_each_entry(devlink_rate, &devlink->rate_list, list) {
407cc7194eSJiri Pirko if (devlink_rate_is_node(devlink_rate) &&
417cc7194eSJiri Pirko !strcmp(node_name, devlink_rate->name))
427cc7194eSJiri Pirko return devlink_rate;
437cc7194eSJiri Pirko }
447cc7194eSJiri Pirko return ERR_PTR(-ENODEV);
457cc7194eSJiri Pirko }
467cc7194eSJiri Pirko
477cc7194eSJiri Pirko static struct devlink_rate *
devlink_rate_node_get_from_attrs(struct devlink * devlink,struct nlattr ** attrs)487cc7194eSJiri Pirko devlink_rate_node_get_from_attrs(struct devlink *devlink, struct nlattr **attrs)
497cc7194eSJiri Pirko {
507cc7194eSJiri Pirko const char *rate_node_name;
517cc7194eSJiri Pirko size_t len;
527cc7194eSJiri Pirko
537cc7194eSJiri Pirko if (!attrs[DEVLINK_ATTR_RATE_NODE_NAME])
547cc7194eSJiri Pirko return ERR_PTR(-EINVAL);
557cc7194eSJiri Pirko rate_node_name = nla_data(attrs[DEVLINK_ATTR_RATE_NODE_NAME]);
567cc7194eSJiri Pirko len = strlen(rate_node_name);
577cc7194eSJiri Pirko /* Name cannot be empty or decimal number */
587cc7194eSJiri Pirko if (!len || strspn(rate_node_name, "0123456789") == len)
597cc7194eSJiri Pirko return ERR_PTR(-EINVAL);
607cc7194eSJiri Pirko
617cc7194eSJiri Pirko return devlink_rate_node_get_by_name(devlink, rate_node_name);
627cc7194eSJiri Pirko }
637cc7194eSJiri Pirko
647cc7194eSJiri Pirko static struct devlink_rate *
devlink_rate_node_get_from_info(struct devlink * devlink,struct genl_info * info)657cc7194eSJiri Pirko devlink_rate_node_get_from_info(struct devlink *devlink, struct genl_info *info)
667cc7194eSJiri Pirko {
677cc7194eSJiri Pirko return devlink_rate_node_get_from_attrs(devlink, info->attrs);
687cc7194eSJiri Pirko }
697cc7194eSJiri Pirko
707cc7194eSJiri Pirko static struct devlink_rate *
devlink_rate_get_from_info(struct devlink * devlink,struct genl_info * info)717cc7194eSJiri Pirko devlink_rate_get_from_info(struct devlink *devlink, struct genl_info *info)
727cc7194eSJiri Pirko {
737cc7194eSJiri Pirko struct nlattr **attrs = info->attrs;
747cc7194eSJiri Pirko
757cc7194eSJiri Pirko if (attrs[DEVLINK_ATTR_PORT_INDEX])
767cc7194eSJiri Pirko return devlink_rate_leaf_get_from_info(devlink, info);
777cc7194eSJiri Pirko else if (attrs[DEVLINK_ATTR_RATE_NODE_NAME])
787cc7194eSJiri Pirko return devlink_rate_node_get_from_info(devlink, info);
797cc7194eSJiri Pirko else
807cc7194eSJiri Pirko return ERR_PTR(-EINVAL);
817cc7194eSJiri Pirko }
827cc7194eSJiri Pirko
devlink_rate_put_tc_bws(struct sk_buff * msg,u32 * tc_bw)83566e8f10SCarolina Jubran static int devlink_rate_put_tc_bws(struct sk_buff *msg, u32 *tc_bw)
84566e8f10SCarolina Jubran {
85566e8f10SCarolina Jubran struct nlattr *nla_tc_bw;
86566e8f10SCarolina Jubran int i;
87566e8f10SCarolina Jubran
88566e8f10SCarolina Jubran for (i = 0; i < DEVLINK_RATE_TCS_MAX; i++) {
89566e8f10SCarolina Jubran nla_tc_bw = nla_nest_start(msg, DEVLINK_ATTR_RATE_TC_BWS);
90566e8f10SCarolina Jubran if (!nla_tc_bw)
91566e8f10SCarolina Jubran return -EMSGSIZE;
92566e8f10SCarolina Jubran
93*1bbdb81aSCarolina Jubran if (nla_put_u8(msg, DEVLINK_RATE_TC_ATTR_INDEX, i) ||
94*1bbdb81aSCarolina Jubran nla_put_u32(msg, DEVLINK_RATE_TC_ATTR_BW, tc_bw[i]))
95566e8f10SCarolina Jubran goto nla_put_failure;
96566e8f10SCarolina Jubran
97566e8f10SCarolina Jubran nla_nest_end(msg, nla_tc_bw);
98566e8f10SCarolina Jubran }
99566e8f10SCarolina Jubran return 0;
100566e8f10SCarolina Jubran
101566e8f10SCarolina Jubran nla_put_failure:
102566e8f10SCarolina Jubran nla_nest_cancel(msg, nla_tc_bw);
103566e8f10SCarolina Jubran return -EMSGSIZE;
104566e8f10SCarolina Jubran }
105566e8f10SCarolina Jubran
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)1067cc7194eSJiri Pirko static int devlink_nl_rate_fill(struct sk_buff *msg,
1077cc7194eSJiri Pirko struct devlink_rate *devlink_rate,
1087cc7194eSJiri Pirko enum devlink_command cmd, u32 portid, u32 seq,
1097cc7194eSJiri Pirko int flags, struct netlink_ext_ack *extack)
1107cc7194eSJiri Pirko {
1117cc7194eSJiri Pirko struct devlink *devlink = devlink_rate->devlink;
1127cc7194eSJiri Pirko void *hdr;
1137cc7194eSJiri Pirko
1147cc7194eSJiri Pirko hdr = genlmsg_put(msg, portid, seq, &devlink_nl_family, flags, cmd);
1157cc7194eSJiri Pirko if (!hdr)
1167cc7194eSJiri Pirko return -EMSGSIZE;
1177cc7194eSJiri Pirko
1187cc7194eSJiri Pirko if (devlink_nl_put_handle(msg, devlink))
1197cc7194eSJiri Pirko goto nla_put_failure;
1207cc7194eSJiri Pirko
1217cc7194eSJiri Pirko if (nla_put_u16(msg, DEVLINK_ATTR_RATE_TYPE, devlink_rate->type))
1227cc7194eSJiri Pirko goto nla_put_failure;
1237cc7194eSJiri Pirko
1247cc7194eSJiri Pirko if (devlink_rate_is_leaf(devlink_rate)) {
1257cc7194eSJiri Pirko if (nla_put_u32(msg, DEVLINK_ATTR_PORT_INDEX,
1267cc7194eSJiri Pirko devlink_rate->devlink_port->index))
1277cc7194eSJiri Pirko goto nla_put_failure;
1287cc7194eSJiri Pirko } else if (devlink_rate_is_node(devlink_rate)) {
1297cc7194eSJiri Pirko if (nla_put_string(msg, DEVLINK_ATTR_RATE_NODE_NAME,
1307cc7194eSJiri Pirko devlink_rate->name))
1317cc7194eSJiri Pirko goto nla_put_failure;
1327cc7194eSJiri Pirko }
1337cc7194eSJiri Pirko
134a788acf1SPrzemek Kitszel if (devlink_nl_put_u64(msg, DEVLINK_ATTR_RATE_TX_SHARE,
135a788acf1SPrzemek Kitszel devlink_rate->tx_share))
1367cc7194eSJiri Pirko goto nla_put_failure;
1377cc7194eSJiri Pirko
138a788acf1SPrzemek Kitszel if (devlink_nl_put_u64(msg, DEVLINK_ATTR_RATE_TX_MAX,
139a788acf1SPrzemek Kitszel devlink_rate->tx_max))
1407cc7194eSJiri Pirko goto nla_put_failure;
1417cc7194eSJiri Pirko
1427cc7194eSJiri Pirko if (nla_put_u32(msg, DEVLINK_ATTR_RATE_TX_PRIORITY,
1437cc7194eSJiri Pirko devlink_rate->tx_priority))
1447cc7194eSJiri Pirko goto nla_put_failure;
1457cc7194eSJiri Pirko
1467cc7194eSJiri Pirko if (nla_put_u32(msg, DEVLINK_ATTR_RATE_TX_WEIGHT,
1477cc7194eSJiri Pirko devlink_rate->tx_weight))
1487cc7194eSJiri Pirko goto nla_put_failure;
1497cc7194eSJiri Pirko
1507cc7194eSJiri Pirko if (devlink_rate->parent)
1517cc7194eSJiri Pirko if (nla_put_string(msg, DEVLINK_ATTR_RATE_PARENT_NODE_NAME,
1527cc7194eSJiri Pirko devlink_rate->parent->name))
1537cc7194eSJiri Pirko goto nla_put_failure;
1547cc7194eSJiri Pirko
155566e8f10SCarolina Jubran if (devlink_rate_put_tc_bws(msg, devlink_rate->tc_bw))
156566e8f10SCarolina Jubran goto nla_put_failure;
157566e8f10SCarolina Jubran
1587cc7194eSJiri Pirko genlmsg_end(msg, hdr);
1597cc7194eSJiri Pirko return 0;
1607cc7194eSJiri Pirko
1617cc7194eSJiri Pirko nla_put_failure:
1627cc7194eSJiri Pirko genlmsg_cancel(msg, hdr);
1637cc7194eSJiri Pirko return -EMSGSIZE;
1647cc7194eSJiri Pirko }
1657cc7194eSJiri Pirko
devlink_rate_notify(struct devlink_rate * devlink_rate,enum devlink_command cmd)1667cc7194eSJiri Pirko static void devlink_rate_notify(struct devlink_rate *devlink_rate,
1677cc7194eSJiri Pirko enum devlink_command cmd)
1687cc7194eSJiri Pirko {
1697cc7194eSJiri Pirko struct devlink *devlink = devlink_rate->devlink;
1707cc7194eSJiri Pirko struct sk_buff *msg;
1717cc7194eSJiri Pirko int err;
1727cc7194eSJiri Pirko
1737cc7194eSJiri Pirko WARN_ON(cmd != DEVLINK_CMD_RATE_NEW && cmd != DEVLINK_CMD_RATE_DEL);
1747cc7194eSJiri Pirko
175cddbff47SJiri Pirko if (!devl_is_registered(devlink) || !devlink_nl_notify_need(devlink))
1767cc7194eSJiri Pirko return;
1777cc7194eSJiri Pirko
1787cc7194eSJiri Pirko msg = nlmsg_new(NLMSG_DEFAULT_SIZE, GFP_KERNEL);
1797cc7194eSJiri Pirko if (!msg)
1807cc7194eSJiri Pirko return;
1817cc7194eSJiri Pirko
1827cc7194eSJiri Pirko err = devlink_nl_rate_fill(msg, devlink_rate, cmd, 0, 0, 0, NULL);
1837cc7194eSJiri Pirko if (err) {
1847cc7194eSJiri Pirko nlmsg_free(msg);
1857cc7194eSJiri Pirko return;
1867cc7194eSJiri Pirko }
1877cc7194eSJiri Pirko
1885648de0bSJiri Pirko devlink_nl_notify_send(devlink, msg);
1897cc7194eSJiri Pirko }
1907cc7194eSJiri Pirko
devlink_rates_notify_register(struct devlink * devlink)1917cc7194eSJiri Pirko void devlink_rates_notify_register(struct devlink *devlink)
1927cc7194eSJiri Pirko {
1937cc7194eSJiri Pirko struct devlink_rate *rate_node;
1947cc7194eSJiri Pirko
1957cc7194eSJiri Pirko list_for_each_entry(rate_node, &devlink->rate_list, list)
1967cc7194eSJiri Pirko devlink_rate_notify(rate_node, DEVLINK_CMD_RATE_NEW);
1977cc7194eSJiri Pirko }
1987cc7194eSJiri Pirko
devlink_rates_notify_unregister(struct devlink * devlink)1997cc7194eSJiri Pirko void devlink_rates_notify_unregister(struct devlink *devlink)
2007cc7194eSJiri Pirko {
2017cc7194eSJiri Pirko struct devlink_rate *rate_node;
2027cc7194eSJiri Pirko
2037cc7194eSJiri Pirko list_for_each_entry_reverse(rate_node, &devlink->rate_list, list)
2047cc7194eSJiri Pirko devlink_rate_notify(rate_node, DEVLINK_CMD_RATE_DEL);
2057cc7194eSJiri Pirko }
2067cc7194eSJiri Pirko
2077cc7194eSJiri Pirko static int
devlink_nl_rate_get_dump_one(struct sk_buff * msg,struct devlink * devlink,struct netlink_callback * cb,int flags)2087cc7194eSJiri Pirko devlink_nl_rate_get_dump_one(struct sk_buff *msg, struct devlink *devlink,
2097cc7194eSJiri Pirko struct netlink_callback *cb, int flags)
2107cc7194eSJiri Pirko {
2117cc7194eSJiri Pirko struct devlink_nl_dump_state *state = devlink_dump_state(cb);
2127cc7194eSJiri Pirko struct devlink_rate *devlink_rate;
2137cc7194eSJiri Pirko int idx = 0;
2147cc7194eSJiri Pirko int err = 0;
2157cc7194eSJiri Pirko
2167cc7194eSJiri Pirko list_for_each_entry(devlink_rate, &devlink->rate_list, list) {
2177cc7194eSJiri Pirko enum devlink_command cmd = DEVLINK_CMD_RATE_NEW;
2187cc7194eSJiri Pirko u32 id = NETLINK_CB(cb->skb).portid;
2197cc7194eSJiri Pirko
2207cc7194eSJiri Pirko if (idx < state->idx) {
2217cc7194eSJiri Pirko idx++;
2227cc7194eSJiri Pirko continue;
2237cc7194eSJiri Pirko }
2247cc7194eSJiri Pirko err = devlink_nl_rate_fill(msg, devlink_rate, cmd, id,
2257cc7194eSJiri Pirko cb->nlh->nlmsg_seq, flags, NULL);
2267cc7194eSJiri Pirko if (err) {
2277cc7194eSJiri Pirko state->idx = idx;
2287cc7194eSJiri Pirko break;
2297cc7194eSJiri Pirko }
2307cc7194eSJiri Pirko idx++;
2317cc7194eSJiri Pirko }
2327cc7194eSJiri Pirko
2337cc7194eSJiri Pirko return err;
2347cc7194eSJiri Pirko }
2357cc7194eSJiri Pirko
devlink_nl_rate_get_dumpit(struct sk_buff * skb,struct netlink_callback * cb)2367cc7194eSJiri Pirko int devlink_nl_rate_get_dumpit(struct sk_buff *skb, struct netlink_callback *cb)
2377cc7194eSJiri Pirko {
2387cc7194eSJiri Pirko return devlink_nl_dumpit(skb, cb, devlink_nl_rate_get_dump_one);
2397cc7194eSJiri Pirko }
2407cc7194eSJiri Pirko
devlink_nl_rate_get_doit(struct sk_buff * skb,struct genl_info * info)2417cc7194eSJiri Pirko int devlink_nl_rate_get_doit(struct sk_buff *skb, struct genl_info *info)
2427cc7194eSJiri Pirko {
2437cc7194eSJiri Pirko struct devlink *devlink = info->user_ptr[0];
2447cc7194eSJiri Pirko struct devlink_rate *devlink_rate;
2457cc7194eSJiri Pirko struct sk_buff *msg;
2467cc7194eSJiri Pirko int err;
2477cc7194eSJiri Pirko
2487cc7194eSJiri Pirko devlink_rate = devlink_rate_get_from_info(devlink, info);
2497cc7194eSJiri Pirko if (IS_ERR(devlink_rate))
2507cc7194eSJiri Pirko return PTR_ERR(devlink_rate);
2517cc7194eSJiri Pirko
2527cc7194eSJiri Pirko msg = nlmsg_new(NLMSG_DEFAULT_SIZE, GFP_KERNEL);
2537cc7194eSJiri Pirko if (!msg)
2547cc7194eSJiri Pirko return -ENOMEM;
2557cc7194eSJiri Pirko
2567cc7194eSJiri Pirko err = devlink_nl_rate_fill(msg, devlink_rate, DEVLINK_CMD_RATE_NEW,
2577cc7194eSJiri Pirko info->snd_portid, info->snd_seq, 0,
2587cc7194eSJiri Pirko info->extack);
2597cc7194eSJiri Pirko if (err) {
2607cc7194eSJiri Pirko nlmsg_free(msg);
2617cc7194eSJiri Pirko return err;
2627cc7194eSJiri Pirko }
2637cc7194eSJiri Pirko
2647cc7194eSJiri Pirko return genlmsg_reply(msg, info);
2657cc7194eSJiri Pirko }
2667cc7194eSJiri Pirko
2677cc7194eSJiri Pirko static bool
devlink_rate_is_parent_node(struct devlink_rate * devlink_rate,struct devlink_rate * parent)2687cc7194eSJiri Pirko devlink_rate_is_parent_node(struct devlink_rate *devlink_rate,
2697cc7194eSJiri Pirko struct devlink_rate *parent)
2707cc7194eSJiri Pirko {
2717cc7194eSJiri Pirko while (parent) {
2727cc7194eSJiri Pirko if (parent == devlink_rate)
2737cc7194eSJiri Pirko return true;
2747cc7194eSJiri Pirko parent = parent->parent;
2757cc7194eSJiri Pirko }
2767cc7194eSJiri Pirko return false;
2777cc7194eSJiri Pirko }
2787cc7194eSJiri Pirko
2797cc7194eSJiri Pirko static int
devlink_nl_rate_parent_node_set(struct devlink_rate * devlink_rate,struct genl_info * info,struct nlattr * nla_parent)2807cc7194eSJiri Pirko devlink_nl_rate_parent_node_set(struct devlink_rate *devlink_rate,
2817cc7194eSJiri Pirko struct genl_info *info,
2827cc7194eSJiri Pirko struct nlattr *nla_parent)
2837cc7194eSJiri Pirko {
2847cc7194eSJiri Pirko struct devlink *devlink = devlink_rate->devlink;
2857cc7194eSJiri Pirko const char *parent_name = nla_data(nla_parent);
2867cc7194eSJiri Pirko const struct devlink_ops *ops = devlink->ops;
2877cc7194eSJiri Pirko size_t len = strlen(parent_name);
2887cc7194eSJiri Pirko struct devlink_rate *parent;
2897cc7194eSJiri Pirko int err = -EOPNOTSUPP;
2907cc7194eSJiri Pirko
2917cc7194eSJiri Pirko parent = devlink_rate->parent;
2927cc7194eSJiri Pirko
2937cc7194eSJiri Pirko if (parent && !len) {
2947cc7194eSJiri Pirko if (devlink_rate_is_leaf(devlink_rate))
2957cc7194eSJiri Pirko err = ops->rate_leaf_parent_set(devlink_rate, NULL,
2967cc7194eSJiri Pirko devlink_rate->priv, NULL,
2977cc7194eSJiri Pirko info->extack);
2987cc7194eSJiri Pirko else if (devlink_rate_is_node(devlink_rate))
2997cc7194eSJiri Pirko err = ops->rate_node_parent_set(devlink_rate, NULL,
3007cc7194eSJiri Pirko devlink_rate->priv, NULL,
3017cc7194eSJiri Pirko info->extack);
3027cc7194eSJiri Pirko if (err)
3037cc7194eSJiri Pirko return err;
3047cc7194eSJiri Pirko
3057cc7194eSJiri Pirko refcount_dec(&parent->refcnt);
3067cc7194eSJiri Pirko devlink_rate->parent = NULL;
3077cc7194eSJiri Pirko } else if (len) {
3087cc7194eSJiri Pirko parent = devlink_rate_node_get_by_name(devlink, parent_name);
3097cc7194eSJiri Pirko if (IS_ERR(parent))
3107cc7194eSJiri Pirko return -ENODEV;
3117cc7194eSJiri Pirko
3127cc7194eSJiri Pirko if (parent == devlink_rate) {
3137cc7194eSJiri Pirko NL_SET_ERR_MSG(info->extack, "Parent to self is not allowed");
3147cc7194eSJiri Pirko return -EINVAL;
3157cc7194eSJiri Pirko }
3167cc7194eSJiri Pirko
3177cc7194eSJiri Pirko if (devlink_rate_is_node(devlink_rate) &&
3187cc7194eSJiri Pirko devlink_rate_is_parent_node(devlink_rate, parent->parent)) {
3197cc7194eSJiri Pirko NL_SET_ERR_MSG(info->extack, "Node is already a parent of parent node.");
3207cc7194eSJiri Pirko return -EEXIST;
3217cc7194eSJiri Pirko }
3227cc7194eSJiri Pirko
3237cc7194eSJiri Pirko if (devlink_rate_is_leaf(devlink_rate))
3247cc7194eSJiri Pirko err = ops->rate_leaf_parent_set(devlink_rate, parent,
3257cc7194eSJiri Pirko devlink_rate->priv, parent->priv,
3267cc7194eSJiri Pirko info->extack);
3277cc7194eSJiri Pirko else if (devlink_rate_is_node(devlink_rate))
3287cc7194eSJiri Pirko err = ops->rate_node_parent_set(devlink_rate, parent,
3297cc7194eSJiri Pirko devlink_rate->priv, parent->priv,
3307cc7194eSJiri Pirko info->extack);
3317cc7194eSJiri Pirko if (err)
3327cc7194eSJiri Pirko return err;
3337cc7194eSJiri Pirko
3347cc7194eSJiri Pirko if (devlink_rate->parent)
3357cc7194eSJiri Pirko /* we're reassigning to other parent in this case */
3367cc7194eSJiri Pirko refcount_dec(&devlink_rate->parent->refcnt);
3377cc7194eSJiri Pirko
3387cc7194eSJiri Pirko refcount_inc(&parent->refcnt);
3397cc7194eSJiri Pirko devlink_rate->parent = parent;
3407cc7194eSJiri Pirko }
3417cc7194eSJiri Pirko
3427cc7194eSJiri Pirko return 0;
3437cc7194eSJiri Pirko }
3447cc7194eSJiri Pirko
devlink_nl_rate_tc_bw_parse(struct nlattr * parent_nest,u32 * tc_bw,unsigned long * bitmap,struct netlink_ext_ack * extack)345566e8f10SCarolina Jubran static int devlink_nl_rate_tc_bw_parse(struct nlattr *parent_nest, u32 *tc_bw,
346566e8f10SCarolina Jubran unsigned long *bitmap,
347566e8f10SCarolina Jubran struct netlink_ext_ack *extack)
348566e8f10SCarolina Jubran {
349*1bbdb81aSCarolina Jubran struct nlattr *tb[DEVLINK_RATE_TC_ATTR_MAX + 1];
350566e8f10SCarolina Jubran u8 tc_index;
351566e8f10SCarolina Jubran int err;
352566e8f10SCarolina Jubran
353*1bbdb81aSCarolina Jubran err = nla_parse_nested(tb, DEVLINK_RATE_TC_ATTR_MAX, parent_nest,
354566e8f10SCarolina Jubran devlink_dl_rate_tc_bws_nl_policy, extack);
355566e8f10SCarolina Jubran if (err)
356566e8f10SCarolina Jubran return err;
357566e8f10SCarolina Jubran
358*1bbdb81aSCarolina Jubran if (!tb[DEVLINK_RATE_TC_ATTR_INDEX]) {
359566e8f10SCarolina Jubran NL_SET_ERR_ATTR_MISS(extack, parent_nest,
360*1bbdb81aSCarolina Jubran DEVLINK_RATE_TC_ATTR_INDEX);
361566e8f10SCarolina Jubran return -EINVAL;
362566e8f10SCarolina Jubran }
363566e8f10SCarolina Jubran
364*1bbdb81aSCarolina Jubran tc_index = nla_get_u8(tb[DEVLINK_RATE_TC_ATTR_INDEX]);
365566e8f10SCarolina Jubran
366*1bbdb81aSCarolina Jubran if (!tb[DEVLINK_RATE_TC_ATTR_BW]) {
367566e8f10SCarolina Jubran NL_SET_ERR_ATTR_MISS(extack, parent_nest,
368*1bbdb81aSCarolina Jubran DEVLINK_RATE_TC_ATTR_BW);
369566e8f10SCarolina Jubran return -EINVAL;
370566e8f10SCarolina Jubran }
371566e8f10SCarolina Jubran
372566e8f10SCarolina Jubran if (test_and_set_bit(tc_index, bitmap)) {
373566e8f10SCarolina Jubran NL_SET_ERR_MSG_FMT(extack,
374566e8f10SCarolina Jubran "Duplicate traffic class index specified (%u)",
375566e8f10SCarolina Jubran tc_index);
376566e8f10SCarolina Jubran return -EINVAL;
377566e8f10SCarolina Jubran }
378566e8f10SCarolina Jubran
379*1bbdb81aSCarolina Jubran tc_bw[tc_index] = nla_get_u32(tb[DEVLINK_RATE_TC_ATTR_BW]);
380566e8f10SCarolina Jubran
381566e8f10SCarolina Jubran return 0;
382566e8f10SCarolina Jubran }
383566e8f10SCarolina Jubran
devlink_nl_rate_tc_bw_set(struct devlink_rate * devlink_rate,struct genl_info * info)384566e8f10SCarolina Jubran static int devlink_nl_rate_tc_bw_set(struct devlink_rate *devlink_rate,
385566e8f10SCarolina Jubran struct genl_info *info)
386566e8f10SCarolina Jubran {
387566e8f10SCarolina Jubran DECLARE_BITMAP(bitmap, DEVLINK_RATE_TCS_MAX) = {};
388566e8f10SCarolina Jubran struct devlink *devlink = devlink_rate->devlink;
389566e8f10SCarolina Jubran const struct devlink_ops *ops = devlink->ops;
390566e8f10SCarolina Jubran u32 tc_bw[DEVLINK_RATE_TCS_MAX] = {};
391566e8f10SCarolina Jubran int rem, err = -EOPNOTSUPP, i;
392566e8f10SCarolina Jubran struct nlattr *attr;
393566e8f10SCarolina Jubran
394566e8f10SCarolina Jubran nlmsg_for_each_attr_type(attr, DEVLINK_ATTR_RATE_TC_BWS, info->nlhdr,
395566e8f10SCarolina Jubran GENL_HDRLEN, rem) {
396566e8f10SCarolina Jubran err = devlink_nl_rate_tc_bw_parse(attr, tc_bw, bitmap,
397566e8f10SCarolina Jubran info->extack);
398566e8f10SCarolina Jubran if (err)
399566e8f10SCarolina Jubran return err;
400566e8f10SCarolina Jubran }
401566e8f10SCarolina Jubran
402566e8f10SCarolina Jubran for (i = 0; i < DEVLINK_RATE_TCS_MAX; i++) {
403566e8f10SCarolina Jubran if (!test_bit(i, bitmap)) {
404566e8f10SCarolina Jubran NL_SET_ERR_MSG_FMT(info->extack,
405566e8f10SCarolina Jubran "Bandwidth values must be specified for all %u traffic classes",
406566e8f10SCarolina Jubran DEVLINK_RATE_TCS_MAX);
407566e8f10SCarolina Jubran return -EINVAL;
408566e8f10SCarolina Jubran }
409566e8f10SCarolina Jubran }
410566e8f10SCarolina Jubran
411566e8f10SCarolina Jubran if (devlink_rate_is_leaf(devlink_rate))
412566e8f10SCarolina Jubran err = ops->rate_leaf_tc_bw_set(devlink_rate, devlink_rate->priv,
413566e8f10SCarolina Jubran tc_bw, info->extack);
414566e8f10SCarolina Jubran else if (devlink_rate_is_node(devlink_rate))
415566e8f10SCarolina Jubran err = ops->rate_node_tc_bw_set(devlink_rate, devlink_rate->priv,
416566e8f10SCarolina Jubran tc_bw, info->extack);
417566e8f10SCarolina Jubran
418566e8f10SCarolina Jubran if (err)
419566e8f10SCarolina Jubran return err;
420566e8f10SCarolina Jubran
421566e8f10SCarolina Jubran memcpy(devlink_rate->tc_bw, tc_bw, sizeof(tc_bw));
422566e8f10SCarolina Jubran
423566e8f10SCarolina Jubran return 0;
424566e8f10SCarolina Jubran }
425566e8f10SCarolina Jubran
devlink_nl_rate_set(struct devlink_rate * devlink_rate,const struct devlink_ops * ops,struct genl_info * info)4267cc7194eSJiri Pirko static int devlink_nl_rate_set(struct devlink_rate *devlink_rate,
4277cc7194eSJiri Pirko const struct devlink_ops *ops,
4287cc7194eSJiri Pirko struct genl_info *info)
4297cc7194eSJiri Pirko {
4307cc7194eSJiri Pirko struct nlattr *nla_parent, **attrs = info->attrs;
4317cc7194eSJiri Pirko int err = -EOPNOTSUPP;
4327cc7194eSJiri Pirko u32 priority;
4337cc7194eSJiri Pirko u32 weight;
4347cc7194eSJiri Pirko u64 rate;
4357cc7194eSJiri Pirko
4367cc7194eSJiri Pirko if (attrs[DEVLINK_ATTR_RATE_TX_SHARE]) {
4377cc7194eSJiri Pirko rate = nla_get_u64(attrs[DEVLINK_ATTR_RATE_TX_SHARE]);
4387cc7194eSJiri Pirko if (devlink_rate_is_leaf(devlink_rate))
4397cc7194eSJiri Pirko err = ops->rate_leaf_tx_share_set(devlink_rate, devlink_rate->priv,
4407cc7194eSJiri Pirko rate, info->extack);
4417cc7194eSJiri Pirko else if (devlink_rate_is_node(devlink_rate))
4427cc7194eSJiri Pirko err = ops->rate_node_tx_share_set(devlink_rate, devlink_rate->priv,
4437cc7194eSJiri Pirko rate, info->extack);
4447cc7194eSJiri Pirko if (err)
4457cc7194eSJiri Pirko return err;
4467cc7194eSJiri Pirko devlink_rate->tx_share = rate;
4477cc7194eSJiri Pirko }
4487cc7194eSJiri Pirko
4497cc7194eSJiri Pirko if (attrs[DEVLINK_ATTR_RATE_TX_MAX]) {
4507cc7194eSJiri Pirko rate = nla_get_u64(attrs[DEVLINK_ATTR_RATE_TX_MAX]);
4517cc7194eSJiri Pirko if (devlink_rate_is_leaf(devlink_rate))
4527cc7194eSJiri Pirko err = ops->rate_leaf_tx_max_set(devlink_rate, devlink_rate->priv,
4537cc7194eSJiri Pirko rate, info->extack);
4547cc7194eSJiri Pirko else if (devlink_rate_is_node(devlink_rate))
4557cc7194eSJiri Pirko err = ops->rate_node_tx_max_set(devlink_rate, devlink_rate->priv,
4567cc7194eSJiri Pirko rate, info->extack);
4577cc7194eSJiri Pirko if (err)
4587cc7194eSJiri Pirko return err;
4597cc7194eSJiri Pirko devlink_rate->tx_max = rate;
4607cc7194eSJiri Pirko }
4617cc7194eSJiri Pirko
4627cc7194eSJiri Pirko if (attrs[DEVLINK_ATTR_RATE_TX_PRIORITY]) {
4637cc7194eSJiri Pirko priority = nla_get_u32(attrs[DEVLINK_ATTR_RATE_TX_PRIORITY]);
4647cc7194eSJiri Pirko if (devlink_rate_is_leaf(devlink_rate))
4657cc7194eSJiri Pirko err = ops->rate_leaf_tx_priority_set(devlink_rate, devlink_rate->priv,
4667cc7194eSJiri Pirko priority, info->extack);
4677cc7194eSJiri Pirko else if (devlink_rate_is_node(devlink_rate))
4687cc7194eSJiri Pirko err = ops->rate_node_tx_priority_set(devlink_rate, devlink_rate->priv,
4697cc7194eSJiri Pirko priority, info->extack);
4707cc7194eSJiri Pirko
4717cc7194eSJiri Pirko if (err)
4727cc7194eSJiri Pirko return err;
4737cc7194eSJiri Pirko devlink_rate->tx_priority = priority;
4747cc7194eSJiri Pirko }
4757cc7194eSJiri Pirko
4767cc7194eSJiri Pirko if (attrs[DEVLINK_ATTR_RATE_TX_WEIGHT]) {
4777cc7194eSJiri Pirko weight = nla_get_u32(attrs[DEVLINK_ATTR_RATE_TX_WEIGHT]);
4787cc7194eSJiri Pirko if (devlink_rate_is_leaf(devlink_rate))
4797cc7194eSJiri Pirko err = ops->rate_leaf_tx_weight_set(devlink_rate, devlink_rate->priv,
4807cc7194eSJiri Pirko weight, info->extack);
4817cc7194eSJiri Pirko else if (devlink_rate_is_node(devlink_rate))
4827cc7194eSJiri Pirko err = ops->rate_node_tx_weight_set(devlink_rate, devlink_rate->priv,
4837cc7194eSJiri Pirko weight, info->extack);
4847cc7194eSJiri Pirko
4857cc7194eSJiri Pirko if (err)
4867cc7194eSJiri Pirko return err;
4877cc7194eSJiri Pirko devlink_rate->tx_weight = weight;
4887cc7194eSJiri Pirko }
4897cc7194eSJiri Pirko
4907cc7194eSJiri Pirko nla_parent = attrs[DEVLINK_ATTR_RATE_PARENT_NODE_NAME];
4917cc7194eSJiri Pirko if (nla_parent) {
4927cc7194eSJiri Pirko err = devlink_nl_rate_parent_node_set(devlink_rate, info,
4937cc7194eSJiri Pirko nla_parent);
4947cc7194eSJiri Pirko if (err)
4957cc7194eSJiri Pirko return err;
4967cc7194eSJiri Pirko }
4977cc7194eSJiri Pirko
498566e8f10SCarolina Jubran if (attrs[DEVLINK_ATTR_RATE_TC_BWS]) {
499566e8f10SCarolina Jubran err = devlink_nl_rate_tc_bw_set(devlink_rate, info);
500566e8f10SCarolina Jubran if (err)
501566e8f10SCarolina Jubran return err;
502566e8f10SCarolina Jubran }
503566e8f10SCarolina Jubran
5047cc7194eSJiri Pirko return 0;
5057cc7194eSJiri Pirko }
5067cc7194eSJiri Pirko
devlink_rate_set_ops_supported(const struct devlink_ops * ops,struct genl_info * info,enum devlink_rate_type type)5077cc7194eSJiri Pirko static bool devlink_rate_set_ops_supported(const struct devlink_ops *ops,
5087cc7194eSJiri Pirko struct genl_info *info,
5097cc7194eSJiri Pirko enum devlink_rate_type type)
5107cc7194eSJiri Pirko {
5117cc7194eSJiri Pirko struct nlattr **attrs = info->attrs;
5127cc7194eSJiri Pirko
5137cc7194eSJiri Pirko if (type == DEVLINK_RATE_TYPE_LEAF) {
5147cc7194eSJiri Pirko if (attrs[DEVLINK_ATTR_RATE_TX_SHARE] && !ops->rate_leaf_tx_share_set) {
5157cc7194eSJiri Pirko NL_SET_ERR_MSG(info->extack, "TX share set isn't supported for the leafs");
5167cc7194eSJiri Pirko return false;
5177cc7194eSJiri Pirko }
5187cc7194eSJiri Pirko if (attrs[DEVLINK_ATTR_RATE_TX_MAX] && !ops->rate_leaf_tx_max_set) {
5197cc7194eSJiri Pirko NL_SET_ERR_MSG(info->extack, "TX max set isn't supported for the leafs");
5207cc7194eSJiri Pirko return false;
5217cc7194eSJiri Pirko }
5227cc7194eSJiri Pirko if (attrs[DEVLINK_ATTR_RATE_PARENT_NODE_NAME] &&
5237cc7194eSJiri Pirko !ops->rate_leaf_parent_set) {
5247cc7194eSJiri Pirko NL_SET_ERR_MSG(info->extack, "Parent set isn't supported for the leafs");
5257cc7194eSJiri Pirko return false;
5267cc7194eSJiri Pirko }
5277cc7194eSJiri Pirko if (attrs[DEVLINK_ATTR_RATE_TX_PRIORITY] && !ops->rate_leaf_tx_priority_set) {
5287cc7194eSJiri Pirko NL_SET_ERR_MSG_ATTR(info->extack,
5297cc7194eSJiri Pirko attrs[DEVLINK_ATTR_RATE_TX_PRIORITY],
5307cc7194eSJiri Pirko "TX priority set isn't supported for the leafs");
5317cc7194eSJiri Pirko return false;
5327cc7194eSJiri Pirko }
5337cc7194eSJiri Pirko if (attrs[DEVLINK_ATTR_RATE_TX_WEIGHT] && !ops->rate_leaf_tx_weight_set) {
5347cc7194eSJiri Pirko NL_SET_ERR_MSG_ATTR(info->extack,
5357cc7194eSJiri Pirko attrs[DEVLINK_ATTR_RATE_TX_WEIGHT],
5367cc7194eSJiri Pirko "TX weight set isn't supported for the leafs");
5377cc7194eSJiri Pirko return false;
5387cc7194eSJiri Pirko }
539566e8f10SCarolina Jubran if (attrs[DEVLINK_ATTR_RATE_TC_BWS] &&
540566e8f10SCarolina Jubran !ops->rate_leaf_tc_bw_set) {
541566e8f10SCarolina Jubran NL_SET_ERR_MSG_ATTR(info->extack,
542566e8f10SCarolina Jubran attrs[DEVLINK_ATTR_RATE_TC_BWS],
543566e8f10SCarolina Jubran "TC bandwidth set isn't supported for the leafs");
544566e8f10SCarolina Jubran return false;
545566e8f10SCarolina Jubran }
5467cc7194eSJiri Pirko } else if (type == DEVLINK_RATE_TYPE_NODE) {
5477cc7194eSJiri Pirko if (attrs[DEVLINK_ATTR_RATE_TX_SHARE] && !ops->rate_node_tx_share_set) {
5487cc7194eSJiri Pirko NL_SET_ERR_MSG(info->extack, "TX share set isn't supported for the nodes");
5497cc7194eSJiri Pirko return false;
5507cc7194eSJiri Pirko }
5517cc7194eSJiri Pirko if (attrs[DEVLINK_ATTR_RATE_TX_MAX] && !ops->rate_node_tx_max_set) {
5527cc7194eSJiri Pirko NL_SET_ERR_MSG(info->extack, "TX max set isn't supported for the nodes");
5537cc7194eSJiri Pirko return false;
5547cc7194eSJiri Pirko }
5557cc7194eSJiri Pirko if (attrs[DEVLINK_ATTR_RATE_PARENT_NODE_NAME] &&
5567cc7194eSJiri Pirko !ops->rate_node_parent_set) {
5577cc7194eSJiri Pirko NL_SET_ERR_MSG(info->extack, "Parent set isn't supported for the nodes");
5587cc7194eSJiri Pirko return false;
5597cc7194eSJiri Pirko }
5607cc7194eSJiri Pirko if (attrs[DEVLINK_ATTR_RATE_TX_PRIORITY] && !ops->rate_node_tx_priority_set) {
5617cc7194eSJiri Pirko NL_SET_ERR_MSG_ATTR(info->extack,
5627cc7194eSJiri Pirko attrs[DEVLINK_ATTR_RATE_TX_PRIORITY],
5637cc7194eSJiri Pirko "TX priority set isn't supported for the nodes");
5647cc7194eSJiri Pirko return false;
5657cc7194eSJiri Pirko }
5667cc7194eSJiri Pirko if (attrs[DEVLINK_ATTR_RATE_TX_WEIGHT] && !ops->rate_node_tx_weight_set) {
5677cc7194eSJiri Pirko NL_SET_ERR_MSG_ATTR(info->extack,
5687cc7194eSJiri Pirko attrs[DEVLINK_ATTR_RATE_TX_WEIGHT],
5697cc7194eSJiri Pirko "TX weight set isn't supported for the nodes");
5707cc7194eSJiri Pirko return false;
5717cc7194eSJiri Pirko }
572566e8f10SCarolina Jubran if (attrs[DEVLINK_ATTR_RATE_TC_BWS] &&
573566e8f10SCarolina Jubran !ops->rate_node_tc_bw_set) {
574566e8f10SCarolina Jubran NL_SET_ERR_MSG_ATTR(info->extack,
575566e8f10SCarolina Jubran attrs[DEVLINK_ATTR_RATE_TC_BWS],
576566e8f10SCarolina Jubran "TC bandwidth set isn't supported for the nodes");
577566e8f10SCarolina Jubran return false;
578566e8f10SCarolina Jubran }
5797cc7194eSJiri Pirko } else {
5807cc7194eSJiri Pirko WARN(1, "Unknown type of rate object");
5817cc7194eSJiri Pirko return false;
5827cc7194eSJiri Pirko }
5837cc7194eSJiri Pirko
5847cc7194eSJiri Pirko return true;
5857cc7194eSJiri Pirko }
5867cc7194eSJiri Pirko
devlink_nl_rate_set_doit(struct sk_buff * skb,struct genl_info * info)58753590934SJiri Pirko int devlink_nl_rate_set_doit(struct sk_buff *skb, struct genl_info *info)
5887cc7194eSJiri Pirko {
5897cc7194eSJiri Pirko struct devlink *devlink = info->user_ptr[0];
5907cc7194eSJiri Pirko struct devlink_rate *devlink_rate;
5917cc7194eSJiri Pirko const struct devlink_ops *ops;
5927cc7194eSJiri Pirko int err;
5937cc7194eSJiri Pirko
5947cc7194eSJiri Pirko devlink_rate = devlink_rate_get_from_info(devlink, info);
5957cc7194eSJiri Pirko if (IS_ERR(devlink_rate))
5967cc7194eSJiri Pirko return PTR_ERR(devlink_rate);
5977cc7194eSJiri Pirko
5987cc7194eSJiri Pirko ops = devlink->ops;
5997cc7194eSJiri Pirko if (!ops || !devlink_rate_set_ops_supported(ops, info, devlink_rate->type))
6007cc7194eSJiri Pirko return -EOPNOTSUPP;
6017cc7194eSJiri Pirko
6027cc7194eSJiri Pirko err = devlink_nl_rate_set(devlink_rate, ops, info);
6037cc7194eSJiri Pirko
6047cc7194eSJiri Pirko if (!err)
6057cc7194eSJiri Pirko devlink_rate_notify(devlink_rate, DEVLINK_CMD_RATE_NEW);
6067cc7194eSJiri Pirko return err;
6077cc7194eSJiri Pirko }
6087cc7194eSJiri Pirko
devlink_nl_rate_new_doit(struct sk_buff * skb,struct genl_info * info)60953590934SJiri Pirko int devlink_nl_rate_new_doit(struct sk_buff *skb, struct genl_info *info)
6107cc7194eSJiri Pirko {
6117cc7194eSJiri Pirko struct devlink *devlink = info->user_ptr[0];
6127cc7194eSJiri Pirko struct devlink_rate *rate_node;
6137cc7194eSJiri Pirko const struct devlink_ops *ops;
6147cc7194eSJiri Pirko int err;
6157cc7194eSJiri Pirko
6167cc7194eSJiri Pirko ops = devlink->ops;
6177cc7194eSJiri Pirko if (!ops || !ops->rate_node_new || !ops->rate_node_del) {
6187cc7194eSJiri Pirko NL_SET_ERR_MSG(info->extack, "Rate nodes aren't supported");
6197cc7194eSJiri Pirko return -EOPNOTSUPP;
6207cc7194eSJiri Pirko }
6217cc7194eSJiri Pirko
6227cc7194eSJiri Pirko if (!devlink_rate_set_ops_supported(ops, info, DEVLINK_RATE_TYPE_NODE))
6237cc7194eSJiri Pirko return -EOPNOTSUPP;
6247cc7194eSJiri Pirko
6257cc7194eSJiri Pirko rate_node = devlink_rate_node_get_from_attrs(devlink, info->attrs);
6267cc7194eSJiri Pirko if (!IS_ERR(rate_node))
6277cc7194eSJiri Pirko return -EEXIST;
6287cc7194eSJiri Pirko else if (rate_node == ERR_PTR(-EINVAL))
6297cc7194eSJiri Pirko return -EINVAL;
6307cc7194eSJiri Pirko
6317cc7194eSJiri Pirko rate_node = kzalloc(sizeof(*rate_node), GFP_KERNEL);
6327cc7194eSJiri Pirko if (!rate_node)
6337cc7194eSJiri Pirko return -ENOMEM;
6347cc7194eSJiri Pirko
6357cc7194eSJiri Pirko rate_node->devlink = devlink;
6367cc7194eSJiri Pirko rate_node->type = DEVLINK_RATE_TYPE_NODE;
6377cc7194eSJiri Pirko rate_node->name = nla_strdup(info->attrs[DEVLINK_ATTR_RATE_NODE_NAME], GFP_KERNEL);
6387cc7194eSJiri Pirko if (!rate_node->name) {
6397cc7194eSJiri Pirko err = -ENOMEM;
6407cc7194eSJiri Pirko goto err_strdup;
6417cc7194eSJiri Pirko }
6427cc7194eSJiri Pirko
6437cc7194eSJiri Pirko err = ops->rate_node_new(rate_node, &rate_node->priv, info->extack);
6447cc7194eSJiri Pirko if (err)
6457cc7194eSJiri Pirko goto err_node_new;
6467cc7194eSJiri Pirko
6477cc7194eSJiri Pirko err = devlink_nl_rate_set(rate_node, ops, info);
6487cc7194eSJiri Pirko if (err)
6497cc7194eSJiri Pirko goto err_rate_set;
6507cc7194eSJiri Pirko
6517cc7194eSJiri Pirko refcount_set(&rate_node->refcnt, 1);
6527cc7194eSJiri Pirko list_add(&rate_node->list, &devlink->rate_list);
6537cc7194eSJiri Pirko devlink_rate_notify(rate_node, DEVLINK_CMD_RATE_NEW);
6547cc7194eSJiri Pirko return 0;
6557cc7194eSJiri Pirko
6567cc7194eSJiri Pirko err_rate_set:
6577cc7194eSJiri Pirko ops->rate_node_del(rate_node, rate_node->priv, info->extack);
6587cc7194eSJiri Pirko err_node_new:
6597cc7194eSJiri Pirko kfree(rate_node->name);
6607cc7194eSJiri Pirko err_strdup:
6617cc7194eSJiri Pirko kfree(rate_node);
6627cc7194eSJiri Pirko return err;
6637cc7194eSJiri Pirko }
6647cc7194eSJiri Pirko
devlink_nl_rate_del_doit(struct sk_buff * skb,struct genl_info * info)66553590934SJiri Pirko int devlink_nl_rate_del_doit(struct sk_buff *skb, struct genl_info *info)
6667cc7194eSJiri Pirko {
6677cc7194eSJiri Pirko struct devlink *devlink = info->user_ptr[0];
6687cc7194eSJiri Pirko struct devlink_rate *rate_node;
6697cc7194eSJiri Pirko int err;
6707cc7194eSJiri Pirko
6717cc7194eSJiri Pirko rate_node = devlink_rate_node_get_from_info(devlink, info);
6727cc7194eSJiri Pirko if (IS_ERR(rate_node))
6737cc7194eSJiri Pirko return PTR_ERR(rate_node);
6747cc7194eSJiri Pirko
6757cc7194eSJiri Pirko if (refcount_read(&rate_node->refcnt) > 1) {
6767cc7194eSJiri Pirko NL_SET_ERR_MSG(info->extack, "Node has children. Cannot delete node.");
6777cc7194eSJiri Pirko return -EBUSY;
6787cc7194eSJiri Pirko }
6797cc7194eSJiri Pirko
6807cc7194eSJiri Pirko devlink_rate_notify(rate_node, DEVLINK_CMD_RATE_DEL);
6817cc7194eSJiri Pirko err = devlink->ops->rate_node_del(rate_node, rate_node->priv,
6827cc7194eSJiri Pirko info->extack);
6837cc7194eSJiri Pirko if (rate_node->parent)
6847cc7194eSJiri Pirko refcount_dec(&rate_node->parent->refcnt);
6857cc7194eSJiri Pirko list_del(&rate_node->list);
6867cc7194eSJiri Pirko kfree(rate_node->name);
6877cc7194eSJiri Pirko kfree(rate_node);
6887cc7194eSJiri Pirko return err;
6897cc7194eSJiri Pirko }
6907cc7194eSJiri Pirko
devlink_rate_nodes_check(struct devlink * devlink,u16 mode,struct netlink_ext_ack * extack)6917cc7194eSJiri Pirko int devlink_rate_nodes_check(struct devlink *devlink, u16 mode,
6927cc7194eSJiri Pirko struct netlink_ext_ack *extack)
6937cc7194eSJiri Pirko {
6947cc7194eSJiri Pirko struct devlink_rate *devlink_rate;
6957cc7194eSJiri Pirko
6967cc7194eSJiri Pirko list_for_each_entry(devlink_rate, &devlink->rate_list, list)
6977cc7194eSJiri Pirko if (devlink_rate_is_node(devlink_rate)) {
6987cc7194eSJiri Pirko NL_SET_ERR_MSG(extack, "Rate node(s) exists.");
6997cc7194eSJiri Pirko return -EBUSY;
7007cc7194eSJiri Pirko }
7017cc7194eSJiri Pirko return 0;
7027cc7194eSJiri Pirko }
7037cc7194eSJiri Pirko
7047cc7194eSJiri Pirko /**
7057cc7194eSJiri Pirko * devl_rate_node_create - create devlink rate node
7067cc7194eSJiri Pirko * @devlink: devlink instance
7077cc7194eSJiri Pirko * @priv: driver private data
7087cc7194eSJiri Pirko * @node_name: name of the resulting node
7097cc7194eSJiri Pirko * @parent: parent devlink_rate struct
7107cc7194eSJiri Pirko *
7117cc7194eSJiri Pirko * Create devlink rate object of type node
7127cc7194eSJiri Pirko */
7137cc7194eSJiri Pirko struct devlink_rate *
devl_rate_node_create(struct devlink * devlink,void * priv,char * node_name,struct devlink_rate * parent)7147cc7194eSJiri Pirko devl_rate_node_create(struct devlink *devlink, void *priv, char *node_name,
7157cc7194eSJiri Pirko struct devlink_rate *parent)
7167cc7194eSJiri Pirko {
7177cc7194eSJiri Pirko struct devlink_rate *rate_node;
7187cc7194eSJiri Pirko
7197cc7194eSJiri Pirko rate_node = devlink_rate_node_get_by_name(devlink, node_name);
7207cc7194eSJiri Pirko if (!IS_ERR(rate_node))
7217cc7194eSJiri Pirko return ERR_PTR(-EEXIST);
7227cc7194eSJiri Pirko
7237cc7194eSJiri Pirko rate_node = kzalloc(sizeof(*rate_node), GFP_KERNEL);
7247cc7194eSJiri Pirko if (!rate_node)
7257cc7194eSJiri Pirko return ERR_PTR(-ENOMEM);
7267cc7194eSJiri Pirko
7277cc7194eSJiri Pirko if (parent) {
7287cc7194eSJiri Pirko rate_node->parent = parent;
7297cc7194eSJiri Pirko refcount_inc(&rate_node->parent->refcnt);
7307cc7194eSJiri Pirko }
7317cc7194eSJiri Pirko
7327cc7194eSJiri Pirko rate_node->type = DEVLINK_RATE_TYPE_NODE;
7337cc7194eSJiri Pirko rate_node->devlink = devlink;
7347cc7194eSJiri Pirko rate_node->priv = priv;
7357cc7194eSJiri Pirko
7367cc7194eSJiri Pirko rate_node->name = kstrdup(node_name, GFP_KERNEL);
7377cc7194eSJiri Pirko if (!rate_node->name) {
7387cc7194eSJiri Pirko kfree(rate_node);
7397cc7194eSJiri Pirko return ERR_PTR(-ENOMEM);
7407cc7194eSJiri Pirko }
7417cc7194eSJiri Pirko
7427cc7194eSJiri Pirko refcount_set(&rate_node->refcnt, 1);
7437cc7194eSJiri Pirko list_add(&rate_node->list, &devlink->rate_list);
7447cc7194eSJiri Pirko devlink_rate_notify(rate_node, DEVLINK_CMD_RATE_NEW);
7457cc7194eSJiri Pirko return rate_node;
7467cc7194eSJiri Pirko }
7477cc7194eSJiri Pirko EXPORT_SYMBOL_GPL(devl_rate_node_create);
7487cc7194eSJiri Pirko
7497cc7194eSJiri Pirko /**
7507cc7194eSJiri Pirko * devl_rate_leaf_create - create devlink rate leaf
7517cc7194eSJiri Pirko * @devlink_port: devlink port object to create rate object on
7527cc7194eSJiri Pirko * @priv: driver private data
7537cc7194eSJiri Pirko * @parent: parent devlink_rate struct
7547cc7194eSJiri Pirko *
7557cc7194eSJiri Pirko * Create devlink rate object of type leaf on provided @devlink_port.
7567cc7194eSJiri Pirko */
devl_rate_leaf_create(struct devlink_port * devlink_port,void * priv,struct devlink_rate * parent)7577cc7194eSJiri Pirko int devl_rate_leaf_create(struct devlink_port *devlink_port, void *priv,
7587cc7194eSJiri Pirko struct devlink_rate *parent)
7597cc7194eSJiri Pirko {
7607cc7194eSJiri Pirko struct devlink *devlink = devlink_port->devlink;
7617cc7194eSJiri Pirko struct devlink_rate *devlink_rate;
7627cc7194eSJiri Pirko
7637cc7194eSJiri Pirko devl_assert_locked(devlink_port->devlink);
7647cc7194eSJiri Pirko
7657cc7194eSJiri Pirko if (WARN_ON(devlink_port->devlink_rate))
7667cc7194eSJiri Pirko return -EBUSY;
7677cc7194eSJiri Pirko
7687cc7194eSJiri Pirko devlink_rate = kzalloc(sizeof(*devlink_rate), GFP_KERNEL);
7697cc7194eSJiri Pirko if (!devlink_rate)
7707cc7194eSJiri Pirko return -ENOMEM;
7717cc7194eSJiri Pirko
7727cc7194eSJiri Pirko if (parent) {
7737cc7194eSJiri Pirko devlink_rate->parent = parent;
7747cc7194eSJiri Pirko refcount_inc(&devlink_rate->parent->refcnt);
7757cc7194eSJiri Pirko }
7767cc7194eSJiri Pirko
7777cc7194eSJiri Pirko devlink_rate->type = DEVLINK_RATE_TYPE_LEAF;
7787cc7194eSJiri Pirko devlink_rate->devlink = devlink;
7797cc7194eSJiri Pirko devlink_rate->devlink_port = devlink_port;
7807cc7194eSJiri Pirko devlink_rate->priv = priv;
7817cc7194eSJiri Pirko list_add_tail(&devlink_rate->list, &devlink->rate_list);
7827cc7194eSJiri Pirko devlink_port->devlink_rate = devlink_rate;
7837cc7194eSJiri Pirko devlink_rate_notify(devlink_rate, DEVLINK_CMD_RATE_NEW);
7847cc7194eSJiri Pirko
7857cc7194eSJiri Pirko return 0;
7867cc7194eSJiri Pirko }
7877cc7194eSJiri Pirko EXPORT_SYMBOL_GPL(devl_rate_leaf_create);
7887cc7194eSJiri Pirko
7897cc7194eSJiri Pirko /**
7907cc7194eSJiri Pirko * devl_rate_leaf_destroy - destroy devlink rate leaf
7917cc7194eSJiri Pirko *
7927cc7194eSJiri Pirko * @devlink_port: devlink port linked to the rate object
7937cc7194eSJiri Pirko *
7947cc7194eSJiri Pirko * Destroy the devlink rate object of type leaf on provided @devlink_port.
7957cc7194eSJiri Pirko */
devl_rate_leaf_destroy(struct devlink_port * devlink_port)7967cc7194eSJiri Pirko void devl_rate_leaf_destroy(struct devlink_port *devlink_port)
7977cc7194eSJiri Pirko {
7987cc7194eSJiri Pirko struct devlink_rate *devlink_rate = devlink_port->devlink_rate;
7997cc7194eSJiri Pirko
8007cc7194eSJiri Pirko devl_assert_locked(devlink_port->devlink);
8017cc7194eSJiri Pirko if (!devlink_rate)
8027cc7194eSJiri Pirko return;
8037cc7194eSJiri Pirko
8047cc7194eSJiri Pirko devlink_rate_notify(devlink_rate, DEVLINK_CMD_RATE_DEL);
8057cc7194eSJiri Pirko if (devlink_rate->parent)
8067cc7194eSJiri Pirko refcount_dec(&devlink_rate->parent->refcnt);
8077cc7194eSJiri Pirko list_del(&devlink_rate->list);
8087cc7194eSJiri Pirko devlink_port->devlink_rate = NULL;
8097cc7194eSJiri Pirko kfree(devlink_rate);
8107cc7194eSJiri Pirko }
8117cc7194eSJiri Pirko EXPORT_SYMBOL_GPL(devl_rate_leaf_destroy);
8127cc7194eSJiri Pirko
8137cc7194eSJiri Pirko /**
8147cc7194eSJiri Pirko * devl_rate_nodes_destroy - destroy all devlink rate nodes on device
8157cc7194eSJiri Pirko * @devlink: devlink instance
8167cc7194eSJiri Pirko *
8177cc7194eSJiri Pirko * Unset parent for all rate objects and destroy all rate nodes
8187cc7194eSJiri Pirko * on specified device.
8197cc7194eSJiri Pirko */
devl_rate_nodes_destroy(struct devlink * devlink)8207cc7194eSJiri Pirko void devl_rate_nodes_destroy(struct devlink *devlink)
8217cc7194eSJiri Pirko {
8227cc7194eSJiri Pirko static struct devlink_rate *devlink_rate, *tmp;
8237cc7194eSJiri Pirko const struct devlink_ops *ops = devlink->ops;
8247cc7194eSJiri Pirko
8257cc7194eSJiri Pirko devl_assert_locked(devlink);
8267cc7194eSJiri Pirko
8277cc7194eSJiri Pirko list_for_each_entry(devlink_rate, &devlink->rate_list, list) {
8287cc7194eSJiri Pirko if (!devlink_rate->parent)
8297cc7194eSJiri Pirko continue;
8307cc7194eSJiri Pirko
8317cc7194eSJiri Pirko refcount_dec(&devlink_rate->parent->refcnt);
8327cc7194eSJiri Pirko if (devlink_rate_is_leaf(devlink_rate))
8337cc7194eSJiri Pirko ops->rate_leaf_parent_set(devlink_rate, NULL, devlink_rate->priv,
8347cc7194eSJiri Pirko NULL, NULL);
8357cc7194eSJiri Pirko else if (devlink_rate_is_node(devlink_rate))
8367cc7194eSJiri Pirko ops->rate_node_parent_set(devlink_rate, NULL, devlink_rate->priv,
8377cc7194eSJiri Pirko NULL, NULL);
8387cc7194eSJiri Pirko }
8397cc7194eSJiri Pirko list_for_each_entry_safe(devlink_rate, tmp, &devlink->rate_list, list) {
8407cc7194eSJiri Pirko if (devlink_rate_is_node(devlink_rate)) {
8417cc7194eSJiri Pirko ops->rate_node_del(devlink_rate, devlink_rate->priv, NULL);
8427cc7194eSJiri Pirko list_del(&devlink_rate->list);
8437cc7194eSJiri Pirko kfree(devlink_rate->name);
8447cc7194eSJiri Pirko kfree(devlink_rate);
8457cc7194eSJiri Pirko }
8467cc7194eSJiri Pirko }
8477cc7194eSJiri Pirko }
8487cc7194eSJiri Pirko EXPORT_SYMBOL_GPL(devl_rate_nodes_destroy);
849