2019-04-05 20:31:34 +03:00
|
|
|
// SPDX-License-Identifier: GPL-2.0
|
2014-07-05 01:34:38 +04:00
|
|
|
/* Copyright 2011-2014 Autronica Fire and Security AS
|
2013-10-31 00:10:47 +04:00
|
|
|
*
|
|
|
|
* Author(s):
|
2014-07-05 01:34:38 +04:00
|
|
|
* 2011-2014 Arvid Brodin, arvid.brodin@alten.se
|
2013-10-31 00:10:47 +04:00
|
|
|
*
|
|
|
|
* Routines for handling Netlink messages for HSR.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "hsr_netlink.h"
|
|
|
|
#include <linux/kernel.h>
|
|
|
|
#include <net/rtnetlink.h>
|
|
|
|
#include <net/genetlink.h>
|
|
|
|
#include "hsr_main.h"
|
|
|
|
#include "hsr_device.h"
|
|
|
|
#include "hsr_framereg.h"
|
|
|
|
|
|
|
|
static const struct nla_policy hsr_policy[IFLA_HSR_MAX + 1] = {
|
|
|
|
[IFLA_HSR_SLAVE1] = { .type = NLA_U32 },
|
|
|
|
[IFLA_HSR_SLAVE2] = { .type = NLA_U32 },
|
|
|
|
[IFLA_HSR_MULTICAST_SPEC] = { .type = NLA_U8 },
|
2016-04-13 14:52:22 +03:00
|
|
|
[IFLA_HSR_VERSION] = { .type = NLA_U8 },
|
2016-04-19 14:34:28 +03:00
|
|
|
[IFLA_HSR_SUPERVISION_ADDR] = { .len = ETH_ALEN },
|
2013-11-30 02:38:16 +04:00
|
|
|
[IFLA_HSR_SEQ_NR] = { .type = NLA_U16 },
|
2013-10-31 00:10:47 +04:00
|
|
|
};
|
|
|
|
|
|
|
|
/* Here, it seems a netdevice has already been allocated for us, and the
|
|
|
|
* hsr_dev_setup routine has been executed. Nice!
|
|
|
|
*/
|
|
|
|
static int hsr_newlink(struct net *src_net, struct net_device *dev,
|
2017-06-26 00:55:59 +03:00
|
|
|
struct nlattr *tb[], struct nlattr *data[],
|
|
|
|
struct netlink_ext_ack *extack)
|
2013-10-31 00:10:47 +04:00
|
|
|
{
|
|
|
|
struct net_device *link[2];
|
2016-04-13 14:52:22 +03:00
|
|
|
unsigned char multicast_spec, hsr_version;
|
2013-10-31 00:10:47 +04:00
|
|
|
|
2014-07-05 01:42:00 +04:00
|
|
|
if (!data) {
|
|
|
|
netdev_info(dev, "HSR: No slave devices specified\n");
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
2013-10-31 00:10:47 +04:00
|
|
|
if (!data[IFLA_HSR_SLAVE1]) {
|
2014-07-05 01:42:00 +04:00
|
|
|
netdev_info(dev, "HSR: Slave1 device not specified\n");
|
2013-10-31 00:10:47 +04:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
2019-04-05 20:31:23 +03:00
|
|
|
link[0] = __dev_get_by_index(src_net,
|
|
|
|
nla_get_u32(data[IFLA_HSR_SLAVE1]));
|
2013-10-31 00:10:47 +04:00
|
|
|
if (!data[IFLA_HSR_SLAVE2]) {
|
2014-07-05 01:42:00 +04:00
|
|
|
netdev_info(dev, "HSR: Slave2 device not specified\n");
|
2013-10-31 00:10:47 +04:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
2019-04-05 20:31:23 +03:00
|
|
|
link[1] = __dev_get_by_index(src_net,
|
|
|
|
nla_get_u32(data[IFLA_HSR_SLAVE2]));
|
2013-10-31 00:10:47 +04:00
|
|
|
|
|
|
|
if (!link[0] || !link[1])
|
|
|
|
return -ENODEV;
|
|
|
|
if (link[0] == link[1])
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
if (!data[IFLA_HSR_MULTICAST_SPEC])
|
|
|
|
multicast_spec = 0;
|
|
|
|
else
|
|
|
|
multicast_spec = nla_get_u8(data[IFLA_HSR_MULTICAST_SPEC]);
|
|
|
|
|
2016-04-13 14:52:22 +03:00
|
|
|
if (!data[IFLA_HSR_VERSION])
|
|
|
|
hsr_version = 0;
|
|
|
|
else
|
|
|
|
hsr_version = nla_get_u8(data[IFLA_HSR_VERSION]);
|
|
|
|
|
|
|
|
return hsr_dev_finalize(dev, link, multicast_spec, hsr_version);
|
2013-10-31 00:10:47 +04:00
|
|
|
}
|
|
|
|
|
2013-11-30 02:38:16 +04:00
|
|
|
static int hsr_fill_info(struct sk_buff *skb, const struct net_device *dev)
|
|
|
|
{
|
2014-07-05 01:34:38 +04:00
|
|
|
struct hsr_priv *hsr;
|
2014-07-05 01:38:05 +04:00
|
|
|
struct hsr_port *port;
|
2014-07-05 01:37:27 +04:00
|
|
|
int res;
|
2013-11-30 02:38:16 +04:00
|
|
|
|
2014-07-05 01:34:38 +04:00
|
|
|
hsr = netdev_priv(dev);
|
2013-11-30 02:38:16 +04:00
|
|
|
|
2014-07-05 01:37:27 +04:00
|
|
|
res = 0;
|
2013-11-30 02:38:16 +04:00
|
|
|
|
2014-07-05 01:37:27 +04:00
|
|
|
rcu_read_lock();
|
2014-07-05 01:38:05 +04:00
|
|
|
port = hsr_port_get_hsr(hsr, HSR_PT_SLAVE_A);
|
|
|
|
if (port)
|
|
|
|
res = nla_put_u32(skb, IFLA_HSR_SLAVE1, port->dev->ifindex);
|
2014-07-05 01:37:27 +04:00
|
|
|
rcu_read_unlock();
|
|
|
|
if (res)
|
|
|
|
goto nla_put_failure;
|
|
|
|
|
|
|
|
rcu_read_lock();
|
2014-07-05 01:38:05 +04:00
|
|
|
port = hsr_port_get_hsr(hsr, HSR_PT_SLAVE_B);
|
|
|
|
if (port)
|
|
|
|
res = nla_put_u32(skb, IFLA_HSR_SLAVE2, port->dev->ifindex);
|
2014-07-05 01:37:27 +04:00
|
|
|
rcu_read_unlock();
|
|
|
|
if (res)
|
|
|
|
goto nla_put_failure;
|
2013-11-30 02:38:16 +04:00
|
|
|
|
|
|
|
if (nla_put(skb, IFLA_HSR_SUPERVISION_ADDR, ETH_ALEN,
|
2014-07-05 01:34:38 +04:00
|
|
|
hsr->sup_multicast_addr) ||
|
|
|
|
nla_put_u16(skb, IFLA_HSR_SEQ_NR, hsr->sequence_nr))
|
2013-11-30 02:38:16 +04:00
|
|
|
goto nla_put_failure;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
nla_put_failure:
|
|
|
|
return -EMSGSIZE;
|
|
|
|
}
|
|
|
|
|
2013-10-31 00:10:47 +04:00
|
|
|
static struct rtnl_link_ops hsr_link_ops __read_mostly = {
|
|
|
|
.kind = "hsr",
|
|
|
|
.maxtype = IFLA_HSR_MAX,
|
|
|
|
.policy = hsr_policy,
|
|
|
|
.priv_size = sizeof(struct hsr_priv),
|
|
|
|
.setup = hsr_dev_setup,
|
|
|
|
.newlink = hsr_newlink,
|
2013-11-30 02:38:16 +04:00
|
|
|
.fill_info = hsr_fill_info,
|
2013-10-31 00:10:47 +04:00
|
|
|
};
|
|
|
|
|
|
|
|
/* attribute policy */
|
|
|
|
static const struct nla_policy hsr_genl_policy[HSR_A_MAX + 1] = {
|
2016-04-19 14:34:28 +03:00
|
|
|
[HSR_A_NODE_ADDR] = { .len = ETH_ALEN },
|
|
|
|
[HSR_A_NODE_ADDR_B] = { .len = ETH_ALEN },
|
2013-10-31 00:10:47 +04:00
|
|
|
[HSR_A_IFINDEX] = { .type = NLA_U32 },
|
|
|
|
[HSR_A_IF1_AGE] = { .type = NLA_U32 },
|
|
|
|
[HSR_A_IF2_AGE] = { .type = NLA_U32 },
|
|
|
|
[HSR_A_IF1_SEQ] = { .type = NLA_U16 },
|
|
|
|
[HSR_A_IF2_SEQ] = { .type = NLA_U16 },
|
|
|
|
};
|
|
|
|
|
2016-10-24 15:40:03 +03:00
|
|
|
static struct genl_family hsr_genl_family;
|
2013-10-31 00:10:47 +04:00
|
|
|
|
2013-11-19 18:19:39 +04:00
|
|
|
static const struct genl_multicast_group hsr_mcgrps[] = {
|
|
|
|
{ .name = "hsr-network", },
|
2013-10-31 00:10:47 +04:00
|
|
|
};
|
|
|
|
|
|
|
|
/* This is called if for some node with MAC address addr, we only get frames
|
|
|
|
* over one of the slave interfaces. This would indicate an open network ring
|
|
|
|
* (i.e. a link has failed somewhere).
|
|
|
|
*/
|
2014-07-05 01:34:38 +04:00
|
|
|
void hsr_nl_ringerror(struct hsr_priv *hsr, unsigned char addr[ETH_ALEN],
|
2014-07-05 01:38:05 +04:00
|
|
|
struct hsr_port *port)
|
2013-10-31 00:10:47 +04:00
|
|
|
{
|
|
|
|
struct sk_buff *skb;
|
|
|
|
void *msg_head;
|
2014-07-05 01:38:05 +04:00
|
|
|
struct hsr_port *master;
|
2013-10-31 00:10:47 +04:00
|
|
|
int res;
|
|
|
|
|
|
|
|
skb = genlmsg_new(NLMSG_GOODSIZE, GFP_ATOMIC);
|
|
|
|
if (!skb)
|
|
|
|
goto fail;
|
|
|
|
|
2019-04-05 20:31:23 +03:00
|
|
|
msg_head = genlmsg_put(skb, 0, 0, &hsr_genl_family, 0,
|
|
|
|
HSR_C_RING_ERROR);
|
2013-10-31 00:10:47 +04:00
|
|
|
if (!msg_head)
|
|
|
|
goto nla_put_failure;
|
|
|
|
|
|
|
|
res = nla_put(skb, HSR_A_NODE_ADDR, ETH_ALEN, addr);
|
|
|
|
if (res < 0)
|
|
|
|
goto nla_put_failure;
|
|
|
|
|
2014-07-05 01:38:05 +04:00
|
|
|
res = nla_put_u32(skb, HSR_A_IFINDEX, port->dev->ifindex);
|
2013-10-31 00:10:47 +04:00
|
|
|
if (res < 0)
|
|
|
|
goto nla_put_failure;
|
|
|
|
|
|
|
|
genlmsg_end(skb, msg_head);
|
2013-11-19 18:19:39 +04:00
|
|
|
genlmsg_multicast(&hsr_genl_family, skb, 0, 0, GFP_ATOMIC);
|
2013-10-31 00:10:47 +04:00
|
|
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
nla_put_failure:
|
|
|
|
kfree_skb(skb);
|
|
|
|
|
|
|
|
fail:
|
2014-07-05 01:38:05 +04:00
|
|
|
rcu_read_lock();
|
|
|
|
master = hsr_port_get_hsr(hsr, HSR_PT_MASTER);
|
|
|
|
netdev_warn(master->dev, "Could not send HSR ring error message\n");
|
|
|
|
rcu_read_unlock();
|
2013-10-31 00:10:47 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
/* This is called when we haven't heard from the node with MAC address addr for
|
|
|
|
* some time (just before the node is removed from the node table/list).
|
|
|
|
*/
|
2014-07-05 01:34:38 +04:00
|
|
|
void hsr_nl_nodedown(struct hsr_priv *hsr, unsigned char addr[ETH_ALEN])
|
2013-10-31 00:10:47 +04:00
|
|
|
{
|
|
|
|
struct sk_buff *skb;
|
|
|
|
void *msg_head;
|
2014-07-05 01:38:05 +04:00
|
|
|
struct hsr_port *master;
|
2013-10-31 00:10:47 +04:00
|
|
|
int res;
|
|
|
|
|
|
|
|
skb = genlmsg_new(NLMSG_GOODSIZE, GFP_ATOMIC);
|
|
|
|
if (!skb)
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
msg_head = genlmsg_put(skb, 0, 0, &hsr_genl_family, 0, HSR_C_NODE_DOWN);
|
|
|
|
if (!msg_head)
|
|
|
|
goto nla_put_failure;
|
|
|
|
|
|
|
|
res = nla_put(skb, HSR_A_NODE_ADDR, ETH_ALEN, addr);
|
|
|
|
if (res < 0)
|
|
|
|
goto nla_put_failure;
|
|
|
|
|
|
|
|
genlmsg_end(skb, msg_head);
|
2013-11-19 18:19:39 +04:00
|
|
|
genlmsg_multicast(&hsr_genl_family, skb, 0, 0, GFP_ATOMIC);
|
2013-10-31 00:10:47 +04:00
|
|
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
nla_put_failure:
|
|
|
|
kfree_skb(skb);
|
|
|
|
|
|
|
|
fail:
|
2014-07-05 01:38:05 +04:00
|
|
|
rcu_read_lock();
|
|
|
|
master = hsr_port_get_hsr(hsr, HSR_PT_MASTER);
|
|
|
|
netdev_warn(master->dev, "Could not send HSR node down\n");
|
|
|
|
rcu_read_unlock();
|
2013-10-31 00:10:47 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
/* HSR_C_GET_NODE_STATUS lets userspace query the internal HSR node table
|
|
|
|
* about the status of a specific node in the network, defined by its MAC
|
|
|
|
* address.
|
|
|
|
*
|
|
|
|
* Input: hsr ifindex, node mac address
|
|
|
|
* Output: hsr ifindex, node mac address (copied from request),
|
|
|
|
* age of latest frame from node over slave 1, slave 2 [ms]
|
|
|
|
*/
|
|
|
|
static int hsr_get_node_status(struct sk_buff *skb_in, struct genl_info *info)
|
|
|
|
{
|
|
|
|
/* For receiving */
|
|
|
|
struct nlattr *na;
|
2014-07-05 01:38:05 +04:00
|
|
|
struct net_device *hsr_dev;
|
2013-10-31 00:10:47 +04:00
|
|
|
|
|
|
|
/* For sending */
|
|
|
|
struct sk_buff *skb_out;
|
|
|
|
void *msg_head;
|
2014-07-05 01:34:38 +04:00
|
|
|
struct hsr_priv *hsr;
|
2014-07-05 01:38:05 +04:00
|
|
|
struct hsr_port *port;
|
2013-10-31 00:10:47 +04:00
|
|
|
unsigned char hsr_node_addr_b[ETH_ALEN];
|
|
|
|
int hsr_node_if1_age;
|
|
|
|
u16 hsr_node_if1_seq;
|
|
|
|
int hsr_node_if2_age;
|
|
|
|
u16 hsr_node_if2_seq;
|
|
|
|
int addr_b_ifindex;
|
|
|
|
int res;
|
|
|
|
|
|
|
|
if (!info)
|
|
|
|
goto invalid;
|
|
|
|
|
|
|
|
na = info->attrs[HSR_A_IFINDEX];
|
|
|
|
if (!na)
|
|
|
|
goto invalid;
|
|
|
|
na = info->attrs[HSR_A_NODE_ADDR];
|
|
|
|
if (!na)
|
|
|
|
goto invalid;
|
|
|
|
|
|
|
|
hsr_dev = __dev_get_by_index(genl_info_net(info),
|
2019-04-05 20:31:23 +03:00
|
|
|
nla_get_u32(info->attrs[HSR_A_IFINDEX]));
|
2013-10-31 00:10:47 +04:00
|
|
|
if (!hsr_dev)
|
|
|
|
goto invalid;
|
|
|
|
if (!is_hsr_master(hsr_dev))
|
|
|
|
goto invalid;
|
|
|
|
|
|
|
|
/* Send reply */
|
|
|
|
skb_out = genlmsg_new(NLMSG_GOODSIZE, GFP_KERNEL);
|
|
|
|
if (!skb_out) {
|
|
|
|
res = -ENOMEM;
|
|
|
|
goto fail;
|
|
|
|
}
|
|
|
|
|
|
|
|
msg_head = genlmsg_put(skb_out, NETLINK_CB(skb_in).portid,
|
2019-04-05 20:31:26 +03:00
|
|
|
info->snd_seq, &hsr_genl_family, 0,
|
|
|
|
HSR_C_SET_NODE_STATUS);
|
2013-10-31 00:10:47 +04:00
|
|
|
if (!msg_head) {
|
|
|
|
res = -ENOMEM;
|
|
|
|
goto nla_put_failure;
|
|
|
|
}
|
|
|
|
|
|
|
|
res = nla_put_u32(skb_out, HSR_A_IFINDEX, hsr_dev->ifindex);
|
|
|
|
if (res < 0)
|
|
|
|
goto nla_put_failure;
|
|
|
|
|
2014-07-05 01:34:38 +04:00
|
|
|
hsr = netdev_priv(hsr_dev);
|
|
|
|
res = hsr_get_node_data(hsr,
|
2019-04-05 20:31:23 +03:00
|
|
|
(unsigned char *)
|
|
|
|
nla_data(info->attrs[HSR_A_NODE_ADDR]),
|
|
|
|
hsr_node_addr_b,
|
|
|
|
&addr_b_ifindex,
|
|
|
|
&hsr_node_if1_age,
|
|
|
|
&hsr_node_if1_seq,
|
|
|
|
&hsr_node_if2_age,
|
|
|
|
&hsr_node_if2_seq);
|
2013-10-31 00:10:47 +04:00
|
|
|
if (res < 0)
|
2013-11-14 23:12:54 +04:00
|
|
|
goto nla_put_failure;
|
2013-10-31 00:10:47 +04:00
|
|
|
|
|
|
|
res = nla_put(skb_out, HSR_A_NODE_ADDR, ETH_ALEN,
|
2019-04-05 20:31:26 +03:00
|
|
|
nla_data(info->attrs[HSR_A_NODE_ADDR]));
|
2013-10-31 00:10:47 +04:00
|
|
|
if (res < 0)
|
|
|
|
goto nla_put_failure;
|
|
|
|
|
|
|
|
if (addr_b_ifindex > -1) {
|
|
|
|
res = nla_put(skb_out, HSR_A_NODE_ADDR_B, ETH_ALEN,
|
2019-04-05 20:31:23 +03:00
|
|
|
hsr_node_addr_b);
|
2013-10-31 00:10:47 +04:00
|
|
|
if (res < 0)
|
|
|
|
goto nla_put_failure;
|
|
|
|
|
2019-04-05 20:31:23 +03:00
|
|
|
res = nla_put_u32(skb_out, HSR_A_ADDR_B_IFINDEX,
|
|
|
|
addr_b_ifindex);
|
2013-10-31 00:10:47 +04:00
|
|
|
if (res < 0)
|
|
|
|
goto nla_put_failure;
|
|
|
|
}
|
|
|
|
|
|
|
|
res = nla_put_u32(skb_out, HSR_A_IF1_AGE, hsr_node_if1_age);
|
|
|
|
if (res < 0)
|
|
|
|
goto nla_put_failure;
|
|
|
|
res = nla_put_u16(skb_out, HSR_A_IF1_SEQ, hsr_node_if1_seq);
|
|
|
|
if (res < 0)
|
|
|
|
goto nla_put_failure;
|
2014-07-05 01:37:27 +04:00
|
|
|
rcu_read_lock();
|
2014-07-05 01:38:05 +04:00
|
|
|
port = hsr_port_get_hsr(hsr, HSR_PT_SLAVE_A);
|
|
|
|
if (port)
|
|
|
|
res = nla_put_u32(skb_out, HSR_A_IF1_IFINDEX,
|
|
|
|
port->dev->ifindex);
|
2014-07-05 01:37:27 +04:00
|
|
|
rcu_read_unlock();
|
2013-10-31 00:10:47 +04:00
|
|
|
if (res < 0)
|
|
|
|
goto nla_put_failure;
|
|
|
|
|
|
|
|
res = nla_put_u32(skb_out, HSR_A_IF2_AGE, hsr_node_if2_age);
|
|
|
|
if (res < 0)
|
|
|
|
goto nla_put_failure;
|
|
|
|
res = nla_put_u16(skb_out, HSR_A_IF2_SEQ, hsr_node_if2_seq);
|
|
|
|
if (res < 0)
|
|
|
|
goto nla_put_failure;
|
2014-07-05 01:37:27 +04:00
|
|
|
rcu_read_lock();
|
2014-07-05 01:38:05 +04:00
|
|
|
port = hsr_port_get_hsr(hsr, HSR_PT_SLAVE_B);
|
|
|
|
if (port)
|
|
|
|
res = nla_put_u32(skb_out, HSR_A_IF2_IFINDEX,
|
|
|
|
port->dev->ifindex);
|
2014-07-05 01:37:27 +04:00
|
|
|
rcu_read_unlock();
|
|
|
|
if (res < 0)
|
|
|
|
goto nla_put_failure;
|
2013-10-31 00:10:47 +04:00
|
|
|
|
|
|
|
genlmsg_end(skb_out, msg_head);
|
|
|
|
genlmsg_unicast(genl_info_net(info), skb_out, info->snd_portid);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
invalid:
|
2017-04-12 15:34:04 +03:00
|
|
|
netlink_ack(skb_in, nlmsg_hdr(skb_in), -EINVAL, NULL);
|
2013-10-31 00:10:47 +04:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
nla_put_failure:
|
|
|
|
kfree_skb(skb_out);
|
|
|
|
/* Fall through */
|
|
|
|
|
|
|
|
fail:
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2014-07-05 01:41:03 +04:00
|
|
|
/* Get a list of MacAddressA of all nodes known to this node (including self).
|
2013-10-31 00:10:47 +04:00
|
|
|
*/
|
|
|
|
static int hsr_get_node_list(struct sk_buff *skb_in, struct genl_info *info)
|
|
|
|
{
|
|
|
|
/* For receiving */
|
|
|
|
struct nlattr *na;
|
|
|
|
struct net_device *hsr_dev;
|
|
|
|
|
|
|
|
/* For sending */
|
|
|
|
struct sk_buff *skb_out;
|
|
|
|
void *msg_head;
|
2014-07-05 01:34:38 +04:00
|
|
|
struct hsr_priv *hsr;
|
2013-10-31 00:10:47 +04:00
|
|
|
void *pos;
|
|
|
|
unsigned char addr[ETH_ALEN];
|
|
|
|
int res;
|
|
|
|
|
|
|
|
if (!info)
|
|
|
|
goto invalid;
|
|
|
|
|
|
|
|
na = info->attrs[HSR_A_IFINDEX];
|
|
|
|
if (!na)
|
|
|
|
goto invalid;
|
|
|
|
|
|
|
|
hsr_dev = __dev_get_by_index(genl_info_net(info),
|
|
|
|
nla_get_u32(info->attrs[HSR_A_IFINDEX]));
|
|
|
|
if (!hsr_dev)
|
|
|
|
goto invalid;
|
|
|
|
if (!is_hsr_master(hsr_dev))
|
|
|
|
goto invalid;
|
|
|
|
|
|
|
|
/* Send reply */
|
|
|
|
skb_out = genlmsg_new(NLMSG_GOODSIZE, GFP_KERNEL);
|
|
|
|
if (!skb_out) {
|
|
|
|
res = -ENOMEM;
|
|
|
|
goto fail;
|
|
|
|
}
|
|
|
|
|
|
|
|
msg_head = genlmsg_put(skb_out, NETLINK_CB(skb_in).portid,
|
2019-04-05 20:31:26 +03:00
|
|
|
info->snd_seq, &hsr_genl_family, 0,
|
|
|
|
HSR_C_SET_NODE_LIST);
|
2013-10-31 00:10:47 +04:00
|
|
|
if (!msg_head) {
|
|
|
|
res = -ENOMEM;
|
|
|
|
goto nla_put_failure;
|
|
|
|
}
|
|
|
|
|
|
|
|
res = nla_put_u32(skb_out, HSR_A_IFINDEX, hsr_dev->ifindex);
|
|
|
|
if (res < 0)
|
|
|
|
goto nla_put_failure;
|
|
|
|
|
2014-07-05 01:34:38 +04:00
|
|
|
hsr = netdev_priv(hsr_dev);
|
2013-10-31 00:10:47 +04:00
|
|
|
|
|
|
|
rcu_read_lock();
|
2014-07-05 01:34:38 +04:00
|
|
|
pos = hsr_get_next_node(hsr, NULL, addr);
|
2013-10-31 00:10:47 +04:00
|
|
|
while (pos) {
|
|
|
|
res = nla_put(skb_out, HSR_A_NODE_ADDR, ETH_ALEN, addr);
|
|
|
|
if (res < 0) {
|
|
|
|
rcu_read_unlock();
|
|
|
|
goto nla_put_failure;
|
|
|
|
}
|
2014-07-05 01:34:38 +04:00
|
|
|
pos = hsr_get_next_node(hsr, pos, addr);
|
2013-10-31 00:10:47 +04:00
|
|
|
}
|
|
|
|
rcu_read_unlock();
|
|
|
|
|
|
|
|
genlmsg_end(skb_out, msg_head);
|
|
|
|
genlmsg_unicast(genl_info_net(info), skb_out, info->snd_portid);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
invalid:
|
2017-04-12 15:34:04 +03:00
|
|
|
netlink_ack(skb_in, nlmsg_hdr(skb_in), -EINVAL, NULL);
|
2013-10-31 00:10:47 +04:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
nla_put_failure:
|
|
|
|
kfree_skb(skb_out);
|
|
|
|
/* Fall through */
|
|
|
|
|
|
|
|
fail:
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2013-11-14 20:14:46 +04:00
|
|
|
static const struct genl_ops hsr_ops[] = {
|
2013-11-14 20:14:40 +04:00
|
|
|
{
|
|
|
|
.cmd = HSR_C_GET_NODE_STATUS,
|
2019-04-26 15:07:31 +03:00
|
|
|
.validate = GENL_DONT_VALIDATE_STRICT | GENL_DONT_VALIDATE_DUMP,
|
2013-11-14 20:14:40 +04:00
|
|
|
.flags = 0,
|
|
|
|
.doit = hsr_get_node_status,
|
|
|
|
.dumpit = NULL,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
.cmd = HSR_C_GET_NODE_LIST,
|
2019-04-26 15:07:31 +03:00
|
|
|
.validate = GENL_DONT_VALIDATE_STRICT | GENL_DONT_VALIDATE_DUMP,
|
2013-11-14 20:14:40 +04:00
|
|
|
.flags = 0,
|
|
|
|
.doit = hsr_get_node_list,
|
|
|
|
.dumpit = NULL,
|
|
|
|
},
|
2013-10-31 00:10:47 +04:00
|
|
|
};
|
|
|
|
|
2016-10-24 15:40:05 +03:00
|
|
|
static struct genl_family hsr_genl_family __ro_after_init = {
|
2016-10-24 15:40:03 +03:00
|
|
|
.hdrsize = 0,
|
|
|
|
.name = "HSR",
|
|
|
|
.version = 1,
|
|
|
|
.maxattr = HSR_A_MAX,
|
genetlink: make policy common to family
Since maxattr is common, the policy can't really differ sanely,
so make it common as well.
The only user that did in fact manage to make a non-common policy
is taskstats, which has to be really careful about it (since it's
still using a common maxattr!). This is no longer supported, but
we can fake it using pre_doit.
This reduces the size of e.g. nl80211.o (which has lots of commands):
text data bss dec hex filename
398745 14323 2240 415308 6564c net/wireless/nl80211.o (before)
397913 14331 2240 414484 65314 net/wireless/nl80211.o (after)
--------------------------------
-832 +8 0 -824
Which is obviously just 8 bytes for each command, and an added 8
bytes for the new policy pointer. I'm not sure why the ops list is
counted as .text though.
Most of the code transformations were done using the following spatch:
@ops@
identifier OPS;
expression POLICY;
@@
struct genl_ops OPS[] = {
...,
{
- .policy = POLICY,
},
...
};
@@
identifier ops.OPS;
expression ops.POLICY;
identifier fam;
expression M;
@@
struct genl_family fam = {
.ops = OPS,
.maxattr = M,
+ .policy = POLICY,
...
};
This also gets rid of devlink_nl_cmd_region_read_dumpit() accessing
the cb->data as ops, which we want to change in a later genl patch.
Signed-off-by: Johannes Berg <johannes.berg@intel.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2019-03-22 00:51:02 +03:00
|
|
|
.policy = hsr_genl_policy,
|
2016-10-24 15:40:03 +03:00
|
|
|
.module = THIS_MODULE,
|
|
|
|
.ops = hsr_ops,
|
|
|
|
.n_ops = ARRAY_SIZE(hsr_ops),
|
|
|
|
.mcgrps = hsr_mcgrps,
|
|
|
|
.n_mcgrps = ARRAY_SIZE(hsr_mcgrps),
|
|
|
|
};
|
|
|
|
|
2013-10-31 00:10:47 +04:00
|
|
|
int __init hsr_netlink_init(void)
|
|
|
|
{
|
|
|
|
int rc;
|
|
|
|
|
|
|
|
rc = rtnl_link_register(&hsr_link_ops);
|
|
|
|
if (rc)
|
|
|
|
goto fail_rtnl_link_register;
|
|
|
|
|
2016-10-24 15:40:03 +03:00
|
|
|
rc = genl_register_family(&hsr_genl_family);
|
2013-10-31 00:10:47 +04:00
|
|
|
if (rc)
|
|
|
|
goto fail_genl_register_family;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
fail_genl_register_family:
|
|
|
|
rtnl_link_unregister(&hsr_link_ops);
|
|
|
|
fail_rtnl_link_register:
|
|
|
|
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
void __exit hsr_netlink_exit(void)
|
|
|
|
{
|
|
|
|
genl_unregister_family(&hsr_genl_family);
|
|
|
|
rtnl_link_unregister(&hsr_link_ops);
|
|
|
|
}
|
|
|
|
|
|
|
|
MODULE_ALIAS_RTNL_LINK("hsr");
|