1 // SPDX-License-Identifier: GPL-2.0
2 /* Copyright 2022 NXP
3  */
4 #include <linux/netdevice.h>
5 #include <net/rtnetlink.h>
6 
7 #include "dsa_priv.h"
8 
9 static const struct nla_policy dsa_policy[IFLA_DSA_MAX + 1] = {
10 	[IFLA_DSA_MASTER]	= { .type = NLA_U32 },
11 };
12 
dsa_changelink(struct net_device * dev,struct nlattr * tb[],struct nlattr * data[],struct netlink_ext_ack * extack)13 static int dsa_changelink(struct net_device *dev, struct nlattr *tb[],
14 			  struct nlattr *data[],
15 			  struct netlink_ext_ack *extack)
16 {
17 	int err;
18 
19 	if (!data)
20 		return 0;
21 
22 	if (data[IFLA_DSA_MASTER]) {
23 		u32 ifindex = nla_get_u32(data[IFLA_DSA_MASTER]);
24 		struct net_device *master;
25 
26 		master = __dev_get_by_index(dev_net(dev), ifindex);
27 		if (!master)
28 			return -EINVAL;
29 
30 		err = dsa_slave_change_master(dev, master, extack);
31 		if (err)
32 			return err;
33 	}
34 
35 	return 0;
36 }
37 
dsa_get_size(const struct net_device * dev)38 static size_t dsa_get_size(const struct net_device *dev)
39 {
40 	return nla_total_size(sizeof(u32)) +	/* IFLA_DSA_MASTER  */
41 	       0;
42 }
43 
dsa_fill_info(struct sk_buff * skb,const struct net_device * dev)44 static int dsa_fill_info(struct sk_buff *skb, const struct net_device *dev)
45 {
46 	struct net_device *master = dsa_slave_to_master(dev);
47 
48 	if (nla_put_u32(skb, IFLA_DSA_MASTER, master->ifindex))
49 		return -EMSGSIZE;
50 
51 	return 0;
52 }
53 
54 struct rtnl_link_ops dsa_link_ops __read_mostly = {
55 	.kind			= "dsa",
56 	.priv_size		= sizeof(struct dsa_port),
57 	.maxtype		= IFLA_DSA_MAX,
58 	.policy			= dsa_policy,
59 	.changelink		= dsa_changelink,
60 	.get_size		= dsa_get_size,
61 	.fill_info		= dsa_fill_info,
62 	.netns_refund		= true,
63 };
64