diff options
| author | Mark Brown <broonie@kernel.org> | 2016-02-09 21:20:39 +0300 | 
|---|---|---|
| committer | Mark Brown <broonie@kernel.org> | 2016-02-09 21:20:39 +0300 | 
| commit | fcdcc79628a1919bde9acf239e364f65bab6327c (patch) | |
| tree | 5499be387cf3028c90ac083b1cf866ebed7bf7e0 /net/mpls/af_mpls.c | |
| parent | 7a8d44bc89e5cddcd5c0704a11a90484d36ba6ba (diff) | |
| parent | a0a90718f18264dc904d34a580f332006f5561e9 (diff) | |
| download | linux-fcdcc79628a1919bde9acf239e364f65bab6327c.tar.xz | |
Merge branch 'topic/acpi' of git://git.kernel.org/pub/scm/linux/kernel/git/broonie/spi into spi-pxa2xx
Diffstat (limited to 'net/mpls/af_mpls.c')
| -rw-r--r-- | net/mpls/af_mpls.c | 228 | 
1 files changed, 188 insertions, 40 deletions
| diff --git a/net/mpls/af_mpls.c b/net/mpls/af_mpls.c index c70d750148b6..b18c5ed42d95 100644 --- a/net/mpls/af_mpls.c +++ b/net/mpls/af_mpls.c @@ -27,6 +27,8 @@   */  #define MAX_MP_SELECT_LABELS 4 +#define MPLS_NEIGH_TABLE_UNSPEC (NEIGH_LINK_TABLE + 1) +  static int zero = 0;  static int label_limit = (1 << 20) - 1; @@ -96,22 +98,15 @@ bool mpls_pkt_too_big(const struct sk_buff *skb, unsigned int mtu)  }  EXPORT_SYMBOL_GPL(mpls_pkt_too_big); -static struct mpls_nh *mpls_select_multipath(struct mpls_route *rt, -					     struct sk_buff *skb, bool bos) +static u32 mpls_multipath_hash(struct mpls_route *rt, +			       struct sk_buff *skb, bool bos)  {  	struct mpls_entry_decoded dec;  	struct mpls_shim_hdr *hdr;  	bool eli_seen = false;  	int label_index; -	int nh_index = 0;  	u32 hash = 0; -	/* No need to look further into packet if there's only -	 * one path -	 */ -	if (rt->rt_nhn == 1) -		goto out; -  	for (label_index = 0; label_index < MAX_MP_SELECT_LABELS && !bos;  	     label_index++) {  		if (!pskb_may_pull(skb, sizeof(*hdr) * label_index)) @@ -165,7 +160,38 @@ static struct mpls_nh *mpls_select_multipath(struct mpls_route *rt,  		}  	} -	nh_index = hash % rt->rt_nhn; +	return hash; +} + +static struct mpls_nh *mpls_select_multipath(struct mpls_route *rt, +					     struct sk_buff *skb, bool bos) +{ +	int alive = ACCESS_ONCE(rt->rt_nhn_alive); +	u32 hash = 0; +	int nh_index = 0; +	int n = 0; + +	/* No need to look further into packet if there's only +	 * one path +	 */ +	if (rt->rt_nhn == 1) +		goto out; + +	if (alive <= 0) +		return NULL; + +	hash = mpls_multipath_hash(rt, skb, bos); +	nh_index = hash % alive; +	if (alive == rt->rt_nhn) +		goto out; +	for_nexthops(rt) { +		if (nh->nh_flags & (RTNH_F_DEAD | RTNH_F_LINKDOWN)) +			continue; +		if (n == nh_index) +			return nh; +		n++; +	} endfor_nexthops(rt); +  out:  	return &rt->rt_nh[nh_index];  } @@ -317,7 +343,13 @@ static int mpls_forward(struct sk_buff *skb, struct net_device *dev,  		}  	} -	err = neigh_xmit(nh->nh_via_table, out_dev, mpls_nh_via(rt, nh), skb); +	/* If via wasn't specified then send out using device address */ +	if (nh->nh_via_table == MPLS_NEIGH_TABLE_UNSPEC) +		err = neigh_xmit(NEIGH_LINK_TABLE, out_dev, +				 out_dev->dev_addr, skb); +	else +		err = neigh_xmit(nh->nh_via_table, out_dev, +				 mpls_nh_via(rt, nh), skb);  	if (err)  		net_dbg_ratelimited("%s: packet transmission failed: %d\n",  				    __func__, err); @@ -365,6 +397,7 @@ static struct mpls_route *mpls_rt_alloc(int num_nh, u8 max_alen)  		     GFP_KERNEL);  	if (rt) {  		rt->rt_nhn = num_nh; +		rt->rt_nhn_alive = num_nh;  		rt->rt_max_alen = max_alen_aligned;  	} @@ -534,8 +567,22 @@ static int mpls_nh_assign_dev(struct net *net, struct mpls_route *rt,  	if (!mpls_dev_get(dev))  		goto errout; +	if ((nh->nh_via_table == NEIGH_LINK_TABLE) && +	    (dev->addr_len != nh->nh_via_alen)) +		goto errout; +  	RCU_INIT_POINTER(nh->nh_dev, dev); +	if (!(dev->flags & IFF_UP)) { +		nh->nh_flags |= RTNH_F_DEAD; +	} else { +		unsigned int flags; + +		flags = dev_get_flags(dev); +		if (!(flags & (IFF_RUNNING | IFF_LOWER_UP))) +			nh->nh_flags |= RTNH_F_LINKDOWN; +	} +  	return 0;  errout: @@ -570,6 +617,9 @@ static int mpls_nh_build_from_cfg(struct mpls_route_config *cfg,  	if (err)  		goto errout; +	if (nh->nh_flags & (RTNH_F_DEAD | RTNH_F_LINKDOWN)) +		rt->rt_nhn_alive--; +  	return 0;  errout: @@ -577,8 +627,8 @@ errout:  }  static int mpls_nh_build(struct net *net, struct mpls_route *rt, -			 struct mpls_nh *nh, int oif, -			 struct nlattr *via, struct nlattr *newdst) +			 struct mpls_nh *nh, int oif, struct nlattr *via, +			 struct nlattr *newdst)  {  	int err = -ENOMEM; @@ -592,10 +642,14 @@ static int mpls_nh_build(struct net *net, struct mpls_route *rt,  			goto errout;  	} -	err = nla_get_via(via, &nh->nh_via_alen, &nh->nh_via_table, -			  __mpls_nh_via(rt, nh)); -	if (err) -		goto errout; +	if (via) { +		err = nla_get_via(via, &nh->nh_via_alen, &nh->nh_via_table, +				  __mpls_nh_via(rt, nh)); +		if (err) +			goto errout; +	} else { +		nh->nh_via_table = MPLS_NEIGH_TABLE_UNSPEC; +	}  	err = mpls_nh_assign_dev(net, rt, nh, oif);  	if (err) @@ -677,15 +731,14 @@ static int mpls_nh_build_multi(struct mpls_route_config *cfg,  			nla_newdst = nla_find(attrs, attrlen, RTA_NEWDST);  		} -		if (!nla_via) -			goto errout; -  		err = mpls_nh_build(cfg->rc_nlinfo.nl_net, rt, nh, -				    rtnh->rtnh_ifindex, nla_via, -				    nla_newdst); +				    rtnh->rtnh_ifindex, nla_via, nla_newdst);  		if (err)  			goto errout; +		if (nh->nh_flags & (RTNH_F_DEAD | RTNH_F_LINKDOWN)) +			rt->rt_nhn_alive--; +  		rtnh = rtnh_next(rtnh, &remaining);  		nhs++;  	} endfor_nexthops(rt); @@ -875,34 +928,74 @@ free:  	return ERR_PTR(err);  } -static void mpls_ifdown(struct net_device *dev) +static void mpls_ifdown(struct net_device *dev, int event)  {  	struct mpls_route __rcu **platform_label;  	struct net *net = dev_net(dev); -	struct mpls_dev *mdev;  	unsigned index;  	platform_label = rtnl_dereference(net->mpls.platform_label);  	for (index = 0; index < net->mpls.platform_labels; index++) {  		struct mpls_route *rt = rtnl_dereference(platform_label[index]); +  		if (!rt)  			continue; -		for_nexthops(rt) { + +		change_nexthops(rt) {  			if (rtnl_dereference(nh->nh_dev) != dev)  				continue; -			nh->nh_dev = NULL; +			switch (event) { +			case NETDEV_DOWN: +			case NETDEV_UNREGISTER: +				nh->nh_flags |= RTNH_F_DEAD; +				/* fall through */ +			case NETDEV_CHANGE: +				nh->nh_flags |= RTNH_F_LINKDOWN; +				ACCESS_ONCE(rt->rt_nhn_alive) = rt->rt_nhn_alive - 1; +				break; +			} +			if (event == NETDEV_UNREGISTER) +				RCU_INIT_POINTER(nh->nh_dev, NULL);  		} endfor_nexthops(rt);  	} -	mdev = mpls_dev_get(dev); -	if (!mdev) -		return; -	mpls_dev_sysctl_unregister(mdev); +	return; +} + +static void mpls_ifup(struct net_device *dev, unsigned int nh_flags) +{ +	struct mpls_route __rcu **platform_label; +	struct net *net = dev_net(dev); +	unsigned index; +	int alive; + +	platform_label = rtnl_dereference(net->mpls.platform_label); +	for (index = 0; index < net->mpls.platform_labels; index++) { +		struct mpls_route *rt = rtnl_dereference(platform_label[index]); + +		if (!rt) +			continue; + +		alive = 0; +		change_nexthops(rt) { +			struct net_device *nh_dev = +				rtnl_dereference(nh->nh_dev); -	RCU_INIT_POINTER(dev->mpls_ptr, NULL); +			if (!(nh->nh_flags & nh_flags)) { +				alive++; +				continue; +			} +			if (nh_dev != dev) +				continue; +			alive++; +			nh->nh_flags &= ~nh_flags; +		} endfor_nexthops(rt); + +		ACCESS_ONCE(rt->rt_nhn_alive) = alive; +	} -	kfree_rcu(mdev, rcu); +	return;  }  static int mpls_dev_notify(struct notifier_block *this, unsigned long event, @@ -910,9 +1003,9 @@ static int mpls_dev_notify(struct notifier_block *this, unsigned long event,  {  	struct net_device *dev = netdev_notifier_info_to_dev(ptr);  	struct mpls_dev *mdev; +	unsigned int flags; -	switch(event) { -	case NETDEV_REGISTER: +	if (event == NETDEV_REGISTER) {  		/* For now just support ethernet devices */  		if ((dev->type == ARPHRD_ETHER) ||  		    (dev->type == ARPHRD_LOOPBACK)) { @@ -920,10 +1013,39 @@ static int mpls_dev_notify(struct notifier_block *this, unsigned long event,  			if (IS_ERR(mdev))  				return notifier_from_errno(PTR_ERR(mdev));  		} -		break; +		return NOTIFY_OK; +	} + +	mdev = mpls_dev_get(dev); +	if (!mdev) +		return NOTIFY_OK; +	switch (event) { +	case NETDEV_DOWN: +		mpls_ifdown(dev, event); +		break; +	case NETDEV_UP: +		flags = dev_get_flags(dev); +		if (flags & (IFF_RUNNING | IFF_LOWER_UP)) +			mpls_ifup(dev, RTNH_F_DEAD | RTNH_F_LINKDOWN); +		else +			mpls_ifup(dev, RTNH_F_DEAD); +		break; +	case NETDEV_CHANGE: +		flags = dev_get_flags(dev); +		if (flags & (IFF_RUNNING | IFF_LOWER_UP)) +			mpls_ifup(dev, RTNH_F_DEAD | RTNH_F_LINKDOWN); +		else +			mpls_ifdown(dev, event); +		break;  	case NETDEV_UNREGISTER: -		mpls_ifdown(dev); +		mpls_ifdown(dev, event); +		mdev = mpls_dev_get(dev); +		if (mdev) { +			mpls_dev_sysctl_unregister(mdev); +			RCU_INIT_POINTER(dev->mpls_ptr, NULL); +			kfree_rcu(mdev, rcu); +		}  		break;  	case NETDEV_CHANGENAME:  		mdev = mpls_dev_get(dev); @@ -1118,6 +1240,7 @@ static int rtm_to_route_config(struct sk_buff *skb,  struct nlmsghdr *nlh,  	cfg->rc_label		= LABEL_NOT_SPECIFIED;  	cfg->rc_protocol	= rtm->rtm_protocol; +	cfg->rc_via_table	= MPLS_NEIGH_TABLE_UNSPEC;  	cfg->rc_nlflags		= nlh->nlmsg_flags;  	cfg->rc_nlinfo.portid	= NETLINK_CB(skb).portid;  	cfg->rc_nlinfo.nlh	= nlh; @@ -1231,15 +1354,22 @@ static int mpls_dump_route(struct sk_buff *skb, u32 portid, u32 seq, int event,  		    nla_put_labels(skb, RTA_NEWDST, nh->nh_labels,  				   nh->nh_label))  			goto nla_put_failure; -		if (nla_put_via(skb, nh->nh_via_table, mpls_nh_via(rt, nh), +		if (nh->nh_via_table != MPLS_NEIGH_TABLE_UNSPEC && +		    nla_put_via(skb, nh->nh_via_table, mpls_nh_via(rt, nh),  				nh->nh_via_alen))  			goto nla_put_failure;  		dev = rtnl_dereference(nh->nh_dev);  		if (dev && nla_put_u32(skb, RTA_OIF, dev->ifindex))  			goto nla_put_failure; +		if (nh->nh_flags & RTNH_F_LINKDOWN) +			rtm->rtm_flags |= RTNH_F_LINKDOWN; +		if (nh->nh_flags & RTNH_F_DEAD) +			rtm->rtm_flags |= RTNH_F_DEAD;  	} else {  		struct rtnexthop *rtnh;  		struct nlattr *mp; +		int dead = 0; +		int linkdown = 0;  		mp = nla_nest_start(skb, RTA_MULTIPATH);  		if (!mp) @@ -1253,11 +1383,21 @@ static int mpls_dump_route(struct sk_buff *skb, u32 portid, u32 seq, int event,  			dev = rtnl_dereference(nh->nh_dev);  			if (dev)  				rtnh->rtnh_ifindex = dev->ifindex; +			if (nh->nh_flags & RTNH_F_LINKDOWN) { +				rtnh->rtnh_flags |= RTNH_F_LINKDOWN; +				linkdown++; +			} +			if (nh->nh_flags & RTNH_F_DEAD) { +				rtnh->rtnh_flags |= RTNH_F_DEAD; +				dead++; +			} +  			if (nh->nh_labels && nla_put_labels(skb, RTA_NEWDST,  							    nh->nh_labels,  							    nh->nh_label))  				goto nla_put_failure; -			if (nla_put_via(skb, nh->nh_via_table, +			if (nh->nh_via_table != MPLS_NEIGH_TABLE_UNSPEC && +			    nla_put_via(skb, nh->nh_via_table,  					mpls_nh_via(rt, nh),  					nh->nh_via_alen))  				goto nla_put_failure; @@ -1266,6 +1406,11 @@ static int mpls_dump_route(struct sk_buff *skb, u32 portid, u32 seq, int event,  			rtnh->rtnh_len = nlmsg_get_pos(skb) - (void *)rtnh;  		} endfor_nexthops(rt); +		if (linkdown == rt->rt_nhn) +			rtm->rtm_flags |= RTNH_F_LINKDOWN; +		if (dead == rt->rt_nhn) +			rtm->rtm_flags |= RTNH_F_DEAD; +  		nla_nest_end(skb, mp);  	} @@ -1319,7 +1464,8 @@ static inline size_t lfib_nlmsg_size(struct mpls_route *rt)  		if (nh->nh_dev)  			payload += nla_total_size(4); /* RTA_OIF */ -		payload += nla_total_size(2 + nh->nh_via_alen); /* RTA_VIA */ +		if (nh->nh_via_table != MPLS_NEIGH_TABLE_UNSPEC) /* RTA_VIA */ +			payload += nla_total_size(2 + nh->nh_via_alen);  		if (nh->nh_labels) /* RTA_NEWDST */  			payload += nla_total_size(nh->nh_labels * 4);  	} else { @@ -1328,7 +1474,9 @@ static inline size_t lfib_nlmsg_size(struct mpls_route *rt)  		for_nexthops(rt) {  			nhsize += nla_total_size(sizeof(struct rtnexthop)); -			nhsize += nla_total_size(2 + nh->nh_via_alen); +			/* RTA_VIA */ +			if (nh->nh_via_table != MPLS_NEIGH_TABLE_UNSPEC) +				nhsize += nla_total_size(2 + nh->nh_via_alen);  			if (nh->nh_labels)  				nhsize += nla_total_size(nh->nh_labels * 4);  		} endfor_nexthops(rt); | 
