| // SPDX-License-Identifier: GPL-2.0 |
| /* Copyright 2022 NXP |
| */ |
| #include <linux/netdevice.h> |
| #include <net/rtnetlink.h> |
| |
| #include "netlink.h" |
| #include "user.h" |
| |
| static const struct nla_policy dsa_policy[IFLA_DSA_MAX + 1] = { |
| [IFLA_DSA_CONDUIT] = { .type = NLA_U32 }, |
| }; |
| |
| static int dsa_changelink(struct net_device *dev, struct nlattr *tb[], |
| struct nlattr *data[], |
| struct netlink_ext_ack *extack) |
| { |
| int err; |
| |
| if (!data) |
| return 0; |
| |
| if (data[IFLA_DSA_CONDUIT]) { |
| u32 ifindex = nla_get_u32(data[IFLA_DSA_CONDUIT]); |
| struct net_device *conduit; |
| |
| conduit = __dev_get_by_index(dev_net(dev), ifindex); |
| if (!conduit) |
| return -EINVAL; |
| |
| err = dsa_user_change_conduit(dev, conduit, extack); |
| if (err) |
| return err; |
| } |
| |
| return 0; |
| } |
| |
| static size_t dsa_get_size(const struct net_device *dev) |
| { |
| return nla_total_size(sizeof(u32)) + /* IFLA_DSA_CONDUIT */ |
| 0; |
| } |
| |
| static int dsa_fill_info(struct sk_buff *skb, const struct net_device *dev) |
| { |
| struct net_device *conduit = dsa_user_to_conduit(dev); |
| |
| if (nla_put_u32(skb, IFLA_DSA_CONDUIT, conduit->ifindex)) |
| return -EMSGSIZE; |
| |
| return 0; |
| } |
| |
| struct rtnl_link_ops dsa_link_ops __read_mostly = { |
| .kind = "dsa", |
| .priv_size = sizeof(struct dsa_port), |
| .maxtype = IFLA_DSA_MAX, |
| .policy = dsa_policy, |
| .changelink = dsa_changelink, |
| .get_size = dsa_get_size, |
| .fill_info = dsa_fill_info, |
| .netns_refund = true, |
| }; |