lists.openwall.net   lists  /  announce  owl-users  owl-dev  john-users  john-dev  passwdqc-users  yescrypt  popa3d-users  /  oss-security  kernel-hardening  musl  sabotage  tlsify  passwords  /  crypt-dev  xvendor  /  Bugtraq  Full-Disclosure  linux-kernel  linux-netdev  linux-ext4  linux-hardening  linux-cve-announce  PHC 
Open Source and information security mailing list archives
 
Hash Suite: Windows password security audit tool. GUI, reports in PDF.
[<prev] [next>] [<thread-prev] [thread-next>] [day] [month] [year] [list]
Message-ID: <20140121133426.GC3015@minipsycho.orion>
Date:	Tue, 21 Jan 2014 14:34:26 +0100
From:	Jiri Pirko <jiri@...nulli.us>
To:	Scott Feldman <sfeldma@...ulusnetworks.com>
Cc:	vfalico@...hat.com, fubar@...ibm.com, andy@...yhouse.net,
	netdev@...r.kernel.org, roopa@...ulusnetworks.com,
	shm@...ulusnetworks.com, dingtianhong@...wei.com
Subject: Re: [PATCH net-next v2 2/2] bonding: add netlink attributes to slave
 link dev

Fri, Jan 17, 2014 at 07:57:56AM CET, sfeldma@...ulusnetworks.com wrote:
>If link is IFF_SLAVE, extend link dev netlink attributes to include
>slave attributes with new IFLA_SLAVE nest.  Add netlink notification
>(RTM_NEWLINK) when slave status changes from backup to active, or
>visa-versa.
>
>Adds new ndo_get_slave op to net_device_ops to fill skb with IFLA_SLAVE
>attributes.  Currently only used by bonding driver, but could be
>used by other aggregating devices with slaves.
>
>Signed-off-by: Scott Feldman <sfeldma@...ulusnetworks.com>
>---
> drivers/net/bonding/bond_main.c    |    1 +
> drivers/net/bonding/bond_netlink.c |   36 ++++++++++++++++++++++++
> drivers/net/bonding/bonding.h      |   11 ++++++-
> include/linux/netdevice.h          |    5 +++
> include/uapi/linux/if_link.h       |   13 +++++++++
> net/core/rtnetlink.c               |   54 ++++++++++++++++++++++++++++++++++++
> 6 files changed, 118 insertions(+), 2 deletions(-)
>
>diff --git a/drivers/net/bonding/bond_main.c b/drivers/net/bonding/bond_main.c
>index df85cec..3220b48 100644
>--- a/drivers/net/bonding/bond_main.c
>+++ b/drivers/net/bonding/bond_main.c
>@@ -3883,6 +3883,7 @@ static const struct net_device_ops bond_netdev_ops = {
> #endif
> 	.ndo_add_slave		= bond_enslave,
> 	.ndo_del_slave		= bond_release,
>+	.ndo_get_slave		= bond_get_slave,
> 	.ndo_fix_features	= bond_fix_features,
> };
> 
>diff --git a/drivers/net/bonding/bond_netlink.c b/drivers/net/bonding/bond_netlink.c
>index 555c783..21c6488 100644
>--- a/drivers/net/bonding/bond_netlink.c
>+++ b/drivers/net/bonding/bond_netlink.c
>@@ -22,6 +22,42 @@
> #include <linux/reciprocal_div.h>
> #include "bonding.h"
> 
>+int bond_get_slave(struct net_device *slave_dev, struct sk_buff *skb)
>+{
>+	struct slave *slave = bond_slave_get_rtnl(slave_dev);
>+	const struct aggregator *agg;
>+
>+	if (nla_put_u8(skb, IFLA_SLAVE_STATE, bond_slave_state(slave)))
>+		goto nla_put_failure;
>+
>+	if (nla_put_u8(skb, IFLA_SLAVE_MII_STATUS, slave->link))
>+		goto nla_put_failure;
>+
>+	if (nla_put_u32(skb, IFLA_SLAVE_LINK_FAILURE_COUNT,
>+			slave->link_failure_count))
>+		goto nla_put_failure;
>+
>+	if (nla_put(skb, IFLA_SLAVE_PERM_HWADDR,
>+		    slave_dev->addr_len, slave->perm_hwaddr))
>+		goto nla_put_failure;
>+
>+	if (nla_put_u16(skb, IFLA_SLAVE_QUEUE_ID, slave->queue_id))
>+		goto nla_put_failure;
>+
>+	if (slave->bond->params.mode == BOND_MODE_8023AD) {
>+		agg = SLAVE_AD_INFO(slave).port.aggregator;
>+		if (agg)
>+			if (nla_put_u16(skb, IFLA_SLAVE_AD_AGGREGATOR_ID,
>+					agg->aggregator_identifier))
>+				goto nla_put_failure;
>+	}
>+
>+	return 0;
>+
>+nla_put_failure:
>+	return -EMSGSIZE;
>+}
>+
> static const struct nla_policy bond_policy[IFLA_BOND_MAX + 1] = {
> 	[IFLA_BOND_MODE]		= { .type = NLA_U8 },
> 	[IFLA_BOND_ACTIVE_SLAVE]	= { .type = NLA_U32 },
>diff --git a/drivers/net/bonding/bonding.h b/drivers/net/bonding/bonding.h
>index 309757d..8a935f8 100644
>--- a/drivers/net/bonding/bonding.h
>+++ b/drivers/net/bonding/bonding.h
>@@ -285,12 +285,18 @@ static inline bool bond_is_lb(const struct bonding *bond)
> 
> static inline void bond_set_active_slave(struct slave *slave)
> {
>-	slave->backup = 0;
>+	if (slave->backup) {
>+		slave->backup = 0;
>+		rtmsg_ifinfo(RTM_NEWLINK, slave->dev, 0, GFP_KERNEL);
>+	}
> }
> 
> static inline void bond_set_backup_slave(struct slave *slave)
> {
>-	slave->backup = 1;
>+	if (!slave->backup) {
>+		slave->backup = 1;
>+		rtmsg_ifinfo(RTM_NEWLINK, slave->dev, 0, GFP_KERNEL);
>+	}
> }
> 
> static inline int bond_slave_state(struct slave *slave)
>@@ -426,6 +432,7 @@ int bond_sysfs_slave_add(struct slave *slave);
> void bond_sysfs_slave_del(struct slave *slave);
> int bond_enslave(struct net_device *bond_dev, struct net_device *slave_dev);
> int bond_release(struct net_device *bond_dev, struct net_device *slave_dev);
>+int bond_get_slave(struct net_device *slave_dev, struct sk_buff *skb);
> int bond_xmit_hash(struct bonding *bond, struct sk_buff *skb, int count);
> int bond_parse_parm(const char *mode_arg, const struct bond_parm_tbl *tbl);
> int bond_parm_tbl_lookup(int mode, const struct bond_parm_tbl *tbl);
>diff --git a/include/linux/netdevice.h b/include/linux/netdevice.h
>index d7668b88..22c8698 100644
>--- a/include/linux/netdevice.h
>+++ b/include/linux/netdevice.h
>@@ -908,6 +908,9 @@ struct netdev_phys_port_id {
>  * int (*ndo_del_slave)(struct net_device *dev, struct net_device *slave_dev);
>  *	Called to release previously enslaved netdev.
>  *
>+ * int (*ndo_get_slave)(struct net_device *slave_dev, struct sk_buff *skb);
>+ *	Called to fill netlink skb with slave info.
>+ *
>  *      Feature/offload setting functions.
>  * netdev_features_t (*ndo_fix_features)(struct net_device *dev,
>  *		netdev_features_t features);
>@@ -1080,6 +1083,8 @@ struct net_device_ops {
> 						 struct net_device *slave_dev);
> 	int			(*ndo_del_slave)(struct net_device *dev,
> 						 struct net_device *slave_dev);
>+	int			(*ndo_get_slave)(struct net_device *slave_dev,
>+						 struct sk_buff *skb);
> 	netdev_features_t	(*ndo_fix_features)(struct net_device *dev,
> 						    netdev_features_t features);
> 	int			(*ndo_set_features)(struct net_device *dev,
>diff --git a/include/uapi/linux/if_link.h b/include/uapi/linux/if_link.h
>index 3e6bd3c..ba2f3bf 100644
>--- a/include/uapi/linux/if_link.h
>+++ b/include/uapi/linux/if_link.h
>@@ -144,6 +144,7 @@ enum {
> 	IFLA_NUM_RX_QUEUES,
> 	IFLA_CARRIER,
> 	IFLA_PHYS_PORT_ID,
>+	IFLA_SLAVE,
> 	__IFLA_MAX
> };
> 
>@@ -368,6 +369,18 @@ enum {
> 
> #define IFLA_BOND_AD_INFO_MAX	(__IFLA_BOND_AD_INFO_MAX - 1)
> 
>+enum {
>+	IFLA_SLAVE_STATE,
>+	IFLA_SLAVE_MII_STATUS,
>+	IFLA_SLAVE_LINK_FAILURE_COUNT,
>+	IFLA_SLAVE_PERM_HWADDR,
>+	IFLA_SLAVE_QUEUE_ID,
>+	IFLA_SLAVE_AD_AGGREGATOR_ID,
>+	__IFLA_SLAVE_MAX,
>+};


The names are not right in my opinion. "BOND" should be mentioned there.

>+
>+#define IFLA_SLAVE_MAX	(__IFLA_SLAVE_MAX - 1)
>+
> /* SR-IOV virtual function management section */
> 
> enum {
>diff --git a/net/core/rtnetlink.c b/net/core/rtnetlink.c
>index e6e7d58..4f85de7 100644
>--- a/net/core/rtnetlink.c
>+++ b/net/core/rtnetlink.c
>@@ -721,6 +721,28 @@ static size_t rtnl_port_size(const struct net_device *dev)
> 		return port_self_size;
> }
> 
>+static size_t rtnl_bond_slave_size(const struct net_device *dev)
>+{
>+	struct net_device *bond;
>+	size_t slave_size =
>+		nla_total_size(sizeof(struct nlattr)) +	/* IFLA_SLAVE */
>+		nla_total_size(1) +	/* IFLA_SLAVE_STATE */
>+		nla_total_size(1) +	/* IFLA_SLAVE_MII_STATUS */
>+		nla_total_size(4) +	/* IFLA_SLAVE_LINK_FAILURE_COUNT */
>+		nla_total_size(MAX_ADDR_LEN) +	/* IFLA_SLAVE_PERM_HWADDR */
>+		nla_total_size(2) +	/* IFLA_SLAVE_QUEUE_ID */
>+		nla_total_size(2) +	/* IFLA_SLAVE_AD_AGGREGATOR_ID */
>+		0;
>+
>+	if (netif_is_bond_slave((struct net_device *)dev)) {
>+		bond = netdev_master_upper_dev_get((struct net_device *)dev);
>+		if (bond && bond->netdev_ops->ndo_get_slave)
>+			return slave_size;
>+	}
>+
>+	return 0;
>+}
>+
> static noinline size_t if_nlmsg_size(const struct net_device *dev,
> 				     u32 ext_filter_mask)
> {
>@@ -750,6 +772,7 @@ static noinline size_t if_nlmsg_size(const struct net_device *dev,
> 	       + rtnl_port_size(dev) /* IFLA_VF_PORTS + IFLA_PORT_SELF */
> 	       + rtnl_link_get_size(dev) /* IFLA_LINKINFO */
> 	       + rtnl_link_get_af_size(dev) /* IFLA_AF_SPEC */
>+	       + rtnl_bond_slave_size(dev) /* IFLA_SLAVE */
> 	       + nla_total_size(MAX_PHYS_PORT_ID_LEN); /* IFLA_PHYS_PORT_ID */
> }
> 
>@@ -847,6 +870,34 @@ static int rtnl_phys_port_id_fill(struct sk_buff *skb, struct net_device *dev)
> 	return 0;
> }
> 
>+static size_t rtnl_bond_slave_fill(struct sk_buff *skb, struct net_device *dev)
>+{
>+	struct net_device *bond;
>+	struct nlattr *nest;
>+	int err;
>+
>+	if (!netif_is_bond_slave(dev))
>+		return 0;
>+
>+	bond = netdev_master_upper_dev_get(dev);
>+	if (!bond || !bond->netdev_ops->ndo_get_slave)
>+		return 0;
>+
>+	nest = nla_nest_start(skb, IFLA_SLAVE);
>+	if (!nest)
>+		return -EMSGSIZE;
>+
>+	err = bond->netdev_ops->ndo_get_slave(dev, skb);
>+	if (err) {
>+		nla_nest_cancel(skb, nest);
>+		return (err == -EMSGSIZE) ? err : 0;
>+	}
>+
>+	nla_nest_end(skb, nest);
>+
>+	return 0;
>+}
>+
> static int rtnl_fill_ifinfo(struct sk_buff *skb, struct net_device *dev,
> 			    int type, u32 pid, u32 seq, u32 change,
> 			    unsigned int flags, u32 ext_filter_mask)
>@@ -1001,6 +1052,9 @@ static int rtnl_fill_ifinfo(struct sk_buff *skb, struct net_device *dev,
> 	if (rtnl_port_fill(skb, dev))
> 		goto nla_put_failure;
> 
>+	if (rtnl_bond_slave_fill(skb, dev))
>+		goto nla_put_failure;
>+

I must say I do not like this at all. This should be done in a generic
way. By a callback registered by bonding and possibly other master-slave
device types.

I have something in mind, will try to prepare patch soon.

Jiri


> 	if (dev->rtnl_link_ops) {
> 		if (rtnl_link_fill(skb, dev) < 0)
> 			goto nla_put_failure;
>
>--
>To unsubscribe from this list: send the line "unsubscribe netdev" in
>the body of a message to majordomo@...r.kernel.org
>More majordomo info at  http://vger.kernel.org/majordomo-info.html
--
To unsubscribe from this list: send the line "unsubscribe netdev" in
the body of a message to majordomo@...r.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html

Powered by blists - more mailing lists

Powered by Openwall GNU/*/Linux Powered by OpenVZ